MonthlyCapacityLoadAppService.cs 56 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042
  1. using Azure.Identity;
  2. using Business.Core.Utilities;
  3. using Business.Domain;
  4. using Business.Dto;
  5. using Business.EntityFrameworkCore.SqlRepositories;
  6. using Business.ResourceExamineManagement.Dto;
  7. using Business.SaleForecast;
  8. using Business.StructuredDB.WMS;
  9. using Microsoft.CodeAnalysis.CSharp.Syntax;
  10. using Spire.Pdf.General.Render.Decode.Jpeg2000.j2k.quantization;
  11. using Spire.Pdf.General.Render.Decode.Jpeg2000.j2k.roi.encoder;
  12. using System;
  13. using System.Collections.Generic;
  14. using System.Linq;
  15. using System.Runtime.CompilerServices;
  16. using System.Threading.Tasks;
  17. using ThoughtWorks.QRCode.Geom;
  18. using Volo.Abp.Application.Services;
  19. using Volo.Abp.DependencyInjection;
  20. using Volo.Abp.Domain.Repositories;
  21. using Volo.Abp.MultiTenancy;
  22. using Volo.Abp.Uow;
  23. using Volo.Abp.Validation.Localization;
  24. namespace Business.SaleForecastManagement
  25. {
  26. /// <summary>
  27. /// 产能分析
  28. /// </summary>
  29. public class MonthlyCapacityLoadAppService : ApplicationService, IMonthlyCapacityLoadAppService, ITransientDependency
  30. {
  31. #region 服务
  32. /// <summary>
  33. /// 物料
  34. /// </summary>
  35. private IRepository<ic_item, long> _ic_item;
  36. /// <summary>
  37. /// 生产线明细
  38. /// </summary>
  39. private ISqlRepository<ProdLineDetail> _prodLineDetail;
  40. /// <summary>
  41. /// 工作日历数据
  42. /// </summary>
  43. private ISqlRepository<ShopCalendarWorkCtr> _shopCalendarWorkCtr;
  44. /// <summary>
  45. /// 产线休息时间记录表
  46. /// </summary>
  47. private ISqlRepository<QualityLineWorkDetail> _qualityLineWorkDetail;
  48. /// <summary>
  49. /// 节假日记录表
  50. /// </summary>
  51. private ISqlRepository<HolidayMaster> _holidayMaster;
  52. /// <summary>
  53. /// 工作单元
  54. /// </summary>
  55. private readonly IUnitOfWorkManager _unitOfWorkManager;
  56. /// <summary>
  57. /// 日志
  58. /// </summary>
  59. private readonly ICurrentTenant _currentTenant;
  60. /// <summary>
  61. /// 年度生产需求大纲
  62. /// </summary>
  63. private IRepository<YearDemandManagement, long> _yearDemandManagement;
  64. /// <summary>
  65. /// 年度生产需求大纲历史记录
  66. /// </summary>
  67. private IRepository<YearDemandManagementHistory, long> _yearDemandManagementHistory;
  68. /// <summary>
  69. /// 国内终端预测
  70. /// </summary>
  71. private IRepository<DomesticTerminalFcst, long> _domesticTerminalFcst;
  72. /// <summary>
  73. /// 海外销售预测
  74. /// </summary>
  75. private IRepository<OverseasSaleFcst, long> _overseasSaleFcst;
  76. /// <summary>
  77. /// 平台预测收集
  78. /// </summary>
  79. private IRepository<PlatformFcstCollect, long> _platformFcstCollect;
  80. /// <summary>
  81. /// 标准物料规格型号设置表
  82. /// </summary>
  83. private IRepository<StandardItemModelSet,long> _standardItemModelSet;
  84. /// <summary>
  85. /// 库存明细表
  86. /// </summary>
  87. private ISqlRepository<LocationDetail> _locationDetail;
  88. /// <summary>
  89. /// 工单
  90. /// </summary>
  91. private ISqlRepository<WorkOrdMaster> _workOrdMaster;
  92. /// <summary>
  93. /// 整体需求计划
  94. /// </summary>
  95. private IRepository<OverallDemandPlan, long> _overallDemandPlan;
  96. /// <summary>
  97. /// 月度产能共识表
  98. /// </summary>
  99. private IRepository<MonthlyProdCapacity, long> _monthlyProdCapacity;
  100. /// <summary>
  101. /// 月度产能共识明细表
  102. /// </summary>
  103. private IRepository<MonthlyProdCapacityDtl, long> _monthlyProdCapacityDtl;
  104. /// <summary>
  105. /// SKU版本维护表
  106. /// </summary>
  107. private IRepository<SkuVersionSet, long> _skuVersionSet;
  108. /// <summary>
  109. /// 计划订单表
  110. /// </summary>
  111. private IRepository<crm_planorder, long> _crm_planorder;
  112. /// <summary>
  113. /// 平台库存监控月份
  114. /// </summary>
  115. private IRepository<PlatStockMonitorSetting, long> _platStockMonitorSetting;
  116. /// <summary>
  117. /// 平台库存表
  118. /// </summary>
  119. private IRepository<WMS_PlatformInventory, long> _platformInventory;
  120. /// <summary>
  121. /// 参数设置表
  122. /// </summary>
  123. private ISqlRepository<GeneralizedCodeMaster> _generalizedCodeMaster;
  124. /// <summary>
  125. /// SAP库存表
  126. /// </summary>
  127. private ISqlRepository<SAPInv> _SAPInv;
  128. /// <summary>
  129. /// 雪花算法
  130. /// </summary>
  131. SnowFlake help = new SnowFlake();
  132. #endregion
  133. #region 构造函数
  134. /// <summary>
  135. /// 构造函数
  136. /// </summary>
  137. public MonthlyCapacityLoadAppService(
  138. IRepository<ic_item, long> ic_item,
  139. ISqlRepository<ProdLineDetail> prodLineDetail,
  140. ISqlRepository<ShopCalendarWorkCtr> shopCalendarWorkCtr,
  141. ISqlRepository<QualityLineWorkDetail> qualityLineWorkDetail,
  142. ISqlRepository<HolidayMaster> holidayMaster,
  143. IRepository<MonthlyProdCapacity, long> monthlyProdCapacity,
  144. IUnitOfWorkManager unitOfWorkManager,
  145. ICurrentTenant currentTenant,
  146. IRepository<StandardItemModelSet,long> standardItemModelSet,
  147. IRepository<YearDemandManagement, long> yearDemandManagement,
  148. IRepository<DomesticTerminalFcst, long> domesticTerminalFcst,
  149. IRepository<OverseasSaleFcst, long> overseasSaleFcst,
  150. IRepository<PlatformFcstCollect, long> platformFcstCollect,
  151. IRepository<YearDemandManagementHistory, long> yearDemandManagementHistory,
  152. ISqlRepository<LocationDetail> locationDetail,
  153. ISqlRepository<WorkOrdMaster> workOrdMaster,
  154. IRepository<OverallDemandPlan, long> overallDemandPlan,
  155. IRepository<MonthlyProdCapacityDtl, long> monthlyProdCapacityDtl,
  156. IRepository<SkuVersionSet, long> skuVersionSet,
  157. IRepository<crm_planorder, long> crm_planorder,
  158. IRepository<PlatStockMonitorSetting, long> platStockMonitorSetting,
  159. IRepository<WMS_PlatformInventory, long> platformInventory,
  160. ISqlRepository<GeneralizedCodeMaster> generalizedCodeMaster,
  161. ISqlRepository<SAPInv> SAPInv
  162. )
  163. {
  164. _ic_item = ic_item;
  165. _prodLineDetail = prodLineDetail;
  166. _shopCalendarWorkCtr = shopCalendarWorkCtr;
  167. _qualityLineWorkDetail = qualityLineWorkDetail;
  168. _holidayMaster = holidayMaster;
  169. _monthlyProdCapacity = monthlyProdCapacity;
  170. _unitOfWorkManager = unitOfWorkManager;
  171. _currentTenant = currentTenant;
  172. _standardItemModelSet = standardItemModelSet;
  173. _yearDemandManagement= yearDemandManagement;
  174. _domesticTerminalFcst= domesticTerminalFcst;
  175. _overseasSaleFcst = overseasSaleFcst;
  176. _platformFcstCollect= platformFcstCollect;
  177. _yearDemandManagementHistory= yearDemandManagementHistory;
  178. _locationDetail = locationDetail;
  179. _workOrdMaster = workOrdMaster;
  180. _overallDemandPlan = overallDemandPlan;
  181. _monthlyProdCapacityDtl = monthlyProdCapacityDtl;
  182. _skuVersionSet= skuVersionSet;
  183. _crm_planorder= crm_planorder;
  184. _platStockMonitorSetting = platStockMonitorSetting;
  185. _platformInventory = platformInventory;
  186. _generalizedCodeMaster = generalizedCodeMaster;
  187. _SAPInv = SAPInv;
  188. }
  189. #endregion
  190. /// <summary>
  191. /// 月度需求预测更新
  192. /// </summary>
  193. /// <param name="input"></param>
  194. /// <returns></returns>
  195. /// <exception cref="NotImplementedException"></exception>
  196. public async Task<string> DemandAnalysis(InputDto input)
  197. {
  198. //1.0 获取当前年年度生产需求大纲
  199. List<YearDemandManagement> yearDemands = _yearDemandManagement.GetListAsync(p => p.Year == input.year && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result.OrderBy(p=>p.OrderNum).ToList();
  200. //1.0 获取下一年年度生产大纲
  201. List<YearDemandManagement> nextYearDemands = _yearDemandManagement.GetListAsync(p => p.Year == (input.year+1) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result.OrderBy(p => p.OrderNum).ToList();
  202. //记录历史版本:当前年+下一年
  203. var updateMonth = input.year.ToString() + "-" + input.month.ToString("00");
  204. var histories = ObjectMapper.Map<List<YearDemandManagement>, List<YearDemandManagementHistory>>(yearDemands);
  205. histories.ForEach(p => {
  206. p.UpdateMonth = updateMonth;
  207. });
  208. var nextHistories = ObjectMapper.Map<List<YearDemandManagement>, List<YearDemandManagementHistory>>(nextYearDemands);
  209. nextHistories.ForEach(p => {
  210. p.UpdateMonth = updateMonth;
  211. });
  212. //1.1、获取海外销售预测数据
  213. List<OverseasSaleFcst> overseasSales = _overseasSaleFcst.GetListAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  214. //1.2、获取平台预测收集数据
  215. List<PlatformFcstCollect> platformFcsts = _platformFcstCollect.GetListAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  216. //1.3、获取国内终端预测-T1汇总数据
  217. List<DomesticTerminalFcst> domesticFcst = _domesticTerminalFcst.GetListAsync(p =>p.TypeEnum == 2 && p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  218. //1.3.1、获取T2-海王,T2-国科数据
  219. List<DomesticTerminalFcst> T2Fcsts = _domesticTerminalFcst.GetListAsync(p => (p.TypeEnum == 4 || p.TypeEnum == 5) && p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  220. //1.4、根据年度生产大纲,获取规格型号对应的补货周期
  221. List<string> allModels = yearDemands.Select(p=>p.Model).Distinct().ToList();
  222. if (nextYearDemands.Any())
  223. {
  224. allModels.AddRange(nextYearDemands.Select(p => p.Model).ToList());
  225. allModels = allModels.Distinct().ToList();
  226. }
  227. List<StandardItemModelSet> standards = _standardItemModelSet.GetListAsync(p => allModels.Contains(p.Model) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  228. //1.5、获取规格型号对应的标准SKU数据:获取最小包装单位
  229. List<string> itemNums = standards.Select(p=>p.ItemNumber).Distinct().ToList();
  230. List<ic_item> items = _ic_item.GetListAsync(p=> itemNums.Contains(p.number) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  231. //1.6、获取成品库存、灭菌库存、在制库存(会从SAP同步的库存表更新到LocationDetail、ic_item表中)
  232. //List<SAPInv> sAPInvs = _SAPInv.Select(p => p.WERKS == input.factory_id.ToString() && itemNums.Contains(p.MATNR));
  233. List<LocationDetail> locations = _locationDetail.Select(p => p.Domain == input.factory_id.ToString() && itemNums.Contains(p.ItemNum) && p.IsActive);
  234. //1.7、获取节假日设置
  235. List<HolidayMaster> holidays = _holidayMaster.Select(p => (p.Dated.Value.Year == input.year || p.Dated.Value.Year == (input.year + 1)) && p.Domain == input.factory_id.ToString() && p.IsActive);
  236. //1.8、获取平台库存监控月份设置
  237. List<PlatStockMonitorSetting> monitorSettings = _platStockMonitorSetting.GetListAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  238. //1.9、获取客户要求交期在N+1月的销售订单
  239. //List<>
  240. #region 数据校验
  241. //1、校验当前年的年度生产大纲是否导入
  242. if (!yearDemands.Any())
  243. {
  244. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "请导入" + input.year + "年的年度生产需求后再操作!", _currentTenant.Id.ToString());
  245. return "NO|请导入" + input.year + "年的年度生产需求后再操作!";
  246. }
  247. //2、如果当前月份大于10,则需校验下一年的年度生产大纲
  248. if (input.month > 10 && !nextYearDemands.Any())
  249. {
  250. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "请导入" + (input.year+1) + "年的年度生产需求后再操作!", _currentTenant.Id.ToString());
  251. return "NO|请导入" + (input.year + 1) + "年的年度生产需求后再操作!";
  252. }
  253. //3、校验当前月份的海外销售预测是否导入
  254. if (!overseasSales.Any())
  255. {
  256. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "请导入" + input.year + "年" + input.month + "月海外销售预测后再操作!", _currentTenant.Id.ToString());
  257. return "NO|请导入" + input.year + "年" + input.month + "月海外销售预测后再操作!";
  258. }
  259. //4、校验当前月份的平台预测收集是否导入
  260. if (!platformFcsts.Any())
  261. {
  262. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "请导入" + input.year + "年" + input.month + "月平台需求收集后再操作!", _currentTenant.Id.ToString());
  263. return "NO|请导入" + input.year + "年" + input.month + "月平台需求收集后再操作!";
  264. }
  265. //5、校验当前月份的国内终端预测是否导入
  266. if (!domesticFcst.Any())
  267. {
  268. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "请导入" + input.year + "年" + input.month + "月国内终端预测后再操作!", _currentTenant.Id.ToString());
  269. return "NO|请导入" + input.year + "年" + input.month + "月国内终端预测后再操作!";
  270. }
  271. //6、校验终端+平台导入的规格型号是否已维护标准SKU
  272. if (allModels.Count() != standards.Count())
  273. {
  274. allModels.RemoveAll(standards.Select(p=>p.Model).ToList());
  275. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "规格型号【" + string.Join(",",allModels) + "】没有维护标准SKU,请维护后再操作!", _currentTenant.Id.ToString());
  276. return "NO|规格型号【" + string.Join(",", allModels) + "】没有维护标准SKU,请维护后再操作!";
  277. }
  278. //7、校验是否已维护补货周期
  279. var zeros = standards.Where(p => p.ReplenishCycle == 0).ToList();
  280. if (zeros.Any())
  281. {
  282. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "规格型号【" + string.Join(",", zeros.Select(p=>p.Model).ToList()) + "】没有维护补货周期,请维护后再操作!", _currentTenant.Id.ToString());
  283. return "NO|规格型号【" + string.Join(",", zeros.Select(p => p.Model).ToList()) + "】没有维护补货周期,请维护后再操作!";
  284. }
  285. #endregion
  286. //计算当前年月的N0,N+1,N+2
  287. List<string> planMons = new List<string>();
  288. int newYear = input.month == 12 ? input.year + 1 : input.year;
  289. int newMonth = input.month == 12 ? 1 : input.month + 1;
  290. string strN1 = newYear.ToString() + "-" + newMonth.ToString("00");
  291. planMons.Add(strN1);
  292. newYear = newMonth == 12 ? newYear + 1 : newYear;
  293. newMonth = newMonth == 12 ? 1 : newMonth + 1;
  294. string strN2 = newYear.ToString() + "-" + newMonth.ToString("00");
  295. planMons.Add(strN2);
  296. //需要回写的数据
  297. List<YearDemandManagement> updates = new List<YearDemandManagement>();
  298. #region 计算国内终端预测,回写年度生产大纲
  299. //获取导入的T1、平台数据
  300. List<string> gnModels = new List<string>();
  301. if (domesticFcst.Any())
  302. {
  303. gnModels.AddRange(domesticFcst.Select(p => p.Model).ToList());
  304. }
  305. if (platformFcsts.Any())
  306. {
  307. gnModels.AddRange(platformFcsts.Select(p => p.Model).ToList());
  308. }
  309. gnModels = gnModels.Distinct().ToList();
  310. foreach (var gnModel in gnModels)
  311. {
  312. //N+1月
  313. YearDemandManagement monthN1 = new YearDemandManagement();
  314. //N+2月
  315. YearDemandManagement monthN2 = new YearDemandManagement();
  316. //获取需要回写的数据
  317. if (input.month < 11)
  318. {
  319. monthN1 = yearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN1);
  320. monthN2 = yearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN2);
  321. }
  322. if (input.month == 11)
  323. {
  324. monthN1 = yearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN1);
  325. monthN2 = nextYearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN2);
  326. }
  327. if (input.month == 12)
  328. {
  329. monthN1 = nextYearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN1);
  330. monthN2 = nextYearDemands.FirstOrDefault(p => p.Area == "国内" && p.Model == gnModel && p.PlanMonth == strN2);
  331. }
  332. if (monthN1 == null || monthN2 == null)
  333. {
  334. continue;
  335. }
  336. //T1
  337. var sumT1N1 = domesticFcst.Where(p => p.Model == gnModel && p.PlanMonth == strN1).Sum(p=>p.Qty);
  338. var sumT1N2 = domesticFcst.Where(p => p.Model == gnModel && p.PlanMonth == strN2).Sum(p => p.Qty);
  339. //平台
  340. var sumPN1 = platformFcsts.Where(p => p.Model == gnModel && p.PlanMonth == strN1).Sum(p => p.Qty);
  341. var sumPN2 = platformFcsts.Where(p => p.Model == gnModel && p.PlanMonth == strN2).Sum(p => p.Qty);
  342. //合计
  343. var sumN1 = sumT1N1+ sumPN1;
  344. var sumN2 = sumT1N2 + sumPN2;
  345. //当前规格型号对应标准SKU的最小包装单位、补货周期
  346. var curStd = standards.FirstOrDefault(p => p.Model == gnModel);
  347. decimal packQty = 1m;//最小包装单位
  348. decimal cycle = 0m;//补货周期
  349. //获取成品库存、在制库存、灭菌库存,参与计算
  350. decimal cpQty = 0m;
  351. decimal zzQty = 0m;
  352. decimal mjQty = 0m;
  353. if (curStd != null)
  354. {
  355. var curItem = items.FirstOrDefault(p => p.number == curStd.ItemNumber);
  356. packQty = curItem == null ? 1 : (curItem.minpackqty.GetValueOrDefault() == 0.0m ? 1 : curItem.minpackqty.Value);
  357. cycle = curStd.ReplenishCycle;
  358. //成品库存
  359. cpQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "8001").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  360. //在制库存
  361. zzQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "8000").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  362. //灭菌库存
  363. mjQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "5008").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  364. }
  365. //N+1月使用N+2月的再订货点参与计算
  366. decimal rop = CalcRop(strN2+"-01", sumN2, packQty, holidays, cycle);
  367. monthN1.Qty = Math.Ceiling((sumN1 / 2 + sumN2 / 2 + rop - cpQty - zzQty - mjQty) / packQty) * packQty;
  368. //N+2月使用本月的需求量,下一月的工作天数
  369. rop = CalcRop(Convert.ToDateTime(strN2 + "-01").AddMonths(1).ToString("yyyy-MM-dd"), sumN2, packQty, holidays, cycle);
  370. monthN2.Qty = Math.Ceiling((sumN2 + rop - cpQty - zzQty - mjQty) / packQty) * packQty;
  371. //负数置0
  372. monthN1.Qty = monthN1.Qty < 0 ? 0m : monthN1.Qty;
  373. monthN2.Qty = monthN2.Qty < 0 ? 0m : monthN2.Qty;
  374. //记录需要回写的数据
  375. updates.Add(monthN1);
  376. updates.Add(monthN2);
  377. }
  378. #endregion
  379. #region 海外销售预测,回写年度生产大纲
  380. List<string> hwModels = overseasSales.Select(p => p.Model).Distinct().ToList();
  381. foreach (var hwModel in hwModels)
  382. {
  383. //N+1月
  384. YearDemandManagement monthN1 = new YearDemandManagement();
  385. //N+2月
  386. YearDemandManagement monthN2 = new YearDemandManagement();
  387. //获取需要回写的数据
  388. if (input.month < 11)
  389. {
  390. monthN1 = yearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN1);
  391. monthN2 = yearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN2);
  392. }
  393. if (input.month == 11)
  394. {
  395. monthN1 = yearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN1);
  396. monthN2 = nextYearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN2);
  397. }
  398. if (input.month == 12)
  399. {
  400. monthN1 = nextYearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN1);
  401. monthN2 = nextYearDemands.FirstOrDefault(p => p.Area == "海外" && p.Model == hwModel && p.PlanMonth == strN2);
  402. }
  403. if (monthN1 == null || monthN2 == null) {
  404. continue;
  405. }
  406. var sumN1 = overseasSales.Where(p => p.Model == hwModel && p.PlanMonth == strN1).Sum(p => p.Qty);
  407. var sumN2 = overseasSales.Where(p => p.Model == hwModel && p.PlanMonth == strN2).Sum(p => p.Qty);
  408. //当前规格型号对应标准SKU的最小包装单位、补货周期
  409. var curStd = standards.FirstOrDefault(p => p.Model == hwModel);
  410. decimal packQty = 1m;//最小包装单位
  411. //获取成品库存、在制库存、灭菌库存,参与计算
  412. decimal cpQty = 0m;
  413. decimal zzQty = 0m;
  414. decimal mjQty = 0m;
  415. if (curStd != null)
  416. {
  417. var curItem = items.FirstOrDefault(p => p.number == curStd.ItemNumber);
  418. packQty = curItem == null ? 1 : (curItem.minpackqty.GetValueOrDefault() == 0.0m ? 1 : curItem.minpackqty.Value);
  419. //成品库存
  420. cpQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "8001").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  421. //在制库存
  422. zzQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "8000").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  423. //灭菌库存
  424. mjQty = locations.Where(p => p.ItemNum == curStd.ItemNumber && p.Location == "5008").Sum(p => Convert.ToDecimal(p.QtyOnHand));
  425. }
  426. //海外生产需求量=当月的50%+下一月的50%-成品库存-在制库存-灭菌库存
  427. //TODO:获取成品库存、在制库存、灭菌库存,参与运算
  428. //计算N+1月,N+2月
  429. monthN1.Qty = Math.Ceiling((sumN1 / 2 + sumN2 / 2 - cpQty - zzQty - mjQty) / packQty) * packQty;
  430. monthN2.Qty = Math.Ceiling((sumN2 - cpQty - zzQty - mjQty) / packQty) *packQty; ;
  431. //负数置0
  432. monthN1.Qty = monthN1.Qty < 0 ? 0m : monthN1.Qty;
  433. monthN2.Qty = monthN2.Qty < 0 ? 0m : monthN2.Qty;
  434. //记录需要回写的数据
  435. updates.Add(monthN1);
  436. updates.Add(monthN2);
  437. }
  438. #endregion
  439. //生成月度产能共识
  440. var capacityDto = CapacityAnalysis(input, updates, planMons);
  441. //生成下一个月的补货计划
  442. var nextMonPFcsts = platformFcsts.Where(p => p.PlanMonth == strN2).ToList();
  443. var nextMonDFcsts = domesticFcst.Where(p => p.PlanMonth == strN2).ToList();
  444. var nextMonT2Fcsts = T2Fcsts.Where(p => p.PlanMonth == strN2).ToList();
  445. var nextMonMonitors = monitorSettings.Where(p => p.PlanMonth == strN1).ToList();
  446. var replenishs = MonthlyReplenish(input,strN1, nextMonPFcsts, nextMonDFcsts, nextMonT2Fcsts, standards, items, holidays, nextMonMonitors, locations);
  447. //保存数据
  448. using (var unitOfWork = _unitOfWorkManager.Begin(false, true))
  449. {
  450. try
  451. {
  452. //判断是否存在当前年月的历史版本,不存在则插入
  453. var dbHistory = _yearDemandManagementHistory.GetListAsync(p=>p.Year ==input.year && p.UpdateMonth == updateMonth && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id).Result;
  454. if (!dbHistory.Any())
  455. {
  456. await _yearDemandManagementHistory.InsertManyAsync(histories);
  457. }
  458. if (input.month > 10)
  459. {
  460. //判断是否存在下一年月的历史版本,不存在则插入
  461. dbHistory = _yearDemandManagementHistory.GetListAsync(p =>(p.Year == input.year + 1) && p.UpdateMonth == updateMonth && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id).Result;
  462. if (!dbHistory.Any())
  463. {
  464. await _yearDemandManagementHistory.InsertManyAsync(nextHistories);
  465. }
  466. }
  467. //回写年度生产大纲
  468. await _yearDemandManagement.UpdateManyAsync(updates);
  469. //先删除产能共识、产能共识明细
  470. await _monthlyProdCapacity.HardDeleteAsync(p=>p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id);
  471. await _monthlyProdCapacityDtl.HardDeleteAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id);
  472. await _monthlyProdCapacity.InsertManyAsync(capacityDto.mains);
  473. await _monthlyProdCapacityDtl.InsertManyAsync(capacityDto.details);
  474. //生成下一月的补货计划
  475. await _crm_planorder.HardDeleteAsync(p=>p.PlanMonth == strN1 && p.Source =="系统运算" && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id);
  476. await _crm_planorder.InsertManyAsync(replenishs);
  477. await unitOfWork.CompleteAsync();
  478. }
  479. catch (Exception e)
  480. {
  481. unitOfWork.Dispose();
  482. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "【" + input.year + "年" + input.month + "月】月度需求预测更新失败:" + e.Message, _currentTenant.Id.ToString());
  483. return "NO|" + e.Message;
  484. };
  485. }
  486. return "OK|刷新成功!";
  487. }
  488. /// <summary>
  489. /// 生成月度产能共识
  490. /// </summary>
  491. /// <param name="input"></param>
  492. /// <returns></returns>
  493. public MonthlyCapacityDto CapacityAnalysis(InputDto input,List<YearDemandManagement> yearDemands, List<string> planMons)
  494. {
  495. //1、获取数据
  496. //1.1 根据规格型号获取物料数据
  497. List<string> models = yearDemands.Select(p => p.Model).Distinct().ToList();
  498. List<StandardItemModelSet> standards = _standardItemModelSet.GetListAsync(p => models.Contains(p.Model) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id).Result;
  499. //1.3 根据物料编码获取产线数据
  500. List<ProdLineDetail> lineDtls = _prodLineDetail.Select(p => standards.Select(m => m.ItemNumber).Contains(p.Part) && p.Domain == input.factory_id.ToString() && p.IsActive).OrderBy(p => p.Line).ToList();
  501. //1.4 根据产线获取工作日历数据和产线休息配置数据
  502. List<ShopCalendarWorkCtr> calendars = _shopCalendarWorkCtr.Select(p => lineDtls.Select(m => m.Line).Contains(p.ProdLine) && p.Domain == input.factory_id.ToString() && p.IsActive);
  503. List<QualityLineWorkDetail> lineWorks = _qualityLineWorkDetail.Select(p => lineDtls.Select(m => m.Line).Contains(p.ProdLine) && p.Domain == input.factory_id.ToString() && p.IsActive);
  504. //1.5 获取当前年和下一年的节假日配置数据
  505. List<HolidayMaster> holidays = _holidayMaster.Select(p => (p.Dated.Value.Year == input.year || p.Dated.Value.Year == (input.year + 1)) && p.Domain == input.factory_id.ToString() && p.IsActive);
  506. //月度产能共识表
  507. List<MonthlyProdCapacity> capacities = new List<MonthlyProdCapacity>();
  508. //月度产能共识明细表
  509. List<MonthlyProdCapacityDtl> capacityDtls = new List<MonthlyProdCapacityDtl>();
  510. //产线
  511. List<string> lines = lineDtls.Select(p => p.Line).Distinct().ToList();
  512. foreach (var item in lines)
  513. {
  514. //计算每天工作时间
  515. var curCal = calendars.FirstOrDefault(p => p.ProdLine == item);
  516. if (curCal == null)
  517. {
  518. continue;
  519. }
  520. foreach (var pm in planMons)
  521. {
  522. //添加月度产能共识产能效率数据
  523. MonthlyProdCapacity dtl = new MonthlyProdCapacity();
  524. dtl.Year = input.year;
  525. dtl.Month = input.month;
  526. dtl.PlanMonth = pm;
  527. dtl.ProdLine = item;
  528. dtl.DailyWorks = curCal.ShiftsHours1 + curCal.ShiftsHours2 + curCal.ShiftsHours3 + curCal.ShiftsHours4;
  529. dtl.FlightQty = 1;
  530. //计算当月工作天数
  531. var curHoildays = holidays.Where(p => p.Dated.Value.Year == Convert.ToInt16(pm.Substring(0,4)) && p.Dated.Value.Month == Convert.ToInt16(pm.Substring(5, 2))).ToList();
  532. //当月天数
  533. int days = DateTime.DaysInMonth(input.year, input.month);
  534. //当月周末天数
  535. int weekDays = CalcWeekDays(days, Convert.ToDateTime(pm + "-01"));
  536. dtl.MonthWorks = days - weekDays - curHoildays.Where(p => p.Ufld1 == "休假").Count() + curHoildays.Where(p => p.Ufld1 == "调班").Count();
  537. dtl.AvailableTimes = dtl.DailyWorks * dtl.MonthWorks;
  538. //计算产线耗时
  539. var curLines = lineDtls.Where(p => p.Line == item).ToList();
  540. decimal sumTimes = 0.00m;
  541. List<ProdLineDto> lineDtos = new List<ProdLineDto>();
  542. foreach (var ld in curLines)
  543. {
  544. if (lineDtos.Exists(p=>p.Part == ld.Part && p.Line == ld.Line))
  545. {
  546. continue;
  547. }
  548. var curStand = standards.FirstOrDefault(p => p.ItemNumber == ld.Part);
  549. if (curStand == null)
  550. {
  551. continue;
  552. }
  553. var curDemands = yearDemands.Where(p => p.Model == curStand.Model && p.PlanMonth == pm).ToList();
  554. var line = curLines.Where(p=>p.Part == ld.Part && p.Line == ld.Line).OrderByDescending(p => p.Op).First();
  555. sumTimes += line.Rate == 0 ? 0 : (Math.Ceiling(curDemands.Sum(p => p.Qty) / line.Rate));
  556. MonthlyProdCapacityDtl capacityDtl = new MonthlyProdCapacityDtl();
  557. capacityDtl.Year = input.year;
  558. capacityDtl.Month = input.month;
  559. capacityDtl.PlanMonth = pm;
  560. capacityDtl.ProdLine = item;
  561. capacityDtl.Model = curStand.Model;
  562. capacityDtl.Qty = curDemands.Sum(p => p.Qty);
  563. capacityDtl.tenant_id = input.tenant_id;
  564. capacityDtl.company_id = input.company_id;
  565. capacityDtl.factory_id = input.factory_id;
  566. capacityDtl.org_id = input.org_id;
  567. capacityDtl.create_by = input.create_by;
  568. capacityDtl.create_by_name = input.create_by_name;
  569. capacityDtl.create_time = DateTime.Now;
  570. capacityDtls.Add(capacityDtl);
  571. lineDtos.Add(new ProdLineDto {
  572. Part = ld.Part,
  573. Line= ld.Line
  574. });
  575. }
  576. dtl.NeedWorks = sumTimes;
  577. dtl.ProdRate = 100;
  578. dtl.Rate = dtl.AvailableTimes == 0 ? 0 : Math.Floor(dtl.NeedWorks / dtl.AvailableTimes * 100);
  579. dtl.IsOverTime = dtl.NeedWorks > dtl.AvailableTimes ? "是" : "否";
  580. dtl.OverTimes = dtl.IsOverTime == "是" ? (dtl.NeedWorks - dtl.AvailableTimes) : 0;
  581. dtl.tenant_id = input.tenant_id;
  582. dtl.company_id= input.company_id;
  583. dtl.factory_id = input.factory_id;
  584. dtl.org_id= input.org_id;
  585. dtl.create_by= input.create_by;
  586. dtl.create_by_name = input.create_by_name;
  587. dtl.create_time = DateTime.Now;
  588. capacities.Add(dtl);
  589. }
  590. }
  591. return new MonthlyCapacityDto{
  592. mains = capacities,
  593. details = capacityDtls
  594. };
  595. }
  596. /// <summary>
  597. /// T1、平台自动补货
  598. /// </summary>
  599. /// <param name="input"></param>
  600. /// <param name="strN1">计划补货月份:N+1月</param>
  601. /// <param name="platformFcsts">平台预测</param>
  602. /// <param name="domesticFcsts">T1预测</param>
  603. /// <param name="standards">标准SKU设置</param>
  604. /// <param name="items">物料</param>
  605. /// <param name="holidays">节假日</param>
  606. /// <param name="monitorSettings">库存监控月份设置</param>
  607. /// <param name="locations">SAP同步库存</param>
  608. /// <returns></returns>
  609. public List<crm_planorder> MonthlyReplenish(InputDto input,string strN1, List<PlatformFcstCollect> platformFcsts, List<DomesticTerminalFcst> domesticFcsts, List<DomesticTerminalFcst> T2Fcsts, List<StandardItemModelSet> standards, List<ic_item> items, List<HolidayMaster> holidays, List<PlatStockMonitorSetting> monitorSettings, List<LocationDetail> locations)
  610. {
  611. List<crm_planorder> planorders = new List<crm_planorder>();
  612. //获取T1中规格型号对应的不同版本的物料编码
  613. var T1Models = domesticFcsts.Select(p => p.Model).Distinct().ToList();
  614. List<SkuVersionSet> skus = _skuVersionSet.GetListAsync(p=> T1Models.Contains(p.Model) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result;
  615. //获取T1库存
  616. //List<string> itemNums = skus.Select(p=>p.ItemNum).Distinct().ToList();
  617. //List<LocationDetail> locationDetails = _locationDetail.Select(p=> itemNums.Contains(p.ItemNum) && p.Domain == input.factory_id.ToString() && p.IsActive);
  618. //获取临期库存设置
  619. GeneralizedCodeMaster master = _generalizedCodeMaster.Select(p => p.Domain == input.factory_id.ToString() && p.IsActive && p.Val == "LongPeriodItemPlanMonth").FirstOrDefault();
  620. int month = master == null ? 0 : Convert.ToInt16(master.UDeci1);
  621. DateTime strTime = Convert.ToDateTime(platformFcsts[0].PlanMonth + "-01").AddMonths(month);
  622. //获取平台库存
  623. var PModels = platformFcsts.Select(p => p.Model).Distinct().ToList();
  624. List<WMS_PlatformInventory> pInventories = _platformInventory.GetListAsync(p=> PModels.Contains(p.SpecificationModel) && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && p.PeriodOfValidity >= strTime).Result;
  625. //计算T1补货
  626. foreach (var item in T1Models)
  627. {
  628. //获取当前规格型号对应的预测数据
  629. var curFcsts = domesticFcsts.Where(p=>p.Model == item).ToList();
  630. //当前规格型号对应的库存数量
  631. decimal sumQty = 0.00m;
  632. //当前的物料版本
  633. var curSkus = skus.Where(p => p.Model == item).ToList();
  634. if (curSkus.Any())
  635. {
  636. //sumQty = sAPInvs.Where(p => curSkus.Select(p => p.ItemNum).Contains(p.MATNR)).Sum(p => Convert.ToDecimal(p.LABST));
  637. sumQty = locations.Where(p => curSkus.Select(p => p.ItemNum).Contains(p.ItemNum) && p.Location == "8001").Sum(p => p.QtyOnHand);
  638. }
  639. //计算Rop
  640. //当前规格型号对应标准SKU的最小包装单位、补货周期
  641. var curStd = standards.FirstOrDefault(p => p.Model == item);
  642. decimal packQty = 1m;//最小包装单位
  643. decimal cycle = 0m;//补货周期
  644. if (curStd != null)
  645. {
  646. var curItem = items.FirstOrDefault(p => p.number == curStd.ItemNumber);
  647. packQty = curItem == null ? 1 : (curItem.minpackqty.GetValueOrDefault() == 0.0m ? 1 : curItem.minpackqty.Value);
  648. cycle = curStd.ReplenishCycle;
  649. }
  650. //N+1月使用N+2月的再订货点参与计算
  651. decimal rop = CalcRop(curFcsts[0].PlanMonth + "-01", curFcsts.Sum(p=>p.Qty), packQty, holidays, cycle);
  652. if(sumQty < rop)
  653. {
  654. planorders.Add(new crm_planorder {
  655. PlanMonth = strN1,
  656. Model = curFcsts[0].Model,
  657. ItemNum = curStd?.ItemNumber,
  658. ProdLine = curFcsts[0].ProdLine,
  659. ProdType = "",
  660. Qty = rop,
  661. Type = "计划单-T1直发补货"
  662. });
  663. }
  664. }
  665. //计算平台补货:海王
  666. var hwFcsts = platformFcsts.Where(p => p.Platform == "海王").ToList();
  667. var hwModels = hwFcsts.Select(p=>p.Model).Distinct().ToList();
  668. foreach (var item in hwModels)
  669. {
  670. //获取当前规格型号对应的平台预测数据
  671. var curFcsts = hwFcsts.Where(p => p.Model == item).ToList();
  672. //当前规格型号对应的库存数量
  673. decimal sumQty = 0.00m;
  674. //当前规格型号对应标准SKU的最小包装单位、补货周期
  675. var curStd = standards.FirstOrDefault(p => p.Model == item);
  676. decimal packQty = 1m;//最小包装单位
  677. decimal cycle = 0m;//补货周期
  678. if (curStd != null)
  679. {
  680. var curItem = items.FirstOrDefault(p => p.number == curStd.ItemNumber);
  681. packQty = curItem == null ? 1 : (curItem.minpackqty.GetValueOrDefault() == 0.0m ? 1 : curItem.minpackqty.Value);
  682. cycle = curStd.ReplenishCycle;
  683. }
  684. //T2海王预测数据
  685. var curT2Hws = T2Fcsts.Where(p => p.Model == item && p.TypeEnum == 4).ToList();
  686. //计算Rop:N+1月使用N+2月的再订货点参与计算
  687. decimal rop = CalcRop(curFcsts[0].PlanMonth + "-01", curT2Hws.Sum(p => p.Qty), packQty, holidays, cycle);
  688. //获取海王当前规格型号库存
  689. sumQty = pInventories.Where(p => p.SpecificationModel == item && p.Code == "HW0001").Sum(p=>p.InventoryQuantity);
  690. //获取N+1月的库存监控设置
  691. var curMonitor = monitorSettings.FirstOrDefault(p => p.Platform == "海王" && p.ProdType == curFcsts[0].ProdType);
  692. if (curMonitor == null)
  693. {
  694. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "海王未维护产品类型为【"+ curFcsts[0].ProdType + "】的库存监控月份", _currentTenant.Id.ToString());
  695. continue;
  696. }
  697. //最小覆盖月份库存
  698. var minQty = rop * curMonitor.MinTimes;
  699. //最大库存月份库存
  700. var maxQty = rop * curMonitor.MaxTimes;
  701. //需补货数量
  702. decimal needQty = 0m;
  703. if (sumQty >= minQty && sumQty < maxQty)//大于等于最低库存覆盖月,小于最高库存覆盖月
  704. {
  705. planorders.Add(new crm_planorder
  706. {
  707. PlanMonth = strN1,
  708. Model = curFcsts[0].Model,
  709. ItemNum = curStd?.ItemNumber,
  710. ProdLine = curT2Hws[0].ProdLine,
  711. ProdType = "",
  712. Qty = maxQty - sumQty,
  713. Type = "计划单-T2平台补货"
  714. });
  715. }
  716. if (sumQty < minQty)//低于最低库存覆盖月
  717. {
  718. while (sumQty < minQty)
  719. {
  720. sumQty += rop;
  721. needQty += rop;
  722. }
  723. planorders.Add(new crm_planorder
  724. {
  725. PlanMonth = strN1,
  726. Model = curFcsts[0].Model,
  727. ItemNum = curStd?.ItemNumber,
  728. ProdLine = curT2Hws[0].ProdLine,
  729. ProdType = "",
  730. Qty = needQty,
  731. Type = "计划单-T2平台补货"
  732. });
  733. }
  734. }
  735. //计算平台补货:国科
  736. var gkFcsts = platformFcsts.Where(p => p.Platform == "国科").ToList();
  737. var gkModels = gkFcsts.Select(p => p.Model).Distinct().ToList();
  738. foreach (var item in gkModels)
  739. {
  740. //获取当前规格型号对应的预测数据
  741. var curFcsts = gkFcsts.Where(p => p.Model == item).ToList();
  742. //当前规格型号对应的库存数量
  743. decimal sumQty = 0.00m;
  744. //当前规格型号对应标准SKU的最小包装单位、补货周期
  745. var curStd = standards.FirstOrDefault(p => p.Model == item);
  746. decimal packQty = 1m;//最小包装单位
  747. decimal cycle = 0m;//补货周期
  748. if (curStd != null)
  749. {
  750. var curItem = items.FirstOrDefault(p => p.number == curStd.ItemNumber);
  751. packQty = curItem == null ? 1 : (curItem.minpackqty.GetValueOrDefault() == 0.0m ? 1 : curItem.minpackqty.Value);
  752. cycle = curStd.ReplenishCycle;
  753. }
  754. //T2国科预测数据
  755. var curT2Gks = T2Fcsts.Where(p => p.Model == item && p.TypeEnum == 5).ToList();
  756. //计算Rop:N+1月使用N+2月的再订货点参与计算
  757. decimal rop = CalcRop(curFcsts[0].PlanMonth + "-01", curT2Gks.Sum(p => p.Qty), packQty, holidays, cycle);
  758. //获取国科当前规格型号库存
  759. sumQty = pInventories.Where(p => p.SpecificationModel == item && p.Code == "GK0001").Sum(p => p.InventoryQuantity);
  760. //获取N+1月的库存监控设置
  761. var curMonitor = monitorSettings.FirstOrDefault(p => p.Platform == "国科" && p.ProdType == curFcsts[0].ProdType);
  762. if (curMonitor == null)
  763. {
  764. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("DemandAnalysis", "国科未维护产品类型为【" + curFcsts[0].ProdType + "】的库存监控月份", _currentTenant.Id.ToString());
  765. continue;
  766. }
  767. //最小覆盖月份库存
  768. var minQty = rop * curMonitor.MinTimes;
  769. //最大库存月份库存
  770. var maxQty = rop * curMonitor.MaxTimes;
  771. //需补货数量
  772. decimal needQty = 0m;
  773. if (sumQty >= minQty && sumQty < maxQty)
  774. {
  775. planorders.Add(new crm_planorder
  776. {
  777. PlanMonth = strN1,
  778. Model = curFcsts[0].Model,
  779. ItemNum = curStd?.ItemNumber,
  780. ProdLine = curT2Gks[0].ProdLine,
  781. ProdType = "",
  782. Qty = maxQty - sumQty,
  783. Type = "计划单-T2平台补货"
  784. });
  785. }
  786. if (sumQty < minQty)
  787. {
  788. while (sumQty < minQty)
  789. {
  790. sumQty += rop;
  791. needQty += rop;
  792. }
  793. planorders.Add(new crm_planorder
  794. {
  795. PlanMonth = strN1,
  796. Model = curFcsts[0].Model,
  797. ItemNum = curStd?.ItemNumber,
  798. ProdLine = curT2Gks[0].ProdLine,
  799. ProdType = "",
  800. Qty = needQty,
  801. Type = "计划单-T2平台补货"
  802. });
  803. }
  804. }
  805. planorders.ForEach(p => {
  806. p.Source = "系统运算";
  807. p.tenant_id = input.tenant_id;
  808. p.company_id = input.company_id;
  809. p.factory_id = input.factory_id;
  810. p.org_id = input.org_id;
  811. p.create_by = input.create_by;
  812. p.create_by_name = input.create_by_name;
  813. p.create_time = DateTime.Now;
  814. });
  815. return planorders;
  816. }
  817. /// <summary>
  818. /// 计算再订货点
  819. /// </summary>
  820. /// <param name="planMonth">计划年月(yyyy-MM-dd)</param>
  821. /// <param name="qty">需求量</param>
  822. /// <param name="qty">最小包装数量</param>
  823. /// <param name="holidays">节假日</param>
  824. /// <param name="replenishCycle">补货周期</param>
  825. /// <returns></returns>
  826. private decimal CalcRop(string planMonth,decimal qty,decimal packQty,List<HolidayMaster> holidays,decimal replenishCycle)
  827. {
  828. decimal rop = 0.0m;
  829. //获取当月天数
  830. int year = Convert.ToInt16(planMonth.Substring(0, 4));
  831. int month = Convert.ToInt16(planMonth.Substring(5, 2));
  832. int days = DateTime.DaysInMonth(year, month);
  833. //计算当前月的周末天数
  834. int weeks = CalcWeekDays(days, Convert.ToDateTime(planMonth));
  835. //获取当前年月的节假日设置
  836. var curHolidays = holidays.Where(p => p.Dated.Value.Year == year && p.Dated.Value.Month == month).ToList();
  837. //当月工作天数
  838. int workDays = days - weeks - curHolidays.Where(p => p.Ufld1 == "休假").Count() + curHolidays.Where(p => p.Ufld1 == "调班").Count();
  839. //rop = (需求量/月工作天数*补货周期)=>按照最小包装单位元整
  840. rop = Math.Ceiling(qty / workDays * replenishCycle / packQty) * packQty;
  841. return rop;
  842. }
  843. /// <summary>
  844. /// 计算当月有多少个周末
  845. /// </summary>
  846. /// <param name="days"></param>
  847. /// <param name="startDay"></param>
  848. /// <returns></returns>
  849. private int CalcWeekDays(int days, DateTime startDay)
  850. {
  851. int sumDays = 0;
  852. for (int i = 0; i < days; i++)
  853. {
  854. int weekDays = (int)startDay.AddDays(i).DayOfWeek;
  855. if (weekDays == 0 || weekDays == 6)
  856. {
  857. sumDays++;
  858. }
  859. }
  860. return sumDays;
  861. }
  862. /// <summary>
  863. /// 生成整体需求计划
  864. /// </summary>
  865. /// <param name="input"></param>
  866. /// <returns></returns>
  867. /// <exception cref="NotImplementedException"></exception>
  868. public async Task<string> OverallDemandPlan(InputDto input)
  869. {
  870. //1.1、获取海外销售预测数据
  871. List<OverseasSaleFcst> overseasSales = _overseasSaleFcst.GetListAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result.OrderBy(p => p.OrderNum).ToList();
  872. //1.2、获取平台预测收集数据
  873. List<PlatformFcstCollect> platformFcsts = _platformFcstCollect.GetListAsync(p => p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result.OrderBy(p=>p.OrderNum).ToList();
  874. //1.3、获取国内终端预测-T1汇总数据
  875. List<DomesticTerminalFcst> domesticFcsts = _domesticTerminalFcst.GetListAsync(p => p.TypeEnum == 2 && p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id && !p.IsDeleted).Result.OrderBy(p => p.OrderNum).ToList();
  876. //计算当前年月的N+1,N+2
  877. string strN0 = input.year.ToString() + "-" + input.month.ToString("00");
  878. int newYear = input.month == 12 ? input.year + 1 : input.year;
  879. int newMonth = input.month == 12 ? 1 : input.month + 1;
  880. string strN1 = newYear.ToString() + "-" + newMonth.ToString("00");
  881. newYear = newMonth == 12 ? newYear + 1 : newYear;
  882. newMonth = newMonth == 12 ? 1 : newMonth + 1;
  883. string strN2 = newYear.ToString() + "-" + newMonth.ToString("00");
  884. //整体需求计划
  885. List<OverallDemandPlan> plans = new List<OverallDemandPlan>();
  886. //计算海外
  887. List<string> hwModels = overseasSales.Select(p => p.Model).Distinct().ToList();
  888. int OrderNum = 1;
  889. foreach (var item in hwModels)
  890. {
  891. var curFcsts = overseasSales.Where(p => p.Model == item).ToList();
  892. OverallDemandPlan plan = new OverallDemandPlan();
  893. plan.Area = "海外";
  894. plan.Model = item;
  895. plan.PlanMonth = strN0;
  896. plan.Qty = curFcsts.Where(p => p.PlanMonth == strN0).Sum(p => p.Qty);
  897. plan.OrderNum = OrderNum;
  898. plans.Add(plan);
  899. plan = new OverallDemandPlan();
  900. plan.Area = "海外";
  901. plan.Model = item;
  902. plan.PlanMonth = strN1;
  903. plan.Qty = curFcsts.Where(p => p.PlanMonth == strN1).Sum(p => p.Qty);
  904. plan.OrderNum = OrderNum;
  905. plans.Add(plan);
  906. plan = new OverallDemandPlan();
  907. plan.Area = "海外";
  908. plan.Model = item;
  909. plan.PlanMonth = strN2;
  910. plan.Qty = curFcsts.Where(p => p.PlanMonth == strN2).Sum(p => p.Qty);
  911. plan.OrderNum = OrderNum;
  912. plans.Add(plan);
  913. OrderNum += 1;
  914. }
  915. //计算国内
  916. List<string> gnModels = domesticFcsts.Select(p=>p.Model).ToList();
  917. gnModels.AddRange(platformFcsts.Select(p => p.Model).ToList());
  918. gnModels = gnModels.Distinct().ToList();
  919. OrderNum = 1;
  920. foreach (var item in gnModels)
  921. {
  922. var curDFcsts = domesticFcsts.Where(p => p.Model == item).ToList();
  923. var curPFcsts = platformFcsts.Where(p => p.Model == item).ToList();
  924. OverallDemandPlan plan = new OverallDemandPlan();
  925. plan.Area = "国内";
  926. plan.Model = item;
  927. plan.PlanMonth = strN0;
  928. plan.Qty = curDFcsts.Where(p => p.PlanMonth == strN0).Sum(p => p.Qty) + curPFcsts.Where(p => p.PlanMonth == strN0).Sum(p => p.Qty);
  929. plan.OrderNum = OrderNum;
  930. plans.Add(plan);
  931. plan = new OverallDemandPlan();
  932. plan.Area = "国内";
  933. plan.Model = item;
  934. plan.PlanMonth = strN1;
  935. plan.Qty = curDFcsts.Where(p => p.PlanMonth == strN1).Sum(p => p.Qty) + curPFcsts.Where(p => p.PlanMonth == strN1).Sum(p => p.Qty);
  936. plan.OrderNum = OrderNum;
  937. plans.Add(plan);
  938. plan = new OverallDemandPlan();
  939. plan.Area = "国内";
  940. plan.Model = item;
  941. plan.PlanMonth = strN2;
  942. plan.Qty = curDFcsts.Where(p => p.PlanMonth == strN2).Sum(p => p.Qty) + curPFcsts.Where(p => p.PlanMonth == strN2).Sum(p => p.Qty);
  943. plan.OrderNum = OrderNum;
  944. plans.Add(plan);
  945. OrderNum += 1;
  946. }
  947. plans.ForEach(p => {
  948. p.Year= input.year;
  949. p.Month= input.month;
  950. p.tenant_id = input.tenant_id;
  951. p.company_id = input.company_id;
  952. p.factory_id= input.factory_id;
  953. p.org_id= input.org_id;
  954. p.create_by = input.create_by;
  955. p.create_by_name= input.create_by_name;
  956. p.create_time = DateTime.Now;
  957. });
  958. //保存数据
  959. using (var unitOfWork = _unitOfWorkManager.Begin(false, true))
  960. {
  961. try
  962. {
  963. //先删除
  964. await _overallDemandPlan.HardDeleteAsync(p=>p.Year == input.year && p.Month == input.month && p.tenant_id == input.tenant_id && p.company_id == input.company_id && p.factory_id == input.factory_id);
  965. //保存整体需求计划
  966. await _overallDemandPlan.InsertManyAsync(plans);
  967. await unitOfWork.CompleteAsync();
  968. }
  969. catch (Exception e)
  970. {
  971. unitOfWork.Dispose();
  972. new NLogHelper("MonthlyCapacityLoadAppService").WriteLog("OverallDemandPlan", "生成【" + input.year + "年" + input.month + "月】整体需求计划失败:" + e.Message, _currentTenant.Id.ToString());
  973. return "NO|" + e.Message;
  974. };
  975. }
  976. return "OK|刷新成功!";
  977. }
  978. }
  979. }