File: TargetingPackTests.cs
Web Access
Project: src\src\Framework\test\Microsoft.AspNetCore.App.UnitTests.csproj (Microsoft.AspNetCore.App.UnitTests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using System.IO.Compression;
using System.Reflection;
using System.Reflection.Metadata;
using System.Reflection.PortableExecutable;
using System.Runtime.CompilerServices;
using System.Xml.Linq;
using Microsoft.AspNetCore.InternalTesting;
using NuGet.Versioning;
using Xunit.Abstractions;
 
namespace Microsoft.AspNetCore;
 
public class TargetingPackTests
{
    private readonly string _expectedRid;
    private readonly string _targetingPackTfm;
    private readonly string _targetingPackRoot;
    private readonly ITestOutputHelper _output;
 
    public TargetingPackTests(ITestOutputHelper output)
    {
        _output = output;
        _expectedRid = TestData.GetSharedFxRuntimeIdentifier();
        _targetingPackTfm = TestData.GetDefaultNetCoreTargetFramework();
        _targetingPackRoot = Path.Combine(
            Environment.GetEnvironmentVariable("DOTNET_ROOT"),
            "packs",
            "Microsoft.AspNetCore.App.Ref",
            TestData.GetSharedFxVersion());
    }
 
    [Fact]
    public void TargetingPackContainsListedAssemblies()
    {
        var actualAssemblies = Directory.GetFiles(Path.Combine(_targetingPackRoot, "ref", _targetingPackTfm), "*.dll")
            .Select(Path.GetFileNameWithoutExtension)
            .ToHashSet();
        var listedTargetingPackAssemblies = TestData.ListedTargetingPackAssemblies.ToHashSet();
 
        _output.WriteLine("==== actual assemblies ====");
        _output.WriteLine(string.Join('\n', actualAssemblies.OrderBy(i => i)));
        _output.WriteLine("==== expected assemblies ====");
        _output.WriteLine(string.Join('\n', listedTargetingPackAssemblies.OrderBy(i => i)));
 
        var missing = listedTargetingPackAssemblies.Except(actualAssemblies);
        var unexpected = actualAssemblies.Except(listedTargetingPackAssemblies);
 
        _output.WriteLine("==== missing assemblies from the framework ====");
        _output.WriteLine(string.Join('\n', missing));
        _output.WriteLine("==== unexpected assemblies in the framework ====");
        _output.WriteLine(string.Join('\n', unexpected));
 
        Assert.Empty(missing);
        Assert.Empty(unexpected);
    }
 
    [Fact]
    public void RefAssembliesHaveExpectedAssemblyVersions()
    {
        // Assemblies from this repo and dotnet/runtime don't always have identical assembly versions.
        var repoAssemblies = TestData.GetAspNetCoreTargetingPackDependencies()
            .Split(';', StringSplitOptions.RemoveEmptyEntries)
            .ToHashSet();
 
        var versionStringWithoutPrereleaseTag = TestData.GetMicrosoftNETCoreAppPackageVersion().Split('-', 2)[0];
        var version = Version.Parse(versionStringWithoutPrereleaseTag);
        var aspnetcoreVersionString = TestData.GetSharedFxVersion().Split('-', 2)[0];
        var aspnetcoreVersion = Version.Parse(aspnetcoreVersionString);
 
        IEnumerable<string> dlls = Directory.GetFiles(Path.Combine(_targetingPackRoot, "ref", _targetingPackTfm), "*.dll", SearchOption.AllDirectories);
        Assert.NotEmpty(dlls);
 
        Assert.All(dlls, path =>
        {
            var expectedVersion = repoAssemblies.Contains(Path.GetFileNameWithoutExtension(path)) ?
                aspnetcoreVersion :
                version;
 
            var fileName = Path.GetFileNameWithoutExtension(path);
            var assemblyName = AssemblyName.GetAssemblyName(path);
            using var fileStream = File.OpenRead(path);
            using var peReader = new PEReader(fileStream, PEStreamOptions.Default);
            var reader = peReader.GetMetadataReader(MetadataReaderOptions.Default);
            var assemblyDefinition = reader.GetAssemblyDefinition();
 
            // Assembly versions should all match Major.Minor.0.0
            if (repoAssemblies.Contains(Path.GetFileNameWithoutExtension(path)))
            {
                // We always align major.minor in assemblies and packages.
                Assert.Equal(expectedVersion.Major, assemblyDefinition.Version.Major);
            }
            else
            {
                // ... but dotnet/runtime has a window between package version and (then) assembly version updates.
                Assert.True(expectedVersion.Major == assemblyDefinition.Version.Major ||
                    expectedVersion.Major - 1 == assemblyDefinition.Version.Major,
                    $"Unexpected Major assembly version '{assemblyDefinition.Version.Major}' is neither " +
                        $"{expectedVersion.Major - 1}' nor '{expectedVersion.Major}'.");
            }
 
            Assert.Equal(expectedVersion.Minor, assemblyDefinition.Version.Minor);
            Assert.Equal(0, assemblyDefinition.Version.Build);
            Assert.Equal(0, assemblyDefinition.Version.Revision);
        });
    }
 
    [Fact]
    public void RefAssemblyReferencesHaveExpectedAssemblyVersions()
    {
        IEnumerable<string> dlls = Directory.GetFiles(Path.Combine(_targetingPackRoot, "ref", _targetingPackTfm), "*.dll", SearchOption.AllDirectories);
        Assert.NotEmpty(dlls);
 
        Assert.All(dlls, path =>
        {
            using var fileStream = File.OpenRead(path);
            using var peReader = new PEReader(fileStream, PEStreamOptions.Default);
            var reader = peReader.GetMetadataReader(MetadataReaderOptions.Default);
 
            Assert.All(reader.AssemblyReferences, handle =>
            {
                var reference = reader.GetAssemblyReference(handle);
                var result = (0 == reference.Version.Revision && 0 == reference.Version.Build);
 
                Assert.True(result, $"In {Path.GetFileName(path)}, {reference.GetAssemblyName()} has unexpected version {reference.Version}.");
            });
        });
    }
 
    [Fact]
    public void PackageOverridesContainsCorrectEntries()
    {
        var packageOverridePath = Path.Combine(_targetingPackRoot, "data", "PackageOverrides.txt");
 
        AssertEx.FileExists(packageOverridePath);
 
        var packageOverrideFileLines = File.ReadAllLines(packageOverridePath);
        var runtimeDependencies = TestData.GetRuntimeTargetingPackDependencies()
            .Split(';', StringSplitOptions.RemoveEmptyEntries)
            .ToHashSet();
        var aspnetcoreDependencies = TestData.GetAspNetCoreTargetingPackDependencies()
            .Split(';', StringSplitOptions.RemoveEmptyEntries)
            .ToHashSet();
 
        Assert.Equal(packageOverrideFileLines.Length, runtimeDependencies.Count + aspnetcoreDependencies.Count);
 
        // PackageOverrides versions should remain at Major.Minor.0 while servicing.
        var netCoreAppPackageVersion = TestData.GetMicrosoftNETCoreAppPackageVersion();
        Assert.True(
            NuGetVersion.TryParse(netCoreAppPackageVersion, out var parsedVersion),
            "MicrosoftNETCoreAppPackageVersion must be convertable to a NuGetVersion.");
        if (parsedVersion.Patch != 0 && !parsedVersion.IsPrerelease)
        {
            netCoreAppPackageVersion = $"{parsedVersion.Major}.{parsedVersion.Minor}.0";
        }
 
        var aspNetCoreAppPackageVersion = TestData.GetReferencePackSharedFxVersion();
        Assert.True(
            NuGetVersion.TryParse(aspNetCoreAppPackageVersion, out parsedVersion),
            "ReferencePackSharedFxVersion must be convertable to a NuGetVersion.");
        if (parsedVersion.Patch != 0 && !parsedVersion.IsPrerelease)
        {
            aspNetCoreAppPackageVersion = $"{parsedVersion.Major}.{parsedVersion.Minor}.0";
        }
 
        Assert.All(packageOverrideFileLines, entry =>
        {
            var packageOverrideParts = entry.Split("|");
            Assert.Equal(2, packageOverrideParts.Length);
 
            var packageName = packageOverrideParts[0];
            var packageVersion = packageOverrideParts[1];
 
            if (runtimeDependencies.Contains(packageName))
            {
                Assert.Equal(netCoreAppPackageVersion, packageVersion);
            }
            else if (aspnetcoreDependencies.Contains(packageName))
            {
                Assert.Equal(aspNetCoreAppPackageVersion, packageVersion);
            }
            else
            {
                Assert.Fail($"{packageName} is not a recognized aspNetCore or runtime dependency");
            }
        });
    }
 
    [Fact]
    public void AssembliesAreReferenceAssemblies()
    {
        IEnumerable<string> dlls = Directory.GetFiles(Path.Combine(_targetingPackRoot, "ref"), "*.dll", SearchOption.AllDirectories);
        Assert.NotEmpty(dlls);
 
        Assert.All(dlls, path =>
        {
            var assemblyName = AssemblyName.GetAssemblyName(path);
            using var fileStream = File.OpenRead(path);
            using var peReader = new PEReader(fileStream, PEStreamOptions.Default);
            var reader = peReader.GetMetadataReader(MetadataReaderOptions.Default);
            var assemblyDefinition = reader.GetAssemblyDefinition();
            var hasRefAssemblyAttribute = assemblyDefinition.GetCustomAttributes().Any(attr =>
            {
                var attribute = reader.GetCustomAttribute(attr);
                var attributeConstructor = reader.GetMemberReference((MemberReferenceHandle)attribute.Constructor);
                var attributeType = reader.GetTypeReference((TypeReferenceHandle)attributeConstructor.Parent);
                return reader.StringComparer.Equals(attributeType.Namespace, typeof(ReferenceAssemblyAttribute).Namespace)
                    && reader.StringComparer.Equals(attributeType.Name, nameof(ReferenceAssemblyAttribute));
            });
 
            Assert.True(hasRefAssemblyAttribute, $"{path} should have {nameof(ReferenceAssemblyAttribute)}");
#pragma warning disable SYSLIB0037 // AssemblyName.ProcessorArchitecture is obsolete
            // MSIL and None represent platform neutral assemblies such that reference assemblies can always be loaded.
            Assert.True(assemblyName.ProcessorArchitecture == ProcessorArchitecture.MSIL ||
                assemblyName.ProcessorArchitecture == ProcessorArchitecture.None
                );
#pragma warning restore SYSLIB0037
        });
    }
 
    [Fact]
    public void PlatformManifestListsAllFiles()
    {
        var platformManifestPath = Path.Combine(_targetingPackRoot, "data", "PlatformManifest.txt");
        var expectedAssemblies = TestData.GetSharedFxDependencies()
            .Split(';', StringSplitOptions.RemoveEmptyEntries)
            .Select(i =>
            {
                var fileName = Path.GetFileName(i);
                return fileName.EndsWith(".dll", StringComparison.Ordinal)
                    ? fileName.Substring(0, fileName.Length - 4)
                    : fileName;
            })
            .ToHashSet();
 
        _output.WriteLine("==== file contents ====");
        _output.WriteLine(File.ReadAllText(platformManifestPath));
        _output.WriteLine("==== expected assemblies ====");
        _output.WriteLine(string.Join('\n', expectedAssemblies.OrderBy(i => i)));
 
        AssertEx.FileExists(platformManifestPath);
 
        var manifestFileLines = File.ReadAllLines(platformManifestPath);
 
        var actualAssemblies = manifestFileLines
            .Where(s => !string.IsNullOrEmpty(s))
            .Select(i =>
            {
                var fileName = i.Split('|')[0];
                return fileName.EndsWith(".dll", StringComparison.Ordinal)
                    ? fileName.Substring(0, fileName.Length - 4)
                    : fileName;
            })
            .ToHashSet();
 
        if (!TestData.VerifyAncmBinary())
        {
            actualAssemblies.Remove("aspnetcorev2_inprocess");
            expectedAssemblies.Remove("aspnetcorev2_inprocess");
        }
 
        var missing = expectedAssemblies.Except(actualAssemblies);
        var unexpected = actualAssemblies.Except(expectedAssemblies);
 
        _output.WriteLine("==== missing assemblies from the manifest ====");
        _output.WriteLine(string.Join('\n', missing));
        _output.WriteLine("==== unexpected assemblies in the manifest ====");
        _output.WriteLine(string.Join('\n', unexpected));
 
        Assert.Empty(missing);
        Assert.Empty(unexpected);
 
        Assert.All(manifestFileLines, line =>
        {
            var parts = line.Split('|');
            Assert.Equal(4, parts.Length);
            Assert.Equal("Microsoft.AspNetCore.App.Ref", parts[1]);
            if (parts[2].Length > 0)
            {
                Assert.True(Version.TryParse(parts[2], out _), "Assembly version must be convertable to System.Version");
            }
            Assert.True(Version.TryParse(parts[3], out _), "File version must be convertable to System.Version");
        });
    }
 
    [Fact]
    public void FrameworkListContainsCorrectEntries()
    {
        var frameworkListPath = Path.Combine(_targetingPackRoot, "data", "FrameworkList.xml");
        var expectedAssemblies = TestData.GetTargetingPackDependencies()
            .Split(';', StringSplitOptions.RemoveEmptyEntries)
            .ToHashSet();
        expectedAssemblies.Remove("aspnetcorev2_inprocess");
 
        AssertEx.FileExists(frameworkListPath);
 
        var frameworkListDoc = XDocument.Load(frameworkListPath);
        var frameworkListEntries = frameworkListDoc.Root.Descendants();
        var managedEntries = frameworkListEntries.Where(i => i.Attribute("Type").Value.Equals("Managed", StringComparison.Ordinal));
        var analyzerEntries = frameworkListEntries.Where(i => i.Attribute("Type").Value.Equals("Analyzer", StringComparison.Ordinal));
 
        var analyzersDir = Path.Combine(_targetingPackRoot, "analyzers");
        var expectedAnalyzers = Directory.Exists(analyzersDir) ?
            Directory.GetFiles(analyzersDir, "*.dll", SearchOption.AllDirectories)
            .Select(p => Path.GetFileNameWithoutExtension(p))
            .Where(f => !f.EndsWith(".resources", StringComparison.OrdinalIgnoreCase))
            .ToHashSet() :
            new HashSet<string>();
 
        CompareFrameworkElements(expectedAssemblies, managedEntries, "managed");
        CompareFrameworkElements(expectedAnalyzers, analyzerEntries, "analyzer");
 
        void CompareFrameworkElements(HashSet<string> expectedAssemblyNames, IEnumerable<XElement> actualElements, string type)
        {
            _output.WriteLine($"==== file contents ({type}) ====");
            _output.WriteLine(string.Join('\n', actualElements.Select(i => i.Attribute("AssemblyName").Value).OrderBy(i => i)));
            _output.WriteLine($"==== expected {type} assemblies ====");
            _output.WriteLine(string.Join('\n', expectedAssemblyNames.OrderBy(i => i)));
 
            var actualAssemblyNames = managedEntries
               .Select(i =>
               {
                   var fileName = i.Attribute("AssemblyName").Value;
                   return fileName.EndsWith(".dll", StringComparison.Ordinal)
                       ? fileName.Substring(0, fileName.Length - 4)
                       : fileName;
               })
               .ToHashSet();
 
            var missing = actualAssemblyNames.Except(actualAssemblyNames);
            var unexpected = actualAssemblyNames.Except(expectedAssemblies);
 
            _output.WriteLine($"==== missing {type} assemblies from the framework list ====");
            _output.WriteLine(string.Join('\n', missing));
            _output.WriteLine($"==== unexpected {type} assemblies in the framework list ====");
            _output.WriteLine(string.Join('\n', unexpected));
 
            Assert.Empty(missing);
            Assert.Empty(unexpected);
        }
 
        Assert.All(frameworkListEntries, i =>
        {
            var assemblyPath = i.Attribute("Path").Value;
            var assemblyVersion = i.Attribute("AssemblyVersion").Value;
            var fileVersion = i.Attribute("FileVersion").Value;
 
            Assert.True(Version.TryParse(assemblyVersion, out _), $"{assemblyPath} has assembly version {assemblyVersion}. Assembly version must be convertable to System.Version");
            Assert.True(Version.TryParse(fileVersion, out _), $"{assemblyPath} has file version {fileVersion}. File version must be convertable to System.Version");
        });
    }
 
    [Fact]
    public void FrameworkListContainsCorrectPaths()
    {
        var frameworkListPath = Path.Combine(_targetingPackRoot, "data", "FrameworkList.xml");
 
        AssertEx.FileExists(frameworkListPath);
 
        var frameworkListDoc = XDocument.Load(frameworkListPath);
        var frameworkListEntries = frameworkListDoc.Root.Descendants();
 
        var packageFolder = SkipOnHelixAttribute.OnHelix() ?
            Environment.GetEnvironmentVariable("HELIX_WORKITEM_ROOT") :
            TestData.GetPackagesFolder();
        var targetingPackPath = Path.Combine(
            packageFolder, "Microsoft.AspNetCore.App.Ref." + TestData.GetSharedFxVersion() + ".nupkg");
        AssertEx.FileExists(targetingPackPath);
 
        ZipArchive archive = ZipFile.OpenRead(targetingPackPath);
 
        var actualPaths = archive.Entries
            .Where(i => i.FullName.EndsWith(".dll", StringComparison.Ordinal) && !i.FullName.EndsWith(".resources.dll", StringComparison.Ordinal))
            .Select(i => i.FullName).ToHashSet();
 
        var expectedPaths = frameworkListEntries.Select(i => i.Attribute("Path").Value).ToHashSet();
 
        _output.WriteLine("==== package contents ====");
        _output.WriteLine(string.Join('\n', actualPaths.OrderBy(i => i)));
        _output.WriteLine("==== expected assemblies ====");
        _output.WriteLine(string.Join('\n', expectedPaths.OrderBy(i => i)));
 
        var missing = expectedPaths.Except(actualPaths);
        var unexpected = actualPaths.Except(expectedPaths);
 
        _output.WriteLine("==== missing assemblies from the runtime list ====");
        _output.WriteLine(string.Join('\n', missing));
        _output.WriteLine("==== unexpected assemblies in the runtime list ====");
        _output.WriteLine(string.Join('\n', unexpected));
 
        Assert.Empty(missing);
        Assert.Empty(unexpected);
    }
 
    [Fact]
    public void FrameworkListContainsAnalyzerLanguage()
    {
        var frameworkListPath = Path.Combine(_targetingPackRoot, "data", "FrameworkList.xml");
 
        AssertEx.FileExists(frameworkListPath);
 
        var frameworkListDoc = XDocument.Load(frameworkListPath);
        var frameworkListEntries = frameworkListDoc.Root.Descendants();
 
        var analyzerEntries = frameworkListEntries.Where(i => i.Attribute("Type").Value.Equals("Analyzer", StringComparison.Ordinal));
 
        Assert.All(analyzerEntries, analyzerEntry =>
        {
            var actualLanguage = analyzerEntry.Attribute("Language")?.Value;
            var assemblyPath = analyzerEntry.Attribute("Path").Value;
 
            string expectedLanguage = Path.GetFileName(Path.GetDirectoryName(assemblyPath));
 
            if (expectedLanguage.Equals("dotnet", StringComparison.OrdinalIgnoreCase))
            {
                expectedLanguage = null;
            }
 
            Assert.Equal(expectedLanguage, actualLanguage);
        });
    }
}