AssemblyInjectStartStopWorker.cs 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. using PCC2.AssemblyInject.Interfaces;
  2. namespace PCC2.AssemblyInject.Services;
  3. internal class AssemblyInjectStartStopWorker<T> : IHostedService
  4. {
  5. private readonly ILogger<AssemblyInjectStartStopWorker<T>> _logger;
  6. private readonly IReadOnlyCollection<IAssemblyInjectSyncInitStarStop> _syncServices;
  7. private readonly IReadOnlyCollection<IAssemblyInjectASyncInitStarStop> _asyncServices;
  8. public AssemblyInjectStartStopWorker(
  9. IServiceProvider serviceProvider,
  10. ILogger<AssemblyInjectStartStopWorker<T>> logger)
  11. {
  12. _logger = logger;
  13. _logger.LogInformation("Activating Singleton Component...");
  14. //单例组件
  15. var serviceTypes = new HashSet<Type>();
  16. foreach (var svcType in typeof(T).Assembly.GetTypes()
  17. .Where(typeof(IAssemblyInjectSingleton).IsAssignableFrom)
  18. .Where(p => p.IsInterface == false && p.IsAbstract == false)
  19. .OrderBy(p => p.Name))
  20. {
  21. var gsi = svcType.GetInterfaces().Where(p => p.IsGenericType && p.GetGenericTypeDefinition() == typeof(IAssemblyInjectSingleton<>)).ToArray();
  22. foreach (var type in gsi)
  23. {
  24. var addAs = type.GetGenericArguments().First();
  25. serviceTypes.Add(addAs);
  26. }
  27. }
  28. foreach (var serviceType in serviceTypes)
  29. {
  30. var services = serviceProvider.GetServices(serviceType).ToArray();
  31. for (var i = 0; i < services.Length; i++)
  32. {
  33. var service = services[i];
  34. if (service != null) _logger.LogInformation($"Singleton Component Active {service.GetType().FullName}");
  35. else logger.LogWarning($"Encounter null in Active {serviceType.FullName}#{i}");
  36. }
  37. }
  38. foreach (var type in typeof(T).Assembly.GetTypes()
  39. .Where(type => typeof(IAssemblyInjectSingleton).IsAssignableFrom(type) && !typeof(IAssemblyInjectSingleton<>).IsAssignableFrom(type))
  40. .Where(p => p.IsInterface == false && p.IsAbstract == false)
  41. .OrderBy(p => p.Name))
  42. {
  43. var service = serviceProvider.GetService(type);
  44. if (service != null)
  45. {
  46. _logger.LogInformation($"Singleton Component Active {service.GetType().FullName}");
  47. }
  48. }
  49. _syncServices = typeof(T).Assembly.GetTypes()
  50. .Where(typeof(IAssemblyInjectSyncInitStarStop).IsAssignableFrom)
  51. .Where(p => p.IsInterface == false && p.IsAbstract == false)
  52. .OrderBy(p => p.Name)
  53. .SelectMany(p =>
  54. {
  55. var gsi = p.GetInterfaces().Where(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IAssemblyInjectSyncInitStarStop<>)).ToArray();
  56. return gsi.Length == 0 ? [p] : gsi.Select(gi => gi.GetGenericArguments().First());
  57. })
  58. .Distinct()
  59. .Select(type => (type, serviceProvider.GetService(type)))
  60. .Where(p =>
  61. {
  62. if (p.Item2 == null)
  63. {
  64. logger.LogWarning($"Encounter null in IAssemblyInjectSyncInitStarStop: {p.type.FullName}");
  65. return false;
  66. }
  67. return true;
  68. }).Select(p => p.Item2)
  69. .Cast<IAssemblyInjectSyncInitStarStop>()
  70. .ToArray();
  71. _asyncServices = typeof(T).Assembly.GetTypes()
  72. .Where(typeof(IAssemblyInjectASyncInitStarStop).IsAssignableFrom)
  73. .Where(p => p.IsInterface == false && p.IsAbstract == false)
  74. .OrderBy(p => p.Name)
  75. .SelectMany(p =>
  76. {
  77. var gsi = p.GetInterfaces().Where(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IAssemblyInjectASyncInitStarStop<>)).ToArray();
  78. return gsi.Length == 0 ? [p] : gsi.Select(gi => gi.GetGenericArguments().First());
  79. })
  80. .Distinct()
  81. .Select(type => (type, serviceProvider.GetService(type)))
  82. .Where(p =>
  83. {
  84. if (p.Item2 == null)
  85. {
  86. logger.LogWarning($"Encounter null in IAssemblyInjectASyncInitStarStop: {p.type.FullName}");
  87. return false;
  88. }
  89. return true;
  90. }).Select(p => p.Item2)
  91. .Cast<IAssemblyInjectASyncInitStarStop>()
  92. .ToArray();
  93. }
  94. public async Task StartAsync(CancellationToken cancellationToken)
  95. {
  96. _logger.LogInformation($"Initialization...");
  97. if (_syncServices.Count == 0) _logger.LogInformation($"No component found for sync init.");
  98. else
  99. {
  100. foreach (var service in _syncServices)
  101. {
  102. _logger.LogInformation($"Initializing sync {service.GetType()}");
  103. try
  104. {
  105. service.Init();
  106. }
  107. catch (Exception e)
  108. {
  109. _logger.LogError(e, nameof(service.Init));
  110. }
  111. }
  112. _logger.LogInformation($"All components sync initialized.");
  113. }
  114. if (_asyncServices.Count == 0) _logger.LogInformation($"No component found for async init.");
  115. else
  116. {
  117. foreach (var service in _asyncServices)
  118. {
  119. _logger.LogInformation($"Initializing async {service.GetType()}");
  120. try
  121. {
  122. await service.InitAsync();
  123. }
  124. catch (Exception e)
  125. {
  126. _logger.LogError(e, nameof(service.InitAsync));
  127. }
  128. }
  129. _logger.LogInformation($"All components initialized async.");
  130. }
  131. _logger.LogInformation($"Starting...");
  132. if (_syncServices.Count == 0) _logger.LogInformation($"No component found for sync start.");
  133. else
  134. {
  135. foreach (var service in _syncServices)
  136. {
  137. _logger.LogInformation($"Starting sync {service.GetType()}");
  138. try
  139. {
  140. service.Start();
  141. }
  142. catch (Exception ex)
  143. {
  144. _logger.LogError(ex, nameof(service.Start));
  145. }
  146. }
  147. _logger.LogInformation($"All components started sync.");
  148. }
  149. if (_asyncServices.Count == 0) _logger.LogInformation($"No component found for async start.");
  150. else
  151. {
  152. foreach (var service in _asyncServices)
  153. {
  154. _logger.LogInformation($"Starting async {service.GetType()}");
  155. try
  156. {
  157. await service.StartAsync();
  158. }
  159. catch (Exception ex)
  160. {
  161. _logger.LogError(ex, nameof(service.StartAsync));
  162. }
  163. }
  164. _logger.LogInformation($"All components started async.");
  165. }
  166. }
  167. public async Task StopAsync(CancellationToken cancellationToken)
  168. {
  169. _logger.LogInformation($"Stopping...");
  170. if (_syncServices.Count == 0) _logger.LogInformation($"No component stopped for sync.");
  171. else
  172. {
  173. foreach (var service in _syncServices)
  174. {
  175. _logger.LogInformation($"Stopping sync {service.GetType()}");
  176. try
  177. {
  178. service.Stop();
  179. }
  180. catch (Exception ex)
  181. {
  182. _logger.LogError(ex, nameof(service.Stop));
  183. }
  184. }
  185. _logger.LogInformation($"All components stopped sync.");
  186. }
  187. if (_asyncServices.Count == 0) _logger.LogInformation($"No component stopped for async.");
  188. else
  189. {
  190. foreach (var service in _asyncServices)
  191. {
  192. _logger.LogInformation($"Stopping async {service.GetType()}");
  193. try
  194. {
  195. await service.StopAsync();
  196. }
  197. catch (Exception ex)
  198. {
  199. _logger.LogError(ex, nameof(service.StopAsync));
  200. }
  201. }
  202. _logger.LogInformation($"All components stopped async.");
  203. }
  204. }
  205. }