|  |  |  | 
|---|
|  |  |  | using AutoMapper; | 
|---|
|  |  |  | using MailKit.Search; | 
|---|
|  |  |  | using Newtonsoft.Json; | 
|---|
|  |  |  | using OfficeOpenXml.FormulaParsing.Excel.Functions.RefAndLookup; | 
|---|
|  |  |  | using SqlSugar; | 
|---|
|  |  |  | using System; | 
|---|
|  |  |  | using System.Collections; | 
|---|
|  |  |  | using System.Collections.Generic; | 
|---|
|  |  |  | using System.Linq; | 
|---|
|  |  |  | using System.Net.Http; | 
|---|
|  |  |  | using System.Text; | 
|---|
|  |  |  | using System.Threading.Tasks; | 
|---|
|  |  |  | using WIDESEA_Common.StockEnum; | 
|---|
|  |  |  | 
|---|
|  |  |  | using WIDESEA_IStockRepository; | 
|---|
|  |  |  | using WIDESEA_IStockService; | 
|---|
|  |  |  | using WIDESEA_Model.Models; | 
|---|
|  |  |  | using WIDESEA_StockRepository; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | namespace WIDESEA_StockService | 
|---|
|  |  |  | { | 
|---|
|  |  |  | 
|---|
|  |  |  | throw new Exception($"æªæ¾å°åºåºåä¿¡æ¯"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<string> locationCodes = _basicRepository.LocationInfoRepository.GetCanOutLocationCodes(outboundOrder.WarehouseId); | 
|---|
|  |  |  | List<string> locationCodes = _basicRepository.LocationInfoRepository.PPGetCanOutLocationCodes(outboundOrder.WarehouseId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return BaseDal.QueryTabs<Dt_StockInfo, Dt_StockInfoDetail, StockSelectViewDTO>((a, b) => a.Id == b.StockId, (a, b) => new StockSelectViewDTO | 
|---|
|  |  |  | { | 
|---|
|  |  |  | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// æ¥è¯¢è®¢åPPç«åºåºåè§å¾ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="orderId"></param> | 
|---|
|  |  |  | /// <param name="materielCode"></param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | public List<PPStockSelectViewDTO> PPGetStockSelectViews(int orderId, string materielCode) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | try | 
|---|
|  |  |  | { | 
|---|
|  |  |  | Dt_MesPPCutOutboundOrder mesPPCutOutboundOrder = _outboundRepository.MesPPCutOutboundOrderRepository.QueryFirst(x => x.Id == orderId); | 
|---|
|  |  |  | if (mesPPCutOutboundOrder == null) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | throw new Exception($"æªæ¾å°åºåºåä¿¡æ¯"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | List<string> locationCodes = _basicRepository.LocationInfoRepository.PPGetCanOutLocationCodes(mesPPCutOutboundOrder.WarehouseId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return BaseDal.QueryTabs<Dt_StockInfo, Dt_StockInfoDetail, PPStockSelectViewDTO>((a, b) => a.Id == b.StockId, (a, b) => new PPStockSelectViewDTO | 
|---|
|  |  |  | { | 
|---|
|  |  |  | LocationCode = a.LocationCode, | 
|---|
|  |  |  | MaterielCode = b.MaterielCode, | 
|---|
|  |  |  | MaterielName = b.MaterielName, | 
|---|
|  |  |  | PalletCode = a.PalletCode, | 
|---|
|  |  |  | Unit = b.Unit, | 
|---|
|  |  |  | CutedWidth = b.CutedWidth, | 
|---|
|  |  |  | UseableQuantity = b.StockQuantity - b.OutboundQuantity | 
|---|
|  |  |  | }, a => locationCodes.Contains(a.LocationCode), b => b.StockQuantity > b.OutboundQuantity && b.MaterielCode == materielCode, x => true).GroupBy(x => x.PalletCode).Select(x => new PPStockSelectViewDTO | 
|---|
|  |  |  | { | 
|---|
|  |  |  | LocationCode = x.FirstOrDefault()?.LocationCode ?? "", | 
|---|
|  |  |  | MaterielCode = x.FirstOrDefault()?.MaterielCode ?? "", | 
|---|
|  |  |  | MaterielName = x.FirstOrDefault()?.MaterielName ?? "", | 
|---|
|  |  |  | Unit = x.FirstOrDefault()?.Unit ?? "", | 
|---|
|  |  |  | CutedWidth = x.Sum(x => x.CutedWidth), | 
|---|
|  |  |  | PalletCode = x.Key, | 
|---|
|  |  |  | UseableQuantity = x.Sum(x => x.UseableQuantity) | 
|---|
|  |  |  | }).ToList(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | catch (Exception ex) | 
|---|
|  |  |  | { | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// æ¥è¯¢è®¢åPPå¹³åºåºåè§å¾ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="orderId"></param> | 
|---|
|  |  |  | /// <param name="materielCode"></param> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | public List<PPStockSelectViewDTO> PPGetPKStockSelectViews(int orderId, string materielCode) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | try | 
|---|
|  |  |  | { | 
|---|
|  |  |  | Dt_MesPPCutOutboundOrder outboundOrder = _outboundRepository.MesPPCutOutboundOrderRepository.QueryFirst(x => x.Id == orderId); | 
|---|
|  |  |  | if (outboundOrder == null) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | throw new Exception($"æªæ¾å°åºåºåä¿¡æ¯"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return BaseDal.QueryTabs<Dt_StockInfo, Dt_StockInfoDetail, PPStockSelectViewDTO>((a, b) => a.Id == b.StockId && a.WarehouseId == outboundOrder.WarehouseId, (a, b) => new PPStockSelectViewDTO | 
|---|
|  |  |  | { | 
|---|
|  |  |  | LocationCode = a.LocationCode, | 
|---|
|  |  |  | MaterielCode = b.MaterielCode, | 
|---|
|  |  |  | MaterielName = b.MaterielName, | 
|---|
|  |  |  | PalletCode = a.PalletCode, | 
|---|
|  |  |  | Unit = b.Unit, | 
|---|
|  |  |  | CutedWidth = b.CutedWidth, | 
|---|
|  |  |  | UseableQuantity = b.StockQuantity - b.OutboundQuantity | 
|---|
|  |  |  | }, a => a.LocationCode.Contains("AGV_PP"), b => b.StockQuantity > b.OutboundQuantity && b.MaterielCode == materielCode, x => true).GroupBy(x => x.PalletCode).Select(x => new PPStockSelectViewDTO | 
|---|
|  |  |  | { | 
|---|
|  |  |  | LocationCode = x.FirstOrDefault()?.LocationCode ?? "", | 
|---|
|  |  |  | MaterielCode = x.FirstOrDefault()?.MaterielCode ?? "", | 
|---|
|  |  |  | MaterielName = x.FirstOrDefault()?.MaterielName ?? "", | 
|---|
|  |  |  | Unit = x.FirstOrDefault()?.Unit ?? "", | 
|---|
|  |  |  | CutedWidth = x.Sum(x => x.CutedWidth), | 
|---|
|  |  |  | PalletCode = x.Key, | 
|---|
|  |  |  | UseableQuantity = x.Sum(x => x.UseableQuantity) | 
|---|
|  |  |  | }).ToList(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | catch (Exception ex) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return null; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public WebResponseContent StockQueryData(SaveModel saveModel) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | 
|---|
|  |  |  | while (needQuantity > 0) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | Dt_StockInfo stockInfo = stockInfos[index]; | 
|---|
|  |  |  | float useableStockQuantity = stockInfo.Details.Where(x => x.MaterielCode == materielCode).Sum(x => x.StockQuantity - x.OutboundQuantity); | 
|---|
|  |  |  | if (useableStockQuantity < needQuantity) | 
|---|
|  |  |  | // è®¡ç®å¯ç¨åºåæ¶è½¬æ¢ä¸ºdecimal | 
|---|
|  |  |  | decimal useableStockQuantity = stockInfo.Details | 
|---|
|  |  |  | .Where(x => x.MaterielCode == materielCode) | 
|---|
|  |  |  | .Sum(x => (decimal)x.StockQuantity - (decimal)x.OutboundQuantity); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // å°needQuantity转æ¢ä¸ºdecimalè¿è¡æ¯è¾ | 
|---|
|  |  |  | if (useableStockQuantity < (decimal)needQuantity && useableStockQuantity > 0) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | stockInfo.Details.ForEach(x => x.OutboundQuantity = x.StockQuantity); | 
|---|
|  |  |  | needQuantity -= useableStockQuantity; | 
|---|
|  |  |  | stockInfo.Details.ForEach(x => | 
|---|
|  |  |  | x.OutboundQuantity = x.StockQuantity); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // ä½¿ç¨decimalè¿è¡è®¡ç®åå转åfloat | 
|---|
|  |  |  | needQuantity = (float)((decimal)needQuantity - useableStockQuantity); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | 
|---|
|  |  |  | { | 
|---|
|  |  |  | if (x.StockQuantity > x.OutboundQuantity && x.MaterielCode == materielCode) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | if (x.StockQuantity - x.OutboundQuantity >= needQuantity) | 
|---|
|  |  |  | // å°ç¸å
³å¼è½¬æ¢ä¸ºdecimalè¿è¡ç²¾ç¡®è®¡ç® | 
|---|
|  |  |  | decimal currentStock = (decimal)x.StockQuantity; | 
|---|
|  |  |  | decimal currentOutbound = (decimal)x.OutboundQuantity; | 
|---|
|  |  |  | decimal currentNeed = (decimal)needQuantity; | 
|---|
|  |  |  | decimal available = currentStock - currentOutbound; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (available >= currentNeed) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | x.OutboundQuantity += needQuantity; | 
|---|
|  |  |  | x.OutboundQuantity = (float)(currentOutbound + currentNeed); | 
|---|
|  |  |  | needQuantity = 0; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | needQuantity -= (x.StockQuantity - x.OutboundQuantity); | 
|---|
|  |  |  | needQuantity = (float)(currentNeed - available); | 
|---|
|  |  |  | x.OutboundQuantity = x.StockQuantity; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | 
|---|
|  |  |  | outStocks.Add(stockInfo); | 
|---|
|  |  |  | index++; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | for (int i = 0; i < stockInfos.Count; i++) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | Dt_StockInfo stockInfo = stockInfos[i]; | 
|---|
|  |  |  | float useableStockQuantity = stockInfo.Details.Where(x => x.MaterielCode == materielCode).Sum(x => x.StockQuantity - x.OutboundQuantity); | 
|---|
|  |  |  | if (useableStockQuantity < needQuantity) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | stockInfo.Details.ForEach(x => x.OutboundQuantity = x.StockQuantity); | 
|---|
|  |  |  | needQuantity -= useableStockQuantity; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | stockInfo.Details.ForEach(x => | 
|---|
|  |  |  | { | 
|---|
|  |  |  | if (x.StockQuantity > x.OutboundQuantity && x.MaterielCode == materielCode) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | if (x.StockQuantity - x.OutboundQuantity >= needQuantity) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | x.OutboundQuantity += needQuantity; | 
|---|
|  |  |  | needQuantity = 0; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | needQuantity -= (x.StockQuantity - x.OutboundQuantity); | 
|---|
|  |  |  | x.OutboundQuantity = x.StockQuantity; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | }); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | outStocks.Add(stockInfo); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | throw new Exception("åºåä¸è¶³"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | residueQuantity = needQuantity; | 
|---|
|  |  |  | return outStocks; | 
|---|
|  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return BaseDal.GetStockInfos(materielCode, batchNo, locationCodes); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | public WebResponseContent MatPicking(SaveModel saveModel) | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public List<Dt_StockInfo> GetUseableStocks(string materielCode, string batchNo, string palletcode, int warehoseId) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | WebResponseContent content= new WebResponseContent(); | 
|---|
|  |  |  | Dt_StockInfo stockInfo = BaseDal.Db.Queryable<Dt_StockInfo>().Where(x => x.PalletCode == palletcode && x.WarehouseId == warehoseId).Includes(x => x.Details).First(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | List<string> locationCodes = _basicRepository.LocationInfoRepository.GetCanOutLocationCodes(stockInfo.LocationCode); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return BaseDal.GetStockInfos(materielCode, batchNo, locationCodes); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | public WebResponseContent UpdateExpirationlabel() | 
|---|
|  |  |  | { | 
|---|
|  |  |  | try | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var today = DateTime.Today; | 
|---|
|  |  |  | int batchSize = 1000; | 
|---|
|  |  |  | int totalUpdated = 0; | 
|---|
|  |  |  | int skipCount = 0; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // åªæ¥è¯¢éè¦çåæ®µï¼åå°æ°æ®ä¼ è¾åå
åå ç¨ | 
|---|
|  |  |  | var query = BaseDal.Db.Queryable<Dt_StockInfoDetail>() | 
|---|
|  |  |  | .InnerJoin<Dt_StockInfo>((detail, master) => detail.StockId == master.Id) | 
|---|
|  |  |  | .Select((detail, master) => new | 
|---|
|  |  |  | { | 
|---|
|  |  |  | MasterId = master.Id, | 
|---|
|  |  |  | master.WarehouseId, | 
|---|
|  |  |  | detail.EffectiveDate, | 
|---|
|  |  |  | CurrentExpirationlabel = master.Expirationlabel | 
|---|
|  |  |  | }); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | while (true) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var batchData = query.Skip(skipCount).Take(batchSize).ToList(); | 
|---|
|  |  |  | if (!batchData.Any()) break; | 
|---|
|  |  |  | var groupedData = batchData.GroupBy(item => item.MasterId) | 
|---|
|  |  |  | .Select(g => new | 
|---|
|  |  |  | { | 
|---|
|  |  |  | MasterId = g.Key, | 
|---|
|  |  |  | WarehouseId = g.First().WarehouseId, | 
|---|
|  |  |  | // åææ©çæææ¥æ | 
|---|
|  |  |  | EarliestEffectiveDate = g.Min(item => | 
|---|
|  |  |  | { | 
|---|
|  |  |  | DateTime.TryParse(item.EffectiveDate, out DateTime date); | 
|---|
|  |  |  | return date; | 
|---|
|  |  |  | }), | 
|---|
|  |  |  | CurrentExpirationlabel = g.First().CurrentExpirationlabel | 
|---|
|  |  |  | }) | 
|---|
|  |  |  | .ToList(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var updateDic = new Dictionary<long, int>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | foreach (var group in groupedData) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | DateTime effectiveDate = group.EarliestEffectiveDate; | 
|---|
|  |  |  | if (effectiveDate == default(DateTime)) // å¤çè§£æå¤±è´¥çæ
åµ | 
|---|
|  |  |  | { | 
|---|
|  |  |  | Console.WriteLine($"主表ID {group.MasterId} ä¸æ æææ¥æï¼è·³è¿"); | 
|---|
|  |  |  | continue; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int newLabel; | 
|---|
|  |  |  | if (effectiveDate < today) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | newLabel = ExpirationlabelEnum.è¿æ.ObjToInt(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else if (group.WarehouseId == 3) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | int daysDiff = (effectiveDate - today).Days; | 
|---|
|  |  |  | newLabel = daysDiff < 60 | 
|---|
|  |  |  | ? ExpirationlabelEnum.临æé¢è¦.ObjToInt() | 
|---|
|  |  |  | : ExpirationlabelEnum.æªä¸´æ.ObjToInt(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | else | 
|---|
|  |  |  | { | 
|---|
|  |  |  | int daysDiff = (effectiveDate - today).Days; | 
|---|
|  |  |  | newLabel = daysDiff < 30 | 
|---|
|  |  |  | ? ExpirationlabelEnum.临æé¢è¦.ObjToInt() | 
|---|
|  |  |  | : ExpirationlabelEnum.æªä¸´æ.ObjToInt(); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (newLabel != group.CurrentExpirationlabel && !updateDic.ContainsKey(group.MasterId)) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | updateDic[group.MasterId] = newLabel; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (updateDic.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var updateBuilder = BaseDal.Db.Updateable<Dt_StockInfo>(); | 
|---|
|  |  |  | var idsToUpdate = updateDic.Keys.ToList(); | 
|---|
|  |  |  | int updateValue = updateDic.First().Value; | 
|---|
|  |  |  | updateBuilder.SetColumns(m => m.Expirationlabel == updateValue) | 
|---|
|  |  |  | .Where(m => idsToUpdate.Contains(m.Id)); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | int batchUpdated = updateBuilder.ExecuteCommand(); | 
|---|
|  |  |  | totalUpdated += batchUpdated; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | Console.WriteLine($"æ¹æ¬¡æ´æ°ï¼{batchUpdated} æ¡ï¼ç´¯è®¡æ´æ°ï¼{totalUpdated} æ¡ï¼æ´æ°æ¡ä»¶ï¼{JsonConvert.SerializeObject(idsToUpdate)}"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | skipCount += batchSize; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | return WebResponseContent.Instance.OK($"æ´æ°æåï¼å
±æ´æ° {totalUpdated} æ¡è®°å½"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | catch (Exception ex) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | content.Error(ex.Message); | 
|---|
|  |  |  | return WebResponseContent.Instance.Error("æ´æ°å¤±è´¥ï¼è¯·è系管çå"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | return content; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// ééæºå¨äººæ¶æ¯æ¨éæµè¯ | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <returns></returns> | 
|---|
|  |  |  | public async Task T0DingTalkText(string webhookUrl, string secret) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | try | 
|---|
|  |  |  | { | 
|---|
|  |  |  | if (webhookUrl == null || secret == null) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | webhookUrl = "https://oapi.dingtalk.com/robot/send?access_token=fbc3aaf4133ea650d8116fb86b3ebfd0c5e0d46775966ce87893a41886bdf9dc"; | 
|---|
|  |  |  | secret = "SECf221842b26356f22ccac84c4e60714e5287408ee8332a8f63503791382c3f5fb"; | 
|---|
|  |  |  | } | 
|---|
|  |  |  | var warehouseIdToName = new Dictionary<int, string> | 
|---|
|  |  |  | { | 
|---|
|  |  |  | { 2, "油墨ä»" }, | 
|---|
|  |  |  | { 3, "æ¿æä»" }, | 
|---|
|  |  |  | { 4, "PPä»" }, | 
|---|
|  |  |  | { 6, "æµè¯æ¶ä»" }, | 
|---|
|  |  |  | { 11, "å¹²èä»" }, | 
|---|
|  |  |  | { 12, "é»çä»" } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var currentTime = DateTime.Now; | 
|---|
|  |  |  | var ninetyDaysAgo = currentTime.AddDays(-90); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var expirationLabel3Stocks = BaseDal.Db.Queryable<Dt_StockInfo>() | 
|---|
|  |  |  | .Where(s => s.Expirationlabel == 3 && s.WarehouseId != 5) | 
|---|
|  |  |  | .ToList(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var over90DaysStocks = BaseDal.Db.Queryable<Dt_StockInfo>() | 
|---|
|  |  |  | .Where(s => s.ModifyDate <= ninetyDaysAgo && s.Expirationlabel != 3 && s.WarehouseId != 5) | 
|---|
|  |  |  | .ToList(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | if (!expirationLabel3Stocks.Any() && !over90DaysStocks.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | return; | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var expirationLabel3StockIds = expirationLabel3Stocks.Select(s => s.Id).ToList(); | 
|---|
|  |  |  | var over90DaysStockIds = over90DaysStocks.Select(s => s.Id).ToList(); | 
|---|
|  |  |  | var expirationLabel3Details = expirationLabel3StockIds.Any() | 
|---|
|  |  |  | ? BaseDal.Db.Queryable<Dt_StockInfoDetail>() | 
|---|
|  |  |  | .Where(d => expirationLabel3StockIds.Contains(d.StockId)) | 
|---|
|  |  |  | .ToList() | 
|---|
|  |  |  | : new List<Dt_StockInfoDetail>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var over90DaysDetails = over90DaysStockIds.Any() | 
|---|
|  |  |  | ? BaseDal.Db.Queryable<Dt_StockInfoDetail>() | 
|---|
|  |  |  | .Where(d => over90DaysStockIds.Contains(d.StockId)) | 
|---|
|  |  |  | .ToList() | 
|---|
|  |  |  | : new List<Dt_StockInfoDetail>(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var stockIdToWarehouseId = expirationLabel3Stocks | 
|---|
|  |  |  | .Concat(over90DaysStocks) | 
|---|
|  |  |  | .ToDictionary(s => s.Id, s => s.WarehouseId); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var expirationLabel3Groups = expirationLabel3Details | 
|---|
|  |  |  | .GroupBy(d => stockIdToWarehouseId[d.StockId]) | 
|---|
|  |  |  | .ToDictionary(g => g.Key, g => g.Select(d => new | 
|---|
|  |  |  | { | 
|---|
|  |  |  | d.MaterielCode, | 
|---|
|  |  |  | d.BatchNo | 
|---|
|  |  |  | }).ToList()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var over90DaysGroups = over90DaysDetails | 
|---|
|  |  |  | .GroupBy(d => stockIdToWarehouseId[d.StockId]) | 
|---|
|  |  |  | .ToDictionary(g => g.Key, g => g.Select(d => new | 
|---|
|  |  |  | { | 
|---|
|  |  |  | d.MaterielCode, | 
|---|
|  |  |  | d.BatchNo | 
|---|
|  |  |  | }).ToList()); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var allWarehouseIds = expirationLabel3Groups.Keys | 
|---|
|  |  |  | .Union(over90DaysGroups.Keys) | 
|---|
|  |  |  | .ToList(); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | using (HttpClient httpClient = new HttpClient()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | foreach (var warehouseId in allWarehouseIds) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | // 9.1 å¤çä»åºåç§°æ¾ç¤ºï¼ä¼å
ç¨æ å°åç§°ï¼æ æ å°æ¶æ¾ç¤ºåå§ID | 
|---|
|  |  |  | var warehouseName = warehouseIdToName.TryGetValue(warehouseId, out var name) | 
|---|
|  |  |  | ? name | 
|---|
|  |  |  | : $"ä»åº{warehouseId}"; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var messageParts = new List<string> | 
|---|
|  |  |  | { | 
|---|
|  |  |  | $"ã{warehouseName}ãç©ææé" | 
|---|
|  |  |  | }; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // æ·»å è¿æç©æä¿¡æ¯ | 
|---|
|  |  |  | if (expirationLabel3Groups.TryGetValue(warehouseId, out var label3Materials) && label3Materials.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var label3Details = label3Materials | 
|---|
|  |  |  | .Select(m => $"â¢ ç©æç¼ç ï¼{m.MaterielCode} | æ¹æ¬¡å·ï¼{m.BatchNo}") | 
|---|
|  |  |  | .Aggregate((current, next) => $"{current}\n{next}"); | 
|---|
|  |  |  | messageParts.Add($"ä¸ãè¿æç©æ\n{label3Details}"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // æ·»å è¶
è¿90天æªä½¿ç¨ç©æä¿¡æ¯ | 
|---|
|  |  |  | if (over90DaysGroups.TryGetValue(warehouseId, out var over90Materials) && over90Materials.Any()) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var over90Details = over90Materials | 
|---|
|  |  |  | .Select(m => $"â¢ ç©æç¼ç ï¼{m.MaterielCode} | æ¹æ¬¡å·ï¼{m.BatchNo}") | 
|---|
|  |  |  | .Aggregate((current, next) => $"{current}\n{next}"); | 
|---|
|  |  |  | messageParts.Add($"äºãè¶
è¿90天æªä½¿ç¨ç©æ\n{over90Details}"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var messageContent = string.Join("\n\n", messageParts); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var timestamp = DateTimeOffset.Now.ToUnixTimeMilliseconds(); | 
|---|
|  |  |  | var sign = GenerateSign(timestamp, secret); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var url = $"{webhookUrl.Split('?')[0]}?access_token={new Uri(webhookUrl).Query.Split('=')[1]}×tamp={timestamp}&sign={sign}"; | 
|---|
|  |  |  |  | 
|---|
|  |  |  | // æå»ºè¯·æ±ä½ï¼ç¬¦åéétextç±»åæ¶æ¯æ ¼å¼ï¼ | 
|---|
|  |  |  | var requestBody = new | 
|---|
|  |  |  | { | 
|---|
|  |  |  | msgtype = "text", | 
|---|
|  |  |  | text = new { content = messageContent } | 
|---|
|  |  |  | }; | 
|---|
|  |  |  | var jsonBody = JsonConvert.SerializeObject(requestBody); | 
|---|
|  |  |  | var content = new StringContent(jsonBody, Encoding.UTF8, "application/json"); | 
|---|
|  |  |  |  | 
|---|
|  |  |  | var response = await httpClient.PostAsync(url, content); | 
|---|
|  |  |  | if (!response.IsSuccessStatusCode) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var errorContent = await response.Content.ReadAsStringAsync(); | 
|---|
|  |  |  | throw new Exception($"ã{warehouseName}ãæ¶æ¯åé失败ï¼ç¶æç ï¼{response.StatusCode}ï¼é误信æ¯ï¼{errorContent}"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | catch (Exception ex) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | throw new Exception($"ééæ¶æ¯æ¨éæ´ä½å¤±è´¥ï¼é误详æ
ï¼{ex.Message}", ex); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  |  | 
|---|
|  |  |  |  | 
|---|
|  |  |  | /// <summary> | 
|---|
|  |  |  | /// çæå ç¾ç¾å | 
|---|
|  |  |  | /// </summary> | 
|---|
|  |  |  | /// <param name="timestamp">æ¶é´æ³</param> | 
|---|
|  |  |  | /// <returns>ç¾å</returns> | 
|---|
|  |  |  | private string GenerateSign(long timestamp,string secret) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var stringToSign = $"{timestamp}\n{secret}"; | 
|---|
|  |  |  | using (var hmacsha256 = new System.Security.Cryptography.HMACSHA256(Encoding.UTF8.GetBytes(secret))) | 
|---|
|  |  |  | { | 
|---|
|  |  |  | var hashBytes = hmacsha256.ComputeHash(Encoding.UTF8.GetBytes(stringToSign)); | 
|---|
|  |  |  | return Convert.ToBase64String(hashBytes).Replace("+", "%2B").Replace("/", "%2F"); | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|
|  |  |  | } | 
|---|