gpt4 book ai didi

c# - 在 Roslyn 代码修复中有条件地添加 "using"语句

转载 作者:行者123 更新时间:2023-11-30 12:55:17 27 4
gpt4 key购买 nike

我正在使用 .NET 编译器 API 在 Roslyn 中编写一些代码分析器/代码修复器。我想要代码修复来转换以下代码:

string.Format("{0} {1}", A, B)

StringExtensions.SafeJoin(" ", A, B)

到目前为止,我有这段代码:

private async Task<Document> UseJoinAsync(Document document, InvocationExpressionSyntax invocationExpr, CancellationToken cancellationToken)
{
var argumentList = invocationExpr.ArgumentList;
var firstArgument = argumentList.Arguments[1];
var secondArgument = argumentList.Arguments[2];

var statement =
InvocationExpression(
MemberAccessExpression(
SyntaxKind.SimpleMemberAccessExpression,
IdentifierName("StringExtensions"), // requires using Trilogy.Miscellaneous
IdentifierName("SafeJoin")))
.WithArgumentList(
ArgumentList(
SeparatedList<ArgumentSyntax>(
new SyntaxNodeOrToken[]
{
Argument(
LiteralExpression(
SyntaxKind.StringLiteralExpression,
Literal(" "))),
Token(SyntaxKind.CommaToken),
firstArgument,
Token(SyntaxKind.CommaToken),
secondArgument
}))).WithLeadingTrivia(invocationExpr.GetLeadingTrivia()).WithTrailingTrivia(invocationExpr.GetTrailingTrivia())
.WithAdditionalAnnotations(Formatter.Annotation);

var root = await document.GetSyntaxRootAsync(cancellationToken);

var newRoot = root.ReplaceNode(invocationExpr, statement);

var newDocument = document.WithSyntaxRoot(newRoot);

return newDocument;
}

但是,我有两个悬而未决的问题:

1) 如何将所需的 using Trilogy.Miscellaneous 添加到文件顶部。

2) 如何检测我的项目是否引用了所需的程序集。在这种情况下,如果我的程序集 Trilogy.Common 没有被引用,我要么不提供代码修复,要么我会建议 string.Join("", A, B) 而不是我自己的 SafeJoin 实现。

更新

我已经通过如下更新我的代码解决了#1...

var newRoot = root.ReplaceNode(invocationExpr, statement);

// Iterate through our usings to see if we've got what we need...
if (root?.Usings.Any(u => u.Name.ToString() == "Trilogy.Miscellaneous") == false)
{
// Create and add the using statement...
var usingStatement = UsingDirective(QualifiedName(IdentifierName("Trilogy"), IdentifierName("Miscellaneous")));
newRoot = newRoot.AddUsings(usingStatement);
}

var newDocument = document.WithSyntaxRoot(newRoot);

return newDocument;

仍然希望对第 2 项有所帮助。

最佳答案

我最终向 RegisterCodeFixesAsync 方法添加了一些代码。我觉得对我所需的程序集的测试有点 hack,所以如果有人发布这个问题,我会接受一个更好的答案。

   public sealed override async Task RegisterCodeFixesAsync(CodeFixContext context)
{
var root = await context.Document.GetSyntaxRootAsync(context.CancellationToken).ConfigureAwait(false);

var diagnostic = context.Diagnostics.First();
var diagnosticSpan = diagnostic.Location.SourceSpan;

// Find the type invocation expression identified by the diagnostic.
var invocationExpr = root.FindToken(
diagnosticSpan.Start).Parent.AncestorsAndSelf()
.OfType<InvocationExpressionSyntax>().First();

// Get the symantec model from the document
var semanticModel = await context.Document.GetSemanticModelAsync(context.CancellationToken).ConfigureAwait(false);

// Check for the assembly we need. I suspect there is a better way...
var hasAssembly = semanticModel.Compilation.ExternalReferences.Any(er => er.Properties.Kind == MetadataImageKind.Assembly && er.Display.EndsWith("Trilogy.Common.dll"));

// Register a code action that will invoke the fix, but only
// if we have the assembly that we need
if (hasAssembly)
context.RegisterCodeFix(
CodeAction.Create(title, c => UseJoinAsync(
context.Document, invocationExpr, c), equivalenceKey: title), diagnostic);
}

关于c# - 在 Roslyn 代码修复中有条件地添加 "using"语句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50553342/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com