EntityFiledAccessAnalyzer.cs 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. using System.Collections.Immutable;
  2. using System.Linq;
  3. using Microsoft.CodeAnalysis;
  4. using Microsoft.CodeAnalysis.CSharp;
  5. using Microsoft.CodeAnalysis.CSharp.Syntax;
  6. using Microsoft.CodeAnalysis.Diagnostics;
  7. namespace ET.Analyzer
  8. {
  9. [DiagnosticAnalyzer(LanguageNames.CSharp)]
  10. public class EntityFiledAccessAnalyzer: DiagnosticAnalyzer
  11. {
  12. private const string Title = "实体字段访问错误";
  13. private const string MessageFormat = "实体: {0} 字段: {1} 只能在实体类生命周期组件或友元类(含有FriendOfAttribute)中访问";
  14. private const string Description = "请使用实体类属性或方法访问其他实体字段.";
  15. private static readonly DiagnosticDescriptor Rule = new DiagnosticDescriptor(DiagnosticIds.EntityFiledAccessAnalyzerRuleId,
  16. Title,
  17. MessageFormat,
  18. DiagnosticCategories.Hotfix,
  19. DiagnosticSeverity.Error, true, Description);
  20. public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics => ImmutableArray.Create(Rule);
  21. public override void Initialize(AnalysisContext context)
  22. {
  23. if (!AnalyzerGlobalSetting.EnableAnalyzer)
  24. {
  25. return;
  26. }
  27. context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None);
  28. context.EnableConcurrentExecution();
  29. context.RegisterCompilationStartAction((analysisContext =>
  30. {
  31. if (AnalyzerHelper.IsAssemblyNeedAnalyze(analysisContext.Compilation.AssemblyName, AnalyzeAssembly.AllModelHotfix))
  32. {
  33. analysisContext.RegisterSemanticModelAction((this.AnalyzeSemanticModel));
  34. }
  35. } ));
  36. }
  37. private void AnalyzeSemanticModel(SemanticModelAnalysisContext analysisContext)
  38. {
  39. foreach (var memberAccessExpressionSyntax in analysisContext.SemanticModel.SyntaxTree.GetRoot().DescendantNodes<MemberAccessExpressionSyntax>())
  40. {
  41. AnalyzeMemberAccessExpression(analysisContext, memberAccessExpressionSyntax);
  42. }
  43. }
  44. private void AnalyzeMemberAccessExpression(SemanticModelAnalysisContext context, MemberAccessExpressionSyntax memberAccessExpressionSyntax)
  45. {
  46. // -----筛选出实体类的字段symbol-----
  47. ISymbol? filedSymbol = context.SemanticModel.GetSymbolInfo(memberAccessExpressionSyntax).Symbol;
  48. if (filedSymbol == null || !(filedSymbol is IFieldSymbol))
  49. {
  50. return;
  51. }
  52. if (filedSymbol.IsStatic)
  53. {
  54. return;
  55. }
  56. if (filedSymbol.ContainingType.BaseType?.ToString() != Definition.EntityType && filedSymbol.ContainingType.BaseType?.ToString() != Definition.LSEntityType)
  57. {
  58. return;
  59. }
  60. // -----筛选出在实体类和实体System外部字段访问-----
  61. // 实体System包括awakeSystem updateSystem等生命周期类和 componentSystem静态方法类
  62. ClassDeclarationSyntax? accessFieldClassDeclaretion = memberAccessExpressionSyntax.GetParentClassDeclaration();
  63. if (accessFieldClassDeclaretion == null)
  64. {
  65. return;
  66. }
  67. INamedTypeSymbol? accessFieldClassSymbol = context.SemanticModel.GetDeclaredSymbol(accessFieldClassDeclaretion);
  68. if (accessFieldClassSymbol == null)
  69. {
  70. return;
  71. }
  72. // 实体基类忽略处理
  73. if (accessFieldClassSymbol.ToString() is Definition.EntityType or Definition.LSEntityType)
  74. {
  75. return;
  76. }
  77. // 允许类内部访问字段
  78. if (accessFieldClassSymbol.ToString()== filedSymbol.ContainingType.ToString() )
  79. {
  80. return;
  81. }
  82. //判断是否在实体类生命周期System中, 这里做了修改,周期System也不允许
  83. //判断是否在实体类的友元类中
  84. if (this.CheckIsEntityFriendOf(accessFieldClassSymbol, filedSymbol.ContainingType))
  85. {
  86. return;
  87. }
  88. var builder = ImmutableDictionary.CreateBuilder<string, string?>();
  89. builder.Add("FriendOfType",filedSymbol.ContainingType.ToString());
  90. Diagnostic diagnostic = Diagnostic.Create(Rule, memberAccessExpressionSyntax.GetLocation(), builder.ToImmutable(),filedSymbol.ContainingType.ToString(),
  91. filedSymbol.Name);
  92. context.ReportDiagnostic(diagnostic);
  93. }
  94. private bool CheckIsEntityFriendOf(INamedTypeSymbol accessFieldTypeSymbol, INamedTypeSymbol entityTypeSymbol)
  95. {
  96. var attributes = accessFieldTypeSymbol.GetAttributes();
  97. foreach (AttributeData? attributeData in attributes)
  98. {
  99. if (!Definition.FriendAttributes.Contains(attributeData.AttributeClass?.ToString()))
  100. {
  101. continue;
  102. }
  103. if (!(attributeData.ConstructorArguments[0].Value is INamedTypeSymbol namedTypeSymbol))
  104. {
  105. continue;
  106. }
  107. if (namedTypeSymbol.ToString() == entityTypeSymbol.ToString())
  108. {
  109. return true;
  110. }
  111. }
  112. return false;
  113. }
  114. }
  115. }