Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add codefix for MSTEST0020 #3798

Merged
merged 14 commits into from
Sep 10, 2024

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -153,4 +153,7 @@
<data name="ReplaceWithTestCleanuFix" xml:space="preserve">
<value>Replace 'Dispose' with a TestCleanup method</value>
</data>
<data name="ReplaceWithConstructorFix" xml:space="preserve">
<value>Replace TestInitialize method with constructor</value>
</data>
</root>
Original file line number Diff line number Diff line change
@@ -0,0 +1,104 @@
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.

using System.Collections.Immutable;
using System.Composition;

using Analyzer.Utilities;

using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.CodeActions;
using Microsoft.CodeAnalysis.CodeFixes;
using Microsoft.CodeAnalysis.CSharp;
using Microsoft.CodeAnalysis.CSharp.Syntax;
using Microsoft.CodeAnalysis.Editing;
using Microsoft.CodeAnalysis.Text;

using MSTest.Analyzers.Helpers;

namespace MSTest.Analyzers;

[ExportCodeFixProvider(LanguageNames.CSharp, Name = nameof(ClassInitializeShouldBeValidFixer))]
[Shared]
public sealed class PreferConstructorOverTestInitializeFixer : CodeFixProvider
{
public override ImmutableArray<string> FixableDiagnosticIds { get; }
= ImmutableArray.Create(DiagnosticIds.PreferConstructorOverTestInitializeRuleId);

public override FixAllProvider GetFixAllProvider()
// See https://github.com/dotnet/roslyn/blob/main/docs/analyzers/FixAllProvider.md for more information on Fix All Providers
=> WellKnownFixAllProviders.BatchFixer;

public override async Task RegisterCodeFixesAsync(CodeFixContext context)
{
SyntaxNode root = await context.Document.GetRequiredSyntaxRootAsync(context.CancellationToken).ConfigureAwait(false);

Diagnostic diagnostic = context.Diagnostics[0];
TextSpan diagnosticSpan = diagnostic.Location.SourceSpan;

SyntaxToken syntaxToken = root.FindToken(diagnosticSpan.Start);
if (syntaxToken.Parent is null)
{
return;
}

// Find the method declaration identified by the diagnostic.
MethodDeclarationSyntax methodDeclaration = syntaxToken.Parent.AncestorsAndSelf().OfType<MethodDeclarationSyntax>().FirstOrDefault();

context.RegisterCodeFix(
CodeAction.Create(
CodeFixResources.ReplaceWithConstructorFix,
c => ReplaceTestInitializeWithConstructorAsync(context.Document, methodDeclaration, c),
nameof(PreferConstructorOverTestInitializeFixer)),
diagnostic);
}

private static async Task<Document> ReplaceTestInitializeWithConstructorAsync(Document document, MethodDeclarationSyntax testInitializeMethod, CancellationToken cancellationToken)
{
DocumentEditor editor = await DocumentEditor.CreateAsync(document, cancellationToken).ConfigureAwait(false);

// Find the class containing the method
if (testInitializeMethod.Parent is ClassDeclarationSyntax containingClass)
{
ConstructorDeclarationSyntax existingConstructor = containingClass.Members
.OfType<ConstructorDeclarationSyntax>()
.FirstOrDefault();

// Move the body of the TestInitialize method
BlockSyntax? testInitializeBody = testInitializeMethod.Body;

if (existingConstructor != null)
{
StatementSyntax[]? testInitializeStatements = testInitializeBody?.Statements.ToArray();
ConstructorDeclarationSyntax newConstructor;

// If a constructor already exists, append the body of the TestInitialize method to it
if (existingConstructor.Body != null)
{
BlockSyntax newConstructorBody = existingConstructor.Body.AddStatements(testInitializeStatements ?? Array.Empty<StatementSyntax>());
newConstructor = existingConstructor.WithBody(newConstructorBody);
}
else
{
newConstructor = existingConstructor.WithBody(testInitializeBody);
}

editor.ReplaceNode(existingConstructor, newConstructor);
}
else
{
// Create a new constructor with the TestInitialize body if one doesn't exist
ConstructorDeclarationSyntax constructor = SyntaxFactory.ConstructorDeclaration(containingClass.Identifier)
.WithModifiers(SyntaxFactory.TokenList(SyntaxFactory.Token(SyntaxKind.PublicKeyword)))
.WithBody(testInitializeBody);

editor.AddMember(containingClass, constructor);
}

// Remove the TestInitialize method
editor.RemoveNode(testInitializeMethod);
}

return editor.GetChangedDocument();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Přidat [TestClass]</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">„[TestClass]“ hinzufügen</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Agregar '[TestClass]'</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Ajouter « [TestClass] »</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Aggiungi '[TestClass]'</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">'[TestClass]' の追加</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">'[TestClass]' 추가</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Dodaj element „[TestClass]”</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Adicionar '[TestClass]'</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">Добавить '[TestClass]'</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">'[TestClass]' Ekle</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">添加“[TestClass]”</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,11 @@
<target state="translated">新增 '[TestClass]'</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithConstructorFix">
<source>Replace TestInitialize method with constructor</source>
<target state="new">Replace TestInitialize method with constructor</target>
<note />
</trans-unit>
<trans-unit id="ReplaceWithFailAssertionFix">
<source>Replace the assertion with 'Assert.Fail()'</source>
<target state="new">Replace the assertion with 'Assert.Fail()'</target>
Expand Down
Loading