PerformanceJob.cs 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. using OASystem.API.OAMethodLib;
  2. using OASystem.Domain.Entities.System;
  3. using OASystem.Domain.ViewModels;
  4. using OASystem.Infrastructure.Tools;
  5. using Quartz;
  6. using SqlSugar;
  7. using Newtonsoft.Json;
  8. namespace OASystem.API.OAMethodLib.Quartz.Jobs
  9. {
  10. /// <summary>
  11. /// 绩效生成定时任务
  12. /// 每月1号5点执行
  13. /// </summary>
  14. public class PerformanceJob : IJob
  15. {
  16. private readonly ILogger<PerformanceJob> _logger;
  17. private readonly IHttpClientFactory _httpClientFactory;
  18. public PerformanceJob(ILogger<PerformanceJob> logger, IHttpClientFactory httpClientFactory)
  19. {
  20. _logger = logger;
  21. _httpClientFactory = httpClientFactory;
  22. }
  23. /// <summary>
  24. /// 绩效生成
  25. /// </summary>
  26. /// <param name="context"></param>
  27. /// <returns></returns>
  28. public async Task Execute(IJobExecutionContext context)
  29. {
  30. _logger.LogInformation("调用绩效生成任务 " + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));
  31. try
  32. {
  33. string baseUrl = "http://132.232.92.186:8888/";
  34. if (!baseUrl.EndsWith("/"))
  35. {
  36. baseUrl = baseUrl.TrimEnd('/');
  37. }
  38. // 创建HttpClient
  39. var httpClient = _httpClientFactory.CreateClient();
  40. httpClient.Timeout = TimeSpan.FromMinutes(30); // 设置超时时间
  41. // 计算上个月的时间范围(因为每月1号执行,所以生成上个月的绩效)
  42. var now = DateTime.Now;
  43. var lastMonth = now.AddMonths(-1);
  44. var startDate = new DateTime(lastMonth.Year, lastMonth.Month, 1);
  45. var endDate = new DateTime(lastMonth.Year, lastMonth.Month, DateTime.DaysInMonth(lastMonth.Year, lastMonth.Month), 23, 59, 59);
  46. _logger.LogInformation($"开始生成绩效数据,时间范围:{startDate:yyyy-MM-dd HH:mm:ss} 至 {endDate:yyyy-MM-dd HH:mm:ss}");
  47. // 获取需要生成绩效的用户列表(通过数据库查询)
  48. var sqlSugar = AutofacIocManager.Instance.GetService<SqlSugarClient>();
  49. if (sqlSugar == null)
  50. {
  51. _logger.LogError("无法获取数据库连接,绩效生成任务终止");
  52. return;
  53. }
  54. // 获取需要生成绩效的用户列表(排除已删除的用户,排除配置中指定的用户ID)
  55. var notidsJson = sqlSugar.Queryable<Sys_SetData>().First(x => x.Id == 1463 && x.IsDel == 0)?.Remark;
  56. var notids = new List<int>();
  57. if (!string.IsNullOrWhiteSpace(notidsJson))
  58. {
  59. try
  60. {
  61. notids = JsonConvert.DeserializeObject<List<int>>(notidsJson) ?? new List<int>();
  62. }
  63. catch
  64. {
  65. _logger.LogWarning("解析排除用户ID配置失败");
  66. }
  67. }
  68. var users = await sqlSugar.Queryable<Sys_Users>()
  69. .LeftJoin<Sys_Department>((u, d) => u.DepId == d.Id)
  70. .LeftJoin<Sys_JobPost>((u, d, jp) => u.JobPostId == jp.Id)
  71. .Where((u, d, jp) => u.IsDel == 0 && !notids.Contains(u.Id))
  72. .Select((u, d, jp) => new { u.Id, u.CnName, DepName = d.DepName ?? "", JobName = jp.JobName ?? "" })
  73. .ToListAsync();
  74. if (users == null || users.Count == 0)
  75. {
  76. _logger.LogWarning("未找到需要生成绩效的用户");
  77. return;
  78. }
  79. _logger.LogInformation($"找到 {users.Count} 个用户需要生成绩效数据");
  80. // 系统管理员ID(用于createUserId参数)
  81. int createUserId = 4; // 管理员ID
  82. // 为每个用户调用绩效分析API
  83. int successCount = 0;
  84. int failCount = 0;
  85. //添加不同岗位的绩效分析API
  86. foreach (var user in users)
  87. {
  88. try
  89. {
  90. // 构建API URL
  91. string apiUrl = $"{baseUrl}/api/PersonnelModule/AiPerformanceAnalysis_AllDepartmentAsync";
  92. // 构建查询参数
  93. var queryParams = new Dictionary<string, string>
  94. {
  95. { "userId", user.Id.ToString() },
  96. { "start", startDate.ToString("yyyy-MM-dd HH:mm:ss") },
  97. { "end", endDate.ToString("yyyy-MM-dd HH:mm:ss") },
  98. { "createUserId", createUserId.ToString() }
  99. };
  100. // 构建完整URL
  101. var queryString = string.Join("&", queryParams.Select(kvp => $"{kvp.Key}={Uri.EscapeDataString(kvp.Value)}"));
  102. string fullUrl = $"{apiUrl}?{queryString}";
  103. //url编码
  104. fullUrl = Uri.EscapeDataString(fullUrl);
  105. _logger.LogInformation($"正在为用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 生成绩效数据...");
  106. _logger.LogInformation($"API URL: {fullUrl}");
  107. // 发送HTTP GET请求
  108. var response = await httpClient.GetAsync(fullUrl);
  109. var responseContent = await response.Content.ReadAsStringAsync();
  110. if (response.IsSuccessStatusCode)
  111. {
  112. // 尝试解析响应结果
  113. try
  114. {
  115. var result = JsonConvert.DeserializeObject<JsonView>(responseContent);
  116. if (result != null && result.Code == 200)
  117. {
  118. _logger.LogInformation($"用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 绩效数据生成成功");
  119. successCount++;
  120. }
  121. else
  122. {
  123. _logger.LogWarning($"用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 绩效数据生成失败:{result?.Msg ?? "未知错误"}");
  124. failCount++;
  125. }
  126. }
  127. catch (Exception ex)
  128. {
  129. _logger.LogError(ex, $"解析用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 的API响应失败");
  130. failCount++;
  131. }
  132. }
  133. else
  134. {
  135. _logger.LogError($"用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 绩效数据生成API调用失败,状态码: {response.StatusCode}, 响应内容: {responseContent}");
  136. failCount++;
  137. }
  138. // 添加延迟,避免请求过于频繁
  139. await Task.Delay(3000); // 延迟3秒
  140. }
  141. catch (Exception ex)
  142. {
  143. _logger.LogError(ex, $"为用户 {user.CnName}(ID:{user.Id}, 部门:{user.DepName}) 生成绩效数据时发生异常");
  144. failCount++;
  145. }
  146. }
  147. _logger.LogInformation($"绩效生成任务完成!成功:{successCount},失败:{failCount},总计:{users.Count}");
  148. }
  149. catch (Exception ex)
  150. {
  151. _logger.LogError(ex, $"调用绩效生成任务失败 ErrorMsg:{ex.Message} " + DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"));
  152. }
  153. await Task.CompletedTask;
  154. }
  155. }
  156. }