S8ExceptionService.cs 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524
  1. using Admin.NET.Plugin.AiDOP.Dto.S8;
  2. using Admin.NET.Plugin.AiDOP.Entity.S0.Manufacturing;
  3. using Admin.NET.Plugin.AiDOP.Entity.S0.Warehouse;
  4. using Admin.NET.Plugin.AiDOP.Entity.S8;
  5. using Admin.NET.Plugin.AiDOP.Infrastructure;
  6. using Admin.NET.Plugin.AiDOP.Infrastructure.S8;
  7. using Microsoft.Extensions.Options;
  8. namespace Admin.NET.Plugin.AiDOP.Service.S8;
  9. public class S8ExceptionService : ITransient
  10. {
  11. private readonly SqlSugarRepository<AdoS8Exception> _rep;
  12. private readonly SqlSugarRepository<AdoS0DepartmentMaster> _deptRep;
  13. private readonly SqlSugarRepository<AdoS0EmployeeMaster> _empRep;
  14. private readonly SqlSugarRepository<SysUser> _sysUserRep;
  15. private readonly S8ImpactMetricsService _impactMetricsService;
  16. // S8-EXCEPTION-DEPT-NAME-HYDRATION-MASTERDATA-FACTORY-FIX-1:部门名水合按 DepartmentMaster 主数据 factory 解析。
  17. private readonly S8MasterDataOptions _masterDataOptions;
  18. // S8-DEMO-IMPACT-SORT-NOTICE-1:影响排序的候选集合上限。
  19. // 超过此阈值时降级为 CreatedAt DESC 走 DB 分页,避免全量取 → 内存排序在生产规模下退化。
  20. private const int ImpactSortCandidateCap = 2000;
  21. // S8-EXCEPTION-DEPT-NAME-HYDRATION-MASTERDATA-FACTORY-FIX-1:未归属部门 codename(与 S8ManualReportService 同协议常量值)。
  22. private const string UnassignedDepartmentCode = "D-UNASSIGNED";
  23. public S8ExceptionService(
  24. SqlSugarRepository<AdoS8Exception> rep,
  25. SqlSugarRepository<AdoS0DepartmentMaster> deptRep,
  26. SqlSugarRepository<AdoS0EmployeeMaster> empRep,
  27. SqlSugarRepository<SysUser> sysUserRep,
  28. S8ImpactMetricsService impactMetricsService,
  29. IOptions<S8MasterDataOptions> masterDataOptions)
  30. {
  31. _rep = rep;
  32. _deptRep = deptRep;
  33. _empRep = empRep;
  34. _sysUserRep = sysUserRep;
  35. _impactMetricsService = impactMetricsService;
  36. _masterDataOptions = masterDataOptions.Value;
  37. }
  38. public async Task<(int total, List<AdoS8ExceptionListItemDto> list)> GetPagedAsync(AdoS8ExceptionQueryDto q)
  39. {
  40. (q.Page, q.PageSize) = PagingGuard.Normalize(q.Page, q.PageSize);
  41. var pendingStatuses = new[] { "NEW", "ASSIGNED", "IN_PROGRESS", "PENDING_VERIFICATION" };
  42. var includeUnclassified = q.IncludeUnclassified == true;
  43. // S8-SLA-TIMEOUT-RUNTIME-1(P3):方法入口取一次 now,供 q.TimeoutFlag 筛选 + 投影写 TimeoutFlag 共用,避免分页前后不一致。
  44. var timeoutNow = DateTime.Now;
  45. var query = _rep.Context.Queryable<AdoS8Exception>()
  46. .LeftJoin<AdoS8SceneConfig>((e, sc) =>
  47. e.TenantId == sc.TenantId && e.FactoryId == sc.FactoryId && e.SceneCode == sc.SceneCode)
  48. .LeftJoin<AdoS8WatchRule>((e, sc, wr) =>
  49. e.TenantId == wr.TenantId && e.FactoryId == wr.FactoryId && e.SourceRuleCode == wr.RuleCode)
  50. .Where((e, sc, wr) => e.TenantId == q.TenantId && e.FactoryId == q.FactoryId && !e.IsDeleted)
  51. .WhereIF(!includeUnclassified, (e, sc, wr) => !SqlFunc.IsNullOrEmpty(e.ExceptionTypeCode))
  52. .WhereIF(!string.IsNullOrWhiteSpace(q.Status), (e, sc, wr) => e.Status == q.Status)
  53. .WhereIF(string.IsNullOrWhiteSpace(q.Status) && q.StatusBucket == "pending",
  54. (e, sc, wr) => pendingStatuses.Contains(e.Status))
  55. // S8-SEVERITY-FOLLOW-SERIOUS-STANDARDIZE-EXEC-1:legacy 兼容 — q.Severity 传 LOW/MEDIUM/HIGH/CRITICAL 时
  56. // 通过 Normalize 映射为 FOLLOW/SERIOUS 再过滤。
  57. .WhereIF(!string.IsNullOrWhiteSpace(q.Severity),
  58. (e, sc, wr) => e.Severity == S8SeverityCode.Normalize(q.Severity))
  59. .WhereIF(!string.IsNullOrWhiteSpace(q.SceneCode), (e, sc, wr) => e.SceneCode == q.SceneCode)
  60. .WhereIF(!string.IsNullOrWhiteSpace(q.ModuleCode), (e, sc, wr) => e.ModuleCode == q.ModuleCode)
  61. // S8-DASHBOARD-DATA-ALIGN-S1S7-1:看板明细表传 OnlyS1S7Modules=true 时与 KPI 口径对齐;
  62. // 异常列表页不传该参数,行为保持不变(仍可见 NULL module 的 legacy 行)。
  63. .WhereIF(q.OnlyS1S7Modules == true, (e, sc, wr) => S8ModuleCode.All.Contains(e.ModuleCode))
  64. .WhereIF(q.DeptId.HasValue, (e, sc, wr) => e.ResponsibleDeptId == q.DeptId!.Value || e.OccurrenceDeptId == q.DeptId!.Value)
  65. // S8-SLA-TIMEOUT-RUNTIME-1(P3):q.TimeoutFlag 由 timeout_flag 字段筛选切到 sla_deadline + status 在线计算。
  66. // q.TimeoutFlag=true → 当前超时;q.TimeoutFlag=false → 未当前超时(含未配 SLA / 已关闭 / 已恢复)。
  67. .WhereIF(q.TimeoutFlag == true, (e, sc, wr) => e.SlaDeadline != null && e.SlaDeadline < timeoutNow && e.Status != "CLOSED" && e.Status != "RECOVERED")
  68. .WhereIF(q.TimeoutFlag == false, (e, sc, wr) => !(e.SlaDeadline != null && e.SlaDeadline < timeoutNow && e.Status != "CLOSED" && e.Status != "RECOVERED"))
  69. .WhereIF(q.BeginTime.HasValue, (e, sc, wr) => e.CreatedAt >= q.BeginTime!.Value)
  70. .WhereIF(q.EndTime.HasValue, (e, sc, wr) => e.CreatedAt <= q.EndTime!.Value)
  71. .WhereIF(!string.IsNullOrWhiteSpace(q.ProcessNodeCode), (e, sc, wr) => e.ProcessNodeCode == q.ProcessNodeCode)
  72. .WhereIF(!string.IsNullOrWhiteSpace(q.RelatedObjectCode), (e, sc, wr) => e.RelatedObjectCode == q.RelatedObjectCode)
  73. .WhereIF(!string.IsNullOrWhiteSpace(q.OrderFlowCode), (e, sc, wr) => e.OrderFlowCode == q.OrderFlowCode)
  74. .WhereIF(!string.IsNullOrWhiteSpace(q.StageCode), (e, sc, wr) => e.StageCode == q.StageCode)
  75. .WhereIF(!string.IsNullOrWhiteSpace(q.RuleMechanism), (e, sc, wr) => e.RuleMechanism == q.RuleMechanism)
  76. .WhereIF(q.RecoveredStatus == "RECOVERED", (e, sc, wr) => e.RecoveredAt != null)
  77. .WhereIF(q.RecoveredStatus == "ACTIVE", (e, sc, wr) => e.RecoveredAt == null)
  78. .WhereIF(!string.IsNullOrWhiteSpace(q.RuleType), (e, sc, wr) => wr.RuleType == q.RuleType)
  79. .WhereIF(!string.IsNullOrWhiteSpace(q.Keyword),
  80. (e, sc, wr) => e.Title.Contains(q.Keyword!) || e.ExceptionCode.Contains(q.Keyword!));
  81. var total = await query.CountAsync();
  82. // S8-DEMO-IMPACT-SORT-NOTICE-1:候选集合 <= ImpactSortCandidateCap 时走"全量取 → 内存填影响 → 排序 → 分页",
  83. // 支持 impactScore / repeatCount30d / cumulativeLossHours30d 等运行期字段排序;超过则降级 DB 分页 createdAt DESC。
  84. var useImpactSort = total > 0 && total <= ImpactSortCandidateCap;
  85. List<AdoS8ExceptionListItemDto> list;
  86. if (useImpactSort)
  87. {
  88. // 先按 query 命中集合一次性取候选,order 仅 createdAt DESC 用于稳定性,后续内存覆写排序。
  89. var candidates = await query
  90. .OrderBy((e, sc, wr) => e.CreatedAt, OrderByType.Desc)
  91. .Take(ImpactSortCandidateCap)
  92. .Select((e, sc, wr) => new AdoS8ExceptionListItemDto
  93. {
  94. Id = e.Id,
  95. FactoryId = e.FactoryId,
  96. ExceptionCode = e.ExceptionCode,
  97. Title = e.Title,
  98. Status = e.Status,
  99. Severity = e.Severity,
  100. PriorityScore = e.PriorityScore,
  101. PriorityLevel = e.PriorityLevel,
  102. SceneCode = e.SceneCode,
  103. SceneName = sc.SceneName,
  104. ModuleCode = e.ModuleCode,
  105. ResponsibleDeptId = e.ResponsibleDeptId,
  106. OccurrenceDeptId = e.OccurrenceDeptId,
  107. AssigneeId = e.AssigneeId,
  108. SlaDeadline = e.SlaDeadline,
  109. TimeoutFlag = e.SlaDeadline != null && e.SlaDeadline < timeoutNow && e.Status != "CLOSED" && e.Status != "RECOVERED",
  110. CreatedAt = e.CreatedAt,
  111. ClosedAt = e.ClosedAt,
  112. ExceptionTypeCode = e.ExceptionTypeCode,
  113. RecoveredAt = e.RecoveredAt,
  114. SourceRuleCode = e.SourceRuleCode,
  115. SourceObjectType = e.SourceObjectType,
  116. SourceObjectId = e.SourceObjectId,
  117. DedupKey = e.DedupKey,
  118. LastDetectedAt = e.LastDetectedAt,
  119. RuleType = wr.RuleType,
  120. RelatedObjectCode = e.RelatedObjectCode,
  121. OrderFlowCode = e.OrderFlowCode,
  122. StageCode = e.StageCode,
  123. RuleMechanism = e.RuleMechanism,
  124. })
  125. .ToListAsync();
  126. // DTO 后处理:损失时间 / 是否超时关闭 / 标签。
  127. foreach (var r in candidates)
  128. {
  129. r.StatusLabel = S8Labels.StatusLabel(r.Status);
  130. r.SeverityLabel = S8Labels.SeverityLabel(r.Severity);
  131. r.ModuleName = string.IsNullOrWhiteSpace(r.ModuleCode) ? null : S8ModuleCode.Label(r.ModuleCode!);
  132. var (lossHours, isOverdueClosed) = ComputeClosureMetrics(r.CreatedAt, r.ClosedAt, r.SlaDeadline);
  133. r.LossHours = lossHours;
  134. r.IsOverdueClosed = isOverdueClosed;
  135. }
  136. // 影响统计批量填充(30 天窗口聚合一次性 GROUP BY,O(n) 回填,无 N+1)。
  137. var impactRows = candidates
  138. .Select(r => new ExceptionImpactRow
  139. {
  140. Id = r.Id,
  141. ExceptionTypeCode = r.ExceptionTypeCode,
  142. Severity = r.Severity,
  143. CreatedAt = r.CreatedAt,
  144. ClosedAt = r.ClosedAt,
  145. TimeoutFlag = r.TimeoutFlag,
  146. })
  147. .ToList();
  148. await _impactMetricsService.FillBatchAsync(q.TenantId, q.FactoryId, impactRows);
  149. var impactById = impactRows.ToDictionary(x => x.Id);
  150. foreach (var r in candidates)
  151. {
  152. if (!impactById.TryGetValue(r.Id, out var snap)) continue;
  153. r.RepeatCount30d = snap.RepeatCount30d;
  154. r.CumulativeLossHours30d = snap.CumulativeLossHours30d;
  155. r.ImpactScore = snap.ImpactScore;
  156. r.SuggestedAttentionLevel = snap.SuggestedAttentionLevel;
  157. r.SuggestedAttentionLabel = snap.SuggestedAttentionLabel;
  158. r.ImpactReason = snap.ImpactReason;
  159. }
  160. // 内存排序(白名单,非白名单回退 createdAt DESC)。
  161. var sorted = ApplyImpactSort(candidates, q.SortField, q.SortOrder);
  162. // 内存分页。
  163. list = sorted
  164. .Skip((q.Page - 1) * q.PageSize)
  165. .Take(q.PageSize)
  166. .ToList();
  167. }
  168. else
  169. {
  170. // 降级路径:候选 > ImpactSortCandidateCap,DB 分页 createdAt DESC,不填充影响字段(前端展示 0/null)。
  171. list = await query
  172. .OrderBy((e, sc, wr) => e.CreatedAt, OrderByType.Desc)
  173. .Select((e, sc, wr) => new AdoS8ExceptionListItemDto
  174. {
  175. Id = e.Id,
  176. FactoryId = e.FactoryId,
  177. ExceptionCode = e.ExceptionCode,
  178. Title = e.Title,
  179. Status = e.Status,
  180. Severity = e.Severity,
  181. PriorityScore = e.PriorityScore,
  182. PriorityLevel = e.PriorityLevel,
  183. SceneCode = e.SceneCode,
  184. SceneName = sc.SceneName,
  185. ModuleCode = e.ModuleCode,
  186. ResponsibleDeptId = e.ResponsibleDeptId,
  187. OccurrenceDeptId = e.OccurrenceDeptId,
  188. AssigneeId = e.AssigneeId,
  189. SlaDeadline = e.SlaDeadline,
  190. TimeoutFlag = e.SlaDeadline != null && e.SlaDeadline < timeoutNow && e.Status != "CLOSED" && e.Status != "RECOVERED",
  191. CreatedAt = e.CreatedAt,
  192. ClosedAt = e.ClosedAt,
  193. ExceptionTypeCode = e.ExceptionTypeCode,
  194. RecoveredAt = e.RecoveredAt,
  195. SourceRuleCode = e.SourceRuleCode,
  196. SourceObjectType = e.SourceObjectType,
  197. SourceObjectId = e.SourceObjectId,
  198. DedupKey = e.DedupKey,
  199. LastDetectedAt = e.LastDetectedAt,
  200. RuleType = wr.RuleType,
  201. RelatedObjectCode = e.RelatedObjectCode,
  202. OrderFlowCode = e.OrderFlowCode,
  203. StageCode = e.StageCode,
  204. RuleMechanism = e.RuleMechanism,
  205. })
  206. .ToPageListAsync(q.Page, q.PageSize);
  207. foreach (var r in list)
  208. {
  209. r.StatusLabel = S8Labels.StatusLabel(r.Status);
  210. r.SeverityLabel = S8Labels.SeverityLabel(r.Severity);
  211. r.ModuleName = string.IsNullOrWhiteSpace(r.ModuleCode) ? null : S8ModuleCode.Label(r.ModuleCode!);
  212. var (lossHours, isOverdueClosed) = ComputeClosureMetrics(r.CreatedAt, r.ClosedAt, r.SlaDeadline);
  213. r.LossHours = lossHours;
  214. r.IsOverdueClosed = isOverdueClosed;
  215. }
  216. }
  217. await FillDisplayNamesAsync(list, q.FactoryId);
  218. return (total, list);
  219. }
  220. // S8-DEMO-IMPACT-SORT-NOTICE-1:白名单排序;severity 自定义键(SERIOUS=2 / FOLLOW=1 / 其他=0)。
  221. // sortField 空或非白名单 → createdAt DESC;sortOrder 非 asc/desc → desc。
  222. private static List<AdoS8ExceptionListItemDto> ApplyImpactSort(
  223. List<AdoS8ExceptionListItemDto> rows, string? sortField, string? sortOrder)
  224. {
  225. var asc = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase);
  226. var key = (sortField ?? string.Empty).Trim();
  227. return key switch
  228. {
  229. "impactScore" => asc
  230. ? rows.OrderBy(r => r.ImpactScore).ThenByDescending(r => r.CreatedAt).ToList()
  231. : rows.OrderByDescending(r => r.ImpactScore).ThenByDescending(r => r.CreatedAt).ToList(),
  232. "repeatCount30d" => asc
  233. ? rows.OrderBy(r => r.RepeatCount30d).ThenByDescending(r => r.CreatedAt).ToList()
  234. : rows.OrderByDescending(r => r.RepeatCount30d).ThenByDescending(r => r.CreatedAt).ToList(),
  235. "cumulativeLossHours30d" => asc
  236. ? rows.OrderBy(r => r.CumulativeLossHours30d).ThenByDescending(r => r.CreatedAt).ToList()
  237. : rows.OrderByDescending(r => r.CumulativeLossHours30d).ThenByDescending(r => r.CreatedAt).ToList(),
  238. "severity" => asc
  239. ? rows.OrderBy(r => SeveritySortKey(r.Severity)).ThenByDescending(r => r.CreatedAt).ToList()
  240. : rows.OrderByDescending(r => SeveritySortKey(r.Severity)).ThenByDescending(r => r.CreatedAt).ToList(),
  241. "priorityScore" => asc
  242. ? rows.OrderBy(r => r.PriorityScore).ThenByDescending(r => r.CreatedAt).ToList()
  243. : rows.OrderByDescending(r => r.PriorityScore).ThenByDescending(r => r.CreatedAt).ToList(),
  244. "createdAt" => asc
  245. ? rows.OrderBy(r => r.CreatedAt).ToList()
  246. : rows.OrderByDescending(r => r.CreatedAt).ToList(),
  247. _ => rows.OrderByDescending(r => r.CreatedAt).ToList(),
  248. };
  249. }
  250. private static int SeveritySortKey(string severity) => severity switch
  251. {
  252. "SERIOUS" => 2,
  253. "FOLLOW" => 1,
  254. _ => 0,
  255. };
  256. public async Task<object> GetFilterOptionsAsync(long tenantId, long factoryId)
  257. {
  258. var scenes = await _rep.Context.Queryable<AdoS8SceneConfig>()
  259. .Where(x => x.TenantId == tenantId && x.FactoryId == factoryId && x.Enabled)
  260. .OrderBy(x => x.SortNo)
  261. .Select(x => new { value = x.SceneCode, label = x.SceneName })
  262. .ToListAsync();
  263. var departments = await _deptRep.AsQueryable()
  264. .Where(x => x.FactoryRefId == factoryId)
  265. .OrderBy(x => x.Department)
  266. .Take(500)
  267. .Select(x => new { value = x.Id, label = x.Descr ?? x.Department })
  268. .ToListAsync();
  269. // S8-EXCEPTION-MODULE-DISPLAY-1:业务展示主口径切到 module_code(S1-S7)。
  270. // scenes 保留兼容期,前端筛选已切到 modules。
  271. var modules = S8ModuleCode.All
  272. .Select(code => new { value = code, label = S8ModuleCode.Label(code) })
  273. .ToList();
  274. return new
  275. {
  276. statuses = S8Labels.StatusOptions(),
  277. severities = S8Labels.SeverityOptions(),
  278. scenes,
  279. modules,
  280. departments
  281. };
  282. }
  283. public async Task<AdoS8ExceptionDetailDto?> GetDetailAsync(long id, long tenantId, long factoryId)
  284. {
  285. // S8-SLA-TIMEOUT-RUNTIME-1(P3):详情 TimeoutFlag 与 list 同口径,运行时计算。
  286. var timeoutNow = DateTime.Now;
  287. var rows = await _rep.Context.Queryable<AdoS8Exception>()
  288. .LeftJoin<AdoS8SceneConfig>((e, sc) =>
  289. e.TenantId == sc.TenantId && e.FactoryId == sc.FactoryId && e.SceneCode == sc.SceneCode)
  290. .LeftJoin<AdoS8WatchRule>((e, sc, wr) =>
  291. e.TenantId == wr.TenantId && e.FactoryId == wr.FactoryId && e.SourceRuleCode == wr.RuleCode)
  292. .Where((e, sc, wr) => e.Id == id && e.TenantId == tenantId && e.FactoryId == factoryId && !e.IsDeleted)
  293. .Select((e, sc, wr) => new AdoS8ExceptionDetailDto
  294. {
  295. Id = e.Id,
  296. FactoryId = e.FactoryId,
  297. ExceptionCode = e.ExceptionCode,
  298. Title = e.Title,
  299. Description = e.Description,
  300. Status = e.Status,
  301. Severity = e.Severity,
  302. PriorityScore = e.PriorityScore,
  303. PriorityLevel = e.PriorityLevel,
  304. SceneCode = e.SceneCode,
  305. SceneName = sc.SceneName,
  306. ModuleCode = e.ModuleCode,
  307. SourceType = e.SourceType,
  308. OccurrenceDeptId = e.OccurrenceDeptId,
  309. ResponsibleDeptId = e.ResponsibleDeptId,
  310. ResponsibleGroupId = e.ResponsibleGroupId,
  311. AssigneeId = e.AssigneeId,
  312. ReporterId = e.ReporterId,
  313. SlaDeadline = e.SlaDeadline,
  314. // S8-SLA-TIMEOUT-RUNTIME-1(P3):详情展示 TimeoutFlag = 在线计算。
  315. TimeoutFlag = e.SlaDeadline != null && e.SlaDeadline < timeoutNow && e.Status != "CLOSED" && e.Status != "RECOVERED",
  316. CreatedAt = e.CreatedAt,
  317. ClosedAt = e.ClosedAt,
  318. AssignedAt = e.AssignedAt,
  319. UpdatedAt = e.UpdatedAt,
  320. ActiveFlowInstanceId = e.ActiveFlowInstanceId,
  321. ActiveFlowBizType = e.ActiveFlowBizType,
  322. VerifierId = e.VerifierId,
  323. VerificationAssignedAt = e.VerificationAssignedAt,
  324. VerifiedAt = e.VerifiedAt,
  325. VerificationResult = e.VerificationResult,
  326. VerificationRemark = e.VerificationRemark,
  327. SourceRuleId = e.SourceRuleId,
  328. RelatedObjectCode = e.RelatedObjectCode,
  329. DedupKey = e.DedupKey,
  330. LastDetectedAt = e.LastDetectedAt,
  331. RecoveredAt = e.RecoveredAt,
  332. SourceRuleCode = e.SourceRuleCode,
  333. SourceObjectType = e.SourceObjectType,
  334. SourceObjectId = e.SourceObjectId,
  335. ExceptionTypeCode = e.ExceptionTypeCode,
  336. RuleType = wr.RuleType,
  337. OrderFlowCode = e.OrderFlowCode,
  338. StageCode = e.StageCode,
  339. RuleMechanism = e.RuleMechanism,
  340. })
  341. .Take(1)
  342. .ToListAsync();
  343. if (rows.Count == 0) return null;
  344. var d = rows[0];
  345. d.StatusLabel = S8Labels.StatusLabel(d.Status);
  346. d.SeverityLabel = S8Labels.SeverityLabel(d.Severity);
  347. d.ModuleName = string.IsNullOrWhiteSpace(d.ModuleCode) ? null : S8ModuleCode.Label(d.ModuleCode!);
  348. // S8-DEMO-CORE-FIELD-COMPLETE-1:详情与列表同口径,DTO 后处理计算损失时间 / 是否超时关闭。
  349. var (lossHours, isOverdueClosed) = ComputeClosureMetrics(d.CreatedAt, d.ClosedAt, d.SlaDeadline);
  350. d.LossHours = lossHours;
  351. d.IsOverdueClosed = isOverdueClosed;
  352. await FillDisplayNamesAsync(new[] { d }, factoryId);
  353. return d;
  354. }
  355. // S8-DEMO-CORE-FIELD-COMPLETE-1:损失时间(小时,1 位小数)+ 是否超时关闭(冻结判定)。
  356. // 未关闭 → LossHours=null;未关闭或无 SLA → IsOverdueClosed=null。与运行时 TimeoutFlag 不复用。
  357. private static (decimal? lossHours, bool? isOverdueClosed) ComputeClosureMetrics(
  358. DateTime createdAt, DateTime? closedAt, DateTime? slaDeadline)
  359. {
  360. decimal? lossHours = closedAt == null
  361. ? (decimal?)null
  362. : Math.Round((decimal)(closedAt.Value - createdAt).TotalHours, 1);
  363. bool? isOverdueClosed = (closedAt == null || slaDeadline == null)
  364. ? (bool?)null
  365. : closedAt.Value > slaDeadline.Value;
  366. return (lossHours, isOverdueClosed);
  367. }
  368. // S8-DEPT-DISPLAY-CONSISTENCY-1(P0-A-2/A-3):list 行也水合 OccurrenceDeptName;
  369. // 部门查询加 factory_ref_id 二次约束,避免跨 factory 同名 / 同 RecID 错位。
  370. private async Task FillDisplayNamesAsync(IEnumerable<AdoS8ExceptionListItemDto> rows, long factoryId)
  371. {
  372. var list = rows.ToList();
  373. if (list.Count == 0) return;
  374. var deptIds = list
  375. .Select(x => x.ResponsibleDeptId)
  376. .Concat(list.Select(x => x.OccurrenceDeptId))
  377. .Where(x => x > 0)
  378. .Distinct()
  379. .ToList();
  380. var empIds = list
  381. .Select(x => x.AssigneeId ?? 0L)
  382. .Concat(list.OfType<AdoS8ExceptionDetailDto>().Select(x => x.ReporterId ?? 0L))
  383. .Concat(list.OfType<AdoS8ExceptionDetailDto>().Select(x => x.VerifierId ?? 0L))
  384. .Where(x => x > 0)
  385. .Distinct()
  386. .ToList();
  387. // S8-EXCEPTION-DEPT-NAME-HYDRATION-MASTERDATA-FACTORY-FIX-1:
  388. // 部门主数据位于 S0 域 factory(与 S8 运营 factoryId 不同源),且 DepartmentMaster.tenant_id 属 S0 域租户。
  389. // 用主数据 factory + ClearFilter(factory_ref_id + 具体 dept_id 集合做硬边界,等价安全,无跨租户泄漏),
  390. // 取代旧的“运营 factoryId + 全局租户过滤”——后者必然查空 → 全部回落“部门ID:N”。
  391. var masterDataFactoryId = await ResolveMasterDataDepartmentFactoryRefIdAsync(factoryId);
  392. var deptMap = deptIds.Count == 0
  393. ? new Dictionary<long, string>()
  394. : (await _deptRep.AsQueryable().ClearFilter()
  395. .Where(x => deptIds.Contains(x.Id) && x.FactoryRefId == masterDataFactoryId)
  396. .Select(x => new { x.Id, Name = x.Descr ?? x.Department })
  397. .ToListAsync())
  398. .ToDictionary(x => x.Id, x => x.Name);
  399. // EmployeeMaster.tenant_id 与 SysUser.TenantId 历史错位(797403760988229 ≠ 系统租户),
  400. // 走全局 multi-tenant filter 会全过滤为空 → 详情页处理人/检验人姓名显示数字 ID。
  401. // 与 S8MasterDataAdapter.GetEmployeesAsync / S8TaskFlowService.GetEmployeeSysUserIdAsync 同口径,
  402. // 局部 ClearFilter + employee.Id 主键集合作为安全边界,无跨租户泄漏。
  403. var empMap = empIds.Count == 0
  404. ? new Dictionary<long, string>()
  405. : (await _empRep.AsQueryable().ClearFilter()
  406. .Where(x => empIds.Contains(x.Id))
  407. .Select(x => new { x.Id, Name = x.Name ?? x.Employee })
  408. .ToListAsync())
  409. .ToDictionary(x => x.Id, x => x.Name);
  410. // S8-REPORTER-IDSPACE-FIX-1(P0-B-1):ReporterId 新协议 = EmployeeMaster.RecID(与 assignee/verifier 同 idspace)。
  411. // 水合优先级:EmployeeMaster.Name → fallback SysUser.RealName/Account(兼容旧数据 reporter=SysUser.Id)。
  412. // 旧注释 “ReporterId 现在记录 sysUser.Id” 已失效,由本批协议替代。
  413. var reporterUserIds = list.OfType<AdoS8ExceptionDetailDto>()
  414. .Select(x => x.ReporterId ?? 0L)
  415. .Where(x => x > 0)
  416. .Distinct()
  417. .ToList();
  418. var reporterUserMap = reporterUserIds.Count == 0
  419. ? new Dictionary<long, string>()
  420. : (await _sysUserRep.AsQueryable().ClearFilter()
  421. .Where(u => reporterUserIds.Contains(u.Id))
  422. .Select(u => new { u.Id, u.RealName, u.Account })
  423. .ToListAsync())
  424. .ToDictionary(
  425. u => u.Id,
  426. u => !string.IsNullOrWhiteSpace(u.RealName) ? u.RealName : u.Account);
  427. foreach (var row in list)
  428. {
  429. row.ResponsibleDeptName = ResolveDeptName(row.ResponsibleDeptId, deptMap);
  430. row.OccurrenceDeptName = ResolveDeptName(row.OccurrenceDeptId, deptMap);
  431. row.AssigneeName = row.AssigneeId.HasValue ? empMap.GetValueOrDefault(row.AssigneeId.Value) : null;
  432. if (row is AdoS8ExceptionDetailDto detail)
  433. {
  434. // S8-REPORTER-IDSPACE-FIX-1:empMap(EmployeeMaster.RecID)优先,reporterUserMap(SysUser.Id)兜底兼容旧数据。
  435. detail.ReporterName = detail.ReporterId.HasValue
  436. ? (empMap.GetValueOrDefault(detail.ReporterId.Value) ?? reporterUserMap.GetValueOrDefault(detail.ReporterId.Value))
  437. : null;
  438. detail.VerifierName = detail.VerifierId.HasValue ? empMap.GetValueOrDefault(detail.VerifierId.Value) : null;
  439. }
  440. }
  441. }
  442. // S8-DEPT-DISPLAY-CONSISTENCY-1(P0-A-3):dept fallback —— 0/缺失=未归属,查不到名=部门ID:{id}。
  443. private static string? ResolveDeptName(long deptId, Dictionary<long, string> deptMap)
  444. {
  445. if (deptId <= 0) return "未归属";
  446. return deptMap.TryGetValue(deptId, out var name) && !string.IsNullOrWhiteSpace(name)
  447. ? name
  448. : $"部门ID:{deptId}";
  449. }
  450. // S8-EXCEPTION-DEPT-NAME-HYDRATION-MASTERDATA-FACTORY-FIX-1:解析 DepartmentMaster 主数据所在 factory_ref_id。
  451. // 优先读配置 S8MasterData.DepartmentFactoryRefId;缺失(<=0)则推导“唯一 active D-UNASSIGNED 的 FactoryRefId”。
  452. // 显示路径不抛错:推导 0 条 / 不唯一时退回运营 factoryId(保持旧“部门ID:N”兜底,绝不让列表/详情查询失败)。
  453. private async Task<long> ResolveMasterDataDepartmentFactoryRefIdAsync(long fallbackFactoryId)
  454. {
  455. var configured = _masterDataOptions.DepartmentFactoryRefId;
  456. if (configured > 0) return configured;
  457. var factories = await _deptRep.AsQueryable().ClearFilter()
  458. .Where(x => x.Department == UnassignedDepartmentCode && x.IsActive)
  459. .Select(x => x.FactoryRefId)
  460. .Distinct()
  461. .ToListAsync();
  462. return factories.Count == 1 ? factories[0] : fallbackFactoryId;
  463. }
  464. /// <summary>
  465. /// S2-EW:debug-only 软删 RelatedObjectCode 以 "TEST_G09_" 开头的异常。
  466. /// 调用方仅限 <see cref="Controllers.S8.AdoS8WatchDebugController"/>,由其 _debugEndpointEnabled 守门。
  467. /// 实现纪律:
  468. /// - 仅 SET IsDeleted=true,不动业务字段(status/closed_at/source_rule_id 等)。
  469. /// - 前缀字面量硬编码,不接受外部 prefix;防止前缀注入。
  470. /// - 不联动 timeline / decision / evidence;业务读路径以父 IsDeleted 屏蔽。
  471. /// </summary>
  472. public async Task<int> SoftDeleteTestPrefixAsync(long tenantId, long factoryId)
  473. {
  474. const string prefix = "TEST_G09_";
  475. var affected = await _rep.AsUpdateable()
  476. .SetColumns(e => new AdoS8Exception { IsDeleted = true })
  477. .Where(e => e.TenantId == tenantId
  478. && e.FactoryId == factoryId
  479. && !e.IsDeleted
  480. && e.RelatedObjectCode != null
  481. && e.RelatedObjectCode.StartsWith(prefix))
  482. .ExecuteCommandAsync();
  483. System.Diagnostics.Trace.TraceWarning(
  484. $"[S2-EW] soft-deleted {affected} TEST_G09_ exceptions, tenant={tenantId} factory={factoryId}");
  485. return affected;
  486. }
  487. }