dependabot-nuget 0.301.1 → 0.302.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (19) hide show
  1. checksums.yaml +4 -4
  2. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/DependencyDiscovery.props +4 -1
  3. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/DependencyDiscovery.targets +19 -4
  4. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Discover/DiscoveryWorker.cs +85 -81
  5. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Discover/SdkProjectDiscovery.cs +177 -28
  6. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/FrameworkChecker/FrameworkCompatibilityService.cs +15 -6
  7. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/FrameworkChecker/SupportedFrameworks.cs +6 -4
  8. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/PackagesConfigUpdater.cs +1 -1
  9. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/{WebApplicationTargetsConditionPatcher.cs → SpecialImportsConditionPatcher.cs} +18 -11
  10. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/XmlFilePreAndPostProcessor.cs +26 -11
  11. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Utilities/MSBuildHelper.cs +37 -11
  12. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Analyze/AnalyzeWorkerTests.cs +54 -0
  13. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Discover/DiscoveryWorkerTests.PackagesConfig.cs +54 -0
  14. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Discover/DiscoveryWorkerTests.Project.cs +94 -0
  15. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/FrameworkChecker/FrameworkCompatibilityServiceFacts.cs +1 -1
  16. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Update/SpecialFilePatcherTests.cs +99 -0
  17. data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Update/UpdateWorkerTests.PackageReference.cs +38 -0
  18. data/lib/dependabot/nuget/file_parser.rb +22 -19
  19. metadata +7 -6
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 956d84d6354497a9a36a640e0851ae0c97b739f32d8a4fcedc670eb3dc9f2175
4
- data.tar.gz: bdabf60631928f0bdb355d92b8f12466764349c5e482b9da073679b733531be2
3
+ metadata.gz: aa40dad694a6849be3c9da853f6d075f0db1f44f9cc9338e07586405e5f02bcc
4
+ data.tar.gz: 0f7ddac2d69e9a1dce70c7000b3159c0daf337816c93b9498678d58fa3753afc
5
5
  SHA512:
6
- metadata.gz: 677106b7e78e73b53bc3238a802e56028f18e33104b3ef7e15107abff684241bc87cfcf44f6c64470bf32ae3e0e3dded3c45f3799c422838239f06f9edd70b3a
7
- data.tar.gz: b906f03ce619ac2d45a907325e7b2a249976ef34056ef8e0d06db38f3c3f8a9a02be72c6e8210749cee1fcc03c4ca9afa9c0b5659838625ecd76bc00c121e19a
6
+ metadata.gz: ee07bf8ca5e089cc11ec1e077e2982b0525ac071f77f1411c3f2fdda7ffe378e6ecfa9d00c2762dee45dc65dd5fd76bcea3f1e2688dae83a1e431470744695f3
7
+ data.tar.gz: 5f0837e22360fb06fca6271bbd335e921e72be1d79d1b36802ca7aa80cecfb92c0218ae7b1a1914de7b45bde64d5c5ff34c4f1450c2405c30ac981d0dd90ea67
@@ -10,7 +10,10 @@
10
10
  -->
11
11
  <_DefaultTargetPlatformVersion Condition="'$(_DefaultTargetPlatformVersion)' == ''">0.0</_DefaultTargetPlatformVersion>
12
12
  <DesignTimeBuild>true</DesignTimeBuild>
13
+ <GenerateDependencyFile>true</GenerateDependencyFile>
14
+ <NuGetInteractive>false</NuGetInteractive>
15
+ <RunAnalyzers>false</RunAnalyzers>
13
16
  <EnableWindowsTargeting Condition="$(TargetFramework.Contains('-windows'))">true</EnableWindowsTargeting>
14
- <TargetPlatformVersion Condition="$(TargetPlatformVersion) == '' AND $(TargetFramework.Contains('-'))">$(_DefaultTargetPlatformVersion)</TargetPlatformVersion>
17
+ <TargetPlatformVersion Condition="$(TargetFramework.Contains('-')) AND '$(TargetPlatformVersion)' == ''">$(_DefaultTargetPlatformVersion)</TargetPlatformVersion>
15
18
  </PropertyGroup>
16
19
  </Project>
@@ -1,15 +1,30 @@
1
1
  <Project>
2
2
  <PropertyGroup>
3
- <!-- Dependency discovery requires a non-zero value for $(TargetPlatformVersion) -->
3
+ <!-- Dependency discovery requires a custom value for $(TargetPlatformVersion). Minimum version of unspecified Windows targeting is 7.0. -->
4
4
  <_DefaultTargetPlatformVersion>1.0</_DefaultTargetPlatformVersion>
5
+ <_DefaultTargetPlatformVersion Condition="$(TargetFramework.ToLowerInvariant().EndsWith('-windows'))">7.0</_DefaultTargetPlatformVersion>
6
+ <_DiscoverDependenciesDependsOn>ResolveAssemblyReferences</_DiscoverDependenciesDependsOn>
7
+
8
+ <!-- only SDK-style projects set `$(NETCoreSdkVersion)`; legacy projects don't have it -->
9
+ <_IsLegacyStyleProject Condition="'$(NETCoreSdkVersion)' == ''">true</_IsLegacyStyleProject>
10
+
11
+ <!-- These targets only exists for SDK-style projects -->
12
+ <_DiscoverDependenciesDependsOn Condition="'$(_IsLegacyStyleProject)' != 'true'">$(_DiscoverDependenciesDependsOn);GenerateBuildDependencyFile;ResolvePackageAssets</_DiscoverDependenciesDependsOn>
13
+
14
+ <!-- For non-SDK-style projects, use a different target defined below -->
15
+ <_DiscoverDependenciesDependsOn Condition="'$(_IsLegacyStyleProject)' == 'true'">$(_DiscoverDependenciesDependsOn);_DependencyDiscovery_LegacyProjects</_DiscoverDependenciesDependsOn>
5
16
  </PropertyGroup>
6
17
 
7
18
  <Import Project="DependencyDiscovery.props" />
8
19
 
9
- <Target Name="_DiscoverDependencies" DependsOnTargets="ResolveAssemblyReferences;GenerateBuildDependencyFile;ResolvePackageAssets">
20
+ <Target Name="_DiscoverDependencies" DependsOnTargets="$(_DiscoverDependenciesDependsOn)">
10
21
  <!--
11
- The targets ResolveAssemblyReferences and GenerateBuildDependencyFile are sufficient for projects targeting .NET Standard or .NET Core.
12
- The target ResolvePackageAssets is necessary for projects targeting .NET Framework.
22
+ This is purely a place to collect the `DependsOnTargets` attribute.
13
23
  -->
14
24
  </Target>
25
+
26
+ <Target Name="_DependencyDiscovery_LegacyProjects" Condition="'@(PackageReference)' != ''">
27
+ <!-- pseudo-sdk-style project; legacy, but with PackageReference elements - dependencies need to be resolved differently; output a sentinel value to notify the dependency discovery -->
28
+ <Message Text="_DependencyDiscovery_LegacyProjects::UseTemporaryProject" Importance="High" />
29
+ </Target>
15
30
  </Project>
@@ -10,6 +10,7 @@ using Microsoft.Build.Exceptions;
10
10
  using NuGet.Frameworks;
11
11
 
12
12
  using NuGetUpdater.Core.Run.ApiModel;
13
+ using NuGetUpdater.Core.Updater;
13
14
  using NuGetUpdater.Core.Utilities;
14
15
 
15
16
  namespace NuGetUpdater.Core.Discover;
@@ -324,101 +325,104 @@ public partial class DiscoveryWorker : IDiscoveryWorker
324
325
 
325
326
  _processedProjectPaths.Add(actualProjectPath);
326
327
 
327
- var relativeProjectPath = Path.GetRelativePath(workspacePath, actualProjectPath).NormalizePathToUnix();
328
- var packagesConfigResult = await PackagesConfigDiscovery.Discover(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
329
- var projectResults = await SdkProjectDiscovery.DiscoverAsync(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
330
-
331
- // Determine if there were unrestored MSBuildSdks
332
- var msbuildSdks = projectResults.SelectMany(p => p.Dependencies.Where(d => d.Type == DependencyType.MSBuildSdk)).ToImmutableArray();
333
- if (msbuildSdks.Length > 0)
328
+ using (new SpecialImportsConditionPatcher(actualProjectPath))
334
329
  {
335
- // If new SDKs were restored, then we need to rerun SdkProjectDiscovery.
336
- if (await TryRestoreMSBuildSdksAsync(repoRootPath, workspacePath, msbuildSdks, _logger))
337
- {
338
- projectResults = await SdkProjectDiscovery.DiscoverAsync(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
339
- }
340
- }
330
+ var relativeProjectPath = Path.GetRelativePath(workspacePath, actualProjectPath).NormalizePathToUnix();
331
+ var packagesConfigResult = await PackagesConfigDiscovery.Discover(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
332
+ var projectResults = await SdkProjectDiscovery.DiscoverAsync(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
341
333
 
342
- foreach (var projectResult in projectResults)
343
- {
344
- if (results.ContainsKey(projectResult.FilePath))
334
+ // Determine if there were unrestored MSBuildSdks
335
+ var msbuildSdks = projectResults.SelectMany(p => p.Dependencies.Where(d => d.Type == DependencyType.MSBuildSdk)).ToImmutableArray();
336
+ if (msbuildSdks.Length > 0)
345
337
  {
346
- continue;
338
+ // If new SDKs were restored, then we need to rerun SdkProjectDiscovery.
339
+ if (await TryRestoreMSBuildSdksAsync(repoRootPath, workspacePath, msbuildSdks, _logger))
340
+ {
341
+ projectResults = await SdkProjectDiscovery.DiscoverAsync(repoRootPath, workspacePath, actualProjectPath, _experimentsManager, _logger);
342
+ }
347
343
  }
348
344
 
349
- // If we had packages.config dependencies, merge them with the project dependencies
350
- if (projectResult.FilePath == relativeProjectPath && packagesConfigResult is not null)
345
+ foreach (var projectResult in projectResults)
351
346
  {
352
- var packagesConfigDependencies = packagesConfigResult.Dependencies
353
- .Select(d => d with { TargetFrameworks = projectResult.TargetFrameworks })
354
- .ToImmutableArray();
347
+ if (results.ContainsKey(projectResult.FilePath))
348
+ {
349
+ continue;
350
+ }
355
351
 
356
- results[projectResult.FilePath] = projectResult with
352
+ // If we had packages.config dependencies, merge them with the project dependencies
353
+ if (projectResult.FilePath == relativeProjectPath && packagesConfigResult is not null)
357
354
  {
358
- Dependencies = [.. projectResult.Dependencies, .. packagesConfigDependencies],
359
- };
360
- }
361
- else
362
- {
363
- results[projectResult.FilePath] = projectResult;
364
- }
365
- }
355
+ var packagesConfigDependencies = packagesConfigResult.Dependencies
356
+ .Select(d => d with { TargetFrameworks = projectResult.TargetFrameworks })
357
+ .ToImmutableArray();
366
358
 
367
- if (packagesConfigResult is not null)
368
- {
369
- // we might have to merge this dependency with some others
370
- if (results.TryGetValue(relativeProjectPath, out var existingProjectDiscovery))
371
- {
372
- // merge SDK and packages.config results
373
- var mergedDependencies = existingProjectDiscovery.Dependencies.Concat(packagesConfigResult.Dependencies)
374
- .DistinctBy(d => d.Name, StringComparer.OrdinalIgnoreCase)
375
- .OrderBy(d => d.Name)
376
- .ToImmutableArray();
377
- var mergedTargetFrameworks = existingProjectDiscovery.TargetFrameworks.Concat(packagesConfigResult.TargetFrameworks)
378
- .Select(t =>
359
+ results[projectResult.FilePath] = projectResult with
379
360
  {
380
- try
381
- {
382
- var tfm = NuGetFramework.Parse(t);
383
- return tfm.GetShortFolderName();
384
- }
385
- catch
386
- {
387
- return string.Empty;
388
- }
389
- })
390
- .Where(tfm => !string.IsNullOrEmpty(tfm))
391
- .Distinct()
392
- .OrderBy(tfm => tfm)
393
- .ToImmutableArray();
394
- var mergedProperties = existingProjectDiscovery.Properties; // packages.config discovery doesn't produce properties
395
- var mergedImportedFiles = existingProjectDiscovery.ImportedFiles; // packages.config discovery doesn't produce imported files
396
- var mergedAdditionalFiles = existingProjectDiscovery.AdditionalFiles.Concat(packagesConfigResult.AdditionalFiles)
397
- .Distinct(StringComparer.OrdinalIgnoreCase)
398
- .OrderBy(f => f)
399
- .ToImmutableArray();
400
- var mergedResult = new ProjectDiscoveryResult()
361
+ Dependencies = [.. projectResult.Dependencies, .. packagesConfigDependencies],
362
+ };
363
+ }
364
+ else
401
365
  {
402
- FilePath = existingProjectDiscovery.FilePath,
403
- Dependencies = mergedDependencies,
404
- TargetFrameworks = mergedTargetFrameworks,
405
- Properties = mergedProperties,
406
- ImportedFiles = mergedImportedFiles,
407
- AdditionalFiles = mergedAdditionalFiles,
408
- };
409
- results[relativeProjectPath] = mergedResult;
366
+ results[projectResult.FilePath] = projectResult;
367
+ }
410
368
  }
411
- else
369
+
370
+ if (packagesConfigResult is not null)
412
371
  {
413
- // add packages.config results
414
- results[relativeProjectPath] = new ProjectDiscoveryResult()
372
+ // we might have to merge this dependency with some others
373
+ if (results.TryGetValue(relativeProjectPath, out var existingProjectDiscovery))
415
374
  {
416
- FilePath = relativeProjectPath,
417
- Dependencies = packagesConfigResult.Dependencies,
418
- TargetFrameworks = packagesConfigResult.TargetFrameworks,
419
- ImportedFiles = [], // no imported files resolved for packages.config scenarios
420
- AdditionalFiles = packagesConfigResult.AdditionalFiles,
421
- };
375
+ // merge SDK and packages.config results
376
+ var mergedDependencies = existingProjectDiscovery.Dependencies.Concat(packagesConfigResult.Dependencies)
377
+ .DistinctBy(d => d.Name, StringComparer.OrdinalIgnoreCase)
378
+ .OrderBy(d => d.Name)
379
+ .ToImmutableArray();
380
+ var mergedTargetFrameworks = existingProjectDiscovery.TargetFrameworks.Concat(packagesConfigResult.TargetFrameworks)
381
+ .Select(t =>
382
+ {
383
+ try
384
+ {
385
+ var tfm = NuGetFramework.Parse(t);
386
+ return tfm.GetShortFolderName();
387
+ }
388
+ catch
389
+ {
390
+ return string.Empty;
391
+ }
392
+ })
393
+ .Where(tfm => !string.IsNullOrEmpty(tfm))
394
+ .Distinct()
395
+ .OrderBy(tfm => tfm)
396
+ .ToImmutableArray();
397
+ var mergedProperties = existingProjectDiscovery.Properties; // packages.config discovery doesn't produce properties
398
+ var mergedImportedFiles = existingProjectDiscovery.ImportedFiles; // packages.config discovery doesn't produce imported files
399
+ var mergedAdditionalFiles = existingProjectDiscovery.AdditionalFiles.Concat(packagesConfigResult.AdditionalFiles)
400
+ .Distinct(StringComparer.OrdinalIgnoreCase)
401
+ .OrderBy(f => f)
402
+ .ToImmutableArray();
403
+ var mergedResult = new ProjectDiscoveryResult()
404
+ {
405
+ FilePath = existingProjectDiscovery.FilePath,
406
+ Dependencies = mergedDependencies,
407
+ TargetFrameworks = mergedTargetFrameworks,
408
+ Properties = mergedProperties,
409
+ ImportedFiles = mergedImportedFiles,
410
+ AdditionalFiles = mergedAdditionalFiles,
411
+ };
412
+ results[relativeProjectPath] = mergedResult;
413
+ }
414
+ else
415
+ {
416
+ // add packages.config results
417
+ results[relativeProjectPath] = new ProjectDiscoveryResult()
418
+ {
419
+ FilePath = relativeProjectPath,
420
+ Dependencies = packagesConfigResult.Dependencies,
421
+ TargetFrameworks = packagesConfigResult.TargetFrameworks,
422
+ ImportedFiles = [], // no imported files resolved for packages.config scenarios
423
+ AdditionalFiles = packagesConfigResult.AdditionalFiles,
424
+ };
425
+ }
422
426
  }
423
427
  }
424
428
  }
@@ -1,5 +1,4 @@
1
1
  using System.Collections.Immutable;
2
- using System.Reflection;
3
2
  using System.Xml.Linq;
4
3
  using System.Xml.XPath;
5
4
 
@@ -22,6 +21,11 @@ internal static class SdkProjectDiscovery
22
21
  "PackageReference"
23
22
  };
24
23
 
24
+ private static readonly HashSet<string> PackageVersionItemNames = new HashSet<string>(StringComparer.Ordinal)
25
+ {
26
+ "PackageVersion"
27
+ };
28
+
25
29
  // the items listed below represent collection names that NuGet will resolve a package into, along with the metadata value names to get the package name and version
26
30
  private static readonly Dictionary<string, (string NameMetadata, string VersionMetadata)> ResolvedPackageItemNames = new Dictionary<string, (string, string)>(StringComparer.OrdinalIgnoreCase)
27
31
  {
@@ -69,26 +73,30 @@ internal static class SdkProjectDiscovery
69
73
  // the following collection feature heavily; the shape is described as follows
70
74
 
71
75
  Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesPerProject = new(PathComparer.Instance);
72
- // projectPath tfm packageName, packageVersion
76
+ // projectPath tfm packageName packageVersion
73
77
 
74
- Dictionary<string, HashSet<string>> topLevelPackagesPerProject = new(PathComparer.Instance);
75
- // projectPath, packageNames
78
+ Dictionary<string, Dictionary<string, HashSet<string>>> topLevelPackagesPerProject = new(PathComparer.Instance);
79
+ // projectPath tfm, packageNames
80
+
81
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> explicitPackageVersionsPerProject = new(PathComparer.Instance);
82
+ // projectPath, tfm, packageName, packageVersion
76
83
 
77
84
  Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesReplacedBySdkPerProject = new(PathComparer.Instance);
78
- // projectPath tfm packageName, packageVersion
85
+ // projectPath tfm packageName packageVersion
79
86
 
80
87
  Dictionary<string, Dictionary<string, string>> resolvedProperties = new(PathComparer.Instance);
81
- // projectPath propertyName, propertyValue
88
+ // projectPath propertyName propertyValue
82
89
 
83
90
  Dictionary<string, HashSet<string>> importedFiles = new(PathComparer.Instance);
84
- // projectPath, importedFiles
91
+ // projectPath importedFiles
85
92
 
86
93
  Dictionary<string, HashSet<string>> referencedProjects = new(PathComparer.Instance);
87
- // projectPath, referencedProjects
94
+ // projectPath referencedProjects
88
95
 
89
96
  Dictionary<string, HashSet<string>> additionalFiles = new(PathComparer.Instance);
90
- // projectPath, additionalFiles
97
+ // projectPath additionalFiles
91
98
 
99
+ var requiresManualPackageResolution = false;
92
100
  var tfms = await MSBuildHelper.GetTargetFrameworkValuesFromProject(repoRootPath, startingProjectPath, experimentsManager, logger);
93
101
  foreach (var tfm in tfms)
94
102
  {
@@ -100,7 +108,7 @@ internal static class SdkProjectDiscovery
100
108
  var (exitCode, stdOut, stdErr) = await MSBuildHelper.HandleGlobalJsonAsync(startingProjectDirectory, repoRootPath, experimentsManager, async () =>
101
109
  {
102
110
  // the built-in target `GenerateBuildDependencyFile` forces resolution of all NuGet packages, but doesn't invoke a full build
103
- var dependencyDiscoveryTargetsPath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location)!, "DependencyDiscovery.targets");
111
+ var dependencyDiscoveryTargetsPath = MSBuildHelper.GetFileFromRuntimeDirectory("DependencyDiscovery.targets");
104
112
  var args = new List<string>()
105
113
  {
106
114
  "build",
@@ -127,10 +135,10 @@ internal static class SdkProjectDiscovery
127
135
  return (exitCode, stdOut, stdErr);
128
136
  }, logger, retainMSBuildSdks: true);
129
137
  MSBuildHelper.ThrowOnError(stdOut);
130
- if (stdOut.Contains("""error MSB4057: The target "GenerateBuildDependencyFile" does not exist in the project."""))
138
+ if (stdOut.Contains("_DependencyDiscovery_LegacyProjects::UseTemporaryProject"))
131
139
  {
132
- // this can happen if it's a non-SDK-style project; totally normal, not worth examining the binlog
133
- return [];
140
+ // special case - legacy project with <PackageReference> elements; this requires extra handling below
141
+ requiresManualPackageResolution = true;
134
142
  }
135
143
  if (exitCode != 0)
136
144
  {
@@ -181,7 +189,7 @@ internal static class SdkProjectDiscovery
181
189
  }
182
190
  break;
183
191
  case NamedNode namedNode when namedNode is AddItem or RemoveItem:
184
- ProcessResolvedPackageReference(namedNode, packagesPerProject, topLevelPackagesPerProject, experimentsManager);
192
+ ProcessResolvedPackageReference(namedNode, packagesPerProject, topLevelPackagesPerProject, explicitPackageVersionsPerProject, experimentsManager);
185
193
 
186
194
  if (namedNode is AddItem addItem)
187
195
  {
@@ -304,8 +312,52 @@ internal static class SdkProjectDiscovery
304
312
  }
305
313
  }
306
314
 
315
+ if (requiresManualPackageResolution)
316
+ {
317
+ // we were able to collect all <PackageReference> elements, but no transitive dependencies were resolved
318
+ // to do this we create a temporary project with all of the top-level project elements, resolve _again_, then rebuild the proper result
319
+ packagesPerProject = await RebuildPackagesPerProject(
320
+ repoRootPath,
321
+ startingProjectPath,
322
+ tfms,
323
+ packagesPerProject,
324
+ explicitPackageVersionsPerProject,
325
+ experimentsManager,
326
+ logger
327
+ );
328
+ }
329
+
307
330
  // and done
308
- var projectDiscoveryResults = packagesPerProject.Keys.OrderBy(p => p).Select(projectPath =>
331
+ var projectDiscoveryResults = BuildResults(
332
+ repoRootPath,
333
+ workspacePath,
334
+ packagesPerProject,
335
+ explicitPackageVersionsPerProject,
336
+ packagesReplacedBySdkPerProject,
337
+ topLevelPackagesPerProject,
338
+ resolvedProperties,
339
+ referencedProjects,
340
+ importedFiles,
341
+ additionalFiles
342
+ );
343
+ return projectDiscoveryResults;
344
+ }
345
+
346
+ private static ImmutableArray<ProjectDiscoveryResult> BuildResults(
347
+ string repoRootPath,
348
+ string workspacePath,
349
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesPerProject,
350
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> packageVersionsPerProject,
351
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesReplacedBySdkPerProject,
352
+ Dictionary<string, Dictionary<string, HashSet<string>>> topLevelPackagesPerProject,
353
+ Dictionary<string, Dictionary<string, string>> resolvedProperties,
354
+ Dictionary<string, HashSet<string>> referencedProjects,
355
+ Dictionary<string, HashSet<string>> importedFiles,
356
+ Dictionary<string, HashSet<string>> additionalFiles
357
+ )
358
+ {
359
+ var projectDiscoveryResults = new List<ProjectDiscoveryResult>();
360
+ foreach (var projectPath in packagesPerProject.Keys.OrderBy(p => p)) //packagesPerProject.Keys.OrderBy(p => p).Select(projectPath =>
309
361
  {
310
362
  // gather some project-level information
311
363
  var packagesByTfm = packagesPerProject[projectPath];
@@ -341,7 +393,10 @@ internal static class SdkProjectDiscovery
341
393
  var localPropertyDefinitionElements = doc.Root!.XPathSelectElements("/Project/PropertyGroup/*");
342
394
  var projectPropertyNames = localPropertyDefinitionElements.Select(e => e.Name.LocalName).ToHashSet(StringComparer.OrdinalIgnoreCase);
343
395
  var projectRelativePath = Path.GetRelativePath(workspacePath, projectPath);
344
- var topLevelPackageNames = topLevelPackagesPerProject.GetOrAdd(projectPath, () => new(StringComparer.OrdinalIgnoreCase));
396
+ var topLevelPackageNames = topLevelPackagesPerProject
397
+ .GetOrAdd(projectPath, () => new(StringComparer.OrdinalIgnoreCase))
398
+ .SelectMany(kvp => kvp.Value)
399
+ .ToHashSet(StringComparer.OrdinalIgnoreCase);
345
400
 
346
401
  // create dependencies
347
402
  var tfms = packagesByTfm.Keys.OrderBy(tfm => tfm).ToImmutableArray();
@@ -379,7 +434,7 @@ internal static class SdkProjectDiscovery
379
434
  .OrderBy(p => p)
380
435
  .ToImmutableArray();
381
436
 
382
- return new ProjectDiscoveryResult()
437
+ var projectDiscoveryResult = new ProjectDiscoveryResult()
383
438
  {
384
439
  FilePath = projectRelativePath,
385
440
  Dependencies = dependencies,
@@ -389,14 +444,65 @@ internal static class SdkProjectDiscovery
389
444
  ImportedFiles = imported,
390
445
  AdditionalFiles = additional,
391
446
  };
392
- }).ToImmutableArray();
393
- return projectDiscoveryResults;
447
+ projectDiscoveryResults.Add(projectDiscoveryResult);
448
+ }
449
+ return projectDiscoveryResults.ToImmutableArray();
450
+ }
451
+
452
+ private static async Task<Dictionary<string, Dictionary<string, Dictionary<string, string>>>> RebuildPackagesPerProject(
453
+ string repoRootPath,
454
+ string projectPath,
455
+ ImmutableArray<string> targetFrameworks,
456
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesPerProject,
457
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> explicitPackageVersionsPerProject,
458
+ ExperimentsManager experimentsManager,
459
+ ILogger logger
460
+ )
461
+ {
462
+ var tempDirectory = Directory.CreateTempSubdirectory("legacy-package-reference-resolution_");
463
+ try
464
+ {
465
+ // gather top level dependencies from topLevelPackagesPerProject
466
+ // TODO: we don't currently partition dependencies by TFM; this will have to be redone when that's supported
467
+ var topLevelDependencies = explicitPackageVersionsPerProject
468
+ .GetOrAdd(projectPath, () => new(StringComparer.OrdinalIgnoreCase))
469
+ .SelectMany(kvp => kvp.Value)
470
+ .Select(kvp => new Dependency(kvp.Key, kvp.Value, DependencyType.PackageReference, TargetFrameworks: targetFrameworks))
471
+ .ToImmutableArray();
472
+
473
+ var tempProjectPath = await MSBuildHelper.CreateTempProjectAsync(tempDirectory, repoRootPath, projectPath, targetFrameworks, topLevelDependencies, experimentsManager, logger);
474
+ var tempProjectDirectory = Path.GetDirectoryName(tempProjectPath)!;
475
+ var rediscoveredDependencies = await DiscoverWithBinLogAsync(tempProjectDirectory, tempProjectDirectory, tempProjectPath, experimentsManager, logger);
476
+ var rediscoveredDependenciesForThisProject = rediscoveredDependencies.Single(); // we started with a single temp project, this will be the only result
477
+
478
+ // re-build packagesPerProject
479
+ var rebuiltPackagesPerProject = packagesPerProject.ToDictionary(PathComparer.Instance); // shallow copy
480
+ rebuiltPackagesPerProject[projectPath] = new(StringComparer.OrdinalIgnoreCase); // rebuild for this project
481
+ var rebuiltPackagesForThisProject = rebuiltPackagesPerProject[projectPath];
482
+ foreach (var tfm in targetFrameworks)
483
+ {
484
+ rebuiltPackagesForThisProject[tfm] = rediscoveredDependenciesForThisProject.Dependencies.ToDictionary(d => d.Name, d => d.Version!, StringComparer.OrdinalIgnoreCase);
485
+ }
486
+
487
+ return rebuiltPackagesPerProject;
488
+ }
489
+ finally
490
+ {
491
+ try
492
+ {
493
+ tempDirectory.Delete(recursive: true);
494
+ }
495
+ catch
496
+ {
497
+ }
498
+ }
394
499
  }
395
500
 
396
501
  private static void ProcessResolvedPackageReference(
397
502
  NamedNode node,
398
503
  Dictionary<string, Dictionary<string, Dictionary<string, string>>> packagesPerProject, // projectPath -> tfm -> (packageName, packageVersion)
399
- Dictionary<string, HashSet<string>> topLevelPackagesPerProject,
504
+ Dictionary<string, Dictionary<string, HashSet<string>>> topLevelPackagesPerProject, // projectPath -> tfm -> packageName
505
+ Dictionary<string, Dictionary<string, Dictionary<string, string>>> packageVersionsPerProject, // projectPath -> tfm -> (packageName, packageVersion)
400
506
  ExperimentsManager experimentsManager
401
507
  )
402
508
  {
@@ -416,16 +522,28 @@ internal static class SdkProjectDiscovery
416
522
  continue;
417
523
  }
418
524
 
419
- var topLevelPackages = topLevelPackagesPerProject.GetOrAdd(projectEvaluation.ProjectFile, () => new(StringComparer.OrdinalIgnoreCase));
420
-
421
- if (doRemoveOperation)
525
+ var tfm = GetPropertyValueFromProjectEvaluation(projectEvaluation, "TargetFramework");
526
+ if (tfm is not null)
422
527
  {
423
- topLevelPackages.Remove(packageName);
424
- }
528
+ var topLevelPackages = topLevelPackagesPerProject.GetOrAdd(projectEvaluation.ProjectFile, () => new(StringComparer.OrdinalIgnoreCase));
529
+ var topLevelPackagesPerTfm = topLevelPackages.GetOrAdd(tfm, () => new(StringComparer.OrdinalIgnoreCase));
425
530
 
426
- if (doAddOperation)
427
- {
428
- topLevelPackages.Add(packageName);
531
+ if (doRemoveOperation)
532
+ {
533
+ topLevelPackagesPerTfm.Remove(packageName);
534
+ }
535
+
536
+ if (doAddOperation)
537
+ {
538
+ topLevelPackagesPerTfm.Add(packageName);
539
+ var packageVersion = GetChildMetadataValue(child, "Version");
540
+ if (packageVersion is not null)
541
+ {
542
+ var packagesPerTfm = packageVersionsPerProject.GetOrAdd(projectEvaluation.ProjectFile, () => new(StringComparer.OrdinalIgnoreCase));
543
+ var packageVersions = packagesPerTfm.GetOrAdd(tfm, () => new(StringComparer.OrdinalIgnoreCase));
544
+ packageVersions[packageName] = packageVersion;
545
+ }
546
+ }
429
547
  }
430
548
  }
431
549
  }
@@ -469,6 +587,37 @@ internal static class SdkProjectDiscovery
469
587
  }
470
588
  }
471
589
  }
590
+ else if (PackageVersionItemNames.Contains(node.Name))
591
+ {
592
+ foreach (var child in node.Children.OfType<Item>())
593
+ {
594
+ var projectEvaluation = GetNearestProjectEvaluation(node);
595
+ if (projectEvaluation is not null)
596
+ {
597
+ var tfm = GetPropertyValueFromProjectEvaluation(projectEvaluation, "TargetFramework");
598
+ if (tfm is not null)
599
+ {
600
+ var packageName = child.Name;
601
+ var packageVersions = packageVersionsPerProject.GetOrAdd(projectEvaluation.ProjectFile, () => new(StringComparer.OrdinalIgnoreCase));
602
+ var packageVersionsPerTfm = packageVersions.GetOrAdd(tfm, () => new(StringComparer.OrdinalIgnoreCase));
603
+
604
+ if (doRemoveOperation)
605
+ {
606
+ packageVersionsPerTfm.Remove(packageName);
607
+ }
608
+
609
+ if (doAddOperation)
610
+ {
611
+ var packageVersion = GetChildMetadataValue(child, "Version");
612
+ if (packageVersion is not null)
613
+ {
614
+ packageVersionsPerTfm[packageName] = packageVersion;
615
+ }
616
+ }
617
+ }
618
+ }
619
+ }
620
+ }
472
621
  }
473
622
 
474
623
  private static string? GetChildMetadataValue(TreeNode node, string metadataItemName)
@@ -61,14 +61,23 @@ public class FrameworkCompatibilityService
61
61
  }
62
62
  }
63
63
 
64
- matrix.Add(
65
- SupportedFrameworks.Net60Windows7,
66
- new HashSet<NuGetFramework>
64
+ // e.g., explicitly allow a project targeting `net9.0-windows` to consume packages targeting `net9.0-windows7.0`
65
+ foreach (var packageFramework in SupportedFrameworks.TfmFilters.NetTfms)
66
+ {
67
+ if (packageFramework.Version.Major <= 5)
68
+ {
69
+ // the TFM `net5.0-windows7.0` isn't valid
70
+ continue;
71
+ }
72
+
73
+ var packageFrameworkWithWindowsVersion = new NuGetFramework(packageFramework.Framework, packageFramework.Version, "windows", FrameworkConstants.Version7);
74
+ var compatibleVersions = SupportedFrameworks.TfmFilters.NetTfms.Where(t => t.Version.Major >= packageFrameworkWithWindowsVersion.Version.Major).ToArray();
75
+ foreach (var compatibleVersion in compatibleVersions)
67
76
  {
68
- SupportedFrameworks.Net60Windows, SupportedFrameworks.Net60Windows7,
69
- SupportedFrameworks.Net70Windows, SupportedFrameworks.Net70Windows7
77
+ var compatibleWindowsTargetWithoutVersion = new NuGetFramework(compatibleVersion.Framework, compatibleVersion.Version, "windows", FrameworkConstants.EmptyVersion);
78
+ matrix[packageFrameworkWithWindowsVersion].Add(compatibleWindowsTargetWithoutVersion);
70
79
  }
71
- );
80
+ }
72
81
 
73
82
  return matrix;
74
83
  }
@@ -72,6 +72,8 @@ namespace NuGetGallery.Frameworks
72
72
 
73
73
  public static readonly NuGetFramework Net60Windows7 = new NuGetFramework(FrameworkIdentifiers.NetCoreApp, Version6, "windows", Version7);
74
74
  public static readonly NuGetFramework Net70Windows7 = new NuGetFramework(FrameworkIdentifiers.NetCoreApp, Version7, "windows", Version7);
75
+ public static readonly NuGetFramework Net80Windows7 = new NuGetFramework(FrameworkIdentifiers.NetCoreApp, Version8, "windows", Version7);
76
+ public static readonly NuGetFramework Net90Windows7 = new NuGetFramework(FrameworkIdentifiers.NetCoreApp, Version9, "windows", Version7);
75
77
 
76
78
  public static readonly IReadOnlyList<NuGetFramework> AllSupportedNuGetFrameworks;
77
79
 
@@ -83,10 +85,10 @@ namespace NuGetGallery.Frameworks
83
85
  Native,
84
86
  Net11, Net2, Net35, Net4, Net403, Net45, Net451, Net452, Net46, Net461, Net462, Net463, Net47, Net471, Net472, Net48, Net481,
85
87
  Net50, Net50Windows,
86
- Net60, Net60Android, Net60Ios, Net60MacCatalyst, Net60MacOs, Net60TvOs, Net60Windows,
87
- Net70, Net70Android, Net70Ios, Net70MacCatalyst, Net70MacOs, Net70TvOs, Net70Windows,
88
- Net80, Net80Android, Net80Ios, Net80MacCatalyst, Net80MacOs, Net80TvOs, Net80Windows,
89
- Net90, Net90Android, Net90Ios, Net90MacCatalyst, Net90MacOs, Net90TvOs, Net90Windows,
88
+ Net60, Net60Android, Net60Ios, Net60MacCatalyst, Net60MacOs, Net60TvOs, Net60Windows, Net60Windows7,
89
+ Net70, Net70Android, Net70Ios, Net70MacCatalyst, Net70MacOs, Net70TvOs, Net70Windows, Net70Windows7,
90
+ Net80, Net80Android, Net80Ios, Net80MacCatalyst, Net80MacOs, Net80TvOs, Net80Windows, Net80Windows7,
91
+ Net90, Net90Android, Net90Ios, Net90MacCatalyst, Net90MacOs, Net90TvOs, Net90Windows, Net90Windows7,
90
92
  NetCore, NetCore45, NetCore451,
91
93
  NetCoreApp10, NetCoreApp11, NetCoreApp20, NetCoreApp21, NetCoreApp22, NetCoreApp30, NetCoreApp31,
92
94
  NetMf,
@@ -86,7 +86,7 @@ internal static partial class PackagesConfigUpdater
86
86
  }
87
87
  }
88
88
 
89
- using (new WebApplicationTargetsConditionPatcher(projectPath))
89
+ using (new SpecialImportsConditionPatcher(projectPath))
90
90
  {
91
91
  RunNugetUpdate(updateArgs, restoreArgs, projectDirectory ?? packagesDirectory, logger);
92
92
  }