是否可以在Roslyn中修改SyntaxTree并运行编辑后的代码?

时间:2018-09-22 19:04:11

标签: c# roslyn

我正在使用Roslyn更改代码,但是,更改后生成了新的SyntaxNode,但是我找不到执行此新代码的方法。我发现的唯一一个方法是从新的ToString获取Root并使用新的字符串调用EvaluateAsync。应该有一种方法可以胜过这个,因为我已经编译了一个新代码。

static void Main(string[] args)
{
    var expression = "System.Console.WriteLine(\"Test\")";
    var compile = CSharpScript.Create<EntityRepresentation>(expression).GetCompilation();
    var root = compile.SyntaxTrees.Single().GetRoot();

    var descentands = root.DescendantNodes().Where(n =>
    {
        if (n is ArgumentSyntax)
            return true;
        return false;
    }).ToList();

    var otherRoot = root.ReplaceNodes(descentands, (n1, n2) =>
    {
        var argumentName = Argument(LiteralExpression(SyntaxKind.StringLiteralExpression, Literal("NewValue")));
        return argumentName;
    });

    var newCode = otherRoot.ToString();

    // Faz o que estou querendo, contudo não me parece a melhor maneira
    var result = CSharpScript.EvaluateAsync(newCode).Result;
}

1 个答案:

答案 0 :(得分:2)

不幸的是,从语法树创建Script对象的方法是Microsoft程序集的内部。

但是,您不必编译两次-您可以在第一次分析后再进行第二次编译。

var expression = "System.Console.WriteLine(\"Test\")";
var origTree = CSharpSyntaxTree.ParseText(expression, 
                  CSharpParseOptions.Default.WithKind(SourceCodeKind.Script));
var root = origTree.GetRoot();

// -Snip- tree manipulation

var script = CSharpScript.Create(otherRoot.ToString());
var errors = script.Compile();
if(errors.Any(x => x.Severity == DiagnosticSeverity.Error)) {
    throw new Exception($"Compilation errors:\n{string.Join("\n", errors.Select(x => x.GetMessage()))}");
}
await script.RunAsync();