Build.cs 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. using System;
  2. using System.Diagnostics;
  3. using System.IO;
  4. using System.IO.Compression;
  5. using System.Linq;
  6. using System.Runtime.InteropServices;
  7. using System.Threading.Tasks;
  8. using System.Xml.Linq;
  9. using Nuke.Common;
  10. using Nuke.Common.Tooling;
  11. using Nuke.Common.Tools.DotNet;
  12. using Nuke.Common.Tools.Npm;
  13. using static Nuke.Common.EnvironmentInfo;
  14. using static Nuke.Common.IO.FileSystemTasks;
  15. using static Nuke.Common.IO.PathConstruction;
  16. using static Nuke.Common.Tools.MSBuild.MSBuildTasks;
  17. using static Nuke.Common.Tools.DotNet.DotNetTasks;
  18. using static Nuke.Common.Tools.Xunit.XunitTasks;
  19. using static Nuke.Common.Tools.VSWhere.VSWhereTasks;
  20. using static Serilog.Log;
  21. using MicroCom.CodeGenerator;
  22. using Nuke.Common.IO;
  23. /*
  24. Before editing this file, install support plugin for your IDE,
  25. running and debugging a particular target (optionally without deps) would be way easier
  26. ReSharper/Rider - https://plugins.jetbrains.com/plugin/10803-nuke-support
  27. VSCode - https://marketplace.visualstudio.com/items?itemName=nuke.support
  28. */
  29. partial class Build : NukeBuild
  30. {
  31. BuildParameters Parameters { get; set; }
  32. [PackageExecutable("Microsoft.DotNet.ApiCompat.Tool", "Microsoft.DotNet.ApiCompat.Tool.dll", Framework = "net6.0")]
  33. Tool ApiCompatTool;
  34. [PackageExecutable("Microsoft.DotNet.GenAPI.Tool", "Microsoft.DotNet.GenAPI.Tool.dll", Framework = "net8.0")]
  35. Tool ApiGenTool;
  36. protected override void OnBuildInitialized()
  37. {
  38. Parameters = new BuildParameters(this, ScheduledTargets.Contains(BuildToNuGetCache));
  39. Information("Building version {0} of Avalonia ({1}) using version {2} of Nuke.",
  40. Parameters.Version,
  41. Parameters.Configuration,
  42. typeof(NukeBuild).Assembly.GetName().Version.ToString());
  43. if (Parameters.IsLocalBuild)
  44. {
  45. Information("Repository Name: " + Parameters.RepositoryName);
  46. Information("Repository Branch: " + Parameters.RepositoryBranch);
  47. }
  48. Information("Configuration: " + Parameters.Configuration);
  49. Information("IsLocalBuild: " + Parameters.IsLocalBuild);
  50. Information("IsRunningOnUnix: " + Parameters.IsRunningOnUnix);
  51. Information("IsRunningOnWindows: " + Parameters.IsRunningOnWindows);
  52. Information("IsRunningOnAzure:" + Parameters.IsRunningOnAzure);
  53. Information("IsPullRequest: " + Parameters.IsPullRequest);
  54. Information("IsMainRepo: " + Parameters.IsMainRepo);
  55. Information("IsMasterBranch: " + Parameters.IsMasterBranch);
  56. Information("IsReleaseBranch: " + Parameters.IsReleaseBranch);
  57. Information("IsReleasable: " + Parameters.IsReleasable);
  58. Information("IsMyGetRelease: " + Parameters.IsMyGetRelease);
  59. Information("IsNuGetRelease: " + Parameters.IsNuGetRelease);
  60. void ExecWait(string preamble, string command, string args)
  61. {
  62. Console.WriteLine(preamble);
  63. Process.Start(new ProcessStartInfo(command, args) {UseShellExecute = false}).WaitForExit();
  64. }
  65. ExecWait("dotnet version:", "dotnet", "--info");
  66. ExecWait("dotnet workloads:", "dotnet", "workload list");
  67. Information("Processor count: " + Environment.ProcessorCount);
  68. Information("Available RAM: " + GC.GetGCMemoryInfo().TotalAvailableMemoryBytes / 0x100000 + "MB");
  69. }
  70. DotNetConfigHelper ApplySettingCore(DotNetConfigHelper c)
  71. {
  72. if (Parameters.IsRunningOnAzure)
  73. c.AddProperty("JavaSdkDirectory", GetVariable<string>("JAVA_HOME_11_X64"));
  74. c.AddProperty("PackageVersion", Parameters.Version)
  75. .SetConfiguration(Parameters.Configuration)
  76. .SetVerbosity(DotNetVerbosity.Minimal);
  77. if (Parameters.IsPackingToLocalCache)
  78. c
  79. .AddProperty("ForcePackAvaloniaNative", "True")
  80. .AddProperty("SkipObscurePlatforms", "True")
  81. .AddProperty("SkipBuildingSamples", "True")
  82. .AddProperty("SkipBuildingTests", "True");
  83. return c;
  84. }
  85. DotNetBuildSettings ApplySetting(DotNetBuildSettings c, Configure<DotNetBuildSettings> configurator = null) =>
  86. ApplySettingCore(c).Build.Apply(configurator);
  87. DotNetPackSettings ApplySetting(DotNetPackSettings c, Configure<DotNetPackSettings> configurator = null) =>
  88. ApplySettingCore(c).Pack.Apply(configurator);
  89. DotNetTestSettings ApplySetting(DotNetTestSettings c, Configure<DotNetTestSettings> configurator = null) =>
  90. ApplySettingCore(c).Test.Apply(configurator);
  91. Target Clean => _ => _.Executes(() =>
  92. {
  93. Parameters.BuildDirs.ForEach(DeleteDirectory);
  94. Parameters.BuildDirs.ForEach(EnsureCleanDirectory);
  95. EnsureCleanDirectory(Parameters.ArtifactsDir);
  96. EnsureCleanDirectory(Parameters.NugetIntermediateRoot);
  97. EnsureCleanDirectory(Parameters.NugetRoot);
  98. EnsureCleanDirectory(Parameters.ZipRoot);
  99. EnsureCleanDirectory(Parameters.TestResultsRoot);
  100. });
  101. Target CompileHtmlPreviewer => _ => _
  102. .DependsOn(Clean)
  103. .OnlyWhenStatic(() => !Parameters.SkipPreviewer)
  104. .Executes(() =>
  105. {
  106. var webappDir = RootDirectory / "src" / "Avalonia.DesignerSupport" / "Remote" / "HtmlTransport" / "webapp";
  107. NpmTasks.NpmInstall(c => c
  108. .SetProcessWorkingDirectory(webappDir)
  109. .SetProcessArgumentConfigurator(a => a.Add("--silent")));
  110. NpmTasks.NpmRun(c => c
  111. .SetProcessWorkingDirectory(webappDir)
  112. .SetCommand("dist"));
  113. });
  114. Target CompileNative => _ => _
  115. .DependsOn(Clean)
  116. .DependsOn(GenerateCppHeaders)
  117. .OnlyWhenStatic(() => EnvironmentInfo.IsOsx)
  118. .Executes(() =>
  119. {
  120. var project = $"{RootDirectory}/native/Avalonia.Native/src/OSX/Avalonia.Native.OSX.xcodeproj/";
  121. var args = $"-project {project} -configuration {Parameters.Configuration} CONFIGURATION_BUILD_DIR={RootDirectory}/Build/Products/Release";
  122. ProcessTasks.StartProcess("xcodebuild", args).AssertZeroExitCode();
  123. });
  124. Target Compile => _ => _
  125. .DependsOn(Clean, CompileNative)
  126. .DependsOn(CompileHtmlPreviewer)
  127. .Executes(() =>
  128. {
  129. DotNetBuild(c => ApplySetting(c)
  130. .SetProjectFile(Parameters.MSBuildSolution)
  131. );
  132. });
  133. void RunCoreTest(string projectName)
  134. {
  135. Information($"Running tests from {projectName}");
  136. var project = RootDirectory.GlobFiles(@$"**\{projectName}.csproj").FirstOrDefault()
  137. ?? throw new InvalidOperationException($"Project {projectName} doesn't exist");
  138. // Nuke and MSBuild tools have build-in helpers to get target frameworks from the project.
  139. // Unfortunately, it gets broken with every second SDK update, so we had to do it manually.
  140. var fileXml = XDocument.Parse(File.ReadAllText(project));
  141. var targetFrameworks = fileXml.Descendants("TargetFrameworks")
  142. .FirstOrDefault()?.Value.Split(';').Select(f => f.Trim());
  143. if (targetFrameworks is null)
  144. {
  145. var targetFramework = fileXml.Descendants("TargetFramework").FirstOrDefault()?.Value;
  146. if (targetFramework is not null)
  147. {
  148. targetFrameworks = new[] { targetFramework };
  149. }
  150. }
  151. if (targetFrameworks is null)
  152. {
  153. throw new InvalidOperationException("No target frameworks were found in the test project");
  154. }
  155. foreach (var fw in targetFrameworks)
  156. {
  157. var tfm = fw;
  158. if (tfm == "$(AvsCurrentTargetFramework)")
  159. {
  160. tfm = "net8.0";
  161. }
  162. if (tfm == "$(AvsLegacyTargetFrameworks)")
  163. {
  164. tfm = "net6.0";
  165. }
  166. if (tfm.StartsWith("net4")
  167. && (RuntimeInformation.IsOSPlatform(OSPlatform.Linux) || RuntimeInformation.IsOSPlatform(OSPlatform.OSX))
  168. && Environment.GetEnvironmentVariable("FORCE_LINUX_TESTS") != "1")
  169. {
  170. Information($"Skipping {projectName} ({tfm}) tests on *nix - https://github.com/mono/mono/issues/13969");
  171. continue;
  172. }
  173. Information($"Running for {projectName} ({tfm}) ...");
  174. DotNetTest(c => ApplySetting(c)
  175. .SetProjectFile(project)
  176. .SetFramework(tfm)
  177. .EnableNoBuild()
  178. .EnableNoRestore()
  179. .When(Parameters.PublishTestResults, _ => _
  180. .SetLoggers("trx")
  181. .SetResultsDirectory(Parameters.TestResultsRoot)));
  182. }
  183. }
  184. Target RunHtmlPreviewerTests => _ => _
  185. .DependsOn(CompileHtmlPreviewer)
  186. .OnlyWhenStatic(() => !(Parameters.SkipPreviewer || Parameters.SkipTests))
  187. .Executes(() =>
  188. {
  189. var webappTestDir = RootDirectory / "tests" / "Avalonia.DesignerSupport.Tests" / "Remote" / "HtmlTransport" / "webapp";
  190. NpmTasks.NpmInstall(c => c
  191. .SetProcessWorkingDirectory(webappTestDir)
  192. .SetProcessArgumentConfigurator(a => a.Add("--silent")));
  193. NpmTasks.NpmRun(c => c
  194. .SetProcessWorkingDirectory(webappTestDir)
  195. .SetCommand("test"));
  196. });
  197. Target RunCoreLibsTests => _ => _
  198. .OnlyWhenStatic(() => !Parameters.SkipTests)
  199. .DependsOn(Compile)
  200. .Executes(() =>
  201. {
  202. RunCoreTest("Avalonia.Base.UnitTests");
  203. RunCoreTest("Avalonia.Controls.UnitTests");
  204. RunCoreTest("Avalonia.Controls.DataGrid.UnitTests");
  205. RunCoreTest("Avalonia.Markup.UnitTests");
  206. RunCoreTest("Avalonia.Markup.Xaml.UnitTests");
  207. RunCoreTest("Avalonia.Skia.UnitTests");
  208. RunCoreTest("Avalonia.ReactiveUI.UnitTests");
  209. RunCoreTest("Avalonia.Headless.NUnit.UnitTests");
  210. RunCoreTest("Avalonia.Headless.XUnit.UnitTests");
  211. });
  212. Target RunRenderTests => _ => _
  213. .OnlyWhenStatic(() => !Parameters.SkipTests)
  214. .DependsOn(Compile)
  215. .Executes(() =>
  216. {
  217. RunCoreTest("Avalonia.Skia.RenderTests");
  218. if (Parameters.IsRunningOnWindows)
  219. RunCoreTest("Avalonia.Direct2D1.RenderTests");
  220. });
  221. Target RunToolsTests => _ => _
  222. .OnlyWhenStatic(() => !Parameters.SkipTests)
  223. .DependsOn(Compile)
  224. .Executes(() =>
  225. {
  226. RunCoreTest("Avalonia.Generators.Tests");
  227. if (Parameters.IsRunningOnWindows)
  228. RunCoreTest("Avalonia.DesignerSupport.Tests");
  229. });
  230. Target RunLeakTests => _ => _
  231. .OnlyWhenStatic(() => !Parameters.SkipTests && Parameters.IsRunningOnWindows)
  232. .DependsOn(Compile)
  233. .Executes(() =>
  234. {
  235. void DoMemoryTest()
  236. {
  237. RunCoreTest("Avalonia.LeakTests");
  238. }
  239. ControlFlow.ExecuteWithRetry(DoMemoryTest, delay: TimeSpan.FromMilliseconds(3));
  240. });
  241. Target ZipFiles => _ => _
  242. .After(CreateNugetPackages, Compile, RunCoreLibsTests, Package)
  243. .Executes(() =>
  244. {
  245. var data = Parameters;
  246. Zip(data.ZipNuGetArtifacts, data.NugetRoot);
  247. });
  248. Target CreateIntermediateNugetPackages => _ => _
  249. .DependsOn(Compile)
  250. .After(RunTests)
  251. .Executes(() =>
  252. {
  253. DotNetPack(c => ApplySetting(c).SetProject(Parameters.MSBuildSolution));
  254. });
  255. Target CreateNugetPackages => _ => _
  256. .DependsOn(CreateIntermediateNugetPackages)
  257. .Executes(() =>
  258. {
  259. BuildTasksPatcher.PatchBuildTasksInPackage(Parameters.NugetIntermediateRoot / "Avalonia.Build.Tasks." +
  260. Parameters.Version + ".nupkg");
  261. var config = Numerge.MergeConfiguration.LoadFile(RootDirectory / "nukebuild" / "numerge.config");
  262. EnsureCleanDirectory(Parameters.NugetRoot);
  263. if(!Numerge.NugetPackageMerger.Merge(Parameters.NugetIntermediateRoot, Parameters.NugetRoot, config,
  264. new NumergeNukeLogger()))
  265. throw new Exception("Package merge failed");
  266. RefAssemblyGenerator.GenerateRefAsmsInPackage(
  267. Parameters.NugetRoot / $"Avalonia.{Parameters.Version}.nupkg",
  268. Parameters.NugetRoot / $"Avalonia.{Parameters.Version}.snupkg");
  269. });
  270. Target ValidateApiDiff => _ => _
  271. .DependsOn(CreateNugetPackages)
  272. .Executes(async () =>
  273. {
  274. await Task.WhenAll(
  275. Directory.GetFiles(Parameters.NugetRoot, "*.nupkg").Select(nugetPackage => ApiDiffHelper.ValidatePackage(
  276. ApiCompatTool, nugetPackage, Parameters.ApiValidationBaseline,
  277. Parameters.ApiValidationSuppressionFiles, Parameters.UpdateApiValidationSuppression)));
  278. });
  279. Target OutputApiDiff => _ => _
  280. .DependsOn(CreateNugetPackages)
  281. .Executes(async () =>
  282. {
  283. await Task.WhenAll(
  284. Directory.GetFiles(Parameters.NugetRoot, "*.nupkg").Select(nugetPackage => ApiDiffHelper.GetDiff(
  285. ApiGenTool, RootDirectory / "api" / "diff",
  286. nugetPackage, Parameters.ApiValidationBaseline)));
  287. });
  288. Target RunTests => _ => _
  289. .DependsOn(RunCoreLibsTests)
  290. .DependsOn(RunRenderTests)
  291. .DependsOn(RunToolsTests)
  292. .DependsOn(RunHtmlPreviewerTests)
  293. .DependsOn(RunLeakTests);
  294. Target Package => _ => _
  295. .DependsOn(RunTests)
  296. .DependsOn(CreateNugetPackages)
  297. .DependsOn(ValidateApiDiff);
  298. Target CiAzureLinux => _ => _
  299. .DependsOn(RunTests);
  300. Target CiAzureOSX => _ => _
  301. .DependsOn(Package)
  302. .DependsOn(ZipFiles);
  303. Target CiAzureWindows => _ => _
  304. .DependsOn(Package)
  305. .DependsOn(ZipFiles);
  306. Target BuildToNuGetCache => _ => _
  307. .DependsOn(CreateNugetPackages)
  308. .Executes(() =>
  309. {
  310. if (!Parameters.IsPackingToLocalCache)
  311. throw new InvalidOperationException();
  312. foreach (var path in Parameters.NugetRoot.GlobFiles("*.nupkg"))
  313. {
  314. using var f = File.Open(path.ToString(), FileMode.Open, FileAccess.Read);
  315. using var zip = new ZipArchive(f, ZipArchiveMode.Read);
  316. var nuspecEntry = zip.Entries.First(e => e.FullName.EndsWith(".nuspec") && e.FullName == e.Name);
  317. var packageId = XDocument.Load(nuspecEntry.Open()).Document.Root
  318. .Elements().First(x => x.Name.LocalName == "metadata")
  319. .Elements().First(x => x.Name.LocalName == "id").Value;
  320. var packagePath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile),
  321. ".nuget",
  322. "packages",
  323. packageId.ToLowerInvariant(),
  324. BuildParameters.LocalBuildVersion);
  325. if (Directory.Exists(packagePath))
  326. Directory.Delete(packagePath, true);
  327. Directory.CreateDirectory(packagePath);
  328. zip.ExtractToDirectory(packagePath);
  329. File.WriteAllText(Path.Combine(packagePath, ".nupkg.metadata"), @"{
  330. ""version"": 2,
  331. ""contentHash"": ""e900dFK7jHJ2WcprLcgJYQoOMc6ejRTwAAMi0VGOFbSczcF98ZDaqwoQIiyqpAwnja59FSbV+GUUXfc3vaQ2Jg=="",
  332. ""source"": ""https://api.nuget.org/v3/index.json""
  333. }");
  334. }
  335. });
  336. Target GenerateCppHeaders => _ => _.Executes(() =>
  337. {
  338. var file = MicroComCodeGenerator.Parse(
  339. File.ReadAllText(RootDirectory / "src" / "Avalonia.Native" / "avn.idl"));
  340. File.WriteAllText(RootDirectory / "native" / "Avalonia.Native" / "inc" / "avalonia-native.h",
  341. file.GenerateCppHeader());
  342. });
  343. public static int Main() =>
  344. RuntimeInformation.IsOSPlatform(OSPlatform.Windows)
  345. ? Execute<Build>(x => x.Package)
  346. : Execute<Build>(x => x.RunTests);
  347. }
  348. public static class ToolSettingsExtensions
  349. {
  350. public static T Apply<T>(this T settings, Configure<T> configurator)
  351. {
  352. return configurator != null ? configurator(settings) : settings;
  353. }
  354. }