From 9d22672b4681fc7e01a00790bbd58d9f0e112968 Mon Sep 17 00:00:00 2001 From: Justin Emgarten Date: Mon, 6 Feb 2017 23:09:39 -0800 Subject: [PATCH] Handle duplicate MSBuild items during restore This fix improves the robustness of Restore, allowing it to gracefully handle when MSBuild generates duplicate items for the same project. Fixes https://github.com/NuGet/Home/issues/4316 --- .../Utility/MSBuildRestoreUtility.cs | 9 +- .../MSBuildRestoreUtilityTests.cs | 163 ++++++++++++++++++ 2 files changed, 170 insertions(+), 2 deletions(-) diff --git a/src/NuGet.Core/NuGet.Commands/RestoreCommand/Utility/MSBuildRestoreUtility.cs b/src/NuGet.Core/NuGet.Commands/RestoreCommand/Utility/MSBuildRestoreUtility.cs index dd2d0b1342d..8dd9fcb88fc 100644 --- a/src/NuGet.Core/NuGet.Commands/RestoreCommand/Utility/MSBuildRestoreUtility.cs +++ b/src/NuGet.Core/NuGet.Commands/RestoreCommand/Utility/MSBuildRestoreUtility.cs @@ -62,7 +62,9 @@ public static DependencyGraphSpec GetDependencySpec(IEnumerable it } // Add projects - foreach (var spec in itemsById.Values.Select(GetPackageSpec)) + var validProjectSpecs = itemsById.Values.Select(GetPackageSpec).Where(e => e != null); + + foreach (var spec in validProjectSpecs) { if (spec.RestoreMetadata.ProjectStyle == ProjectStyle.PackageReference || spec.RestoreMetadata.ProjectStyle == ProjectStyle.ProjectJson @@ -118,7 +120,10 @@ public static PackageSpec GetPackageSpec(IEnumerable items) PackageSpec result = null; - var specItem = items.SingleOrDefault(item => + // There should only be one ProjectSpec per project in the item set, + // but if multiple do appear take only the first one in an effort + // to handle this gracefully. + var specItem = items.FirstOrDefault(item => "projectSpec".Equals(item.GetProperty("Type"), StringComparison.OrdinalIgnoreCase)); diff --git a/test/NuGet.Core.Tests/NuGet.Commands.Test/MSBuildRestoreUtilityTests.cs b/test/NuGet.Core.Tests/NuGet.Commands.Test/MSBuildRestoreUtilityTests.cs index 4160c5ee035..bfbf19958e6 100644 --- a/test/NuGet.Core.Tests/NuGet.Commands.Test/MSBuildRestoreUtilityTests.cs +++ b/test/NuGet.Core.Tests/NuGet.Commands.Test/MSBuildRestoreUtilityTests.cs @@ -884,6 +884,169 @@ public void MSBuildRestoreUtility_GetPackageSpec_NetCore_Conditionals() } } + [Fact] + public void MSBuildRestoreUtility_GetPackageSpec_NetCore_VerifyDuplicateItemsAreIgnored() + { + using (var workingDir = TestDirectory.Create()) + { + // Arrange + var projectRoot = Path.Combine(workingDir, "a"); + var projectPath = Path.Combine(projectRoot, "a.csproj"); + var outputPath = Path.Combine(projectRoot, "obj"); + + var items = new List>(); + + var specItem = new Dictionary() + { + { "Type", "ProjectSpec" }, + { "ProjectName", "a" }, + { "ProjectStyle", "PackageReference" }, + { "OutputPath", outputPath }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectPath", projectPath }, + { "TargetFrameworks", "net46;netstandard1.6" }, + { "CrossTargeting", "true" }, + }; + + // Add each item twice + items.Add(specItem); + items.Add(specItem); + + // A -> B + var projectRef = new Dictionary() + { + { "Type", "ProjectReference" }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectReferenceUniqueName", "AA2C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectPath", "otherProjectPath.csproj" }, + { "TargetFrameworks", "netstandard1.6" }, + { "CrossTargeting", "true" }, + }; + + items.Add(projectRef); + items.Add(projectRef); + + // Package references + // A netstandard1.6 -> Z + var packageRef1 = new Dictionary() + { + { "Type", "Dependency" }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "Id", "z" }, + { "VersionRange", "2.0.0" }, + { "TargetFrameworks", "netstandard1.6" }, + { "CrossTargeting", "true" }, + }; + + items.Add(packageRef1); + items.Add(packageRef1); + + // B ALL -> Y + var packageRef2 = new Dictionary() + { + { "Type", "Dependency" }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "Id", "y" }, + { "VersionRange", "[1.0.0]" }, + { "TargetFrameworks", "netstandard1.6;net46" }, + { "CrossTargeting", "true" }, + }; + + items.Add(packageRef2); + items.Add(packageRef2); + + // Framework assembly + var frameworkAssembly = new Dictionary() + { + { "Type", "FrameworkAssembly" }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "Id", "System.IO" }, + { "TargetFrameworks", "net46" }, + { "CrossTargeting", "true" }, + }; + + items.Add(frameworkAssembly); + items.Add(frameworkAssembly); + + // TFM info + var tfmInfo = new Dictionary() + { + { "Type", "TargetFrameworkInformation" }, + { "ProjectUniqueName", "482C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "PackageTargetFallback", "portable-net45+win8;dnxcore50;;" }, + { "TargetFramework", "netstandard16" } + }; + + items.Add(tfmInfo); + items.Add(tfmInfo); + + var wrappedItems = items.Select(CreateItems).ToList(); + + // Act + var dgSpec = MSBuildRestoreUtility.GetDependencySpec(wrappedItems); + var projectSpec = dgSpec.Projects.Single(e => e.Name == "a"); + + // Assert + Assert.Equal(0, projectSpec.Dependencies.Count); + Assert.Equal(1, dgSpec.Projects.Count); + Assert.Equal("System.IO|y", string.Join("|", projectSpec.GetTargetFramework(NuGetFramework.Parse("net46")).Dependencies.Select(e => e.Name))); + Assert.Equal("z|y", string.Join("|", projectSpec.GetTargetFramework(NuGetFramework.Parse("netstandard1.6")).Dependencies.Select(e => e.Name))); + Assert.Equal(2, projectSpec.RestoreMetadata.TargetFrameworks.Count); + } + } + + [Fact] + public void MSBuildRestoreUtility_GetPackageSpec_NetCore_IgnoreBadItemWithMismatchedIds() + { + using (var workingDir = TestDirectory.Create()) + { + // Arrange + var projectRoot = Path.Combine(workingDir, "a"); + var projectPath = Path.Combine(projectRoot, "a.csproj"); + var outputPath = Path.Combine(projectRoot, "obj"); + + var items = new List>(); + + var specItem = new Dictionary() + { + { "Type", "ProjectSpec" }, + { "ProjectName", "a" }, + { "ProjectStyle", "PackageReference" }, + { "OutputPath", outputPath }, + { "ProjectUniqueName", "AA2C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectPath", projectPath }, + { "TargetFrameworks", "net46;netstandard1.6" }, + { "CrossTargeting", "true" }, + }; + + items.Add(specItem); + + // A -> B + var projectRef = new Dictionary() + { + { "Type", "ProjectReference" }, + + // This ID does not match the project! + { "ProjectUniqueName", "BB2C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectReferenceUniqueName", "CC2C20DE-DFF9-4BD0-B90A-BD3201AA351A" }, + { "ProjectPath", "otherProjectPath.csproj" }, + { "TargetFrameworks", "netstandard1.6" }, + { "CrossTargeting", "true" }, + }; + + items.Add(projectRef); + + var wrappedItems = items.Select(CreateItems).ToList(); + + // Act + var dgSpec = MSBuildRestoreUtility.GetDependencySpec(wrappedItems); + var projectSpec = dgSpec.Projects.Single(e => e.Name == "a"); + + // Assert + Assert.NotNull(projectSpec); + } + } + [Fact] public void MSBuildRestoreUtility_GetPackageSpec_UAP_P2P() {