dotnet-build-analysis
dotnet-build-analysis
Help agents interpret and act on MSBuild build output. Covers error code prefixes, NuGet restore failures, analyzer warning interpretation, multi-targeting build differences, and "works locally, fails in CI" diagnosis patterns. Each subsection includes example output, diagnosis steps, and a fix pattern.
Scope
- MSBuild error code prefix interpretation (CS, MSB, NU, CA, IDE, NETSDK)
- NuGet restore failure diagnosis and resolution
- Analyzer warning triage and suppression guidance
- CI vs local build drift diagnosis
Out of scope
- Writing or modifying .csproj files -- see [skill:dotnet-csproj-reading]
- Project structure decisions -- see [skill:dotnet-project-structure]
- Common agent code mistakes -- see [skill:dotnet-agent-gotchas]
Prerequisites
.NET 8.0+ SDK. MSBuild (included with .NET SDK). Understanding of SDK-style project format.
Cross-references: [skill:dotnet-agent-gotchas] for common code mistakes that cause build errors, [skill:dotnet-csproj-reading] for project file structure and modification, [skill:dotnet-project-structure] for project organization and SDK selection.
Error Code Prefixes
MSBuild output uses standardized prefixes to indicate the error source. Understanding the prefix tells you which system produced the error and where to look for fixes.
CS -- C# Compiler Errors and Warnings
Produced by the Roslyn C# compiler. These are language-level issues in source code.
Example output:
src/MyApp.Api/Services/OrderService.cs(42,17): error CS0246: The type or namespace name 'OrderDto' could not be found (are you missing a using directive or an assembly reference?)
src/MyApp.Api/Models/User.cs(15,9): warning CS8618: Non-nullable property 'Name' must contain a non-null value when exiting constructor. Consider adding the 'required' modifier or declaring the property as nullable.
Diagnosis:
- Parse the file path and line number from the error --
src/MyApp.Api/Services/OrderService.csline 42, column 17. - CS0246 means a type is missing. Check: is the type defined? Is the namespace imported? Is the project referencing the assembly that contains it?
- CS8618 is a nullable reference type warning. The property needs a
requiredmodifier, nullable annotation (string?), or constructor initialization.
Fix pattern:
- CS0xxx (syntax/type errors): Fix source code at the indicated location. Add
usingdirectives, fix type names, add missing references. - CS8xxx (nullable warnings): Add null annotations, null checks, or
requiredmodifiers. Do NOT suppress with#pragmaor!operator.
MSB -- MSBuild Engine Errors
Produced by the MSBuild build engine itself. These indicate project file problems, target failures, or build system misconfiguration.
Example output:
error MSB4019: The imported project "C:\Program Files\dotnet\sdk\9.0.100\Microsoft\VisualStudio\v17.0\WebApplications\Microsoft.WebApplication.targets" was not found. Confirm that the expression in the Import declaration "..." is correct.
error MSB3644: The reference assemblies for .NETFramework,Version=v4.8 were not found. You might need to install the developer pack for this framework version.
Diagnosis:
- MSB4019: An MSBuild
.targetsfile is missing. This usually means wrong SDK type, missing workload, or corrupt SDK installation. - MSB3644: Targeting a framework version whose targeting pack is not installed. Common when a project targets .NET Framework but only .NET (Core) SDK is installed.
Fix pattern:
- MSB4019: Verify
<Project Sdk="...">is correct (e.g.,Microsoft.NET.Sdk.Webfor ASP.NET Core). Rundotnet workload listand install missing workloads. - MSB3xxx: Check
<TargetFramework>value. Ensure the required SDK or targeting pack is installed.
NU -- NuGet Errors and Warnings
Produced by the NuGet package manager during restore or pack operations.
Example output:
error NU1101: Unable to find package Newtonsoft.Json.Extensions. No packages exist with this id in source(s): nuget.org
warning NU1603: Microsoft.EntityFrameworkCore 9.0.0 depends on Microsoft.Extensions.Caching.Memory (>= 9.0.0) but version Microsoft.Extensions.Caching.Memory 8.0.1 was resolved. Approve the package to suppress this warning.
error NU1605: Detected package downgrade: Microsoft.Extensions.Logging from 9.0.0 to 8.0.1. Reference the package directly from the project to select a different version.
Diagnosis:
- NU1101: Package ID does not exist. Check spelling, verify the package source is configured, check if the package was renamed or deprecated.
- NU1603: Transitive dependency version conflict. A package wants a newer version than what is resolved.
- NU1605: Explicit downgrade detected. Two packages require different versions of the same dependency.
Fix pattern:
- NU1101: Fix the package name. Search nuget.org for the correct ID.
- NU1603/NU1605: Add a direct
<PackageReference>for the conflicting package at a compatible version, or use central package management to pin versions.
IDE -- IDE/Roslyn Analyzer Code Style Diagnostics
Produced by Roslyn IDE analyzers for code style enforcement. These are usually warnings, not errors (unless .editorconfig escalates them).
Example output:
src/MyApp.Api/Program.cs(1,1): warning IDE0005: Using directive is unnecessary.
src/MyApp.Api/Models/Order.cs(8,12): warning IDE0044: Make field readonly
src/MyApp.Api/Services/Report.cs(22,5): warning IDE0058: Expression value is never used
Diagnosis:
- IDE0005: Unused
usingdirective. Safe to remove. - IDE0044: Field can be
readonlybecause it is only assigned in the constructor. - IDE0058: A method return value is discarded. Either assign it or use
_ = ...to explicitly discard.
Fix pattern:
- IDE analyzers enforce code style. Fix them by applying the suggested change. Configure severity in
.editorconfigto promote warnings to errors for CI enforcement.
CA -- .NET Code Analysis (FxCop/Microsoft.CodeAnalysis.NetAnalyzers)
Produced by the .NET code analysis SDK analyzers for API design, performance, reliability, and security rules.
Example output:
src/MyApp.Api/Services/CacheService.cs(34,9): warning CA1848: Use the LoggerMessage delegates instead of calling 'LoggerExtensions.LogInformation(ILogger, string?, params object?[])'. Using LoggerMessage delegates provides better performance.
src/MyApp.Api/Controllers/UserController.cs(12,5): warning CA2007: Consider calling ConfigureAwait on the awaited task
src/MyApp.Api/Crypto/HashService.cs(8,9): warning CA5351: Do Not Use Broken Cryptographic Algorithms (MD5)
Diagnosis:
- CA1848: High-performance logging. Use
[LoggerMessage]source generator attributes instead of string interpolation in log calls. - CA2007:
ConfigureAwait(false)guidance for library code. Not applicable to ASP.NET Core app code (noSynchronizationContext). - CA5351: Security-critical. MD5 is broken for cryptographic purposes. Switch to SHA-256 or SHA-512.
Fix pattern:
- CA1xxx (design): Apply suggested API changes. These improve API consistency.
- CA2xxx (reliability/performance): Fix per suggestion. CA2007 can be suppressed in ASP.NET Core apps via
.editorconfig. - CA5xxx (security): Always fix. These flag real security vulnerabilities.
NuGet Restore Failures
NuGet restore is the first build step. When it fails, no compilation occurs. These are the most common restore failure patterns.
Pattern: Package Not Found
Example output:
Determining projects to restore...
Writing assets file to disk. Path: /src/MyApp.Api/obj/project.assets.json
/src/MyApp.Api/MyApp.Api.csproj : error NU1101: Unable to find package MyCompany.Shared.Models. No packages exist with this id in source(s): nuget.org
Failed to restore /src/MyApp.Api/MyApp.Api.csproj (in 2.14 sec).
Diagnosis:
- Is the package ID spelled correctly? NuGet IDs are case-insensitive but must be exact.
- Is the package from a private feed? Check
nuget.configfor feed configuration. NuGet searches feeds hierarchically upward from the project directory. - Is
packageSourceMappingconfigured? If so, the package must be mapped to a source that contains it.MyCompany.*patterns take precedence over*wildcard.
Fix pattern:
# Check configured sources
dotnet nuget list source
# Check if nuget.config exists (searches upward from project dir)
ls nuget.config ../nuget.config ../../nuget.config 2>/dev/null
<!-- Add private feed to nuget.config -->
<configuration>
<packageSources>
<add key="nuget.org" value="https://api.nuget.org/v3/index.json" />
<add key="MyCompany" value="https://pkgs.dev.azure.com/myorg/_packaging/myfeed/nuget/v3/index.json" />
</packageSources>
<packageSourceMapping>
<packageSource key="nuget.org">
<package pattern="*" />
</packageSource>
<packageSource key="MyCompany">
<package pattern="MyCompany.*" />
</packageSource>
</packageSourceMapping>
</configuration>
Pattern: Version Conflict
Example output:
error NU1107: Version conflict detected for Microsoft.Extensions.DependencyInjection.Abstractions.
MyApp.Api -> Microsoft.EntityFrameworkCore 9.0.0 -> Microsoft.Extensions.DependencyInjection.Abstractions (>= 9.0.0)
MyApp.Api -> Microsoft.Extensions.Hosting 8.0.1 -> Microsoft.Extensions.DependencyInjection.Abstractions (>= 8.0.1)
Diagnosis:
- Two dependency chains require different major versions of the same package.
- Trace each chain to find which top-level package is pinned at an older version.
- The fix is usually upgrading the older top-level package.
Fix pattern:
<!-- Upgrade the older top-level package to match -->
<PackageReference Include="Microsoft.Extensions.Hosting" Version="9.0.0" />
<!-- Or add a direct reference to force a specific version -->
<PackageReference Include="Microsoft.Extensions.DependencyInjection.Abstractions" Version="9.0.0" />
Pattern: Authentication Failure on Private Feed
Example output:
Retrying 'FindPackagesByIdAsyncCore' for source 'https://pkgs.dev.azure.com/myorg/_packaging/myfeed/nuget/v3/index.json'.
Response status code does not indicate success: 401 (Unauthorized).
error NU1301: Unable to load the service index for source https://pkgs.dev.azure.com/myorg/_packaging/myfeed/nuget/v3/index.json.
Diagnosis:
- Credentials are missing or expired for the private feed.
- In CI, check that the credential provider or PAT is configured.
- Locally, run
dotnet nuget update sourcewith credentials or use Azure Artifacts Credential Provider.
Fix pattern:
# Install Azure Artifacts Credential Provider (see official docs for platform-specific steps):
# https://github.com/microsoft/artifacts-credprovider#setup
# Windows: iex "& { $(irm https://aka.ms/install-artifacts-credprovider.ps1) }"
# macOS/Linux: sh -c "$(curl -fsSL https://aka.ms/install-artifacts-credprovider.sh)"
# Or add credentials explicitly to a specific source
dotnet nuget update source MyCompany --username az --password $PAT --store-password-in-clear-text
Analyzer Warning Interpretation
Analyzer warnings are produced by Roslyn analyzers bundled with the SDK or added via NuGet. Understanding when to fix vs. when to configure severity is critical.
Example Output
src/MyApp.Api/Controllers/OrdersController.cs(27,5): warning CA2007: Consider calling ConfigureAwait on the awaited task [/src/MyApp.Api/MyApp.Api.csproj]
src/MyApp.Api/Services/OrderService.cs(15,16): warning CA1062: In externally visible method 'OrderService.Process(string)', validate parameter 'input' is non-null before using it [/src/MyApp.Api/MyApp.Api.csproj]
src/MyApp.Api/Models/UserDto.cs(8,12): warning IDE0032: Use auto-implemented property [/src/MyApp.Api/MyApp.Api.csproj]
Diagnosis:
- Identify the prefix:
CA= Code Analysis (.NET analyzers),IDE= IDE code style analyzers. - Check severity: warnings don't break builds unless
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>is set. - Determine if the rule applies to your project type (e.g., CA2007 is irrelevant in ASP.NET Core — no SynchronizationContext).
- Decide: fix the code, configure severity in
.editorconfig, or suppress with documented justification.
Fix pattern:
- Fix the code when the analyzer identifies a real issue (CA1062 — add null validation or use
ArgumentNullException.ThrowIfNull). - Configure severity in
.editorconfigwhen the rule doesn't apply project-wide (see below). - Suppress inline only with documented justification (see "When Suppression Is Acceptable" below).
Severity Levels
| Severity | Build Impact | Action |
|---|---|---|
| Error | Build fails | Must fix before build succeeds |
| Warning | Build succeeds (unless <TreatWarningsAsErrors>true</TreatWarningsAsErrors>) |
Fix or configure in .editorconfig |
| Suggestion | Build succeeds; shown in IDE | Fix when practical |
| Hidden | Not shown; available via code fix | Ignore unless actively refactoring |
Configuring Severity
Use .editorconfig to control analyzer behavior across the project:
# .editorconfig (place at solution root)
[*.cs]
# Promote nullable warnings to errors (recommended)
dotnet_diagnostic.CS8600.severity = error
dotnet_diagnostic.CS8602.severity = error
dotnet_diagnostic.CS8603.severity = error
# Suppress ConfigureAwait warning in ASP.NET Core apps (no SynchronizationContext)
dotnet_diagnostic.CA2007.severity = none
# Promote security warnings to errors
dotnet_diagnostic.CA5350.severity = error
dotnet_diagnostic.CA5351.severity = error
When Suppression Is Acceptable
Suppression is acceptable ONLY when:
- The analyzer cannot understand the code's safety guarantee (e.g., a custom guard clause that ensures non-null).
- The rule does not apply to the project type (e.g., CA2007 in ASP.NET Core apps).
- A documented justification is provided.
// ACCEPTABLE: documented justification
[SuppressMessage("Reliability", "CA2007:ConfigureAwait",
Justification = "ASP.NET Core has no SynchronizationContext")]
public async Task<Order> GetOrderAsync(int id, CancellationToken ct)
{
return await _repo.GetByIdAsync(id, ct);
}
// NOT ACCEPTABLE: no justification, hides a real issue
#pragma warning disable CA1062
public void Process(string input) { } // input could be null
#pragma warning restore CA1062
Multi-Targeting Build Output
When a project targets multiple frameworks, MSBuild builds each TFM separately. Errors may appear for only one target.
Example Output
MyApp.Shared -> /src/MyApp.Shared/bin/Debug/net8.0/MyApp.Shared.dll
src/MyApp.Shared/Services/FeatureService.cs(18,30): error CS1061: 'FrozenDictionary<string, int>' does not contain a definition for 'GetAlternateLookup' [/src/MyApp.Shared/MyApp.Shared.csproj -> net8.0]
MyApp.Shared -> /src/MyApp.Shared/bin/Debug/net9.0/MyApp.Shared.dll
Build succeeded for net9.0.
Build FAILED for net8.0.
Diagnosis:
- The error tag
[...csproj -> net8.0]shows which TFM failed.net9.0succeeded. GetAlternateLookupwas added in .NET 9. The code uses an API not available in .NET 8.- The fix requires conditional compilation or an alternative API for the older TFM.
Fix pattern:
// Use preprocessor directives for TFM-specific code
#if NET9_0_OR_GREATER
var lookup = frozenDict.GetAlternateLookup<ReadOnlySpan<char>>();
return lookup.TryGetValue(key, out var value) ? value : default;
#else
return frozenDict.TryGetValue(key.ToString(), out var value) ? value : default;
#endif
<!-- Or constrain the feature to specific TFMs in the project file -->
<PropertyGroup>
<TargetFrameworks>net8.0;net9.0</TargetFrameworks>
</PropertyGroup>
<!-- TFM-conditional package reference -->
<ItemGroup Condition="'$(TargetFramework)' == 'net8.0'">
<PackageReference Include="System.Collections.Immutable" Version="8.0.0" />
</ItemGroup>
Reading Multi-Target Output
Key patterns for identifying TFM-specific issues:
[ProjectPath -> TFM]suffix on every diagnostic line identifies the target.Build succeeded for netX.0/Build FAILED for netX.0summary at the end.- Restore output shows all TFMs:
Restored ... (net8.0, net9.0). - Build output paths include the TFM:
bin/Debug/net8.0/vsbin/Debug/net9.0/.
CI Drift: Works Locally, Fails in CI
The most frustrating build failures are ones that pass locally but fail in CI. These are almost always caused by environmental differences.
Pattern: Different SDK Version
Example scenario:
Local: dotnet --version -> 9.0.200
CI: dotnet --version -> 9.0.100
Build error in CI:
error CS8652: The feature 'field keyword' is currently in Preview and *unsupported*.
Diagnosis:
- The local SDK (9.0.200) includes a language preview feature that the CI SDK (9.0.100) does not.
- A
global.jsonfile is either missing or not pinning the SDK version.
Fix pattern:
// global.json -- pin SDK version for consistent builds
{
"sdk": {
"version": "9.0.200",
"rollForward": "latestPatch"
}
}
Pattern: Missing Workload in CI
Example scenario:
CI error:
error NETSDK1147: To build this project, the following workloads must be installed: maui-android
Diagnosis:
- MAUI/Aspire/WASM workloads installed locally but not in the CI image.
- CI pipeline needs explicit workload install step.
Fix pattern:
# GitHub Actions example
- name: Install .NET workloads
run: dotnet workload install maui-android maui-ios
Pattern: Implicit NuGet Sources
Example scenario:
Local restore succeeds (using cached packages).
CI error:
error NU1101: Unable to find package MyCompany.Internal.Lib.
Diagnosis:
- Local machine has the package in the global NuGet cache from a previous restore.
- CI starts with a clean cache and cannot find the package because the private feed is not configured.
- A
nuget.configfile is missing from the repository, or CI lacks feed credentials.
Fix pattern:
- Add
nuget.configto the repository root with all required package sources. - Configure CI to authenticate to private feeds (credential provider, PAT, or managed identity).
- Do NOT rely on global NuGet cache for CI builds.
Pattern: OS-Specific Path Differences
Example scenario:
Local (Windows): Build succeeds
CI (Linux): error MSB4018: The "ResolveAssemblyReference" task failed.
Could not find file '/src/MyApp/../Shared/MyLib.dll'
Diagnosis:
- Windows file system is case-insensitive; Linux is case-sensitive.
- A file reference uses different casing than the actual file on disk.
- Or backslash path separators in MSBuild properties that Linux cannot resolve.
Fix pattern:
- Ensure file and directory names match the case used in project references exactly.
- Use forward slashes (
/) in.csprojpaths -- MSBuild normalizes them on all platforms. - Test in a Linux container locally with
docker runbefore pushing.
Pattern: TreatWarningsAsErrors in CI Only
Example scenario:
Local: Build succeeds with 3 warnings
CI: error CS8602: Dereference of a possibly null reference.
(because CI sets TreatWarningsAsErrors=true)
Diagnosis:
- CI pipeline or
Directory.Build.propsenables<TreatWarningsAsErrors>true</TreatWarningsAsErrors>via a CI-only condition. - Developers see warnings locally but never fix them because the build succeeds.
Fix pattern:
- Enable
<TreatWarningsAsErrors>true</TreatWarningsAsErrors>unconditionally inDirectory.Build.propsso local and CI builds behave identically. - Fix all warnings. Do NOT add CI-only MSBuild properties that diverge from local behavior.
Slopwatch Anti-Patterns
These patterns indicate an agent is hiding build problems rather than fixing them. Flag these during code review. See [skill:dotnet-slopwatch] for the automated quality gate that detects these patterns.
Warning Suppressions
<!-- RED FLAG: blanket NoWarn in .csproj -->
<PropertyGroup>
<NoWarn>CS8600;CS8602;CS8603;CS8604;IL2026;IL2046</NoWarn>
</PropertyGroup>
// RED FLAG: pragma disable without justification
#pragma warning disable CS8618
public class UserModel
{
public string Name { get; set; } // non-nullable not initialized
public string Email { get; set; }
}
#pragma warning restore CS8618
Fix: Remove <NoWarn> entries and fix the underlying issues. If suppression is truly needed, use .editorconfig with per-rule severity and a comment explaining why.
Silenced Analyzers Without Justification
// RED FLAG: suppressing security analyzer with no explanation
[SuppressMessage("Security", "CA5351")]
public byte[] HashData(byte[] input)
{
using var md5 = MD5.Create(); // insecure algorithm
return md5.ComputeHash(input);
}
# RED FLAG: disabling entire analyzer categories in .editorconfig
[*.cs]
dotnet_diagnostic.CA5350.severity = none
dotnet_diagnostic.CA5351.severity = none
dotnet_diagnostic.CA5358.severity = none
Fix: Replace insecure algorithms (MD5 -> SHA-256). If suppression is unavoidable (e.g., interop with a system requiring MD5), add a Justification string explaining the constraint.
Cross-References
- [skill:dotnet-agent-gotchas] -- common agent coding mistakes that produce build errors
- [skill:dotnet-csproj-reading] -- project file structure, PropertyGroup/ItemGroup conventions
- [skill:dotnet-project-structure] -- SDK selection, project organization, multi-project solutions