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 ParameterMustBeLastIfNonScalarAnalyzer : AnalyzerBase { public ParameterMustBeLastIfNonScalarAnalyzer() : base( "Parameters of non-scalar types must be last in order", "This parameter has a non-scalar type so it must be last in order (its order must be highest within the command).") { } private static bool IsScalar(ITypeSymbol type) => type.DisplayNameMatches("string") || type.DisplayNameMatches("System.String") || !type.AllInterfaces .Select(i => i.ConstructedFrom) .Any(s => s.DisplayNameMatches("System.Collections.Generic.IEnumerable")); private void Analyze( SyntaxNodeAnalysisContext context, PropertyDeclarationSyntax propertyDeclaration, IPropertySymbol property) { if (property.ContainingType is null) return; if (IsScalar(property.Type)) return; var parameter = CommandParameterSymbol.TryResolve(property); if (parameter is null) return; var otherProperties = property .ContainingType .GetMembers() .OfType() .Where(m => !m.Equals(property, SymbolEqualityComparer.Default)) .ToArray(); foreach (var otherProperty in otherProperties) { var otherParameter = CommandParameterSymbol.TryResolve(otherProperty); if (otherParameter is null) continue; if (otherParameter.Order > parameter.Order) { context.ReportDiagnostic(CreateDiagnostic(propertyDeclaration.GetLocation())); } } } public override void Initialize(AnalysisContext context) { base.Initialize(context); context.HandlePropertyDeclaration(Analyze); } } }