npx claudepluginhub dotnet/skills --plugin dotnet-upgradeThis skill uses the workspace's default tool permissions.
Migrate a .NET 9 project or solution to .NET 10, systematically resolving all breaking changes. The outcome is a project targeting `net10.0` that builds cleanly, passes tests, and accounts for every behavioral, source-incompatible, and binary-incompatible change introduced in the .NET 10 release.
references/aspnet-core-dotnet9to10.mdreferences/containers-interop-dotnet9to10.mdreferences/core-libraries-dotnet9to10.mdreferences/cryptography-dotnet9to10.mdreferences/csharp-compiler-dotnet9to10.mdreferences/efcore-dotnet9to10.mdreferences/extensions-hosting-dotnet9to10.mdreferences/sdk-msbuild-dotnet9to10.mdreferences/serialization-networking-dotnet9to10.mdreferences/winforms-wpf-dotnet9to10.mdGuides Next.js Cache Components and Partial Prerendering (PPR) with cacheComponents enabled. Implements 'use cache', cacheLife(), cacheTag(), revalidateTag(), static/dynamic optimization, and cache debugging.
Migrates code, prompts, and API calls from Claude Sonnet 4.0/4.5 or Opus 4.1 to Opus 4.5, updating model strings on Anthropic, AWS, GCP, Azure platforms.
Automates semantic versioning and release workflow for Claude Code plugins: bumps versions in package.json, marketplace.json, plugin.json; verifies builds; creates git tags, GitHub releases, changelogs.
Migrate a .NET 9 project or solution to .NET 10, systematically resolving all breaking changes. The outcome is a project targeting net10.0 that builds cleanly, passes tests, and accounts for every behavioral, source-incompatible, and binary-incompatible change introduced in the .NET 10 release.
TargetFramework from net9.0 to net10.0System.Linq.Async package to the built-in System.Linq.AsyncEnumerablenet10.0 and builds cleanly — migration is donemigrate-dotnet8-to-dotnet9 skill first to reach net9.0, then return to this skill for the net9.0 → net10.0 migration| Input | Required | Description |
|---|---|---|
| Project or solution path | Yes | The .csproj, .sln, or .slnx entry point to migrate |
| Build command | No | How to build (e.g., dotnet build, a repo build script). Auto-detect if not provided |
| Test command | No | How to run tests (e.g., dotnet test). Auto-detect if not provided |
| Project type hints | No | Whether the project uses ASP.NET Core, EF Core, WinForms, WPF, containers, etc. Auto-detect from PackageReferences and SDK attributes if not provided |
Answer directly from the loaded reference documents. Do not search the filesystem or fetch web pages for breaking change information — the references contain the authoritative details. Focus on identifying which breaking changes apply and providing concrete fixes. Exception: If you suspect a security vulnerability (CVE) may apply to the project's dependencies, check for published security advisories — the reference documents may not cover post-publication CVEs.
Commit strategy: Commit at each logical boundary — after updating the TFM (Step 2), after resolving build errors (Step 3), after addressing behavioral changes (Step 4), and after updating infrastructure (Step 5). This keeps each commit focused and reviewable.
.sln/.slnx files, or individual .csproj files.dotnet --version to confirm the .NET 10 SDK is installed. If it is not, stop and inform the user.Microsoft.NET.Sdk.Web → ASP.NET Core; Microsoft.NET.Sdk.WindowsDesktop with <UseWPF> or <UseWindowsForms> → WPF/WinFormsMicrosoft.EntityFrameworkCore.* → EF Core; Microsoft.Data.Sqlite → Sqlite; Microsoft.Extensions.Hosting → Generic Host / BackgroundServiceSystem.Linq.Async package reference → AsyncEnumerable migration neededSystem.Text.Json usage with polymorphism → Serialization changes relevantdotnet build --no-incremental or delete bin/obj) on the current net9.0 target to establish a clean baseline. Record any pre-existing warnings.In each .csproj (or Directory.Build.props if centralized), change:
<TargetFramework>net9.0</TargetFramework>
to:
<TargetFramework>net10.0</TargetFramework>
For multi-targeted projects, add net10.0 to <TargetFrameworks> or replace net9.0.
Update all Microsoft.Extensions.*, Microsoft.AspNetCore.*, Microsoft.EntityFrameworkCore.*, and other Microsoft package references to their 10.0.x versions. If using Central Package Management (Directory.Packages.props), update versions there.
Run dotnet restore. Watch for:
<PackageReference> if so.<PackageReference> must have a Version (or use CPM).dotnet restore now audits transitive deps) — review any new vulnerability warnings.Run a clean build. Collect all errors and new warnings. These will be addressed in Step 3.
Work through compilation errors and new warnings systematically. Load the appropriate reference documents based on the project type:
| If the project uses… | Load reference |
|---|---|
| Any .NET 10 project | references/csharp-compiler-dotnet9to10.md |
| Any .NET 10 project | references/core-libraries-dotnet9to10.md |
| Any .NET 10 project | references/sdk-msbuild-dotnet9to10.md |
| ASP.NET Core | references/aspnet-core-dotnet9to10.md |
| Entity Framework Core | references/efcore-dotnet9to10.md |
| Cryptography APIs | references/cryptography-dotnet9to10.md |
| Microsoft.Extensions.Hosting, BackgroundService, configuration | references/extensions-hosting-dotnet9to10.md |
| System.Text.Json, XmlSerializer, HttpClient, MailAddress, Uri | references/serialization-networking-dotnet9to10.md |
| Windows Forms or WPF | references/winforms-wpf-dotnet9to10.md |
| Docker containers, single-file apps, native interop | references/containers-interop-dotnet9to10.md |
Common source-incompatible changes to check for:
System.Linq.Async conflicts — Remove the System.Linq.Async package reference or upgrade to v7.0.0. If consumed transitively, add <ExcludeAssets>compile</ExcludeAssets>. Rename SelectAwait calls to Select where needed.
New obsoletion warnings (SYSLIB0058–SYSLIB0062):
SYSLIB0058: Replace SslStream.KeyExchangeAlgorithm/CipherAlgorithm/HashAlgorithm with NegotiatedCipherSuite — if the old properties were used to reject weak TLS ciphers, preserve equivalent validation logic using the new APISYSLIB0059: Replace SystemEvents.EventsThreadShutdown with AppDomain.ProcessExitSYSLIB0060: Replace Rfc2898DeriveBytes constructors with Rfc2898DeriveBytes.Pbkdf2SYSLIB0061: Replace Queryable.MaxBy/MinBy overloads taking IComparer<TSource> with ones taking IComparer<TKey>SYSLIB0062: Replace XsltSettings.EnableScript usageC# 14 field keyword in property accessors — The identifier field is now a contextual keyword inside property get/set/init accessors. Local variables named field cause CS9272 (error). Class members named field referenced without this. cause CS9258 (warning). Fix by renaming (e.g., fieldValue) or escaping with @field. See references/csharp-compiler-dotnet9to10.md.
C# 14 extension contextual keyword — Types, aliases, or type parameters named extension are disallowed. Rename or escape with @extension.
C# 14 overload resolution with span parameters — Expression trees containing .Contains() on arrays may now bind to MemoryExtensions.Contains instead of Enumerable.Contains. Enumerable.Reverse on arrays may resolve to the in-place Span extension. Fix by casting to IEnumerable<T>, using .AsEnumerable(), or explicit static invocations. See references/csharp-compiler-dotnet9to10.md for full details.
ASP.NET Core obsoletions (if applicable):
WebHostBuilder, IWebHost, WebHost are obsolete — migrate to Host.CreateDefaultBuilder or WebApplication.CreateBuilderIActionContextAccessor / ActionContextAccessor obsoleteWithOpenApi extension method deprecatedIncludeOpenAPIAnalyzers property deprecatedIPNetwork and ForwardedHeadersOptions.KnownNetworks obsoleteMicrosoft.Extensions.ApiDescription.Client package deprecatedMicrosoft.OpenApi v2.x breaking changes — Microsoft.AspNetCore.OpenApi 10.0 pulls in Microsoft.OpenApi v2.x which restructures namespaces and models. OpenApiString/OpenApiAny types are removed (use JsonNode), OpenApiSecurityScheme.Reference replaced by OpenApiSecuritySchemeReference, collections on OpenAPI model objects may be null, and OpenApiSchema.Nullable is removed. See references/aspnet-core-dotnet9to10.md for migration patterns.SDK changes:
dotnet new sln now defaults to SLNX format — use --format sln if the old format is neededdnx.ps1 removed from .NET SDKproject.json no longer supported in dotnet restoreEF Core source changes (if applicable) — See references/efcore-dotnet9to10.md for:
ExecuteUpdateAsync now accepts a regular lambda (expression tree construction code must be rewritten)IDiscriminatorPropertySetConvention signature changedIRelationalCommandDiagnosticsLogger methods add logCommandText parameterWinForms/WPF source changes (if applicable):
MenuItem and ContextMenu typesHtmlElement.InsertAdjacentElementColumnDefinitions and RowDefinitions are disallowed in WPFCryptography source changes (if applicable):
MLDsa and SlhDsa members renamed from SecretKey to PrivateKey (e.g., ExportMLDsaSecretKey → ExportMLDsaPrivateKey, SecretKeySizeInBytes → PrivateKeySizeInBytes)Rfc2898DeriveBytes constructors are obsolete (SYSLIB0060) — replace with static Rfc2898DeriveBytes.Pbkdf2(password, salt, iterations, hashAlgorithm, outputLength)CoseSigner.Key can now be null — check for null before useX509Certificate.GetKeyAlgorithmParameters() and PublicKey.EncodedParameters can return nullCLR_OPENSSL_VERSION_OVERRIDE to DOTNET_OPENSSL_VERSION_OVERRIDEBuild again after each batch of fixes. Repeat until the build is clean.
Behavioral changes do not cause build errors but may change runtime behavior. Review each applicable item and determine whether the previous behavior was relied upon.
High-impact behavioral changes (check first):
SIGTERM signal handling removed — The .NET runtime no longer registers default SIGTERM handlers. If you rely on AppDomain.ProcessExit or AssemblyLoadContext.Unloading being raised on SIGTERM:
PosixSignalRegistration.Create(PosixSignal.SIGTERM, _ => Environment.Exit(0)) explicitlyBackgroundService.ExecuteAsync runs entirely on a background thread — The synchronous portion before the first await no longer blocks startup. If startup ordering matters, move that code to StartAsync or the constructor, or implement IHostedLifecycleService.
Configuration null values are now preserved — JSON null values are no longer converted to empty strings. Properties initialized with non-default values will be overwritten with null. Review configuration binding code.
Microsoft.Data.Sqlite DateTimeOffset changes (all High impact):
GetDateTimeOffset without an offset now assumes UTC (previously assumed local)DateTimeOffset into REAL columns now converts to UTC firstGetDateTime with an offset now returns UTC with DateTimeKind.UtcAppContext.SetSwitch("Microsoft.Data.Sqlite.Pre10TimeZoneHandling", true) as a temporary workaroundEF Core parameterized collections — .Contains() on collections now uses multiple scalar parameters instead of JSON/OPENJSON. May affect query performance for large collections. Mitigation: UseParameterizedCollectionMode(ParameterTranslationMode.Parameter) to revert.
EF Core JSON data type on Azure SQL — Azure SQL and compatibility level ≥170 now use the json data type instead of nvarchar(max). A migration will be generated to alter existing columns. Mitigation: set compatibility level to 160 or use HasColumnType("nvarchar(max)") explicitly.
System.Text.Json property name conflict validation — Polymorphic types with properties conflicting with metadata names ($type, $id, $ref) now throw InvalidOperationException. Add [JsonIgnore] to conflicting properties.
Other behavioral changes to review:
BufferedStream.WriteByte no longer implicitly flushes — add explicit Flush() calls if neededDriveInfo.DriveFormat returns actual Linux filesystem type namesDirectoryControl parsing is more stringentDllImportSearchPath.AssemblyDirectory only searches the assembly directoryMailAddress enforces validation for consecutive dotsUri length limits removed — add explicit length validation if Uri was used to reject oversized input from untrusted sourcesXmlSerializer no longer ignores [Obsolete] properties — audit obsolete properties for sensitive data and add [XmlIgnore] to prevent unintended data exposuredotnet restore audits transitive packagesdotnet watch logs to stderr instead of stdoutdotnet CLI commands log non-command-relevant data to stderrreferences/sdk-msbuild-dotnet9to10.md)StatusStrip uses System RenderMode by default (WinForms)TreeView checkbox image truncation fix (WinForms)DynamicResource incorrect usage causes crash (WPF)Dockerfiles: Update base images. Default tags now use Ubuntu instead of Debian. Debian images are no longer shipped for .NET 10.
# Before
FROM mcr.microsoft.com/dotnet/sdk:9.0 AS build
FROM mcr.microsoft.com/dotnet/aspnet:9.0
# After
FROM mcr.microsoft.com/dotnet/sdk:10.0 AS build
FROM mcr.microsoft.com/dotnet/aspnet:10.0
CI/CD pipelines: Update SDK version references. If using global.json, update:
{
"sdk": {
"version": "10.0.100"
}
}
Environment variables renamed:
DOTNET_OPENSSL_VERSION_OVERRIDE replaces the old nameDOTNET_ICU_VERSION_OVERRIDE replaces the old nameNUGET_ENABLE_ENHANCED_HTTP_RETRY has been removedOpenSSL requirements: OpenSSL 1.1.1 or later is now required on Unix. OpenSSL cryptographic primitives are no longer supported on macOS.
Solution file format: If dotnet new sln is used in scripts, note it now generates SLNX format. Pass --format sln if the old format is needed.
dotnet build --no-incrementaldotnet test.Contains() on collectionsSslStream API migration (SYSLIB0058)[XmlIgnore], [JsonIgnore])Uri was used as a length gatetrueApplication Name that does not leak version infodotnet restore vulnerability audit findings are addressed, not suppressedThe references/ folder contains detailed breaking change information organized by technology area. Load only the references relevant to the project being migrated:
| Reference file | When to load |
|---|---|
references/csharp-compiler-dotnet9to10.md | Always (C# 14 compiler breaking changes — field keyword, extension keyword, span overloads) |
references/core-libraries-dotnet9to10.md | Always (applies to all .NET 10 projects) |
references/sdk-msbuild-dotnet9to10.md | Always (SDK and build tooling changes) |
references/aspnet-core-dotnet9to10.md | Project uses ASP.NET Core |
references/efcore-dotnet9to10.md | Project uses Entity Framework Core or Microsoft.Data.Sqlite |
references/cryptography-dotnet9to10.md | Project uses System.Security.Cryptography or X.509 certificates |
references/extensions-hosting-dotnet9to10.md | Project uses Generic Host, BackgroundService, or Microsoft.Extensions.Configuration |
references/serialization-networking-dotnet9to10.md | Project uses System.Text.Json, XmlSerializer, HttpClient, or networking APIs |
references/winforms-wpf-dotnet9to10.md | Project uses Windows Forms or WPF |
references/containers-interop-dotnet9to10.md | Project uses Docker containers, single-file publishing, or native interop (P/Invoke) |