using ServiceCenter.Extensions; using ServiceCenter.SqlSugars; using System.ComponentModel; using WCS.Core; using WCS.Entity; using WCS.WorkEngineering.Extensions; using WCS.WorkEngineering.Protocol.Station; using WCS.WorkEngineering.Worlds; using DeviceFlags = WCS.WorkEngineering.Extensions.DeviceFlags; using TaskStatus = WCS.Entity.TaskStatus; namespace WCS.WorkEngineering.Systems { /// /// 分线计算09 /// [BelongTo(typeof(NoInteractionWorld))] [Description("分线计算09")] public class 分线计算09 : DeviceSystem> { protected override bool ParallelDo => true; protected override bool SaveLogsToFile => true; public override void Do(Device obj) { SqlSugarHelper.Do(_db => { //获取所有待执行的任务 var db = _db.Default; var taskList = db.Queryable().Where(x => x.Status == TaskStatus.WaitingToExecute && x.Type == TaskType.SetPlate && x.AddrFrom == "Robot").Take(1).ToList(); //开始对任务进行处理 foreach (var taskInfo in taskList) { //TODO:暂时不来考虑动态计算可前往的目标 var nextAdd = taskInfo.WarehouseCode switch { "1N" => "455", "1S" => "455", "2N" => "455", "2S" => "455", "3N" => "455", "3S" => "455", _ => "0" }; //获取这个地址的下一个地址集合 var cacheLineDevList = Device.All.First(x => x.Code == nextAdd).Targets.Where(x => x.HasFlag(DeviceFlags.桁架缓存放行点)); var cacheLineCodes = cacheLineDevList.Select(x => x.Code.ToShort()); var cacheLineList = db.Queryable().Includes(x => x.Locations).ToList(); #region 跟据缓存信息寻找可以到达的缓存点 //找到当前任务可用的缓存线信息 var cacheLine = cacheLineList.Where(x => x.Locations.Any(l => l is { InStock: false, IsEmpty: false })).FirstOrDefault(x => cacheLineCodes.Contains(x.LocationNo) && x.MatCodeList.Contains(taskInfo.MatCode) && !x.InStock); if (cacheLine != null)//这个任务可以直接去一条线体,不需要新建缓存信息 { //找到这条线体中序号最小的一条位信息 非空置且无货 var cacheLoc = cacheLine.Locations.Where(x => x is { InStock: false, IsEmpty: false }).MinBy(x => x.XYNo); if (cacheLoc != null) { cacheLoc = db.Queryable().Single(x => x.Id == cacheLoc.Id); cacheLoc.InStock = true; cacheLoc.TaskId = taskInfo.ID; cacheLoc.EditTime = DateTime.Now; db.Updateable(cacheLoc).ExecuteCommand(); //WCS任务相关信息 taskInfo.Status = TaskStatus.FinishOfShunt; taskInfo.AddrNext = cacheLine.LocationNo.ToString(); taskInfo.EditWho = "WCS"; taskInfo.EditTime = DateTime.Now; taskInfo.AddWCS_TASK_DTL(_db, obj.Entity.Code, taskInfo.AddrNext, $"完成分库计算,目标地址:{cacheLine.LocationNo}"); db.Updateable(taskInfo).ExecuteCommand(); taskInfo.UpdateRedisHash(); return; } } #endregion 跟据缓存信息寻找可以到达的缓存点 #region 初始化一个信息的缓存信息 //找到所有当前轮子可以去的垛形 var palletizingList = db.Queryable() .Includes(x => x.Layers, r => r.Rows, l => l.Locs) .Where(x => x.MatCodeList.Contains(taskInfo.MatCode) && !x.Finish).ToList().Where(x => x.Layers.SelectMany(x => x.Rows).Any(r => r.LineCode == null)).OrderBy(x => x.AddTime).ToList(); //如果没有对应的垛形信息就初始化一个垛形信息 if (palletizingList.Count <= 0) { taskInfo.InitStackStructure(); return; } foreach (var palletizing in palletizingList) { //TODO:当前不考虑不按顺序码垛的情况 //按码垛顺序找到一个最近的没有码垛成功的行, //先找层:未空置、未结束 TODO:暂不考虑所有层全部结束,但是垛形信息未清除的情况 var palletizingLayer = palletizing.Layers .Where(x => !x.IsEmpty) .Where(x => !x.Finish) .Where(x => x.MatCodeList.Contains(taskInfo.MatCode)) .Where(x => x.Rows.Any(r => r.CacheLineId == 0)) .MinBy(x => x.LayerNo); //如果没有哪一层需要这个物料号,就初始化一个新的垛形信息 if (palletizingLayer == null) { taskInfo.InitStackStructure(); return; } //再找行:未空置、未结束 var palletizingRow = palletizingLayer.Rows.Where(x => x is { IsEmpty: false, Finish: false } && x.MatCodeList.Contains(taskInfo.MatCode) && x.CacheLineId == 0) .MinBy(x => x.RowNo); //如果没有哪一行需要这个物料号,就初始化一个新的垛形信息 if (palletizingRow == null) { taskInfo.InitStackStructure(); return; } //走到这一步就表示没有哪一段线体缓存了当前物料,需要选一段新的线体进行缓存 //TODO:暂时不处理就近分线的逻辑 //获取一个当前可以使用的分配锁 var devCode = cacheLineDevList.Select(x => x.Code.ToShort()).FirstOrDefault(x => !cacheLineList.Select(s => s.LocationNo).Contains(x)); if (devCode == 0) { World.Log($"无可用线体:{taskInfo.ID}"); return; } //开始初始化缓存位信息 cacheLine = new WCS_CacheLine() { LocationNo = devCode, AddTime = DateTime.Now, PalletizingRowId = palletizingRow.Id, InStock = false, Put = false, MatCodeList = palletizingRow.MatCodeList }; var res = db.Insertable(cacheLine).ExecuteReturnEntity(); palletizingRow = db.Queryable().Includes(x => x.Locs).Single(x => x.Id == palletizingRow.Id); palletizingRow.Locs = palletizingRow.Locs.OrderBy(x => x.XYNo).ToList(); palletizingRow.CacheLineId = res.Id; palletizingRow.EditTime = DateTime.Now; db.Updateable(palletizingRow).ExecuteCommand(); for (var i = 0; i < palletizingRow.Locs.Count; i++) { var loc = new WCS_CacheLineLoc() { XYNo = palletizingRow.Locs[i].XYNo, InStock = i == 0, IsEmpty = palletizingRow.Locs[i].IsEmpty, MatCode = palletizingRow.Locs[i].MatCode, TaskId = i == 0 ? taskInfo.ID : 0, CacheLineId = res.Id }; db.Insertable(loc).ExecuteCommand(); } taskInfo.Status = TaskStatus.FinishOfShunt; taskInfo.AddrNext = devCode.ToString(); taskInfo.EditWho = "WCS"; taskInfo.EditTime = DateTime.Now; taskInfo.AddWCS_TASK_DTL(_db, obj.Entity.Code, taskInfo.AddrNext, $"完成分库计算,目标地址:{cacheLine.LocationNo}"); db.Updateable(taskInfo).ExecuteCommand(); taskInfo.UpdateRedisHash(); return; } #endregion 初始化一个信息的缓存信息 } }); } public override bool Select(Device dev) { return dev.Code == nameof(分线计算09); } } public class LineCache { /// /// 位置 /// public string Location { get; set; } /// /// 是否放行 /// public bool IsPut { get; set; } /// /// 具体缓存信息 /// public List LineInfos { get; set; } } /// /// 线体缓存组员 /// public class LineCacheInfo { /// /// 序号 /// public string Index { get; set; } /// /// 物料号 /// public string MatCode { get; set; } /// /// 任务号 /// public int TaskNumber { get; set; } /// /// 是否有货 /// public bool InStock { get; set; } /// /// 是否空置 /// public bool IsEmpty { get; set; } } }