| | |
| | | using HH.WCS.QingXigongchang.process; |
| | | using HH.WCS.QingXigongchang.util; |
| | | using HH.WCS.QingXigongchang.wms; |
| | | using Newtonsoft.Json; |
| | | using System; |
| | | using System.Collections.Generic; |
| | | using System.Linq; |
| | | using System.Web.Configuration; |
| | | using System.Xml.Schema; |
| | | using static HH.WCS.QingXigongchang.api.ApiModel; |
| | | |
| | |
| | | var list = TaskHelper.GetTaskListByState("未执行"); |
| | | if (list.Count > 0) |
| | | { |
| | | //foreach (var mst in list) |
| | | //{ |
| | | // var dinfo = Settings.GetDeviceInfoList().FindAll(x => x.deviceType == 13); |
| | | // if (dinfo.Find(x => x.location.Contains(mst.S_START_LOC)) != null) |
| | | // mst.N_PRIORITY = 99; |
| | | // else |
| | | // if (dinfo.Find(x => x.location.Contains(mst.S_END_LOC)) != null) |
| | | // mst.N_PRIORITY = 98; |
| | | //} |
| | | //list = list.OrderByDescending(x => x.N_PRIORITY).ToList(); |
| | | |
| | | list.ForEach(task => |
| | | { |
| | | if (!TaskProcess.Intercept(task)) |
| | |
| | | } |
| | | } |
| | | |
| | | /// <summary> |
| | | /// 60分钟更新一次库容 |
| | | /// </summary> |
| | | /// <param name="min"></param> |
| | | public static void CheckKuRong(int min) |
| | | { |
| | | try |
| | | { |
| | | Console.WriteLine("CheckKurRong"); |
| | | //如果表内没有数据就初始化一下,有数据了modify时间和当前时间超过30min再统计一次 |
| | | var db = new SqlHelper<object>().GetInstance(); |
| | | var info = db.Queryable<KuRong>().First(); |
| | | if (info == null || DateTime.Now.Subtract(info.T_MODIFY).TotalMinutes > min) |
| | | { |
| | | //1、查到小板库区下面所有的库位 |
| | | var totalX = 0; |
| | | var totalRows = 0; |
| | | var list = new List<KuRong>(); |
| | | var areas = db.Queryable<Location>().Select(x => x.S_AREA_CODE).ToList().FindAll(x => System.Text.RegularExpressions.Regex.IsMatch(x, @"QX-\d+_[A-Z]+")).Select(x => x.Split('_')[0]).Distinct().ToList(); //GetAreas(); |
| | | areas.ForEach(area => |
| | | { |
| | | totalX += db.Queryable<Location>().Count(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_X")); |
| | | totalRows += db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_X")).Select(b => b.N_ROW).Distinct().Count(); |
| | | var aresList = db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_")).Select(b => b.S_AREA_CODE).Distinct().ToList(); |
| | | aresList = aresList.Select(a => a.Split('_')[0]).Distinct().ToList(); |
| | | aresList.ForEach(a => |
| | | { |
| | | //b => b.S_AREA_CODE == a + "_X" |
| | | Console.WriteLine("area:" + a); |
| | | // && b.S_AREA_CODE.EndsWith("X") |
| | | var rows = db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(a)).Select(b => b.N_ROW).Distinct().ToList(); |
| | | if (rows.Count > 0) |
| | | { |
| | | Console.WriteLine(string.Join(",", rows)); |
| | | rows.ForEach(b => |
| | | { |
| | | var kr = new KuRong() { StorageLoc = a.Split('-')[1] + "-" + b, T_MODIFY = DateTime.Now }; |
| | | Console.WriteLine(kr.StorageLoc); |
| | | //每一排要查一下有没有托盘 |
| | | var listLoc = db.Queryable<Location>().Where(c => c.S_AREA_CODE.Contains(c.S_AREA_CODE) && c.N_ROW == b && c.N_CURRENT_NUM > 0).ToList(); |
| | | if (!listLoc.Any()) |
| | | { |
| | | //没托盘,默认最大容量是小板的数量 |
| | | var count = db.Queryable<Location>().Count(c => c.S_AREA_CODE == a + "_X" && c.N_ROW == b) * 2 * 2;//2层2托 |
| | | kr.MaxCapacity = count; |
| | | kr.AvailableCapacity = count; |
| | | kr.ProductItem = ""; |
| | | } |
| | | else |
| | | { |
| | | //有托盘 判断是叠2层还是3层,要获取物料信息 |
| | | var lcr = db.Queryable<LocCntrRel>().Includes(it => it.CntrItemRel).Where(l => l.S_LOC_CODE == listLoc[0].S_LOC_CODE).First(); |
| | | if (lcr != null && lcr.CntrItemRel != null) |
| | | { |
| | | var maxLayer = ContainerHelper.GetItem(ass => ass.S_ITEM_NAME == lcr.CntrItemRel.S_ITEM_NAME && ass.S_ITEM_CODE == lcr.CntrItemRel.S_ITEM_CODE)?.MaxLayer ?? listLoc[0].N_CAPACITY; |
| | | var total = db.Queryable<Location>().Count(c => c.S_AREA_CODE == listLoc[0].S_AREA_CODE && c.N_ROW == b) * maxLayer * 2; |
| | | var count = db.Queryable<Location>().Where(c => c.S_AREA_CODE == listLoc[0].S_AREA_CODE && c.N_ROW == b && c.N_CURRENT_NUM > 0).Sum(c => c.N_CURRENT_NUM) * 2; // 指定统计字段 |
| | | |
| | | kr.ProductItem = lcr.CntrItemRel.S_ITEM_CODE; |
| | | kr.MaxCapacity = total; |
| | | kr.FilledCapacity = count; |
| | | kr.AvailableCapacity = total - count; |
| | | } |
| | | } |
| | | list.Add(kr); |
| | | }); |
| | | } |
| | | else |
| | | { |
| | | Console.WriteLine("area:" + a + ":未获取到排"); |
| | | } |
| | | }); |
| | | |
| | | }); |
| | | var ordersCount = db.Queryable<WorkOrder>().Where(a => a.SQL_State == "执行中").Count(); |
| | | var totalDesiredCapacity = ordersCount * 4 * 3 * (totalX / totalRows); |
| | | LogHelper.Error($"成品小板区总货位是{totalX},总排数是{totalRows},当前执行工单数量是{ordersCount}", null); |
| | | var totalAvailableCapacity = list.Sum(a => a.AvailableCapacity); |
| | | var totalAssignableCapacity = totalAvailableCapacity - totalDesiredCapacity; |
| | | list.ForEach(a => { a.DesiredCapacity = totalDesiredCapacity; a.CRC = totalAvailableCapacity; a.RAC = totalAssignableCapacity; }); |
| | | if (info == null) |
| | | { |
| | | /*SQL Server 1000 条/批次 单个 INSERT 语句最多支持 1000 条 VALUES 子句,超出会报语法错误。*/ |
| | | db.Insertable(list).ExecuteCommand(); |
| | | } |
| | | else |
| | | { |
| | | //更新 |
| | | list.ForEach(a => |
| | | { |
| | | var old = db.Queryable<KuRong>().Where(b => b.StorageLoc == a.StorageLoc).First(); |
| | | if (old != null) |
| | | { |
| | | old.MaxCapacity = a.MaxCapacity; |
| | | old.AvailableCapacity = a.AvailableCapacity; |
| | | old.FilledCapacity = a.FilledCapacity; |
| | | old.ProductItem = a.ProductItem; |
| | | old.DesiredCapacity = a.DesiredCapacity; |
| | | old.CRC = a.CRC; |
| | | old.RAC = a.RAC; |
| | | old.T_MODIFY = DateTime.Now; |
| | | db.Updateable(old).UpdateColumns(it => new { it.MaxCapacity, it.FilledCapacity, it.AvailableCapacity, it.ProductItem, it.T_MODIFY }).ExecuteCommand(); |
| | | } |
| | | else |
| | | { |
| | | db.Insertable(a).ExecuteCommand(); |
| | | } |
| | | }); |
| | | } |
| | | } |
| | | } |
| | | catch (Exception ex) |
| | | { |
| | | Console.WriteLine(ex.Message); |
| | | LogHelper.Error(ex.Message, ex); |
| | | } |
| | | |
| | | //统计当前产线 |
| | | } |
| | | |
| | | internal static void Kuronggg() |
| | | { |
| | | int min = 60; |
| | | try |
| | | { |
| | | LogHelper.Info("Run>>CheckKurRong"); |
| | | //如果表内没有数据就初始化一下,有数据了modify时间和当前时间超过30min再统计一次 |
| | | var db = new SqlHelper<object>().GetInstance(); |
| | | var info = db.Queryable<KuRong>().First(); |
| | | if (info == null || DateTime.Now.Subtract(info.T_MODIFY).TotalMinutes > min) |
| | | { |
| | | //1、查到小板库区下面所有的库位 |
| | | var totalX = 0; |
| | | var totalRows = 0; |
| | | var list = new List<KuRong>(); |
| | | var areas = db.Queryable<Location>().Select(x => x.S_AREA_CODE).ToList().FindAll(x => x != null && System.Text.RegularExpressions.Regex.IsMatch(x, @"QX-\d+_[A-Z]+")).Select(x => x.Split('_')[0]).Distinct().ToList(); //GetAreas(); |
| | | LogHelper.Info($"KurRongCalc List {JsonConvert.SerializeObject(areas)}"); |
| | | areas.ForEach(area => |
| | | { |
| | | totalX += db.Queryable<Location>().Count(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_X")); |
| | | totalRows += db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_X")).Select(b => b.N_ROW).Distinct().Count(); |
| | | var aresList = db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(area) && b.S_AREA_CODE.Contains("_")).Select(b => b.S_AREA_CODE).Distinct().ToList(); |
| | | aresList = aresList.Select(a => a.Split('_')[0]).Distinct().ToList(); |
| | | LogHelper.Info($"KurRongCalc aresList {JsonConvert.SerializeObject(aresList)}"); |
| | | aresList.ForEach(a => |
| | | { |
| | | //b => b.S_AREA_CODE == a + "_X" |
| | | Console.WriteLine("area:" + a); |
| | | // && b.S_AREA_CODE.EndsWith("X") |
| | | var rows = db.Queryable<Location>().Where(b => b.S_AREA_CODE.StartsWith(a)).Select(b => b.N_ROW).Distinct().ToList(); |
| | | if (rows.Count > 0) |
| | | { |
| | | Console.WriteLine(string.Join(",", rows)); |
| | | rows.ForEach(b => |
| | | { |
| | | var kr = new KuRong() { StorageLoc = a.Split('-')[1] + "-" + b, T_MODIFY = DateTime.Now }; |
| | | Console.WriteLine(kr.StorageLoc); |
| | | //每一排要查一下有没有托盘 |
| | | var listLoc = db.Queryable<Location>().Where(c => c.S_AREA_CODE.Contains(a) && c.N_ROW == b).ToList();//&& c.N_CURRENT_NUM > 0 |
| | | |
| | | //如果有正常物料排,就用, 没有或者 空排 ,就用小板 |
| | | var norAreacode = ""; |
| | | foreach (var arl in listLoc.GroupBy(x => x.S_AREA_CODE)) |
| | | { |
| | | if (arl.ToList().Find(x => x.S_LOCK_STATE.Contains("空间锁")) == null) |
| | | { |
| | | if (arl.ToList().Find(x => x.N_CURRENT_NUM > 0) != null) |
| | | { |
| | | norAreacode = arl.Key; |
| | | break; |
| | | } |
| | | } |
| | | } |
| | | LogHelper.Info($"KurRongCalc [{a}]-{b} 统计是否正常{norAreacode}"); |
| | | if (string.IsNullOrEmpty(norAreacode)) |
| | | { |
| | | norAreacode = a + "_X"; |
| | | //没托盘,默认最大容量是小板的数量 |
| | | var count = db.Queryable<Location>().Count(c => c.S_AREA_CODE == norAreacode && c.N_ROW == b) * 2 * 2;//2层2托 |
| | | kr.MaxCapacity = count; |
| | | kr.AvailableCapacity = count; |
| | | kr.ProductItem = ""; |
| | | } |
| | | else |
| | | { |
| | | var locode = listLoc.Find(x => x.S_AREA_CODE == norAreacode && x.N_CURRENT_NUM > 0); |
| | | //有托盘 判断是叠2层还是3层,要获取物料信息 |
| | | var lcr = db.Queryable<LocCntrRel>().Includes(it => it.CntrItemRel).Where(l => l.S_LOC_CODE == locode.S_LOC_CODE).First(); |
| | | LogHelper.Info($"KurRongCalc {norAreacode}-{locode.S_LOC_CODE}-获取到托盘物料信息?:{lcr != null}"); |
| | | if (lcr != null && lcr.CntrItemRel != null) |
| | | { |
| | | var maxLayer = ContainerHelper.GetItem(ass => ass.S_ITEM_NAME == lcr.CntrItemRel.S_ITEM_NAME && ass.S_ITEM_CODE == lcr.CntrItemRel.S_ITEM_CODE)?.MaxLayer ?? 0; |
| | | |
| | | //LogHelper.Info($"KurRongCalc {norAreacode}-{lcr.CntrItemRel.S_ITEM_NAME}-{lcr.CntrItemRel.S_ITEM_CODE}-maxlayer:{maxLayer}-{locode.S_LOC_CODE}-{locode.N_CAPACITY}"); |
| | | if (maxLayer <= 0) |
| | | { |
| | | maxLayer = locode.N_CAPACITY; |
| | | } |
| | | var total = db.Queryable<Location>().Count(c => c.S_AREA_CODE == norAreacode && c.N_ROW == b) * maxLayer * 2; |
| | | //LogHelper.Info($"KurRongCalc {norAreacode}-{locode.S_LOC_CODE}"); |
| | | var count = db.Queryable<Location>().Where(c => c.S_AREA_CODE == norAreacode && c.N_ROW == b && c.N_CURRENT_NUM > 0).Sum(c => c.N_CURRENT_NUM) * 2; // 指定统计字段 |
| | | |
| | | kr.ProductItem = lcr.CntrItemRel.S_ITEM_CODE; |
| | | if(!string.IsNullOrEmpty(lcr.CntrItemRel.S_ITEM_NAME)) |
| | | kr.ProductItem = lcr.CntrItemRel.S_ITEM_NAME; |
| | | kr.MaxCapacity = total; |
| | | kr.FilledCapacity = count; |
| | | kr.AvailableCapacity = total - count; |
| | | } |
| | | } |
| | | list.Add(kr); |
| | | }); |
| | | } |
| | | else |
| | | { |
| | | Console.WriteLine("area:" + a + ":未获取到排"); |
| | | } |
| | | }); |
| | | |
| | | }); |
| | | |
| | | var ordersCount = 0;// db.Queryable<WorkOrder>().Where(a => a.SQL_State == "执行中").Count(); |
| | | foreach (var pline in Settings.GetDeviceInfoList().FindAll(x => x.deviceType == 9)) |
| | | { |
| | | ordersCount += db.Queryable<WorkOrder>().Where(a => a.SQL_PLineNo == pline.deviceName && a.SQL_State == "执行中").Count(); |
| | | } |
| | | var totalDesiredCapacity = ordersCount * 4 * 3 * (totalX / totalRows); |
| | | LogHelper.Error($"成品小板区总货位是{totalX},总排数是{totalRows},当前执行工单数量是{ordersCount}", new Exception("Kurong")); |
| | | var totalAvailableCapacity = list.Sum(a => a.AvailableCapacity); |
| | | var totalAssignableCapacity = totalAvailableCapacity - totalDesiredCapacity; |
| | | list.ForEach(a => { a.DesiredCapacity = totalDesiredCapacity; a.CRC = totalAvailableCapacity; a.RAC = totalAssignableCapacity; }); |
| | | if (info == null) |
| | | { |
| | | /*SQL Server 1000 条/批次 单个 INSERT 语句最多支持 1000 条 VALUES 子句,超出会报语法错误。*/ |
| | | for (int i = 0; i < list.Count; i += 1000) |
| | | { |
| | | db.Insertable(list.Skip(i).Take(1000).ToList()).ExecuteCommand(); |
| | | } |
| | | } |
| | | else |
| | | { |
| | | //更新 |
| | | list.ForEach(a => |
| | | { |
| | | var old = db.Queryable<KuRong>().Where(b => b.StorageLoc == a.StorageLoc).First(); |
| | | if (old != null) |
| | | { |
| | | old.MaxCapacity = a.MaxCapacity; |
| | | old.AvailableCapacity = a.AvailableCapacity; |
| | | old.FilledCapacity = a.FilledCapacity; |
| | | old.ProductItem = a.ProductItem; |
| | | old.DesiredCapacity = a.DesiredCapacity; |
| | | old.CRC = a.CRC; |
| | | old.RAC = a.RAC; |
| | | old.T_MODIFY = DateTime.Now; |
| | | db.Updateable(old).UpdateColumns(it => new { it.MaxCapacity, it.FilledCapacity, it.AvailableCapacity, it.ProductItem, it.T_MODIFY }).ExecuteCommand(); |
| | | } |
| | | else |
| | | { |
| | | db.Insertable(a).ExecuteCommand(); |
| | | } |
| | | }); |
| | | } |
| | | } |
| | | |
| | | LogHelper.Info("OVER<<CheckKurRong"); |
| | | } |
| | | catch (Exception ex) |
| | | { |
| | | Console.WriteLine(ex.Message); |
| | | LogHelper.Error(ex.Message, ex); |
| | | } |
| | | |
| | | } |
| | | } |
| | | } |