SysCommonService.cs 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. // Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
  2. //
  3. // 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
  4. //
  5. // 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
  6. using Microsoft.AspNetCore.Mvc.ApiExplorer;
  7. using Org.BouncyCastle.Crypto.Parameters;
  8. using Org.BouncyCastle.Utilities.Encoders;
  9. using Swashbuckle.AspNetCore.SwaggerGen;
  10. namespace Admin.NET.Core.Service;
  11. /// <summary>
  12. /// 系统通用服务 🧩
  13. /// </summary>
  14. [ApiDescriptionSettings(Order = 101)]
  15. [AllowAnonymous]
  16. public class SysCommonService : IDynamicApiController, ITransient
  17. {
  18. private readonly IApiDescriptionGroupCollectionProvider _apiProvider;
  19. private readonly SqlSugarRepository<SysUser> _sysUserRep;
  20. private readonly CDConfigOptions _cdConfigOptions;
  21. private readonly UserManager _userManager;
  22. private readonly HttpClient _httpClient;
  23. public SysCommonService(IApiDescriptionGroupCollectionProvider apiProvider,
  24. SqlSugarRepository<SysUser> sysUserRep,
  25. IOptions<CDConfigOptions> giteeOptions,
  26. IHttpClientFactory httpClientFactory,
  27. UserManager userManager)
  28. {
  29. _sysUserRep = sysUserRep;
  30. _apiProvider = apiProvider;
  31. _userManager = userManager;
  32. _cdConfigOptions = giteeOptions.Value;
  33. _httpClient = httpClientFactory.CreateClient();
  34. }
  35. /// <summary>
  36. /// 获取国密公钥私钥对 🏆
  37. /// </summary>
  38. /// <returns></returns>
  39. [DisplayName("获取国密公钥私钥对")]
  40. public SmKeyPairOutput GetSmKeyPair()
  41. {
  42. var kp = GM.GenerateKeyPair();
  43. var privateKey = Hex.ToHexString(((ECPrivateKeyParameters)kp.Private).D.ToByteArray()).ToUpper();
  44. var publicKey = Hex.ToHexString(((ECPublicKeyParameters)kp.Public).Q.GetEncoded()).ToUpper();
  45. return new SmKeyPairOutput
  46. {
  47. PrivateKey = privateKey,
  48. PublicKey = publicKey,
  49. };
  50. }
  51. /// <summary>
  52. /// 获取所有接口/动态API 🔖
  53. /// </summary>
  54. /// <returns></returns>
  55. [DisplayName("获取所有接口/动态API")]
  56. public List<ApiOutput> GetApiList()
  57. {
  58. var apiList = new List<ApiOutput>();
  59. foreach (var item in _apiProvider.ApiDescriptionGroups.Items)
  60. {
  61. foreach (var apiDescription in item.Items)
  62. {
  63. var displayName = apiDescription.TryGetMethodInfo(out MethodInfo apiMethodInfo) ? apiMethodInfo.GetCustomAttribute<DisplayNameAttribute>(true)?.DisplayName : "";
  64. apiList.Add(new ApiOutput
  65. {
  66. GroupName = item.GroupName,
  67. DisplayName = displayName,
  68. RouteName = apiDescription.RelativePath
  69. });
  70. }
  71. }
  72. return apiList;
  73. }
  74. /// <summary>
  75. /// 下载标记错误的临时Excel(全局)
  76. /// </summary>
  77. /// <returns></returns>
  78. [DisplayName("下载标记错误的临时Excel(全局)")]
  79. public async Task<IActionResult> DownloadErrorExcelTemp([FromQuery] string fileName = null)
  80. {
  81. var userId = App.User?.FindFirst(ClaimConst.UserId)?.Value;
  82. var resultStream = App.GetRequiredService<SysCacheService>().Get<MemoryStream>(CacheConst.KeyExcelTemp + userId);
  83. if (resultStream == null) throw Oops.Oh("错误标记文件已过期。");
  84. return await Task.FromResult(new FileStreamResult(resultStream, "application/octet-stream")
  85. {
  86. FileDownloadName = $"{(string.IsNullOrEmpty(fileName) ? "错误标记_" + DateTime.Now.ToString("yyyyMMddhhmmss") : fileName)}.xlsx"
  87. });
  88. }
  89. /// <summary>
  90. /// 加密字符串 🔖
  91. /// </summary>
  92. /// <returns></returns>
  93. [SuppressMonitor]
  94. [DisplayName("加密字符串")]
  95. public dynamic EncryptPlainText([Required] string plainText)
  96. {
  97. return CryptogramUtil.Encrypt(plainText);
  98. }
  99. /// <summary>
  100. /// 接口压测 🔖
  101. /// </summary>
  102. /// <returns></returns>
  103. [DisplayName("接口压测")]
  104. public async Task<StressTestOutput> StressTest(StressTestInput input)
  105. {
  106. // 限制仅超管用户才能使用此功能
  107. if (!_userManager.SuperAdmin) throw Oops.Oh(ErrorCodeEnum.SA001);
  108. var stopwatch = new Stopwatch();
  109. var responseTimes = new List<double>(); //响应时间集合
  110. input.RequestMethod = input.RequestMethod.ToUpper();
  111. long totalRequests = 0, successfulRequests = 0, failedRequests = 0;
  112. stopwatch.Start();
  113. var semaphore = new SemaphoreSlim(input.MaxDegreeOfParallelism!.Value > 0 ? input.MaxDegreeOfParallelism.Value : Environment.ProcessorCount);
  114. #region 参数构建
  115. // 构建基础URI(不包括路径和查询参数)
  116. var baseUriBuilder = new UriBuilder(input.RequestUri);
  117. var queryString = HttpUtility.ParseQueryString(baseUriBuilder.Query);
  118. // 替换路径参数到baseUriBuilder.Path
  119. foreach (var param in input.PathParameters)
  120. {
  121. baseUriBuilder.Path = baseUriBuilder.Path.Replace($"{{{param.Key}}}", param.Value, StringComparison.OrdinalIgnoreCase);
  122. }
  123. // 构建Query参数
  124. foreach (var param in input.QueryParameters)
  125. {
  126. queryString[param.Key] = param.Value;
  127. }
  128. baseUriBuilder.Query = queryString.ToString() ?? string.Empty;
  129. var fullUri = baseUriBuilder.Uri;
  130. // 创建一次性的HttpRequestMessage模板
  131. HttpRequestMessage requestTemplate = CreateRequestMessage(input, fullUri);
  132. #endregion 参数构建
  133. var tasks = Enumerable.Range(0, input.NumberOfRounds!.Value * input.NumberOfRequests!.Value).Select(async _ =>
  134. {
  135. await semaphore.WaitAsync();
  136. try
  137. {
  138. var requestStopwatch = new Stopwatch();
  139. requestStopwatch.Start();
  140. using (var request = requestTemplate.DeepCopy())
  141. {
  142. if (!string.Equals(input.RequestMethod, "GET", StringComparison.OrdinalIgnoreCase) && input.RequestParameters.Any())
  143. {
  144. var content = new FormUrlEncodedContent(input.RequestParameters);
  145. request.Content = content;
  146. }
  147. using (var response = await _httpClient.SendAsync(request))
  148. {
  149. response.EnsureSuccessStatusCode(); // 抛出错误状态码异常
  150. requestStopwatch.Stop();
  151. responseTimes.Add(requestStopwatch.Elapsed.TotalMilliseconds);
  152. Interlocked.Increment(ref successfulRequests);
  153. }
  154. }
  155. }
  156. catch
  157. {
  158. Interlocked.Increment(ref failedRequests);
  159. }
  160. finally
  161. {
  162. Interlocked.Increment(ref totalRequests);
  163. semaphore.Release();
  164. }
  165. });
  166. await Task.WhenAll(tasks);
  167. stopwatch.Stop();
  168. var totalTimeInSeconds = stopwatch.Elapsed.TotalSeconds;
  169. var qps = totalTimeInSeconds > 0 ? totalRequests / totalTimeInSeconds : 0;
  170. var orderResponseTimes = responseTimes.OrderBy(t => t).ToList();
  171. var averageResponseTime = responseTimes.Any() ? responseTimes.Average() : 0;
  172. var minResponseTime = responseTimes.Any() ? responseTimes.Min() : 0;
  173. var maxResponseTime = responseTimes.Any() ? responseTimes.Max() : 0;
  174. return new StressTestOutput
  175. {
  176. TotalRequests = totalRequests,
  177. TotalTimeInSeconds = totalTimeInSeconds,
  178. SuccessfulRequests = successfulRequests,
  179. FailedRequests = failedRequests,
  180. QueriesPerSecond = qps,
  181. MinResponseTime = minResponseTime,
  182. MaxResponseTime = maxResponseTime,
  183. AverageResponseTime = averageResponseTime,
  184. Percentile10ResponseTime = CalculatePercentile(orderResponseTimes, 0.1),
  185. Percentile25ResponseTime = CalculatePercentile(orderResponseTimes, 0.25),
  186. Percentile50ResponseTime = CalculatePercentile(orderResponseTimes, 0.5),
  187. Percentile75ResponseTime = CalculatePercentile(orderResponseTimes, 0.75),
  188. Percentile90ResponseTime = CalculatePercentile(orderResponseTimes, 0.9),
  189. Percentile99ResponseTime = CalculatePercentile(orderResponseTimes, 0.99),
  190. Percentile999ResponseTime = CalculatePercentile(orderResponseTimes, 0.999)
  191. };
  192. }
  193. /// <summary>
  194. /// 创建请求消息
  195. /// </summary>
  196. /// <param name="input">输入参数</param>
  197. /// <param name="fullUri">url</param>
  198. /// <returns></returns>
  199. private HttpRequestMessage CreateRequestMessage(StressTestInput input, Uri fullUri)
  200. {
  201. HttpRequestMessage request = input.RequestMethod switch
  202. {
  203. "GET" => new HttpRequestMessage(HttpMethod.Get, fullUri),
  204. "PUT" => new HttpRequestMessage(HttpMethod.Put, fullUri),
  205. "POST" => new HttpRequestMessage(HttpMethod.Post, fullUri),
  206. "DELETE" => new HttpRequestMessage(HttpMethod.Delete, fullUri),
  207. _ => throw Oops.Bah("请求方式异常")
  208. };
  209. // 设置请求头
  210. foreach (var header in input.Headers)
  211. {
  212. request.Headers.Add(header.Key, header.Value);
  213. }
  214. return request;
  215. }
  216. /// <summary>
  217. /// 计算百分位请求耗时
  218. /// </summary>
  219. /// <param name="times">请求耗时列表</param>
  220. /// <param name="percentile">百分位</param>
  221. /// <returns></returns>
  222. private double CalculatePercentile(List<double> times, double percentile)
  223. {
  224. if (!times.Any()) return 0;
  225. var index = (int)Math.Ceiling(percentile * times.Count) - 1;
  226. return times[index < times.Count ? index : times.Count - 1];
  227. }
  228. }