一楼入库.cs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. using DBHelper;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol;
  8. using WCS.Entity.Protocol.SRM;
  9. using WCS.Service.Extensions;
  10. using WCS.Service.Helpers;
  11. namespace WCS.Service.Works.Station
  12. {
  13. [WorkTitle(typeof(ProductHandler), "一楼入库扫码")]
  14. internal class 一楼入库 : DeviceWork<StationDevice>
  15. {
  16. private readonly string Conv_1028 = "1028";
  17. private readonly string Conv_1029 = "1029";
  18. protected override bool SelectDevice(WCS_DEVICE dev)
  19. {
  20. return dev.CODE == Conv_1028 || dev.CODE == Conv_1029;
  21. }
  22. protected override void Do(StationDevice dev)
  23. {
  24. var code = dev.Entity.CODE;
  25. if (dev.WhetherToExecute(IstationRequest.扫码入库)) return;
  26. var bcr = dev.Entity.BCR();
  27. var barcode = bcr.Content.Trim('\r');
  28. if (barcode == "") throw new Exception(LogHelper.SpliceLogMessage("扫码失败", dev.Entity.CODE, WCS_EXCEPTIONTYPE.逻辑异常, GetType()));
  29. var info = WMS.I_WCS_GetInTask(barcode, dev.Entity.CODE);
  30. var next = dev.Entity.CODE == Conv_1028 ? "1030" : "1031";
  31. var station = dev.Entity.CODE;
  32. DB.Do(db =>
  33. {
  34. var task = new WCS_TASK();
  35. task.BARCODE = info.ContainerCode;
  36. task.TYPE = TaskType.入库;
  37. task.STATUS = TaskStatus.执行中;
  38. task.ADDRFROM = dev.Entity.CODE;
  39. task.ADDRTO = info.EndPostion;
  40. task.STARTTIME = DateTime.Now;
  41. task.UPDATEUSER = "WCS";
  42. task.UPDATETIME = DateTime.Now;
  43. task.WMSTASK = int.Parse(info.WMSTaskNum);
  44. task.TaskGroupKey = info.TaskGroupKey;
  45. task.ADDRNEXT = next;
  46. db.Default.Set<WCS_TASK>().Add(task);
  47. db.Default.SaveChanges();
  48. dev.Data.Tasknum = task.ID;
  49. dev.Data.Goodsstart = task.ADDRFROM.ToShort();
  50. dev.Data.Goodsend = task.ADDRNEXT.ToShort();
  51. dev.Data.CmdType = IstationCmdType.扫码入库;
  52. dev.Data.VoucherNo++;
  53. var msg = $"下达从{dev.Entity.CODE}移动至{next}的PLC指令。";
  54. msg += $"[{dev.Data.Tasknum}][{dev.Data.Goodsstart}][{dev.Data.Goodsend}][{dev.Data.VoucherNo}[{dev.Data2.VoucherNo}]";
  55. task.CreateStatusLog(db, msg, this.GetType());
  56. });
  57. }
  58. }
  59. [WorkTitle(typeof(ProductHandler), "一楼分配巷道")]
  60. internal class 巷道分配 : Work<StationDeviceGroup>
  61. {
  62. private readonly string ConvGroup_1030 = "G1030";
  63. private readonly string Conv_1030 = "1030";
  64. private readonly string RGV8 = "RGV8";
  65. protected override bool SelectDevice(WCS_DEVICE dev)
  66. {
  67. return dev.CODE == ConvGroup_1030;
  68. }
  69. protected override void Do(StationDeviceGroup obj)
  70. {
  71. var code = obj.Entity.CODE;
  72. //当前组有一个运行的设备就停止执行
  73. if (obj.WhetherToExecute()) return;
  74. //获取需要进行巷道分配的设备
  75. var devs = obj.TaskedDeviceGetNextAddress() ?? throw new Exception(LogHelper.SpliceLogMessage("无可用任务", obj.Entity.CODE, WCS_EXCEPTIONTYPE.逻辑异常, GetType()));
  76. DB.Do(db =>
  77. {
  78. var taskIds = devs.Select(p => p.Data2.Tasknum);
  79. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskIds.Any(v => v == p.ID)).ToList();
  80. var res = WMS.GetTunnelList(tasks.Select(v => v.WMSTASK.ToString()).ToList(), code);
  81. if (string.IsNullOrEmpty(res.TunnelNum)) throw new Exception(LogHelper.SpliceLogMessage($"WMS未返回巷道", obj.Entity.CODE, WCS_EXCEPTIONTYPE.逻辑异常, GetType()));
  82. var tunnelNo = res.TunnelNum.Split(',').Select(v => "TY" + v).ToList();
  83. var tunnels = Device.Where(v => tunnelNo.Contains(v.CODE)).ToList();
  84. List<TunnelInfo> tunnelInfos = new List<TunnelInfo>();
  85. foreach (var item in tunnels)
  86. {
  87. //找到空闲的取货点 先找到下一个路径点是当前巷道的设备
  88. var q = Device.Where(p => p.PATHPOINTS.Any(d => d.NEXT == item))
  89. .SelectMany(p => p.PATHPOINTS) //将所有取货点的路线信息提取到一个集合中
  90. .Select(p => p.PREV.Create<StationDevice>())//取出所有取货点的前一个路径点
  91. .Where(p => !p.Data3.Status.HasFlag(StationStatus.运行状态位) && !p.Data2.Status.HasFlag(IstationStatus.光电状态) && p.Data2.Tasknum < 10000)//筛选出空闲的路径点,此处因输送机都是一个动力,因此可以先找路径点再找设备组
  92. .Where(p => Device.Where(d => d.CODE.StartsWith("G")).Any(d => d.DEVICEGROUP.Any(c => c.MEMBER == p.Entity)))//找到有效路径点对应的设备组
  93. .Distinct().FirstOrDefault();//去一次重
  94. if (q == null) continue;
  95. var dev = Device.Where(p => p.CODE.StartsWith("G") && p.DEVICEGROUP.Any(d => d.MEMBER == q.Entity)).OrderBy(p => p.CODE).FirstOrDefault();
  96. tunnelInfos.Add(new TunnelInfo
  97. {
  98. Tunnel = item,
  99. taskIN = dev, //找到放货点设备所在组
  100. SRM = Device.Where(p => p.IsSC()).FirstOrDefault(p => item.ROUTES.Any(d => d.NEXT.CODE == p.CODE)).Create<SRMDevice>()
  101. });
  102. }
  103. //筛选出优先级最高的可用巷道
  104. var tunnelInfo = tunnelInfos.Where(v => { try { return v.SRM.Data3.SCAlarm == 0 && v.SRM.Data2.SRMMode == SCMode.远程; } catch { return false; } })
  105. .OrderBy(v => tunnelNo.IndexOf(v.Tunnel.CODE)).FirstOrDefault();
  106. if (tunnelInfo == null) throw new Exception(LogHelper.SpliceLogMessage("无可用巷道", obj.Entity.CODE, WCS_EXCEPTIONTYPE.逻辑异常, GetType()));
  107. //开始向设备中写入任务信息
  108. foreach (var dev in devs)
  109. {
  110. var task = tasks.FirstOrDefault(p => p.ID == dev.Data2.Tasknum);
  111. if ((task.TaskGroupKey.Contains($"{task.WMSTASK}_") || task.TaskGroupKey.Contains($"_{task.WMSTASK}"))) continue;
  112. if (dev.Data2.Goodsend != task.ADDRNEXT.ToShort()) continue;
  113. dev.Data.Tasknum = task.ID;
  114. dev.Data.Goodsstart = Conv_1030.ToShort();
  115. dev.Data.Goodsend = tunnelInfo.taskIN.CODE.Replace("G", "").ToShort();
  116. dev.Data.CmdType = IstationCmdType.分配目标地址;
  117. dev.Data.VoucherNo++;
  118. task.DEVICE = tunnelInfo.SRM.Entity.CODE;
  119. task.TUNNEL = tunnelInfo.Tunnel.CODE;
  120. task.ADDRNEXT = dev.Data.Goodsend.ToString();
  121. task.TaskGroupKey = res.WMSTaskGroupKey;
  122. task.ADDRTO = task.DEVICE;
  123. db.Default.SaveChanges();
  124. var msg = $"下达从{Conv_1030}移动至{dev.Data.Goodsend}的PLC指令。同时将任务分配至[{ task.TUNNEL }]-[{task.DEVICE}]";
  125. msg += $"[{dev.Data.Tasknum}][{Conv_1030}][{dev.Data.Goodsend}][{tunnelInfo.SRM.Entity.CODE}][{dev.Data.VoucherNo}[{dev.Data2.VoucherNo}]";
  126. task.CreateStatusLog(db, msg, this.GetType());
  127. }
  128. });
  129. }
  130. }
  131. [WorkTitle(typeof(ProductHandler), "一楼RGV放货结束分配目标地址")]
  132. internal class 一楼RGV放货结束分配目标地址 : DeviceWork<StationDevice>
  133. {
  134. protected override void Do(StationDevice obj)
  135. {
  136. var code = obj.Entity.CODE;
  137. if (obj.WhetherToExecute(IstationRequest.请求分配目标地址)) return;
  138. DB.Do(db =>
  139. {
  140. var task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == obj.Data2.Tasknum);
  141. switch (task.ADDRFROM)
  142. {
  143. case "1028":
  144. break;
  145. case "1029":
  146. break;
  147. default:
  148. break;
  149. }
  150. var next = obj.Entity.GetPath(task.ADDRTO);
  151. obj.Data.Tasknum = task.ID;
  152. obj.Data.Goodsstart = obj.Entity.CODE.ToShort();
  153. obj.Data.Goodsend = next.FirstOrDefault().CODE.ToShort();
  154. obj.Data.CmdType = IstationCmdType.分配目标地址;
  155. obj.Data.VoucherNo++;
  156. task.ADDRNEXT = next.FirstOrDefault().CODE;
  157. db.Default.SaveChanges();
  158. var msg = $"下达从{obj.Data.Goodsstart}移动至{obj.Data.Goodsend}的PLC指令";
  159. msg += $"[{obj.Data.Tasknum}][{obj.Data.Goodsstart}][{obj.Data.Goodsend}][{obj.Data.VoucherNo}[{obj.Data2.VoucherNo}]";
  160. task.CreateStatusLog(db, msg, this.GetType());
  161. });
  162. }
  163. protected override bool SelectDevice(WCS_DEVICE dev)
  164. {
  165. return devCodes.Contains(dev.CODE);
  166. }
  167. private List<string> devCodes = new List<string>() {
  168. "1035",
  169. "1036",
  170. "1044",
  171. "1045",
  172. "1053",
  173. "1054",
  174. "1062",
  175. "1063",
  176. };
  177. }
  178. }