EntityMemberDeclarationAnalyzer.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.Immutable;
  4. using System.Linq;
  5. using Microsoft.CodeAnalysis;
  6. using Microsoft.CodeAnalysis.CSharp;
  7. using Microsoft.CodeAnalysis.CSharp.Syntax;
  8. using Microsoft.CodeAnalysis.Diagnostics;
  9. namespace ET.Analyzer
  10. {
  11. [DiagnosticAnalyzer(LanguageNames.CSharp)]
  12. public class EntityMemberDeclarationAnalyzer: DiagnosticAnalyzer
  13. {
  14. public override ImmutableArray<DiagnosticDescriptor> SupportedDiagnostics =>ImmutableArray.Create(EntityDelegateDeclarationAnalyzerRule.Rule,
  15. EntityFieldDeclarationInEntityAnalyzerRule.Rule, LSEntityFloatMemberAnalyzer.Rule, EntityComponentChildAnalyzerRule.Rule);
  16. public override void Initialize(AnalysisContext context)
  17. {
  18. if (!AnalyzerGlobalSetting.EnableAnalyzer)
  19. {
  20. return;
  21. }
  22. context.ConfigureGeneratedCodeAnalysis(GeneratedCodeAnalysisFlags.None);
  23. context.EnableConcurrentExecution();
  24. context.RegisterCompilationStartAction((analysisContext =>
  25. {
  26. if (AnalyzerHelper.IsAssemblyNeedAnalyze(analysisContext.Compilation.AssemblyName,AnalyzeAssembly.AllModel))
  27. {
  28. analysisContext.RegisterSemanticModelAction((this.AnalyzeSemanticModel));
  29. }
  30. } ));
  31. }
  32. private void AnalyzeSemanticModel(SemanticModelAnalysisContext analysisContext)
  33. {
  34. foreach (var classDeclarationSyntax in analysisContext.SemanticModel.SyntaxTree.GetRoot().DescendantNodes<ClassDeclarationSyntax>())
  35. {
  36. var classTypeSymbol = analysisContext.SemanticModel.GetDeclaredSymbol(classDeclarationSyntax);
  37. if (classTypeSymbol!=null)
  38. {
  39. Analyzer(analysisContext, classTypeSymbol);
  40. }
  41. }
  42. }
  43. private void Analyzer(SemanticModelAnalysisContext context, INamedTypeSymbol namedTypeSymbol)
  44. {
  45. var baseType = namedTypeSymbol.BaseType?.ToString();
  46. // 筛选出实体类
  47. if (baseType== Definition.EntityType)
  48. {
  49. AnalyzeDelegateMember(context, namedTypeSymbol);
  50. AnalyzeEntityMember(context, namedTypeSymbol);
  51. AnalyzeComponentChildAttr(context, namedTypeSymbol);
  52. }else if (baseType == Definition.LSEntityType)
  53. {
  54. AnalyzeDelegateMember(context, namedTypeSymbol);
  55. AnalyzeEntityMember(context, namedTypeSymbol);
  56. AnalyzeFloatMemberInLSEntity(context,namedTypeSymbol);
  57. AnalyzeComponentChildAttr(context, namedTypeSymbol);
  58. }
  59. }
  60. /// <summary>
  61. /// 检查委托成员
  62. /// </summary>
  63. private void AnalyzeDelegateMember(SemanticModelAnalysisContext context,INamedTypeSymbol namedTypeSymbol)
  64. {
  65. foreach (var member in namedTypeSymbol.GetMembers())
  66. {
  67. if (member is IFieldSymbol fieldSymbol && fieldSymbol.Type.BaseType?.ToString()==typeof(MulticastDelegate).FullName)
  68. {
  69. ReportDiagnostic(fieldSymbol,fieldSymbol.Name);
  70. continue;
  71. }
  72. if (member is IPropertySymbol propertySymbol && propertySymbol.Type.BaseType?.ToString()==typeof(MulticastDelegate).FullName)
  73. {
  74. ReportDiagnostic(propertySymbol,propertySymbol.Name);
  75. continue;
  76. }
  77. }
  78. void ReportDiagnostic(ISymbol symbol,string delegateName)
  79. {
  80. foreach (var syntaxReference in symbol.DeclaringSyntaxReferences)
  81. {
  82. var syntax = syntaxReference.GetSyntax();
  83. Diagnostic diagnostic = Diagnostic.Create(EntityDelegateDeclarationAnalyzerRule.Rule, syntax.GetLocation(),namedTypeSymbol.Name,delegateName);
  84. context.ReportDiagnostic(diagnostic);
  85. }
  86. }
  87. }
  88. /// <summary>
  89. /// 检查实体成员
  90. /// </summary>
  91. private void AnalyzeEntityMember(SemanticModelAnalysisContext context, INamedTypeSymbol namedTypeSymbol)
  92. {
  93. foreach (var member in namedTypeSymbol.GetMembers())
  94. {
  95. if (member is not IFieldSymbol fieldSymbol)
  96. {
  97. continue;
  98. }
  99. // 忽略静态字段 允许单例实体类
  100. if (fieldSymbol.IsStatic)
  101. {
  102. continue;
  103. }
  104. // 字段类型是否是实体类型数组
  105. if (fieldSymbol.Type is IArrayTypeSymbol arrayTypeSymbol)
  106. {
  107. if (arrayTypeSymbol.ElementType.IsETEntity())
  108. {
  109. var syntaxReference = fieldSymbol.DeclaringSyntaxReferences.FirstOrDefault();
  110. if (syntaxReference==null)
  111. {
  112. continue;
  113. }
  114. Diagnostic diagnostic = Diagnostic.Create(EntityFieldDeclarationInEntityAnalyzerRule.Rule, syntaxReference.GetSyntax().GetLocation(),namedTypeSymbol.Name,fieldSymbol.Name);
  115. context.ReportDiagnostic(diagnostic);
  116. }
  117. continue;
  118. }
  119. if (fieldSymbol.Type is not INamedTypeSymbol namedTypeSymbol2)
  120. {
  121. continue;
  122. }
  123. // 字段类型是否是实体类
  124. if (namedTypeSymbol2.IsETEntity())
  125. {
  126. var syntaxReference = fieldSymbol.DeclaringSyntaxReferences.FirstOrDefault();
  127. if (syntaxReference==null)
  128. {
  129. continue;
  130. }
  131. Diagnostic diagnostic = Diagnostic.Create(EntityFieldDeclarationInEntityAnalyzerRule.Rule, syntaxReference.GetSyntax().GetLocation(),namedTypeSymbol.Name,fieldSymbol.Name);
  132. context.ReportDiagnostic(diagnostic);
  133. continue;
  134. }
  135. // 字段类型是否是含实体类参数的泛型类
  136. if (namedTypeSymbol2.IsGenericType&&GenericTypeHasEntityTypeArgs(namedTypeSymbol2))
  137. {
  138. var syntaxReference = fieldSymbol.DeclaringSyntaxReferences.FirstOrDefault();
  139. if (syntaxReference==null)
  140. {
  141. continue;
  142. }
  143. Diagnostic diagnostic = Diagnostic.Create(EntityFieldDeclarationInEntityAnalyzerRule.Rule, syntaxReference.GetSyntax().GetLocation(),namedTypeSymbol.Name,fieldSymbol.Name);
  144. context.ReportDiagnostic(diagnostic);
  145. }
  146. }
  147. }
  148. /// <summary>
  149. /// 检查LSEntity中 是否有浮点数字段
  150. /// </summary>
  151. private void AnalyzeFloatMemberInLSEntity(SemanticModelAnalysisContext context, INamedTypeSymbol namedTypeSymbol)
  152. {
  153. foreach (var member in namedTypeSymbol.GetMembers())
  154. {
  155. INamedTypeSymbol? memberType = null;
  156. if (member is IFieldSymbol fieldSymbol)
  157. {
  158. memberType = fieldSymbol.Type as INamedTypeSymbol;
  159. }
  160. if (member is IPropertySymbol propertySymbol)
  161. {
  162. memberType = propertySymbol.Type as INamedTypeSymbol;
  163. }
  164. if (memberType==null)
  165. {
  166. continue;
  167. }
  168. if (memberType.SpecialType is SpecialType.System_Single or SpecialType.System_Double)
  169. {
  170. var syntaxReference = member.DeclaringSyntaxReferences.FirstOrDefault();
  171. if (syntaxReference==null)
  172. {
  173. continue;
  174. }
  175. Diagnostic diagnostic = Diagnostic.Create(LSEntityFloatMemberAnalyzer.Rule, syntaxReference.GetSyntax().GetLocation(),namedTypeSymbol.Name,member.Name);
  176. context.ReportDiagnostic(diagnostic);
  177. continue;
  178. }
  179. if (memberType.IsGenericType && GenericTypeHasFloatTypeArgs(memberType))
  180. {
  181. var syntaxReference = member.DeclaringSyntaxReferences.FirstOrDefault();
  182. if (syntaxReference==null)
  183. {
  184. continue;
  185. }
  186. Diagnostic diagnostic = Diagnostic.Create(LSEntityFloatMemberAnalyzer.Rule, syntaxReference.GetSyntax().GetLocation(),namedTypeSymbol.Name,member.Name);
  187. context.ReportDiagnostic(diagnostic);
  188. continue;
  189. }
  190. }
  191. }
  192. /// <summary>
  193. /// 泛型类 是否含有浮点数类型参数
  194. /// 对于嵌套泛型参数 递归判断
  195. /// </summary>
  196. private bool GenericTypeHasFloatTypeArgs(INamedTypeSymbol namedTypeSymbol)
  197. {
  198. var typeArgs = namedTypeSymbol.TypeArguments;
  199. foreach (var typeSymbol in typeArgs)
  200. {
  201. if (typeSymbol is not INamedTypeSymbol namedTypeSymbol2)
  202. {
  203. break;
  204. }
  205. if (namedTypeSymbol2.IsGenericType)
  206. {
  207. if (GenericTypeHasFloatTypeArgs(namedTypeSymbol2))
  208. {
  209. return true;
  210. }
  211. }
  212. else
  213. {
  214. if (namedTypeSymbol2.SpecialType is SpecialType.System_Single or SpecialType.System_Double)
  215. {
  216. return true;
  217. }
  218. }
  219. }
  220. return false;
  221. }
  222. /// <summary>
  223. /// 实体类是否同时标记为component child
  224. /// </summary>
  225. private void AnalyzeComponentChildAttr(SemanticModelAnalysisContext context, INamedTypeSymbol namedTypeSymbol)
  226. {
  227. bool hasComponentOf = namedTypeSymbol.HasAttribute(Definition.ComponentOfAttribute);
  228. bool hasChildOf = namedTypeSymbol.HasAttribute(Definition.ChildOfAttribute);
  229. if (hasComponentOf && hasChildOf)
  230. {
  231. var syntax = namedTypeSymbol.DeclaringSyntaxReferences.First().GetSyntax() as ClassDeclarationSyntax;
  232. Diagnostic diagnostic = Diagnostic.Create(EntityComponentChildAnalyzerRule.Rule, syntax?.Identifier.GetLocation(),namedTypeSymbol.Name);
  233. context.ReportDiagnostic(diagnostic);
  234. }
  235. }
  236. /// <summary>
  237. /// 泛型类 是否含有的实体类型参数
  238. /// 对于嵌套泛型参数 递归判断
  239. /// </summary>
  240. private bool GenericTypeHasEntityTypeArgs(INamedTypeSymbol namedTypeSymbol)
  241. {
  242. if (namedTypeSymbol.IsEntityRefOrEntityWeakRef())
  243. {
  244. return false;
  245. }
  246. var typeArgs = namedTypeSymbol.TypeArguments;
  247. foreach (var typeSymbol in typeArgs)
  248. {
  249. if (typeSymbol is not INamedTypeSymbol namedTypeSymbol2)
  250. {
  251. break;
  252. }
  253. if (namedTypeSymbol2.IsGenericType)
  254. {
  255. if (GenericTypeHasEntityTypeArgs(namedTypeSymbol2))
  256. {
  257. return true;
  258. }
  259. }
  260. else
  261. {
  262. if (namedTypeSymbol2.IsETEntity())
  263. {
  264. return true;
  265. }
  266. }
  267. }
  268. return false;
  269. }
  270. }
  271. }