-
Notifications
You must be signed in to change notification settings - Fork 258
NuGet Restore No Op
Currently in all 4 NuGet Clients (NuGet.exe, dotnet.exe, VS Client, MSBuild) we do a full restore at every restore call The goal of this feature is to optimize restore so that if the restore result is up-to-date, the corresponding restore operation is not executed.
The benefits are a more seamless restore experience, avoid doing redundant work.
Project Model | NuGet.Exe | VS Client | dotnet.exe | MSBuild |
---|---|---|---|---|
Packages.config | Not covered. NuGet.exe restores packages.config projects first | Not covered. VS Client restores packages.config projects first | Not covered. packages.config projects are ignored | Not covered. MSBuild restores packages.config projects first |
project.json | Covered. | Covered. | Covered. | Covered. |
Package Reference | Covered. | Covered. | Covered. | Covered. |
The no-op restore will run by default. If you wish to force a restore, you can use -SkipNoOp as such:
- NuGet.exe restore MyProject.csproj -SkipNoOp
- Dotnet.exe restore MyProject.csproj -SkipNoOp
- MSBuild.exe /t:restore MyProject.csproj /p:SkipNoOp
- Visual Studio - NuGet Package Manager - TBD
-
Inputs affecting NuGet restore of a project:
- projectName.csproj
- NuGet.config(s)
- Properties passed in msbuild
- Environment variables
- NUGET_PACKAGES & NUGET_FALLBACK_PACKAGES
- Environment variables that affect msbuild and change pack ref/proj refs
-
Output of NuGet restore:
- project.assets.json
- projectName.csproj.nuget.props
- projectName.csproj.nuget.g.targets
We need to answer 3 questions to determine whether the restore operation will be a no-op.
-
Are all variables for resolving the packages correct?
- Package References
- Project References
- Target Frameworks
- Runtime Identifiers
- Guardrails
-
Are all the correct packages there?
- Are the same sources used?
- NuGet.Config
- -S SourceName
- Are the same sources used?
-
Have the NuGet Global Packages Folder & Fallback folder paths changed?
- NuGet.Config
- Environment Variables
- Do the hashes from project.assets.json match the ones found in the Global Packages/ Fallback Folders
-
Does restore no-op for all child projects?
- Self-explanatory, due to the transitive nature of package references, all child projects of the project being restore need to no-op too.
Leverage the existing dgspec graph.
Contents of the dg spec:
- Dependencies
- Target Frameworks
- RIDs
- Guardrails
The dg spec almost covers all the factors we need to check to perform the no-op. To make it complete we need to add the nuget settings. That include the resolved source and the Environment variables, more specifically NUGET_PACKAGES & NUGET_FALLBACK_PACKAGES folders.
Content that need to be added to the dg spec:
- Sources
- NUGET_PACKAGES path
- NUGET_FALLBACK_PACKAGES path
Once we have generated the dg spec, we will hash the contents and write it into a json file. projectName.nuget.json
Example content:
{
"dgspec" : "BA7816BF8F01CFEA414140DE5DAE2223B00361A396177A9CB410FF61F20015AD"
}
This json file will live in BaseIntermediateOutputPath next to the project.assets.json file. At each restore, we will compare the evaluated hash of the dg spec with the one currently in the BaseIntermediateOutputPath if existing. If the hashes are the same, we will then verify that the assets.json file exists, and then we will verify all the packages exist on disk. If all those conditions are fulfilled we will no-op.
Step by step illustration of the added restore logic:
- Read Settings (Includes Sources & resolve the paths for the packages folders)
- Build DG SPEC
- Hash the contents of the dg spec
- Compare hash value with the dgspec value in projectName.nuget.json if existing. If true GOTO 5
- Verify that the project.assets.json file exists. If true GOTO 6
- Verify that all packages exist on disk (in packages folder). If this condition is satisfied too, we stop here since we have a no-op restore, otherwise we continue 7.Restore as usual
- If the restore is successful we write the projectName.nuget.json with the new hash
Of course in case of a recursive restore we will need to do the same logic for all of the projects.
The No-Op feature does not cover all cases.
Since the No-Op evaluation is fully offline, it does not cover the cases where a floating version of a package is requested. If a new version of the package is uploaded, or a version of the package is unlisted it might change the package resolution behavior.
Example:
<ItemGroup>
<!-- ... -->
<PackageReference Include="My.Awesome.Package" Version="3.6.*" />
<!-- ... -->
</ItemGroup>
At the moment of the first restore, version="3.6.0" was the latest version on the package source. Sometime between that restore and the new restore, version="3.6.1" was uploaded to the source. Running a full restore would resolve 3.6.1 and downloaded that version of the package. However since the restore inputs have not changed, restore will think no action needs to be taken.
Check out the proposals in the accepted
& proposed
folders on the repository, and active PRs for proposals being discussed today.