ProductionScheduleGenerationService.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486
  1. namespace Admin.NET.Plugin.AiDOP.Production;
  2. using Admin.NET.Plugin.AiDOP.WorkOrder;
  3. /// <summary>生产排程生成:为待排工单写入 PeriodSequenceDet(工作日历 + 工作中心冲突避让)。</summary>
  4. public class ProductionScheduleGenerationService : ITransient
  5. {
  6. private readonly ISqlSugarClient _db;
  7. private readonly WorkOrderKittingCheckService _kittingCheck;
  8. public ProductionScheduleGenerationService(ISqlSugarClient db, WorkOrderKittingCheckService kittingCheck)
  9. {
  10. _db = db;
  11. _kittingCheck = kittingCheck;
  12. }
  13. public async Task<ScheduleGenerationResult> GenerateAsync(long tenantId, string? domain, string account)
  14. {
  15. var workOrders = await LoadPendingWorkOrdersAsync(tenantId);
  16. if (workOrders.Count == 0)
  17. return new ScheduleGenerationResult { Message = "没有待排产的工单(状态 p/r)" };
  18. return await ScheduleWorkOrdersAsync(tenantId, domain, account, workOrders);
  19. }
  20. public async Task<ScheduleGenerationResult> RegenerateForWorkOrderAsync(
  21. long tenantId,
  22. string workOrd,
  23. string? domain,
  24. string account)
  25. {
  26. var rows = await _db.Ado.SqlQueryAsync<PendingWorkOrderRow>(
  27. """
  28. SELECT RecID AS RecId, WorkOrd, ItemNum, `Domain`, QtyOrded, OrdDate, DueDate, Priority, Urgent
  29. FROM WorkOrdMaster
  30. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  31. LIMIT 1
  32. """,
  33. new SugarParameter("@TenantId", tenantId),
  34. new SugarParameter("@WorkOrd", workOrd.Trim()));
  35. var wo = rows.FirstOrDefault();
  36. if (wo is null)
  37. return new ScheduleGenerationResult { Message = $"工单 {workOrd} 不存在" };
  38. return await ScheduleWorkOrdersAsync(tenantId, domain, account, new List<PendingWorkOrderRow> { wo });
  39. }
  40. private async Task<ScheduleGenerationResult> ScheduleWorkOrdersAsync(
  41. long tenantId,
  42. string? domain,
  43. string account,
  44. List<PendingWorkOrderRow> workOrders)
  45. {
  46. var now = DateTime.Now;
  47. var calendar = await LoadWorkCenterCalendarAsync(tenantId);
  48. var occupiedSlots = await LoadOccupiedSlotsAsync(tenantId);
  49. var usedCalendar = calendar.Count > 0;
  50. var scheduledCount = 0;
  51. var rowCount = 0;
  52. var skipped = new List<string>();
  53. // 按优先级顺序跟踪各物料已占用库存量(ItemNumber → 已占用数量)
  54. var consumedStock = new Dictionary<string, decimal>(StringComparer.OrdinalIgnoreCase);
  55. // 清理旧的占用记录,确保每次排程重新计算
  56. const long scheduleBangId = 1;
  57. await _db.Ado.ExecuteCommandAsync(
  58. "DELETE FROM ic_item_stockoccupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  59. new SugarParameter("@TenantId", tenantId),
  60. new SugarParameter("@BangId", scheduleBangId));
  61. await _db.Ado.ExecuteCommandAsync(
  62. "DELETE FROM srm_po_occupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  63. new SugarParameter("@TenantId", tenantId),
  64. new SugarParameter("@BangId", scheduleBangId));
  65. foreach (var wo in workOrders)
  66. {
  67. // 1. 执行齐套检查,刷新资源检查记录(写入 b_examine_result / b_bom_child_examine)
  68. try
  69. {
  70. await _kittingCheck.CheckSingleAsync(tenantId, wo.WorkOrd, account, scheduleBangId);
  71. }
  72. catch (Exception ex)
  73. {
  74. skipped.Add($"齐套检查失败[{wo.WorkOrd}]: {ex.Message}");
  75. }
  76. // 2. 刷新齐套数量(LocationStock),扣减已被高优先级工单占用的库存
  77. var locationStock = await CalcLocationStockAsync(tenantId, wo.WorkOrd, wo.QtyOrded ?? 0, consumedStock);
  78. if (locationStock.HasValue)
  79. {
  80. await _db.Ado.ExecuteCommandAsync(
  81. """
  82. UPDATE WorkOrdMaster
  83. SET LocationStock = @Stock, UpdateTime = @Now
  84. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  85. """,
  86. new SugarParameter("@Stock", locationStock.Value),
  87. new SugarParameter("@Now", now),
  88. new SugarParameter("@TenantId", tenantId),
  89. new SugarParameter("@WorkOrd", wo.WorkOrd));
  90. }
  91. var routings = await LoadRoutingsAsync(tenantId, wo.WorkOrd);
  92. if (routings.Count == 0)
  93. {
  94. skipped.Add($"{wo.WorkOrd}(无工艺路线)");
  95. continue;
  96. }
  97. var woDomain = ResolveDomain(wo.Domain, domain, tenantId);
  98. await DeactivateExistingScheduleAsync(tenantId, wo.WorkOrd, woDomain);
  99. var planStart = (wo.OrdDate ?? now).Date;
  100. var planEnd = (wo.DueDate ?? planStart.AddDays(Math.Max(routings.Count, 7))).Date;
  101. if (planEnd < planStart)
  102. planEnd = planStart;
  103. var cursor = planStart;
  104. var seq = 1;
  105. for (var i = 0; i < routings.Count; i++)
  106. {
  107. var routing = routings[i];
  108. DateTime planDate;
  109. if (usedCalendar)
  110. {
  111. planDate = ResolveNextAvailablePlanDate(cursor, routing.WorkCtr, calendar, occupiedSlots);
  112. cursor = planDate.AddDays(1);
  113. }
  114. else
  115. {
  116. var spanDays = Math.Max((planEnd - planStart).Days, 0);
  117. planDate = routings.Count <= 1
  118. ? planStart
  119. : planStart.AddDays(spanDays * i / Math.Max(routings.Count - 1, 1));
  120. }
  121. if (planDate > planEnd)
  122. planEnd = planDate;
  123. var slotKey = BuildOccupancyKey(routing.WorkCtr, planDate);
  124. occupiedSlots.Add(slotKey);
  125. var recId = await NextPeriodRecIdAsync();
  126. await InsertScheduleRowAsync(
  127. recId, woDomain, routing, wo, planDate, seq, account, now, tenantId);
  128. seq++;
  129. rowCount++;
  130. }
  131. scheduledCount++;
  132. }
  133. return new ScheduleGenerationResult
  134. {
  135. WorkOrderCount = scheduledCount,
  136. ScheduleRowCount = rowCount,
  137. SkippedWorkOrders = skipped,
  138. UsedWorkCenterCalendar = usedCalendar,
  139. Message = scheduledCount > 0
  140. ? $"已为 {scheduledCount} 个工单生成 {rowCount} 条工序排程"
  141. : "未生成排程,请确认工单已同步工艺路线"
  142. };
  143. }
  144. private async Task InsertScheduleRowAsync(
  145. int recId,
  146. string woDomain,
  147. RoutingRow routing,
  148. PendingWorkOrderRow wo,
  149. DateTime planDate,
  150. int seq,
  151. string account,
  152. DateTime now,
  153. long tenantId)
  154. {
  155. await _db.Ado.ExecuteCommandAsync(
  156. """
  157. INSERT INTO PeriodSequenceDet (
  158. RecID, `Domain`, Site, ItemNum, Line, Op, WorkCtr, ProdDate, PlanDate,
  159. Sequence, OrdQty, CompQty, WorkOrds, Status, Employee,
  160. CreateUser, CreateTime, UpdateUser, UpdateTime,
  161. IsActive, IsConfirm, BusinessID, tenant_id
  162. ) VALUES (
  163. @RecId, @Domain, @Site, @ItemNum, @Line, @Op, @WorkCtr, @ProdDate, @PlanDate,
  164. @Sequence, @OrdQty, 0, @WorkOrd, '', '',
  165. @User, @Now, @User, @Now,
  166. 1, 0, @BusinessId, @TenantId
  167. )
  168. """,
  169. new SugarParameter("@RecId", recId),
  170. new SugarParameter("@Domain", woDomain.Length > 8 ? woDomain[..8] : woDomain),
  171. new SugarParameter("@Site", routing.Site ?? (object)DBNull.Value),
  172. new SugarParameter("@ItemNum", wo.ItemNum ?? string.Empty),
  173. new SugarParameter("@Line", routing.ProdLine ?? (object)DBNull.Value),
  174. new SugarParameter("@Op", routing.Op),
  175. new SugarParameter("@WorkCtr", routing.WorkCtr ?? (object)DBNull.Value),
  176. new SugarParameter("@ProdDate", planDate),
  177. new SugarParameter("@PlanDate", planDate),
  178. new SugarParameter("@Sequence", seq),
  179. new SugarParameter("@OrdQty", wo.QtyOrded ?? 0),
  180. new SugarParameter("@WorkOrd", wo.WorkOrd),
  181. new SugarParameter("@BusinessId", wo.RecId),
  182. new SugarParameter("@User", account.Length > 24 ? account[..24] : account),
  183. new SugarParameter("@Now", now),
  184. new SugarParameter("@TenantId", tenantId));
  185. }
  186. private static DateTime ResolveNextAvailablePlanDate(
  187. DateTime start,
  188. string? workCtr,
  189. Dictionary<string, HashSet<int>> calendar,
  190. HashSet<string> occupiedSlots)
  191. {
  192. var date = start.Date;
  193. for (var guard = 0; guard < 366; guard++)
  194. {
  195. if (IsWorkDay(date, workCtr, calendar))
  196. {
  197. var key = BuildOccupancyKey(workCtr, date);
  198. if (!occupiedSlots.Contains(key))
  199. return date;
  200. }
  201. date = date.AddDays(1);
  202. }
  203. return start.Date;
  204. }
  205. private static bool IsWorkDay(DateTime date, string? workCtr, Dictionary<string, HashSet<int>> calendar)
  206. {
  207. if (calendar.Count == 0)
  208. return true;
  209. var key = NormalizeWorkCtr(workCtr);
  210. if (!calendar.TryGetValue(key, out var days) || days.Count == 0)
  211. return date.DayOfWeek is not DayOfWeek.Saturday and not DayOfWeek.Sunday;
  212. var weekDay = (int)date.DayOfWeek;
  213. return days.Contains(weekDay);
  214. }
  215. private static string BuildOccupancyKey(string? workCtr, DateTime date) =>
  216. $"{NormalizeWorkCtr(workCtr)}|{date:yyyy-MM-dd}";
  217. private static string NormalizeWorkCtr(string? workCtr) =>
  218. string.IsNullOrWhiteSpace(workCtr) ? "*" : workCtr.Trim();
  219. private async Task<Dictionary<string, HashSet<int>>> LoadWorkCenterCalendarAsync(long tenantId)
  220. {
  221. var rows = await _db.Ado.SqlQueryAsync<CalendarRow>(
  222. """
  223. SELECT TRIM(IFNULL(WorkCtr,'')) AS WorkCtr, WeekDay
  224. FROM ShopCalendarWorkCtr
  225. WHERE tenant_id = @TenantId
  226. AND IFNULL(IsActive, 0) = 1
  227. AND IFNULL(IsWorkDay, 0) = 1
  228. AND WeekDay IS NOT NULL
  229. """,
  230. new SugarParameter("@TenantId", tenantId));
  231. var map = new Dictionary<string, HashSet<int>>(StringComparer.OrdinalIgnoreCase);
  232. foreach (var row in rows)
  233. {
  234. if (row.WeekDay is null) continue;
  235. var key = NormalizeWorkCtr(row.WorkCtr);
  236. if (!map.TryGetValue(key, out var set))
  237. {
  238. set = new HashSet<int>();
  239. map[key] = set;
  240. }
  241. set.Add(row.WeekDay.Value);
  242. }
  243. return map;
  244. }
  245. private async Task<HashSet<string>> LoadOccupiedSlotsAsync(long tenantId)
  246. {
  247. var rows = await _db.Ado.SqlQueryAsync<OccupiedSlotRow>(
  248. """
  249. SELECT TRIM(IFNULL(WorkCtr,'')) AS WorkCtr, DATE(PlanDate) AS PlanDate
  250. FROM PeriodSequenceDet
  251. WHERE tenant_id = @TenantId AND IFNULL(IsActive, 0) = 1 AND PlanDate IS NOT NULL
  252. """,
  253. new SugarParameter("@TenantId", tenantId));
  254. return rows
  255. .Where(x => x.PlanDate.HasValue)
  256. .Select(x => BuildOccupancyKey(x.WorkCtr, x.PlanDate!.Value))
  257. .ToHashSet(StringComparer.OrdinalIgnoreCase);
  258. }
  259. private async Task<List<PendingWorkOrderRow>> LoadPendingWorkOrdersAsync(long tenantId)
  260. {
  261. return await _db.Ado.SqlQueryAsync<PendingWorkOrderRow>(
  262. """
  263. SELECT RecID AS RecId, WorkOrd, ItemNum, `Domain`, QtyOrded, OrdDate, DueDate, Priority, Urgent
  264. FROM WorkOrdMaster
  265. WHERE tenant_id = @TenantId
  266. AND LOWER(TRIM(IFNULL(Status,''))) IN ('p', 'r')
  267. ORDER BY IFNULL(Urgent, 0) DESC, IFNULL(Priority, 0) DESC, DueDate, WorkOrd
  268. """,
  269. new SugarParameter("@TenantId", tenantId));
  270. }
  271. private async Task<List<RoutingRow>> LoadRoutingsAsync(long tenantId, string workOrd)
  272. {
  273. return await _db.Ado.SqlQueryAsync<RoutingRow>(
  274. """
  275. SELECT OP AS Op, ProdLine, WorkCtr, WorkCtr AS Site
  276. FROM WorkOrdRouting
  277. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd AND IFNULL(IsActive, 0) = 1
  278. ORDER BY (OP + 0), OP
  279. """,
  280. new SugarParameter("@TenantId", tenantId),
  281. new SugarParameter("@WorkOrd", workOrd));
  282. }
  283. public async Task DeactivateExistingScheduleAsync(long tenantId, string workOrd, string domain)
  284. {
  285. await _db.Ado.ExecuteCommandAsync(
  286. """
  287. UPDATE PeriodSequenceDet
  288. SET IsActive = 0, UpdateTime = @Now
  289. WHERE tenant_id = @TenantId AND WorkOrds = @WorkOrd AND `Domain` = @Domain AND IFNULL(IsActive, 0) = 1
  290. """,
  291. new SugarParameter("@Now", DateTime.Now),
  292. new SugarParameter("@TenantId", tenantId),
  293. new SugarParameter("@WorkOrd", workOrd),
  294. new SugarParameter("@Domain", domain.Length > 8 ? domain[..8] : domain));
  295. }
  296. private static string ResolveDomain(string? woDomain, string? requestDomain, long tenantId)
  297. {
  298. if (!string.IsNullOrWhiteSpace(woDomain))
  299. return woDomain.Trim();
  300. if (!string.IsNullOrWhiteSpace(requestDomain))
  301. return requestDomain.Trim();
  302. return tenantId.ToString();
  303. }
  304. private async Task<int> NextPeriodRecIdAsync()
  305. {
  306. var max = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) FROM PeriodSequenceDet");
  307. return max + 1;
  308. }
  309. public sealed class ScheduleGenerationResult
  310. {
  311. public int WorkOrderCount { get; set; }
  312. public int ScheduleRowCount { get; set; }
  313. public bool UsedWorkCenterCalendar { get; set; }
  314. public List<string> SkippedWorkOrders { get; set; } = new();
  315. public string Message { get; set; } = string.Empty;
  316. }
  317. /// <summary>
  318. /// 计算工单齐套数量:MIN(可用库存 / qty) 得可生产成品数,超过工单需求量则取工单需求量。
  319. /// 若工单领料单已全部发完料,则齐套数量 = 工单需求量。
  320. /// consumedStock 跟踪已被高优先级工单占用的库存量,避免重复占用。
  321. /// </summary>
  322. private async Task<decimal?> CalcLocationStockAsync(
  323. long tenantId, string workOrd, decimal qtyOrded,
  324. Dictionary<string, decimal> consumedStock)
  325. {
  326. // 检查领料单是否已全部发完料
  327. var allIssued = await _db.Ado.GetIntAsync(
  328. """
  329. SELECT CASE WHEN COUNT(*) > 0 THEN 1 ELSE 0 END
  330. FROM NbrMaster nm
  331. WHERE nm.tenant_id = @TenantId AND nm.WorkOrd = @WorkOrd
  332. AND nm.Type = 'SM' AND IFNULL(nm.IsActive, 0) = 1
  333. AND NOT EXISTS (
  334. SELECT 1 FROM NbrDetail nd
  335. WHERE nd.tenant_id = nm.tenant_id AND nd.Nbr = nm.Nbr
  336. AND nd.Type = 'SM' AND IFNULL(nd.IsActive, 0) = 1
  337. AND IFNULL(nd.QtyRec, 0) < IFNULL(nd.QtyOrd, 0)
  338. )
  339. """,
  340. new SugarParameter("@TenantId", tenantId),
  341. new SugarParameter("@WorkOrd", workOrd));
  342. if (allIssued > 0 && qtyOrded > 0)
  343. return qtyOrded;
  344. // 从最新资源检查获取各物料的 use_qty 和 qty
  345. var materials = await _db.Ado.SqlQueryAsync<LocationStockMaterialRow>(
  346. """
  347. SELECT
  348. bce.item_number AS ItemNumber,
  349. IFNULL(bce.use_qty, 0) AS UseQty,
  350. IFNULL(bce.qty, 0) AS Qty
  351. FROM b_examine_result ber
  352. INNER JOIN b_bom_child_examine bce ON ber.Id = bce.examine_id AND bce.is_use = 1
  353. WHERE ber.tenant_id = @TenantId
  354. AND ber.IsDeleted = 0
  355. AND ber.morder_no = @WorkOrd
  356. AND ber.Id = (
  357. SELECT br.Id FROM b_examine_result br
  358. WHERE br.tenant_id = @TenantId AND br.morder_no = @WorkOrd AND br.IsDeleted = 0
  359. ORDER BY br.create_time DESC LIMIT 1
  360. )
  361. AND IFNULL(bce.qty, 0) > 0
  362. AND IFNULL(bce.erp_cls, 3) = 3
  363. """,
  364. new SugarParameter("@TenantId", tenantId),
  365. new SugarParameter("@WorkOrd", workOrd));
  366. if (materials.Count == 0)
  367. return null;
  368. // MIN(可用库存 / qty) = 扣减已占用后的瓶颈物料可生产成品数
  369. decimal? minProducible = null;
  370. foreach (var mat in materials)
  371. {
  372. if (mat.Qty <= 0) continue;
  373. var alreadyConsumed = consumedStock.TryGetValue(mat.ItemNumber, out var c) ? c : 0m;
  374. var availableUseQty = Math.Max(0m, mat.UseQty - alreadyConsumed);
  375. var producible = availableUseQty / mat.Qty;
  376. if (minProducible is null || producible < minProducible)
  377. minProducible = producible;
  378. }
  379. if (minProducible is null)
  380. return null;
  381. // 可生产数 > 工单需求量则取工单需求量
  382. var result = Math.Ceiling(minProducible.Value);
  383. if (qtyOrded > 0 && result > qtyOrded)
  384. result = qtyOrded;
  385. // 记录本工单占用的库存量
  386. foreach (var mat in materials)
  387. {
  388. if (mat.Qty <= 0) continue;
  389. var alreadyConsumed = consumedStock.TryGetValue(mat.ItemNumber, out var c) ? c : 0m;
  390. var availableUseQty = Math.Max(0m, mat.UseQty - alreadyConsumed);
  391. var consume = Math.Min(availableUseQty, result * mat.Qty);
  392. if (!consumedStock.ContainsKey(mat.ItemNumber))
  393. consumedStock[mat.ItemNumber] = 0m;
  394. consumedStock[mat.ItemNumber] += consume;
  395. }
  396. return result;
  397. }
  398. private sealed class LocationStockMaterialRow
  399. {
  400. public string ItemNumber { get; set; } = string.Empty;
  401. public decimal UseQty { get; set; }
  402. public decimal Qty { get; set; }
  403. }
  404. private sealed class PendingWorkOrderRow
  405. {
  406. public long RecId { get; set; }
  407. public string WorkOrd { get; set; } = string.Empty;
  408. public string? ItemNum { get; set; }
  409. public string? Domain { get; set; }
  410. public decimal? QtyOrded { get; set; }
  411. public DateTime? OrdDate { get; set; }
  412. public DateTime? DueDate { get; set; }
  413. public decimal? Priority { get; set; }
  414. public int? Urgent { get; set; }
  415. }
  416. private sealed class RoutingRow
  417. {
  418. public int Op { get; set; }
  419. public string? ProdLine { get; set; }
  420. public string? WorkCtr { get; set; }
  421. public string? Site { get; set; }
  422. }
  423. private sealed class CalendarRow
  424. {
  425. public string? WorkCtr { get; set; }
  426. public int? WeekDay { get; set; }
  427. }
  428. private sealed class OccupiedSlotRow
  429. {
  430. public string? WorkCtr { get; set; }
  431. public DateTime? PlanDate { get; set; }
  432. }
  433. }