How to migrate your Blazor Server app to .NET 8
It’s one thing to start a brand new project with .NET 8.
But what if you have an existing Blazor Server project which you want to upgrade?
How easy is it to migrate to the new framework, and what should you watch out for along the way?
Let’s find out!
Please note: this article has been updated to reflect some minor changes in .NET 8 Release Candidate 2.
Up the .NET version#
This first part is by far the easiest.
You can change your existing Blazor Server app to target .NET 8 with one change in your project’s .csproj file.
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup> <TargetFramework>net8.0</TargetFramework> </PropertyGroup>
</Project>
The TargetFramework
is the key here, and indicates you want this project to run using .NET 8.
If you have Microsoft NuGet packages references, you will probably want to up them too.
<ItemGroup> <PackageReference Include="Microsoft.EntityFrameworkCore" Version="8.0.0-8.0.0-rc.2.23480.1" /></ItemGroup>
In this case I’ve updated the reference to Microsoft.EntityFrameworkCore to the latest publicly available .NET 8 release candidate.
If you have other projects in your solution you’ll likely to need to upgrade them to .NET 8 too at this point.
If, at this point, you run into compile errors, it’s probably because of any NuGet packages you’ve upgraded.
You’ll need to tackle each of those compile errors as you encounter them (and obviously the details will vary depending on which NuGet packages you’re using).
Once you can compile, it’s worth running any automated tests you have, then check your app in the browser to make sure everything still works!
Switch to ‘SSR first mode’#
That’s the boring bit done, now to embrace .NET 8’s SSR render mode.
I’ve found the easiest way to think about .NET 8 static server-side rendering mode is as the foundation on which everything else is built.
If your app were a cake, SSR would be the sponge base
In practice this means telling .NET you want to route incoming requests to your Razor components first and foremost. Then let it take matters on from there.
Specifically, you need a little config to tell it to route incoming requests to one specific Razor component (of your choosing).
Program.cs
builder.Services.AddRazorComponents() .AddInteractiveServerComponents();
...
app.MapRazorComponents<App>() .AddInteractiveServerRenderMode();
The key here is the call to AddRazorComponents
.
While here I’ve also added interactive server components and enabled interactive server render mode (that’s Blazor server to you and me).
The next few steps can be a little confusing so let’s outline what we’re going to do, then walk through the exact steps, and take a moment to review what we’re doing (and why).
First we need to create an App
razor component.
This is where we indicated (in the code above) we wanted to route incoming requests, to see if they can be handled via a razor component.
Secondly we need to migrate our existing code from _Host.cshtml page across to the new App.razor component.
That’s because App.razor is going to become the new root page for our application.
Migrate from _Host.cshtml to App.razor#
Traditionally Blazor Server apps have been served via an initial Razor Page, usually called _Host.cshtml.
This acts as the entry point for your web app (and is where you reference stylesheets, scripts etc.)
You also likely have a separate App.razor component which handles routing for your Razor Components.
The usual flow is something like this:
In .NET 8 we can switch out that _Host.cshtml razor page for a razor component instead.
The aim is to change the flow to this:
The easiest way to change this flow is to:
- Rename your existing App.razor component file to Routes.razor
- Rename _Host.cshtml page to App.razor
Then make a few small tweaks to your new App.razor component.
You’re aiming for something like this:
App.razor
<!DOCTYPE html><html lang="en"><head> <meta charset="utf-8"/> <meta name="viewport" content="width=device-width, initial-scale=1.0"/> <base href="/" /> <link rel="stylesheet" href="css/bootstrap/bootstrap.min.css"/> <link href="css/site.css" rel="stylesheet"/> <link href="BlazorMigrationDemo.styles.css" rel="stylesheet"/> <link rel="icon" type="image/png" href="favicon.png"/> <HeadOutlet/></head>
<body> <Routes/>
<div id="blazor-error-ui"> An unhandled error has occurred. <a href="" class="reload">Reload</a> <a class="dismiss">🗙</a> </div>
<script src="_framework/blazor.web.js"></script>
</body></html>