我正在使用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块。问题是什么,我在这里错过了什么?
答案 0 :(得分:1)
为什么要解析自己的文件并构建自己的编译?如果您已经拥有Workspace
Solution
,Project
和Document
的{{1}},则可以使用document.GetSyntaxTreeAsync()
,document.GetSemanticModelAsync()
和{ {1}}了解这些。
您似乎在为一些新代码添加语法树,但您可以通过添加该语法树并获取新的document.Project.GetCompilationAsync()
来实现。 e.g。
Project
无论如何,我怀疑您所说的代码存在的问题是您存储的语义模型是与project = project.AddDocument("generatedfile", code).Project;
中的第一个语法树对应的语法模型,但您永远不会将syntaxTreesOfDocuments
(您拥有的syntaxTree
字符串的树)添加到该列表中,因此它不是编译的一部分,并且绝对不是{{1}的树。 1}}操作。