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.
- checksums.yaml +4 -4
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/DependencyDiscovery.props +4 -1
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/DependencyDiscovery.targets +19 -4
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Discover/DiscoveryWorker.cs +85 -81
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Discover/SdkProjectDiscovery.cs +177 -28
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/FrameworkChecker/FrameworkCompatibilityService.cs +15 -6
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/FrameworkChecker/SupportedFrameworks.cs +6 -4
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/PackagesConfigUpdater.cs +1 -1
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/{WebApplicationTargetsConditionPatcher.cs → SpecialImportsConditionPatcher.cs} +18 -11
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Updater/XmlFilePreAndPostProcessor.cs +26 -11
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/Utilities/MSBuildHelper.cs +37 -11
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Analyze/AnalyzeWorkerTests.cs +54 -0
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Discover/DiscoveryWorkerTests.PackagesConfig.cs +54 -0
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Discover/DiscoveryWorkerTests.Project.cs +94 -0
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/FrameworkChecker/FrameworkCompatibilityServiceFacts.cs +1 -1
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Update/SpecialFilePatcherTests.cs +99 -0
- data/helpers/lib/NuGetUpdater/NuGetUpdater.Core.Test/Update/UpdateWorkerTests.PackageReference.cs +38 -0
- data/lib/dependabot/nuget/file_parser.rb +22 -19
- metadata +7 -6
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: aa40dad694a6849be3c9da853f6d075f0db1f44f9cc9338e07586405e5f02bcc
|
4
|
+
data.tar.gz: 0f7ddac2d69e9a1dce70c7000b3159c0daf337816c93b9498678d58fa3753afc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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="$(
|
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
|
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="
|
20
|
+
<Target Name="_DiscoverDependencies" DependsOnTargets="$(_DiscoverDependenciesDependsOn)">
|
10
21
|
<!--
|
11
|
-
|
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
|
-
|
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
|
-
|
336
|
-
|
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
|
-
|
343
|
-
|
344
|
-
if (
|
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
|
-
|
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
|
-
|
350
|
-
if (projectResult.FilePath == relativeProjectPath && packagesConfigResult is not null)
|
345
|
+
foreach (var projectResult in projectResults)
|
351
346
|
{
|
352
|
-
|
353
|
-
|
354
|
-
|
347
|
+
if (results.ContainsKey(projectResult.FilePath))
|
348
|
+
{
|
349
|
+
continue;
|
350
|
+
}
|
355
351
|
|
356
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
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 =
|
403
|
-
|
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
|
-
|
369
|
+
|
370
|
+
if (packagesConfigResult is not null)
|
412
371
|
{
|
413
|
-
//
|
414
|
-
results
|
372
|
+
// we might have to merge this dependency with some others
|
373
|
+
if (results.TryGetValue(relativeProjectPath, out var existingProjectDiscovery))
|
415
374
|
{
|
416
|
-
|
417
|
-
|
418
|
-
|
419
|
-
|
420
|
-
|
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
|
-
//
|
76
|
+
// projectPath tfm packageName packageVersion
|
73
77
|
|
74
|
-
Dictionary<string, HashSet<string
|
75
|
-
//
|
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
|
-
//
|
85
|
+
// projectPath tfm packageName packageVersion
|
79
86
|
|
80
87
|
Dictionary<string, Dictionary<string, string>> resolvedProperties = new(PathComparer.Instance);
|
81
|
-
//
|
88
|
+
// projectPath propertyName propertyValue
|
82
89
|
|
83
90
|
Dictionary<string, HashSet<string>> importedFiles = new(PathComparer.Instance);
|
84
|
-
//
|
91
|
+
// projectPath importedFiles
|
85
92
|
|
86
93
|
Dictionary<string, HashSet<string>> referencedProjects = new(PathComparer.Instance);
|
87
|
-
//
|
94
|
+
// projectPath referencedProjects
|
88
95
|
|
89
96
|
Dictionary<string, HashSet<string>> additionalFiles = new(PathComparer.Instance);
|
90
|
-
//
|
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 =
|
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(""
|
138
|
+
if (stdOut.Contains("_DependencyDiscovery_LegacyProjects::UseTemporaryProject"))
|
131
139
|
{
|
132
|
-
//
|
133
|
-
|
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 =
|
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
|
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
|
-
|
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
|
-
|
393
|
-
|
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
|
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
|
420
|
-
|
421
|
-
if (doRemoveOperation)
|
525
|
+
var tfm = GetPropertyValueFromProjectEvaluation(projectEvaluation, "TargetFramework");
|
526
|
+
if (tfm is not null)
|
422
527
|
{
|
423
|
-
topLevelPackages.
|
424
|
-
|
528
|
+
var topLevelPackages = topLevelPackagesPerProject.GetOrAdd(projectEvaluation.ProjectFile, () => new(StringComparer.OrdinalIgnoreCase));
|
529
|
+
var topLevelPackagesPerTfm = topLevelPackages.GetOrAdd(tfm, () => new(StringComparer.OrdinalIgnoreCase));
|
425
530
|
|
426
|
-
|
427
|
-
|
428
|
-
|
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)
|
data/helpers/lib/NuGetUpdater/NuGetUpdater.Core/FrameworkChecker/FrameworkCompatibilityService.cs
CHANGED
@@ -61,14 +61,23 @@ public class FrameworkCompatibilityService
|
|
61
61
|
}
|
62
62
|
}
|
63
63
|
|
64
|
-
|
65
|
-
|
66
|
-
|
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
|
-
|
69
|
-
|
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
|
89
|
+
using (new SpecialImportsConditionPatcher(projectPath))
|
90
90
|
{
|
91
91
|
RunNugetUpdate(updateArgs, restoreArgs, projectDirectory ?? packagesDirectory, logger);
|
92
92
|
}
|