ApiStationMng.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Threading.Tasks;
  6. using Newtonsoft.Json;
  7. using Sers.Core.Module.Api.ApiDesc;
  8. using Sers.Gover.Base.Model;
  9. using Sers.Gover.Persistence;
  10. using Sers.ServiceCenter.Entity;
  11. using Vit.Core.Module.Log;
  12. using Vit.Core.Util.ConfigurationManager;
  13. using Vit.Extensions;
  14. using Vit.Extensions.Json_Extensions;
  15. namespace Sers.Gover.Base
  16. {
  17. [JsonObject(MemberSerialization.OptIn)]
  18. public class ApiStationMng
  19. {
  20. [JsonProperty]
  21. ConcurrentDictionary<string, ApiStationData> apiStations;
  22. GoverApiCenterService goverManage;
  23. public ApiStationMng()
  24. {
  25. }
  26. public ApiStationMng Init(GoverApiCenterService goverManage)
  27. {
  28. this.goverManage = goverManage;
  29. if (null == apiStations)
  30. {
  31. apiStations = new ConcurrentDictionary<string, ApiStationData>();
  32. }
  33. return this;
  34. }
  35. #region 查询
  36. public IEnumerable<SsApiDesc> ApiDesc_GetAll()
  37. {
  38. var apiServices = (from apiStation in apiStations.Values
  39. from apiService in apiStation.apiServices.Values
  40. select apiService);
  41. var res = apiServices.Select((apiService)=>
  42. {
  43. var apiDesc = apiService.apiDesc.ConvertBySerialize<SsApiDesc>();
  44. apiDesc.ext = new { apiService.counter};
  45. return apiDesc;
  46. });
  47. return res;
  48. }
  49. #endregion
  50. #region ApiStation
  51. [System.Runtime.CompilerServices.MethodImpl(System.Runtime.CompilerServices.MethodImplOptions.AggressiveInlining)]
  52. public List<ApiStationData> ApiStation_GetAll()
  53. {
  54. return apiStations.Values.OrderBy(m=>m.stationName).ToList();
  55. }
  56. public bool ApiStation_Pause(string stationName)
  57. {
  58. lock (this)
  59. {
  60. if (!apiStations.TryGetValue(stationName, out var apiStationData))
  61. {
  62. return false;
  63. }
  64. if (apiStationData.eStatus == EServiceStationStatus.暂停)
  65. {
  66. return true;
  67. }
  68. apiStationData.eStatus=EServiceStationStatus.暂停;
  69. foreach (var apiNode in apiStationData.apiServices.Values.SelectMany((m) => m.apiNodes))
  70. {
  71. apiNode.StopReason_Add(goverManage.apiLoadBalancingMng, "stopByApiStation");
  72. }
  73. }
  74. return true;
  75. }
  76. public bool ApiStation_Start(string stationName)
  77. {
  78. lock (this)
  79. {
  80. if (!apiStations.TryGetValue(stationName, out var apiStationData))
  81. {
  82. return false;
  83. }
  84. if (apiStationData.eStatus == EServiceStationStatus.正常)
  85. {
  86. return true;
  87. }
  88. apiStationData.eStatus = EServiceStationStatus.正常;
  89. foreach (var apiNode in apiStationData.apiServices.Values.SelectMany((m)=>m.apiNodes))
  90. {
  91. apiNode.StopReason_Remove(goverManage.apiLoadBalancingMng, "stopByApiStation");
  92. }
  93. }
  94. return true;
  95. }
  96. ApiStationData ApiStation_Get(string route)
  97. {
  98. var stationName = route.Split('/')[1];
  99. if (apiStations.TryGetValue(stationName, out var apiStationData))
  100. {
  101. return apiStationData;
  102. }
  103. return null;
  104. }
  105. public ApiStationData ApiStation_GetOrAddByName(string stationName)
  106. {
  107. return apiStations.GetOrAdd(stationName, (n) => new ApiStationData() { stationName = n });
  108. }
  109. public ApiStationData ApiStation_GetOrAddByRoute(string route)
  110. {
  111. var stationName = route.Split('/')[1];
  112. return ApiStation_GetOrAddByName(stationName);
  113. }
  114. void ApiStation_Remove(ApiStationData apiStation)
  115. {
  116. apiStations.TryRemove(apiStation.stationName,out _);
  117. }
  118. #endregion
  119. #region ApiService
  120. /// <summary>
  121. /// 移除离线的ApiService
  122. /// </summary>
  123. public void ApiService_RemoveOffline()
  124. {
  125. List<ApiStationData> changedApiStationList;
  126. lock (this)
  127. {
  128. var apiServiceItems = (from apiStation in apiStations.Values
  129. from apiService in apiStation.apiServices.Values
  130. where apiService.apiNodeCount==0
  131. select (apiStation,apiService)).ToList();
  132. foreach (var (apiStation, apiService) in apiServiceItems)
  133. {
  134. apiStation.ApiService_Remove(apiService.apiDesc.ServiceKeyGet());
  135. }
  136. changedApiStationList = apiServiceItems.Select(m => m.apiStation).Distinct().ToList();
  137. foreach (var apiStation in changedApiStationList)
  138. {
  139. if (apiStation.apiServiceCount == 0)
  140. {
  141. ApiStation_Remove(apiStation);
  142. }
  143. }
  144. }
  145. #region 持久化对应ApiStation中的所有ApiDesc(异步执行)
  146. Task.Run(() =>
  147. {
  148. try
  149. {
  150. foreach (var apiStation in changedApiStationList)
  151. {
  152. Persistence_ApiDesc.ApiDesc_SaveApiStationToJsonFile(apiStation);
  153. }
  154. }
  155. catch (Exception ex)
  156. {
  157. Logger.Error(ex);
  158. }
  159. });
  160. #endregion
  161. }
  162. #endregion
  163. #region ApiNode
  164. static bool Config_ApiRegistEvent_Print = (false != Appsettings.json.GetByPath<bool?>("Sers.ServiceCenter.ApiRegistEvent_Print"));
  165. void ApiNode_Add(string route, ApiNode apiNode)
  166. {
  167. try
  168. {
  169. if (Config_ApiRegistEvent_Print)
  170. {
  171. Logger.Info("[ApiCenterService]Add ApiNode", apiNode.apiDesc.ServiceKeyGet());
  172. }
  173. //ApiStation 添加ApiNode
  174. var apiStation = ApiStation_GetOrAddByRoute(route);
  175. apiStation.ApiNode_Add(apiNode);
  176. if (apiStation.IsActive())
  177. {
  178. apiNode.Start(goverManage.apiLoadBalancingMng);
  179. }
  180. }
  181. catch (Exception ex)
  182. {
  183. Logger.Error(ex);
  184. }
  185. }
  186. void ApiNode_Remove(ApiNode apiNode)
  187. {
  188. var route = apiNode?.apiDesc?.route;
  189. if (string.IsNullOrWhiteSpace(route)) return;
  190. if (Config_ApiRegistEvent_Print)
  191. {
  192. Logger.Info("[ApiCenterService]Remove ApiNode", apiNode.apiDesc.ServiceKeyGet());
  193. }
  194. var apiStation = ApiStation_Get(route);
  195. if (apiStation == null) return;
  196. //ApiStation 注销ApiNode
  197. apiStation.ApiNode_Remove(apiNode);
  198. //if (apiStation.apiNodeCount == 0)
  199. //{
  200. // ApiStation_Remove(apiStation);
  201. //}
  202. //ApiLoadBalancingMng 注销ApiNode
  203. apiNode.Stop(goverManage.apiLoadBalancingMng);
  204. }
  205. #endregion
  206. #region ServiceStation
  207. #region ServiceStation_Add
  208. public void ServiceStation_Add(ServiceStation serviceStation)
  209. {
  210. //注册apiNode
  211. lock (this)
  212. {
  213. foreach (var apiNode in serviceStation.apiNodes)
  214. {
  215. var route = apiNode?.apiDesc?.route;
  216. if (string.IsNullOrWhiteSpace(route)) continue;
  217. apiNode.serviceStation = serviceStation;
  218. ApiNode_Add(route, apiNode);
  219. }
  220. }
  221. #region 持久化对应ApiStation中的所有ApiDesc(异步执行)
  222. Task.Run(() =>
  223. {
  224. try
  225. {
  226. foreach (var apiStationName in serviceStation.ApiStationNames_Get())
  227. {
  228. if (apiStations.TryGetValue(apiStationName, out var apiStation))
  229. {
  230. Persistence_ApiDesc.ApiDesc_SaveApiStationToJsonFile(apiStation);
  231. }
  232. }
  233. }
  234. catch (Exception ex)
  235. {
  236. Logger.Error(ex);
  237. }
  238. });
  239. #endregion
  240. }
  241. #endregion
  242. #region ServiceStation_Remove
  243. public void ServiceStation_Remove(ServiceStation serviceStation)
  244. {
  245. lock (this)
  246. {
  247. //注销apiNode
  248. foreach (var apiNode in serviceStation.apiNodes)
  249. {
  250. ApiNode_Remove(apiNode);
  251. }
  252. }
  253. }
  254. #endregion
  255. #region ServiceStation_Pause
  256. public void ServiceStation_Pause(ServiceStation serviceStation)
  257. {
  258. lock (this)
  259. {
  260. foreach (var apiNode in serviceStation.apiNodes)
  261. {
  262. apiNode.StopReason_Add(goverManage.apiLoadBalancingMng, "stopByServiceStation");
  263. }
  264. }
  265. }
  266. #endregion
  267. #region ServiceStation_Start
  268. public void ServiceStation_Start(ServiceStation serviceStation)
  269. {
  270. lock (this)
  271. {
  272. foreach (var apiNode in serviceStation.apiNodes)
  273. {
  274. apiNode.StopReason_Remove(goverManage.apiLoadBalancingMng, "stopByServiceStation");
  275. }
  276. }
  277. }
  278. #endregion
  279. #endregion
  280. }
  281. }