data:image/s3,"s3://crabby-images/174f9/174f9cf4a2bbdfbf1409eb3943d45d2f0e1a2262" alt=""
The Struggle with Vue and ASP.NET Core
There is a wonderful template in Visual Studio to create a FullStack application that serves Vue 3 as the frontend and .NET core as the backend. Microsoft even provides a tutorial on it.
data:image/s3,"s3://crabby-images/c9cbc/c9cbc6252dcde485114e02b5541ae931dce14a46" alt="Screenshot of a 'Create a new project' window in Visual Studio, showing a search for 'vue' projects. The interface displays recent project templates with a note that recently accessed templates will be listed. Below, under 'Other results based on your search,' there are four project templates listed: 'Vue App' in TypeScript, 'Vue App' in JavaScript, 'Vue and ASP.NET Core' in JavaScript, and 'Vue and ASP.NET Core' in TypeScript. Each template is described as either a standalone Vue project or a full-stack application with a Vue frontend and ASP.NET Core backend, available for Windows, macOS, Linux, and web platforms."
But as usual, once you want to customize, things get complicated.
I’ve spent hours trying to understand why I could not change the URL to hit the server, and I finally found the issue. It’s no surprise that this was a CORS and HTTPS/Certificate issue.
The project is pretty basic, and we will concentrate on these four files: HellowWorld.vue, vite.config.ts, WeatherForecastController.cs, and Program.cs.
data:image/s3,"s3://crabby-images/4fd6a/4fd6a7a0f0a75ce7f0ba61effa5dafd765b1c263" alt="Screenshot of a Solution Explorer window in Visual Studio Code, displaying the file structure of a project named 'VueApp1' with two sub-projects: 'vueapp1.client' and 'VueApp1.Server'. The 'vueapp1.client' section shows a Vue.js application structure, including folders like 'public', 'src' with subfolders 'assets', 'components', and 'icons', and files such as 'HelloWorld.vue', 'TheWelcome.vue', 'main.ts', and configuration files like 'package.json' and 'vite.config.ts'. The 'VueApp1.Server' section displays a C# server-side project structure, including files like 'WeatherForecastController.cs', 'Program.cs', and 'appsettings.json'."
In HelloWorld.vue, there is a fetchData() function.
data:image/s3,"s3://crabby-images/228ce/228ce2a4f4830330fa99b84277f5d6baeda3edd0" alt="Screenshot of a Visual Studio Code window split into two panels. The left panel shows the file structure of the 'vueapp1.client' project, highlighting the 'HelloWorld.vue' file in a Vue.js application, with folders like 'public', 'src', 'assets', 'components', and 'icons', along with TypeScript files and configuration files. The right panel displays the code inside 'HelloWorld.vue', showing JavaScript/TypeScript code with a 'watch' function monitoring route changes, and an asynchronous 'fetchData' method that makes a POST request to 'weatherforecast', handles the response, and updates the loading state."
And inside is a fetch()
call. This is important because, by default, this call hits the backend controller namespace VueApp1.Server.Controllers
.
This seems pretty straightforward, but everything will stop working if you change
var response = await fetch('weatherforecast');
to
var response = await fetch('/api/weatherforecast');
You may get a NETWORK_ERR
or a No Valid JSON
error, but to me it didn’t make sense why that was happening.
Even changing from a relative path to an absolute path (like below) did not resolve the issue.
var response = await fetch('https://localhost:7185/api/weatherforecast');
To understand this, we have to look at vite.config.ts
.
data:image/s3,"s3://crabby-images/efee1/efee12e194c7995c6fb560cc99ca198678a42c80" alt="Screenshot of a Visual Studio Code window split into two panels. The left panel displays the file structure of the 'vueapp1.client' project, with the 'vite.config.ts' file highlighted, showing folders like 'src', 'public', and 'tsconfig.ts', along with configuration files. The right panel shows the code inside 'vite.config.ts', written in TypeScript, configuring a Vite development server. The code sets up environment variables for ASP.NET Core, defines server settings including a proxy for 'weatherforecast', specifies port 61402, and includes HTTPS configurations with certificate paths."
In here, some magic is happening; specifically, a server
is being setup/configured.
In here, there is a proxy
option. This is where the baseUrl
is being defined. The target is set just a few lines above as:
This little setup is really here to enable you to debug “easier” and not have to worry about certs or CORS at all. But to enable that, the system has some settings that matter. The biggest is this:
Do you remember the HelloWorld
file?
This is the endpoint that uses the server setup. With the target
URL set, the fetch
URL gets translated to: https://localhost:7185/weatherforecast
This will hit the corresponding controller in the backend, but everything broke when we changed this:
The issue was that we have to also change the server.proxy
configuration to this:
And then, of course, your controller also needs to be updated to have the corresponding route.
data:image/s3,"s3://crabby-images/67fde/67fded333e8513b9ad9b95fc15313ec4c764dd94" alt="Screenshot of code in Visual Studio Code, showing a portion of a C# class definition for 'WeatherForecastController'."
Running the app now should work again.
Great!
Just one more thing: it will break again when you try to hit the absolute URL.
WHY?
Because it’s not using the server configuration from before. As part of that configuration, there are certificates installed:
Those certificates are needed for SSL traffic in debugging.
Luckily, there is a simple fix: enable CORS in your backend for the port from the server (61402 in this case).
data:image/s3,"s3://crabby-images/41df3/41df3de46435c6c9255a925eb2a99e6e7ac7e90e" alt="Screenshot of a Solution Explorer window in Visual Studio, displaying the file structure of the 'VueApp1.Server' project."
Adding that will allow you to use the absolute URL as well, and you could technically go without the Vue server setup.
That’s it!
It took me way too long to get to a running debugging state that made sense, so I wanted to share this, hoping it would save you some time and suffering.
Please note that this whole setup is for debugging/development only. At the very top of Server Options, it says:
data:image/s3,"s3://crabby-images/2868e/2868e6c290f5ccb78cd44284645f6d6bd24f413a" alt="Screenshot of a section header in a software configuration or documentation interface, titled 'Server Options' in white text on a dark background."
So, you will still have to use real SSL configurations in your production environment, which is a good thing.