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

Attribute Binding Roslyn Analyzers #301

Merged
merged 6 commits into from
May 3, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion src/Analyzers/AnalyzerReleases.Unshipped.md
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
; Unshipped analyzer release
; Unshipped analyzer release
; https://github.com/dotnet/roslyn-analyzers/blob/main/src/Microsoft.CodeAnalysis.Analyzers/ReleaseTrackingAnalyzers.Help.md

### New Rules
Expand All @@ -8,3 +8,6 @@ Rule ID | Category | Severity | Notes
DURABLE0001 | Orchestration | Warning | DateTimeOrchestrationAnalyzer
DURABLE0002 | Orchestration | Warning | GuidOrchestrationAnalyzer
DURABLE0003 | Orchestration | Warning | DelayOrchestrationAnalyzer
DURABLE1001 | Attribute Binding | Error | OrchestrationTriggerBindingAnalyzer
DURABLE1002 | Attribute Binding | Error | DurableClientBindingAnalyzer
DURABLE1003 | Attribute Binding | Error | EntityTriggerBindingAnalyzer
5 changes: 5 additions & 0 deletions src/Analyzers/AnalyzersCategories.cs
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,9 @@ static class AnalyzersCategories
/// The category for the orchestration related analyzers.
/// </summary>
public const string Orchestration = "Orchestration";

/// <summary>
/// The category for the attribute binding related analyzers.
/// </summary>
public const string AttributeBinding = "Attribute Binding";
}
51 changes: 51 additions & 0 deletions src/Analyzers/AttributeBinding/DurableClientBindingAnalyzer.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using System.Collections.Immutable;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Diagnostics;

namespace Microsoft.DurableTask.Analyzers.AttributeBinding;

/// <summary>
/// Analyzer that matches 'DurableClientAttribute' with 'DurableTaskClient' parameters.
/// </summary>
[DiagnosticAnalyzer(LanguageNames.CSharp)]
public sealed class DurableClientBindingAnalyzer : MatchingAttributeBindingAnalyzer
{
/// <summary>
/// Diagnostic ID supported for the analyzer.
/// </summary>
public const string DiagnosticId = "DURABLE1002";
allantargino marked this conversation as resolved.
Show resolved Hide resolved

static readonly LocalizableString Title = new LocalizableResourceString(nameof(Resources.DurableClientBindingAnalyzerTitle), Resources.ResourceManager, typeof(Resources));
static readonly LocalizableString MessageFormat = new LocalizableResourceString(nameof(Resources.DurableClientBindingAnalyzerMessageFormat), Resources.ResourceManager, typeof(Resources));

static readonly DiagnosticDescriptor Rule = new(
DiagnosticId,
Title,
MessageFormat,
AnalyzersCategories.AttributeBinding,
DiagnosticSeverity.Error,
isEnabledByDefault: true);

/// <inheritdoc/>
public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => [Rule];

/// <inheritdoc/>
protected override ExpectedBinding GetExpectedBinding(KnownTypeSymbols knownTypeSymbols)
{
return new ExpectedBinding()
{
Attribute = knownTypeSymbols.DurableClientAttribute,
Type = knownTypeSymbols.DurableTaskClient,
};
}

/// <inheritdoc/>
protected override void ReportDiagnostic(SymbolAnalysisContext ctx, ExpectedBinding expected, IParameterSymbol parameter)
{
string wrongType = parameter.Type.ToDisplayString(SymbolDisplayFormat.CSharpShortErrorMessageFormat);
ctx.ReportDiagnostic(RoslynExtensions.BuildDiagnostic(Rule, parameter, wrongType));
}
}
51 changes: 51 additions & 0 deletions src/Analyzers/AttributeBinding/EntityTriggerBindingAnalyzer.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using System.Collections.Immutable;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Diagnostics;

namespace Microsoft.DurableTask.Analyzers.AttributeBinding;

/// <summary>
/// Analyzer that matches 'EntityTriggerAttribute' with 'TaskEntityDispatcher' parameters.
/// </summary>
[DiagnosticAnalyzer(LanguageNames.CSharp)]
public sealed class EntityTriggerBindingAnalyzer : MatchingAttributeBindingAnalyzer
{
/// <summary>
/// Diagnostic ID supported for the analyzer.
/// </summary>
public const string DiagnosticId = "DURABLE1003";

static readonly LocalizableString Title = new LocalizableResourceString(nameof(Resources.EntityTriggerBindingAnalyzerTitle), Resources.ResourceManager, typeof(Resources));
static readonly LocalizableString MessageFormat = new LocalizableResourceString(nameof(Resources.EntityTriggerBindingAnalyzerMessageFormat), Resources.ResourceManager, typeof(Resources));

static readonly DiagnosticDescriptor Rule = new(
DiagnosticId,
Title,
MessageFormat,
AnalyzersCategories.AttributeBinding,
DiagnosticSeverity.Error,
isEnabledByDefault: true);

/// <inheritdoc/>
public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => [Rule];

/// <inheritdoc/>
protected override ExpectedBinding GetExpectedBinding(KnownTypeSymbols knownTypeSymbols)
{
return new ExpectedBinding()
{
Attribute = knownTypeSymbols.EntityTriggerAttribute,
Type = knownTypeSymbols.TaskEntityDispatcher,
};
}

/// <inheritdoc/>
protected override void ReportDiagnostic(SymbolAnalysisContext ctx, ExpectedBinding expected, IParameterSymbol parameter)
{
string wrongType = parameter.Type.ToDisplayString(SymbolDisplayFormat.CSharpShortErrorMessageFormat);
ctx.ReportDiagnostic(RoslynExtensions.BuildDiagnostic(Rule, parameter, wrongType));
}
}
79 changes: 79 additions & 0 deletions src/Analyzers/AttributeBinding/MatchingAttributeBindingAnalyzer.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Diagnostics;

namespace Microsoft.DurableTask.Analyzers.AttributeBinding;

/// <summary>
/// Expected attribute binding for a given parameter type.
/// </summary>
public struct ExpectedBinding
{
/// <summary>
/// Gets or sets the expected attribute.
/// </summary>
public INamedTypeSymbol? Attribute { get; set; }

/// <summary>
/// Gets or sets the expected type.
/// </summary>
public INamedTypeSymbol? Type { get; set; }
}

/// <summary>
/// Analyzer that inspects the parameter type of a method to ensure it matches the expected attribute binding.
/// It expects one parameter in the DiagnosticRule message template, so the analyzer can report the wrong type.
/// </summary>
public abstract class MatchingAttributeBindingAnalyzer : DiagnosticAnalyzer
{
/// <inheritdoc/>
public override void Initialize(AnalysisContext context)
{
context.EnableConcurrentExecution();
context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None);

context.RegisterCompilationStartAction(
ctx =>
{
KnownTypeSymbols knownTypeSymbols = new(ctx.Compilation);
allantargino marked this conversation as resolved.
Show resolved Hide resolved

ExpectedBinding expectedBinding = this.GetExpectedBinding(knownTypeSymbols);
if (expectedBinding.Attribute is null || expectedBinding.Type is null)
{
return;
}

ctx.RegisterSymbolAction(c => this.Analyze(c, expectedBinding), SymbolKind.Parameter);
});
}

/// <summary>
/// Gets the expected attribute binding and the related type to be used during parameters analysis.
/// </summary>
/// <param name="knownTypeSymbols">The set of well-known types.</param>
/// <returns>The expected binding for this analyzer.</returns>
protected abstract ExpectedBinding GetExpectedBinding(KnownTypeSymbols knownTypeSymbols);

/// <summary>
/// After an incorrect attribute/type matching is found, this method is called so the concrete implementation can report a diagnostic.
/// </summary>
/// <param name="ctx">Context for a symbol action. Allows reporting a diagnostic.</param>
/// <param name="expected">Expected binding for a attribute/type.</param>
allantargino marked this conversation as resolved.
Show resolved Hide resolved
/// <param name="parameter">Analyzed parameter symbol.</param>
protected abstract void ReportDiagnostic(SymbolAnalysisContext ctx, ExpectedBinding expected, IParameterSymbol parameter);

void Analyze(SymbolAnalysisContext ctx, ExpectedBinding expected)
{
IParameterSymbol parameter = (IParameterSymbol)ctx.Symbol;

if (parameter.GetAttributes().Any(a => expected.Attribute!.Equals(a.AttributeClass, SymbolEqualityComparer.Default)))
{
if (!parameter.Type.Equals(expected.Type, SymbolEqualityComparer.Default))
{
this.ReportDiagnostic(ctx, expected, parameter);
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using System.Collections.Immutable;
using Microsoft.CodeAnalysis;
using Microsoft.CodeAnalysis.Diagnostics;

namespace Microsoft.DurableTask.Analyzers.AttributeBinding;

/// <summary>
/// Analyzer that matches 'OrchestrationTriggerAttribute' with 'TaskOrchestrationContext' parameters.
/// </summary>
[DiagnosticAnalyzer(LanguageNames.CSharp)]
public sealed class OrchestrationTriggerBindingAnalyzer : MatchingAttributeBindingAnalyzer
{
/// <summary>
/// Diagnostic ID supported for the analyzer.
/// </summary>
public const string DiagnosticId = "DURABLE1001";

static readonly LocalizableString Title = new LocalizableResourceString(nameof(Resources.OrchestrationTriggerBindingAnalyzerTitle), Resources.ResourceManager, typeof(Resources));
static readonly LocalizableString MessageFormat = new LocalizableResourceString(nameof(Resources.OrchestrationTriggerBindingAnalyzerMessageFormat), Resources.ResourceManager, typeof(Resources));

static readonly DiagnosticDescriptor Rule = new(
DiagnosticId,
Title,
MessageFormat,
AnalyzersCategories.AttributeBinding,
DiagnosticSeverity.Error,
isEnabledByDefault: true);

/// <inheritdoc/>
public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => [Rule];

/// <inheritdoc/>
protected override ExpectedBinding GetExpectedBinding(KnownTypeSymbols knownTypeSymbols)
{
return new ExpectedBinding()
{
Attribute = knownTypeSymbols.FunctionOrchestrationAttribute,
Type = knownTypeSymbols.TaskOrchestrationContext,
};
}

/// <inheritdoc/>
protected override void ReportDiagnostic(SymbolAnalysisContext ctx, ExpectedBinding expected, IParameterSymbol parameter)
{
string wrongType = parameter.Type.ToDisplayString(SymbolDisplayFormat.CSharpShortErrorMessageFormat);
ctx.ReportDiagnostic(RoslynExtensions.BuildDiagnostic(Rule, parameter, wrongType));
}
}
30 changes: 30 additions & 0 deletions src/Analyzers/KnownTypeSymbols.cs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,11 @@ public sealed class KnownTypeSymbols(Compilation compilation)
INamedTypeSymbol? taskOrchestratorInterface;
INamedTypeSymbol? taskOrchestratorBaseClass;
INamedTypeSymbol? durableTaskRegistry;
INamedTypeSymbol? taskOrchestrationContext;
INamedTypeSymbol? durableClientAttribute;
INamedTypeSymbol? durableTaskClient;
INamedTypeSymbol? entityTriggerAttribute;
INamedTypeSymbol? taskEntityDispatcher;
INamedTypeSymbol? guid;
INamedTypeSymbol? thread;
INamedTypeSymbol? task;
Expand Down Expand Up @@ -50,6 +55,31 @@ public sealed class KnownTypeSymbols(Compilation compilation)
/// </summary>
public INamedTypeSymbol? DurableTaskRegistry => this.GetOrResolveFullyQualifiedType("Microsoft.DurableTask.DurableTaskRegistry", ref this.durableTaskRegistry);

/// <summary>
/// Gets a TaskOrchestrationContext type symbol.
/// </summary>
public INamedTypeSymbol? TaskOrchestrationContext => this.GetOrResolveFullyQualifiedType("Microsoft.DurableTask.TaskOrchestrationContext", ref this.taskOrchestrationContext);

/// <summary>
/// Gets a DurableClientAttribute type symbol.
/// </summary>
public INamedTypeSymbol? DurableClientAttribute => this.GetOrResolveFullyQualifiedType("Microsoft.Azure.Functions.Worker.DurableClientAttribute", ref this.durableClientAttribute);
allantargino marked this conversation as resolved.
Show resolved Hide resolved

/// <summary>
/// Gets a DurableTaskClient type symbol.
/// </summary>
public INamedTypeSymbol? DurableTaskClient => this.GetOrResolveFullyQualifiedType("Microsoft.DurableTask.Client.DurableTaskClient", ref this.durableTaskClient);

/// <summary>
/// Gets an EntityTriggerAttribute type symbol.
/// </summary>
public INamedTypeSymbol? EntityTriggerAttribute => this.GetOrResolveFullyQualifiedType("Microsoft.Azure.Functions.Worker.EntityTriggerAttribute", ref this.entityTriggerAttribute);

/// <summary>
/// Gets a TaskEntityDispatcher type symbol.
/// </summary>
public INamedTypeSymbol? TaskEntityDispatcher => this.GetOrResolveFullyQualifiedType("Microsoft.Azure.Functions.Worker.TaskEntityDispatcher", ref this.taskEntityDispatcher);

/// <summary>
/// Gets a Guid type symbol.
/// </summary>
Expand Down
18 changes: 18 additions & 0 deletions src/Analyzers/Resources.resx
Original file line number Diff line number Diff line change
Expand Up @@ -135,4 +135,22 @@
<data name="DelayOrchestrationAnalyzerTitle" xml:space="preserve">
<value>Thread.Sleep and Task.Delay calls are not allowed inside an orchestrator</value>
</data>
<data name="OrchestrationTriggerBindingAnalyzerMessageFormat" xml:space="preserve">
<value>[OrchestrationTrigger] is associated with the wrong type '{0}', it must be used with TaskOrchestrationContext</value>
</data>
<data name="OrchestrationTriggerBindingAnalyzerTitle" xml:space="preserve">
<value>[OrchestrationTrigger] must be used with TaskOrchestrationContext</value>
</data>
<data name="EntityTriggerBindingAnalyzerMessageFormat" xml:space="preserve">
<value>[EntityTrigger] is associated with the wrong type '{0}', it must be used with TaskEntityDispatcher</value>
</data>
<data name="EntityTriggerBindingAnalyzerTitle" xml:space="preserve">
<value>[EntityTrigger] must be used with TaskOrchestrationContext</value>
</data>
<data name="DurableClientBindingAnalyzerMessageFormat" xml:space="preserve">
<value>[DurableClient] is associated with the wrong type '{0}', it must be used with DurableTaskClient</value>
</data>
<data name="DurableClientBindingAnalyzerTitle" xml:space="preserve">
<value>[DurableClient] must be used with DurableTaskClient</value>
</data>
</root>
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using Microsoft.DurableTask.Analyzers.AttributeBinding;

namespace Microsoft.DurableTask.Analyzers.Tests.AttributeBinding;

public class DurableClientBindingAnalyzerTests : MatchingAttributeBindingSpecificationTests<DurableClientBindingAnalyzer>
{
protected override string ExpectedDiagnosticId => DurableClientBindingAnalyzer.DiagnosticId;

protected override string ExpectedAttribute => "[DurableClient]";

protected override string ExpectedType => "DurableTaskClient";
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT License.

using Microsoft.DurableTask.Analyzers.AttributeBinding;

namespace Microsoft.DurableTask.Analyzers.Tests.AttributeBinding;

public class EntityTriggerBindingAnalyzerTests : MatchingAttributeBindingSpecificationTests<EntityTriggerBindingAnalyzer>
{
protected override string ExpectedDiagnosticId => EntityTriggerBindingAnalyzer.DiagnosticId;

protected override string ExpectedAttribute => "[EntityTrigger]";

protected override string ExpectedType => "TaskEntityDispatcher";
}
Loading
Loading