RGVWork.cs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. using DBHelper;
  2. using System;
  3. using System.Linq;
  4. using WCS.Core;
  5. using WCS.Entity;
  6. using WCS.Entity.Protocol;
  7. namespace WCS.Service.Works
  8. {
  9. [WorkTitle(typeof(ProductHandler), "穿梭车", true)]
  10. public class RGVWork : DeviceWork<Device<IRGV521, IRGV520, IRGV523>>
  11. {
  12. protected override void Do(Device<IRGV521, IRGV520, IRGV523> dev)
  13. {
  14. if (dev.Data3.Fault != 0)
  15. {
  16. if (dev.Entity.WakeupOn(5000))
  17. {
  18. WMS.DevInfo(dev.Entity.CODE, dev.Data3.Fault.ToString());
  19. }
  20. }
  21. if (Ltc.Do(dev, v => v.Data2.Trigger != 0))
  22. return;
  23. if (Ltc.Do(dev, v => v.Data.WorkMode != RGVMode.自动))
  24. return;
  25. if (Ltc.Do(dev, v => v.Data.SystemStatus != RGVRunStatus.空闲))
  26. return;
  27. var startStations = Device.Where(v => v.ROUTES.Any(d => d.NEXT.CODE == dev.Entity.CODE))//.Where(v => v.CODE != "2323")
  28. .Select(v => v.Device<IStation521, IStation520>()).ToArray();
  29. startStations = Ltc.Do(startStations, (arr) => arr.Where(v => v.Data.GOODSEND > 0 && v.Data.PH_STATUS).ToArray());
  30. if (dev.Entity.CODE == "RGV1")
  31. startStations = Ltc.Do(startStations, (arr) => arr.Where(v => v.Data.GOODSEND > 0 && v.Data.PH_STATUS && !v.Data.REQUEST).ToArray());
  32. if (startStations.Length == 0)
  33. return;
  34. //startStations = Ltc.Do(startStations, dev, (arr, dev) => arr.Where(v => dev.Entity.ROUTES.Any(d => d.NEXT.CODE == v.Data.GOODSEND.ToString())).ToArray());
  35. //if (startStations.Length == 0)
  36. // return;
  37. startStations = Ltc.Do(startStations, arr => arr.Where(v => v.Data.GOODSEND.ToString() != v.Entity.CODE).ToArray());
  38. if (startStations.Length == 0)
  39. return;
  40. var endEmptys = startStations.Where(v =>
  41. {
  42. var end = Device.Find(v.Data.GOODSEND.ToString());
  43. var next = dev.Entity.GetNext(end.CODE).Device<IStation521>();
  44. var res = next.Data.TASKNUM == 0 && next.Data.PH_STATUS == false;
  45. if (res && end.CODE == "2324")
  46. {
  47. res = Device.Find("2071", "2072", "2073", "2074", "2075", "2076", "2077", "2078", "2079", "2017", "2019", "2022", "2024", "2029").Select(v => v.Device<IStation521>())
  48. .Count(v => v.Data.PH_STATUS == false && v.Data.TASKNUM == 0) > 3;
  49. if (!res)
  50. {
  51. Configs.UploadException(end.CODE, "库前拥堵,RGV暂停入库");
  52. }
  53. }
  54. return res;
  55. }).ToArray();
  56. short taskType = 1;
  57. var st = Ltc.Do(endEmptys, arr => arr.OrderBy(v => v.Entity.CODE == "2182" || v.Entity.CODE == "2181" || v.Entity.CODE == "2081" || v.Entity.CODE == "2325" ? 1 : 2)
  58. .ThenBy(v => v.Data.GetUpdateTime())
  59. .FirstOrDefault());
  60. if (st == null)
  61. {
  62. st = startStations.OrderBy(v => v.Entity.CODE == "2182" || v.Entity.CODE == "2181" || v.Entity.CODE == "2081" || v.Entity.CODE == "2325" ? 1 : 2)
  63. .ThenBy(v => v.Data.GetUpdateTime())
  64. .FirstOrDefault();
  65. taskType = 3;
  66. }
  67. if (taskType == 3)
  68. {
  69. var target = st.Entity.Code();
  70. if (dev.Data.DestPosition != target)
  71. {
  72. dev.Data2.TaskType = taskType;
  73. dev.Data2.TaskID = 1;
  74. dev.Data2.StartPosition = 0;
  75. dev.Data2.DestPosition = short.Parse(st.Entity.CODE);
  76. dev.Data2.Trigger = 1;
  77. }
  78. }
  79. else
  80. {
  81. var next = dev.Entity.GetNext(st.Data.GOODSEND.ToString());
  82. if (next == null)
  83. return;
  84. DB.Do(db =>
  85. {
  86. var task = db.Default.Set<WCS_TASK>().Find(st.Data.TASKNUM);
  87. if (task != null)
  88. {
  89. if (next.CODE == task.ADDRTO)
  90. {
  91. //task.ADDRCURRENT = dev.Entity.CODE;
  92. task.UPDATETIME = DateTime.Now;
  93. db.Default.SaveChanges();
  94. }
  95. }
  96. dev.Data2.TaskType = taskType;
  97. dev.Data2.TaskID = st.Data.TASKNUM;
  98. dev.Data2.StartPosition = short.Parse(st.Entity.CODE);
  99. dev.Data2.DestPosition = next.Code();
  100. dev.Data2.Trigger = 1;
  101. });
  102. }
  103. }
  104. protected override bool SelectDevice(WCS_DEVICE dev)
  105. {
  106. return dev.CODE is "RGV1" or "RGV2" or "RGV3" or "RGV5";
  107. }
  108. }
  109. [WorkTitle(typeof(ProductHandler), "双工位RGV", true)]
  110. public class 双工位RGV : DeviceWork<Device<IDRGV521, IDRGV520, IDRGV523>>
  111. {
  112. protected override void Do(Device<IDRGV521, IDRGV520, IDRGV523> dev)
  113. {
  114. if (dev.Data3.Fault != 0)
  115. {
  116. if (dev.Entity.WakeupOn(5000))
  117. {
  118. WMS.DevInfo(dev.Entity.CODE, dev.Data3.Fault.ToString());
  119. }
  120. }
  121. if (Ltc.Do(dev, v => v.Data2.Trigger_1 != 0 && v.Data2.Trigger_2 != 0)) return;
  122. if (Ltc.Do(dev, v => v.Data.WorkMode != RGVMode.自动)) return;
  123. if (Ltc.Do(dev, v => v.Data.SystemStatus != RGVRunStatus.空闲)) return;
  124. #region 放货任务
  125. //判断小车是否有工位有光电,小车空闲自动且有光电代表需要执行放货任务
  126. if (dev.Data.PH_Status_1 || dev.Data.PH_Status_2)
  127. {
  128. if (dev.Data.PH_Status_1) //工位一需要放货
  129. {
  130. if (dev.Data.TaskID_1 < 10000) throw new Exception("有光电无任务号");
  131. string next = "";
  132. //获取任务
  133. DB.Do(db =>
  134. {
  135. var task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == dev.Data.TaskID_1) ?? throw new Exception($"WCS不存在任务{dev.Data.TaskID_1}");
  136. var next = dev.Entity.GetNext(task.ADDRTO).CODE;
  137. });
  138. //TODO:放货站点有货检查
  139. dev.Data2.TaskID_1 = dev.Data.TaskID_1;
  140. dev.Data2.TaskType_1 = Convert.ToInt16(DRGVTaskType.放货);
  141. dev.Data2.StartPosition_1 = Convert.ToInt16(next);
  142. dev.Data.RES = dev.Data2.StartPosition_1;
  143. dev.Data2.Trigger_1 = 1;
  144. }
  145. else if (dev.Data.PH_Status_2)//工位二需要放货
  146. {
  147. if (dev.Data.TaskID_2 < 10000) throw new Exception("有光电无任务号");
  148. string next = "";
  149. //获取任务
  150. DB.Do(db =>
  151. {
  152. var task = db.Default.Set<WCS_TASK>().FirstOrDefault(v => v.ID == dev.Data.TaskID_2) ?? throw new Exception($"WCS不存在任务{dev.Data.TaskID_2}");
  153. var next = dev.Entity.GetNext(task.ADDRTO).CODE;
  154. });
  155. //TODO:放货站点有货检查
  156. dev.Data2.TaskID_2 = dev.Data.TaskID_2;
  157. dev.Data2.TaskType_2 = Convert.ToInt16(DRGVTaskType.放货);
  158. dev.Data2.StartPosition_2 = Convert.ToInt16(next);
  159. dev.Data.RES = dev.Data2.StartPosition_2;
  160. dev.Data2.Trigger_2 = 1;
  161. }
  162. }
  163. #endregion 放货任务
  164. #region 取货任务
  165. //获取所有需要取货的站台
  166. //获取下一个地址为当前RGV的站台,此处不考虑一台车跑全程
  167. var validPickupPoints = Device.Where(v => v.ROUTES.Any(p => p.NEXT.CODE == dev.Entity.CODE)).ToList()
  168. .Select(v => v.Device<IStation521, IStation520, IStation523>())
  169. .Where(v => v.Data.PH_STATUS && v.Data.TASKNUM > 10000 && !v.Data2.CONFIRM && v.Data.GOODSEND != Convert.ToInt16(v.Entity.CODE))//需要取货的站点
  170. .Where(v => //是否可以进行取货,主要用于判断放货位是否有货
  171. {
  172. var end = Device.Find(v.Data.GOODSEND.ToString()).Device<IStation521>();
  173. var res = end.Data.TASKNUM == 0 && end.Data.PH_STATUS == false;
  174. return res;
  175. });
  176. if (!validPickupPoints.Any()) return;//不需要取货
  177. Device<IStation521, IStation520, IStation523> validPickupPoint = null;
  178. //是否包含小车当前所在站点,写死,如有变动必须改代码
  179. //这一块的作用主要是用于解决小车就近取货问题
  180. if (dev.Data.RES is 3035 or 3036) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3035" or "3036");
  181. else if (dev.Data.RES is 3041 or 3040) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3041" or "3040");
  182. else if (dev.Data.RES is 3047 or 3048) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3047" or "3048");
  183. else if (dev.Data.RES is 3052 or 2108) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3052" or "2108");
  184. //当前站点没有需要取货的
  185. if (validPickupPoint == null) validPickupPoint = validPickupPoints.OrderBy(v => v.UpdateTime).FirstOrDefault() ?? throw new Exception("没有站台需要取货");
  186. if (validPickupPoint.Entity.CODE is "" or "" or "") //一工位取货点
  187. {
  188. dev.Data2.TaskID_1 = dev.Data.TaskID_1;
  189. dev.Data2.TaskType_1 = Convert.ToInt16(DRGVTaskType.取货);
  190. dev.Data2.StartPosition_1 = Convert.ToInt16(validPickupPoint.Data.GOODSEND);
  191. dev.Data2.Trigger_1 = 1;
  192. }
  193. else if (validPickupPoint.Entity.CODE is "" or "" or "") //二工位取货点
  194. {
  195. dev.Data2.TaskID_2 = dev.Data.TaskID_2;
  196. dev.Data2.TaskType_2 = Convert.ToInt16(DRGVTaskType.取货);
  197. dev.Data2.StartPosition_2 = Convert.ToInt16(validPickupPoint.Data.GOODSEND);
  198. dev.Data2.Trigger_2 = 1;
  199. }
  200. #endregion 取货任务
  201. }
  202. protected override bool SelectDevice(WCS_DEVICE dev)
  203. {
  204. return dev.CODE is "RGV4" or "RGV6";
  205. }
  206. }
  207. [WorkTitle(typeof(ProductHandler), "连廊单工位RGV", true)]
  208. public class 连廊RGV : DeviceWork<Device<IRGV521, IRGV520, IRGV523>>
  209. {
  210. protected override void Do(Device<IRGV521, IRGV520, IRGV523> dev)
  211. {
  212. if (dev.Data3.Fault != 0)
  213. {
  214. if (dev.Entity.WakeupOn(5000))
  215. {
  216. WMS.DevInfo(dev.Entity.CODE, dev.Data3.Fault.ToString());
  217. }
  218. }
  219. if (Ltc.Do(dev, v => v.Data2.Trigger != 0)) return;
  220. if (Ltc.Do(dev, v => v.Data.WorkMode != RGVMode.自动)) return;
  221. if (Ltc.Do(dev, v => v.Data.SystemStatus != RGVRunStatus.空闲)) return;
  222. //获取所有需要取货的站台
  223. //获取下一个地址为当前RGV的站台,此处不考虑一台车跑全程
  224. var validPickupPoints = Device.Where(v => v.ROUTES.Any(p => p.NEXT.CODE == dev.Entity.CODE)).ToList()
  225. .Select(v => v.Device<IStation521, IStation520, IStation523>())
  226. .Where(v => v.Data.PH_STATUS && v.Data.TASKNUM > 10000 && !v.Data2.CONFIRM && v.Data.GOODSEND != Convert.ToInt16(v.Entity.CODE))//需要取货的站点
  227. .Where(v => //是否可以进行取货,主要用于判断放货位是否有货
  228. {
  229. var end = Device.Find(v.Data.GOODSEND.ToString()).Device<IStation521>();
  230. var res = end.Data.TASKNUM == 0 && end.Data.PH_STATUS == false;
  231. return res;
  232. });
  233. if (!validPickupPoints.Any()) return;//不需要取货
  234. Device<IStation521, IStation520, IStation523> validPickupPoint = null;
  235. //是否包含小车当前所在站点,写死,如有变动必须改代码
  236. //这一块的作用主要是用于解决小车就近取货问题
  237. if (dev.Data.RES is 3025 or 3026) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3025" or "3026");
  238. else if (dev.Data.RES is 3021 or 3022) validPickupPoint = validPickupPoints.FirstOrDefault(v => v.Entity.CODE is "3021" or "3022");
  239. //当前站点没有需要取货的
  240. if (validPickupPoint == null) validPickupPoint = validPickupPoints.OrderBy(v => v.UpdateTime).FirstOrDefault() ?? throw new Exception("没有站台需要取货");
  241. dev.Data2.TaskID = validPickupPoint.Data.TASKNUM;
  242. dev.Data2.StartPosition = Convert.ToInt16(validPickupPoint.Entity.CODE);
  243. dev.Data2.DestPosition = validPickupPoint.Data.GOODSEND;
  244. dev.Data2.Trigger = 1;
  245. }
  246. protected override bool SelectDevice(WCS_DEVICE dev)
  247. {
  248. return dev.CODE is "RGV7" or "RGV8";
  249. }
  250. }
  251. }