integrating-sdk-and-msbuild
SDK-MSBuild Integration Patterns
MSBuild operates as a component within the .NET SDK. This boundary is the most complex integration point in the .NET build stack, spanning MSBuild (engine), SDK (target implementations), NuGet (restore), and Roslyn (compilation).
The Evaluation Boundary
Understanding MSBuild's evaluation order is critical for SDK target authoring:
1. Environment variables
2. Global properties (from CLI: -p:Foo=Bar)
3. Project-level properties (file order, with imports):
┌─ Sdk.props (SDK defaults)
├─ Directory.Build.props (user overrides BEFORE project)
├─ <Project> properties (the .csproj itself)
├─ Directory.Build.targets (user overrides AFTER project)
└─ Sdk.targets (SDK target definitions)
4. Item definitions
5. Items (including SDK default globs)
Key Import Order Rules
- SDK props import BEFORE user project — SDK defaults can be overridden by the user
- SDK targets import AFTER user project — SDK targets see user-specified properties
Directory.Build.propsis imported fromMicrosoft.Common.propsas an early user extension point after core defaults are computed — use it for solution-wide customization- Property defaults set in SDK must not override user-specified values — always use
Condition="'$(Prop)' == ''"
<!-- CORRECT: SDK default that respects user override -->
<OutputType Condition="'$(OutputType)' == ''">Library</OutputType>
<!-- WRONG: Unconditional set clobbers user's .csproj -->
<OutputType>Library</OutputType>
Restore and Build Separation
Restore and Build must never run in the same evaluation. The restore phase generates .g.props and .g.targets files that must be imported during evaluation — but they don't exist until restore completes.
dotnet buildimplicitly runs restore then build as separate invocationsdotnet build --no-restoreskips restore, assuming it already happened- Running both targets in one invocation (
/t:Restore;Build) is a known anti-pattern that causes intermittent failures
Project Reference Protocol
The project-reference protocol spans MSBuild, SDK, and NuGet. It is the most complex integration boundary.
How It Works
- Outer build dispatches to
_GetProjectReferenceTargetFrameworkPropertiesto determine inner build parameters - Inner build runs with the resolved
TargetFramework(singular) for each referenced project GetTargetPathreturns the output assembly for the referencing project to consume
Rules
- Protocol changes must be coordinated across MSBuild, SDK, and NuGet teams
- Multi-targeting projects (
<TargetFrameworks>) dispatch multiple inner builds - The outer build must not assume a single target framework
SetTargetFrameworkis how the outer build communicates the chosen framework to inner builds
Target Authoring in SDK Context
Extension Points
SDK provides well-known extension points for targets:
| Extension Point | Use For |
|---|---|
$(BuildDependsOn) |
Adding to the Build chain |
$(CompileDependsOn) |
Pre-compilation steps |
$(PublishDependsOn) |
Publish pipeline additions |
$(PackDependsOn) |
NuGet pack pipeline additions |
BeforeTargets="Build" |
Use sparingly; prefer DependsOnTargets |
Ordering Rules
- Use
DependsOnTargetsfor required predecessors — it's explicit and predictable BeforeTargets/AfterTargetsshould be used sparingly — they create implicit ordering that's hard to debug- Incremental build targets need precise
InputsandOutputs— incorrect declarations cause either rebuild-every-time or stale-output bugs - Test with multi-targeting — target chains execute once per
TargetFrameworkin the inner build
Cross-Repo Coordination
Changes that touch the MSBuild-SDK boundary often require coordinated PRs:
- MSBuild engine change → may need SDK target updates
- SDK target change → may need MSBuild API additions
- NuGet restore change → affects both MSBuild evaluation and SDK targets
Coordination Protocol
- File an issue in both repos describing the cross-cutting change
- Land the MSBuild change first (lower in the stack)
- Update SDK to consume the new MSBuild via dependency flow
- Test end-to-end with the SDK's MSBuild integration tests
Design-Time Builds
Visual Studio uses design-time builds with different target contracts:
- Design-time builds call
ResolveProjectReferencesbut notBuild - They set
$(DesignTimeBuild)=trueand$(BuildingProject)=false - Targets that should not run during design-time must check these properties
- Design-time builds must be fast — avoid expensive I/O or compilation
Common Integration Bugs
| Symptom | Likely Cause |
|---|---|
| Property has wrong value | Import ordering — check if SDK prop overrides user setting |
| Target runs in wrong order | Missing DependsOnTargets declaration |
| Build works, restore fails | Evaluation-time dependency on restore-generated files |
| Works single-target, fails multi-target | Target assumes single $(TargetFramework) |
| CLI build works, VS build fails | Design-time build target contract violation |