EntityFiledAccessAnalyzer.cs 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  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 (analysisContext.Compilation.AssemblyName==AnalyzeAssembly.UnityCodes)
  32. {
  33. analysisContext.RegisterSemanticModelAction((modelAnalysisContext =>
  34. {
  35. if (AnalyzerHelper.IsSemanticModelNeedAnalyze(modelAnalysisContext.SemanticModel,UnityCodesPath.AllModelHotfix))
  36. {
  37. AnalyzeSemanticModel(modelAnalysisContext);
  38. }
  39. } ));
  40. return;
  41. }
  42. if (AnalyzerHelper.IsAssemblyNeedAnalyze(analysisContext.Compilation.AssemblyName, AnalyzeAssembly.AllModelHotfix))
  43. {
  44. analysisContext.RegisterSemanticModelAction((this.AnalyzeSemanticModel));
  45. }
  46. } ));
  47. }
  48. private void AnalyzeSemanticModel(SemanticModelAnalysisContext analysisContext)
  49. {
  50. foreach (var memberAccessExpressionSyntax in analysisContext.SemanticModel.SyntaxTree.GetRoot().DescendantNodes<MemberAccessExpressionSyntax>())
  51. {
  52. AnalyzeMemberAccessExpression(analysisContext, memberAccessExpressionSyntax);
  53. }
  54. }
  55. private void AnalyzeMemberAccessExpression(SemanticModelAnalysisContext context, MemberAccessExpressionSyntax memberAccessExpressionSyntax)
  56. {
  57. // -----筛选出实体类的字段symbol-----
  58. ISymbol? filedSymbol = context.SemanticModel.GetSymbolInfo(memberAccessExpressionSyntax).Symbol;
  59. if (filedSymbol == null || !(filedSymbol is IFieldSymbol))
  60. {
  61. return;
  62. }
  63. if (filedSymbol.IsStatic)
  64. {
  65. return;
  66. }
  67. if (filedSymbol.ContainingType.BaseType?.ToString() != Definition.EntityType && filedSymbol.ContainingType.BaseType?.ToString() != Definition.LSEntityType)
  68. {
  69. return;
  70. }
  71. // -----筛选出在实体类和实体System外部字段访问-----
  72. // 实体System包括awakeSystem updateSystem等生命周期类和 componentSystem静态方法类
  73. ClassDeclarationSyntax? accessFieldClassDeclaretion = memberAccessExpressionSyntax.GetParentClassDeclaration();
  74. if (accessFieldClassDeclaretion == null)
  75. {
  76. return;
  77. }
  78. INamedTypeSymbol? accessFieldClassSymbol = context.SemanticModel.GetDeclaredSymbol(accessFieldClassDeclaretion);
  79. if (accessFieldClassSymbol == null)
  80. {
  81. return;
  82. }
  83. // 实体基类忽略处理
  84. if (accessFieldClassSymbol.ToString() is Definition.EntityType or Definition.LSEntityType)
  85. {
  86. return;
  87. }
  88. // 允许类内部访问字段
  89. if (accessFieldClassSymbol.ToString()== filedSymbol.ContainingType.ToString() )
  90. {
  91. return;
  92. }
  93. //判断是否在实体类生命周期System中, 这里做了修改,周期System也不允许
  94. //判断是否在实体类的友元类中
  95. if (this.CheckIsEntityFriendOf(accessFieldClassSymbol, filedSymbol.ContainingType))
  96. {
  97. return;
  98. }
  99. var builder = ImmutableDictionary.CreateBuilder<string, string?>();
  100. builder.Add("FriendOfType",filedSymbol.ContainingType.ToString());
  101. Diagnostic diagnostic = Diagnostic.Create(Rule, memberAccessExpressionSyntax.GetLocation(), builder.ToImmutable(),filedSymbol.ContainingType.ToString(),
  102. filedSymbol.Name);
  103. context.ReportDiagnostic(diagnostic);
  104. }
  105. private bool CheckIsEntityFriendOf(INamedTypeSymbol accessFieldTypeSymbol, INamedTypeSymbol entityTypeSymbol)
  106. {
  107. var attributes = accessFieldTypeSymbol.GetAttributes();
  108. foreach (AttributeData? attributeData in attributes)
  109. {
  110. if (attributeData.AttributeClass?.ToString() != Definition.FriendOfAttribute)
  111. {
  112. continue;
  113. }
  114. if (!(attributeData.ConstructorArguments[0].Value is INamedTypeSymbol namedTypeSymbol))
  115. {
  116. continue;
  117. }
  118. if (namedTypeSymbol.ToString() == entityTypeSymbol.ToString())
  119. {
  120. return true;
  121. }
  122. }
  123. return false;
  124. }
  125. }
  126. }