AidopKanbanController.cs 55 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318
  1. using Admin.NET.Core;
  2. using Admin.NET.Plugin.AiDOP.Entity;
  3. using Admin.NET.Plugin.AiDOP.FinishedWarehouse;
  4. using Admin.NET.Plugin.AiDOP.Infrastructure;
  5. using Admin.NET.Plugin.AiDOP.Manufacturing;
  6. using Admin.NET.Plugin.AiDOP.MaterialWarehouse;
  7. using Admin.NET.Plugin.AiDOP.Order;
  8. using Admin.NET.Plugin.AiDOP.ProcurementExecution;
  9. using Admin.NET.Plugin.AiDOP.Production;
  10. using Admin.NET.Plugin.AiDOP.Supply;
  11. using Admin.NET.Plugin.AiDOP.SmartOps;
  12. using SqlSugar;
  13. namespace Admin.NET.Plugin.AiDOP.Controllers;
  14. [ApiController]
  15. [Route("api/[controller]")]
  16. [AllowAnonymous]
  17. [NonUnify]
  18. public partial class AidopKanbanController : ControllerBase
  19. {
  20. private readonly ISqlSugarClient _db;
  21. private readonly S1MdpSyncTransformService _s1MdpSyncTransformService;
  22. private readonly S2MdpSyncTransformService _s2MdpSyncTransformService;
  23. private readonly S3MdpSyncTransformService _s3MdpSyncTransformService;
  24. private readonly S4MdpSyncTransformService _s4MdpSyncTransformService;
  25. private readonly S5MdpSyncTransformService _s5MdpSyncTransformService;
  26. private readonly S6MdpSyncTransformService _s6MdpSyncTransformService;
  27. private readonly S7MdpSyncTransformService _s7MdpSyncTransformService;
  28. private readonly SmartOpsKpiAtomicQueryService _atomicQuery;
  29. private readonly OutsourceIssueMdpSyncService _outsourceIssueMdpSyncService;
  30. private readonly PurchaseReceiptMdpSyncService _purchaseReceiptMdpSyncService;
  31. public AidopKanbanController(
  32. ISqlSugarClient db,
  33. S1MdpSyncTransformService s1MdpSyncTransformService,
  34. S2MdpSyncTransformService s2MdpSyncTransformService,
  35. S3MdpSyncTransformService s3MdpSyncTransformService,
  36. S4MdpSyncTransformService s4MdpSyncTransformService,
  37. S5MdpSyncTransformService s5MdpSyncTransformService,
  38. S6MdpSyncTransformService s6MdpSyncTransformService,
  39. S7MdpSyncTransformService s7MdpSyncTransformService,
  40. SmartOpsKpiAtomicQueryService atomicQuery,
  41. OutsourceIssueMdpSyncService outsourceIssueMdpSyncService,
  42. PurchaseReceiptMdpSyncService purchaseReceiptMdpSyncService)
  43. {
  44. _db = db;
  45. _s1MdpSyncTransformService = s1MdpSyncTransformService;
  46. _s2MdpSyncTransformService = s2MdpSyncTransformService;
  47. _s3MdpSyncTransformService = s3MdpSyncTransformService;
  48. _s4MdpSyncTransformService = s4MdpSyncTransformService;
  49. _s5MdpSyncTransformService = s5MdpSyncTransformService;
  50. _s6MdpSyncTransformService = s6MdpSyncTransformService;
  51. _s7MdpSyncTransformService = s7MdpSyncTransformService;
  52. _atomicQuery = atomicQuery;
  53. _outsourceIssueMdpSyncService = outsourceIssueMdpSyncService;
  54. _purchaseReceiptMdpSyncService = purchaseReceiptMdpSyncService;
  55. }
  56. [HttpGet("home-l1")]
  57. public async Task<IActionResult> GetHomeL1([FromQuery] long factoryId = 1)
  58. {
  59. var tenantId = AidopTenantHelper.GetTenantId(HttpContext);
  60. const string sql = """
  61. SELECT v.module_code AS ModuleCode, v.metric_code AS MetricCode,
  62. v.metric_value AS MetricValue, v.target_value AS TargetValue,
  63. v.status_color AS StatusColor, v.trend_flag AS TrendFlag,
  64. k.Direction, k.YellowThreshold, k.RedThreshold
  65. FROM ado_s9_kpi_value_l1_day v
  66. LEFT JOIN ado_smart_ops_kpi_master k
  67. ON k.MetricCode = v.metric_code COLLATE utf8mb4_general_ci AND k.TenantId = v.tenant_id
  68. WHERE v.tenant_id=@tenantId AND v.factory_id=@factoryId AND v.is_deleted=0
  69. AND v.biz_date = (SELECT MAX(biz_date) FROM ado_s9_kpi_value_l1_day
  70. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0)
  71. ORDER BY v.module_code
  72. """;
  73. var raw = await _db.Ado.SqlQueryAsync<HomeL1RawDto>(sql, new { tenantId, factoryId });
  74. var rows = raw.Select(r => new HomeL1Dto
  75. {
  76. ModuleCode = r.ModuleCode,
  77. MetricCode = r.MetricCode,
  78. MetricValue = r.MetricValue,
  79. TargetValue = r.TargetValue,
  80. TrendFlag = r.TrendFlag,
  81. StatusColor = AidopS4KpiMerge.AchievementLevel(
  82. r.MetricValue, r.TargetValue,
  83. r.Direction ?? "higher_is_better",
  84. r.YellowThreshold, r.RedThreshold)
  85. }).ToList();
  86. return Ok(rows);
  87. }
  88. [HttpGet("s8-alerts")]
  89. public async Task<IActionResult> GetS8Alerts([FromQuery] long factoryId = 1)
  90. {
  91. var tenantId = AidopTenantHelper.GetTenantId(HttpContext);
  92. const string sql = """
  93. SELECT DATE_FORMAT(alert_time, '%H:%i') AS Time,
  94. IFNULL(SUBSTRING_INDEX(metric_code, '_', 1), 'S8') AS Module,
  95. IFNULL(message, '异常告警') AS Message,
  96. IFNULL(level_code, 'medium') AS LevelCode
  97. FROM ado_s8_alert_record
  98. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0
  99. ORDER BY alert_time DESC
  100. LIMIT 6
  101. """;
  102. var rows = await _db.Ado.SqlQueryAsync<S8AlertDto>(sql, new { tenantId, factoryId });
  103. return Ok(rows.Select(x => new
  104. {
  105. time = x.Time,
  106. module = x.Module,
  107. message = x.Message,
  108. level = x.LevelCode,
  109. levelLabel = x.LevelCode switch
  110. {
  111. "critical" => "严重",
  112. "high" => "高",
  113. "medium" => "中",
  114. _ => "一般"
  115. }
  116. }));
  117. }
  118. [HttpGet("module-detail")]
  119. public async Task<IActionResult> GetModuleDetail([FromQuery] string moduleCode = "S1", [FromQuery] long factoryId = 1)
  120. {
  121. var tenantId = AidopTenantHelper.GetTenantId(HttpContext);
  122. moduleCode = string.IsNullOrWhiteSpace(moduleCode) ? "S1" : moduleCode.ToUpperInvariant();
  123. const string l2Sql = """
  124. SELECT v.module_code AS ModuleCode, v.metric_code AS MetricCode,
  125. COALESCE(NULLIF(k.MetricName,''), v.metric_code) AS MetricName,
  126. v.metric_value AS MetricValue, v.target_value AS TargetValue,
  127. v.status_color AS StatusColor, v.trend_flag AS TrendFlag, v.biz_date AS StatDate
  128. FROM ado_s9_kpi_value_l2_day v
  129. LEFT JOIN ado_smart_ops_kpi_master k ON k.TenantId=v.tenant_id AND k.MetricCode=v.metric_code AND k.ModuleCode=v.module_code
  130. WHERE v.tenant_id=@tenantId AND v.factory_id=@factoryId AND v.is_deleted=0 AND v.module_code=@moduleCode
  131. AND v.biz_date=(SELECT MAX(biz_date) FROM ado_s9_kpi_value_l2_day
  132. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND module_code=@moduleCode AND is_deleted=0)
  133. ORDER BY v.metric_code
  134. LIMIT 30
  135. """;
  136. const string l3Sql = """
  137. SELECT v.module_code AS ModuleCode, v.metric_code AS MetricCode,
  138. COALESCE(NULLIF(k.MetricName,''), v.metric_code) AS MetricName,
  139. v.metric_value AS MetricValue, v.target_value AS TargetValue,
  140. v.status_color AS StatusColor, v.trend_flag AS TrendFlag, v.biz_date AS StatDate,
  141. p.MetricCode AS ParentMetricCode
  142. FROM ado_s9_kpi_value_l3_day v
  143. LEFT JOIN ado_smart_ops_kpi_master k ON k.TenantId=v.tenant_id AND k.MetricCode=v.metric_code AND k.ModuleCode=v.module_code
  144. LEFT JOIN ado_smart_ops_kpi_master p ON p.Id=k.ParentId
  145. WHERE v.tenant_id=@tenantId AND v.factory_id=@factoryId AND v.is_deleted=0 AND v.module_code=@moduleCode
  146. AND v.biz_date=(SELECT MAX(biz_date) FROM ado_s9_kpi_value_l3_day
  147. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND module_code=@moduleCode AND is_deleted=0)
  148. ORDER BY v.metric_code
  149. LIMIT 60
  150. """;
  151. const string alertSql = """
  152. SELECT DATE_FORMAT(alert_time, '%H:%i:%s') AS Time,
  153. IFNULL(level_code, 'medium') AS LevelCode,
  154. IFNULL(message, '异常告警') AS Message
  155. FROM ado_s8_alert_record
  156. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0
  157. AND (UPPER(IFNULL(module_code,''))=@moduleCode OR UPPER(IFNULL(SUBSTRING_INDEX(metric_code, '_', 1), ''))=@moduleCode)
  158. ORDER BY alert_time DESC
  159. LIMIT 20
  160. """;
  161. const string l2FallbackSql = """
  162. SELECT '' AS ModuleCode, metric_code AS MetricCode, metric_code AS MetricName, metric_value AS MetricValue,
  163. target_value AS TargetValue, status_color AS StatusColor, trend_flag AS TrendFlag, NULL AS StatDate
  164. FROM ado_s9_kpi_value_l2_day
  165. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0
  166. ORDER BY id DESC
  167. LIMIT 30
  168. """;
  169. const string l3FallbackSql = """
  170. SELECT '' AS ModuleCode, metric_code AS MetricCode, metric_code AS MetricName, metric_value AS MetricValue,
  171. target_value AS TargetValue, status_color AS StatusColor, trend_flag AS TrendFlag, NULL AS StatDate
  172. FROM ado_s9_kpi_value_l3_day
  173. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0
  174. ORDER BY id DESC
  175. LIMIT 60
  176. """;
  177. var l2 = new List<KpiDetailDto>();
  178. var l3 = new List<KpiDetailDto>();
  179. var alerts = new List<S8AlertDto>();
  180. try { l2 = await _db.Ado.SqlQueryAsync<KpiDetailDto>(l2Sql, new { moduleCode, tenantId, factoryId }); }
  181. catch { l2 = await _db.Ado.SqlQueryAsync<KpiDetailDto>(l2FallbackSql, new { tenantId, factoryId }); }
  182. try { l3 = await _db.Ado.SqlQueryAsync<KpiDetailDto>(l3Sql, new { moduleCode, tenantId, factoryId }); }
  183. catch { l3 = await _db.Ado.SqlQueryAsync<KpiDetailDto>(l3FallbackSql, new { tenantId, factoryId }); }
  184. try { alerts = await _db.Ado.SqlQueryAsync<S8AlertDto>(alertSql, new { moduleCode, tenantId, factoryId }); }
  185. catch { alerts = new List<S8AlertDto>(); }
  186. var schedules = new List<S2ScheduleDto>();
  187. var s2Alerts = new List<S8AlertDto>();
  188. var decomposition = new List<S2DecompositionDto>();
  189. var trend = new List<S2TrendDto>();
  190. var distribution = new List<S2DistributionDto>();
  191. var branchKpis = new List<S3BranchKpiGroupDto>();
  192. S2SyncStatusDto? syncStatus = null;
  193. if (moduleCode == "S2")
  194. {
  195. syncStatus = await GetS2SyncStatusAsync();
  196. s2Alerts = await GetS2DerivedAlertsAsync(tenantId, factoryId);
  197. if (s2Alerts.Count > 0)
  198. alerts = s2Alerts;
  199. decomposition = await GetS2DecompositionAsync(tenantId, factoryId, l2);
  200. trend = await GetS2TrendAsync(tenantId, factoryId);
  201. distribution = await GetS2DistributionAsync(tenantId, factoryId);
  202. try
  203. {
  204. schedules = await _db.Ado.SqlQueryAsync<S2ScheduleDto>(
  205. """
  206. SELECT work_order AS OrderNo,
  207. COALESCE(NULLIF(item_name, ''), item_code, work_order) AS Product,
  208. COALESCE(NULLIF(sales_order_no, ''), work_order) AS Customer,
  209. prod_line AS ProductionLine,
  210. qty_ordered AS Quantity,
  211. due_date AS DeliveryDate,
  212. first_start_time AS StartTime,
  213. schedule_cycle_days AS CycleDays,
  214. CASE WHEN schedule_satisfaction_flag = 1 THEN 100
  215. WHEN due_date IS NULL OR last_plan_date IS NULL THEN 70
  216. ELSE 65 END AS Satisfaction,
  217. CASE WHEN schedule_satisfaction_flag = 1 THEN '已锁定'
  218. WHEN last_plan_date IS NULL THEN '待排程'
  219. ELSE '资源异常' END AS Status
  220. FROM dwd_order_schedule_trans
  221. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  222. AND calc_batch_id=(SELECT calc_batch_id FROM dwd_order_schedule_trans
  223. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  224. ORDER BY calc_time DESC, id DESC LIMIT 1)
  225. ORDER BY urgent_flag DESC, due_date IS NULL, due_date, id DESC
  226. LIMIT 30
  227. """,
  228. new { tenantId, factoryId });
  229. }
  230. catch
  231. {
  232. schedules = new List<S2ScheduleDto>();
  233. }
  234. }
  235. if (moduleCode == "S3")
  236. {
  237. syncStatus = await GetMdpSyncStatusAsync("S3_MDP_SYNC_TRANSFORM");
  238. var s3Alerts = await GetS3DerivedAlertsAsync(tenantId, factoryId);
  239. if (s3Alerts.Count > 0)
  240. alerts = s3Alerts;
  241. decomposition = await GetS3DecompositionAsync(tenantId, factoryId, l2);
  242. branchKpis = await GetS3BranchKpisAsync(tenantId, factoryId, l2);
  243. trend = await GetS3TrendAsync(tenantId, factoryId);
  244. distribution = await GetS3DistributionAsync(tenantId, factoryId);
  245. }
  246. if (moduleCode == "S4")
  247. {
  248. syncStatus = await GetMdpSyncStatusAsync("S4_MDP_SYNC_TRANSFORM");
  249. var s4Alerts = await GetS4DerivedAlertsAsync(tenantId, factoryId);
  250. if (s4Alerts.Count > 0)
  251. alerts = s4Alerts;
  252. trend = await GetS4TrendAsync(tenantId, factoryId);
  253. distribution = await GetS4DistributionAsync(tenantId, factoryId);
  254. }
  255. var drillGroups = BuildKpiDrillGroups(moduleCode, l2, l3);
  256. if (moduleCode == "S2")
  257. decomposition = EnrichS2DecompositionWithL3(decomposition, l3);
  258. return Ok(new
  259. {
  260. moduleCode,
  261. l2,
  262. l3,
  263. drillGroups,
  264. syncStatus,
  265. decomposition,
  266. branchKpis,
  267. trend,
  268. distribution,
  269. schedules,
  270. alerts = alerts.Select(x => new
  271. {
  272. time = x.Time,
  273. message = x.Message,
  274. level = x.LevelCode
  275. })
  276. });
  277. }
  278. private static List<KpiDrillGroupDto> BuildKpiDrillGroups(string moduleCode, List<KpiDetailDto> l2, List<KpiDetailDto> l3)
  279. {
  280. if (l3.Count == 0) return new List<KpiDrillGroupDto>();
  281. var parentTitles = l2.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  282. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  283. .ToDictionary(g => g.Key, g => g.First().MetricName ?? g.Key, StringComparer.OrdinalIgnoreCase);
  284. return l3.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  285. .GroupBy(u => string.IsNullOrWhiteSpace(u.ParentMetricCode) ? "_root" : u.ParentMetricCode!, StringComparer.OrdinalIgnoreCase)
  286. .Select(g =>
  287. {
  288. var parentCode = g.Key == "_root" ? null : g.Key;
  289. var title = parentCode != null && parentTitles.TryGetValue(parentCode, out var parentName)
  290. ? parentName
  291. : parentCode ?? $"{moduleCode} L3";
  292. return new KpiDrillGroupDto
  293. {
  294. ParentMetricCode = parentCode,
  295. Title = title,
  296. Items = g.Select(item => new KpiDrillItemDto
  297. {
  298. MetricCode = item.MetricCode,
  299. MetricName = item.MetricName ?? item.MetricCode,
  300. MetricValue = item.MetricValue,
  301. TargetValue = item.TargetValue,
  302. StatusColor = item.StatusColor,
  303. TrendFlag = item.TrendFlag,
  304. StatDate = item.StatDate?.ToString("yyyy-MM-dd")
  305. }).ToList()
  306. };
  307. })
  308. .OrderBy(x => x.ParentMetricCode)
  309. .ToList();
  310. }
  311. private static List<S2DecompositionDto> EnrichS2DecompositionWithL3(List<S2DecompositionDto> decomposition, List<KpiDetailDto> l3)
  312. {
  313. if (l3.Count == 0) return decomposition;
  314. var latestL3 = l3.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  315. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  316. .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);
  317. foreach (var section in decomposition)
  318. {
  319. if (section.Title == "工序排程")
  320. {
  321. section.Metrics = new List<string>
  322. {
  323. $"1. 周期:{FormatMetric(latestL3, "S2_L3_001", "天")}",
  324. $"2. 满足率:{FormatMetric(latestL3, "S2_L3_002", "%")}",
  325. $"3. 人效:{FormatMetric(latestL3, "S2_L3_003", "道/人")}"
  326. };
  327. }
  328. else if (section.Title == "资源排程")
  329. {
  330. section.Metrics = new List<string>
  331. {
  332. $"1. 设备周期:{FormatMetric(latestL3, "S2_L3_004", "天")}",
  333. $"2. 设备满足率:{FormatMetric(latestL3, "S2_L3_005", "%")}",
  334. $"3. 设备人效:{FormatMetric(latestL3, "S2_L3_006", "台/人")}"
  335. };
  336. }
  337. }
  338. return decomposition;
  339. }
  340. [HttpPost("s1-mdp/refresh")]
  341. public async Task<IActionResult> RefreshS1Mdp(CancellationToken cancellationToken)
  342. {
  343. var result = await _s1MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  344. return Ok(new
  345. {
  346. ok = true,
  347. result.BatchId,
  348. result.StageRows,
  349. result.StandardRows,
  350. result.DwdRows,
  351. result.KpiRows,
  352. result.AtomicRows
  353. });
  354. }
  355. [HttpPost("s2-mdp/refresh")]
  356. public async Task<IActionResult> RefreshS2Mdp(CancellationToken cancellationToken)
  357. {
  358. var result = await _s2MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  359. return Ok(new
  360. {
  361. ok = true,
  362. result.BatchId,
  363. result.StageRows,
  364. result.StandardRows,
  365. result.DwdRows,
  366. result.KpiRows
  367. });
  368. }
  369. [HttpPost("s3-mdp/refresh")]
  370. public async Task<IActionResult> RefreshS3Mdp(CancellationToken cancellationToken)
  371. {
  372. var result = await _s3MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  373. return Ok(new
  374. {
  375. ok = true,
  376. result.BatchId,
  377. result.StageRows,
  378. result.StandardRows,
  379. result.DwdRows,
  380. result.KpiRows
  381. });
  382. }
  383. [HttpPost("s4-mdp/refresh")]
  384. public async Task<IActionResult> RefreshS4Mdp(CancellationToken cancellationToken)
  385. {
  386. var result = await _s4MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  387. return Ok(new
  388. {
  389. ok = true,
  390. result.BatchId,
  391. result.StageRows,
  392. result.StandardRows,
  393. result.DwdRows,
  394. result.KpiRows
  395. });
  396. }
  397. /// <summary>
  398. /// S5 物料仓储 KPI 手动刷新(T8 SQL Server 跨库读 → DOP DWD/KPI)。
  399. /// 路径 A:方老师 v5.4 KPI #16/#17/#18/#19 原 SQL 直发 T8。
  400. /// 边界:只读 SELECT;仅 refresh 内触发;禁止看板实时查 T8;禁止高频调度。
  401. /// </summary>
  402. [HttpPost("s5-mdp/refresh")]
  403. public async Task<IActionResult> RefreshS5Mdp(CancellationToken cancellationToken)
  404. {
  405. var result = await _s5MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  406. return Ok(new
  407. {
  408. ok = true,
  409. result.BatchId,
  410. result.StageRows,
  411. result.StandardRows,
  412. result.DwdRows,
  413. result.KpiRows,
  414. result.PerKpiDwdRows,
  415. result.PerKpiKpiRows,
  416. result.KpiDenominatorStatus,
  417. result.SourceZtid
  418. });
  419. }
  420. /// <summary>
  421. /// S5 委外发料单 数据中台只读链路手动刷新(DOP 内部:NbrMaster/NbrDetail Type='CA' → mdp_stg → mdp_std)。
  422. /// 独立于 s5-mdp/refresh 的 KPI 管线;只读源、只写 mdp 表;不碰 T8;CA 为 0 行时成功、处理数 0。
  423. /// </summary>
  424. [HttpPost("s5-outsource-issue-mdp/refresh")]
  425. public async Task<IActionResult> RefreshS5OutsourceIssueMdp(CancellationToken cancellationToken)
  426. {
  427. var result = await _outsourceIssueMdpSyncService.RunFullAsync(cancellationToken, "MANUAL");
  428. return Ok(new
  429. {
  430. ok = true,
  431. result.BatchId,
  432. result.HeadStageRows,
  433. result.DetailStageRows,
  434. result.HeadStandardRows,
  435. result.DetailStandardRows
  436. });
  437. }
  438. /// <summary>
  439. /// S5 采购收货单 数据中台只读链路手动刷新(DOP 内部方案 1:PurOrdRctDetail/Master RctType='rc' → mdp_std_purchase_receipt)。
  440. /// 独立于 s5-mdp/refresh 的 KPI 管线;只读源、只写 mdp_std_purchase_receipt;不读/不改 S3 mdp_stg_receipt、S4 ado_s4_receipt;不碰 T8;rc 为 0 行时成功、处理数 0。
  441. /// </summary>
  442. [HttpPost("s5-purchase-receipt-mdp/refresh")]
  443. public async Task<IActionResult> RefreshS5PurchaseReceiptMdp(CancellationToken cancellationToken)
  444. {
  445. var result = await _purchaseReceiptMdpSyncService.RunFullAsync(cancellationToken, "MANUAL");
  446. return Ok(new
  447. {
  448. ok = true,
  449. batchId = result.BatchId,
  450. stdRows = result.StdRows
  451. });
  452. }
  453. /// <summary>
  454. /// S6 生产执行 KPI 手动刷新(T8 SQL Server 跨库读 → DOP DWD/KPI)。
  455. /// 路径 A:方老师 v5.4 KPI #22/#23 原 SQL 直发 T8。
  456. /// </summary>
  457. [HttpPost("s6-mdp/refresh")]
  458. public async Task<IActionResult> RefreshS6Mdp(CancellationToken cancellationToken)
  459. {
  460. var result = await _s6MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  461. return Ok(new
  462. {
  463. ok = true,
  464. result.BatchId,
  465. result.StageRows,
  466. result.StandardRows,
  467. result.DwdRows,
  468. result.KpiRows,
  469. result.PerKpiDwdRows,
  470. result.PerKpiKpiRows,
  471. result.KpiDenominatorStatus,
  472. result.SourceZtid
  473. });
  474. }
  475. /// <summary>
  476. /// S7 成品仓储 KPI 手动刷新(T8 SQL Server 跨库读 → DOP DWD/KPI)。
  477. /// 路径 A:方老师 v5.4 KPI #25/#26/#27 原 SQL 直发 T8。
  478. /// </summary>
  479. [HttpPost("s7-mdp/refresh")]
  480. public async Task<IActionResult> RefreshS7Mdp(CancellationToken cancellationToken)
  481. {
  482. var result = await _s7MdpSyncTransformService.RunFullAsync(cancellationToken, "MANUAL");
  483. return Ok(new
  484. {
  485. ok = true,
  486. result.BatchId,
  487. result.StageRows,
  488. result.StandardRows,
  489. result.DwdRows,
  490. result.KpiRows,
  491. result.PerKpiDwdRows,
  492. result.PerKpiKpiRows,
  493. result.KpiDenominatorStatus,
  494. result.SourceZtid
  495. });
  496. }
  497. [HttpPost("mdp/refresh/{moduleCode}")]
  498. public async Task<IActionResult> RefreshModuleMdp(string moduleCode, CancellationToken cancellationToken)
  499. {
  500. var mc = string.IsNullOrWhiteSpace(moduleCode) ? string.Empty : moduleCode.Trim().ToUpperInvariant();
  501. return mc switch
  502. {
  503. "S1" => await RefreshS1Mdp(cancellationToken),
  504. "S2" => await RefreshS2Mdp(cancellationToken),
  505. "S3" => await RefreshS3Mdp(cancellationToken),
  506. "S4" => await RefreshS4Mdp(cancellationToken),
  507. _ => Ok(new { ok = true, moduleCode = mc, message = "当前模块暂无 MDP 刷新任务,仅刷新动态看板数据。" })
  508. };
  509. }
  510. private async Task<S2SyncStatusDto?> GetS2SyncStatusAsync()
  511. {
  512. return await GetMdpSyncStatusAsync("S2_MDP_SYNC_TRANSFORM");
  513. }
  514. private async Task<S2SyncStatusDto?> GetMdpSyncStatusAsync(string jobCode)
  515. {
  516. try
  517. {
  518. return await _db.Ado.SqlQuerySingleAsync<S2SyncStatusDto>(
  519. """
  520. SELECT batch_id AS BatchId, status AS Status, stage_rows AS StageRows,
  521. standard_rows AS StandardRows, dwd_rows AS DwdRows,
  522. start_time AS StartTime, end_time AS EndTime, error_message AS ErrorMessage
  523. FROM mdp_transform_run_log
  524. WHERE job_code=@JobCode
  525. ORDER BY start_time DESC, id DESC
  526. LIMIT 1
  527. """,
  528. new SugarParameter("@JobCode", jobCode));
  529. }
  530. catch
  531. {
  532. return null;
  533. }
  534. }
  535. private async Task<List<S2DecompositionDto>> GetS2DecompositionAsync(long tenantId, long factoryId, List<KpiDetailDto> l2)
  536. {
  537. var latestL1 = new Dictionary<string, KpiDetailDto>(StringComparer.OrdinalIgnoreCase);
  538. try
  539. {
  540. var rows = await _db.Ado.SqlQueryAsync<KpiDetailDto>(
  541. """
  542. SELECT v.module_code AS ModuleCode, v.metric_code AS MetricCode, k.MetricName AS MetricName,
  543. v.metric_value AS MetricValue, v.target_value AS TargetValue,
  544. v.status_color AS StatusColor, v.trend_flag AS TrendFlag, v.biz_date AS StatDate
  545. FROM ado_s9_kpi_value_l1_day v
  546. LEFT JOIN ado_smart_ops_kpi_master k ON k.TenantId=v.tenant_id AND k.MetricCode=v.metric_code
  547. WHERE v.tenant_id=@tenantId AND v.factory_id=@factoryId AND v.module_code='S2' AND v.is_deleted=0
  548. AND v.biz_date=(SELECT MAX(biz_date) FROM ado_s9_kpi_value_l1_day
  549. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND module_code='S2' AND is_deleted=0)
  550. """,
  551. new { tenantId, factoryId });
  552. latestL1 = rows.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  553. .ToDictionary(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase);
  554. }
  555. catch
  556. {
  557. latestL1 = new Dictionary<string, KpiDetailDto>(StringComparer.OrdinalIgnoreCase);
  558. }
  559. var latestL2 = l2.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  560. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  561. .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);
  562. var operationSummary = await GetS2OperationSummaryAsync(tenantId, factoryId);
  563. var resourceSummary = await GetS2ResourceSummaryAsync(tenantId, factoryId);
  564. return new List<S2DecompositionDto>
  565. {
  566. new()
  567. {
  568. Title = "订单排程",
  569. Active = true,
  570. Metrics = new List<string>
  571. {
  572. $"1. 周期:{FormatMetric(latestL1, "S2_L1_001", "天")}",
  573. $"2. 满足率:{FormatMetric(latestL1, "S2_L1_002", "%")}",
  574. $"3. 在制库存:{FormatMetric(latestL1, "S2_L1_004", "天")}",
  575. $"4. 人效:{FormatMetric(latestL1, "S2_L1_003", "单/人")}"
  576. }
  577. },
  578. new()
  579. {
  580. Title = "工单排程",
  581. Metrics = new List<string>
  582. {
  583. $"1. 周期:{FormatMetric(latestL2, "S2_L2_001", "天")}",
  584. $"2. 满足率:{FormatMetric(latestL2, "S2_L2_002", "%")}",
  585. $"3. 人效:{FormatMetric(latestL2, "S2_L2_003", "单/人")}"
  586. }
  587. },
  588. new()
  589. {
  590. Title = "工序排程",
  591. Metrics = new List<string>
  592. {
  593. $"1. 工序数:{operationSummary.OperationCount}",
  594. $"2. 完成量:{Math.Round(operationSummary.CompletedQty ?? 0, 2)}",
  595. $"3. 排程量:{Math.Round(operationSummary.ScheduledQty ?? 0, 2)}"
  596. }
  597. },
  598. new()
  599. {
  600. Title = "资源排程",
  601. Metrics = new List<string>
  602. {
  603. $"1. 人员数:{Math.Round(resourceSummary.PersonCount ?? 0, 2)}",
  604. $"2. 异常数:{resourceSummary.RiskCount}",
  605. $"3. 产线数:{resourceSummary.LineCount}"
  606. }
  607. }
  608. };
  609. }
  610. private async Task<S2OperationSummaryDto> GetS2OperationSummaryAsync(long tenantId, long factoryId)
  611. {
  612. try
  613. {
  614. return await _db.Ado.SqlQuerySingleAsync<S2OperationSummaryDto>(
  615. """
  616. SELECT SUM(operation_count) AS OperationCount,
  617. SUM(completed_op_qty) AS CompletedQty,
  618. SUM(scheduled_qty) AS ScheduledQty
  619. FROM dwd_order_schedule_trans
  620. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  621. AND calc_batch_id=(SELECT calc_batch_id FROM dwd_order_schedule_trans
  622. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  623. ORDER BY calc_time DESC, id DESC LIMIT 1)
  624. """,
  625. new { tenantId, factoryId }) ?? new S2OperationSummaryDto();
  626. }
  627. catch
  628. {
  629. return new S2OperationSummaryDto();
  630. }
  631. }
  632. private async Task<S2ResourceSummaryDto> GetS2ResourceSummaryAsync(long tenantId, long factoryId)
  633. {
  634. try
  635. {
  636. return await _db.Ado.SqlQuerySingleAsync<S2ResourceSummaryDto>(
  637. """
  638. SELECT SUM(resource_person_count) AS PersonCount,
  639. SUM(CASE WHEN schedule_satisfaction_flag=0 THEN 1 ELSE 0 END) AS RiskCount,
  640. COUNT(DISTINCT COALESCE(NULLIF(prod_line,''), NULLIF(site_code,''), '未分配')) AS LineCount
  641. FROM dwd_order_schedule_trans
  642. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  643. AND calc_batch_id=(SELECT calc_batch_id FROM dwd_order_schedule_trans
  644. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  645. ORDER BY calc_time DESC, id DESC LIMIT 1)
  646. """,
  647. new { tenantId, factoryId }) ?? new S2ResourceSummaryDto();
  648. }
  649. catch
  650. {
  651. return new S2ResourceSummaryDto();
  652. }
  653. }
  654. private async Task<List<S2TrendDto>> GetS2TrendAsync(long tenantId, long factoryId)
  655. {
  656. try
  657. {
  658. return await _db.Ado.SqlQueryAsync<S2TrendDto>(
  659. """
  660. SELECT DATE_FORMAT(d.biz_date, '%m-%d') AS DateLabel,
  661. MAX(CASE WHEN d.metric_code='S2_L1_001' THEN d.metric_value END) AS CycleDays,
  662. MAX(CASE WHEN d.metric_code='S2_L1_002' THEN d.metric_value END) AS SatisfactionPct
  663. FROM ado_s9_kpi_value_l1_day d
  664. WHERE d.tenant_id=@tenantId AND d.factory_id=@factoryId AND d.module_code='S2' AND d.is_deleted=0
  665. AND d.metric_code IN ('S2_L1_001','S2_L1_002')
  666. GROUP BY d.biz_date
  667. ORDER BY d.biz_date DESC
  668. LIMIT 7
  669. """,
  670. new { tenantId, factoryId });
  671. }
  672. catch
  673. {
  674. return new List<S2TrendDto>();
  675. }
  676. }
  677. private async Task<List<S2DistributionDto>> GetS2DistributionAsync(long tenantId, long factoryId)
  678. {
  679. try
  680. {
  681. return await _db.Ado.SqlQueryAsync<S2DistributionDto>(
  682. """
  683. SELECT COALESCE(NULLIF(prod_line,''), NULLIF(site_code,''), '未分配') AS Name,
  684. ROUND(100 * SUM(schedule_satisfaction_flag) / NULLIF(COUNT(1), 0), 2) AS SatisfactionPct,
  685. COUNT(1) AS TotalCount,
  686. SUM(CASE WHEN schedule_satisfaction_flag=0 THEN 1 ELSE 0 END) AS RiskCount
  687. FROM dwd_order_schedule_trans
  688. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  689. AND calc_batch_id=(SELECT calc_batch_id FROM dwd_order_schedule_trans
  690. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  691. ORDER BY calc_time DESC, id DESC LIMIT 1)
  692. GROUP BY COALESCE(NULLIF(prod_line,''), NULLIF(site_code,''), '未分配')
  693. ORDER BY SatisfactionPct, TotalCount DESC
  694. LIMIT 8
  695. """,
  696. new { tenantId, factoryId });
  697. }
  698. catch
  699. {
  700. return new List<S2DistributionDto>();
  701. }
  702. }
  703. private async Task<List<S8AlertDto>> GetS2DerivedAlertsAsync(long tenantId, long factoryId)
  704. {
  705. try
  706. {
  707. return await _db.Ado.SqlQueryAsync<S8AlertDto>(
  708. """
  709. SELECT DATE_FORMAT(calc_time, '%H:%i:%s') AS Time,
  710. CASE
  711. WHEN last_plan_date IS NULL THEN 'high'
  712. WHEN due_date IS NOT NULL AND last_plan_date > due_date THEN 'critical'
  713. WHEN IFNULL(resource_person_count, 0) <= 0 THEN 'medium'
  714. ELSE 'info'
  715. END AS LevelCode,
  716. CONCAT('工单 ', work_order, ' ',
  717. CASE
  718. WHEN last_plan_date IS NULL THEN '尚未形成排程'
  719. WHEN due_date IS NOT NULL AND last_plan_date > due_date THEN '排程晚于交期'
  720. WHEN IFNULL(resource_person_count, 0) <= 0 THEN '缺少资源人员配置'
  721. ELSE '排程状态正常'
  722. END) AS Message
  723. FROM dwd_order_schedule_trans
  724. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  725. AND calc_batch_id=(SELECT calc_batch_id FROM dwd_order_schedule_trans
  726. WHERE tenant_id=@tenantId AND factory_id=@factoryId
  727. ORDER BY calc_time DESC, id DESC LIMIT 1)
  728. AND (last_plan_date IS NULL OR (due_date IS NOT NULL AND last_plan_date > due_date) OR IFNULL(resource_person_count, 0) <= 0)
  729. ORDER BY FIELD(LevelCode, 'critical', 'high', 'medium', 'info'), due_date IS NULL, due_date
  730. LIMIT 20
  731. """,
  732. new { tenantId, factoryId });
  733. }
  734. catch
  735. {
  736. return new List<S8AlertDto>();
  737. }
  738. }
  739. private async Task<List<S2DecompositionDto>> GetS3DecompositionAsync(long tenantId, long factoryId, List<KpiDetailDto> l2)
  740. {
  741. var l1 = await GetLatestModuleL1Async("S3", tenantId, factoryId);
  742. var latestL2 = l2.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  743. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  744. .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);
  745. var readiness = await GetS3ReadinessSummaryAsync(tenantId);
  746. var delivery = await GetS3DeliverySummaryAsync(tenantId);
  747. return new List<S2DecompositionDto>
  748. {
  749. new()
  750. {
  751. Title = "物料需求计划",
  752. Active = true,
  753. Metrics = new List<string>
  754. {
  755. $"1. 周期:{FormatMetric(l1, "S3_L1_001", "天")}",
  756. $"2. 满足率:{Math.Round(readiness.ReadyRatePct ?? GetMetricValue(l1, "S3_L1_002") ?? 0, 2)}%",
  757. $"3. 覆盖SKU:{readiness.ComponentCount}",
  758. $"4. 缺口数量:{Math.Round(readiness.ShortageQty ?? 0, 2)}"
  759. }
  760. },
  761. new()
  762. {
  763. Title = "物料交货计划",
  764. Metrics = new List<string>
  765. {
  766. $"1. 周期:{FormatMetric(latestL2, "S3_L2_004", "天")}",
  767. $"2. 满足率:{FormatMetric(latestL2, "S3_L2_005", "%")}",
  768. $"3. 风险供应商:{delivery.RiskSupplierCount}",
  769. $"4. 待交数量:{Math.Round(delivery.RemainingQty ?? 0, 2)}"
  770. }
  771. }
  772. };
  773. }
  774. private async Task<List<S3BranchKpiGroupDto>> GetS3BranchKpisAsync(long tenantId, long factoryId, List<KpiDetailDto> l2)
  775. {
  776. var l1 = await GetLatestModuleL1Async("S3", tenantId, factoryId);
  777. var latestL2 = l2.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  778. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  779. .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);
  780. var readiness = await GetS3ReadinessSummaryAsync(tenantId);
  781. var delivery = await GetS3DeliverySummaryAsync(tenantId);
  782. var demandCycle = GetMetricValue(l1, "S3_L1_001");
  783. var demandReadyRate = readiness.ReadyRatePct ?? GetMetricValue(l1, "S3_L1_002");
  784. var deliveryCycle = GetMetricValue(latestL2, "S3_L2_004");
  785. var deliveryRate = GetMetricValue(latestL2, "S3_L2_005");
  786. return new List<S3BranchKpiGroupDto>
  787. {
  788. new()
  789. {
  790. Branch = "MRP",
  791. Title = "物料需求计划",
  792. Items = new List<S3BranchKpiDto>
  793. {
  794. BuildBranchKpi("需求计划周期", demandCycle, "d", InverseBar(demandCycle, 15m)),
  795. BuildBranchKpi("需求计划满足率", demandReadyRate, "%", PercentBar(demandReadyRate)),
  796. BuildBranchKpi("计划覆盖SKU", readiness.ComponentCount, "SKU", PercentBar(readiness.ComponentCount, Math.Max(readiness.ComponentCount, 1)))
  797. }
  798. },
  799. new()
  800. {
  801. Branch = "MDP",
  802. Title = "物料交货计划",
  803. Items = new List<S3BranchKpiDto>
  804. {
  805. BuildBranchKpi("交货计划周期", deliveryCycle, "d", InverseBar(deliveryCycle, 10.5m)),
  806. BuildBranchKpi("交货计划满足率", deliveryRate, "%", PercentBar(deliveryRate)),
  807. BuildBranchKpi("风险供应商", delivery.RiskSupplierCount, "家", InverseBar(delivery.RiskSupplierCount, Math.Max(delivery.SupplierCount, 1)))
  808. }
  809. }
  810. };
  811. }
  812. private async Task<List<S2TrendDto>> GetS3TrendAsync(long tenantId, long factoryId)
  813. {
  814. try
  815. {
  816. return await _db.Ado.SqlQueryAsync<S2TrendDto>(
  817. """
  818. SELECT DATE_FORMAT(d.biz_date, '%m-%d') AS DateLabel,
  819. MAX(CASE WHEN d.metric_code='S3_L1_001' THEN d.metric_value END) AS CycleDays,
  820. MAX(CASE WHEN d.metric_code='S3_L1_002' THEN d.metric_value END) AS SatisfactionPct
  821. FROM ado_s9_kpi_value_l1_day d
  822. WHERE d.tenant_id=@tenantId AND d.factory_id=@factoryId AND d.module_code='S3' AND d.is_deleted=0
  823. AND d.metric_code IN ('S3_L1_001','S3_L1_002')
  824. GROUP BY d.biz_date
  825. ORDER BY d.biz_date DESC
  826. LIMIT 7
  827. """,
  828. new { tenantId, factoryId });
  829. }
  830. catch
  831. {
  832. return new List<S2TrendDto>();
  833. }
  834. }
  835. private async Task<List<S2DistributionDto>> GetS3DistributionAsync(long tenantId, long factoryId)
  836. {
  837. _ = factoryId;
  838. try
  839. {
  840. return await _db.Ado.SqlQueryAsync<S2DistributionDto>(
  841. """
  842. SELECT COALESCE(NULLIF(supplier_name,''), NULLIF(supplier_code,''), '未分配') AS Name,
  843. ROUND(100 * SUM(CASE WHEN IFNULL(schedule_qty,0) >= IFNULL(order_qty,0) AND IFNULL(order_qty,0) > 0 THEN 1 ELSE 0 END) / NULLIF(COUNT(1), 0), 2) AS SatisfactionPct,
  844. COUNT(1) AS TotalCount,
  845. SUM(CASE WHEN risk_level IN ('HIGH','MEDIUM') THEN 1 ELSE 0 END) AS RiskCount
  846. FROM dwd_supplier_delivery
  847. WHERE tenant_id=@tenantId
  848. AND stat_date=(SELECT MAX(stat_date) FROM dwd_supplier_delivery WHERE tenant_id=@tenantId)
  849. GROUP BY COALESCE(NULLIF(supplier_name,''), NULLIF(supplier_code,''), '未分配')
  850. ORDER BY RiskCount DESC, SatisfactionPct, TotalCount DESC
  851. LIMIT 8
  852. """,
  853. new { tenantId });
  854. }
  855. catch
  856. {
  857. return new List<S2DistributionDto>();
  858. }
  859. }
  860. private async Task<List<S8AlertDto>> GetS3DerivedAlertsAsync(long tenantId, long factoryId)
  861. {
  862. _ = factoryId;
  863. try
  864. {
  865. return await _db.Ado.SqlQueryAsync<S8AlertDto>(
  866. """
  867. SELECT DATE_FORMAT(calc_time, '%H:%i:%s') AS Time,
  868. CASE WHEN risk_level='HIGH' THEN 'critical'
  869. WHEN risk_level='MEDIUM' THEN 'warning'
  870. ELSE 'info' END AS LevelCode,
  871. CONCAT('供应商 ', COALESCE(NULLIF(supplier_name,''), supplier_code, '未分配'),
  872. ' 物料 ', COALESCE(NULLIF(item_code,''), '未指定'),
  873. ':', COALESCE(NULLIF(risk_reason,''), risk_type, '供应风险')) AS Message
  874. FROM dwd_supplier_risk
  875. WHERE tenant_id=@tenantId
  876. AND stat_date=(SELECT MAX(stat_date) FROM dwd_supplier_risk WHERE tenant_id=@tenantId)
  877. ORDER BY FIELD(risk_level, 'HIGH', 'MEDIUM', 'LOW'), risk_count DESC, calc_time DESC
  878. LIMIT 20
  879. """,
  880. new { tenantId });
  881. }
  882. catch
  883. {
  884. return new List<S8AlertDto>();
  885. }
  886. }
  887. private async Task<List<S2TrendDto>> GetS4TrendAsync(long tenantId, long factoryId)
  888. {
  889. try
  890. {
  891. return await _db.Ado.SqlQueryAsync<S2TrendDto>(
  892. """
  893. SELECT DATE_FORMAT(d.biz_date, '%m-%d') AS DateLabel,
  894. MAX(CASE WHEN d.metric_code='S4_L1_001' THEN d.metric_value END) AS CycleDays,
  895. MAX(CASE WHEN d.metric_code='S4_L1_002' THEN d.metric_value END) AS SatisfactionPct
  896. FROM ado_s9_kpi_value_l1_day d
  897. WHERE d.tenant_id=@tenantId AND d.factory_id=@factoryId AND d.module_code='S4' AND d.is_deleted=0
  898. AND d.metric_code IN ('S4_L1_001','S4_L1_002')
  899. GROUP BY d.biz_date
  900. ORDER BY d.biz_date DESC
  901. LIMIT 7
  902. """,
  903. new { tenantId, factoryId });
  904. }
  905. catch
  906. {
  907. return new List<S2TrendDto>();
  908. }
  909. }
  910. private async Task<List<S2DistributionDto>> GetS4DistributionAsync(long tenantId, long factoryId)
  911. {
  912. _ = factoryId;
  913. try
  914. {
  915. return await _db.Ado.SqlQueryAsync<S2DistributionDto>(
  916. """
  917. SELECT COALESCE(NULLIF(supplier_name,''), NULLIF(supplier_code,''), '未分配') AS Name,
  918. ROUND(100 * SUM(CASE WHEN IFNULL(receipt_qty,0) >= IFNULL(order_qty,0) AND IFNULL(order_qty,0) > 0 THEN 1
  919. WHEN delivery_status='COMPLETED' THEN 1 ELSE 0 END) / NULLIF(COUNT(1), 0), 2) AS SatisfactionPct,
  920. COUNT(1) AS TotalCount,
  921. SUM(CASE WHEN risk_level IN ('HIGH','MEDIUM') THEN 1 ELSE 0 END) AS RiskCount
  922. FROM dwd_supplier_delivery
  923. WHERE tenant_id=@tenantId
  924. AND stat_date=(SELECT MAX(stat_date) FROM dwd_supplier_delivery WHERE tenant_id=@tenantId)
  925. GROUP BY COALESCE(NULLIF(supplier_name,''), NULLIF(supplier_code,''), '未分配')
  926. ORDER BY RiskCount DESC, SatisfactionPct, TotalCount DESC
  927. LIMIT 8
  928. """,
  929. new { tenantId });
  930. }
  931. catch
  932. {
  933. return new List<S2DistributionDto>();
  934. }
  935. }
  936. private async Task<List<S8AlertDto>> GetS4DerivedAlertsAsync(long tenantId, long factoryId)
  937. {
  938. _ = factoryId;
  939. try
  940. {
  941. var fromDwd = await _db.Ado.SqlQueryAsync<S8AlertDto>(
  942. """
  943. SELECT DATE_FORMAT(calc_time, '%H:%i:%s') AS Time,
  944. CASE WHEN risk_level='HIGH' THEN 'critical'
  945. WHEN risk_level='MEDIUM' THEN 'warning'
  946. ELSE 'info' END AS LevelCode,
  947. CONCAT('采购订单 ', COALESCE(po_no, '—'), ' 供应商 ', COALESCE(NULLIF(supplier_name,''), supplier_code, '未分配'),
  948. ':', COALESCE(delivery_status, 'OPEN'), ' 剩余 ', CAST(IFNULL(remaining_qty,0) AS CHAR)) AS Message
  949. FROM dwd_supplier_delivery
  950. WHERE tenant_id=@tenantId
  951. AND stat_date=(SELECT MAX(stat_date) FROM dwd_supplier_delivery WHERE tenant_id=@tenantId)
  952. AND risk_level IN ('HIGH','MEDIUM')
  953. ORDER BY FIELD(risk_level, 'HIGH', 'MEDIUM'), remaining_qty DESC
  954. LIMIT 15
  955. """,
  956. new { tenantId });
  957. if (fromDwd.Count > 0)
  958. return fromDwd;
  959. }
  960. catch
  961. {
  962. // fall through to S8 alerts
  963. }
  964. try
  965. {
  966. return await _db.Ado.SqlQueryAsync<S8AlertDto>(
  967. """
  968. SELECT DATE_FORMAT(alert_time, '%H:%i:%s') AS Time,
  969. IFNULL(level_code, 'medium') AS LevelCode,
  970. IFNULL(message, '采购执行异常') AS Message
  971. FROM ado_s8_alert_record
  972. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND is_deleted=0
  973. AND (UPPER(IFNULL(module_code,''))='S4' OR metric_code LIKE 'S4%')
  974. ORDER BY alert_time DESC
  975. LIMIT 20
  976. """,
  977. new { tenantId, factoryId });
  978. }
  979. catch
  980. {
  981. return new List<S8AlertDto>();
  982. }
  983. }
  984. private async Task<Dictionary<string, KpiDetailDto>> GetLatestModuleL1Async(string moduleCode, long tenantId, long factoryId)
  985. {
  986. try
  987. {
  988. var rows = await _db.Ado.SqlQueryAsync<KpiDetailDto>(
  989. """
  990. SELECT v.module_code AS ModuleCode, v.metric_code AS MetricCode, k.MetricName AS MetricName,
  991. v.metric_value AS MetricValue, v.target_value AS TargetValue,
  992. v.status_color AS StatusColor, v.trend_flag AS TrendFlag, v.biz_date AS StatDate
  993. FROM ado_s9_kpi_value_l1_day v
  994. LEFT JOIN ado_smart_ops_kpi_master k ON k.TenantId=v.tenant_id AND k.MetricCode=v.metric_code
  995. WHERE v.tenant_id=@tenantId AND v.factory_id=@factoryId AND v.module_code=@moduleCode AND v.is_deleted=0
  996. AND v.biz_date=(SELECT MAX(biz_date) FROM ado_s9_kpi_value_l1_day
  997. WHERE tenant_id=@tenantId AND factory_id=@factoryId AND module_code=@moduleCode AND is_deleted=0)
  998. """,
  999. new { moduleCode, tenantId, factoryId });
  1000. return rows.Where(u => !string.IsNullOrWhiteSpace(u.MetricCode))
  1001. .GroupBy(u => u.MetricCode!, StringComparer.OrdinalIgnoreCase)
  1002. .ToDictionary(g => g.Key, g => g.First(), StringComparer.OrdinalIgnoreCase);
  1003. }
  1004. catch
  1005. {
  1006. return new Dictionary<string, KpiDetailDto>(StringComparer.OrdinalIgnoreCase);
  1007. }
  1008. }
  1009. private async Task<S3ReadinessSummaryDto> GetS3ReadinessSummaryAsync(long tenantId)
  1010. {
  1011. try
  1012. {
  1013. return await _db.Ado.SqlQuerySingleAsync<S3ReadinessSummaryDto>(
  1014. """
  1015. SELECT COUNT(DISTINCT component_item_code) AS ComponentCount,
  1016. ROUND(100 * SUM(CASE WHEN IFNULL(shortage_qty,0) <= 0 OR UPPER(IFNULL(ready_status,'')) IN ('READY','SUFFICIENT','OK') THEN 1 ELSE 0 END) / NULLIF(COUNT(1), 0), 2) AS ReadyRatePct,
  1017. SUM(IFNULL(shortage_qty, 0)) AS ShortageQty
  1018. FROM dwd_material_readiness
  1019. WHERE tenant_id=@tenantId
  1020. AND stat_date=(SELECT MAX(stat_date) FROM dwd_material_readiness WHERE tenant_id=@tenantId)
  1021. """,
  1022. new { tenantId }) ?? new S3ReadinessSummaryDto();
  1023. }
  1024. catch
  1025. {
  1026. return new S3ReadinessSummaryDto();
  1027. }
  1028. }
  1029. private async Task<S3DeliverySummaryDto> GetS3DeliverySummaryAsync(long tenantId)
  1030. {
  1031. try
  1032. {
  1033. return await _db.Ado.SqlQuerySingleAsync<S3DeliverySummaryDto>(
  1034. """
  1035. SELECT COUNT(DISTINCT supplier_code) AS SupplierCount,
  1036. COUNT(DISTINCT CASE WHEN risk_level IN ('HIGH','MEDIUM') THEN supplier_code END) AS RiskSupplierCount,
  1037. SUM(IFNULL(remaining_qty, 0)) AS RemainingQty
  1038. FROM dwd_supplier_delivery
  1039. WHERE tenant_id=@tenantId
  1040. AND stat_date=(SELECT MAX(stat_date) FROM dwd_supplier_delivery WHERE tenant_id=@tenantId)
  1041. """,
  1042. new { tenantId }) ?? new S3DeliverySummaryDto();
  1043. }
  1044. catch
  1045. {
  1046. return new S3DeliverySummaryDto();
  1047. }
  1048. }
  1049. private static string FormatMetric(Dictionary<string, KpiDetailDto> rows, string metricCode, string unit)
  1050. {
  1051. if (!rows.TryGetValue(metricCode, out var row) || row.MetricValue == null)
  1052. return $"--{unit}";
  1053. return $"{Math.Round(row.MetricValue.Value, 2)}{unit}";
  1054. }
  1055. private static decimal? GetMetricValue(Dictionary<string, KpiDetailDto> rows, string metricCode)
  1056. {
  1057. return rows.TryGetValue(metricCode, out var row) ? row.MetricValue : null;
  1058. }
  1059. private static S3BranchKpiDto BuildBranchKpi(string label, decimal? value, string unit, decimal barPct)
  1060. {
  1061. return new S3BranchKpiDto
  1062. {
  1063. Label = label,
  1064. Value = value,
  1065. Unit = unit,
  1066. BarPct = barPct,
  1067. Status = barPct >= 90 ? "success" : barPct >= 70 ? "warning" : "danger"
  1068. };
  1069. }
  1070. private static decimal PercentBar(decimal? value, decimal max = 100m)
  1071. {
  1072. if (value == null || max <= 0)
  1073. return 0;
  1074. return Math.Round(Math.Max(0, Math.Min(100, value.Value / max * 100)), 2);
  1075. }
  1076. private static decimal InverseBar(decimal? value, decimal target)
  1077. {
  1078. if (value == null || value <= 0 || target <= 0)
  1079. return 0;
  1080. return Math.Round(Math.Max(0, Math.Min(100, target / value.Value * 100)), 2);
  1081. }
  1082. /// <summary>
  1083. /// 智慧运营看板基础查询下拉:产品、订单号、产线(兼容旧接口;优先 S0 主数据)。
  1084. /// </summary>
  1085. [HttpGet("smart-ops-filter-options")]
  1086. public async Task<IActionResult> GetSmartOpsFilterOptions([FromQuery] long factoryId = 1)
  1087. {
  1088. _ = factoryId;
  1089. var tenantId = AidopTenantHelper.GetTenantId(HttpContext);
  1090. const int cap = 200;
  1091. var productItems = await SearchProductsAsync(tenantId, null, cap);
  1092. var orderItems = await SearchOrderNosAsync(tenantId, null, cap);
  1093. var lineItems = await SearchProductionLinesAsync(tenantId, null, cap);
  1094. return Ok(new
  1095. {
  1096. products = productItems.Select(x => x.Value).Distinct(StringComparer.Ordinal).OrderBy(x => x, StringComparer.Ordinal).ToList(),
  1097. orderNos = orderItems.Select(x => x.Value).Distinct(StringComparer.Ordinal).OrderBy(x => x, StringComparer.Ordinal).ToList(),
  1098. productionLines = lineItems.Select(x => x.Value).Distinct(StringComparer.Ordinal).OrderBy(x => x, StringComparer.Ordinal).ToList()
  1099. });
  1100. }
  1101. private sealed class SmartOpsOptionRow
  1102. {
  1103. public string? Value { get; set; }
  1104. }
  1105. private sealed class HomeL1Dto
  1106. {
  1107. public string? ModuleCode { get; set; }
  1108. public string? MetricCode { get; set; }
  1109. public decimal? MetricValue { get; set; }
  1110. public decimal? TargetValue { get; set; }
  1111. public string? StatusColor { get; set; }
  1112. public string? TrendFlag { get; set; }
  1113. }
  1114. private sealed class HomeL1RawDto
  1115. {
  1116. public string? ModuleCode { get; set; }
  1117. public string? MetricCode { get; set; }
  1118. public decimal? MetricValue { get; set; }
  1119. public decimal? TargetValue { get; set; }
  1120. public string? StatusColor { get; set; }
  1121. public string? TrendFlag { get; set; }
  1122. public string? Direction { get; set; }
  1123. public decimal? YellowThreshold { get; set; }
  1124. public decimal? RedThreshold { get; set; }
  1125. }
  1126. private sealed class S8AlertDto
  1127. {
  1128. public string? Time { get; set; }
  1129. public string? Module { get; set; }
  1130. public string? Message { get; set; }
  1131. public string? LevelCode { get; set; }
  1132. }
  1133. private sealed class KpiDetailDto
  1134. {
  1135. public string? ModuleCode { get; set; }
  1136. public string? MetricCode { get; set; }
  1137. public string? MetricName { get; set; }
  1138. public decimal? MetricValue { get; set; }
  1139. public decimal? TargetValue { get; set; }
  1140. public string? StatusColor { get; set; }
  1141. public string? TrendFlag { get; set; }
  1142. public DateTime? StatDate { get; set; }
  1143. public string? ParentMetricCode { get; set; }
  1144. }
  1145. private sealed class KpiDrillGroupDto
  1146. {
  1147. public string? ParentMetricCode { get; set; }
  1148. public string Title { get; set; } = string.Empty;
  1149. public List<KpiDrillItemDto> Items { get; set; } = new();
  1150. }
  1151. private sealed class KpiDrillItemDto
  1152. {
  1153. public string? MetricCode { get; set; }
  1154. public string? MetricName { get; set; }
  1155. public decimal? MetricValue { get; set; }
  1156. public decimal? TargetValue { get; set; }
  1157. public string? StatusColor { get; set; }
  1158. public string? TrendFlag { get; set; }
  1159. public string? StatDate { get; set; }
  1160. }
  1161. private sealed class S2ScheduleDto
  1162. {
  1163. public string? OrderNo { get; set; }
  1164. public string? Customer { get; set; }
  1165. public string? Product { get; set; }
  1166. public string? ProductionLine { get; set; }
  1167. public decimal? Quantity { get; set; }
  1168. public DateTime? DeliveryDate { get; set; }
  1169. public DateTime? StartTime { get; set; }
  1170. public decimal? CycleDays { get; set; }
  1171. public decimal? Satisfaction { get; set; }
  1172. public string? Status { get; set; }
  1173. }
  1174. private sealed class S2SyncStatusDto
  1175. {
  1176. public string? BatchId { get; set; }
  1177. public string? Status { get; set; }
  1178. public int? StageRows { get; set; }
  1179. public int? StandardRows { get; set; }
  1180. public int? DwdRows { get; set; }
  1181. public DateTime? StartTime { get; set; }
  1182. public DateTime? EndTime { get; set; }
  1183. public string? ErrorMessage { get; set; }
  1184. }
  1185. private sealed class S2DecompositionDto
  1186. {
  1187. public string Title { get; set; } = string.Empty;
  1188. public bool Active { get; set; }
  1189. public List<string> Metrics { get; set; } = new();
  1190. }
  1191. private sealed class S2TrendDto
  1192. {
  1193. public string? DateLabel { get; set; }
  1194. public decimal? CycleDays { get; set; }
  1195. public decimal? SatisfactionPct { get; set; }
  1196. }
  1197. private sealed class S2DistributionDto
  1198. {
  1199. public string? Name { get; set; }
  1200. public decimal? SatisfactionPct { get; set; }
  1201. public int TotalCount { get; set; }
  1202. public int RiskCount { get; set; }
  1203. }
  1204. private sealed class S2OperationSummaryDto
  1205. {
  1206. public int OperationCount { get; set; }
  1207. public decimal? CompletedQty { get; set; }
  1208. public decimal? ScheduledQty { get; set; }
  1209. }
  1210. private sealed class S2ResourceSummaryDto
  1211. {
  1212. public decimal? PersonCount { get; set; }
  1213. public int RiskCount { get; set; }
  1214. public int LineCount { get; set; }
  1215. }
  1216. private sealed class S3BranchKpiGroupDto
  1217. {
  1218. public string Branch { get; set; } = string.Empty;
  1219. public string Title { get; set; } = string.Empty;
  1220. public List<S3BranchKpiDto> Items { get; set; } = new();
  1221. }
  1222. private sealed class S3BranchKpiDto
  1223. {
  1224. public string Label { get; set; } = string.Empty;
  1225. public decimal? Value { get; set; }
  1226. public string Unit { get; set; } = string.Empty;
  1227. public decimal BarPct { get; set; }
  1228. public string Status { get; set; } = "info";
  1229. }
  1230. private sealed class S3ReadinessSummaryDto
  1231. {
  1232. public int ComponentCount { get; set; }
  1233. public decimal? ReadyRatePct { get; set; }
  1234. public decimal? ShortageQty { get; set; }
  1235. }
  1236. private sealed class S3DeliverySummaryDto
  1237. {
  1238. public int SupplierCount { get; set; }
  1239. public int RiskSupplierCount { get; set; }
  1240. public decimal? RemainingQty { get; set; }
  1241. }
  1242. }