OrderReviewOrchestrationService.cs 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023
  1. using Admin.NET.Plugin.AiDOP.Infrastructure;
  2. using Admin.NET.Plugin.AiDOP.Production;
  3. using Admin.NET.Plugin.AiDOP.Supply;
  4. using Admin.NET.Plugin.AiDOP.WorkOrder;
  5. namespace Admin.NET.Plugin.AiDOP.Order;
  6. /// <summary>
  7. /// S1 订单评审编排:资源检查、状态更新、工单生成、运行日志。
  8. /// </summary>
  9. public class OrderReviewOrchestrationService : ITransient
  10. {
  11. public const string ActionReview = "S1_ORDER_REVIEW";
  12. public const string ActionConfirm = "S1_DELIVERY_CONFIRM";
  13. public const string ActionRefresh = "S1_ORDER_REFRESH_PLAN";
  14. /// <summary>订单评审资源检查批次ID,用于跨工单库存占用递减。</summary>
  15. private const long ReviewBangId = 2;
  16. private readonly ISqlSugarClient _db;
  17. private readonly UserManager _userManager;
  18. private readonly OrderWorkOrderGenerationService _workOrderGen;
  19. private readonly OrderResourceCheckService _resourceCheck;
  20. private readonly WorkOrderMaterialDetailSyncService _materialDetailSync;
  21. private readonly WorkOrderRoutingSyncService _routingSync;
  22. private readonly S1MdpSyncTransformService _mdpSync;
  23. private readonly AidopActionRunLogWriter _runLog;
  24. private readonly ProductionScheduleGenerationService _scheduleGen;
  25. private readonly ProcurementPipelineService _pipeline;
  26. public OrderReviewOrchestrationService(
  27. ISqlSugarClient db,
  28. UserManager userManager,
  29. OrderWorkOrderGenerationService workOrderGen,
  30. OrderResourceCheckService resourceCheck,
  31. WorkOrderMaterialDetailSyncService materialDetailSync,
  32. WorkOrderRoutingSyncService routingSync,
  33. S1MdpSyncTransformService mdpSync,
  34. AidopActionRunLogWriter runLog,
  35. ProductionScheduleGenerationService scheduleGen,
  36. ProcurementPipelineService pipeline)
  37. {
  38. _db = db;
  39. _userManager = userManager;
  40. _workOrderGen = workOrderGen;
  41. _resourceCheck = resourceCheck;
  42. _materialDetailSync = materialDetailSync;
  43. _routingSync = routingSync;
  44. _mdpSync = mdpSync;
  45. _runLog = runLog;
  46. _scheduleGen = scheduleGen;
  47. _pipeline = pipeline;
  48. }
  49. public Task<SeOrderReviewExecuteResult> ReviewAsync(IReadOnlyList<long> orderIds) =>
  50. ExecuteBatchAsync(ActionReview, orderIds, ReviewOneOrderAsync);
  51. public Task<SeOrderReviewExecuteResult> ConfirmDeliveryAsync(IReadOnlyList<long> orderIds) =>
  52. ExecuteBatchAsync(ActionConfirm, orderIds, ConfirmOneOrderAsync);
  53. public Task<SeOrderReviewExecuteResult> RefreshPlanAsync(long orderId, string? reason) =>
  54. ExecuteSingleAsync(ActionRefresh, orderId, async (order, result, warnings, account) =>
  55. {
  56. // ── 第0步:物料编码变更校验 ──
  57. await ValidateMaterialNotChangedAsync(order.Id, order.TenantId);
  58. // ── 第1步:清理旧占用记录,确保跨工单库存递减重新计算 ──
  59. await _db.Ado.ExecuteCommandAsync(
  60. "DELETE FROM ic_item_stockoccupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  61. new SugarParameter("@TenantId", order.TenantId),
  62. new SugarParameter("@BangId", ReviewBangId));
  63. await _db.Ado.ExecuteCommandAsync(
  64. "DELETE FROM srm_po_occupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  65. new SugarParameter("@TenantId", order.TenantId),
  66. new SugarParameter("@BangId", ReviewBangId));
  67. // ── 第2步:加载可重排明细行(确认 / 再评审) ──
  68. var entries = await LoadReviewableEntriesAsync(order.Id, order.TenantId, ["3", "0"]);
  69. if (entries.Count == 0)
  70. throw Oops.Oh("订单没有可重排的确认/再评审明细行");
  71. // ── 第3步:逐条处理(工单 + 资源检查 + 领料单 + 交期更新) ──
  72. foreach (var entry in entries)
  73. {
  74. ValidateEntryForResourceCheck(entry);
  75. var wo = await _workOrderGen.CreateOrUpdateForEntryAsync(order, entry, account, warnings);
  76. if (wo.Created) result.WorkOrderCreatedCount++;
  77. else result.WorkOrderUpdatedCount++;
  78. result.WorkOrders.Add(wo.WorkOrd);
  79. var check = await _resourceCheck.RunForEntryAsync(order, entry, wo.WorkOrd, account, warnings, ReviewBangId);
  80. result.ResourceCheckCount++;
  81. result.ResourceCheckLineCount += check.LineCount;
  82. await _materialDetailSync.EnsureFromResourceCheckAsync(entry.TenantId, wo.WorkOrd, account);
  83. await _routingSync.EnsureFromRoutingAsync(entry.TenantId, wo.WorkOrd, account);
  84. // 当工单状态为下达/投产/暂停(R、W、S)时,更新对应领料单数据
  85. await UpdatePickingListForActiveWorkOrderAsync(entry.TenantId, wo.WorkOrd, account, warnings);
  86. // 根据资源检查结果更新明细行系统建议交期
  87. await UpdateEntrySysCapacityDateAsync(entry.Id, check.KittingTime, account);
  88. }
  89. // ── 第4步:更新明细行进度为3 ──
  90. await UpdateEntriesProgressAsync(entries.Select(e => e.Id).ToList(), "3", account);
  91. // ── 第4.5步:重新进行生产排程(在事务提交前) ──
  92. try
  93. {
  94. var scheduleResult = await _scheduleGen.GenerateAsync(order.TenantId, order.TenantId.ToString(), account);
  95. if (!string.IsNullOrWhiteSpace(scheduleResult.Message))
  96. warnings.Add($"生产排程:{scheduleResult.Message}");
  97. }
  98. catch (Exception ex)
  99. {
  100. warnings.Add($"生产排程失败:{ex.Message}");
  101. }
  102. // ── 第4.5步:同步物料需求(MRP → PR → 采购闭环) ──
  103. try
  104. {
  105. var mrResult = await _pipeline.ExecuteCoreAsync(order.TenantId, account, createFromShortage: true);
  106. if (!string.IsNullOrWhiteSpace(mrResult.Message))
  107. warnings.Add($"物料需求同步:{mrResult.Message}");
  108. }
  109. catch (Exception ex)
  110. {
  111. warnings.Add($"物料需求同步失败:{ex.Message}");
  112. }
  113. // ── 第4.5步:生成交货单(出货计划) ──
  114. await GenerateShippingPlanFromOrderAsync(order, entries, account, warnings);
  115. result.EntryCount = entries.Count;
  116. result.Message = string.IsNullOrWhiteSpace(reason)
  117. ? "3级计划重排完成"
  118. : $"3级计划重排完成:{reason.Trim()}";
  119. });
  120. private async Task<SeOrderReviewExecuteResult> ExecuteBatchAsync(
  121. string actionCode,
  122. IReadOnlyList<long> orderIds,
  123. Func<OrderWorkOrderGenerationService.OrderHeader, SeOrderReviewExecuteResult, List<string>, string, Task> handler)
  124. {
  125. if (orderIds is null || orderIds.Count == 0)
  126. throw Oops.Oh("至少选择一条订单");
  127. var tenantId = _userManager.TenantId > 0
  128. ? _userManager.TenantId
  129. : AidopTenantHelper.Resolve(App.HttpContext);
  130. var account = _userManager.Account ?? "system";
  131. var distinctIds = orderIds.Distinct().ToList();
  132. var aggregate = new SeOrderReviewExecuteResult
  133. {
  134. ActionCode = actionCode,
  135. OrderCount = distinctIds.Count,
  136. Message = "执行成功"
  137. };
  138. var allWarnings = new List<string>();
  139. long? firstLogId = null;
  140. // 评审/重排前清理旧占用记录,确保跨工单库存递减正确
  141. if (actionCode == ActionReview || actionCode == ActionRefresh)
  142. {
  143. await _db.Ado.ExecuteCommandAsync(
  144. "DELETE FROM ic_item_stockoccupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  145. new SugarParameter("@TenantId", tenantId),
  146. new SugarParameter("@BangId", ReviewBangId));
  147. await _db.Ado.ExecuteCommandAsync(
  148. "DELETE FROM srm_po_occupy WHERE tenant_id = @TenantId AND bang_id = @BangId",
  149. new SugarParameter("@TenantId", tenantId),
  150. new SugarParameter("@BangId", ReviewBangId));
  151. }
  152. foreach (var orderId in distinctIds)
  153. {
  154. var order = await LoadOrderAsync(orderId, tenantId)
  155. ?? throw Oops.Oh($"订单 {orderId} 不存在或不属于当前租户");
  156. var runLogId = await _runLog.StartAsync(actionCode, tenantId, "crm_seorder", order.Id, order.BillNo);
  157. if (firstLogId is null)
  158. firstLogId = runLogId;
  159. var perOrder = new SeOrderReviewExecuteResult { ActionCode = actionCode };
  160. var warnings = new List<string>();
  161. try
  162. {
  163. await _db.Ado.BeginTranAsync();
  164. await handler(order, perOrder, warnings, account);
  165. await _db.Ado.CommitTranAsync();
  166. aggregate.EntryCount += perOrder.EntryCount;
  167. aggregate.WorkOrderCreatedCount += perOrder.WorkOrderCreatedCount;
  168. aggregate.WorkOrderUpdatedCount += perOrder.WorkOrderUpdatedCount;
  169. aggregate.WorkOrderClosedCount += perOrder.WorkOrderClosedCount;
  170. aggregate.ResourceCheckCount += perOrder.ResourceCheckCount;
  171. aggregate.ResourceCheckLineCount += perOrder.ResourceCheckLineCount;
  172. aggregate.WorkOrders.AddRange(perOrder.WorkOrders);
  173. allWarnings.AddRange(warnings);
  174. await _runLog.SuccessAsync(runLogId, perOrder.Message, new
  175. {
  176. orderId = order.Id,
  177. billNo = order.BillNo,
  178. perOrder.EntryCount,
  179. perOrder.WorkOrderCreatedCount,
  180. perOrder.WorkOrderUpdatedCount,
  181. workOrders = perOrder.WorkOrders,
  182. warnings
  183. });
  184. }
  185. catch (Exception ex)
  186. {
  187. await _db.Ado.RollbackTranAsync();
  188. await _runLog.FailedAsync(runLogId, ex.Message, new { orderId = order.Id, billNo = order.BillNo });
  189. throw Oops.Oh(ex.Message);
  190. }
  191. }
  192. aggregate.RunLogId = firstLogId ?? 0;
  193. aggregate.Warnings = allWarnings.Distinct().ToList();
  194. aggregate.Message = BuildAggregateMessage(actionCode, aggregate);
  195. if (actionCode == ActionReview && aggregate.ResourceCheckCount > 0)
  196. aggregate.Warnings.AddRange(await TryTriggerMdpRefreshAsync());
  197. return aggregate;
  198. }
  199. private async Task<SeOrderReviewExecuteResult> ExecuteSingleAsync(
  200. string actionCode,
  201. long orderId,
  202. Func<OrderWorkOrderGenerationService.OrderHeader, SeOrderReviewExecuteResult, List<string>, string, Task> handler)
  203. {
  204. var tenantId = _userManager.TenantId > 0
  205. ? _userManager.TenantId
  206. : AidopTenantHelper.Resolve(App.HttpContext);
  207. var account = _userManager.Account ?? "system";
  208. var order = await LoadOrderAsync(orderId, tenantId)
  209. ?? throw Oops.Oh("订单不存在或不属于当前租户");
  210. var result = new SeOrderReviewExecuteResult
  211. {
  212. ActionCode = actionCode,
  213. OrderCount = 1
  214. };
  215. var warnings = new List<string>();
  216. var runLogId = await _runLog.StartAsync(actionCode, tenantId, "crm_seorder", order.Id, order.BillNo);
  217. result.RunLogId = runLogId;
  218. try
  219. {
  220. await _db.Ado.BeginTranAsync();
  221. await handler(order, result, warnings, account);
  222. await _db.Ado.CommitTranAsync();
  223. result.Warnings = warnings;
  224. result.Message = BuildSingleMessage(result);
  225. await _runLog.SuccessAsync(runLogId, result.Message, new
  226. {
  227. orderId = order.Id,
  228. billNo = order.BillNo,
  229. result.EntryCount,
  230. result.WorkOrderCreatedCount,
  231. result.WorkOrderUpdatedCount,
  232. workOrders = result.WorkOrders,
  233. warnings
  234. });
  235. if (actionCode == ActionRefresh && result.ResourceCheckCount > 0)
  236. result.Warnings.AddRange(await TryTriggerMdpRefreshAsync());
  237. return result;
  238. }
  239. catch (Exception ex)
  240. {
  241. await _db.Ado.RollbackTranAsync();
  242. await _runLog.FailedAsync(runLogId, ex.Message, new { orderId = order.Id, billNo = order.BillNo });
  243. throw Oops.Oh(ex.Message);
  244. }
  245. }
  246. private async Task ReviewOneOrderAsync(
  247. OrderWorkOrderGenerationService.OrderHeader order,
  248. SeOrderReviewExecuteResult result,
  249. List<string> warnings,
  250. string account)
  251. {
  252. var entries = await LoadReviewableEntriesAsync(order.Id, order.TenantId, ["1", "0", "2"]);
  253. if (entries.Count == 0)
  254. throw Oops.Oh($"订单 {order.BillNo} 没有可评审的明细行(须为新建,评审,再评审状态)");
  255. foreach (var entry in entries)
  256. {
  257. ValidateEntryForResourceCheck(entry);
  258. if (entry.PlanDate is null)
  259. throw Oops.Oh($"订单行 {entry.EntrySeq} 缺少客户要求交期(plan_date)");
  260. // 1. 先做资源检查(纯 BOM 展开 + 库存计算,不依赖工单),记录库存占用
  261. var (check, lines) = await _resourceCheck.CheckOnlyAsync(order, entry, warnings, ReviewBangId);
  262. result.ResourceCheckCount++;
  263. result.ResourceCheckLineCount += check.LineCount;
  264. // 2. 有缺料 → 才生成工单并写入资源检查结果;库存可满足则跳过工单
  265. if (check.HasShortage)
  266. {
  267. var wo = await _workOrderGen.CreateOrUpdateForEntryAsync(order, entry, account, warnings);
  268. if (wo.Created) result.WorkOrderCreatedCount++;
  269. else result.WorkOrderUpdatedCount++;
  270. result.WorkOrders.Add(wo.WorkOrd);
  271. await _resourceCheck.WriteResultAsync(order, entry, wo.WorkOrd, lines, account);
  272. await _materialDetailSync.EnsureFromResourceCheckAsync(entry.TenantId, wo.WorkOrd, account);
  273. await _routingSync.EnsureFromRoutingAsync(entry.TenantId, wo.WorkOrd, account);
  274. warnings.Add($"订单行 {entry.EntrySeq} 存在缺料(工单 {wo.WorkOrd})");
  275. }
  276. else
  277. {
  278. // 3. 不缺料 → 若之前已生成工单则关闭(Status='C', IsActive=0)
  279. var closedCount = await CloseExistingWorkOrdersAsync(entry.TenantId, entry.Id, account);
  280. if (closedCount > 0)
  281. {
  282. result.WorkOrderClosedCount += closedCount;
  283. warnings.Add($"订单行 {entry.EntrySeq} 库存可满足,已关闭 {closedCount} 个历史工单");
  284. }
  285. }
  286. await UpdateEntryAfterReviewAsync(entry.Id, check.KittingTime, account);
  287. }
  288. result.EntryCount = entries.Count;
  289. result.Message = $"订单 {order.BillNo} 评审完成(资源检查 {result.ResourceCheckCount} 条)";
  290. }
  291. private async Task ConfirmOneOrderAsync(
  292. OrderWorkOrderGenerationService.OrderHeader order,
  293. SeOrderReviewExecuteResult result,
  294. List<string> warnings,
  295. string account)
  296. {
  297. var entries = await LoadReviewableEntriesAsync(order.Id, order.TenantId, ["2"]);
  298. if (entries.Count == 0)
  299. throw Oops.Oh($"订单 {order.BillNo} 没有处于评审状态的明细行,请先完成订单评审");
  300. foreach (var entry in entries)
  301. {
  302. var woCnt = await _db.Ado.GetIntAsync(
  303. """
  304. SELECT COUNT(*) FROM WorkOrdMaster
  305. WHERE tenant_id = @TenantId AND BusinessID = @EntryId
  306. AND LOWER(TRIM(IFNULL(Status,''))) <> 'c'
  307. """,
  308. new SugarParameter("@TenantId", entry.TenantId),
  309. new SugarParameter("@EntryId", entry.Id));
  310. // 无活跃工单:可能从未生成(库存可满足)或已关闭 → 仍允许交期确认
  311. var confirmDate = entry.SysCapacityDate ?? entry.PlanDate;
  312. await UpdateEntryAfterConfirmAsync(entry.Id, confirmDate, account);
  313. result.EntryCount++;
  314. if (woCnt > 0)
  315. {
  316. var workOrd = await LoadWorkOrdForEntryAsync(entry.TenantId, entry.Id);
  317. result.WorkOrders.Add(workOrd);
  318. // 交期确认后将工单状态设为 p
  319. if (!string.IsNullOrWhiteSpace(workOrd))
  320. {
  321. await _db.Ado.ExecuteCommandAsync(
  322. """
  323. UPDATE WorkOrdMaster
  324. SET Status = 'p', UpdateUser = @User, UpdateTime = @Now
  325. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  326. AND IFNULL(TRIM(Status), '') = ''
  327. """,
  328. new SugarParameter("@TenantId", entry.TenantId),
  329. new SugarParameter("@WorkOrd", workOrd),
  330. new SugarParameter("@User", account),
  331. new SugarParameter("@Now", DateTime.Now));
  332. }
  333. }
  334. else
  335. warnings.Add($"订单行 {entry.EntrySeq} 无活跃工单(库存可满足或已关闭),已直接确认交期");
  336. }
  337. result.WorkOrders = result.WorkOrders.Where(x => !string.IsNullOrWhiteSpace(x)).Distinct().ToList();
  338. result.Message = $"订单 {order.BillNo} 交期确认完成";
  339. }
  340. private async Task<string> LoadWorkOrdForEntryAsync(long tenantId, long entryId)
  341. {
  342. var rows = await _db.Ado.SqlQueryAsync<string>(
  343. """
  344. SELECT WorkOrd FROM WorkOrdMaster
  345. WHERE tenant_id = @TenantId AND BusinessID = @EntryId
  346. ORDER BY RecID DESC LIMIT 1
  347. """,
  348. new SugarParameter("@TenantId", tenantId),
  349. new SugarParameter("@EntryId", entryId));
  350. return rows.FirstOrDefault() ?? string.Empty;
  351. }
  352. /// <summary>
  353. /// 关闭指定订单行已有的未关闭工单(WorkOrdMaster.Status='C', IsActive=0;mes_morder.morder_state='关闭')。
  354. /// 返回被关闭的工单数量。
  355. /// </summary>
  356. private async Task<int> CloseExistingWorkOrdersAsync(long tenantId, long entryId, string account)
  357. {
  358. // 查找该订单行下所有未关闭的工单号
  359. var openWorkOrds = await _db.Ado.SqlQueryAsync<string>(
  360. """
  361. SELECT WorkOrd FROM WorkOrdMaster
  362. WHERE tenant_id = @TenantId AND BusinessID = @EntryId
  363. AND LOWER(TRIM(IFNULL(Status,''))) <> 'c'
  364. AND IFNULL(IsActive, 0) = 1
  365. """,
  366. new SugarParameter("@TenantId", tenantId),
  367. new SugarParameter("@EntryId", entryId));
  368. if (openWorkOrds.Count == 0) return 0;
  369. var workOrdList = string.Join(",", openWorkOrds.Select(w => $"'{w}'"));
  370. var now = DateTime.Now;
  371. // 关闭 WorkOrdMaster
  372. await _db.Ado.ExecuteCommandAsync(
  373. $"""
  374. UPDATE WorkOrdMaster
  375. SET Status = 'C', IsActive = 0,
  376. UpdateUser = @User, UpdateTime = @Now
  377. WHERE tenant_id = @TenantId AND BusinessID = @EntryId
  378. AND LOWER(TRIM(IFNULL(Status,''))) <> 'c'
  379. AND IFNULL(IsActive, 0) = 1
  380. """,
  381. new SugarParameter("@TenantId", tenantId),
  382. new SugarParameter("@EntryId", entryId),
  383. new SugarParameter("@User", account),
  384. new SugarParameter("@Now", now));
  385. // 同步关闭 mes_morder
  386. await _db.Ado.ExecuteCommandAsync(
  387. $"""
  388. UPDATE mes_morder
  389. SET morder_state = '关闭',
  390. update_by_name = @User, update_time = @Now
  391. WHERE tenant_id = @TenantId AND morder_no IN ({workOrdList})
  392. AND IFNULL(morder_state, '') <> '关闭'
  393. """,
  394. new SugarParameter("@TenantId", tenantId),
  395. new SugarParameter("@User", account),
  396. new SugarParameter("@Now", now));
  397. return openWorkOrds.Count;
  398. }
  399. private static void ValidateEntryForResourceCheck(OrderWorkOrderGenerationService.OrderEntryLine entry)
  400. {
  401. if (string.IsNullOrWhiteSpace(entry.ItemNumber))
  402. throw Oops.Oh($"订单行 {entry.EntrySeq} 物料编码不能为空");
  403. if (entry.Qty is null or <= 0)
  404. throw Oops.Oh($"订单行 {entry.EntrySeq} 数量必须大于 0");
  405. if (entry.PlanDate is null && entry.SysCapacityDate is null)
  406. throw Oops.Oh($"订单行 {entry.EntrySeq} 缺少计划交期");
  407. }
  408. private async Task<OrderWorkOrderGenerationService.OrderHeader?> LoadOrderAsync(long orderId, long tenantId)
  409. {
  410. var rows = await _db.Ado.SqlQueryAsync<OrderWorkOrderGenerationService.OrderHeader>(
  411. """
  412. SELECT Id, bill_no AS BillNo, custom_no AS CustomNo, urgent AS Urgent,
  413. factory_id AS FactoryId, tenant_id AS TenantId
  414. FROM crm_seorder
  415. WHERE Id = @Id AND tenant_id = @TenantId AND IsDeleted = 0
  416. LIMIT 1
  417. """,
  418. new SugarParameter("@Id", orderId),
  419. new SugarParameter("@TenantId", tenantId));
  420. return rows.FirstOrDefault();
  421. }
  422. private async Task<List<OrderWorkOrderGenerationService.OrderEntryLine>> LoadReviewableEntriesAsync(
  423. long orderId,
  424. long tenantId,
  425. IReadOnlyList<string> progressList)
  426. {
  427. if (progressList.Count == 0)
  428. return new List<OrderWorkOrderGenerationService.OrderEntryLine>();
  429. var inClause = string.Join(", ", progressList.Select((_, i) => $"@P{i}"));
  430. var pars = new List<SugarParameter>
  431. {
  432. new("@OrderId", orderId),
  433. new("@TenantId", tenantId)
  434. };
  435. for (var i = 0; i < progressList.Count; i++)
  436. pars.Add(new SugarParameter($"@P{i}", progressList[i]));
  437. return await _db.Ado.SqlQueryAsync<OrderWorkOrderGenerationService.OrderEntryLine>(
  438. $"""
  439. SELECT
  440. Id, seorder_id AS SeOrderId, bill_no AS BillNo, entry_seq AS EntrySeq,
  441. item_number AS ItemNumber, item_name AS ItemName, specification AS Specification,
  442. unit AS Unit, bom_number AS BomNumber, qty AS Qty,
  443. plan_date AS PlanDate, sys_capacity_date AS SysCapacityDate,
  444. progress AS Progress, urgent AS Urgent,
  445. factory_id AS FactoryId, company_id AS CompanyId, tenant_id AS TenantId
  446. FROM crm_seorderentry
  447. WHERE seorder_id = @OrderId AND tenant_id = @TenantId AND IsDeleted = 0
  448. AND COALESCE(NULLIF(progress, ''), '1') IN ({inClause})
  449. ORDER BY entry_seq, Id
  450. """,
  451. pars);
  452. }
  453. private async Task UpdateEntryAfterReviewAsync(long entryId, DateTime? capacityDate, string account)
  454. {
  455. await _db.Ado.ExecuteCommandAsync(
  456. """
  457. UPDATE crm_seorderentry
  458. SET progress = '2',
  459. sys_capacity_date = @CapacityDate,
  460. update_time = @Now
  461. WHERE Id = @Id AND IsDeleted = 0
  462. """,
  463. new SugarParameter("@CapacityDate", capacityDate ?? (object)DBNull.Value),
  464. new SugarParameter("@Now", DateTime.Now),
  465. new SugarParameter("@Id", entryId));
  466. }
  467. private async Task UpdateEntryAfterConfirmAsync(long entryId, DateTime? confirmDate, string account)
  468. {
  469. await _db.Ado.ExecuteCommandAsync(
  470. """
  471. UPDATE crm_seorderentry
  472. SET progress = '3',
  473. date = COALESCE(date, @ConfirmDate),
  474. update_time = @Now
  475. WHERE Id = @Id AND IsDeleted = 0
  476. """,
  477. new SugarParameter("@ConfirmDate", confirmDate ?? (object)DBNull.Value),
  478. new SugarParameter("@Now", DateTime.Now),
  479. new SugarParameter("@Id", entryId));
  480. }
  481. private async Task UpdateEntriesProgressAsync(IReadOnlyList<long> entryIds, string progress, string account)
  482. {
  483. if (entryIds.Count == 0) return;
  484. var idList = string.Join(",", entryIds);
  485. await _db.Ado.ExecuteCommandAsync(
  486. $"""
  487. UPDATE crm_seorderentry
  488. SET progress = @Progress, update_time = @Now
  489. WHERE Id IN ({idList}) AND IsDeleted = 0
  490. """,
  491. new SugarParameter("@Progress", progress),
  492. new SugarParameter("@Now", DateTime.Now));
  493. }
  494. private static string BuildAggregateMessage(string actionCode, SeOrderReviewExecuteResult r)
  495. {
  496. var woPart = r.WorkOrders.Count > 0
  497. ? $",工单:{string.Join("、", r.WorkOrders.Distinct())}"
  498. : string.Empty;
  499. var closedPart = r.WorkOrderClosedCount > 0
  500. ? $"、关闭 {r.WorkOrderClosedCount}"
  501. : string.Empty;
  502. return actionCode switch
  503. {
  504. ActionReview => $"评审完成 {r.OrderCount} 单、{r.EntryCount} 行,新建工单 {r.WorkOrderCreatedCount}、更新 {r.WorkOrderUpdatedCount}{closedPart}、资源检查 {r.ResourceCheckCount} 条{woPart}",
  505. ActionConfirm => $"交期确认完成 {r.OrderCount} 单、{r.EntryCount} 行{woPart}",
  506. _ => r.Message
  507. };
  508. }
  509. private static string BuildSingleMessage(SeOrderReviewExecuteResult r)
  510. {
  511. if (r.WorkOrders.Count == 0)
  512. return r.Message;
  513. return $"{r.Message},工单:{string.Join("、", r.WorkOrders.Distinct())}";
  514. }
  515. private async Task<List<string>> TryTriggerMdpRefreshAsync()
  516. {
  517. var warnings = new List<string>();
  518. try
  519. {
  520. await _mdpSync.RunFullAsync(triggerType: "ORDER_REVIEW");
  521. }
  522. catch (Exception ex)
  523. {
  524. warnings.Add($"S1 MDP/DWD 刷新未完成:{ex.Message}");
  525. }
  526. return warnings;
  527. }
  528. // ══════════════════════════════════════════════════════════════
  529. // 3级计划重排 — 辅助方法
  530. // ══════════════════════════════════════════════════════════════
  531. /// <summary>
  532. /// 校验订单明细行的物料编码是否与已生成工单的物料编码一致。
  533. /// 若不一致则说明物料信息已变更,不允许重排。
  534. /// </summary>
  535. private async Task ValidateMaterialNotChangedAsync(long orderId, long tenantId)
  536. {
  537. var mismatches = await _db.Ado.SqlQueryAsync<dynamic>(
  538. """
  539. SELECT e.entry_seq AS EntrySeq,
  540. e.item_number AS EntryItemNum,
  541. w.ItemNum AS WoItemNum
  542. FROM crm_seorderentry e
  543. INNER JOIN WorkOrdMaster w
  544. ON w.BusinessID = e.Id AND w.tenant_id = e.tenant_id
  545. AND IFNULL(w.IsActive, 0) = 1
  546. AND LOWER(TRIM(IFNULL(w.Status, ''))) <> 'c'
  547. WHERE e.seorder_id = @OrderId AND e.tenant_id = @TenantId AND e.IsDeleted = 0
  548. AND TRIM(IFNULL(e.item_number, '')) <> ''
  549. AND TRIM(IFNULL(w.ItemNum, '')) <> ''
  550. AND TRIM(e.item_number) <> TRIM(w.ItemNum)
  551. """,
  552. new SugarParameter("@OrderId", orderId),
  553. new SugarParameter("@TenantId", tenantId));
  554. if (mismatches.Count > 0)
  555. throw Oops.Oh("此订单行的物料信息有变更无法重排");
  556. }
  557. /// <summary>
  558. /// 当工单状态为下达(R)/投产(W)/暂停(S)时,更新对应领料单明细数据。
  559. /// 重新从 WorkOrdDetail 汇总物料需求,覆盖 NbrDetail 中的 QtyOrd。
  560. /// </summary>
  561. private async Task UpdatePickingListForActiveWorkOrderAsync(
  562. long tenantId, string workOrd, string account, List<string> warnings)
  563. {
  564. // 获取工单状态
  565. var status = await _db.Ado.GetStringAsync(
  566. """
  567. SELECT IFNULL(LOWER(TRIM(Status)), '') FROM WorkOrdMaster
  568. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd
  569. LIMIT 1
  570. """,
  571. new SugarParameter("@TenantId", tenantId),
  572. new SugarParameter("@WorkOrd", workOrd));
  573. // 仅处理 下达(R)/投产(W)/暂停(S) 状态的工单
  574. if (status is not ("r" or "w" or "s"))
  575. return;
  576. // 查找领料单主记录
  577. var nbrRows = await _db.Ado.SqlQueryAsync<NbrMasterRow>(
  578. """
  579. SELECT RecID, Nbr, `Domain` FROM NbrMaster
  580. WHERE tenant_id = @TenantId AND WorkOrd = @WorkOrd AND Type = 'SM'
  581. AND IFNULL(TransType, '') = ''
  582. AND IFNULL(IsActive, 0) = 1
  583. """,
  584. new SugarParameter("@TenantId", tenantId),
  585. new SugarParameter("@WorkOrd", workOrd));
  586. if (nbrRows.Count == 0)
  587. {
  588. warnings.Add($"工单 {workOrd} 状态为 {status.ToUpper()} 但无领料单,跳过领料单更新");
  589. return;
  590. }
  591. var now = DateTime.Now;
  592. // 从 WorkOrdDetail 汇总最新物料需求
  593. var details = await _db.Ado.SqlQueryAsync<PickDetailRow>(
  594. """
  595. SELECT
  596. d.ItemNum,
  597. SUM(d.QtyRequired) AS QtyRequired,
  598. MAX(IFNULL(d.UM, im.Um)) AS Unit,
  599. MAX(im.Descr) AS ItemName
  600. FROM WorkOrdDetail d
  601. LEFT JOIN ItemMaster im ON d.ItemNum = im.ItemNum AND im.tenant_id = d.tenant_id
  602. WHERE d.tenant_id = @TenantId AND d.WorkOrd = @WorkOrd AND IFNULL(d.IsActive, 0) = 1
  603. GROUP BY d.ItemNum
  604. HAVING SUM(d.QtyRequired) > 0
  605. ORDER BY d.ItemNum
  606. """,
  607. new SugarParameter("@TenantId", tenantId),
  608. new SugarParameter("@WorkOrd", workOrd));
  609. foreach (var nbr in nbrRows)
  610. {
  611. var domain = string.IsNullOrWhiteSpace(nbr.Domain) ? tenantId.ToString() : nbr.Domain!.Trim();
  612. if (domain.Length > 8) domain = domain[..8];
  613. // 加载当前领料单明细行(仅未关闭的)
  614. var existingDetails = await _db.Ado.SqlQueryAsync<NbrDetailRow>(
  615. """
  616. SELECT RecID, ItemNum, QtyOrd, QtyRec, CurrQtyOpened, Line
  617. FROM NbrDetail
  618. WHERE tenant_id = @TenantId AND Nbr = @Nbr AND Type = 'SM'
  619. AND IFNULL(IsActive, 0) = 1
  620. """,
  621. new SugarParameter("@TenantId", tenantId),
  622. new SugarParameter("@Nbr", nbr.Nbr));
  623. var detailMap = details.ToDictionary(d => d.ItemNum.Trim(), d => d);
  624. var existingMap = existingDetails.ToDictionary(d => (d.ItemNum ?? "").Trim(), d => d);
  625. // ── 1. 处理已有明细行:按 ItemNum 匹配 ──
  626. foreach (var existing in existingDetails)
  627. {
  628. var key = (existing.ItemNum ?? "").Trim();
  629. if (detailMap.TryGetValue(key, out var newDetail))
  630. {
  631. // 物料在工单明细中存在
  632. var newQty = newDetail.QtyRequired;
  633. if (existing.QtyRec > 0)
  634. {
  635. // 已发料:判断新需求数是否大于已发料数
  636. if (newQty > existing.QtyRec)
  637. {
  638. // 新需求 > 已发料 → 更新需求数
  639. await _db.Ado.ExecuteCommandAsync(
  640. """
  641. UPDATE NbrDetail
  642. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  643. UM = @UM, ItemName = @ItemName,
  644. UpdateUser = @User, UpdateTime = @Now
  645. WHERE RecID = @RecId
  646. """,
  647. new SugarParameter("@QtyOrd", newQty),
  648. new SugarParameter("@CurrQtyOpened", newQty),
  649. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  650. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  651. new SugarParameter("@User", account),
  652. new SugarParameter("@Now", now),
  653. new SugarParameter("@RecId", existing.RecID));
  654. }
  655. else
  656. {
  657. // 新需求 <= 已发料 → 关闭当前行
  658. await _db.Ado.ExecuteCommandAsync(
  659. """
  660. UPDATE NbrDetail
  661. SET Status = 'C', IsActive = 0,
  662. UpdateUser = @User, UpdateTime = @Now
  663. WHERE RecID = @RecId
  664. """,
  665. new SugarParameter("@User", account),
  666. new SugarParameter("@Now", now),
  667. new SugarParameter("@RecId", existing.RecID));
  668. }
  669. }
  670. else
  671. {
  672. // 未发料 → 直接修改需求数
  673. await _db.Ado.ExecuteCommandAsync(
  674. """
  675. UPDATE NbrDetail
  676. SET QtyOrd = @QtyOrd, CurrQtyOpened = @CurrQtyOpened,
  677. UM = @UM, ItemName = @ItemName,
  678. UpdateUser = @User, UpdateTime = @Now
  679. WHERE RecID = @RecId
  680. """,
  681. new SugarParameter("@QtyOrd", newQty),
  682. new SugarParameter("@CurrQtyOpened", newQty),
  683. new SugarParameter("@UM", (object?)newDetail.Unit ?? DBNull.Value),
  684. new SugarParameter("@ItemName", (object?)newDetail.ItemName ?? DBNull.Value),
  685. new SugarParameter("@User", account),
  686. new SugarParameter("@Now", now),
  687. new SugarParameter("@RecId", existing.RecID));
  688. }
  689. }
  690. else
  691. {
  692. // 物料明细中没有,但领料单明细有 → 关闭当前领料单明细行
  693. await _db.Ado.ExecuteCommandAsync(
  694. """
  695. UPDATE NbrDetail
  696. SET Status = 'C', IsActive = 0,
  697. UpdateUser = @User, UpdateTime = @Now
  698. WHERE RecID = @RecId
  699. """,
  700. new SugarParameter("@User", account),
  701. new SugarParameter("@Now", now),
  702. new SugarParameter("@RecId", existing.RecID));
  703. }
  704. }
  705. // ── 2. 新增:物料明细中有,领料单明细没有的 ──
  706. var nextDetailId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) + 1 FROM NbrDetail");
  707. short newLine = (short)(existingDetails.Count > 0 ? existingDetails.Max(d => d.Line) + 1 : 1);
  708. foreach (var d in details)
  709. {
  710. var key = d.ItemNum.Trim();
  711. if (existingMap.ContainsKey(key))
  712. continue; // 已处理
  713. await _db.Ado.ExecuteCommandAsync(
  714. """
  715. INSERT INTO NbrDetail (
  716. RecID, `Domain`, Type, Nbr, Line, ItemNum, Dimension1, Dimension2,
  717. LocationFrom, LocationTo, QtyFrom, QtyTo, QtyOrd, QtyRec,
  718. CurrQtyOpened, UM, WorkOrd, ItemName, Status,
  719. IsActive, IsConfirm, IsChanged, BusinessID, NbrRecID,
  720. CreateUser, CreateTime, UpdateUser, UpdateTime, tenant_id
  721. ) VALUES (
  722. @RecId, @Domain, 'SM', @Nbr, @Line, @ItemNum, '', '',
  723. '', '', 0, 0, @QtyOrd, 0,
  724. @CurrQtyOpened, @UM, @WorkOrd, @ItemName, '',
  725. 1, 0, 1, 0, @NbrRecId,
  726. @User, @Now, @User, @Now, @TenantId
  727. )
  728. """,
  729. new SugarParameter("@RecId", nextDetailId++),
  730. new SugarParameter("@Domain", domain),
  731. new SugarParameter("@Nbr", nbr.Nbr),
  732. new SugarParameter("@Line", newLine++),
  733. new SugarParameter("@ItemNum", d.ItemNum),
  734. new SugarParameter("@QtyOrd", d.QtyRequired),
  735. new SugarParameter("@CurrQtyOpened", d.QtyRequired),
  736. new SugarParameter("@UM", (object?)d.Unit ?? DBNull.Value),
  737. new SugarParameter("@WorkOrd", workOrd),
  738. new SugarParameter("@ItemName", (object?)d.ItemName ?? DBNull.Value),
  739. new SugarParameter("@NbrRecId", nbr.RecID),
  740. new SugarParameter("@User", account.Length > 24 ? account[..24] : account),
  741. new SugarParameter("@Now", now),
  742. new SugarParameter("@TenantId", tenantId));
  743. }
  744. // ── 3. 更新领料单主记录的更新时间和数量 ──
  745. await _db.Ado.ExecuteCommandAsync(
  746. """
  747. UPDATE NbrMaster
  748. SET QtyOrd = (SELECT IFNULL(SUM(QtyOrd), 0) FROM NbrDetail
  749. WHERE Nbr = @Nbr AND Type = 'SM' AND IFNULL(IsActive, 1) = 1),
  750. UpdateUser = @User, UpdateTime = @Now
  751. WHERE RecID = @RecId
  752. """,
  753. new SugarParameter("@Nbr", nbr.Nbr),
  754. new SugarParameter("@User", account),
  755. new SugarParameter("@Now", now),
  756. new SugarParameter("@RecId", nbr.RecID));
  757. }
  758. }
  759. /// <summary>
  760. /// 根据资源检查结果更新明细行系统建议交期(sys_capacity_date)。
  761. /// </summary>
  762. private async Task UpdateEntrySysCapacityDateAsync(long entryId, DateTime? kittingTime, string account)
  763. {
  764. await _db.Ado.ExecuteCommandAsync(
  765. """
  766. UPDATE crm_seorderentry
  767. SET sys_capacity_date = @CapacityDate,
  768. update_time = @Now
  769. WHERE Id = @Id AND IsDeleted = 0
  770. """,
  771. new SugarParameter("@CapacityDate", kittingTime ?? (object)DBNull.Value),
  772. new SugarParameter("@Now", DateTime.Now),
  773. new SugarParameter("@Id", entryId));
  774. }
  775. /// <summary>
  776. /// 根据订单及明细行生成交货单(出货计划 ShippingPlan / ShippingPlanDetail)。
  777. /// 若该订单已存在出货计划则更新明细,否则新建。
  778. /// </summary>
  779. private async Task GenerateShippingPlanFromOrderAsync(
  780. OrderWorkOrderGenerationService.OrderHeader order,
  781. List<OrderWorkOrderGenerationService.OrderEntryLine> entries,
  782. string account,
  783. List<string> warnings)
  784. {
  785. // 加载订单额外字段(客户名称、国家、日期)
  786. var orderInfo = await _db.Ado.SqlQueryAsync<OrderShippingInfoRow>(
  787. """
  788. SELECT custom_name AS CustomName, country AS Country, date AS OrderDate
  789. FROM crm_seorder
  790. WHERE Id = @Id AND tenant_id = @TenantId AND IsDeleted = 0
  791. LIMIT 1
  792. """,
  793. new SugarParameter("@Id", order.Id),
  794. new SugarParameter("@TenantId", order.TenantId));
  795. var info = orderInfo.FirstOrDefault() ?? new OrderShippingInfoRow();
  796. // 检查是否已存在该订单的出货计划明细
  797. var existingPlanId = await _db.Ado.GetIntAsync(
  798. """
  799. SELECT IFNULL(MAX(plan_id), 0) FROM ShippingPlanDetail
  800. WHERE tenant_id = @TenantId AND seorder_id = @OrderId AND IFNULL(IsActive, 1) = 1
  801. """,
  802. new SugarParameter("@TenantId", order.TenantId),
  803. new SugarParameter("@OrderId", order.Id));
  804. var now = DateTime.Now;
  805. var domain = order.TenantId.ToString();
  806. if (domain.Length > 8) domain = domain[..8];
  807. if (existingPlanId == 0)
  808. {
  809. // ── 新建出货计划主表 ──
  810. var planId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) + 1 FROM ShippingPlan");
  811. await _db.Ado.ExecuteCommandAsync(
  812. """
  813. INSERT INTO ShippingPlan (
  814. RecID, `Domain`, LotSerial, ShippingDate, ShippingSite,
  815. Consignee, Priority, Status, Remark,
  816. IsActive, IsConfirm, CreateUser, CreateTime, tenant_id
  817. ) VALUES (
  818. @PlanId, @Domain, @LotSerial, @ShippingDate, '',
  819. @Consignee, 0, '', '3级计划重排自动生成',
  820. 1, 1, @User, @Now, @TenantId
  821. )
  822. """,
  823. new SugarParameter("@PlanId", planId),
  824. new SugarParameter("@Domain", domain),
  825. new SugarParameter("@LotSerial", order.BillNo ?? string.Empty),
  826. new SugarParameter("@ShippingDate", info.OrderDate ?? (object)DBNull.Value),
  827. new SugarParameter("@Consignee", (object?)info.CustomName ?? DBNull.Value),
  828. new SugarParameter("@User", account),
  829. new SugarParameter("@Now", now),
  830. new SugarParameter("@TenantId", order.TenantId));
  831. // ── 新建出货计划明细 ──
  832. var nextDetailId = await _db.Ado.GetIntAsync("SELECT IFNULL(MAX(RecID), 0) + 1 FROM ShippingPlanDetail");
  833. foreach (var entry in entries)
  834. {
  835. await _db.Ado.ExecuteCommandAsync(
  836. """
  837. INSERT INTO ShippingPlanDetail (
  838. RecID, `Domain`, plan_id, OrdNbr, bill_no,
  839. ItemNum, ItemName, Specification, Qty,
  840. OrdDate, Country, CustomNo, CustomName,
  841. seorder_id, sentry_id, Remark, Status,
  842. IsActive, IsConfirm, CreateUser, CreateTime, tenant_id
  843. ) VALUES (
  844. @RecId, @Domain, @PlanId, @OrdNbr, @BillNo,
  845. @ItemNum, @ItemName, @Spec, @Qty,
  846. @OrdDate, @Country, @CustomNo, @CustomName,
  847. @SeOrderId, @SentryId, '', '',
  848. 1, 1, @User, @Now, @TenantId
  849. )
  850. """,
  851. new SugarParameter("@RecId", nextDetailId++),
  852. new SugarParameter("@Domain", domain),
  853. new SugarParameter("@PlanId", planId),
  854. new SugarParameter("@OrdNbr", order.BillNo ?? string.Empty),
  855. new SugarParameter("@BillNo", entry.BillNo ?? order.BillNo ?? string.Empty),
  856. new SugarParameter("@ItemNum", (object?)entry.ItemNumber ?? DBNull.Value),
  857. new SugarParameter("@ItemName", (object?)entry.ItemName ?? DBNull.Value),
  858. new SugarParameter("@Spec", (object?)entry.Specification ?? DBNull.Value),
  859. new SugarParameter("@Qty", entry.Qty ?? 0),
  860. new SugarParameter("@OrdDate", entry.PlanDate ?? (object)DBNull.Value),
  861. new SugarParameter("@Country", (object?)info.Country ?? DBNull.Value),
  862. new SugarParameter("@CustomNo", (object?)order.CustomNo ?? DBNull.Value),
  863. new SugarParameter("@CustomName", (object?)info.CustomName ?? DBNull.Value),
  864. new SugarParameter("@SeOrderId", order.Id),
  865. new SugarParameter("@SentryId", entry.Id),
  866. new SugarParameter("@User", account),
  867. new SugarParameter("@Now", now),
  868. new SugarParameter("@TenantId", order.TenantId));
  869. }
  870. warnings.Add($"已自动生成出货计划(ID={planId}),共 {entries.Count} 行明细");
  871. }
  872. else
  873. {
  874. // ── 更新已有出货计划明细 ──
  875. await _db.Ado.ExecuteCommandAsync(
  876. """
  877. UPDATE ShippingPlanDetail
  878. SET CustomName = @CustomName, Country = @Country,
  879. UpdateUser = @User, UpdateTime = @Now
  880. WHERE tenant_id = @TenantId AND seorder_id = @OrderId AND IFNULL(IsActive, 1) = 1
  881. """,
  882. new SugarParameter("@CustomName", (object?)info.CustomName ?? DBNull.Value),
  883. new SugarParameter("@Country", (object?)info.Country ?? DBNull.Value),
  884. new SugarParameter("@User", account),
  885. new SugarParameter("@Now", now),
  886. new SugarParameter("@TenantId", order.TenantId),
  887. new SugarParameter("@OrderId", order.Id));
  888. warnings.Add($"已更新出货计划(ID={existingPlanId})的明细数据");
  889. }
  890. }
  891. // ══════════════════════════════════════════════════════════════
  892. // 3级计划重排 — 内部 DTO
  893. // ══════════════════════════════════════════════════════════════
  894. private sealed class NbrMasterRow
  895. {
  896. public int RecID { get; set; }
  897. public string Nbr { get; set; } = string.Empty;
  898. public string? Domain { get; set; }
  899. }
  900. private sealed class NbrDetailRow
  901. {
  902. public int RecID { get; set; }
  903. public string? ItemNum { get; set; }
  904. public decimal QtyOrd { get; set; }
  905. public decimal QtyRec { get; set; }
  906. public decimal CurrQtyOpened { get; set; }
  907. public short Line { get; set; }
  908. }
  909. private sealed class PickDetailRow
  910. {
  911. public string ItemNum { get; set; } = string.Empty;
  912. public decimal QtyRequired { get; set; }
  913. public string? Unit { get; set; }
  914. public string? ItemName { get; set; }
  915. }
  916. private sealed class OrderShippingInfoRow
  917. {
  918. public string? CustomName { get; set; }
  919. public string? Country { get; set; }
  920. public DateTime? OrderDate { get; set; }
  921. }
  922. }