SRMWork.cs 48 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758
  1. using DBHelper;
  2. using Microsoft.EntityFrameworkCore;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Linq;
  6. using WCS.Core;
  7. using WCS.Entity;
  8. using WCS.Entity.Protocol;
  9. using WCS.Entity.Protocol.SRM;
  10. using WCS.Service.Entity;
  11. using WCS.Service.Extensions;
  12. using WCS.Service.Handlers;
  13. using WCS.Service.Helpers;
  14. using WCS.Service.Log;
  15. namespace WCS.Service.Works.SRM
  16. {
  17. [WorkTitle(typeof(SRMHandler), "堆垛机")]
  18. internal class SrmWork : DeviceWork<SRMDevice>
  19. {
  20. protected override void Do(SRMDevice obj)
  21. {
  22. obj.EX(srmDevice =>
  23. {
  24. //状态上抛
  25. DeciceStateHelper.StatusCheck(obj.Entity.CODE, obj.Data2.SRMMode.ToString());
  26. var deviceCode = obj.Entity.CODE;
  27. //先检查堆垛机是否报警
  28. if (srmDevice.Data3.SCAlarm != 0)
  29. {
  30. if (srmDevice.Entity.WakeupOn(5000)) WMS.DevInfo(deviceCode, srmDevice.Data3.SCAlarm.ToString());
  31. InfoLog.INFO_SRMALARM($"{deviceCode}-{srmDevice.Data3.SCAlarm}");
  32. return;
  33. }
  34. if (srmDevice.Data.FinishedACK_1 == 1 || srmDevice.Data.FinishedACK_2 == 1) throw new WarnException($"WCS反馈任务完成信号PLC未清除");
  35. if (srmDevice.Data2.VoucherNo_1 != srmDevice.Data.VoucherNo_1 || srmDevice.Data2.VoucherNo_2 != srmDevice.Data.VoucherNo_2)
  36. {
  37. DB.Do(db =>
  38. {
  39. WCS_TASK task;
  40. if (srmDevice.Data2.VoucherNo_1 != srmDevice.Data.VoucherNo_1)
  41. {
  42. task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == srmDevice.Data.TaskID_1) ?? throw new WarnException($"未找到对应的WCS任务--{srmDevice.Data.TaskID_1}");
  43. if (task.STATUS == TaskStatus.新建)
  44. {
  45. srmDevice.Data.VoucherNo_1 = srmDevice.Data2.VoucherNo_1;
  46. }
  47. }
  48. if (srmDevice.Data2.VoucherNo_2 != srmDevice.Data.VoucherNo_2)
  49. {
  50. task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == srmDevice.Data.TaskID_2) ?? throw new WarnException($"未找到对应的WCS任务--{srmDevice.Data.TaskID_2}");
  51. if (task.STATUS == TaskStatus.新建)
  52. {
  53. srmDevice.Data.VoucherNo_2 = srmDevice.Data2.VoucherNo_2;
  54. }
  55. }
  56. });
  57. throw new WarnException($"等待执行{srmDevice.Data.TaskID_1}-{srmDevice.Data.TaskID_2}");
  58. }
  59. //处理堆垛机已完成的任务
  60. if (srmDevice.Data2.FinishedTask_1 != 0 || srmDevice.Data2.FinishedTask_2 != 0)
  61. {
  62. InfoLog.INFO_SRMINFO($"开始完成任务:[{deviceCode}]-{srmDevice.Data2.FinishedTask_1}-{srmDevice.Data2.FinishedTask_2}");
  63. srmDevice.FinishedTaskHandle();
  64. InfoLog.INFO_SRMINFO($"完成任务处理结束:[{deviceCode}]-{srmDevice.Data2.FinishedTask_1}-{srmDevice.Data2.FinishedTask_2}");
  65. }
  66. if (srmDevice.Data2.SRMMode != SCMode.远程) return;
  67. if (srmDevice.Data2.SRMStatus != SCRunStatus.空闲) return;
  68. bool isTransfer = false; //默认没有移库任务
  69. WCS_TASK enterPriority = null, outPriority = null; //出入库优先级任务
  70. var dumpLibrary = new List<WCS_TASK>();
  71. //再检查是否有等待执行的货物
  72. DB.Do(db =>
  73. {
  74. //获取当前堆垛机的所有未完成任务
  75. var tasks = db.Default.Set<WCS_TASK>().AsNoTracking().Where(v => v.STATUS < TaskStatus.已完成 && (v.DEVICE == deviceCode || v.DEVICEDL == deviceCode)).ToList();
  76. //获取倒库任务
  77. dumpLibrary = tasks.Where(v => v.TYPE == TaskType.倒库).ToList();
  78. #region 没有倒库任务
  79. if (!dumpLibrary.Any())
  80. {
  81. if (tasks.Any(v => v.STATUS == TaskStatus.堆垛机执行)) throw new WarnException($"[{deviceCode}]有正在执行的任务");
  82. //属于当前堆垛机未执行的移库任务
  83. isTransfer = tasks.Any(v => v.TYPE == TaskType.移库 && v.STATUS < TaskStatus.堆垛机执行);
  84. //判断是否存在调整优先级任务,存在初始化isTransfer值 让本次执行优先任务
  85. if (tasks.Any(v => v.TYPE != TaskType.移库 && v.STATUS < TaskStatus.堆垛机执行 && v.Priority > 0)) isTransfer = false;
  86. enterPriority = tasks.Where(v => v.TYPE == TaskType.入库 && v.STATUS < TaskStatus.堆垛机执行 && v.Priority > 0).OrderByDescending(v => v.Priority).FirstOrDefault();
  87. outPriority = tasks.Where(v => v.TYPE == TaskType.出库 && v.STATUS < TaskStatus.堆垛机执行 && v.STATUS!= TaskStatus.执行中 && v.Priority > 0).OrderByDescending(v => v.Priority).FirstOrDefault();
  88. }
  89. #endregion 没有倒库任务
  90. });
  91. //最后一个是否是出库任务
  92. var lastIsOut = srmDevice.Entity.Get<bool>("LastIsOut");
  93. srmDevice.Entity.Set("LastIsOut", !lastIsOut);
  94. #region 倒库任务
  95. if (dumpLibrary.Any())
  96. {
  97. //检查当前堆垛机是起始点还是目标点
  98. var lastOut = srmDevice.Entity.Get<bool>("LastOut");
  99. srmDevice.Entity.Set("LastOut", !lastIsOut);
  100. if (lastOut) //入库
  101. {
  102. var arrIn = srmDevice.GetPickPoint()
  103. .Where(v => Device.Where(d => d.IsConv()).Where(v => v.Is(DF.SRM涂布取货)).Select(d => d.Device<IStation521>()).Any(d => d.Data.Goodsend == v.Entity.Code())) //有正在前往取货点的任务
  104. .ToList();
  105. if (!arrIn.Any()) return; //当前堆垛机无入库任务
  106. //入库口设备信息 找一个有任务有光电且不在运行状态位的取货点 如果找不到代表任务还在输送途中
  107. var station = arrIn.OrderBy(v => v.Data2.Tasknum > 0 && v.Data2.Status.HasFlag(IstationStatus.光电状态) && !v.Data3.Status.HasFlag(StationStatus.运行状态位) ? 0 : 1)
  108. .ThenBy(v => v.UpdateTime)
  109. .FirstOrDefault() ?? throw new WarnException($"[{deviceCode}]等待入库任务输送到位");
  110. //根据上述筛选条件筛选出来的入库任务 找到对应的设备组
  111. var item = Device.Where(v => v.DEVICEGROUP.Any(p => p.MEMBER.CODE == station.Entity.CODE)).Single();
  112. //对数据进行排序,根据CAD图纸规划,取货点两个设备中设备号较大的一个一定对应到堆垛机的一工位货叉
  113. //因此按照降序进行排序,可以保证数组中第一个一定是放到堆垛机一工位的数据
  114. var devise = item.DEVICEGROUP.Select(v => v.MEMBER)
  115. .Select(v => v.Create<StationDevice>())
  116. .OrderByDescending(v => v.Entity.CODE)
  117. .ToArray();
  118. var finishTaskList = new List<FinishTaskList<int>>(); //成功分配货位的任务
  119. //检测有效任务数与实际任务是是否相等
  120. var validDev = devise.Where(v => v.Data2.Tasknum > 10000 && v.Data2.Status.HasFlag(IstationStatus.光电状态) && !v.Data3.Status.HasFlag(StationStatus.运行状态位));
  121. var stationDevices = validDev as StationDevice[] ?? validDev.ToArray();
  122. if (!stationDevices.Any()) throw new DoException("无有效入库任务");
  123. DB.Do(db =>
  124. {
  125. var tasking = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == stationDevices.First().Data2.Tasknum);
  126. var taskList = db.Default.Set<WCS_TASK>().Count(v => v.TaskGroupKey == tasking.TaskGroupKey && v.TYPE == TaskType.倒库);
  127. if (stationDevices.Count() != taskList) throw new WarnException($"任务数量不匹配,设备-{stationDevices.Count()},WCS-{taskList}");
  128. foreach (var dev in stationDevices)
  129. {
  130. var task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.STATUS < TaskStatus.堆垛机执行 && v.ID == dev.Data2.Tasknum) ?? throw new WarnException($"设备有光电有任务且不在运行状态,但WCS找不到任务{dev.Data2.Tasknum}");
  131. var oldTask = task.STATUS;
  132. var tunnel = dev.Entity.ROUTES.First().NEXT.CODE;
  133. I_WCS_GetWareCellResponse loc;
  134. //判定当前设备对应的堆垛机工位
  135. if (dev.Entity.CODE == devise[0].Entity.CODE) loc = WMS.GetLocalIn(task.WMSTASK, tunnel, dev.Entity.CODE, Entity.WareCellForkNum.货叉1); //一工位
  136. else if (dev.Entity.CODE == devise[1].Entity.CODE) loc = WMS.GetLocalIn(task.WMSTASK, tunnel, dev.Entity.CODE, Entity.WareCellForkNum.货叉2); //2工位
  137. else throw new WarnException($"设备{dev.Entity.CODE}无法对应至堆垛机的任一一个工位");
  138. task.UPDATETIME = DateTime.Now;
  139. task.STATUS = TaskStatus.堆垛机执行;
  140. task.ADDRTO = $"{loc.Row}-{loc.Colomn}-{loc.Layer}";
  141. task.DEVICE = deviceCode;
  142. task.TUNNEL = tunnel;
  143. task.CreateStatusLog(db, $"堆垛机开始执行入库", this.GetType());
  144. finishTaskList.Add(new FinishTaskList<int>(task.ID, dev));
  145. }
  146. db.Default.SaveChanges();
  147. });
  148. DB.Do(db =>
  149. {
  150. foreach (var finish in finishTaskList)
  151. {
  152. var task = db.Default.Set<WCS_TASK>().Find(finish.FinishCode);
  153. var addTo = task!.ADDRTO.Split("-");
  154. if (finish.Station.Entity.CODE == devise[0].Entity.CODE) //一工位
  155. {
  156. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]1工位-开始:[{srmDevice.Data.TaskID_1}][{srmDevice.Data.SLine_1}][{srmDevice.Data.ELine_1}][{srmDevice.Data.ECol_1}][{srmDevice.Data.ELayer_1}][{srmDevice.Data.VoucherNo_1}]--[{finishTaskList.Count.ToShort()}]");
  157. srmDevice.Data.TaskID_1 = task.ID;
  158. srmDevice.Data.SLine_1 = finish.Station.Entity.CODE.ToShort();
  159. srmDevice.Data.SCol_1 = 0;
  160. srmDevice.Data.SLayer_1 = 0;
  161. srmDevice.Data.ELine_1 = addTo[0].ToShort();
  162. srmDevice.Data.ECol_1 = addTo[1].ToShort();
  163. srmDevice.Data.ELayer_1 = addTo[2].ToShort();
  164. srmDevice.Data.RES1_1 = finishTaskList.Count.ToShort();
  165. srmDevice.Data.VoucherNo_1++;
  166. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]1工位-结束:[{srmDevice.Data.TaskID_1}][{srmDevice.Data.SLine_1}][{srmDevice.Data.ELine_1}][{srmDevice.Data.ECol_1}][{srmDevice.Data.ELayer_1}][{srmDevice.Data.VoucherNo_1}]--[{finishTaskList.Count.ToShort()}]");
  167. }
  168. else if (finish.Station.Entity.CODE == devise[1].Entity.CODE)
  169. {
  170. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]2工位-开始:[{srmDevice.Data.TaskID_2}][{srmDevice.Data.SLine_2}][{srmDevice.Data.ELine_2}][{srmDevice.Data.ECol_2}][{srmDevice.Data.ELayer_2}][{srmDevice.Data.VoucherNo_2}]--[{finishTaskList.Count.ToShort()}]");
  171. srmDevice.Data.TaskID_2 = task.ID;
  172. srmDevice.Data.SLine_2 = finish.Station.Entity.CODE.ToShort();
  173. srmDevice.Data.SCol_2 = 0;
  174. srmDevice.Data.SLayer_2 = 0;
  175. srmDevice.Data.ELine_2 = addTo[0].ToShort();
  176. srmDevice.Data.ECol_2 = addTo[1].ToShort();
  177. srmDevice.Data.ELayer_2 = addTo[2].ToShort();
  178. srmDevice.Data.RES1_2 = finishTaskList.Count.ToShort();
  179. srmDevice.Data.VoucherNo_2++;
  180. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]2工位-结束:[{srmDevice.Data.TaskID_2}][{srmDevice.Data.SLine_2}][{srmDevice.Data.ELine_2}][{srmDevice.Data.ECol_2}][{srmDevice.Data.ELayer_2}][{srmDevice.Data.VoucherNo_2}]--[{finishTaskList.Count.ToShort()}]");
  181. }
  182. }
  183. });
  184. }
  185. else //出库
  186. {
  187. //获取当前堆垛机所有的放货点
  188. var list = srmDevice.GetDeliveryPoint().Where(v =>
  189. {
  190. //true:满足条件 false:不满足条件
  191. //返回结果为无货的设备 默认无货
  192. var res = true;
  193. //放货点是否有货
  194. if (v.Data.VoucherNo != v.Data2.VoucherNo) res = false;
  195. else if (v.Data3.Status.HasFlag(StationStatus.运行状态位)) res = false;
  196. else if (v.Data2.Status.HasFlag(IstationStatus.光电状态)) res = false;
  197. else if (v.Data2.Request == IstationRequest.堆垛机放货完成请求目标地址) res = false;
  198. else if (v.Data2.Tasknum > 10000) res = false;
  199. if (!v.Entity.Is(DF.SRM涂布放货)) return res;
  200. var devise = new List<StationDevice>();
  201. switch (v.Entity.CODE)
  202. {
  203. case "1283" or "1284":
  204. devise = Device.Where(b => b.CODE is "1281" or "1282").Select(b => b.Create<StationDevice>()).ToList();
  205. break;
  206. case "1290" or "1291" or "1292" or "1293":
  207. devise = Device.Where(b => b.CODE is "1288" or "1289").Select(b => b.Create<StationDevice>()).ToList();
  208. break;
  209. case "1299" or "1300" or "1301" or "1302":
  210. devise = Device.Where(b => b.CODE is "1297" or "1298").Select(b => b.Create<StationDevice>()).ToList();
  211. break;
  212. case "1308" or "1309" or "1310" or "1311":
  213. devise = Device.Where(b => b.CODE is "1306" or "1307").Select(b => b.Create<StationDevice>()).ToList();
  214. break;
  215. }
  216. if (!devise.Any()) return res;
  217. foreach (var stationDevice in devise)
  218. {
  219. //放货点是否有货
  220. if (stationDevice.Data3.Status.HasFlag(StationStatus.运行状态位)) res = false;
  221. else if (stationDevice.Data2.Status.HasFlag(IstationStatus.光电状态)) res = false;
  222. else if (stationDevice.Data2.Request == IstationRequest.堆垛机放货完成请求目标地址) res = false;
  223. else if (stationDevice.Data2.Tasknum > 10000) res = false;
  224. }
  225. return res;
  226. }).Select(v => v.Entity.CODE).ToList();
  227. //没有可用货位
  228. if (!list.Any()) return;
  229. //找到对应的设备组编号
  230. var groupList = Device.Where(v => v.IsDevGroup()).Where(v => v.DEVICEGROUP.Any(b => list.Contains(b.MEMBER.CODE))).Select(v => v.CODE).ToList();
  231. //堆垛机设备
  232. var srm = deviceCode;
  233. var finishTaskList = new List<FinishTaskList<SrmFork, Task>>();
  234. DB.Do(db =>
  235. {
  236. //堆垛机当前是否有正在执行的任务
  237. if (db.Default.Set<WCS_TASK>().Any(d => d.DEVICE == srm && d.STATUS == TaskStatus.堆垛机执行)) throw new WarnException($"[{deviceCode}]有正在执行的出库任务");
  238. //找出等待执行的出库任务
  239. var waitTask = db.Default.Set<WCS_TASK>().Where(v => v.STATUS == TaskStatus.新建)
  240. .Where(v => v.DEVICE == srm)
  241. .Where(v => v.TYPE == TaskType.倒库)
  242. .Where(v => !db.Default.Set<WCS_TASK>().Any(d => d.DEVICE == srm && d.STATUS == TaskStatus.堆垛机执行))
  243. .Where(v => groupList.Contains(v.SRMSTATION)) //站台必须可用
  244. .ToList();
  245. //同时对结果进行排序,分组
  246. var outDepotGrouping = waitTask.OrderByDescending(v => v.Priority).ThenBy(v => v.CREATETIME)
  247. .GroupBy(v => v.TaskGroupKey)
  248. .FirstOrDefault().Select(v => v).ToList();
  249. if (outDepotGrouping == null) return; //用于解决Linq Value cannot be null. (Parameter 'source')
  250. //获取一组任务
  251. var tasks = outDepotGrouping.Select(v => v.Create<Task>()).OrderBy(v => v.Col).ToArray();
  252. for (var i = 0; i < tasks.Count(); i++)
  253. {
  254. var item = tasks[i];
  255. var task = db.Default.Set<WCS_TASK>().Find(item.ID);
  256. var oldTaskStatus = task!.STATUS;
  257. task.STARTTIME = DateTime.Now;
  258. task.UPDATETIME = DateTime.Now;
  259. task.STATUS = TaskStatus.堆垛机执行;
  260. task.DEVICE = deviceCode;
  261. var fork = srmDevice.GetFork(item, i);
  262. //获取站台及下一个地址
  263. task.GetSrmStationAndaddNext(fork);
  264. task.CreateStatusLog(db, "堆垛机开始执行出库", this.GetType());
  265. item.SRMSTATION = task.SRMSTATION;
  266. finishTaskList.Add(new FinishTaskList<SrmFork, Task>(fork, item));
  267. }
  268. db.Default.SaveChanges();
  269. //此处只做标记,表示当前事务已经提交
  270. foreach (var finish in finishTaskList)
  271. {
  272. switch (finish.FinishCode)
  273. {
  274. // 列数较小的放一工位
  275. case SrmFork.货叉1:
  276. obj.Data.TaskID_1 = finish.Station.ID;
  277. break;
  278. //列数较大的放二工位
  279. case SrmFork.货叉2:
  280. obj.Data.TaskID_2 = finish.Station.ID;
  281. break;
  282. default:
  283. throw new ArgumentOutOfRangeException();
  284. }
  285. }
  286. });
  287. ////检查标记好的出库任务,并将出库任务下达至堆垛机
  288. DB.Do(db =>
  289. {
  290. //此处只做标记,表示当前事务已经提交
  291. foreach (var finish in finishTaskList)
  292. {
  293. var task = db.Default.Set<WCS_TASK>().Find(finish.Station.ID).Create<Task>();
  294. switch (finish.FinishCode)
  295. {
  296. // 列数较小的放一工位
  297. case SrmFork.货叉1:
  298. obj.WriteTask1(task, (short)finishTaskList.Count);
  299. break;
  300. //列数较大的放二工位
  301. case SrmFork.货叉2:
  302. obj.WriteTask2(task, (short)finishTaskList.Count);
  303. break;
  304. default:
  305. throw new ArgumentOutOfRangeException();
  306. }
  307. }
  308. });
  309. }
  310. }
  311. #endregion 倒库任务
  312. #region 移库
  313. if (isTransfer) //防止因为无当前堆垛机移库任务导致无法执行其他类型任务
  314. {
  315. var finishTaskList = new List<FinishTaskList<SrmFork, Task>>();
  316. DB.Do(db =>
  317. {
  318. //获取当前堆垛机未执行的任务
  319. var taskList = db.Default.Set<WCS_TASK>().Where(v => v.DEVICE == srmDevice.Entity.CODE && v.TYPE == TaskType.移库 && v.STATUS < TaskStatus.堆垛机执行).OrderBy(p => p.CREATETIME);
  320. var tasks = taskList.ToList().GetOutTask();
  321. if (tasks.Any(v => v.Layer <= 6))
  322. {
  323. if (tasks.Count() >= 2)
  324. {
  325. tasks = tasks.Take(1).ToArray();
  326. }
  327. }
  328. for (var i = 0; i < tasks.Length; i++)
  329. {
  330. var item = tasks[i];
  331. var task = db.Default.Set<WCS_TASK>().Find(item.ID);
  332. var oldTaskStatus = task!.STATUS;
  333. task.STARTTIME = DateTime.Now;
  334. task.UPDATETIME = DateTime.Now;
  335. task.STATUS = WCS.Entity.TaskStatus.堆垛机执行;
  336. task.DEVICE = srmDevice.Entity.CODE;
  337. task.TaskGroupKey = tasks.Length switch
  338. {
  339. 1 => $"{tasks[0].ID}_0",
  340. 2 => $"{tasks[0].ID}_{tasks[1].ID}",
  341. _ => throw new WarnException($"可用任务数异常{tasks.Length}"),
  342. };
  343. var fork = srmDevice.GetFork(item, i);
  344. var msg = "";
  345. msg = fork == SrmFork.货叉1 ? $"状态由[{oldTaskStatus}]变更为[{task.STATUS}]" : $"状态由[{oldTaskStatus}]变更为[{task.STATUS}]";
  346. task.CreateStatusLog(db, $"任务下发堆垛机执行,起始地址[{task.ADDRFROM}],目标地址[{task.ADDRTO}]", this.GetType());
  347. finishTaskList.Add(new FinishTaskList<SrmFork, Task>(fork, item));
  348. }
  349. db.Default.SaveChanges();
  350. //此处只做标记,表示当前事务已经提交
  351. foreach (var finish in finishTaskList)
  352. {
  353. switch (finish.FinishCode)
  354. {
  355. // 列数较小的放一工位
  356. case SrmFork.货叉1:
  357. obj.Data.TaskID_1 = finish.Station.ID;
  358. break;
  359. //列数较大的放二工位
  360. case SrmFork.货叉2:
  361. obj.Data.TaskID_2 = finish.Station.ID;
  362. break;
  363. default:
  364. throw new ArgumentOutOfRangeException();
  365. }
  366. }
  367. });
  368. //检查标记好的出库任务,并将出库任务下达至堆垛机
  369. DB.Do(db =>
  370. {
  371. //此处只做标记,表示当前事务已经提交
  372. foreach (var finish in finishTaskList)
  373. {
  374. var task = db.Default.Set<WCS_TASK>().Find(finish.Station.ID).Create<Task>();
  375. switch (finish.FinishCode)
  376. {
  377. // 列数较小的放一工位
  378. case SrmFork.货叉1:
  379. obj.WriteTask1(task, (short)finishTaskList.Count);
  380. break;
  381. //列数较大的放二工位
  382. case SrmFork.货叉2:
  383. obj.WriteTask2(task, (short)finishTaskList.Count);
  384. break;
  385. default:
  386. throw new ArgumentOutOfRangeException();
  387. }
  388. }
  389. });
  390. }
  391. #endregion 移库
  392. else if (lastIsOut)
  393. {
  394. #region 入库
  395. var floor = srmDevice.Entity.Get<int>("LastInFloor");
  396. floor = floor % 2 + 1;
  397. srmDevice.Entity.Set("LastInFloor", floor);
  398. if (enterPriority != null && outPriority != null && outPriority.Priority > enterPriority.Priority) return;
  399. //获取当前堆垛机的取货点
  400. var arrIn = srmDevice.GetPickPoint();
  401. if (!arrIn.Any()) return; //当前堆垛机无取货点
  402. var lastIn = srmDevice.Entity.Get<int>("LastIn");
  403. StationDevice station = null;
  404. #region 找到一个可用的取货点
  405. if (lastIn is 0 or 1)
  406. {
  407. // 涂布优先为2
  408. srmDevice.Entity.Set("LastIn", 2);
  409. station = arrIn.OrderBy(v => v.Data2.Tasknum > 0 && v.Data2.Status.HasFlag(IstationStatus.光电状态) && !v.Data3.Status.HasFlag(StationStatus.运行状态位) ? 0 : 1)
  410. .ThenBy(v => v.Entity.Is(DF.SRM涂布取货) ? 0 : 1)
  411. .ThenBy(v => v.Entity.Is(DF.SRMBOPP取货) ? 0 : 1)
  412. .FirstOrDefault() ?? throw new WarnException($"[{deviceCode}]等待入库任务输送到位");
  413. }
  414. else if (lastIn == 2)
  415. {
  416. // BOPP优先为1
  417. srmDevice.Entity.Set("LastIn", 1);
  418. station = arrIn.OrderBy(v => v.Data2.Tasknum > 0 && v.Data2.Status.HasFlag(IstationStatus.光电状态) && !v.Data3.Status.HasFlag(StationStatus.运行状态位) ? 0 : 1)
  419. .ThenBy(v => v.Entity.Is(DF.SRMBOPP取货) ? 0 : 1)
  420. .ThenBy(v => v.Entity.Is(DF.SRM涂布取货) ? 0 : 1)
  421. .FirstOrDefault() ?? throw new WarnException($"[{deviceCode}]等待入库任务输送到位");
  422. }
  423. #endregion 找到一个可用的取货点
  424. //根据上述筛选条件筛选出来的入库任务 找到对应的设备组
  425. var item = Device.Where(v => v.DEVICEGROUP.Any(p => p.MEMBER.CODE == station.Entity.CODE)).Single();
  426. //对数据进行排序,根据CAD图纸规划,取货点两个设备中设备号较大的一个一定对应到堆垛机的一工位货叉
  427. //因此按照降序进行排序,可以保证数组中第一个一定是放到堆垛机一工位的数据
  428. var devise = item.DEVICEGROUP.Select(v => v.MEMBER)
  429. .Select(v => v.Create<StationDevice>())
  430. .OrderByDescending(v => v.Entity.CODE)
  431. .ToArray();
  432. var finishTaskList = new List<FinishTaskList<int>>(); //成功分配货位的任务
  433. //检测有效任务数与实际任务是是否相等
  434. var validDev = devise.Where(v => v.Data2.Tasknum > 10000 && v.Data2.Status.HasFlag(IstationStatus.光电状态) && !v.Data3.Status.HasFlag(StationStatus.运行状态位));
  435. var stationDevices = validDev as StationDevice[] ?? validDev.ToArray();
  436. if (!stationDevices.Any()) throw new DoException("无有效入库任务");
  437. DB.Do(db =>
  438. {
  439. var tasks = db.Default.Set<WCS_TASK>().AsNoTracking().Where(v => v.STATUS < TaskStatus.已完成 && v.DEVICE == deviceCode && v.TYPE == TaskType.入库).ToList();
  440. var tasking = tasks.FirstOrDefault(v => v.ID == stationDevices.First().Data2.Tasknum);
  441. var taskList = tasks.Count(v => v.TaskGroupKey == tasking.TaskGroupKey);
  442. if (stationDevices.Count() != taskList) throw new WarnException($"任务数量不匹配,设备-{stationDevices.Count()},WCS-{taskList}");
  443. foreach (var dev in stationDevices)
  444. {
  445. var task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.STATUS < TaskStatus.堆垛机执行 && v.ID == dev.Data2.Tasknum) ?? throw new WarnException($"设备有光电有任务且不在运行状态,但WCS找不到任务{dev.Data2.Tasknum}");
  446. var oldTask = task.STATUS;
  447. var tunnel = dev.Entity.ROUTES.First().NEXT.CODE;
  448. I_WCS_GetWareCellResponse loc;
  449. //判定当前设备对应的堆垛机工位
  450. if (dev.Entity.CODE == devise[0].Entity.CODE) loc = WMS.GetLocalIn(task.WMSTASK, tunnel, dev.Entity.CODE, Entity.WareCellForkNum.货叉1); //一工位
  451. else if (dev.Entity.CODE == devise[1].Entity.CODE) loc = WMS.GetLocalIn(task.WMSTASK, tunnel, dev.Entity.CODE, Entity.WareCellForkNum.货叉2); //2工位
  452. else throw new WarnException($"设备{dev.Entity.CODE}无法对应至堆垛机的任一一个工位");
  453. task.UPDATETIME = DateTime.Now;
  454. task.STATUS = TaskStatus.堆垛机执行;
  455. task.ADDRTO = $"{loc.Row}-{loc.Colomn}-{loc.Layer}";
  456. task.DEVICE = deviceCode;
  457. task.TUNNEL = tunnel;
  458. task.CreateStatusLog(db, $"任务下发堆垛机执行,起始地址[{dev.Entity.CODE}],目标地址[{task.ADDRTO}]", this.GetType());
  459. finishTaskList.Add(new FinishTaskList<int>(task.ID, dev));
  460. }
  461. db.Default.SaveChanges();
  462. });
  463. DB.Do(db =>
  464. {
  465. foreach (var finish in finishTaskList)
  466. {
  467. var task = db.Default.Set<WCS_TASK>().Find(finish.FinishCode);
  468. var addTo = task!.ADDRTO.Split("-");
  469. if (finish.Station.Entity.CODE == devise[0].Entity.CODE) //一工位
  470. {
  471. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]1工位-开始:[{srmDevice.Data.TaskID_1}][{srmDevice.Data.SLine_1}][{srmDevice.Data.ELine_1}][{srmDevice.Data.ECol_1}][{srmDevice.Data.ELayer_1}][{srmDevice.Data.VoucherNo_1}]--[{finishTaskList.Count.ToShort()}]");
  472. srmDevice.Data.TaskID_1 = task.ID;
  473. srmDevice.Data.SLine_1 = finish.Station.Entity.CODE.ToShort();
  474. srmDevice.Data.SCol_1 = 0;
  475. srmDevice.Data.SLayer_1 = 0;
  476. srmDevice.Data.ELine_1 = addTo[0].ToShort();
  477. srmDevice.Data.ECol_1 = addTo[1].ToShort();
  478. srmDevice.Data.ELayer_1 = addTo[2].ToShort();
  479. srmDevice.Data.RES1_1 = finishTaskList.Count.ToShort();
  480. srmDevice.Data.VoucherNo_1++;
  481. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]1工位-结束:[{srmDevice.Data.TaskID_1}][{srmDevice.Data.SLine_1}][{srmDevice.Data.ELine_1}][{srmDevice.Data.ECol_1}][{srmDevice.Data.ELayer_1}][{srmDevice.Data.VoucherNo_1}]--[{finishTaskList.Count.ToShort()}]");
  482. }
  483. else if (finish.Station.Entity.CODE == devise[1].Entity.CODE)
  484. {
  485. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]2工位-开始:[{srmDevice.Data.TaskID_2}][{srmDevice.Data.SLine_2}][{srmDevice.Data.ELine_2}][{srmDevice.Data.ECol_2}][{srmDevice.Data.ELayer_2}][{srmDevice.Data.VoucherNo_2}]--[{finishTaskList.Count.ToShort()}]");
  486. srmDevice.Data.TaskID_2 = task.ID;
  487. srmDevice.Data.SLine_2 = finish.Station.Entity.CODE.ToShort();
  488. srmDevice.Data.SCol_2 = 0;
  489. srmDevice.Data.SLayer_2 = 0;
  490. srmDevice.Data.ELine_2 = addTo[0].ToShort();
  491. srmDevice.Data.ECol_2 = addTo[1].ToShort();
  492. srmDevice.Data.ELayer_2 = addTo[2].ToShort();
  493. srmDevice.Data.RES1_2 = finishTaskList.Count.ToShort();
  494. srmDevice.Data.VoucherNo_2++;
  495. InfoLog.INFO_SRMINFO($"入库--写入堆垛机[{deviceCode}]2工位-结束:[{srmDevice.Data.TaskID_2}][{srmDevice.Data.SLine_2}][{srmDevice.Data.ELine_2}][{srmDevice.Data.ECol_2}][{srmDevice.Data.ELayer_2}][{srmDevice.Data.VoucherNo_2}]--[{finishTaskList.Count.ToShort()}]");
  496. }
  497. }
  498. });
  499. #endregion 入库
  500. }
  501. else
  502. {
  503. #region 出库
  504. var floor = srmDevice.Entity.Get<int>("LastOutFloor");
  505. floor = floor % 2 + 1;
  506. srmDevice.Entity.Set("LastOutFloor", floor);
  507. if (enterPriority != null && outPriority != null && enterPriority.Priority > outPriority.Priority) return;
  508. srmDevice.CheckOutTask();
  509. //获取当前堆垛机所有的放货点
  510. var list = srmDevice.GetDeliveryPoint().Where(v =>
  511. {
  512. //true:满足条件 false:不满足条件
  513. //返回结果为无货的设备 默认无货
  514. var res = true;
  515. if (!v.Entity.Is(DF.SRM月台放货))
  516. {
  517. //放货点是否有货
  518. if (v.Data.VoucherNo != v.Data2.VoucherNo) res = false;
  519. else if (v.Data3.Status.HasFlag(StationStatus.运行状态位)) res = false;
  520. else if (v.Data2.Status.HasFlag(IstationStatus.光电状态)) res = false;
  521. else if (v.Data2.Request == IstationRequest.堆垛机放货完成请求目标地址) res = false;
  522. else if (v.Data2.Tasknum > 10000) res = false;
  523. if (!v.Entity.Is(DF.SRM涂布放货)) return res;
  524. var devise = new List<StationDevice>();
  525. switch (v.Entity.CODE)
  526. {
  527. case "1283" or "1284":
  528. devise = Device.Where(b => b.CODE is "1281" or "1282").Select(b => b.Create<StationDevice>()).ToList();
  529. break;
  530. case "1290" or "1291" or "1292" or "1293":
  531. devise = Device.Where(b => b.CODE is "1288" or "1289").Select(b => b.Create<StationDevice>()).ToList();
  532. break;
  533. case "1299" or "1300" or "1301" or "1302":
  534. devise = Device.Where(b => b.CODE is "1297" or "1298").Select(b => b.Create<StationDevice>()).ToList();
  535. break;
  536. case "1308" or "1309" or "1310" or "1311":
  537. devise = Device.Where(b => b.CODE is "1306" or "1307").Select(b => b.Create<StationDevice>()).ToList();
  538. break;
  539. }
  540. if (!devise.Any()) return res;
  541. foreach (var stationDevice in devise)
  542. {
  543. //放货点是否有货
  544. if (stationDevice.Data3.Status.HasFlag(StationStatus.运行状态位)) res = false;
  545. else if (stationDevice.Data2.Status.HasFlag(IstationStatus.光电状态)) res = false;
  546. else if (stationDevice.Data2.Request == IstationRequest.堆垛机放货完成请求目标地址) res = false;
  547. else if (stationDevice.Data2.Tasknum > 10000) res = false;
  548. }
  549. }
  550. return res;
  551. }).Select(v => v.Entity.CODE).ToList();
  552. //没有可用货位
  553. if (!list.Any()) return;
  554. //找到对应的设备组编号
  555. var groupList = Device.Where(v => v.IsDevGroup()).Where(v => v.DEVICEGROUP.Any(b => list.Contains(b.MEMBER.CODE))).Select(v => v.CODE).ToList();
  556. //堆垛机设备
  557. var srm = deviceCode;
  558. var finishTaskList = new List<FinishTaskList<SrmFork, Task>>();
  559. DB.Do(db =>
  560. {
  561. //堆垛机当前是否有正在执行的任务
  562. if (db.Default.Set<WCS_TASK>().Any(d => d.DEVICE == srm && d.STATUS == TaskStatus.堆垛机执行)) throw new WarnException($"[{deviceCode}]有正在执行的出库任务");
  563. //找出等待执行的出库任务
  564. var waitTask = db.Default.Set<WCS_TASK>().Where(v => v.STATUS == TaskStatus.新建 && v.Priority >= 0)
  565. .Where(v => v.DEVICE == srm)
  566. .Where(v => v.TYPE == TaskType.出库)
  567. .Where(v => !db.Default.Set<WCS_TASK>().Any(d => d.DEVICE == srm && d.STATUS == TaskStatus.堆垛机执行))
  568. .Where(v => groupList.Contains(v.SRMSTATION)) //站台必须可用
  569. .ToList();
  570. //同时对结果进行排序,分组
  571. var maximum = ProtocolProxy.YGWMS150Redis.Get("SaleTaskGroupCount").ToInt();
  572. var cTaskList = db.Default.Set<WCS_TASK>().AsNoTracking()
  573. .Where(d => d.TYPE == TaskType.出库)
  574. .Where(d => d.STATUS > TaskStatus.新建)
  575. .Where(d => d.STATUS < TaskStatus.已完成).ToList();
  576. var outDepotGrouping = waitTask.Where(v => cTaskList.Where(d => d.ADDRTO == v.ADDRTO && d.FLOOR == v.FLOOR).GroupBy(d => d.TaskGroupKey).Count() < maximum || v.ADDRTO == "G1340")
  577. .OrderByDescending(v => v.Priority)
  578. .ThenBy(v => v.ADDRTO == "G1340" ? 0 : 1)
  579. .ThenBy(v => v.FLOOR == floor ? 0 : 1)
  580. .ThenBy(v => cTaskList.Where(d => d.ADDRTO == v.ADDRTO && d.FLOOR == v.FLOOR).GroupBy(d => d.TaskGroupKey).Count())
  581. .ThenBy(v => v.CREATETIME)
  582. .GroupBy(v => v.ADDRTO)
  583. .FirstOrDefault();
  584. if (outDepotGrouping == null) return; //用于解决Linq Value cannot be null. (Parameter 'source')
  585. var outDepotList = outDepotGrouping.Select(v => v).ToList();
  586. //获取两个个可执行任务,此时这两个任务的目标地址是一致的
  587. var tasks = outDepotList.GetOutTask();
  588. #region 校验两个产品是否为同规格
  589. if (tasks.Length == 2)
  590. {
  591. var length = tasks.OrderByDescending(v => v.Length).ToArray();
  592. //较大的长度减去较小的长度,差大于两百表示为不同规格产品
  593. if (length[0].Length - length[1].Length > 200)
  594. {
  595. tasks = tasks.Take(1).ToArray();
  596. }
  597. }
  598. #endregion 校验两个产品是否为同规格
  599. for (var i = 0; i < tasks.Length; i++)
  600. {
  601. var item = tasks[i];
  602. var task = db.Default.Set<WCS_TASK>().Find(item.ID);
  603. var oldTaskStatus = task!.STATUS;
  604. task.STARTTIME = DateTime.Now;
  605. task.UPDATETIME = DateTime.Now;
  606. task.STATUS = WCS.Entity.TaskStatus.堆垛机执行;
  607. task.DEVICE = deviceCode;
  608. task.TaskGroupKey = tasks.Length switch
  609. {
  610. 1 => $"{tasks[0].ID}_0",
  611. 2 => $"{tasks[0].ID}_{tasks[1].ID}",
  612. _ => throw new WarnException($"可用任务数异常{tasks.Length}"),
  613. };
  614. var fork = srmDevice.GetFork(item, i);
  615. //获取站台及下一个地址
  616. task.GetSrmStationAndaddNext(fork);
  617. task.CreateStatusLog(db, $"任务下发堆垛机执行,起点地址[{task.ADDRFROM}],目标地址[{task.SRMSTATION}]", this.GetType());
  618. item.SRMSTATION = task.SRMSTATION;
  619. finishTaskList.Add(new FinishTaskList<SrmFork, Task>(fork, item));
  620. }
  621. db.Default.SaveChanges();
  622. //此处只做标记,表示当前事务已经提交
  623. foreach (var finish in finishTaskList)
  624. {
  625. switch (finish.FinishCode)
  626. {
  627. // 列数较小的放一工位
  628. case SrmFork.货叉1:
  629. obj.Data.TaskID_1 = finish.Station.ID;
  630. break;
  631. //列数较大的放二工位
  632. case SrmFork.货叉2:
  633. obj.Data.TaskID_2 = finish.Station.ID;
  634. break;
  635. default:
  636. throw new ArgumentOutOfRangeException();
  637. }
  638. }
  639. });
  640. ////检查标记好的出库任务,并将出库任务下达至堆垛机
  641. DB.Do(db =>
  642. {
  643. //此处只做标记,表示当前事务已经提交
  644. foreach (var finish in finishTaskList)
  645. {
  646. var task = db.Default.Set<WCS_TASK>().Find(finish.Station.ID).Create<Task>();
  647. switch (finish.FinishCode)
  648. {
  649. // 列数较小的放一工位
  650. case SrmFork.货叉1:
  651. obj.WriteTask1(task, (short)finishTaskList.Count);
  652. break;
  653. //列数较大的放二工位
  654. case SrmFork.货叉2:
  655. obj.WriteTask2(task, (short)finishTaskList.Count);
  656. break;
  657. default:
  658. throw new ArgumentOutOfRangeException();
  659. }
  660. }
  661. });
  662. #endregion 出库
  663. }
  664. });
  665. }
  666. protected override bool SelectDevice(WCS_DEVICE dev)
  667. {
  668. return dev.Is(DF.SRM);
  669. }
  670. }
  671. }