SysCommonService.cs 10 KB

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