mirror of
https://github.com/Tyrrrz/CliFx.git
synced 2025-10-25 15:19:17 +00:00
68 lines
2.3 KiB
C#
68 lines
2.3 KiB
C#
using System.Linq;
|
|
using CliFx.Analyzers.ObjectModel;
|
|
using CliFx.Analyzers.Utils.Extensions;
|
|
using Microsoft.CodeAnalysis;
|
|
using Microsoft.CodeAnalysis.CSharp.Syntax;
|
|
using Microsoft.CodeAnalysis.Diagnostics;
|
|
|
|
namespace CliFx.Analyzers;
|
|
|
|
[DiagnosticAnalyzer(LanguageNames.CSharp)]
|
|
public class OptionMustHaveValidConverterAnalyzer : AnalyzerBase
|
|
{
|
|
public OptionMustHaveValidConverterAnalyzer()
|
|
: base(
|
|
$"Option converters must derive from `{SymbolNames.CliFxBindingConverterClass}`",
|
|
$"Converter specified for this option must derive from a compatible `{SymbolNames.CliFxBindingConverterClass}`."
|
|
) { }
|
|
|
|
private void Analyze(
|
|
SyntaxNodeAnalysisContext context,
|
|
PropertyDeclarationSyntax propertyDeclaration,
|
|
IPropertySymbol property
|
|
)
|
|
{
|
|
var option = CommandOptionSymbol.TryResolve(property);
|
|
if (option is null)
|
|
return;
|
|
|
|
if (option.ConverterType is null)
|
|
return;
|
|
|
|
var converterValueType = option.ConverterType
|
|
.GetBaseTypes()
|
|
.FirstOrDefault(
|
|
t => t.ConstructedFrom.DisplayNameMatches(SymbolNames.CliFxBindingConverterClass)
|
|
)
|
|
?.TypeArguments.FirstOrDefault();
|
|
|
|
// Value returned by the converter must be assignable to the property type
|
|
var isCompatible =
|
|
converterValueType is not null
|
|
&& (
|
|
option.IsScalar()
|
|
// Scalar
|
|
? context.Compilation.IsAssignable(converterValueType, property.Type)
|
|
// Non-scalar (assume we can handle all IEnumerable types for simplicity)
|
|
: property.Type.TryGetEnumerableUnderlyingType() is { } enumerableUnderlyingType
|
|
&& context.Compilation.IsAssignable(
|
|
converterValueType,
|
|
enumerableUnderlyingType
|
|
)
|
|
);
|
|
|
|
if (!isCompatible)
|
|
{
|
|
context.ReportDiagnostic(
|
|
CreateDiagnostic(propertyDeclaration.Identifier.GetLocation())
|
|
);
|
|
}
|
|
}
|
|
|
|
public override void Initialize(AnalysisContext context)
|
|
{
|
|
base.Initialize(context);
|
|
context.HandlePropertyDeclaration(Analyze);
|
|
}
|
|
}
|