mirror of
https://github.com/github/codeql.git
synced 2025-12-16 16:53:25 +01:00
208 lines
9.5 KiB
C#
208 lines
9.5 KiB
C#
using System.Collections.Generic;
|
|
using System.Linq;
|
|
using Semmle.Util;
|
|
using Semmle.Util.Logging;
|
|
|
|
namespace Semmle.Autobuild.Shared
|
|
{
|
|
internal static class MsBuildCommandExtensions
|
|
{
|
|
/// <summary>
|
|
/// Appends a call to msbuild.
|
|
/// </summary>
|
|
/// <param name="cmdBuilder"></param>
|
|
/// <param name="builder"></param>
|
|
/// <returns></returns>
|
|
public static CommandBuilder MsBuildCommand(this CommandBuilder cmdBuilder, IAutobuilder<AutobuildOptionsShared> builder)
|
|
{
|
|
// mono doesn't ship with `msbuild` on Arm-based Macs, but we can fall back to
|
|
// msbuild that ships with `dotnet` which can be invoked with `dotnet msbuild`
|
|
// perhaps we should do this on all platforms?
|
|
return builder.Actions.IsRunningOnAppleSilicon()
|
|
? cmdBuilder.RunCommand("dotnet").Argument("msbuild")
|
|
: cmdBuilder.RunCommand("msbuild");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// A build rule using msbuild.
|
|
/// </summary>
|
|
public class MsBuildRule : IBuildRule<AutobuildOptionsShared>
|
|
{
|
|
/// <summary>
|
|
/// A list of solutions or projects which failed to build.
|
|
/// </summary>
|
|
public readonly List<IProjectOrSolution> FailedProjectsOrSolutions = new();
|
|
|
|
public BuildScript Analyse(IAutobuilder<AutobuildOptionsShared> builder, bool auto)
|
|
{
|
|
if (!builder.ProjectsOrSolutionsToBuild.Any())
|
|
return BuildScript.Failure;
|
|
|
|
if (auto)
|
|
builder.Log(Severity.Info, "Attempting to build using MSBuild");
|
|
|
|
var vsTools = GetVcVarsBatFile(builder);
|
|
|
|
if (vsTools is null && builder.ProjectsOrSolutionsToBuild.Any())
|
|
{
|
|
var firstSolution = builder.ProjectsOrSolutionsToBuild.OfType<ISolution>().FirstOrDefault();
|
|
vsTools = firstSolution is not null
|
|
? BuildTools.FindCompatibleVcVars(builder.Actions, firstSolution)
|
|
: BuildTools.VcVarsAllBatFiles(builder.Actions).OrderByDescending(b => b.ToolsVersion).FirstOrDefault();
|
|
}
|
|
|
|
if (vsTools is null && builder.Actions.IsWindows())
|
|
{
|
|
builder.Log(Severity.Warning, "Could not find a suitable version of VsDevCmd.bat/vcvarsall.bat");
|
|
}
|
|
|
|
// Use `nuget.exe` from source code repo, if present, otherwise first attempt with global
|
|
// `nuget` command, and if that fails, attempt to download `nuget.exe` from nuget.org
|
|
var nuget = builder.GetFilename("nuget.exe").Select(t => t.Item1).FirstOrDefault() ?? "nuget";
|
|
var nugetDownloadPath = builder.Actions.PathCombine(FileUtils.GetTemporaryWorkingDirectory(builder.Actions.GetEnvironmentVariable, builder.Options.Language.UpperCaseName, out var _), ".nuget", "nuget.exe");
|
|
var nugetDownloaded = false;
|
|
|
|
var ret = BuildScript.Success;
|
|
|
|
foreach (var projectOrSolution in builder.ProjectsOrSolutionsToBuild)
|
|
{
|
|
if (builder.Options.NugetRestore)
|
|
{
|
|
BuildScript GetNugetRestoreScript() =>
|
|
new CommandBuilder(builder.Actions).
|
|
RunCommand(nuget).
|
|
Argument("restore").
|
|
QuoteArgument(projectOrSolution.FullPath).
|
|
Argument("-DisableParallelProcessing").
|
|
Script;
|
|
var nugetRestore = GetNugetRestoreScript();
|
|
var msbuildRestoreCommand = new CommandBuilder(builder.Actions).
|
|
MsBuildCommand(builder).
|
|
Argument("/t:restore").
|
|
QuoteArgument(projectOrSolution.FullPath);
|
|
|
|
if (builder.Actions.IsRunningOnAppleSilicon())
|
|
{
|
|
// On Apple Silicon, only try package restore with `dotnet msbuild /t:restore`
|
|
ret &= BuildScript.Try(msbuildRestoreCommand.Script);
|
|
}
|
|
else if (nugetDownloaded)
|
|
{
|
|
ret &= BuildScript.Try(nugetRestore | msbuildRestoreCommand.Script);
|
|
}
|
|
else
|
|
{
|
|
// If `nuget restore` fails, and we have not already attempted to download `nuget.exe`,
|
|
// download it and reattempt `nuget restore`.
|
|
var nugetDownloadAndRestore =
|
|
BuildScript.Bind(DownloadNugetExe(builder, nugetDownloadPath), exitCode =>
|
|
{
|
|
nugetDownloaded = true;
|
|
if (exitCode != 0)
|
|
return BuildScript.Failure;
|
|
|
|
nuget = nugetDownloadPath;
|
|
return GetNugetRestoreScript();
|
|
});
|
|
ret &= BuildScript.Try(nugetRestore | nugetDownloadAndRestore | msbuildRestoreCommand.Script);
|
|
}
|
|
}
|
|
|
|
var command = new CommandBuilder(builder.Actions);
|
|
|
|
if (vsTools is not null)
|
|
{
|
|
command.CallBatFile(vsTools.Path);
|
|
// `vcvarsall.bat` sets a default Platform environment variable,
|
|
// which may not be compatible with the supported platforms of the
|
|
// given project/solution. Unsetting it means that the default platform
|
|
// of the project/solution is used instead.
|
|
command.RunCommand("set Platform=&& type NUL", quoteExe: false);
|
|
}
|
|
|
|
command.MsBuildCommand(builder);
|
|
command.QuoteArgument(projectOrSolution.FullPath);
|
|
|
|
var target = builder.Options.MsBuildTarget ?? "rebuild";
|
|
var platform = builder.Options.MsBuildPlatform ?? (projectOrSolution is ISolution s1 ? s1.DefaultPlatformName : null);
|
|
var configuration = builder.Options.MsBuildConfiguration ?? (projectOrSolution is ISolution s2 ? s2.DefaultConfigurationName : null);
|
|
|
|
command.Argument("/t:" + target);
|
|
if (platform is not null)
|
|
command.Argument(string.Format("/p:Platform=\"{0}\"", platform));
|
|
if (configuration is not null)
|
|
command.Argument(string.Format("/p:Configuration=\"{0}\"", configuration));
|
|
|
|
command.Argument(builder.Options.MsBuildArguments);
|
|
|
|
// append the build script which invokes msbuild to the overall build script `ret`;
|
|
// we insert a check that building the current project or solution was successful:
|
|
// if it was not successful, we add it to `FailedProjectsOrSolutions`
|
|
ret &= BuildScript.OnFailure(command.Script, ret =>
|
|
{
|
|
FailedProjectsOrSolutions.Add(projectOrSolution);
|
|
});
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the BAT file used to initialize the appropriate Visual Studio
|
|
/// version/platform, as specified by the `vstools_version` property in
|
|
/// lgtm.yml.
|
|
///
|
|
/// Returns <code>null</code> when no version is specified.
|
|
/// </summary>
|
|
public static VcVarsBatFile? GetVcVarsBatFile<TAutobuildOptions>(IAutobuilder<TAutobuildOptions> builder) where TAutobuildOptions : AutobuildOptionsShared
|
|
{
|
|
VcVarsBatFile? vsTools = null;
|
|
|
|
if (builder.Options.VsToolsVersion is not null)
|
|
{
|
|
if (int.TryParse(builder.Options.VsToolsVersion, out var msToolsVersion))
|
|
{
|
|
foreach (var b in BuildTools.VcVarsAllBatFiles(builder.Actions))
|
|
{
|
|
builder.Log(Severity.Info, "Found {0} version {1}", b.Path, b.ToolsVersion);
|
|
}
|
|
|
|
vsTools = BuildTools.FindCompatibleVcVars(builder.Actions, msToolsVersion);
|
|
if (vsTools is null)
|
|
builder.Log(Severity.Warning, "Could not find build tools matching version {0}", msToolsVersion);
|
|
else
|
|
builder.Log(Severity.Info, "Setting Visual Studio tools to {0}", vsTools.Path);
|
|
}
|
|
else
|
|
{
|
|
builder.Log(Severity.Error, "The format of vstools_version is incorrect. Please specify an integer.");
|
|
}
|
|
}
|
|
|
|
return vsTools;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a script for downloading `nuget.exe` from nuget.org.
|
|
/// </summary>
|
|
private static BuildScript DownloadNugetExe<TAutobuildOptions>(IAutobuilder<TAutobuildOptions> builder, string path) where TAutobuildOptions : AutobuildOptionsShared =>
|
|
BuildScript.Create(_ =>
|
|
{
|
|
builder.Log(Severity.Info, "Attempting to download nuget.exe");
|
|
return 0;
|
|
})
|
|
&
|
|
BuildScript.DownloadFile(
|
|
FileUtils.NugetExeUrl,
|
|
path,
|
|
e => builder.Log(Severity.Warning, $"Failed to download 'nuget.exe': {e.Message}"))
|
|
&
|
|
BuildScript.Create(_ =>
|
|
{
|
|
builder.Log(Severity.Info, $"Successfully downloaded {path}");
|
|
return 0;
|
|
});
|
|
}
|
|
}
|