// Copyright (c) .NET Foundation and contributors. All rights reserved. // Licensed under the MIT license. See LICENSE file in the project root for full license information. using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Runtime.InteropServices; using System.Xml.Linq; using FluentAssertions; using Microsoft.DotNet.TestFramework; using Microsoft.DotNet.Tools.Test.Utilities; using Xunit; namespace EndToEnd.Tests { public class ProjectBuildTests : TestBase { [Fact] public void ItCanNewRestoreBuildRunCleanMSBuildProject() { var directory = TestAssets.CreateTestDirectory(); string projectDirectory = directory.FullName; string newArgs = "console --no-restore"; new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); string projectPath = Path.Combine(projectDirectory, directory.Name + ".csproj"); var project = XDocument.Load(projectPath); var ns = project.Root.Name.Namespace; project.Root.Element(ns + "PropertyGroup") .Element(ns + "TargetFramework").Value = TestAssetInfo.currentTfm; project.Save(projectPath); new RestoreCommand() .WithWorkingDirectory(projectDirectory) .Execute() .Should().Pass(); new BuildCommand() .WithWorkingDirectory(projectDirectory) .Execute() .Should().Pass(); var runCommand = new RunCommand() .WithWorkingDirectory(projectDirectory) .ExecuteWithCapturedOutput() .Should().Pass().And.HaveStdOutContaining("Hello, World!"); var binDirectory = new DirectoryInfo(projectDirectory).Sub("bin"); binDirectory.Should().HaveFilesMatching("*.dll", SearchOption.AllDirectories); new CleanCommand() .WithWorkingDirectory(projectDirectory) .Execute() .Should().Pass(); binDirectory.Should().NotHaveFilesMatching("*.dll", SearchOption.AllDirectories); } [Fact(Skip ="The current aspnet runtime is built against an 8.0 core runtime")] public void ItCanRunAnAppUsingTheWebSdk() { var directory = TestAssets.CreateTestDirectory(); string projectDirectory = directory.FullName; string newArgs = "console --no-restore"; new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); string projectPath = Path.Combine(projectDirectory, directory.Name + ".csproj"); var project = XDocument.Load(projectPath); var ns = project.Root.Name.Namespace; project.Root.Attribute("Sdk").Value = "Microsoft.NET.Sdk.Web"; project.Root.Element(ns + "PropertyGroup") .Element(ns + "TargetFramework").Value = TestAssetInfo.currentTfm; project.Save(projectPath); new BuildCommand() .WithWorkingDirectory(projectDirectory) .Execute() .Should().Pass(); var runCommand = new RunCommand() .WithWorkingDirectory(projectDirectory) .ExecuteWithCapturedOutput() .Should().Pass().And.HaveStdOutContaining("Hello, World!"); } [WindowsOnlyTheory] // [InlineData("net6.0", true)] // [InlineData("net6.0", false)] [InlineData("current", true)] [InlineData("current", false)] public void ItCanPublishArm64Winforms(string TargetFramework, bool selfContained) { DirectoryInfo directory = TestAssets.CreateTestDirectory(); string projectDirectory = directory.FullName; string TargetFrameworkParameter = ""; if (TargetFramework != "current") { TargetFrameworkParameter = $"-f {TargetFramework}"; } string newArgs = $"winforms {TargetFrameworkParameter} --no-restore"; new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); string selfContainedArgs = selfContained ? " --self-contained" : ""; string publishArgs = "-r win-arm64" + selfContainedArgs; new PublishCommand() .WithWorkingDirectory(projectDirectory) .Execute(publishArgs) .Should().Pass(); var selfContainedPublishDir = new DirectoryInfo(projectDirectory) .Sub("bin").Sub(TargetFramework != "current" ? "Debug" : "Release").GetDirectories().FirstOrDefault() .Sub("win-arm64").Sub("publish"); if (selfContained) { selfContainedPublishDir.Should().HaveFilesMatching("System.Windows.Forms.dll", SearchOption.TopDirectoryOnly); } selfContainedPublishDir.Should().HaveFilesMatching($"{directory.Name}.dll", SearchOption.TopDirectoryOnly); } [WindowsOnlyTheory] // [InlineData("net6.0", true)] // [InlineData("net6.0", false)] [InlineData("current", true)] [InlineData("current", false)] public void ItCanPublishArm64Wpf(string TargetFramework, bool selfContained) { DirectoryInfo directory = TestAssets.CreateTestDirectory(); string projectDirectory = directory.FullName; string TargetFrameworkParameter = ""; if (TargetFramework != "current") { TargetFrameworkParameter = $"-f {TargetFramework}"; } string newArgs = $"wpf {TargetFrameworkParameter} --no-restore"; new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); string selfContainedArgs = selfContained ? " --self-contained" : ""; string publishArgs = "-r win-arm64" + selfContainedArgs; new PublishCommand() .WithWorkingDirectory(projectDirectory) .Execute(publishArgs) .Should().Pass(); var selfContainedPublishDir = new DirectoryInfo(projectDirectory) .Sub("bin").Sub(TargetFramework != "current" ? "Debug" : "Release").GetDirectories().FirstOrDefault() .Sub("win-arm64").Sub("publish"); if (selfContained) { selfContainedPublishDir.Should().HaveFilesMatching("PresentationCore.dll", SearchOption.TopDirectoryOnly); selfContainedPublishDir.Should().HaveFilesMatching("PresentationNative_*.dll", SearchOption.TopDirectoryOnly); } selfContainedPublishDir.Should().HaveFilesMatching($"{directory.Name}.dll", SearchOption.TopDirectoryOnly); } [Theory] // microsoft.dotnet.common.projectemplates templates [InlineData("console")] [InlineData("console", "C#")] [InlineData("console", "VB")] [InlineData("console", "F#")] [InlineData("classlib")] [InlineData("classlib", "C#")] [InlineData("classlib", "VB")] [InlineData("classlib", "F#")] [InlineData("mstest")] [InlineData("nunit")] [InlineData("web")] [InlineData("mvc")] public void ItCanBuildTemplates(string templateName, string language = "") { TestTemplateCreateAndBuild(templateName, language: language); } /// /// The test checks if dotnet new shows curated list correctly after the SDK installation and template insertion. /// [Fact] public void DotnetNewShowsCuratedListCorrectly() { string locale = System.Threading.Thread.CurrentThread.CurrentUICulture.Name; if (!string.IsNullOrWhiteSpace(locale) && !locale.StartsWith("en", StringComparison.OrdinalIgnoreCase)) { Console.WriteLine($"[{nameof(DotnetNewShowsCuratedListCorrectly)}] CurrentUICulture: {locale}"); Console.WriteLine($"[{nameof(DotnetNewShowsCuratedListCorrectly)}] Test is skipped as it supports only 'en' or invariant culture."); return; } string expectedOutput = @"[\-\s]+ [\w \.]+webapp,razor\s+\[C#\][\w\ \/]+ [\w \.]+classlib\s+\[C#\],F#,VB[\w\ \/]+ [\w \.]+console\s+\[C#\],F#,VB[\w\ \/]+ "; if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows)) { expectedOutput += @"[\w \.]+winforms\s+\[C#\],VB[\w\ \/]+ [\w \.]+\wpf\s+\[C#\],VB[\w\ \/]+ "; } //list should end with new line expectedOutput += Environment.NewLine; new NewCommandShim() .Execute() .Should().Pass() .And.HaveStdOutMatching(expectedOutput); } [Theory] // microsoft.dotnet.common.itemtemplates templates [InlineData("globaljson")] [InlineData("nugetconfig")] [InlineData("webconfig")] [InlineData("gitignore")] [InlineData("tool-manifest")] [InlineData("sln")] public void ItCanCreateItemTemplate(string templateName) { DirectoryInfo directory = TestAssets.CreateTestDirectory(identifier: templateName); string projectDirectory = directory.FullName; string newArgs = $"{templateName}"; new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); //check if the template created files Assert.True(directory.Exists); Assert.True(directory.EnumerateFileSystemInfos().Any()); // delete test directory for some tests so we aren't leaving behind non-compliant nuget files if (templateName.Equals("nugetconfig")) { directory.Delete(true); } } [Theory] // microsoft.dotnet.common.itemtemplates templates [InlineData("class")] [InlineData("struct")] [InlineData("enum")] [InlineData("record")] [InlineData("interface")] [InlineData("class", "C#")] [InlineData("class", "VB")] [InlineData("struct", "VB")] [InlineData("enum", "VB")] [InlineData("interface", "VB")] public void ItCanCreateItemTemplateWithProjectRestriction(string templateName, string language = "") { var languageExtensionMap = new Dictionary() { { "", ".cs" }, { "C#", ".cs" }, { "VB", ".vb" } }; DirectoryInfo directory = InstantiateProjectTemplate("classlib", language, withNoRestore: false); string projectDirectory = directory.FullName; string expectedItemName = $"TestItem_{templateName}"; string newArgs = $"{templateName} --name {expectedItemName} --debug:ephemeral-hive"; if (!string.IsNullOrWhiteSpace(language)) { newArgs += $" --language {language}"; } new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); //check if the template created files Assert.True(directory.Exists); Assert.True(directory.EnumerateFileSystemInfos().Any()); Assert.True(directory.GetFile($"{expectedItemName}.{languageExtensionMap[language]}") != null); } [WindowsOnlyTheory] [InlineData("wpf")] [InlineData("winforms")] public void ItCanBuildDesktopTemplates(string templateName) { TestTemplateCreateAndBuild(templateName); } [WindowsOnlyTheory] [InlineData("wpf")] public void ItCanBuildDesktopTemplatesSelfContained(string templateName) { TestTemplateCreateAndBuild(templateName, selfContained: true); } [Theory] [InlineData("web")] [InlineData("console")] public void ItCanBuildTemplatesSelfContained(string templateName) { TestTemplateCreateAndBuild(templateName, selfContained: true); } /// /// The test checks if the template creates the template for correct framework by default. /// For .NET 6 the templates should create the projects targeting net6.0 /// [Theory] [InlineData("console")] [InlineData("console", "C#")] [InlineData("console", "VB")] [InlineData("console", "F#")] [InlineData("classlib")] [InlineData("classlib", "C#")] [InlineData("classlib", "VB")] [InlineData("classlib", "F#")] [InlineData("worker")] [InlineData("worker", "C#")] [InlineData("worker", "F#")] [InlineData("mstest")] [InlineData("mstest", "C#")] [InlineData("mstest", "VB")] [InlineData("mstest", "F#")] [InlineData("nunit")] [InlineData("nunit", "C#")] [InlineData("nunit", "VB")] [InlineData("nunit", "F#")] [InlineData("xunit")] [InlineData("xunit", "C#")] [InlineData("xunit", "VB")] [InlineData("xunit", "F#")] [InlineData("blazorwasm")] [InlineData("web")] [InlineData("web", "C#")] [InlineData("web", "F#")] [InlineData("mvc")] [InlineData("mvc", "C#")] [InlineData("mvc", "F#")] [InlineData("webapi")] [InlineData("webapi", "C#")] [InlineData("webapi", "F#")] [InlineData("webapp")] [InlineData("razorclasslib")] public void ItCanCreateAndBuildTemplatesWithDefaultFramework(string templateName, string language = "") { string framework = DetectExpectedDefaultFramework(templateName); TestTemplateCreateAndBuild(templateName, selfContained: false, language: language, framework: framework); } /// /// [Windows only tests] /// The test checks if the template creates the template for correct framework by default. /// For .NET 6 the templates should create the projects targeting net6.0. /// [WindowsOnlyTheory] [InlineData("wpf")] [InlineData("wpf", "C#")] [InlineData("wpf", "VB")] [InlineData("wpflib")] [InlineData("wpflib", "C#")] [InlineData("wpflib", "VB")] [InlineData("wpfcustomcontrollib")] [InlineData("wpfcustomcontrollib", "C#")] [InlineData("wpfcustomcontrollib", "VB")] [InlineData("wpfusercontrollib")] [InlineData("wpfusercontrollib", "C#")] [InlineData("wpfusercontrollib", "VB")] [InlineData("winforms")] [InlineData("winforms", "C#")] [InlineData("winforms", "VB")] [InlineData("winformslib")] [InlineData("winformslib", "C#")] [InlineData("winformslib", "VB")] [InlineData("winformscontrollib")] [InlineData("winformscontrollib", "C#")] [InlineData("winformscontrollib", "VB")] public void ItCanCreateAndBuildTemplatesWithDefaultFramework_Windows(string templateName, string language = "") { string framework = DetectExpectedDefaultFramework(templateName); TestTemplateCreateAndBuild(templateName, selfContained: false, language: language, framework: $"{framework}-windows"); } /// /// [project is not built on linux-musl] /// The test checks if the template creates the template for correct framework by default. /// For .NET 6 the templates should create the projects targeting net6.0. /// [Theory] [InlineData("grpc")] public void ItCanCreateAndBuildTemplatesWithDefaultFramework_DisableBuildOnLinuxMusl(string templateName) { string framework = DetectExpectedDefaultFramework(templateName); if (RuntimeInformation.RuntimeIdentifier.StartsWith("linux-musl")) { TestTemplateCreateAndBuild(templateName, build: false, framework: framework); } else { TestTemplateCreateAndBuild(templateName, selfContained: true, framework: framework); } } private static string DetectExpectedDefaultFramework(string template = "") { string dotnetFolder = Path.GetDirectoryName(RepoDirectoriesProvider.DotnetUnderTest); string[] runtimeFolders = Directory.GetDirectories(Path.Combine(dotnetFolder, "shared", "Microsoft.NETCore.App")); int latestMajorVersion = runtimeFolders.Select(folder => int.Parse(Path.GetFileName(folder).Split('.').First())).Max(); if (latestMajorVersion == 9) { // TODO: This block need to be updated when every template updates their default tfm. // Currently winforms updated their default templates target but not others. if (template.StartsWith("mstest") || template.StartsWith("winforms") || template.StartsWith("wpf") || template.StartsWith("web") || template.StartsWith("razor") || template.StartsWith("blazor") || template.StartsWith("mvc") || template.StartsWith("worker") || template.StartsWith("grpc") || template.StartsWith("classlib") || template.StartsWith("console") || template.StartsWith("nunit") || template.StartsWith("xunit")) { return $"net8.0"; } return $"net{latestMajorVersion}.0"; } throw new Exception("Unsupported version of SDK"); } private static void TestTemplateCreateAndBuild(string templateName, bool build = true, bool selfContained = false, string language = "", string framework = "", bool deleteTestDirectory = false) { DirectoryInfo directory = InstantiateProjectTemplate(templateName, language); string projectDirectory = directory.FullName; if (!string.IsNullOrWhiteSpace(framework)) { //check if MSBuild TargetFramework property for *proj is set to expected framework string expectedExtension = language switch { "C#" => "*.csproj", "F#" => "*.fsproj", "VB" => "*.vbproj", _ => "*.csproj" }; string projectFile = Directory.GetFiles(projectDirectory, expectedExtension).Single(); XDocument projectXml = XDocument.Load(projectFile); XNamespace ns = projectXml.Root.Name.Namespace; Assert.Equal(framework, projectXml.Root.Element(ns + "PropertyGroup").Element(ns + "TargetFramework").Value); } if (build) { string buildArgs = selfContained ? $"-r {RuntimeInformation.RuntimeIdentifier} --self-contained" : ""; if (!string.IsNullOrWhiteSpace(framework)) { buildArgs += $" --framework {framework}"; } // Remove this (or formalize it) after https://github.com/dotnet/installer/issues/12479 is resolved. if (language == "F#") { buildArgs += $" /p:_NETCoreSdkIsPreview=true"; } string dotnetRoot = Path.GetDirectoryName(RepoDirectoriesProvider.DotnetUnderTest); new BuildCommand() .WithEnvironmentVariable("PATH", dotnetRoot) // override PATH since razor rely on PATH to find dotnet .WithWorkingDirectory(projectDirectory) .Execute(buildArgs) .Should().Pass(); } // delete test directory for some tests so we aren't leaving behind non-compliant package files if (deleteTestDirectory) { directory.Delete(true); } } private static DirectoryInfo InstantiateProjectTemplate(string templateName, string language = "", bool withNoRestore = true) { DirectoryInfo directory = TestAssets.CreateTestDirectory( identifier: string.IsNullOrWhiteSpace(language) ? templateName : $"{templateName}[{language}]"); string projectDirectory = directory.FullName; string newArgs = $"{templateName} --debug:ephemeral-hive {(withNoRestore ? "--no-restore" : "")}"; if (!string.IsNullOrWhiteSpace(language)) { newArgs += $" --language {language}"; } new NewCommandShim() .WithWorkingDirectory(projectDirectory) .Execute(newArgs) .Should().Pass(); return directory; } } }