From 6cae4cfb3af615ee006bf139c0aaea59e2fd7849 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bj=C3=B6rn=20Hellander?= Date: Wed, 24 May 2023 05:59:29 +0200 Subject: [PATCH 1/2] Update SA1212 to also trigger for an init accessor before a getter #3652 --- .../OrderingRules/SA1212CSharp9UnitTests.cs | 82 +++++++++++++++++++ .../OrderingRules/SA1212UnitTests.cs | 2 +- .../Lightup/SyntaxKindEx.cs | 1 + .../SA1212PropertyAccessorsMustFollowOrder.cs | 3 +- documentation/SA1212.md | 4 +- 5 files changed, 88 insertions(+), 4 deletions(-) diff --git a/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs b/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs index 15301157f..39837749c 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs @@ -3,9 +3,91 @@ namespace StyleCop.Analyzers.Test.CSharp9.OrderingRules { + using System.Threading; + using System.Threading.Tasks; using StyleCop.Analyzers.Test.CSharp8.OrderingRules; + using StyleCop.Analyzers.Test.Verifiers; + using Xunit; + using static StyleCop.Analyzers.Test.Verifiers.StyleCopCodeFixVerifier< + StyleCop.Analyzers.OrderingRules.SA1212PropertyAccessorsMustFollowOrder, + StyleCop.Analyzers.OrderingRules.SA1212SA1213CodeFixProvider>; public class SA1212CSharp9UnitTests : SA1212CSharp8UnitTests { + [Fact] + [WorkItem(3652, "https://github.com/DotNetAnalyzers/StyleCopAnalyzers/issues/3652")] + public async Task TestAutoPropertyDeclarationInitBeforeGetterAsync() + { + var testCode = @" +public class Foo +{ + public int Prop { [|init;|] get; } +}"; + + var fixedCode = @" +public class Foo +{ + public int Prop { get; init; } +}"; + + var test = new CSharpTest + { + TestCode = testCode, + FixedCode = fixedCode, + ReferenceAssemblies = GenericAnalyzerTest.ReferenceAssembliesNet50, + }; + await test.RunAsync(CancellationToken.None).ConfigureAwait(false); + } + + [Fact] + [WorkItem(3652, "https://github.com/DotNetAnalyzers/StyleCopAnalyzers/issues/3652")] + public async Task TestPropertyWithBackingFieldDeclarationInitBeforeGetterAsync() + { + var testCode = @" +public class Foo +{ + private int i = 0; + + public int Prop + { + [|init + { + i = value; + }|] + + get + { + return i; + } + } +}"; + + var fixedCode = @" +public class Foo +{ + private int i = 0; + + public int Prop + { + get + { + return i; + } + + init + { + i = value; + } + } +}"; + + var test = new CSharpTest + { + TestCode = testCode, + FixedCode = fixedCode, + ReferenceAssemblies = GenericAnalyzerTest.ReferenceAssembliesNet50, + }; + await test.RunAsync(CancellationToken.None).ConfigureAwait(false); + } } } diff --git a/StyleCop.Analyzers/StyleCop.Analyzers.Test/OrderingRules/SA1212UnitTests.cs b/StyleCop.Analyzers/StyleCop.Analyzers.Test/OrderingRules/SA1212UnitTests.cs index a9e7d07af..987e709b2 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers.Test/OrderingRules/SA1212UnitTests.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers.Test/OrderingRules/SA1212UnitTests.cs @@ -339,7 +339,7 @@ public int Prop } [Fact] - public async Task TestAutoPropertydDeclarationSetterBeforeGetterAsync() + public async Task TestAutoPropertyDeclarationSetterBeforeGetterAsync() { var testCode = @" public class Foo diff --git a/StyleCop.Analyzers/StyleCop.Analyzers/Lightup/SyntaxKindEx.cs b/StyleCop.Analyzers/StyleCop.Analyzers/Lightup/SyntaxKindEx.cs index 677453d6f..62b06c867 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers/Lightup/SyntaxKindEx.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers/Lightup/SyntaxKindEx.cs @@ -62,6 +62,7 @@ internal static class SyntaxKindEx public const SyntaxKind NullableDirectiveTrivia = (SyntaxKind)9055; public const SyntaxKind FunctionPointerType = (SyntaxKind)9056; public const SyntaxKind FunctionPointerParameter = (SyntaxKind)9057; + public const SyntaxKind InitAccessorDeclaration = (SyntaxKind)9060; public const SyntaxKind WithExpression = (SyntaxKind)9061; public const SyntaxKind WithInitializerExpression = (SyntaxKind)9062; public const SyntaxKind RecordDeclaration = (SyntaxKind)9063; diff --git a/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs b/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs index 083f7ce6b..a4b135697 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs @@ -11,6 +11,7 @@ namespace StyleCop.Analyzers.OrderingRules using Microsoft.CodeAnalysis.CSharp; using Microsoft.CodeAnalysis.CSharp.Syntax; using Microsoft.CodeAnalysis.Diagnostics; + using StyleCop.Analyzers.Lightup; /// /// A get accessor appears after a set accessor within a property or indexer. @@ -99,7 +100,7 @@ private static void AnalyzeProperty(SyntaxNodeAnalysisContext context, BasePrope return; } - if (accessors[0].Kind() == SyntaxKind.SetAccessorDeclaration && + if ((accessors[0].Kind() == SyntaxKind.SetAccessorDeclaration || accessors[0].Kind() == SyntaxKindEx.InitAccessorDeclaration) && accessors[1].Kind() == SyntaxKind.GetAccessorDeclaration) { context.ReportDiagnostic(Diagnostic.Create(Descriptor, accessors[0].GetLocation())); diff --git a/documentation/SA1212.md b/documentation/SA1212.md index e735b0c09..2790f8ae4 100644 --- a/documentation/SA1212.md +++ b/documentation/SA1212.md @@ -17,11 +17,11 @@ ## Cause -A get accessor appears after a set accessor within a property or indexer. +A get accessor appears after a set or init accessor within a property or indexer. ## Rule description -A violation of this rule occurs when a get accessor is placed after a set accessor within a property or indexer. To comply with this rule, the get accessor should appear before the set accessor. +A violation of this rule occurs when a get accessor is placed after a set or init accessor within a property or indexer. To comply with this rule, the get accessor should appear first. For example, the following code would raise an instance of this violation: From 280b11ab84d8ee27827064f058d95a52c8b83a29 Mon Sep 17 00:00:00 2001 From: Sam Harwell Date: Wed, 21 Jun 2023 08:41:22 -0500 Subject: [PATCH 2/2] Apply suggestions from code review --- .../OrderingRules/SA1212CSharp9UnitTests.cs | 16 +++++++--------- .../SA1212PropertyAccessorsMustFollowOrder.cs | 2 +- 2 files changed, 8 insertions(+), 10 deletions(-) diff --git a/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs b/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs index 39837749c..9c0dfe925 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers.Test.CSharp9/OrderingRules/SA1212CSharp9UnitTests.cs @@ -5,8 +5,8 @@ namespace StyleCop.Analyzers.Test.CSharp9.OrderingRules { using System.Threading; using System.Threading.Tasks; + using Microsoft.CodeAnalysis.Testing; using StyleCop.Analyzers.Test.CSharp8.OrderingRules; - using StyleCop.Analyzers.Test.Verifiers; using Xunit; using static StyleCop.Analyzers.Test.Verifiers.StyleCopCodeFixVerifier< StyleCop.Analyzers.OrderingRules.SA1212PropertyAccessorsMustFollowOrder, @@ -30,13 +30,12 @@ public class Foo public int Prop { get; init; } }"; - var test = new CSharpTest + await new CSharpTest { TestCode = testCode, FixedCode = fixedCode, - ReferenceAssemblies = GenericAnalyzerTest.ReferenceAssembliesNet50, - }; - await test.RunAsync(CancellationToken.None).ConfigureAwait(false); + ReferenceAssemblies = ReferenceAssemblies.Net.Net50, + }.RunAsync(CancellationToken.None).ConfigureAwait(false); } [Fact] @@ -81,13 +80,12 @@ public int Prop } }"; - var test = new CSharpTest + await new CSharpTest { TestCode = testCode, FixedCode = fixedCode, - ReferenceAssemblies = GenericAnalyzerTest.ReferenceAssembliesNet50, - }; - await test.RunAsync(CancellationToken.None).ConfigureAwait(false); + ReferenceAssemblies = ReferenceAssemblies.Net.Net50, + }.RunAsync(CancellationToken.None).ConfigureAwait(false); } } } diff --git a/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs b/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs index a4b135697..a2285c411 100644 --- a/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs +++ b/StyleCop.Analyzers/StyleCop.Analyzers/OrderingRules/SA1212PropertyAccessorsMustFollowOrder.cs @@ -100,7 +100,7 @@ private static void AnalyzeProperty(SyntaxNodeAnalysisContext context, BasePrope return; } - if ((accessors[0].Kind() == SyntaxKind.SetAccessorDeclaration || accessors[0].Kind() == SyntaxKindEx.InitAccessorDeclaration) && + if ((accessors[0].Kind() is SyntaxKind.SetAccessorDeclaration or SyntaxKindEx.InitAccessorDeclaration) && accessors[1].Kind() == SyntaxKind.GetAccessorDeclaration) { context.ReportDiagnostic(Diagnostic.Create(Descriptor, accessors[0].GetLocation()));