Files
Umbraco-CMS/src/Umbraco.Infrastructure/ModelsBuilder/RoslynCompiler.cs
Nikolaj Geisle 02669e930c V12: Update dependencies to latest (#14204)
* Update dependencies to latest

* revert npoco back to 5.5.0

* Updated Npoco and Serilog

---------

Co-authored-by: Bjarke Berg <mail@bergmania.dk>
2023-05-11 13:32:14 +02:00

86 lines
3.4 KiB
C#

using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.Emit;
using Microsoft.CodeAnalysis.Text;
using Microsoft.Extensions.DependencyModel;
namespace Umbraco.Cms.Infrastructure.ModelsBuilder;
public class RoslynCompiler
{
public const string GeneratedAssemblyName = "ModelsGeneratedAssembly";
private readonly OutputKind _outputKind;
private readonly CSharpParseOptions _parseOptions;
private readonly IEnumerable<MetadataReference> _refs;
/// <summary>
/// Initializes a new instance of the <see cref="RoslynCompiler" /> class.
/// </summary>
/// <remarks>
/// Roslyn compiler which can be used to compile a c# file to a Dll assembly
/// </remarks>
public RoslynCompiler()
{
_outputKind = OutputKind.DynamicallyLinkedLibrary;
_parseOptions =
CSharpParseOptions.Default.WithLanguageVersion(LanguageVersion
.Latest); // What languageversion should we default to?
// In order to dynamically compile the assembly, we need to add all refs from our current
// application. This will also add the correct framework dependencies and we won't have to worry
// about the specific framework that is currently being run.
// This was borrowed from: https://github.com/dotnet/core/issues/2082#issuecomment-442713181
// because we were running into the same error as that thread because we were either:
// - not adding enough of the runtime dependencies OR
// - we were explicitly adding the wrong runtime dependencies
// ... at least that the gist of what I can tell.
if (DependencyContext.Default != null)
{
MetadataReference[] refs =
DependencyContext.Default.CompileLibraries
.SelectMany(cl => cl.ResolveReferencePaths())
.Select(asm => MetadataReference.CreateFromFile(asm))
.ToArray();
_refs = refs.ToList();
}
else
{
_refs = Enumerable.Empty<MetadataReference>();
}
}
/// <summary>
/// Compile a source file to a dll
/// </summary>
/// <param name="pathToSourceFile">Path to the source file containing the code to be compiled.</param>
/// <param name="savePath">The path where the output assembly will be saved.</param>
public void CompileToFile(string pathToSourceFile, string savePath)
{
var sourceCode = File.ReadAllText(pathToSourceFile);
var sourceText = SourceText.From(sourceCode);
SyntaxTree syntaxTree = SyntaxFactory.ParseSyntaxTree(sourceText, _parseOptions);
// Not entirely certain that assemblyIdentityComparer is nececary?
var compilation = CSharpCompilation.Create(
GeneratedAssemblyName,
new[] { syntaxTree },
_refs,
new CSharpCompilationOptions(
_outputKind,
optimizationLevel: OptimizationLevel.Release,
assemblyIdentityComparer: DesktopAssemblyIdentityComparer.Default));
EmitResult emitResult = compilation.Emit(savePath);
if (!emitResult.Success)
{
throw new InvalidOperationException("Roslyn compiler could not create ModelsBuilder dll:\n" +
string.Join("\n", emitResult.Diagnostics.Select(x => x.GetMessage())));
}
}
}