From 940c320e4250685af228641652b5ae47b1dcb4c9 Mon Sep 17 00:00:00 2001 From: Youssef Victor Date: Wed, 18 Dec 2024 11:21:15 +0100 Subject: [PATCH] Fix false positive for PreferTestCleanupOverDispose on non-test classes (#4380) --- .../PreferTestCleanupOverDisposeAnalyzer.cs | 16 +++++++++++----- ...PreferTestCleanupOverDisposeAnalyzerTests.cs | 17 +++++++++++++++++ 2 files changed, 28 insertions(+), 5 deletions(-) diff --git a/src/Analyzers/MSTest.Analyzers/PreferTestCleanupOverDisposeAnalyzer.cs b/src/Analyzers/MSTest.Analyzers/PreferTestCleanupOverDisposeAnalyzer.cs index 1e22fec61a..b63e66970d 100644 --- a/src/Analyzers/MSTest.Analyzers/PreferTestCleanupOverDisposeAnalyzer.cs +++ b/src/Analyzers/MSTest.Analyzers/PreferTestCleanupOverDisposeAnalyzer.cs @@ -40,22 +40,28 @@ public override void Initialize(AnalysisContext context) context.RegisterCompilationStartAction(context => { - if (context.Compilation.TryGetOrCreateTypeByMetadataName(WellKnownTypeNames.SystemIDisposable, out INamedTypeSymbol? idisposableSymbol)) + if (context.Compilation.TryGetOrCreateTypeByMetadataName(WellKnownTypeNames.SystemIDisposable, out INamedTypeSymbol? idisposableSymbol) && + context.Compilation.TryGetOrCreateTypeByMetadataName(WellKnownTypeNames.MicrosoftVisualStudioTestToolsUnitTestingTestClassAttribute, out INamedTypeSymbol? testClassAttributeSymbol)) { INamedTypeSymbol? iasyncDisposableSymbol = context.Compilation.GetOrCreateTypeByMetadataName(WellKnownTypeNames.SystemIAsyncDisposable); INamedTypeSymbol? valueTaskSymbol = context.Compilation.GetOrCreateTypeByMetadataName(WellKnownTypeNames.SystemThreadingTasksValueTask); - context.RegisterSymbolAction(context => AnalyzeSymbol(context, idisposableSymbol, iasyncDisposableSymbol, valueTaskSymbol), SymbolKind.Method); + context.RegisterSymbolAction(context => AnalyzeSymbol(context, testClassAttributeSymbol, idisposableSymbol, iasyncDisposableSymbol, valueTaskSymbol), SymbolKind.Method); } }); } - private static void AnalyzeSymbol(SymbolAnalysisContext context, INamedTypeSymbol idisposableSymbol, INamedTypeSymbol? iasyncDisposableSymbol, + private static void AnalyzeSymbol( + SymbolAnalysisContext context, + INamedTypeSymbol testClassAttributeSymbol, + INamedTypeSymbol idisposableSymbol, + INamedTypeSymbol? iasyncDisposableSymbol, INamedTypeSymbol? valueTaskSymbol) { var methodSymbol = (IMethodSymbol)context.Symbol; - if (methodSymbol.IsAsyncDisposeImplementation(iasyncDisposableSymbol, valueTaskSymbol) - || methodSymbol.IsDisposeImplementation(idisposableSymbol)) + if (methodSymbol.ContainingType.GetAttributes().Any(x => x.AttributeClass.Inherits(testClassAttributeSymbol)) && + (methodSymbol.IsAsyncDisposeImplementation(iasyncDisposableSymbol, valueTaskSymbol) + || methodSymbol.IsDisposeImplementation(idisposableSymbol))) { context.ReportDiagnostic(methodSymbol.CreateDiagnostic(Rule)); } diff --git a/test/UnitTests/MSTest.Analyzers.UnitTests/PreferTestCleanupOverDisposeAnalyzerTests.cs b/test/UnitTests/MSTest.Analyzers.UnitTests/PreferTestCleanupOverDisposeAnalyzerTests.cs index 278c0a417f..19c586ce4c 100644 --- a/test/UnitTests/MSTest.Analyzers.UnitTests/PreferTestCleanupOverDisposeAnalyzerTests.cs +++ b/test/UnitTests/MSTest.Analyzers.UnitTests/PreferTestCleanupOverDisposeAnalyzerTests.cs @@ -10,6 +10,23 @@ namespace MSTest.Analyzers.Test; [TestGroup] public sealed class PreferTestCleanupOverDisposeAnalyzerTests(ITestExecutionContext testExecutionContext) : TestBase(testExecutionContext) { + public async Task WhenNonTestClassHasDispose_NoDiagnostic() + { + string code = """ + using System; + using Microsoft.VisualStudio.TestTools.UnitTesting; + + public class MyNonTestClass : IDisposable + { + public void Dispose() + { + } + } + """; + + await VerifyCS.VerifyCodeFixAsync(code, code); + } + public async Task WhenTestClassHasDispose_Diagnostic() { string code = """