DeliveryScheduleService.cs 64 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484
  1. using Admin.NET.Plugin.AiDOP.Rule;
  2. using Yitter.IdGenerator;
  3. namespace Admin.NET.Plugin.AiDOP.Supply;
  4. /// <summary>
  5. /// 物料交货计划服务
  6. /// </summary>
  7. [ApiDescriptionSettings(Order = 306, Description = "物料交货计划")]
  8. [Route("api/Supply")]
  9. [AllowAnonymous]
  10. [NonUnify]
  11. public class DeliveryScheduleService : IDynamicApiController, ITransient
  12. {
  13. private readonly ISqlSugarClient _db;
  14. private readonly SqlSugarRepository<PolistDeliverySchedule> _rep;
  15. private readonly UserManager _userManager;
  16. private readonly NumberRuleService _numberRuleService;
  17. private readonly PurchaseRequestMergeService _purchaseRequestMergeService;
  18. private readonly PurchaseOrderTransferService _purchaseOrderTransferService;
  19. private readonly PurchaseRequestExternalPushService _purchaseRequestExternalPushService;
  20. private readonly S3OperationLogService _operationLogService;
  21. private readonly RuleConfigService _ruleConfigService;
  22. public DeliveryScheduleService(
  23. ISqlSugarClient db,
  24. SqlSugarRepository<PolistDeliverySchedule> rep,
  25. UserManager userManager,
  26. NumberRuleService numberRuleService,
  27. PurchaseRequestMergeService purchaseRequestMergeService,
  28. PurchaseOrderTransferService purchaseOrderTransferService,
  29. PurchaseRequestExternalPushService purchaseRequestExternalPushService,
  30. S3OperationLogService operationLogService,
  31. RuleConfigService ruleConfigService)
  32. {
  33. _db = db;
  34. _rep = rep;
  35. _userManager = userManager;
  36. _numberRuleService = numberRuleService;
  37. _purchaseRequestMergeService = purchaseRequestMergeService;
  38. _purchaseOrderTransferService = purchaseOrderTransferService;
  39. _purchaseRequestExternalPushService = purchaseRequestExternalPushService;
  40. _operationLogService = operationLogService;
  41. _ruleConfigService = ruleConfigService;
  42. }
  43. [DisplayName("物料交货计划列表")]
  44. [HttpGet("delivery-schedule/list")]
  45. public async Task<object> GetList([FromQuery] DeliveryScheduleListInput input)
  46. {
  47. var page = input.Page <= 0 ? 1 : input.Page;
  48. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  49. var offset = (page - 1) * pageSize;
  50. //if (await ShouldUseStdAsync())
  51. //{
  52. //return await GetStdListAsync(input, page, pageSize, offset);
  53. //}
  54. var pars = new List<SugarParameter>();
  55. var where = new List<string> { "ss.isactive = 1" };
  56. if (_userManager.TenantId > 0)
  57. {
  58. where.Add("ss.tenant_id = @TenantId");
  59. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  60. }
  61. if (!string.IsNullOrWhiteSpace(input.PoNumber))
  62. {
  63. where.Add("ss.ponumber LIKE @PoNumber");
  64. pars.Add(new SugarParameter("@PoNumber", $"%{input.PoNumber.Trim()}%"));
  65. }
  66. if (!string.IsNullOrWhiteSpace(input.DsNum))
  67. {
  68. where.Add("ss.dsnum LIKE @DsNum");
  69. pars.Add(new SugarParameter("@DsNum", $"%{input.DsNum.Trim()}%"));
  70. }
  71. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  72. {
  73. where.Add("ss.itemnum LIKE @ItemNum");
  74. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  75. }
  76. if (!string.IsNullOrWhiteSpace(input.Supplier))
  77. {
  78. where.Add("(ss.suppliercode LIKE @Supplier OR ss.supplier LIKE @Supplier)");
  79. pars.Add(new SugarParameter("@Supplier", $"%{input.Supplier.Trim()}%"));
  80. }
  81. if (!string.IsNullOrWhiteSpace(input.Status))
  82. {
  83. where.Add("ss.status = @Status");
  84. pars.Add(new SugarParameter("@Status", input.Status.Trim()));
  85. }
  86. var orderBy = BuildListOrderBy(input.SortField, input.SortOrder);
  87. var fromSql = $"""
  88. FROM srm_polist_ds ss
  89. LEFT JOIN ItemMaster im ON ss.itemnum = im.ItemNum
  90. WHERE {string.Join(" AND ", where)}
  91. """;
  92. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  93. var list = await _db.Ado.SqlQueryAsync<DeliveryScheduleListRow>(
  94. $"""
  95. SELECT
  96. ss.Id AS Id,
  97. ss.domain AS Domain,
  98. ss.icdsid AS IcdsId,
  99. ss.dsnum AS DsNum,
  100. ss.status AS Status,
  101. ss.itemnum AS ItemNum,
  102. im.Descr AS Descr,
  103. ss.um AS Um,
  104. ss.purgroup AS PurGroup,
  105. ss.suppliercode AS SupplierCode,
  106. ss.supplier AS Supplier,
  107. ss.submitdate AS SubmitDate,
  108. ss.requestdate AS RequestDate,
  109. ss.needdate AS NeedDate,
  110. ss.ponumber AS PoNumber,
  111. ss.poline AS PoLine,
  112. ss.schedqty AS SchedQty,
  113. ss.lastsentdate AS LastSentDate,
  114. ss.lastsentqty AS LastSentQty,
  115. ss.sentqty AS SentQty,
  116. ss.restqty AS RestQty
  117. {fromSql}
  118. ORDER BY {orderBy}
  119. LIMIT {pageSize} OFFSET {offset}
  120. """,
  121. pars);
  122. return new { total, page, pageSize, list };
  123. }
  124. private async Task<bool> ShouldUseStdAsync()
  125. {
  126. return await _db.Ado.GetIntAsync("SELECT COUNT(1) FROM mdp_std_delivery_schedule LIMIT 1") > 0;
  127. }
  128. private async Task<object> GetStdListAsync(DeliveryScheduleListInput input, int page, int pageSize, int offset)
  129. {
  130. var pars = new List<SugarParameter>();
  131. var where = new List<string> { "1=1" };
  132. if (_userManager.TenantId > 0)
  133. {
  134. where.Add("ss.tenant_id = @TenantId");
  135. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  136. }
  137. if (!string.IsNullOrWhiteSpace(input.PoNumber))
  138. {
  139. where.Add("ss.po_no LIKE @PoNumber");
  140. pars.Add(new SugarParameter("@PoNumber", $"%{input.PoNumber.Trim()}%"));
  141. }
  142. if (!string.IsNullOrWhiteSpace(input.DsNum))
  143. {
  144. where.Add("ss.delivery_plan_no LIKE @DsNum");
  145. pars.Add(new SugarParameter("@DsNum", $"%{input.DsNum.Trim()}%"));
  146. }
  147. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  148. {
  149. where.Add("ss.item_code LIKE @ItemNum");
  150. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  151. }
  152. if (!string.IsNullOrWhiteSpace(input.Supplier))
  153. {
  154. where.Add("(ss.supplier_code LIKE @Supplier OR ss.supplier_name LIKE @Supplier)");
  155. pars.Add(new SugarParameter("@Supplier", $"%{input.Supplier.Trim()}%"));
  156. }
  157. if (!string.IsNullOrWhiteSpace(input.Status))
  158. {
  159. where.Add("ss.status = @Status");
  160. pars.Add(new SugarParameter("@Status", input.Status.Trim()));
  161. }
  162. var orderBy = BuildStdListOrderBy(input.SortField, input.SortOrder);
  163. var fromSql = $"FROM mdp_std_delivery_schedule ss LEFT JOIN mdp_std_item im ON ss.tenant_id=im.tenant_id AND ss.item_code=im.item_code WHERE {string.Join(" AND ", where)}";
  164. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) {fromSql}", pars);
  165. var list = await _db.Ado.SqlQueryAsync<DeliveryScheduleListRow>(
  166. $"""
  167. SELECT
  168. ss.id AS Id,
  169. NULL AS Domain,
  170. NULL AS IcdsId,
  171. ss.delivery_plan_no AS DsNum,
  172. ss.status AS Status,
  173. ss.item_code AS ItemNum,
  174. im.item_name AS Descr,
  175. NULL AS Um,
  176. NULL AS PurGroup,
  177. ss.supplier_code AS SupplierCode,
  178. ss.supplier_name AS Supplier,
  179. ss.submit_date AS SubmitDate,
  180. ss.request_date AS RequestDate,
  181. ss.need_date AS NeedDate,
  182. ss.po_no AS PoNumber,
  183. CAST(ss.po_line AS SIGNED) AS PoLine,
  184. ss.schedule_qty AS SchedQty,
  185. ss.last_sent_date AS LastSentDate,
  186. ss.sent_qty AS LastSentQty,
  187. ss.sent_qty AS SentQty,
  188. ss.rest_qty AS RestQty
  189. {fromSql}
  190. ORDER BY {orderBy}
  191. LIMIT {pageSize} OFFSET {offset}
  192. """,
  193. pars);
  194. return new { total, page, pageSize, list, source = "mdp_std_delivery_schedule" };
  195. }
  196. private static string BuildStdListOrderBy(string? sortField, string? sortOrder)
  197. {
  198. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  199. return sortField?.ToLowerInvariant() switch
  200. {
  201. "dsnum" => $"ss.delivery_plan_no {dir}",
  202. "status" => $"ss.status {dir}",
  203. "itemnum" => $"ss.item_code {dir}",
  204. "suppliercode" => $"ss.supplier_code {dir}",
  205. "supplier" => $"ss.supplier_name {dir}",
  206. "submitdate" => $"ss.submit_date {dir}",
  207. "requestdate" => $"ss.request_date {dir}",
  208. "needdate" => $"ss.need_date {dir}",
  209. "ponumber" => $"ss.po_no {dir}",
  210. "poline" => $"ss.po_line {dir}",
  211. "schedqty" => $"ss.schedule_qty {dir}",
  212. "sentqty" => $"ss.sent_qty {dir}",
  213. "restqty" => $"ss.rest_qty {dir}",
  214. _ => "ss.need_date DESC, ss.delivery_plan_no DESC"
  215. };
  216. }
  217. [DisplayName("生成交货单")]
  218. [HttpPost("delivery-schedule/generate")]
  219. public async Task<object> Generate()
  220. {
  221. try
  222. {
  223. var result = await GenerateDeliverySchedulesFromDemandAsync();
  224. await _operationLogService.WriteAsync(new S3OperationLogDto
  225. {
  226. Action = "S3_DELIVERY_GENERATE",
  227. DisplayTitle = "S3 生成交货单",
  228. Message = result.Message,
  229. Result = result,
  230. Success = true
  231. });
  232. return result;
  233. }
  234. catch (Exception ex)
  235. {
  236. await _operationLogService.WriteAsync(new S3OperationLogDto
  237. {
  238. Action = "S3_DELIVERY_GENERATE",
  239. DisplayTitle = "S3 生成交货单失败",
  240. Message = ex.Message,
  241. Exception = ex.ToString(),
  242. Success = false
  243. });
  244. throw;
  245. }
  246. }
  247. [DisplayName("批量添加交货单")]
  248. [HttpPost("delivery-schedule/batch-generate")]
  249. public async Task<object> BatchGenerate([FromBody] DeliveryScheduleGenerateInput input)
  250. {
  251. if (string.IsNullOrWhiteSpace(input.PoNumber))
  252. throw Oops.Oh("采购单号不能为空");
  253. return await GenerateDeliverySchedulesFromPurchaseOrdersAsync(input.PoNumber.Trim());
  254. }
  255. [DisplayName("发布交货单")]
  256. [HttpPost("delivery-schedule/publish")]
  257. public async Task<object> Publish([FromBody] DeliveryScheduleBatchIdsInput input)
  258. {
  259. var ids = ParseIds(input.Ids);
  260. if (!ids.Any()) throw Oops.Oh("请勾选要发布的数据");
  261. var account = _userManager.Account ?? "system";
  262. var tenantId = _userManager.TenantId;
  263. var affected = await _db.Updateable<PolistDeliverySchedule>()
  264. .SetColumns(x => new PolistDeliverySchedule
  265. {
  266. Status = "P",
  267. SubmitDate = DateTime.Now,
  268. UpdateUser = account,
  269. UpdateTime = DateTime.Now
  270. })
  271. .Where(x => ids.Contains(x.Id) && x.IsActive == 1 && x.Status == "N" && (tenantId <= 0 || x.TenantId == tenantId))
  272. .ExecuteCommandAsync();
  273. return new { affected, message = affected > 0 ? "发布成功" : "无可发布数据" };
  274. }
  275. [DisplayName("取消发布交货单")]
  276. [HttpPost("delivery-schedule/unpublish/{id:long}")]
  277. public async Task<object> UnPublish(long id)
  278. {
  279. var tenantId = _userManager.TenantId;
  280. var row = await _rep.GetFirstAsync(x => x.Id == id && x.IsActive == 1 && (tenantId <= 0 || x.TenantId == tenantId)) ?? throw Oops.Oh("交货单不存在");
  281. row.Status = "N";
  282. row.UpdateUser = _userManager.Account;
  283. row.UpdateTime = DateTime.Now;
  284. await _rep.UpdateAsync(row);
  285. return new { message = "取消发布成功" };
  286. }
  287. [DisplayName("作废交货单")]
  288. [HttpPost("delivery-schedule/cancel/{id:long}")]
  289. public async Task<object> Cancel(long id)
  290. {
  291. var tenantId = _userManager.TenantId;
  292. var row = await _rep.GetFirstAsync(x => x.Id == id && x.IsActive == 1 && (tenantId <= 0 || x.TenantId == tenantId)) ?? throw Oops.Oh("交货单不存在");
  293. row.Status = "C";
  294. row.UpdateUser = _userManager.Account;
  295. row.UpdateTime = DateTime.Now;
  296. await _rep.UpdateAsync(row);
  297. return new { message = "作废成功" };
  298. }
  299. [DisplayName("选择采购单分页")]
  300. [HttpGet("delivery-schedule/purchase-order/page")]
  301. public async Task<object> GetPurchaseOrderPage([FromQuery] PurchaseOrderSelectInput input)
  302. {
  303. var page = input.Page <= 0 ? 1 : input.Page;
  304. var pageSize = input.PageSize <= 0 ? 10 : input.PageSize;
  305. var offset = (page - 1) * pageSize;
  306. var pars = new List<SugarParameter>();
  307. var where = new List<string>
  308. {
  309. "IFNULL(m.Status,'') <> 'C'",
  310. "IFNULL(d.Status,'') <> 'C'",
  311. "(IFNULL(ds.dsnum, '') = '' OR (ds.status = 'C' AND d.QtyOrded - d.RctQty > 0))"
  312. };
  313. if (_userManager.TenantId > 0)
  314. {
  315. where.Add("m.tenant_id = @TenantId");
  316. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  317. }
  318. if (!string.IsNullOrWhiteSpace(input.PurOrd))
  319. {
  320. where.Add("m.PurOrd LIKE @PurOrd");
  321. pars.Add(new SugarParameter("@PurOrd", $"%{input.PurOrd.Trim()}%"));
  322. }
  323. if (!string.IsNullOrWhiteSpace(input.ItemNum))
  324. {
  325. where.Add("d.ItemNum LIKE @ItemNum");
  326. pars.Add(new SugarParameter("@ItemNum", $"%{input.ItemNum.Trim()}%"));
  327. }
  328. if (!string.IsNullOrWhiteSpace(input.Supp))
  329. {
  330. where.Add("(m.Supp LIKE @Supp OR s.Name LIKE @Supp)");
  331. pars.Add(new SugarParameter("@Supp", $"%{input.Supp.Trim()}%"));
  332. }
  333. var orderBy = BuildPurchaseOrderOrderBy(input.SortField, input.SortOrder);
  334. var fromSql = $"""
  335. FROM PurOrdMaster m
  336. INNER JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  337. LEFT JOIN srm_polist_ds ds ON d.PurOrd = ds.ponumber AND d.Line = ds.poline AND ds.isactive = 1
  338. LEFT JOIN ConsigneeAddressMaster s ON m.Supp = s.Address AND s.Typed = 'Supp'
  339. WHERE {string.Join(" AND ", where)}
  340. """;
  341. var distinctSql = $"SELECT DISTINCT m.PurOrd, d.Line, d.ItemNum {fromSql}";
  342. var total = await _db.Ado.GetIntAsync($"SELECT COUNT(1) FROM ({distinctSql}) t", pars);
  343. var list = await _db.Ado.SqlQueryAsync<PurchaseOrderSelectRow>(
  344. $"""
  345. SELECT DISTINCT
  346. m.PurOrd AS PurOrd,
  347. d.Line AS Line,
  348. d.ItemNum AS ItemNum,
  349. d.UM AS Um,
  350. m.Supp AS Supp,
  351. s.Name AS Name,
  352. d.DueDate AS DueDate,
  353. d.QtyOrded AS QtyOrded,
  354. m.Buyer AS Buyer
  355. {fromSql}
  356. ORDER BY {orderBy}
  357. LIMIT {pageSize} OFFSET {offset}
  358. """,
  359. pars);
  360. return new { total, page, pageSize, list };
  361. }
  362. private async Task<DeliveryScheduleGenerateResult> GenerateDeliverySchedulesFromDemandAsync()
  363. {
  364. var account = _userManager.Account ?? "system";
  365. var now = DateTime.Now;
  366. var optTime = now;
  367. var ruleResult = await _ruleConfigService.ResolveS3DeliveryGenerateOptionsAsync(_userManager.TenantId, null);
  368. if (!ruleResult.Success)
  369. {
  370. throw Oops.Oh($"S3交货单生成规则配置无效:{string.Join(";", ruleResult.Errors)}");
  371. }
  372. var rules = ruleResult.Options;
  373. var ruleSnapshot = _ruleConfigService.CreateSnapshot("S3", "DELIVERY_GENERATE", _userManager.TenantId, null, ruleResult, now);
  374. var pars = new List<SugarParameter>();
  375. var where = new List<string>
  376. {
  377. "IFNULL(a.status,'') = 'P'",
  378. "IFNULL(a.tosechedqty,0) > 0",
  379. "(IFNULL(a.ishistoryversion,'') = '' OR a.ishistoryversion = 'N')",
  380. "IFNULL(a.IsDeleted,0)=0"
  381. };
  382. if (_userManager.TenantId > 0)
  383. {
  384. where.Add("a.tenant_id = @TenantId");
  385. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  386. }
  387. var demands = await _db.Ado.SqlQueryAsync<DemandScheduleCandidateRow>(
  388. $"""
  389. SELECT
  390. a.Id AS IcdsId,
  391. IFNULL(TRIM(a.itemnum),'') AS ItemNum,
  392. a.requestdate AS RequestDate,
  393. a.arrivaldate AS ArrivalDate,
  394. IFNULL(a.tosechedqty,0) AS DemandQty,
  395. a.tenant_id AS TenantId,
  396. a.factory_id AS FactoryId,
  397. a.company_id AS CompanyId,
  398. IFNULL(TRIM(im.PurMfg),'') AS PurMfg,
  399. IFNULL(TRIM(im.UM),'') AS Um,
  400. IFNULL(TRIM(im.Descr),'') AS ItemName,
  401. ic.Id AS IcItemId
  402. FROM ic_demandschedule a
  403. LEFT JOIN ItemMaster im ON a.itemnum = im.ItemNum
  404. LEFT JOIN ic_item ic ON a.itemnum = ic.number AND (a.tenant_id = ic.tenant_id OR a.tenant_id IS NULL)
  405. WHERE {string.Join(" AND ", where)}
  406. ORDER BY a.arrivaldate, a.requestdate, a.Id
  407. """,
  408. pars);
  409. if (demands.Count == 0)
  410. {
  411. return new DeliveryScheduleGenerateResult
  412. {
  413. DemandCount = 0,
  414. CandidateCount = 0,
  415. CreatedCount = 0,
  416. RuleSnapshot = ruleSnapshot,
  417. Message = "无已发布且待生成交货单的需求计划"
  418. };
  419. }
  420. var demandIds = demands.Select(x => x.IcdsId).Distinct().ToList();
  421. var existingUsages = await QueryExistingScheduleUsagesAsync(demandIds);
  422. var itemNums = demands.Select(x => x.ItemNum).Where(x => !string.IsNullOrWhiteSpace(x)).Distinct().ToList();
  423. var sourceLists = await QuerySourceListsAsync(itemNums);
  424. var supplierCodes = sourceLists.Select(x => x.SupplierCode).Where(x => !string.IsNullOrWhiteSpace(x)).Distinct().ToList();
  425. var poLines = supplierCodes.Count == 0
  426. ? new List<PurchaseOrderAvailableRow>()
  427. : await QueryAvailablePurchaseOrderLinesAsync(itemNums, supplierCodes, rules);
  428. var exceptions = new List<DeliveryExceptionMaster>();
  429. var schedules = new List<PolistDeliverySchedule>();
  430. var purchaseRequests = new List<PurchaseRequestMain>();
  431. var prMergeReducedCount = 0;
  432. var purchaseOrderTransferResult = new PurchaseOrderTransferResult();
  433. var externalPushResult = new PurchaseRequestExternalPushResult();
  434. var allocatedByPoLine = new Dictionary<string, decimal>();
  435. var skippedCount = 0;
  436. foreach (var demand in demands)
  437. {
  438. if (string.IsNullOrWhiteSpace(demand.ItemNum))
  439. {
  440. skippedCount++;
  441. AddDeliveryException(exceptions, demand, optTime, "需求计划缺少物料编码,无法生成交货单", demand.DemandQty);
  442. continue;
  443. }
  444. if (!string.Equals(demand.PurMfg, "P", StringComparison.OrdinalIgnoreCase))
  445. {
  446. skippedCount++;
  447. AddDeliveryException(exceptions, demand, optTime, $"物料[{demand.ItemNum}]不是采购件,不需要生成交货单", demand.DemandQty);
  448. continue;
  449. }
  450. var usedQty = existingUsages.TryGetValue(demand.IcdsId, out var used) ? used : 0m;
  451. var demandRestQty = demand.DemandQty - usedQty;
  452. if (demandRestQty <= 0)
  453. {
  454. skippedCount++;
  455. continue;
  456. }
  457. var itemSources = sourceLists
  458. .Where(x => string.Equals(x.ItemNum, demand.ItemNum, StringComparison.OrdinalIgnoreCase))
  459. .OrderByDescending(x => x.QuotaRate)
  460. .ToList();
  461. if (itemSources.Count == 0)
  462. {
  463. skippedCount++;
  464. AddDeliveryException(exceptions, demand, optTime, $"物料交货计划[{demand.ItemNum}:{FormatDate(demand.RequestDate)}]没有维护货源清单,无法转交货单", demandRestQty);
  465. continue;
  466. }
  467. var quotaTotal = itemSources.Sum(x => x.QuotaRate);
  468. if (Math.Abs(quotaTotal - rules.QuotaRequiredTotal) > rules.QuotaTolerance)
  469. {
  470. skippedCount++;
  471. AddDeliveryException(exceptions, demand, optTime, $"物料交货计划[{demand.ItemNum}:{FormatDate(demand.RequestDate)}]的货源清单供应商配额之和不为{rules.QuotaRequiredTotal:0.####}%,无法转交货单", demandRestQty);
  472. continue;
  473. }
  474. var allocatedForDemand = 0m;
  475. for (var i = 0; i < itemSources.Count; i++)
  476. {
  477. var source = itemSources[i];
  478. var remainingDemand = demandRestQty - allocatedForDemand;
  479. if (remainingDemand <= 0) break;
  480. var supplierNeedQty = i == itemSources.Count - 1
  481. ? remainingDemand
  482. : Math.Min(remainingDemand, Math.Ceiling(demandRestQty * source.QuotaRate / 100m));
  483. if (supplierNeedQty <= 0) continue;
  484. var supplierAllocated = AllocateDemandToPurchaseLines(
  485. demand,
  486. source,
  487. poLines,
  488. allocatedByPoLine,
  489. supplierNeedQty,
  490. remainingDemand,
  491. schedules,
  492. now,
  493. account,
  494. rules);
  495. allocatedForDemand += supplierAllocated;
  496. var shortageQty = supplierNeedQty - supplierAllocated;
  497. if (shortageQty > 0)
  498. {
  499. var prCreated = rules.ShortageCreatePr && TryAddPurchaseRequest(purchaseRequests, demand, source, shortageQty, now, account, rules);
  500. AddDeliveryException(
  501. exceptions,
  502. demand,
  503. optTime,
  504. prCreated
  505. ? $"物料交货计划[{demand.ItemNum}:{FormatDate(demand.RequestDate)}]对应供应商[{source.SupplierName}]采购订单不足,已生成采购申请,后续待合并/转单/推送"
  506. : rules.ShortageCreatePr
  507. ? $"物料交货计划[{demand.ItemNum}:{FormatDate(demand.RequestDate)}]对应供应商[{source.SupplierName}]采购订单不足,需补充 {shortageQty:0.####};因物料或供应商主键缺失,未自动生成PR"
  508. : $"物料交货计划[{demand.ItemNum}:{FormatDate(demand.RequestDate)}]对应供应商[{source.SupplierName}]采购订单不足,需补充 {shortageQty:0.####};规则配置关闭自动生成PR",
  509. shortageQty);
  510. }
  511. }
  512. }
  513. if (schedules.Count == 0 && exceptions.Count == 0)
  514. {
  515. return new DeliveryScheduleGenerateResult
  516. {
  517. DemandCount = demands.Count,
  518. CandidateCount = 0,
  519. CreatedCount = 0,
  520. SkippedCount = skippedCount,
  521. RuleSnapshot = ruleSnapshot,
  522. Message = "没有可生成交货单的需求计划"
  523. };
  524. }
  525. try
  526. {
  527. _db.Ado.BeginTran();
  528. if (purchaseRequests.Count > 0 && rules.EnablePrMerge)
  529. {
  530. var mergeResult = _purchaseRequestMergeService.MergeGeneratedRequests(purchaseRequests);
  531. purchaseRequests = mergeResult.Requests;
  532. prMergeReducedCount = mergeResult.ReducedCount;
  533. }
  534. if (purchaseRequests.Count > 0)
  535. {
  536. await AssignPurchaseRequestNumbersAsync(purchaseRequests, account);
  537. }
  538. if (rules.EnableRequireGoodsToPo && purchaseRequests.Any(x => x.IsRequireGoods == 1))
  539. {
  540. purchaseOrderTransferResult = await _purchaseOrderTransferService.TransferGeneratedRequireGoodsAsync(purchaseRequests, account);
  541. }
  542. if (rules.EnableExternalPushTracking && purchaseRequests.Any(x => x.IsRequireGoods == 0))
  543. {
  544. externalPushResult = await _purchaseRequestExternalPushService.CreateQadTrackingForGeneratedRequestsAsync(purchaseRequests, account);
  545. }
  546. if (schedules.Count > 0)
  547. {
  548. var dsNumbersByKey = new Dictionary<string, string>();
  549. foreach (var group in schedules.GroupBy(x => x.Domain ?? string.Empty))
  550. {
  551. var groupRows = group.ToList();
  552. var dsNumbers = await _numberRuleService.NextBatchInCurrentTransactionAsync("M8", group.Key, groupRows.Count, account);
  553. if (dsNumbers.Count < groupRows.Count || dsNumbers.Any(string.IsNullOrWhiteSpace))
  554. throw Oops.Oh($"当前交货计划号生成失败,请检查交货计划编号规则维护。Domain={group.Key}");
  555. for (var index = 0; index < groupRows.Count; index++)
  556. {
  557. dsNumbersByKey[BuildScheduleTempKey(groupRows[index])] = dsNumbers[index].Trim();
  558. }
  559. }
  560. foreach (var schedule in schedules)
  561. {
  562. schedule.DsNum = dsNumbersByKey[BuildScheduleTempKey(schedule)];
  563. }
  564. await _db.Insertable(schedules).ExecuteCommandAsync();
  565. await UpsertDeliveryScheduleMdpAsync(schedules, now);
  566. }
  567. if (purchaseRequests.Count > 0)
  568. {
  569. await InsertPurchaseRequestsAsync(purchaseRequests);
  570. }
  571. if (exceptions.Count > 0)
  572. {
  573. await _db.Insertable(exceptions).ExecuteCommandAsync();
  574. }
  575. _db.Ado.CommitTran();
  576. }
  577. catch
  578. {
  579. _db.Ado.RollbackTran();
  580. throw;
  581. }
  582. return new DeliveryScheduleGenerateResult
  583. {
  584. DemandCount = demands.Count,
  585. CandidateCount = schedules.Count,
  586. CreatedCount = schedules.Count,
  587. ExceptionCount = exceptions.Count,
  588. PurchaseRequestCount = purchaseRequests.Count,
  589. PurchaseRequestMergeReducedCount = prMergeReducedCount,
  590. PurchaseOrderCount = purchaseOrderTransferResult.CreatedOrderCount,
  591. PurchaseOrderLineCount = purchaseOrderTransferResult.TransferredPrCount,
  592. QadTrackingCount = externalPushResult.TrackingCount,
  593. SkippedCount = skippedCount,
  594. DsNums = schedules.Select(x => x.DsNum).Where(x => !string.IsNullOrWhiteSpace(x)).Take(20).ToList(),
  595. RuleSnapshot = ruleSnapshot,
  596. Message = schedules.Count > 0
  597. ? $"生成交货单成功,共生成 {schedules.Count} 条;自动生成PR {purchaseRequests.Count} 条;合并减少 {prMergeReducedCount} 条;生成DO/PO {purchaseOrderTransferResult.CreatedOrderCount} 单;写QadTracking {externalPushResult.TrackingCount} 条;异常 {exceptions.Count} 条"
  598. : $"未生成交货单,自动生成PR {purchaseRequests.Count} 条,合并减少 {prMergeReducedCount} 条,生成DO/PO {purchaseOrderTransferResult.CreatedOrderCount} 单,写QadTracking {externalPushResult.TrackingCount} 条,已记录异常 {exceptions.Count} 条"
  599. };
  600. }
  601. private async Task<DeliveryScheduleGenerateResult> GenerateDeliverySchedulesFromPurchaseOrdersAsync(string poNumber)
  602. {
  603. var account = _userManager.Account ?? "system";
  604. var pars = new List<SugarParameter>();
  605. var where = new List<string>
  606. {
  607. "IFNULL(m.Status,'') <> 'C'",
  608. "IFNULL(d.Status,'') <> 'C'",
  609. "(IFNULL(d.QtyOrded,0) - IFNULL(d.RctQty,0)) > 0",
  610. """
  611. NOT EXISTS (
  612. SELECT 1
  613. FROM srm_polist_ds ds
  614. WHERE ds.ponumber = d.PurOrd
  615. AND ds.poline = d.Line
  616. AND ds.isactive = 1
  617. AND IFNULL(ds.status,'') <> 'C'
  618. )
  619. """
  620. };
  621. if (_userManager.TenantId > 0)
  622. {
  623. where.Add("m.tenant_id = @TenantId");
  624. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  625. }
  626. if (!string.IsNullOrWhiteSpace(poNumber))
  627. {
  628. where.Add("m.PurOrd = @PoNumber");
  629. pars.Add(new SugarParameter("@PoNumber", poNumber.Trim()));
  630. }
  631. var candidates = await _db.Ado.SqlQueryAsync<DeliveryScheduleCandidateRow>(
  632. $"""
  633. SELECT
  634. IFNULL(TRIM(m.Domain),'') AS Domain,
  635. d.RecID AS IcdsId,
  636. IFNULL(TRIM(d.PurOrd),'') AS PoNumber,
  637. IFNULL(d.Line,0) AS PoLine,
  638. IFNULL(TRIM(d.ItemNum),'') AS ItemNum,
  639. IFNULL(TRIM(d.UM),'') AS Um,
  640. IFNULL(TRIM(m.Buyer),'') AS PurGroup,
  641. IFNULL(TRIM(m.Supp),'') AS SupplierCode,
  642. IFNULL(TRIM(s.Name),'') AS Supplier,
  643. d.DueDate AS NeedDate,
  644. (IFNULL(d.QtyOrded,0) - IFNULL(d.RctQty,0)) AS RestQty,
  645. m.tenant_id AS TenantId
  646. FROM PurOrdMaster m
  647. INNER JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  648. LEFT JOIN ConsigneeAddressMaster s ON m.Supp = s.Address AND s.Typed = 'Supp'
  649. WHERE {string.Join(" AND ", where)}
  650. ORDER BY m.PurOrd, d.Line
  651. """,
  652. pars);
  653. if (candidates.Count == 0)
  654. {
  655. return new DeliveryScheduleGenerateResult
  656. {
  657. CandidateCount = 0,
  658. CreatedCount = 0,
  659. Message = string.IsNullOrWhiteSpace(poNumber)
  660. ? "无可生成交货单的采购行"
  661. : "当前采购单无可生成交货单的采购行"
  662. };
  663. }
  664. try
  665. {
  666. _db.Ado.BeginTran();
  667. var dsNumbersByKey = new Dictionary<string, string>();
  668. foreach (var group in candidates.GroupBy(x => x.Domain ?? string.Empty))
  669. {
  670. var groupRows = group.ToList();
  671. var dsNumbers = await _numberRuleService.NextBatchInCurrentTransactionAsync("M8", group.Key, groupRows.Count, account);
  672. if (dsNumbers.Count < groupRows.Count || dsNumbers.Any(string.IsNullOrWhiteSpace))
  673. throw Oops.Oh($"当前交货计划号生成失败,请检查交货计划编号规则维护。Domain={group.Key}");
  674. for (var index = 0; index < groupRows.Count; index++)
  675. {
  676. dsNumbersByKey[BuildCandidateKey(groupRows[index])] = dsNumbers[index].Trim();
  677. }
  678. }
  679. var now = DateTime.Now;
  680. var schedules = candidates.Select(row => new PolistDeliverySchedule
  681. {
  682. Id = YitIdHelper.NextId(),
  683. Domain = row.Domain ?? string.Empty,
  684. IcdsId = row.IcdsId,
  685. DsNum = dsNumbersByKey[BuildCandidateKey(row)],
  686. Status = "N",
  687. ItemNum = row.ItemNum ?? string.Empty,
  688. Um = row.Um ?? string.Empty,
  689. PurGroup = row.PurGroup ?? string.Empty,
  690. SupplierCode = row.SupplierCode ?? string.Empty,
  691. Supplier = row.Supplier ?? string.Empty,
  692. RequestDate = now,
  693. NeedDate = row.NeedDate,
  694. PoNumber = row.PoNumber ?? string.Empty,
  695. PoLine = row.PoLine,
  696. SchedQty = row.RestQty,
  697. LastSentQty = 0,
  698. SentQty = 0,
  699. RestQty = row.RestQty,
  700. CreateUser = account,
  701. UpdateUser = account,
  702. CreateTime = now,
  703. UpdateTime = now,
  704. Remarks = "由系统按采购行自动生成",
  705. IsActive = 1,
  706. ReturnQty = 0,
  707. TenantId = _userManager.TenantId > 0 ? _userManager.TenantId : row.TenantId
  708. }).ToList();
  709. var createdCount = await _db.Insertable(schedules).ExecuteCommandAsync();
  710. await UpsertDeliveryScheduleMdpAsync(schedules, now);
  711. _db.Ado.CommitTran();
  712. return new DeliveryScheduleGenerateResult
  713. {
  714. CandidateCount = candidates.Count,
  715. CreatedCount = createdCount,
  716. DsNums = schedules.Select(x => x.DsNum).Take(20).ToList(),
  717. Message = createdCount > 0
  718. ? $"生成交货单成功,共生成 {createdCount} 条"
  719. : "未生成交货单,请检查采购行是否已存在有效交货计划"
  720. };
  721. }
  722. catch
  723. {
  724. _db.Ado.RollbackTran();
  725. throw;
  726. }
  727. }
  728. private async Task UpsertDeliveryScheduleMdpAsync(List<PolistDeliverySchedule> schedules, DateTime now)
  729. {
  730. if (schedules.Count == 0) return;
  731. var batchId = $"S3_DELIVERY_{now:yyyyMMddHHmmss}";
  732. foreach (var schedule in schedules)
  733. {
  734. await _db.Ado.ExecuteCommandAsync(
  735. """
  736. INSERT INTO mdp_std_delivery_schedule
  737. (tenant_id, source_system, delivery_plan_no, po_no, po_line, item_code, supplier_code, supplier_name, schedule_qty, sent_qty, rest_qty, return_qty, request_date, need_date, submit_date, last_sent_date, status, source_biz_key, sync_batch_id, sync_time)
  738. VALUES
  739. (@TenantId, 'AIDOP', @DsNum, @PoNumber, @PoLine, @ItemNum, @SupplierCode, @Supplier, @SchedQty, @SentQty, @RestQty, @ReturnQty, @RequestDate, @NeedDate, @SubmitDate, @LastSentDate, @Status, @DsNum, @BatchId, @Now)
  740. ON DUPLICATE KEY UPDATE
  741. po_no=VALUES(po_no),
  742. po_line=VALUES(po_line),
  743. item_code=VALUES(item_code),
  744. supplier_code=VALUES(supplier_code),
  745. supplier_name=VALUES(supplier_name),
  746. schedule_qty=VALUES(schedule_qty),
  747. sent_qty=VALUES(sent_qty),
  748. rest_qty=VALUES(rest_qty),
  749. return_qty=VALUES(return_qty),
  750. request_date=VALUES(request_date),
  751. need_date=VALUES(need_date),
  752. submit_date=VALUES(submit_date),
  753. last_sent_date=VALUES(last_sent_date),
  754. status=VALUES(status),
  755. sync_batch_id=VALUES(sync_batch_id),
  756. sync_time=VALUES(sync_time),
  757. update_time=CURRENT_TIMESTAMP
  758. """,
  759. new SugarParameter("@TenantId", schedule.TenantId ?? 0),
  760. new SugarParameter("@DsNum", schedule.DsNum),
  761. new SugarParameter("@PoNumber", schedule.PoNumber),
  762. new SugarParameter("@PoLine", schedule.PoLine.ToString()),
  763. new SugarParameter("@ItemNum", schedule.ItemNum),
  764. new SugarParameter("@SupplierCode", schedule.SupplierCode),
  765. new SugarParameter("@Supplier", schedule.Supplier),
  766. new SugarParameter("@SchedQty", schedule.SchedQty),
  767. new SugarParameter("@SentQty", schedule.SentQty),
  768. new SugarParameter("@RestQty", schedule.RestQty),
  769. new SugarParameter("@ReturnQty", schedule.ReturnQty),
  770. new SugarParameter("@RequestDate", schedule.RequestDate),
  771. new SugarParameter("@NeedDate", schedule.NeedDate),
  772. new SugarParameter("@SubmitDate", schedule.SubmitDate),
  773. new SugarParameter("@LastSentDate", schedule.LastSentDate),
  774. new SugarParameter("@Status", schedule.Status),
  775. new SugarParameter("@BatchId", batchId),
  776. new SugarParameter("@Now", now));
  777. await _db.Ado.ExecuteCommandAsync(
  778. """
  779. INSERT INTO dwd_supplier_delivery
  780. (tenant_id, stat_date, po_no, po_line, po_type, supplier_code, supplier_name, item_code, item_name, order_qty, schedule_qty, delivery_qty, receipt_qty, return_qty, remaining_qty, due_date, need_date, last_delivery_date, delivery_status, source_system, sync_batch_id, calc_time)
  781. SELECT
  782. @TenantId,
  783. @StatDate,
  784. IFNULL(m.PurOrd, @PoNumber),
  785. CAST(IFNULL(d.Line, @PoLineInt) AS CHAR),
  786. IFNULL(m.Potype, ''),
  787. IFNULL(m.Supp, @SupplierCode),
  788. @Supplier,
  789. IFNULL(d.ItemNum, @ItemNum),
  790. IFNULL(i.Descr, ''),
  791. IFNULL(d.QtyOrded, @SchedQty),
  792. @SchedQty,
  793. @SentQty,
  794. IFNULL(d.QtyReceived, IFNULL(d.RctQty, 0)),
  795. @ReturnQty,
  796. @RestQty,
  797. d.DueDate,
  798. @NeedDate,
  799. @LastSentDate,
  800. CASE
  801. WHEN IFNULL(@Status, '') = 'C' THEN 'CANCELLED'
  802. WHEN IFNULL(@RestQty, 0) <= 0 THEN 'CLOSED'
  803. WHEN IFNULL(@SentQty, 0) > 0 THEN 'PARTIAL'
  804. ELSE 'OPEN'
  805. END,
  806. 'AIDOP',
  807. @BatchId,
  808. @Now
  809. FROM PurOrdMaster m
  810. JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  811. LEFT JOIN ItemMaster i ON d.ItemNum = i.ItemNum
  812. WHERE m.PurOrd = @PoNumber AND d.Line = @PoLineInt
  813. LIMIT 1
  814. ON DUPLICATE KEY UPDATE
  815. supplier_code=VALUES(supplier_code),
  816. supplier_name=VALUES(supplier_name),
  817. item_code=VALUES(item_code),
  818. item_name=VALUES(item_name),
  819. order_qty=VALUES(order_qty),
  820. schedule_qty=VALUES(schedule_qty),
  821. delivery_qty=VALUES(delivery_qty),
  822. receipt_qty=VALUES(receipt_qty),
  823. return_qty=VALUES(return_qty),
  824. remaining_qty=VALUES(remaining_qty),
  825. due_date=VALUES(due_date),
  826. need_date=VALUES(need_date),
  827. last_delivery_date=VALUES(last_delivery_date),
  828. delivery_status=VALUES(delivery_status),
  829. sync_batch_id=VALUES(sync_batch_id),
  830. calc_time=VALUES(calc_time),
  831. update_time=CURRENT_TIMESTAMP
  832. """,
  833. new SugarParameter("@TenantId", schedule.TenantId ?? 0),
  834. new SugarParameter("@StatDate", now.Date),
  835. new SugarParameter("@PoNumber", schedule.PoNumber),
  836. new SugarParameter("@PoLineInt", schedule.PoLine),
  837. new SugarParameter("@SupplierCode", schedule.SupplierCode),
  838. new SugarParameter("@Supplier", schedule.Supplier),
  839. new SugarParameter("@ItemNum", schedule.ItemNum),
  840. new SugarParameter("@SchedQty", schedule.SchedQty),
  841. new SugarParameter("@SentQty", schedule.SentQty),
  842. new SugarParameter("@ReturnQty", schedule.ReturnQty),
  843. new SugarParameter("@RestQty", schedule.RestQty),
  844. new SugarParameter("@NeedDate", schedule.NeedDate),
  845. new SugarParameter("@LastSentDate", schedule.LastSentDate),
  846. new SugarParameter("@Status", schedule.Status),
  847. new SugarParameter("@BatchId", batchId),
  848. new SugarParameter("@Now", now));
  849. }
  850. }
  851. private async Task<Dictionary<long, decimal>> QueryExistingScheduleUsagesAsync(List<long> demandIds)
  852. {
  853. if (demandIds.Count == 0) return new Dictionary<long, decimal>();
  854. var pars = new List<SugarParameter>();
  855. var inClause = BuildInClause(demandIds.Select(x => x.ToString()).ToList(), "DemandId", pars);
  856. var tenantWhere = _userManager.TenantId > 0 ? " AND tenant_id = @TenantId" : string.Empty;
  857. if (_userManager.TenantId > 0)
  858. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  859. var rows = await _db.Ado.SqlQueryAsync<ExistingScheduleUsageRow>(
  860. $"""
  861. SELECT
  862. icdsid AS IcdsId,
  863. SUM(
  864. CASE
  865. WHEN IFNULL(status,'') = 'C' THEN IFNULL(sentqty,0)
  866. WHEN IFNULL(schedqty,0) >= IFNULL(sentqty,0) THEN IFNULL(schedqty,0)
  867. ELSE IFNULL(sentqty,0)
  868. END
  869. ) AS UsedQty
  870. FROM srm_polist_ds
  871. WHERE isactive = 1 AND icdsid IN ({inClause}){tenantWhere}
  872. GROUP BY icdsid
  873. """,
  874. pars);
  875. return rows.ToDictionary(x => x.IcdsId, x => x.UsedQty);
  876. }
  877. private async Task<List<SourceListRow>> QuerySourceListsAsync(List<string> itemNums)
  878. {
  879. if (itemNums.Count == 0) return new List<SourceListRow>();
  880. var pars = new List<SugarParameter>();
  881. var where = new List<string>
  882. {
  883. "IFNULL(sp.IsDeleted,0)=0",
  884. "IFNULL(sp.supplier_number,'') <> ''",
  885. "IFNULL(sp.quota_rate,0) > 0",
  886. """
  887. (
  888. IFNULL(sp.is_active,'') = ''
  889. OR sp.is_active = '否'
  890. OR sp.is_active = 'N'
  891. OR sp.is_active = '0'
  892. OR LOWER(sp.is_active) = 'false'
  893. )
  894. """
  895. };
  896. if (_userManager.TenantId > 0)
  897. {
  898. where.Add("sp.tenant_id = @TenantId");
  899. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  900. }
  901. var inClause = BuildInClause(itemNums, "ItemNum", pars);
  902. where.Add($"sp.number IN ({inClause})");
  903. return await _db.Ado.SqlQueryAsync<SourceListRow>(
  904. $"""
  905. SELECT
  906. IFNULL(TRIM(sp.number),'') AS ItemNum,
  907. IFNULL(TRIM(sp.supplier_number),'') AS SupplierCode,
  908. IFNULL(TRIM(sp.supplier_name),'') AS SupplierName,
  909. IFNULL(TRIM(sp.supplier_type),'') AS SupplierType,
  910. IFNULL(sp.quota_rate,0) AS QuotaRate,
  911. IFNULL(sp.packaging_qty,0) AS PackagingQty,
  912. IFNULL(sp.lead_time,0) AS LeadTime,
  913. IFNULL(sp.IsRequireGoods,0) AS IsRequireGoods,
  914. sp.supplier_id AS SupplierId,
  915. IFNULL(TRIM(sp.order_rector_name),'') AS OrderRectorName,
  916. IFNULL(TRIM(sp.order_rector_num),'') AS OrderRectorNum,
  917. IFNULL(TRIM(sp.purchase_unit),'') AS PurchaseUnit,
  918. sp.currency_type AS CurrencyType
  919. FROM srm_purchase sp
  920. WHERE {string.Join(" AND ", where)}
  921. ORDER BY sp.number, IFNULL(sp.quota_priority, 999999), sp.quota_rate DESC
  922. """,
  923. pars);
  924. }
  925. private async Task<List<PurchaseOrderAvailableRow>> QueryAvailablePurchaseOrderLinesAsync(
  926. List<string> itemNums,
  927. List<string> supplierCodes,
  928. S3DeliveryGenerateRuleOptions rules)
  929. {
  930. var pars = new List<SugarParameter>();
  931. var poBuyerCodesClause = BuildInClause(rules.PoBuyerCodes, "PoBuyerCode", pars);
  932. var doUsagesClause = BuildInClause(rules.DoUsages, "DoUsage", pars);
  933. var where = new List<string>
  934. {
  935. "IFNULL(m.Status,'') <> 'C'",
  936. "IFNULL(d.Status,'') <> 'C'",
  937. "IFNULL(d.ItemNum,'') <> ''",
  938. "IFNULL(m.Supp,'') <> ''",
  939. "(IFNULL(d.QtyOrded,0) - IFNULL(d.RctQty,0)) > 0",
  940. """
  941. (
  942. (m.PurOrd NOT LIKE 'DO%' AND IFNULL(m.Buyer,'') IN ({0}))
  943. OR
  944. (m.PurOrd LIKE 'DO%' AND IFNULL(m.USAGE,'') IN ({1}))
  945. )
  946. """.Replace("{0}", poBuyerCodesClause).Replace("{1}", doUsagesClause)
  947. };
  948. if (_userManager.TenantId > 0)
  949. {
  950. where.Add("m.tenant_id = @TenantId");
  951. pars.Add(new SugarParameter("@TenantId", _userManager.TenantId));
  952. }
  953. where.Add($"d.ItemNum IN ({BuildInClause(itemNums, "PoItemNum", pars)})");
  954. where.Add($"m.Supp IN ({BuildInClause(supplierCodes, "SupplierCode", pars)})");
  955. return await _db.Ado.SqlQueryAsync<PurchaseOrderAvailableRow>(
  956. $"""
  957. SELECT *
  958. FROM (
  959. SELECT
  960. IFNULL(TRIM(m.Domain),'') AS Domain,
  961. IFNULL(TRIM(m.PurOrd),'') AS PoNumber,
  962. IFNULL(d.Line,0) AS PoLine,
  963. d.RecID AS DetailRecId,
  964. IFNULL(TRIM(d.ItemNum),'') AS ItemNum,
  965. IFNULL(TRIM(d.UM),'') AS Um,
  966. IFNULL(TRIM(m.Buyer),'') AS PurGroup,
  967. IFNULL(TRIM(m.Supp),'') AS SupplierCode,
  968. IFNULL(TRIM(s.Name),'') AS SupplierName,
  969. d.DueDate AS DueDate,
  970. IFNULL(d.QtyOrded,0) AS OrderQty,
  971. IFNULL(d.RctQty,0) AS ReceiptQty,
  972. m.tenant_id AS TenantId,
  973. (
  974. IFNULL(d.QtyOrded,0)
  975. - IFNULL(d.RctQty,0)
  976. - IFNULL((
  977. SELECT SUM(
  978. CASE
  979. WHEN IFNULL(ds.status,'') = 'C' THEN IFNULL(ds.sentqty,0)
  980. WHEN IFNULL(ds.schedqty,0) >= IFNULL(ds.sentqty,0) THEN IFNULL(ds.schedqty,0)
  981. ELSE IFNULL(ds.sentqty,0)
  982. END)
  983. FROM srm_polist_ds ds
  984. WHERE ds.ponumber = d.PurOrd
  985. AND ds.poline = d.Line
  986. AND ds.itemnum = d.ItemNum
  987. AND ds.isactive = 1
  988. ), 0)
  989. ) AS AvailableQty
  990. FROM PurOrdMaster m
  991. INNER JOIN PurOrdDetail d ON m.RecID = d.PurOrdRecID
  992. LEFT JOIN ConsigneeAddressMaster s ON m.Supp = s.Address AND s.Typed = 'Supp'
  993. WHERE {string.Join(" AND ", where)}
  994. ) t
  995. WHERE t.AvailableQty > 0
  996. ORDER BY t.ItemNum, t.SupplierCode, CASE WHEN t.PoNumber LIKE 'DO%' THEN 1 ELSE 0 END, t.DueDate, t.DetailRecId
  997. """,
  998. pars);
  999. }
  1000. private static decimal AllocateDemandToPurchaseLines(
  1001. DemandScheduleCandidateRow demand,
  1002. SourceListRow source,
  1003. List<PurchaseOrderAvailableRow> poLines,
  1004. Dictionary<string, decimal> allocatedByPoLine,
  1005. decimal supplierNeedQty,
  1006. decimal remainingDemand,
  1007. List<PolistDeliverySchedule> schedules,
  1008. DateTime now,
  1009. string account,
  1010. S3DeliveryGenerateRuleOptions rules)
  1011. {
  1012. var supplierAllocated = 0m;
  1013. var lines = poLines
  1014. .Where(x =>
  1015. string.Equals(x.ItemNum, demand.ItemNum, StringComparison.OrdinalIgnoreCase)
  1016. && string.Equals(x.SupplierCode, source.SupplierCode, StringComparison.OrdinalIgnoreCase))
  1017. .ToList();
  1018. foreach (var line in lines)
  1019. {
  1020. if (supplierAllocated >= supplierNeedQty) break;
  1021. var key = BuildPoLineKey(line);
  1022. var runAllocated = allocatedByPoLine.TryGetValue(key, out var used) ? used : 0m;
  1023. var availableQty = line.AvailableQty - runAllocated;
  1024. if (availableQty <= 0) continue;
  1025. var requiredQty = supplierNeedQty - supplierAllocated;
  1026. var qty = Math.Min(availableQty, requiredQty);
  1027. if (rules.EnablePackagingRoundUp && source.PackagingQty > 0 && availableQty > qty)
  1028. {
  1029. var roundedQty = Math.Ceiling(qty / source.PackagingQty) * source.PackagingQty;
  1030. if (roundedQty <= availableQty && roundedQty <= remainingDemand - supplierAllocated)
  1031. {
  1032. qty = roundedQty;
  1033. }
  1034. }
  1035. if (qty <= 0) continue;
  1036. allocatedByPoLine[key] = runAllocated + qty;
  1037. supplierAllocated += qty;
  1038. schedules.Add(new PolistDeliverySchedule
  1039. {
  1040. Id = YitIdHelper.NextId(),
  1041. Domain = line.Domain ?? string.Empty,
  1042. IcdsId = demand.IcdsId,
  1043. DsNum = string.Empty,
  1044. Status = "N",
  1045. ItemNum = demand.ItemNum ?? string.Empty,
  1046. Um = line.Um ?? demand.Um ?? string.Empty,
  1047. PurGroup = line.PurGroup ?? string.Empty,
  1048. SupplierCode = source.SupplierCode ?? line.SupplierCode ?? string.Empty,
  1049. Supplier = string.IsNullOrWhiteSpace(source.SupplierName) ? line.SupplierName ?? string.Empty : source.SupplierName,
  1050. RequestDate = demand.ArrivalDate ?? now,
  1051. NeedDate = demand.RequestDate,
  1052. PoNumber = line.PoNumber ?? string.Empty,
  1053. PoLine = line.PoLine,
  1054. SchedQty = qty,
  1055. LastSentQty = 0,
  1056. SentQty = 0,
  1057. RestQty = qty,
  1058. CreateUser = account,
  1059. UpdateUser = account,
  1060. CreateTime = now,
  1061. UpdateTime = now,
  1062. Remarks = "由系统按需求计划、货源配额和采购单可用量自动生成",
  1063. IsActive = 1,
  1064. ReturnQty = 0,
  1065. TenantId = demand.TenantId ?? line.TenantId
  1066. });
  1067. }
  1068. return supplierAllocated;
  1069. }
  1070. private static void AddDeliveryException(
  1071. List<DeliveryExceptionMaster> exceptions,
  1072. DemandScheduleCandidateRow demand,
  1073. DateTime optTime,
  1074. string remark,
  1075. decimal needQty)
  1076. {
  1077. exceptions.Add(new DeliveryExceptionMaster
  1078. {
  1079. RecID = YitIdHelper.NextId(),
  1080. Domain = demand.TenantId?.ToString() ?? string.Empty,
  1081. IcdsId = demand.IcdsId,
  1082. OptTime = optTime,
  1083. ItemNum = demand.ItemNum,
  1084. Remark = remark,
  1085. NeedQty = needQty,
  1086. TenantId = demand.TenantId
  1087. });
  1088. }
  1089. private static bool TryAddPurchaseRequest(
  1090. List<PurchaseRequestMain> purchaseRequests,
  1091. DemandScheduleCandidateRow demand,
  1092. SourceListRow source,
  1093. decimal shortageQty,
  1094. DateTime now,
  1095. string account,
  1096. S3DeliveryGenerateRuleOptions rules)
  1097. {
  1098. if (shortageQty <= 0) return false;
  1099. if (!demand.IcItemId.HasValue || demand.IcItemId.Value <= 0) return false;
  1100. if (!source.SupplierId.HasValue || source.SupplierId.Value <= 0) return false;
  1101. var arriveDate = demand.ArrivalDate ?? demand.RequestDate ?? now.Date;
  1102. var leadDays = source.LeadTime > 0 ? (int)Math.Ceiling(source.LeadTime) : rules.DefaultLeadDays;
  1103. if (leadDays < 0) leadDays = 0;
  1104. var sendDate = arriveDate.Date.AddDays(-leadDays);
  1105. if (sendDate < now.Date) sendDate = now.Date;
  1106. purchaseRequests.Add(new PurchaseRequestMain
  1107. {
  1108. Id = YitIdHelper.NextId(),
  1109. PrPurchaseId = source.SupplierId.Value,
  1110. PrPurchaseNumber = source.SupplierCode,
  1111. PrPurchaseName = source.SupplierName,
  1112. PrPurchaser = source.OrderRectorName,
  1113. PrPurchaserNum = source.OrderRectorNum,
  1114. PrRqty = shortageQty,
  1115. PrAqty = shortageQty,
  1116. PrSqty = shortageQty,
  1117. IcitemId = demand.IcItemId.Value,
  1118. IcitemName = string.IsNullOrWhiteSpace(demand.ItemName) ? demand.ItemNum : demand.ItemName,
  1119. PrSsendDate = sendDate,
  1120. PrSarriveDate = arriveDate,
  1121. PrUnit = string.IsNullOrWhiteSpace(source.PurchaseUnit) ? demand.Um : source.PurchaseUnit,
  1122. State = 1,
  1123. PrType = 3,
  1124. CurrencyType = source.CurrencyType ?? 1,
  1125. CreateBy = null,
  1126. CreateByName = account,
  1127. CreateTime = now,
  1128. UpdateBy = null,
  1129. UpdateByName = account,
  1130. UpdateTime = now,
  1131. TenantId = demand.TenantId ?? 0,
  1132. FactoryId = demand.FactoryId,
  1133. OrgId = demand.FactoryId,
  1134. IsDeleted = false,
  1135. CompanyId = demand.CompanyId ?? 1000,
  1136. IsRequireGoods = source.IsRequireGoods,
  1137. SupplierType = source.SupplierType
  1138. });
  1139. return true;
  1140. }
  1141. private async Task AssignPurchaseRequestNumbersAsync(List<PurchaseRequestMain> purchaseRequests, string account)
  1142. {
  1143. foreach (var group in purchaseRequests.GroupBy(x => x.TenantId.ToString()))
  1144. {
  1145. var groupRows = group.ToList();
  1146. var prNumbers = await _numberRuleService.NextBatchInCurrentTransactionAsync("PR", group.Key, groupRows.Count, account);
  1147. if (prNumbers.Count < groupRows.Count || prNumbers.Any(string.IsNullOrWhiteSpace))
  1148. throw Oops.Oh($"当前采购申请号生成失败,请检查采购申请编号规则维护。Domain={group.Key}");
  1149. for (var index = 0; index < groupRows.Count; index++)
  1150. {
  1151. groupRows[index].PrBillNo = prNumbers[index].Trim();
  1152. }
  1153. }
  1154. }
  1155. private async Task InsertPurchaseRequestsAsync(List<PurchaseRequestMain> purchaseRequests)
  1156. {
  1157. if (purchaseRequests.Count == 0) return;
  1158. foreach (var pr in purchaseRequests)
  1159. {
  1160. await _db.Ado.ExecuteCommandAsync(
  1161. """
  1162. INSERT INTO srm_pr_main
  1163. (Id,pr_billno,pr_purchaseid,pr_purchasenumber,pr_purchasename,pr_purchaser,pr_purchaser_num,
  1164. pr_rqty,pr_aqty,pr_sqty,icitem_id,icitem_name,pr_ssend_date,pr_sarrive_date,pr_unit,state,pr_type,currencytype,
  1165. create_by,create_by_name,create_time,update_by,update_by_name,update_time,tenant_id,factory_id,org_id,IsDeleted,company_id,IsRequireGoods,supplier_type)
  1166. VALUES
  1167. (@Id,@PrBillNo,@PrPurchaseId,@PrPurchaseNumber,@PrPurchaseName,@PrPurchaser,@PrPurchaserNum,
  1168. @PrRqty,@PrAqty,@PrSqty,@IcitemId,@IcitemName,@PrSsendDate,@PrSarriveDate,@PrUnit,@State,@PrType,@CurrencyType,
  1169. @CreateBy,@CreateByName,@CreateTime,@UpdateBy,@UpdateByName,@UpdateTime,@TenantId,@FactoryId,@OrgId,@IsDeleted,@CompanyId,@IsRequireGoods,@SupplierType)
  1170. """,
  1171. new SugarParameter("@Id", pr.Id),
  1172. new SugarParameter("@PrBillNo", pr.PrBillNo),
  1173. new SugarParameter("@PrPurchaseId", pr.PrPurchaseId),
  1174. new SugarParameter("@PrPurchaseNumber", pr.PrPurchaseNumber),
  1175. new SugarParameter("@PrPurchaseName", pr.PrPurchaseName),
  1176. new SugarParameter("@PrPurchaser", pr.PrPurchaser),
  1177. new SugarParameter("@PrPurchaserNum", pr.PrPurchaserNum),
  1178. new SugarParameter("@PrRqty", pr.PrRqty),
  1179. new SugarParameter("@PrAqty", pr.PrAqty),
  1180. new SugarParameter("@PrSqty", pr.PrSqty),
  1181. new SugarParameter("@IcitemId", pr.IcitemId),
  1182. new SugarParameter("@IcitemName", pr.IcitemName),
  1183. new SugarParameter("@PrSsendDate", pr.PrSsendDate),
  1184. new SugarParameter("@PrSarriveDate", pr.PrSarriveDate),
  1185. new SugarParameter("@PrUnit", pr.PrUnit),
  1186. new SugarParameter("@State", pr.State),
  1187. new SugarParameter("@PrType", pr.PrType),
  1188. new SugarParameter("@CurrencyType", pr.CurrencyType),
  1189. new SugarParameter("@CreateBy", pr.CreateBy),
  1190. new SugarParameter("@CreateByName", pr.CreateByName),
  1191. new SugarParameter("@CreateTime", pr.CreateTime),
  1192. new SugarParameter("@UpdateBy", pr.UpdateBy),
  1193. new SugarParameter("@UpdateByName", pr.UpdateByName),
  1194. new SugarParameter("@UpdateTime", pr.UpdateTime),
  1195. new SugarParameter("@TenantId", pr.TenantId),
  1196. new SugarParameter("@FactoryId", pr.FactoryId),
  1197. new SugarParameter("@OrgId", pr.OrgId),
  1198. new SugarParameter("@IsDeleted", pr.IsDeleted ? 1 : 0),
  1199. new SugarParameter("@CompanyId", pr.CompanyId),
  1200. new SugarParameter("@IsRequireGoods", pr.IsRequireGoods),
  1201. new SugarParameter("@SupplierType", pr.SupplierType));
  1202. }
  1203. }
  1204. private static string BuildInClause(List<string> values, string prefix, List<SugarParameter> pars)
  1205. {
  1206. if (values.Count == 0) return "NULL";
  1207. var names = new List<string>();
  1208. for (var i = 0; i < values.Count; i++)
  1209. {
  1210. var name = $"@{prefix}{i}";
  1211. names.Add(name);
  1212. pars.Add(new SugarParameter(name, values[i]));
  1213. }
  1214. return string.Join(",", names);
  1215. }
  1216. private static string BuildPoLineKey(PurchaseOrderAvailableRow row)
  1217. {
  1218. return $"{row.PoNumber}|{row.PoLine}|{row.ItemNum}|{row.DetailRecId}";
  1219. }
  1220. private static string BuildScheduleTempKey(PolistDeliverySchedule row)
  1221. {
  1222. return $"{row.Id}|{row.PoNumber}|{row.PoLine}|{row.IcdsId}";
  1223. }
  1224. private static string FormatDate(DateTime? value) => value?.ToString("yyyy-MM-dd") ?? "";
  1225. private static List<long> ParseIds(string ids)
  1226. {
  1227. return ids.Split(',', StringSplitOptions.RemoveEmptyEntries)
  1228. .Select(x => long.TryParse(x.Trim(), out var id) ? id : 0)
  1229. .Where(x => x > 0)
  1230. .Distinct()
  1231. .ToList();
  1232. }
  1233. private static string BuildCandidateKey(DeliveryScheduleCandidateRow row)
  1234. {
  1235. return $"{row.Domain}|{row.PoNumber}|{row.PoLine}|{row.IcdsId}";
  1236. }
  1237. private static string BuildListOrderBy(string? sortField, string? sortOrder)
  1238. {
  1239. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  1240. return sortField?.ToLowerInvariant() switch
  1241. {
  1242. "dsnum" => $"ss.dsnum {dir}",
  1243. "status" => $"ss.status {dir}",
  1244. "itemnum" => $"ss.itemnum {dir}",
  1245. "requestdate" => $"ss.requestdate {dir}",
  1246. "needdate" => $"ss.needdate {dir}",
  1247. "schedqty" => $"ss.schedqty {dir}",
  1248. "sentqty" => $"ss.sentqty {dir}",
  1249. "restqty" => $"ss.restqty {dir}",
  1250. "submitdate" => $"ss.submitdate {dir}",
  1251. "ponumber" => $"ss.ponumber {dir}",
  1252. _ => "ss.Id DESC"
  1253. };
  1254. }
  1255. private static string BuildPurchaseOrderOrderBy(string? sortField, string? sortOrder)
  1256. {
  1257. var dir = string.Equals(sortOrder, "asc", StringComparison.OrdinalIgnoreCase) ? "ASC" : "DESC";
  1258. return sortField?.ToLowerInvariant() switch
  1259. {
  1260. "purord" => $"m.PurOrd {dir}",
  1261. "line" => $"d.Line {dir}",
  1262. "itemnum" => $"d.ItemNum {dir}",
  1263. "supp" => $"m.Supp {dir}",
  1264. "name" => $"s.Name {dir}",
  1265. "duedate" => $"d.DueDate {dir}",
  1266. "qtyorded" => $"d.QtyOrded {dir}",
  1267. "buyer" => $"m.Buyer {dir}",
  1268. _ => "m.PurOrd DESC, d.Line ASC"
  1269. };
  1270. }
  1271. private sealed class DeliveryScheduleListRow
  1272. {
  1273. public long Id { get; set; }
  1274. public string? Domain { get; set; }
  1275. public long? IcdsId { get; set; }
  1276. public string? DsNum { get; set; }
  1277. public string? Status { get; set; }
  1278. public string? ItemNum { get; set; }
  1279. public string? Descr { get; set; }
  1280. public string? Um { get; set; }
  1281. public string? PurGroup { get; set; }
  1282. public string? SupplierCode { get; set; }
  1283. public string? Supplier { get; set; }
  1284. public DateTime? SubmitDate { get; set; }
  1285. public DateTime? RequestDate { get; set; }
  1286. public DateTime? NeedDate { get; set; }
  1287. public string? PoNumber { get; set; }
  1288. public int? PoLine { get; set; }
  1289. public decimal? SchedQty { get; set; }
  1290. public DateTime? LastSentDate { get; set; }
  1291. public decimal? LastSentQty { get; set; }
  1292. public decimal? SentQty { get; set; }
  1293. public decimal? RestQty { get; set; }
  1294. }
  1295. private sealed class PurchaseOrderSelectRow
  1296. {
  1297. public string? PurOrd { get; set; }
  1298. public int? Line { get; set; }
  1299. public string? ItemNum { get; set; }
  1300. public string? Um { get; set; }
  1301. public string? Supp { get; set; }
  1302. public string? Name { get; set; }
  1303. public DateTime? DueDate { get; set; }
  1304. public decimal? QtyOrded { get; set; }
  1305. public string? Buyer { get; set; }
  1306. }
  1307. private sealed class DeliveryScheduleGenerateResult
  1308. {
  1309. public int DemandCount { get; set; }
  1310. public int CandidateCount { get; set; }
  1311. public int CreatedCount { get; set; }
  1312. public int ExceptionCount { get; set; }
  1313. public int PurchaseRequestCount { get; set; }
  1314. public int PurchaseRequestMergeReducedCount { get; set; }
  1315. public int PurchaseOrderCount { get; set; }
  1316. public int PurchaseOrderLineCount { get; set; }
  1317. public int QadTrackingCount { get; set; }
  1318. public int SkippedCount { get; set; }
  1319. public List<string> DsNums { get; set; } = new();
  1320. public RuleConfigSnapshot<S3DeliveryGenerateRuleOptions>? RuleSnapshot { get; set; }
  1321. public string Message { get; set; } = string.Empty;
  1322. }
  1323. private sealed class DeliveryScheduleCandidateRow
  1324. {
  1325. public string? Domain { get; set; }
  1326. public long IcdsId { get; set; }
  1327. public string? PoNumber { get; set; }
  1328. public int PoLine { get; set; }
  1329. public string? ItemNum { get; set; }
  1330. public string? Um { get; set; }
  1331. public string? PurGroup { get; set; }
  1332. public string? SupplierCode { get; set; }
  1333. public string? Supplier { get; set; }
  1334. public DateTime? NeedDate { get; set; }
  1335. public decimal RestQty { get; set; }
  1336. public long? TenantId { get; set; }
  1337. }
  1338. private sealed class DemandScheduleCandidateRow
  1339. {
  1340. public long IcdsId { get; set; }
  1341. public string? ItemNum { get; set; }
  1342. public DateTime? RequestDate { get; set; }
  1343. public DateTime? ArrivalDate { get; set; }
  1344. public decimal DemandQty { get; set; }
  1345. public long? TenantId { get; set; }
  1346. public long? FactoryId { get; set; }
  1347. public long? CompanyId { get; set; }
  1348. public string? PurMfg { get; set; }
  1349. public string? Um { get; set; }
  1350. public string? ItemName { get; set; }
  1351. public long? IcItemId { get; set; }
  1352. }
  1353. private sealed class ExistingScheduleUsageRow
  1354. {
  1355. public long IcdsId { get; set; }
  1356. public decimal UsedQty { get; set; }
  1357. }
  1358. private sealed class SourceListRow
  1359. {
  1360. public string? ItemNum { get; set; }
  1361. public string? SupplierCode { get; set; }
  1362. public string? SupplierName { get; set; }
  1363. public string? SupplierType { get; set; }
  1364. public decimal QuotaRate { get; set; }
  1365. public decimal PackagingQty { get; set; }
  1366. public decimal LeadTime { get; set; }
  1367. public int IsRequireGoods { get; set; }
  1368. public long? SupplierId { get; set; }
  1369. public string? OrderRectorName { get; set; }
  1370. public string? OrderRectorNum { get; set; }
  1371. public string? PurchaseUnit { get; set; }
  1372. public long? CurrencyType { get; set; }
  1373. }
  1374. private sealed class PurchaseOrderAvailableRow
  1375. {
  1376. public string? Domain { get; set; }
  1377. public string? PoNumber { get; set; }
  1378. public int PoLine { get; set; }
  1379. public long DetailRecId { get; set; }
  1380. public string? ItemNum { get; set; }
  1381. public string? Um { get; set; }
  1382. public string? PurGroup { get; set; }
  1383. public string? SupplierCode { get; set; }
  1384. public string? SupplierName { get; set; }
  1385. public DateTime? DueDate { get; set; }
  1386. public decimal OrderQty { get; set; }
  1387. public decimal ReceiptQty { get; set; }
  1388. public long? TenantId { get; set; }
  1389. public decimal AvailableQty { get; set; }
  1390. }
  1391. }