using System; 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 OptionMustHaveUniqueNameAnalyzer : AnalyzerBase { public OptionMustHaveUniqueNameAnalyzer() : base( "Options must have unique names", "This option's name must be unique within the command (comparison IS NOT case sensitive). " + "Specified name: `{0}`. " + "Property bound to another option with the same name: `{1}`.") { } private void Analyze( SyntaxNodeAnalysisContext context, PropertyDeclarationSyntax propertyDeclaration, IPropertySymbol property) { if (property.ContainingType is null) return; var option = CommandOptionSymbol.TryResolve(property); if (option is null) return; if (string.IsNullOrWhiteSpace(option.Name)) return; var otherProperties = property .ContainingType .GetMembers() .OfType() .Where(m => !m.Equals(property, SymbolEqualityComparer.Default)) .ToArray(); foreach (var otherProperty in otherProperties) { var otherOption = CommandOptionSymbol.TryResolve(otherProperty); if (otherOption is null) continue; if (string.IsNullOrWhiteSpace(otherOption.Name)) continue; if (string.Equals(option.Name, otherOption.Name, StringComparison.OrdinalIgnoreCase)) { context.ReportDiagnostic( CreateDiagnostic( propertyDeclaration.Identifier.GetLocation(), option.Name, otherProperty.Name ) ); } } } public override void Initialize(AnalysisContext context) { base.Initialize(context); context.HandlePropertyDeclaration(Analyze); } }