ProductionSchedulingActionService.cs 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683
  1. using System.Globalization;
  2. using Admin.NET.Plugin.AiDOP.Infrastructure;
  3. using Admin.NET.Plugin.AiDOP.Order;
  4. using Admin.NET.Plugin.AiDOP.WorkOrder;
  5. namespace Admin.NET.Plugin.AiDOP.Production;
  6. /// <summary>工单排产动作:生成排程、优先级调整并重检。</summary>
  7. [ApiDescriptionSettings(Order = 264, Description = "工单排产动作")]
  8. [Route("api/Production")]
  9. [AllowAnonymous]
  10. [NonUnify]
  11. public class ProductionSchedulingActionService : IDynamicApiController, ITransient
  12. {
  13. private readonly ISqlSugarClient _db;
  14. private readonly UserManager _userManager;
  15. private readonly OrderResourceCheckService _resourceCheck;
  16. private readonly ProductionScheduleGenerationService _scheduleGen;
  17. private readonly WorkOrderMaterialDetailSyncService _materialDetailSync;
  18. private readonly WorkOrderRoutingSyncService _routingSync;
  19. private readonly AidopActionRunLogWriter _runLog;
  20. public ProductionSchedulingActionService(
  21. ISqlSugarClient db,
  22. UserManager userManager,
  23. OrderResourceCheckService resourceCheck,
  24. ProductionScheduleGenerationService scheduleGen,
  25. WorkOrderMaterialDetailSyncService materialDetailSync,
  26. WorkOrderRoutingSyncService routingSync,
  27. AidopActionRunLogWriter runLog)
  28. {
  29. _db = db;
  30. _userManager = userManager;
  31. _resourceCheck = resourceCheck;
  32. _scheduleGen = scheduleGen;
  33. _materialDetailSync = materialDetailSync;
  34. _routingSync = routingSync;
  35. _runLog = runLog;
  36. }
  37. /// <summary>生成生产排程计划(写入 PeriodSequenceDet)。</summary>
  38. [DisplayName("生成生产排程")]
  39. [HttpPost("scheduling/generate")]
  40. public async Task<object> GenerateSchedule([FromQuery] string domain, [FromQuery] bool enableCapacityConstraint = false)
  41. {
  42. var tenantId = ResolveTenantId(domain);
  43. var account = _userManager.Account ?? "system";
  44. var logId = await _runLog.StartAsync("S2_SCHEDULE_GENERATE", tenantId, "WorkOrdMaster", null, $"domain={domain}");
  45. try
  46. {
  47. var result = await _scheduleGen.GenerateAsync(tenantId, domain, account, enableCapacityConstraint);
  48. await _runLog.SuccessAsync(logId, result.Message, new
  49. {
  50. tenantId,
  51. domain,
  52. result.WorkOrderCount,
  53. result.ScheduleRowCount,
  54. result.UsedWorkCenterCalendar,
  55. result.SkippedWorkOrders
  56. });
  57. return new
  58. {
  59. message = result.Message,
  60. workOrderCount = result.WorkOrderCount,
  61. scheduleRowCount = result.ScheduleRowCount,
  62. usedWorkCenterCalendar = result.UsedWorkCenterCalendar,
  63. skippedWorkOrders = result.SkippedWorkOrders
  64. };
  65. }
  66. catch (Exception ex)
  67. {
  68. await _runLog.FailedAsync(logId, ex.Message, new { tenantId, domain });
  69. throw;
  70. }
  71. }
  72. /// <summary>优先级/数量/交期调整并重做资源检查。</summary>
  73. [DisplayName("优先级调整并重检")]
  74. [HttpPost("scheduling/update-priority-and-recheck")]
  75. public async Task<object> UpdatePriorityAndRecheck([FromBody] WorkOrderPriorityRecheckInput input)
  76. {
  77. var tenantId = AidopTenantHelper.Resolve(App.HttpContext);
  78. var account = string.IsNullOrWhiteSpace(input.UserAccount)
  79. ? (_userManager.Account ?? "system")
  80. : input.UserAccount.Trim();
  81. var workOrd = input.Workord.Trim();
  82. var logId = await _runLog.StartAsync("S2_PRIORITY_RECHECK", tenantId, "WorkOrdMaster", null, workOrd);
  83. try
  84. {
  85. var before = await LoadWorkOrderSnapshotAsync(tenantId, workOrd);
  86. DateTime? dueDate = null;
  87. if (!string.IsNullOrWhiteSpace(input.Instockdate))
  88. dueDate = DateTime.Parse(input.Instockdate.Trim(), CultureInfo.InvariantCulture).Date;
  89. decimal? qty = null;
  90. if (!string.IsNullOrWhiteSpace(input.Qty) && decimal.TryParse(input.Qty.Trim(), NumberStyles.Any, CultureInfo.InvariantCulture, out var q))
  91. qty = q;
  92. var pars = new List<SugarParameter>
  93. {
  94. new("@WorkOrd", workOrd),
  95. new("@TenantId", tenantId),
  96. new("@Priority", string.IsNullOrWhiteSpace(input.Priority) ? (object)DBNull.Value : input.Priority.Trim()),
  97. new("@LotSerial", string.IsNullOrWhiteSpace(input.LotSerial) ? (object)DBNull.Value : input.LotSerial.Trim()),
  98. new("@Qty", qty ?? (object)DBNull.Value),
  99. new("@DueDate", dueDate ?? (object)DBNull.Value),
  100. new("@User", account),
  101. new("@Now", DateTime.Now)
  102. };
  103. var affected = await _db.Ado.ExecuteCommandAsync(
  104. """
  105. UPDATE WorkOrdMaster
  106. SET Priority = COALESCE(@Priority, Priority),
  107. LotSerial = COALESCE(@LotSerial, LotSerial),
  108. QtyOrded = COALESCE(@Qty, QtyOrded),
  109. DueDate = COALESCE(@DueDate, DueDate),
  110. UpdateUser = @User,
  111. UpdateTime = @Now
  112. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  113. """,
  114. pars);
  115. if (affected == 0)
  116. throw Oops.Oh("工单不存在或未更新");
  117. var link = await LoadWorkOrderEntryLinkAsync(tenantId, workOrd);
  118. var warnings = new List<string>();
  119. var resourceRechecked = false;
  120. if (link is not null && (qty.HasValue || dueDate.HasValue))
  121. {
  122. if (qty.HasValue)
  123. link.Entry.Qty = qty;
  124. if (dueDate.HasValue)
  125. link.Entry.SysCapacityDate = dueDate;
  126. await _resourceCheck.RunForEntryAsync(link.Order, link.Entry, workOrd, account, warnings);
  127. resourceRechecked = true;
  128. // 资源检查后同步工单物料明细(WorkOrdDetail)和工艺路线(WorkOrdRouting)
  129. await _materialDetailSync.EnsureFromResourceCheckAsync(tenantId, workOrd, account);
  130. await _routingSync.EnsureFromRoutingAsync(tenantId, workOrd, account);
  131. }
  132. var qtyChanged = qty.HasValue && before?.QtyOrded != qty;
  133. var dueChanged = dueDate.HasValue && before?.DueDate?.Date != dueDate.Value.Date;
  134. var priorityChanged = !string.IsNullOrWhiteSpace(input.Priority)
  135. && !string.Equals(before?.Priority?.Trim(), input.Priority.Trim(), StringComparison.Ordinal);
  136. // 数量变更时同步更新 mes_morder 和 mes_moentry
  137. if (qtyChanged && qty.HasValue)
  138. {
  139. await UpdateMesOrderQuantityAsync(tenantId, workOrd, qty.Value, account);
  140. // 同步更新 WorkOrdRouting.QtyOrded(EnsureFromRoutingAsync 已有数据时不会更新数量)
  141. await _db.Ado.ExecuteCommandAsync(
  142. """
  143. UPDATE WorkOrdRouting
  144. SET QtyOrded = @Qty,
  145. UpdateUser = @User, UpdateTime = @Now
  146. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  147. AND IFNULL(IsActive, 0) = 1
  148. """,
  149. new SugarParameter("@Qty", qty.Value),
  150. new SugarParameter("@User", account),
  151. new SugarParameter("@Now", DateTime.Now),
  152. new SugarParameter("@TenantId", tenantId),
  153. new SugarParameter("@WorkOrd", workOrd));
  154. }
  155. // 数量变更时同步更新领料单明细(NbrDetail)
  156. if (qtyChanged)
  157. {
  158. await SyncPickingListDetailsAsync(tenantId, workOrd, account, warnings);
  159. }
  160. ProductionScheduleGenerationService.ScheduleGenerationResult? reschedule = null;
  161. if (qtyChanged || dueChanged)
  162. {
  163. reschedule = await _scheduleGen.RegenerateForWorkOrderAsync(tenantId, workOrd, input.Domain, account, input.EnableCapacityConstraint);
  164. warnings.Add(reschedule.Message);
  165. }
  166. else if (priorityChanged)
  167. {
  168. await _scheduleGen.DeactivateExistingScheduleAsync(tenantId, workOrd);
  169. warnings.Add("优先级已变更,原排程已失效,请重新执行批量排程");
  170. }
  171. await _runLog.SuccessAsync(logId, "优先级调整并重检完成", new
  172. {
  173. workOrd,
  174. before,
  175. after = new
  176. {
  177. qty,
  178. dueDate,
  179. priority = input.Priority,
  180. lotSerial = input.LotSerial
  181. },
  182. qtyChanged,
  183. dueChanged,
  184. priorityChanged,
  185. resourceRechecked,
  186. reschedule = reschedule is null
  187. ? null
  188. : new
  189. {
  190. reschedule.WorkOrderCount,
  191. reschedule.ScheduleRowCount,
  192. reschedule.UsedWorkCenterCalendar
  193. },
  194. warnings
  195. });
  196. return new
  197. {
  198. message = "ok",
  199. warnings,
  200. resourceRechecked,
  201. rescheduleTriggered = reschedule is not null,
  202. scheduleRowCount = reschedule?.ScheduleRowCount ?? 0
  203. };
  204. }
  205. catch (Exception ex)
  206. {
  207. await _runLog.FailedAsync(logId, ex.Message, new { workOrd, tenantId });
  208. throw;
  209. }
  210. }
  211. private long ResolveTenantId(string? domain)
  212. {
  213. if (!string.IsNullOrWhiteSpace(domain) && long.TryParse(domain.Trim(), out var tid) && tid > 0)
  214. return tid;
  215. return AidopTenantHelper.Resolve(App.HttpContext);
  216. }
  217. /// <summary>
  218. /// 数量变更时同步更新 mes_morder(制造工单)和 mes_moentry(工单明细)的数量字段。
  219. /// </summary>
  220. private async Task UpdateMesOrderQuantityAsync(long tenantId, string workOrd, decimal newQty, string account)
  221. {
  222. var now = DateTime.Now;
  223. // 更新 mes_morder
  224. await _db.Ado.ExecuteCommandAsync(
  225. """
  226. UPDATE mes_morder
  227. SET need_number = @Qty,
  228. morder_production_number = @Qty,
  229. update_by_name = @User,
  230. update_time = @Now
  231. WHERE morder_no = @MorderNo
  232. AND tenant_id = @TenantId
  233. AND IsDeleted = 0
  234. """,
  235. new SugarParameter("@Qty", newQty),
  236. new SugarParameter("@User", account),
  237. new SugarParameter("@Now", now),
  238. new SugarParameter("@MorderNo", workOrd),
  239. new SugarParameter("@TenantId", tenantId));
  240. // 更新 mes_moentry(remaining_number 仅在未开始生产时才更新)
  241. await _db.Ado.ExecuteCommandAsync(
  242. """
  243. UPDATE mes_moentry
  244. SET need_number = @Qty,
  245. morder_production_number = @Qty,
  246. remaining_number = CASE
  247. WHEN IFNULL(remaining_number, 0) = IFNULL(need_number, 0)
  248. OR IFNULL(morder_production_number, 0) = 0
  249. THEN @Qty
  250. ELSE remaining_number
  251. END,
  252. update_by_name = @User,
  253. update_time = @Now
  254. WHERE moentry_mono = @Mono
  255. AND tenant_id = @TenantId
  256. AND IsDeleted = 0
  257. """,
  258. new SugarParameter("@Qty", newQty),
  259. new SugarParameter("@User", account),
  260. new SugarParameter("@Now", now),
  261. new SugarParameter("@Mono", workOrd),
  262. new SugarParameter("@TenantId", tenantId));
  263. }
  264. /// <summary>
  265. /// 数量变更后同步领料单明细(NbrDetail),按物料逐行比对更新。
  266. /// 规则:
  267. /// 已发料(QtyRec>0) + 新需求>已发料 → 更新 QtyOrd/CurrQtyOpened
  268. /// 已发料(QtyRec>0) + 新需求≤已发料 → 关闭该行
  269. /// 未发料 → 直接修改 QtyOrd/CurrQtyOpened
  270. /// 工单明细有但领料单无 → 新增
  271. /// 领料单有但工单明细无 → 关闭
  272. /// </summary>
  273. private async Task SyncPickingListDetailsAsync(
  274. long tenantId, string workOrd, string account, List<string> warnings)
  275. {
  276. // 获取工单状态
  277. var status = await _db.Ado.GetStringAsync(
  278. """
  279. SELECT IFNULL(LOWER(TRIM(Status)), '') FROM WorkOrdMaster
  280. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  281. LIMIT 1
  282. """,
  283. new SugarParameter("@TenantId", tenantId),
  284. new SugarParameter("@WorkOrd", workOrd));
  285. // 仅处理 下达(R)/投产(W)/暂停(S) 状态的工单
  286. if (status is not ("r" or "w" or "s"))
  287. {
  288. // 初始(P)状态无领料单,无需同步
  289. return;
  290. }
  291. // 查找领料单主记录
  292. var nbrRows = await _db.Ado.SqlQueryAsync<NbrMasterRow>(
  293. """
  294. SELECT RecID, Nbr, `Domain` FROM NbrMaster
  295. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd AND Type = 'SM'
  296. AND IFNULL(TransType, '') = ''
  297. AND IFNULL(IsActive, 0) = 1
  298. """,
  299. new SugarParameter("@TenantId", tenantId),
  300. new SugarParameter("@WorkOrd", workOrd));
  301. if (nbrRows.Count == 0)
  302. {
  303. warnings.Add($"工单 {workOrd} 状态为 {status.ToUpper()} 但无领料单,跳过领料单同步");
  304. return;
  305. }
  306. var now = DateTime.Now;
  307. // 从 WorkOrdDetail 汇总最新物料需求
  308. var details = await _db.Ado.SqlQueryAsync<PickDetailRow>(
  309. """
  310. SELECT
  311. d.ItemNum,
  312. SUM(d.QtyRequired) AS QtyRequired,
  313. MAX(IFNULL(d.UM, im.Um)) AS Unit,
  314. MAX(im.Descr) AS ItemName
  315. FROM WorkOrdDetail d
  316. LEFT JOIN ItemMaster im ON d.ItemNum = im.ItemNum AND im.tenant_id = d.tenant_id
  317. WHERE d.tenant_id = @TenantId AND d.WorkOrd = @WorkOrd AND IFNULL(d.IsActive, 0) = 1
  318. GROUP BY d.ItemNum
  319. HAVING SUM(d.QtyRequired) > 0
  320. ORDER BY d.ItemNum
  321. """,
  322. new SugarParameter("@TenantId", tenantId),
  323. new SugarParameter("@WorkOrd", workOrd));
  324. foreach (var nbr in nbrRows)
  325. {
  326. var domain = string.IsNullOrWhiteSpace(nbr.Domain) ? tenantId.ToString() : nbr.Domain!.Trim();
  327. if (domain.Length > 8) domain = domain[..8];
  328. // 加载当前领料单明细行(仅未关闭的)
  329. var existingDetails = await _db.Ado.SqlQueryAsync<NbrDetailRow>(
  330. """
  331. SELECT RecID, ItemNum, QtyOrd, QtyRec, CurrQtyOpened, Line
  332. FROM NbrDetail
  333. WHERE tenant_id = @TenantId AND Nbr = @Nbr AND Type = 'SM'
  334. AND IFNULL(IsActive, 0) = 1
  335. """,
  336. new SugarParameter("@TenantId", tenantId),
  337. new SugarParameter("@Nbr", nbr.Nbr));
  338. var detailMap = details.ToDictionary(d => d.ItemNum.Trim(), d => d);
  339. var existingMap = existingDetails.ToDictionary(d => (d.ItemNum ?? "").Trim(), d => d);
  340. // ── 1. 处理已有明细行:按 ItemNum 匹配 ──
  341. foreach (var existing in existingDetails)
  342. {
  343. var key = (existing.ItemNum ?? "").Trim();
  344. if (detailMap.TryGetValue(key, out var newDetail))
  345. {
  346. var newQty = newDetail.QtyRequired;
  347. if (existing.QtyRec > 0)
  348. {
  349. if (newQty > existing.QtyRec)
  350. {
  351. // 新需求 > 已发料 → 更新需求数
  352. await _db.Ado.ExecuteCommandAsync(
  353. """
  354. UPDATE NbrDetail
  355. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  356. UM = @UM, ItemName = @ItemName,
  357. UpdateUser = @User, UpdateTime = @Now
  358. WHERE RecID = @RecId
  359. """,
  360. new SugarParameter("@QtyOrd", newQty),
  361. new SugarParameter("@CurrQtyOpened", newQty),
  362. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  363. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  364. new SugarParameter("@User", account),
  365. new SugarParameter("@Now", now),
  366. new SugarParameter("@RecId", existing.RecID));
  367. }
  368. else
  369. {
  370. // 新需求 <= 已发料 → 关闭当前行
  371. await _db.Ado.ExecuteCommandAsync(
  372. """
  373. UPDATE NbrDetail
  374. SET Status = 'C', IsActive = 0,
  375. UpdateUser = @User, UpdateTime = @Now
  376. WHERE RecID = @RecId
  377. """,
  378. new SugarParameter("@User", account),
  379. new SugarParameter("@Now", now),
  380. new SugarParameter("@RecId", existing.RecID));
  381. }
  382. }
  383. else
  384. {
  385. // 未发料 → 直接修改需求数
  386. await _db.Ado.ExecuteCommandAsync(
  387. """
  388. UPDATE NbrDetail
  389. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  390. UM = @UM, ItemName = @ItemName,
  391. UpdateUser = @User, UpdateTime = @Now
  392. WHERE RecID = @RecId
  393. """,
  394. new SugarParameter("@QtyOrd", newQty),
  395. new SugarParameter("@CurrQtyOpened", newQty),
  396. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  397. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  398. new SugarParameter("@User", account),
  399. new SugarParameter("@Now", now),
  400. new SugarParameter("@RecId", existing.RecID));
  401. }
  402. }
  403. else
  404. {
  405. // 物料明细中没有,但领料单明细有 → 关闭当前领料单明细行
  406. await _db.Ado.ExecuteCommandAsync(
  407. """
  408. UPDATE NbrDetail
  409. SET Status = 'C', IsActive = 0,
  410. UpdateUser = @User, UpdateTime = @Now
  411. WHERE RecID = @RecId
  412. """,
  413. new SugarParameter("@User", account),
  414. new SugarParameter("@Now", now),
  415. new SugarParameter("@RecId", existing.RecID));
  416. }
  417. }
  418. // ── 2. 新增:物料明细中有,领料单明细没有的 ──
  419. var nextDetailId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) + 1 FROM NbrDetail");
  420. short newLine = (short)(existingDetails.Count > 0 ? existingDetails.Max(d => d.Line) + 1 : 1);
  421. foreach (var d in details)
  422. {
  423. var key = d.ItemNum.Trim();
  424. if (existingMap.ContainsKey(key))
  425. continue;
  426. await _db.Ado.ExecuteCommandAsync(
  427. """
  428. INSERT INTO NbrDetail (
  429. RecID, `Domain`, Type, Nbr, Line, ItemNum, Dimension1, Dimension2,
  430. LocationFrom, LocationTo, QtyFrom, QtyTo, QtyOrd, QtyRec,
  431. CurrQtyOpened, UM, WorkOrd, ItemName, Status,
  432. IsActive, IsConfirm, IsChanged, BusinessID, NbrRecID,
  433. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  434. ) VALUES (
  435. @RecId, @Domain, 'SM', @Nbr, @Line, @ItemNum, '', '',
  436. '', '', 0, 0, @QtyOrd, 0,
  437. @CurrQtyOpened, @UM, @WorkOrd, @ItemName, '',
  438. 1, 0, 1, 0, @NbrRecId,
  439. @User, @Now, @User, @Now, @TenantId
  440. )
  441. """,
  442. new SugarParameter("@RecId", nextDetailId++),
  443. new SugarParameter("@Domain", domain),
  444. new SugarParameter("@Nbr", nbr.Nbr),
  445. new SugarParameter("@Line", newLine++),
  446. new SugarParameter("@ItemNum", d.ItemNum),
  447. new SugarParameter("@QtyOrd", d.QtyRequired),
  448. new SugarParameter("@CurrQtyOpened", d.QtyRequired),
  449. new SugarParameter("@UM", (object?)d.Unit ?? DBNull.Value),
  450. new SugarParameter("@WorkOrd", workOrd),
  451. new SugarParameter("@ItemName", (object?)d.ItemName ?? DBNull.Value),
  452. new SugarParameter("@NbrRecId", nbr.RecID),
  453. new SugarParameter("@User", account.Length > 24 ? account[..24] : account),
  454. new SugarParameter("@Now", now),
  455. new SugarParameter("@TenantId", tenantId));
  456. }
  457. // ── 3. 更新领料单主记录的更新时间和数量 ──
  458. await _db.Ado.ExecuteCommandAsync(
  459. """
  460. UPDATE NbrMaster
  461. SET QtyOrd = (SELECT IFNULL(SUM(QtyOrd), 0) FROM NbrDetail
  462. WHERE Nbr = @Nbr AND Type = 'SM' AND IFNULL(IsActive, 1) = 1),
  463. UpdateUser = @User, UpdateTime = @Now
  464. WHERE RecID = @RecId
  465. """,
  466. new SugarParameter("@Nbr", nbr.Nbr),
  467. new SugarParameter("@User", account),
  468. new SugarParameter("@Now", now),
  469. new SugarParameter("@RecId", nbr.RecID));
  470. }
  471. }
  472. // ══════════════════════════════════════════════════════════════
  473. // 内部 DTO
  474. // ══════════════════════════════════════════════════════════════
  475. private sealed class NbrMasterRow
  476. {
  477. public int RecID { get; set; }
  478. public string Nbr { get; set; } = string.Empty;
  479. public string? Domain { get; set; }
  480. }
  481. private sealed class NbrDetailRow
  482. {
  483. public int RecID { get; set; }
  484. public string? ItemNum { get; set; }
  485. public decimal QtyOrd { get; set; }
  486. public decimal QtyRec { get; set; }
  487. public decimal CurrQtyOpened { get; set; }
  488. public short Line { get; set; }
  489. }
  490. private sealed class PickDetailRow
  491. {
  492. public string ItemNum { get; set; } = string.Empty;
  493. public decimal QtyRequired { get; set; }
  494. public string? Unit { get; set; }
  495. public string? ItemName { get; set; }
  496. }
  497. private async Task<WorkOrderSnapshotRow?> LoadWorkOrderSnapshotAsync(long tenantId, string workOrd)
  498. {
  499. var rows = await _db.Ado.SqlQueryAsync<WorkOrderSnapshotRow>(
  500. """
  501. SELECT WorkOrd, Priority, QtyOrded, DueDate, LotSerial, `Domain`
  502. FROM WorkOrdMaster
  503. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  504. LIMIT 1
  505. """,
  506. new SugarParameter("@TenantId", tenantId),
  507. new SugarParameter("@WorkOrd", workOrd));
  508. return rows.FirstOrDefault();
  509. }
  510. private async Task<WorkOrderEntryLink?> LoadWorkOrderEntryLinkAsync(long tenantId, string workOrd)
  511. {
  512. var rows = await _db.Ado.SqlQueryAsync<WorkOrderEntryLinkRow>(
  513. """
  514. SELECT
  515. w.BusinessID AS EntryId,
  516. e.seorder_id AS SeOrderId,
  517. e.bill_no AS BillNo,
  518. e.entry_seq AS EntrySeq,
  519. e.item_number AS ItemNumber,
  520. e.item_name AS ItemName,
  521. e.specification AS Specification,
  522. e.unit AS Unit,
  523. e.bom_number AS BomNumber,
  524. e.qty AS Qty,
  525. e.plan_date AS PlanDate,
  526. e.sys_capacity_date AS SysCapacityDate,
  527. e.progress AS Progress,
  528. e.urgent AS Urgent,
  529. e.factory_id AS FactoryId,
  530. e.company_id AS CompanyId,
  531. e.tenant_id AS TenantId,
  532. o.Id AS OrderId,
  533. o.bill_no AS OrderBillNo,
  534. o.custom_no AS CustomNo,
  535. o.urgent AS OrderUrgent,
  536. o.factory_id AS OrderFactoryId
  537. FROM WorkOrdMaster w
  538. INNER JOIN crm_seorderentry e ON e.Id = w.BusinessID AND e.tenant_id = w.tenant_id AND e.IsDeleted = 0
  539. INNER JOIN crm_seorder o ON o.Id = e.seorder_id AND o.tenant_id = e.tenant_id AND o.IsDeleted = 0
  540. WHERE w.tenant_id = @TenantId AND w.WorkOrd = @WorkOrd
  541. LIMIT 1
  542. """,
  543. new SugarParameter("@TenantId", tenantId),
  544. new SugarParameter("@WorkOrd", workOrd));
  545. var row = rows.FirstOrDefault();
  546. if (row is null || row.EntryId <= 0)
  547. return null;
  548. return new WorkOrderEntryLink
  549. {
  550. Order = new OrderWorkOrderGenerationService.OrderHeader
  551. {
  552. Id = row.OrderId,
  553. BillNo = row.OrderBillNo ?? row.BillNo,
  554. CustomNo = row.CustomNo,
  555. Urgent = row.OrderUrgent,
  556. FactoryId = row.OrderFactoryId,
  557. TenantId = tenantId
  558. },
  559. Entry = new OrderWorkOrderGenerationService.OrderEntryLine
  560. {
  561. Id = row.EntryId,
  562. SeOrderId = row.SeOrderId,
  563. BillNo = row.BillNo,
  564. EntrySeq = row.EntrySeq,
  565. ItemNumber = row.ItemNumber,
  566. ItemName = row.ItemName,
  567. Specification = row.Specification,
  568. Unit = row.Unit,
  569. BomNumber = row.BomNumber,
  570. Qty = row.Qty,
  571. PlanDate = row.PlanDate,
  572. SysCapacityDate = row.SysCapacityDate,
  573. Progress = row.Progress,
  574. Urgent = row.Urgent,
  575. FactoryId = row.FactoryId,
  576. CompanyId = row.CompanyId,
  577. TenantId = row.TenantId
  578. }
  579. };
  580. }
  581. private sealed class WorkOrderSnapshotRow
  582. {
  583. public string? WorkOrd { get; set; }
  584. public string? Priority { get; set; }
  585. public decimal? QtyOrded { get; set; }
  586. public DateTime? DueDate { get; set; }
  587. public string? LotSerial { get; set; }
  588. public string? Domain { get; set; }
  589. }
  590. private sealed class WorkOrderEntryLink
  591. {
  592. public OrderWorkOrderGenerationService.OrderHeader Order { get; set; } = new();
  593. public OrderWorkOrderGenerationService.OrderEntryLine Entry { get; set; } = new();
  594. }
  595. private sealed class WorkOrderEntryLinkRow
  596. {
  597. public long EntryId { get; set; }
  598. public long SeOrderId { get; set; }
  599. public string? BillNo { get; set; }
  600. public int? EntrySeq { get; set; }
  601. public string? ItemNumber { get; set; }
  602. public string? ItemName { get; set; }
  603. public string? Specification { get; set; }
  604. public string? Unit { get; set; }
  605. public string? BomNumber { get; set; }
  606. public decimal? Qty { get; set; }
  607. public DateTime? PlanDate { get; set; }
  608. public DateTime? SysCapacityDate { get; set; }
  609. public string? Progress { get; set; }
  610. public int? Urgent { get; set; }
  611. public long? FactoryId { get; set; }
  612. public long? CompanyId { get; set; }
  613. public long TenantId { get; set; }
  614. public long OrderId { get; set; }
  615. public string? OrderBillNo { get; set; }
  616. public string? CustomNo { get; set; }
  617. public int? OrderUrgent { get; set; }
  618. public long? OrderFactoryId { get; set; }
  619. }
  620. }
  621. public class WorkOrderPriorityRecheckInput
  622. {
  623. public string Workord { get; set; } = string.Empty;
  624. public string? Qty { get; set; }
  625. public string? Instockdate { get; set; }
  626. public string? Priority { get; set; }
  627. public string Domain { get; set; } = string.Empty;
  628. public string? UserAccount { get; set; }
  629. public string? LotSerial { get; set; }
  630. public bool EnableCapacityConstraint { get; set; }
  631. }