在roslyn中使用语义模型时,“ArgumentException:语法节点不在语法树中”

时间:2015-01-03 21:35:09

标签: c# roslyn

我正在使用Roslyn收集方法中所有方法的调用,并在方法的相应测试方法中用填充程序替换它们。问题是,尽管使用项目的所有引用和语法树适当地初始化编译对象,但它会抛出一个ArgumentException,它表示"语法节点不在语法树"中。

这是我用来初始化语义模型的代码:

   public TestMethodCodeGenerator(string code, Project project = null)
    {
        if (!string.IsNullOrWhiteSpace(code))
        {
            var syntaxTree = CSharpSyntaxTree.ParseText(code);

            if (null != project)
            {
                var syntaxTreesOfDocumentsInProject = new List<SyntaxTree>();
                foreach (var document in project.Documents)
                {
                    syntaxTreesOfDocumentsInProject.Add(CSharpSyntaxTree.ParseFile(document.FilePath));
                }

                var compilation = CSharpCompilation.Create("Demo").AddReferences(project.MetadataReferences).AddSyntaxTrees(syntaxTreesOfDocumentsInProject);
                this.semanticModel = compilation.GetSemanticModel(syntaxTreesOfDocumentsInProject[0]);
                }
            }
            else 
            {                
                var compilation = CSharpCompilation.Create("Demo").AddSyntaxTrees(syntaxTree);
                this.semanticModel = compilation.GetSemanticModel(syntaxTree);
            }
        }
    }

这就是我使用semanticModel为所有调用生成填充程序的方法:

private string PopulateMethodBodyWithShims(MethodDeclarationSyntax methodDeclarationSyntax)
    {
        if (null != methodDeclarationSyntax)
        {
            var stringBuilder = new StringBuilder();
            var methodBlock = methodDeclarationSyntax.Body;
            foreach (var statement in methodBlock.Statements)
            {
                var invocationSyntax = this.ExtractMethodInvocationSyntaxFromStatement(statement);
                if (null != invocationSyntax)
                {
                    var call = invocationSyntax.Expression as MemberAccessExpressionSyntax;
                    if (null != call)
                    {
                        try
                        {
                            IMethodSymbol methodSymbol;
                            **methodSymbol = this.semanticModel.GetSymbolInfo(call).Symbol as IMethodSymbol;**
                            if (null != methodSymbol)
                            {
                                var shimMethod = this.GenerateShimMethod(methodSymbol);
                                stringBuilder.AppendLine(GeneratedTestClassConstants.IndentationSpaceToken + GeneratedTestClassConstants.IndentationSpaceToken + shimMethod);
                            }
                        }
                        catch (ArgumentException ex)
                        {
                            ////This exception can be thrown if the syntax node is not within the syntax tree
                            var message = ex.Message;
                            throw new ArgumentException(message + " : " + call);
                        }                           
                    }
                }
            }
.
.
.

第二个代码段中突出显示的区域是发生异常的位置,执行流向catch块。问题是什么,我在这里错过了什么?

1 个答案:

答案 0 :(得分:1)

为什么要解析自己的文件并构建自己的编译?如果您已经拥有Workspace SolutionProjectDocument的{​​{1}},则可以使用document.GetSyntaxTreeAsync()document.GetSemanticModelAsync()和{ {1}}了解这些。

您似乎在为一些新代码添加语法树,但您可以通过添加该语法树并获取新的document.Project.GetCompilationAsync()来实现。 e.g。

Project

无论如何,我怀疑您所说的代码存在的问题是您存储的语义模型是与project = project.AddDocument("generatedfile", code).Project;中的第一个语法树对应的语法模型,但您永远不会将syntaxTreesOfDocuments(您拥有的syntaxTree字符串的树)添加到该列表中,因此它不是编译的一部分,并且绝对不是{{1}的树。 1}}操作。