 357242f7b6
			
		
	
	
	357242f7b6
	
	
	
		
			
			- Downgrade lock file validation changes to warning instead of an error - Don't show dependency information. Once we get log levels it'll be verbose. The response file can be used to debug things.
		
			
				
	
	
		
			260 lines
		
	
	
	
		
			10 KiB
			
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			260 lines
		
	
	
	
		
			10 KiB
			
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.IO;
 | |
| using System.Linq;
 | |
| using Microsoft.Dnx.Runtime.Common.CommandLine;
 | |
| using Microsoft.DotNet.Cli.Utils;
 | |
| using Microsoft.Extensions.ProjectModel;
 | |
| using Microsoft.Extensions.ProjectModel.Compilation;
 | |
| using NuGet.Frameworks;
 | |
| 
 | |
| namespace Microsoft.DotNet.Tools.Compiler
 | |
| {
 | |
|     public class Program
 | |
|     {
 | |
|         public static int Main(string[] args)
 | |
|         {
 | |
|             DebugHelper.HandleDebugSwitch(ref args);
 | |
| 
 | |
|             var app = new CommandLineApplication();
 | |
|             app.Name = "dotnet compile";
 | |
|             app.FullName = ".NET Compiler";
 | |
|             app.Description = "Compiler for the .NET Platform";
 | |
|             app.HelpOption("-h|--help");
 | |
| 
 | |
|             var output = app.Option("-o|--output <OUTPUT_DIR>", "Directory in which to place outputs", CommandOptionType.SingleValue);
 | |
|             var framework = app.Option("-f|--framework <FRAMEWORK>", "Compile a specific framework", CommandOptionType.MultipleValue);
 | |
|             var configuration = app.Option("-c|--configuration <CONFIGURATION>", "Configuration under which to build", CommandOptionType.SingleValue);
 | |
|             var project = app.Argument("<PROJECT>", "The project to compile, defaults to the current directory. Can be a path to a project.json or a project directory");
 | |
| 
 | |
|             app.OnExecute(() =>
 | |
|             {
 | |
|                 // Locate the project and get the name and full path
 | |
|                 var path = project.Value;
 | |
|                 if (string.IsNullOrEmpty(path))
 | |
|                 {
 | |
|                     path = Directory.GetCurrentDirectory();
 | |
|                 }
 | |
| 
 | |
|                 // Load project contexts for each framework and compile them
 | |
|                 bool success = true;
 | |
|                 if (framework.HasValue())
 | |
|                 {
 | |
|                     foreach (var context in framework.Values.Select(f => ProjectContext.Create(path, NuGetFramework.Parse(f))))
 | |
|                     {
 | |
|                         success &= Compile(context, configuration.Value() ?? Constants.DefaultConfiguration, output.Value());
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     foreach (var context in ProjectContext.CreateContextForEachFramework(path))
 | |
|                     {
 | |
|                         success &= Compile(context, configuration.Value() ?? Constants.DefaultConfiguration, output.Value());
 | |
|                     }
 | |
|                 }
 | |
|                 return success ? 0 : 1;
 | |
|             });
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 return app.Execute(args);
 | |
|             }
 | |
|             catch (OperationCanceledException ex)
 | |
|             {
 | |
|                 Console.Error.WriteLine(ex.Message);
 | |
|                 return 1;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static bool Compile(ProjectContext context, string configuration, string outputPath)
 | |
|         {
 | |
|             Reporter.Output.WriteLine($"Building {context.RootProject.Identity.Name.Yellow()} for {context.TargetFramework.DotNetFrameworkName.Yellow()}");
 | |
| 
 | |
|             // Create the library exporter
 | |
|             var exporter = context.CreateExporter(configuration);
 | |
| 
 | |
|             bool success = true;
 | |
| 
 | |
|             // Print out dependency diagnostics
 | |
|             foreach (var diag in context.LibraryManager.GetAllDiagnostics())
 | |
|             {
 | |
|                 success &= diag.Severity != DiagnosticMessageSeverity.Error;
 | |
|                 Console.WriteLine(diag.FormattedMessage);
 | |
|             }
 | |
| 
 | |
|             // If there were dependency errors don't bother compiling
 | |
|             if (!success)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             // Gather exports for the project
 | |
|             var dependencies = exporter.GetCompilationDependencies().ToList();
 | |
| 
 | |
|             // Hackily trigger builds of the dependent projects
 | |
|             foreach (var dependency in dependencies.Where(d => d.CompilationAssemblies.Any()))
 | |
|             {
 | |
|                 var projectDependency = dependency.Library as ProjectDescription;
 | |
|                 if (projectDependency != null && !string.Equals(projectDependency.Identity.Name, context.RootProject.Identity.Name, StringComparison.Ordinal))
 | |
|                 {
 | |
|                     var compileResult = Command.Create("dotnet-compile", $"--framework {projectDependency.Framework} --configuration {configuration} {projectDependency.Project.ProjectDirectory}")
 | |
|                         .ForwardStdOut()
 | |
|                         .ForwardStdErr()
 | |
|                         .RunAsync()
 | |
|                         .Result;
 | |
|                     if (compileResult.ExitCode != 0)
 | |
|                     {
 | |
|                         Console.Error.WriteLine($"Failed to compile dependency: {projectDependency.Identity.Name}");
 | |
|                         return false;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // Dump dependency data
 | |
|             // TODO: Turn on only if verbose, we can look at the response
 | |
|             // file anyways
 | |
|             // ShowDependencyInfo(dependencies);
 | |
| 
 | |
|             // Hackily generate the output path
 | |
|             if (string.IsNullOrEmpty(outputPath))
 | |
|             {
 | |
|                 outputPath = Path.Combine(
 | |
|                     context.ProjectFile.ProjectDirectory,
 | |
|                     Constants.BinDirectoryName,
 | |
|                     configuration,
 | |
|                     context.TargetFramework.GetTwoDigitShortFolderName());
 | |
|             }
 | |
|             if (!Directory.Exists(outputPath))
 | |
|             {
 | |
|                 Directory.CreateDirectory(outputPath);
 | |
|             }
 | |
| 
 | |
|             // Get compilation options
 | |
|             var compilationOptions = context.ProjectFile.GetCompilerOptions(context.TargetFramework, configuration);
 | |
|             var outputName = Path.Combine(outputPath, context.ProjectFile.Name + ".dll");
 | |
| 
 | |
|             // Assemble csc args
 | |
|             var cscArgs = new List<string>()
 | |
|             {
 | |
|                 // Default suppressions
 | |
|                 "-nowarn:CS1701",
 | |
|                 "-nowarn:CS1702",
 | |
|                 "-nowarn:CS1705",
 | |
|                 "-nostdlib",
 | |
|                 "-nologo",
 | |
|                 $"-out:\"{outputName}\""
 | |
|             };
 | |
| 
 | |
|             // Add compilation options to the args
 | |
|             ApplyCompilationOptions(compilationOptions, cscArgs);
 | |
| 
 | |
|             foreach (var dependency in dependencies)
 | |
|             {
 | |
|                 cscArgs.AddRange(dependency.CompilationAssemblies.Select(r => $"-r:\"{r}\""));
 | |
|                 cscArgs.AddRange(dependency.SourceReferences);
 | |
|             }
 | |
| 
 | |
|             // Add project source files
 | |
|             cscArgs.AddRange(context.ProjectFile.Files.SourceFiles);
 | |
| 
 | |
|             // Write RSP file
 | |
|             var rsp = Path.Combine(outputPath, "dotnet-compile.csc.rsp");
 | |
|             File.WriteAllLines(rsp, cscArgs);
 | |
| 
 | |
|             // Execute CSC!
 | |
|             var result = RunCsc($"-noconfig @\"{rsp}\"")
 | |
|                 .ForwardStdErr()
 | |
|                 .ForwardStdOut()
 | |
|                 .RunAsync()
 | |
|                 .Result;
 | |
|             return result.ExitCode == 0;
 | |
|         }
 | |
| 
 | |
|         private static Command RunCsc(string cscArgs)
 | |
|         {
 | |
|             // Locate CoreRun
 | |
|             string hostRoot = Environment.GetEnvironmentVariable("DOTNET_CSC_PATH");
 | |
|             if (string.IsNullOrEmpty(hostRoot))
 | |
|             {
 | |
|                 hostRoot = AppContext.BaseDirectory;
 | |
|             }
 | |
|             var corerun = Path.Combine(hostRoot, Constants.CoreRunName);
 | |
|             var cscExe = Path.Combine(hostRoot, "csc.exe");
 | |
|             return File.Exists(corerun) && File.Exists(cscExe)
 | |
|                 ? Command.Create(corerun, $@"""{cscExe}"" {cscArgs}")
 | |
|                 : Command.Create("csc", cscArgs);
 | |
|         }
 | |
| 
 | |
|         private static void ApplyCompilationOptions(CompilerOptions compilationOptions, List<string> cscArgs)
 | |
|         {
 | |
|             var targetType = compilationOptions.EmitEntryPoint.GetValueOrDefault() ? "exe" : "library";
 | |
| 
 | |
|             cscArgs.Add($"-target:{targetType}");
 | |
| 
 | |
|             if (compilationOptions.AllowUnsafe.GetValueOrDefault())
 | |
|             {
 | |
|                 cscArgs.Add("-unsafe+");
 | |
|             }
 | |
| 
 | |
|             cscArgs.AddRange(compilationOptions.Defines.Select(d => $"-d:{d}"));
 | |
| 
 | |
|             if (compilationOptions.Optimize.GetValueOrDefault())
 | |
|             {
 | |
|                 cscArgs.Add("-optimize");
 | |
|             }
 | |
| 
 | |
|             if (!string.IsNullOrEmpty(compilationOptions.Platform))
 | |
|             {
 | |
|                 cscArgs.Add($"-platform:{compilationOptions.Platform}");
 | |
|             }
 | |
| 
 | |
|             if (compilationOptions.WarningsAsErrors.GetValueOrDefault())
 | |
|             {
 | |
|                 cscArgs.Add("-warnaserror");
 | |
|             }
 | |
| 
 | |
|             if (compilationOptions.DelaySign.GetValueOrDefault())
 | |
|             {
 | |
|                 cscArgs.Add("-delaysign+");
 | |
|             }
 | |
| 
 | |
|             if (!string.IsNullOrEmpty(compilationOptions.KeyFile))
 | |
|             {
 | |
|                 cscArgs.Add($"-keyFile:\"{compilationOptions.KeyFile}\"");
 | |
|             }
 | |
| 
 | |
|             // TODO: Support debug portable
 | |
|             cscArgs.Add("-debug:full");
 | |
| 
 | |
|             // TODO: OSS signing
 | |
|         }
 | |
| 
 | |
|         private static void ShowDependencyInfo(IEnumerable<LibraryExport> dependencies)
 | |
|         {
 | |
|             foreach (var dependency in dependencies)
 | |
|             {
 | |
|                 if (!dependency.Library.Resolved)
 | |
|                 {
 | |
|                     Reporter.Error.WriteLine($"  Unable to resolve dependency {dependency.Library.Identity.ToString().Red().Bold()}");
 | |
|                     Reporter.Error.WriteLine("");
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     Reporter.Output.WriteLine($"  Using {dependency.Library.Identity.Type.Value.Cyan().Bold()} dependency {dependency.Library.Identity.ToString().Cyan().Bold()}");
 | |
|                     Reporter.Output.WriteLine($"    Path: {dependency.Library.Path}");
 | |
| 
 | |
|                     foreach (var metadataReference in dependency.CompilationAssemblies)
 | |
|                     {
 | |
|                         Reporter.Output.WriteLine($"    Assembly: {metadataReference}");
 | |
|                     }
 | |
| 
 | |
|                     foreach (var sourceReference in dependency.SourceReferences)
 | |
|                     {
 | |
|                         Reporter.Output.WriteLine($"    Source: {sourceReference}");
 | |
|                     }
 | |
|                     Reporter.Output.WriteLine("");
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |