RGVWorks.cs 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413
  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.RGV;
  8. using WCS.Service.Extensions;
  9. using WCS.Service.Handlers;
  10. using WCS.Service.Log;
  11. namespace WCS.Service.Works.RGV
  12. {
  13. [WorkTitle(typeof(RGVHandler), "直穿RGV")]
  14. public class 直穿RGV : DeviceWork<RGVDevice>
  15. {
  16. private readonly string ConvGroup_1030 = "G1030";
  17. private readonly string RGV8 = "RGV8";
  18. protected override void Do(RGVDevice rgv)
  19. {
  20. rgv.EX(rgv =>
  21. {
  22. //RGV是自动且空闲的
  23. if (rgv.Data2.WorkMode != RGVMode.自动) throw new DoException($"{rgv.Data2.WorkMode}");
  24. if (rgv.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException($"{rgv.Data2.SystemStatus}");
  25. if (rgv.Data.Trigger_1 != rgv.Data2.Trigger_1) throw new WarnException("凭证好不一致");
  26. if (rgv.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new DoException("RGV有光电,无法执行任务");
  27. //RGV状态为 RGV到站&&任务完成 时程序认为RGV当前不在取货点
  28. if (rgv.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站) && rgv.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.任务完成))
  29. {
  30. //下达一个目标为1030的移动任务
  31. rgv.Data.TaskID_1 = 1030;
  32. rgv.Data.DestPosition_1 = 1030;
  33. rgv.Data.TaskType_1 = RGVTaskType.移动;
  34. rgv.Data.Trigger_1++;
  35. }
  36. var obj = Device.Find(ConvGroup_1030).Create<StationDeviceGroup>();
  37. obj.WhetherToExecute();
  38. //筛选出有任务号和起始及目标地址的设备
  39. var dev = obj.RGVGetTaskedDevice() ?? throw new WarnException("无可用任务");
  40. DB.Do(db =>
  41. {
  42. var taskids = dev.Select(v => v.Data2.Tasknum);
  43. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  44. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  45. throw new WarnException("任务组ID不一致");
  46. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  47. throw new WarnException("无任务组ID");
  48. var gw1 = obj.Items.ToArray()[0];
  49. var gw2 = obj.Items.ToArray()[1];
  50. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  51. {
  52. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{obj.Entity.CODE}目标地址不一致");
  53. }
  54. rgv.Data.TaskID_1 = gw1.Data2.Tasknum;
  55. rgv.Data.TaskID_2 = gw2.Data2.Tasknum;
  56. if (gw1.Data2.Tasknum != 0)
  57. {
  58. rgv.Data.StartPosition_1 = gw1.Entity.CODE.ToShort();
  59. rgv.Data.DestPosition_1 = gw1.Data2.Goodsend;
  60. }
  61. else
  62. {
  63. rgv.Data.StartPosition_1 = gw1.Entity.CODE.ToShort();
  64. rgv.Data.DestPosition_1 = gw2.Data2.Goodsend;
  65. }
  66. rgv.Data.Trigger_1++;
  67. foreach (var task in tasks)
  68. {
  69. var msg = $"下达从{rgv.Data.StartPosition_1}移动至{rgv.Data.DestPosition_1}的RGV PLC指令。";
  70. msg += $"[{task.ID}][{rgv.Data.StartPosition_1}][{rgv.Data.DestPosition_1}[{rgv.Data.Trigger_1}]";
  71. task.CreateStatusLog(db, msg, this.GetType());
  72. }
  73. });
  74. });
  75. }
  76. protected override bool SelectDevice(WCS_DEVICE dev)
  77. {
  78. return dev.CODE == RGV8;
  79. }
  80. }
  81. [WorkTitle(typeof(RGVHandler), "涂布环穿")]
  82. public class 涂布环穿 : DeviceWork<RGVDevice>
  83. {
  84. protected override void Do(RGVDevice obj)
  85. {
  86. //11号站台是取货点,但不是待命点,如需取货,从九号站台调车
  87. //2号站台是一个待命点,但是为了保证入库取货效率,所有后面会跟一个车
  88. //如果3号站台的小车拦住了后一个小车的放货任务,检测一次最近的空车距离值,如果大于或等于618500并且小于被拦住小车的位置值再进行调车,用于避免无效空跑
  89. obj.EX(rgvDevice =>
  90. {
  91. if (rgvDevice.Data2.Trigger_1 != rgvDevice.Data.Trigger_1) throw new WarnException($"等待执行任务{rgvDevice.Data2.TaskID_1}--{rgvDevice.Data2.TaskID_2}");
  92. if (rgvDevice.Data2.WorkMode != RGVMode.自动) throw new DoException(rgvDevice.Data2.WorkMode.ToString());
  93. if (rgvDevice.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException(rgvDevice.Data2.SystemStatus.ToString());
  94. //RGV当前是否刚刚完成取货任务,等待放货
  95. if (rgvDevice.IsPut())
  96. {
  97. if (!rgvDevice.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV无光电,无法放货,请检查实际情况");
  98. //取小车上的任务
  99. DB.Do(db =>
  100. {
  101. var taskids = new List<int>() { rgvDevice.Data2.TaskID_1, rgvDevice.Data2.TaskID_2 };
  102. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  103. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  104. throw new WarnException("任务组ID不一致");
  105. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  106. throw new WarnException("无任务组ID");
  107. var destStation = Device.Find(tasks.FirstOrDefault()!.ADDRNEXT).Create<StationDeviceGroup>();
  108. rgvDevice.Put(destStation, rgvDevice.Data2.TaskID_1, rgvDevice.Data2.TaskID_2);
  109. });
  110. return;
  111. }
  112. //当前有rgv的取货站台
  113. var pickStation = rgvDevice.CurrentStation();
  114. //RGV是否在任何一个站台
  115. if (pickStation != null)
  116. {
  117. //是否是取货站台
  118. if (pickStation.Entity.Is(DF.涂布RGV取货设备组))
  119. {
  120. // 筛选出有任务号和起始及目标地址的设备
  121. var devise = pickStation.RGVGetTaskedDevice();
  122. //是否需要取货
  123. if (devise is { Count: > 0 } && pickStation.IsPickUp(rgvDevice))
  124. {
  125. //开始下达取货任务
  126. DB.Do(db =>
  127. {
  128. if (rgvDevice.Data.TaskID_1 != pickStation.Entity.CODE.Replace("G","").ToShort()) throw new WarnException($"目标站台{rgvDevice.Data.TaskID_1}与当前站台{pickStation.Entity.CODE.Replace("G", "").ToShort()}不一致");
  129. if (rgvDevice.Data2.TaskType_1 != RGVTaskType.移动) throw new WarnException($"RGV正在执行{rgvDevice.Data2.TaskType_1}");
  130. if (rgvDevice.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法取货,请检查实际情况");
  131. if (!rgvDevice.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) throw new WarnException("RGV无到站状态,请检查RGV实际状态");
  132. var taskids = devise.Select(v => v.Data2.Tasknum);
  133. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  134. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  135. throw new WarnException("任务组ID不一致");
  136. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  137. throw new WarnException("无任务组ID");
  138. var gw1 = pickStation.Items.ToArray()[0];
  139. var gw2 = pickStation.Items.ToArray()[1];
  140. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  141. {
  142. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{rgvDevice.Entity.CODE}目标地址不一致");
  143. }
  144. InfoLog.INFO_RGVINFO($"[{rgvDevice.Entity.CODE}]--写入RGV取货任务-开始:{rgvDevice.Data.TaskID_1},{rgvDevice.Data.TaskID_2},{rgvDevice.Data.TaskType_1},{rgvDevice.Data.DestPosition_1},{rgvDevice.Data.Trigger_1}");
  145. rgvDevice.Data.TaskID_1 = gw1.Data2.Tasknum;
  146. rgvDevice.Data.TaskID_2 = gw2.Data2.Tasknum;
  147. rgvDevice.Data.TaskType_1 = RGVTaskType.取货;
  148. rgvDevice.Data.DestPosition_1 = pickStation.Entity.CODE.Replace("G", "").ToShort();
  149. InfoLog.INFO_RGVINFO($"[{rgvDevice.Entity.CODE}]--写入RGV取货任务-结束:{rgvDevice.Data.TaskID_1},{rgvDevice.Data.TaskID_2},{rgvDevice.Data.TaskType_1},{rgvDevice.Data.DestPosition_1},{rgvDevice.Data.Trigger_1}");
  150. rgvDevice.Data.Trigger_1++;
  151. foreach (var task in tasks)
  152. {
  153. var msg = $"下达从{rgvDevice.Data.StartPosition_1}移动至{rgvDevice.Data.DestPosition_1}的RGV PLC指令。";
  154. msg += $"[{task.ID}][{rgvDevice.Data.StartPosition_1}][{rgvDevice.Data.DestPosition_1}[{rgvDevice.Data.Trigger_1}]";
  155. task.CreateStatusLog(db, msg, this.GetType());
  156. }
  157. });
  158. return;
  159. }
  160. if (rgvDevice.RGVList.Count + 1 != 6)//有小车进入维修站时启用该逻辑,避免在正常使用时影响效率,RGVList把自身排除在外了所以需要+1
  161. {
  162. //var stationList = rgvDevice.LocationList.Where(v => v.Entity.Is(DF.涂布RGV取货设备组)).Where(v => v.RGVGetTaskedDevice().Any())
  163. // .Where(v =>
  164. // {
  165. // if (v.CurrentRGV() == null) return true;
  166. // return false;
  167. // });
  168. //if (stationList != null)
  169. //{
  170. // var station11 = stationList.MinBy(rgvDevice.Distance);
  171. // //该站台到当前小车的范围中没有车
  172. // if (rgvDevice.RGVList.Any(v => station11!.Position <= v.Position && rgvDevice.Position > v.Position))
  173. // {
  174. // rgvDevice.Move(station11);
  175. // return;
  176. // }
  177. //}
  178. }
  179. if (pickStation.Entity.CODE == "G9")
  180. {
  181. //九站台的小车需要额外检测一下11站台是否需要取货
  182. var station11 = Device.Find("G11").Create<StationDeviceGroup>();
  183. var station11Dev = station11.RGVGetTaskedDevice();
  184. if (station11Dev != null && station11Dev.Count > 0)
  185. {
  186. rgvDevice.Move(station11);
  187. return;
  188. }
  189. }
  190. //找到自己的后一个小车
  191. var afterRgv = rgvDevice.After();
  192. //自己是否阻挡了该小车
  193. if (afterRgv.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.SystemStatus != RGVRunStatus.空闲 && rgvDevice.StopedByMe(afterRgv))
  194. {
  195. switch (pickStation.Entity.CODE)
  196. {
  197. case "G2":
  198. {
  199. if (rgvDevice.RGVList.Count + 1 != 6) break;
  200. //此站台因需要进行排队,因此是否需要赶车取决于是否拦住了两个小车的任务
  201. var afterRgv1 = afterRgv.After();
  202. //如果是空闲或者没有拦住他,就不执行赶车
  203. if (afterRgv1.Data2.WorkMode == RGVMode.自动 && (afterRgv1.Data2.SystemStatus == RGVRunStatus.空闲 || !afterRgv.StopedByMe(afterRgv1))) return;
  204. break;
  205. }
  206. case "G3":
  207. {
  208. if (rgvDevice.RGVList.Count + 1 != 6) break;
  209. //为了减少空跑,需要被阻拦的小车后面的小车任务类型为移动并且位置值要大于618500小于被阻拦小车的位置值
  210. var afterRgv1 = afterRgv.After();
  211. if (afterRgv1.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.TaskType_1 != RGVTaskType.移动 && !(afterRgv1.Position < afterRgv.Position && afterRgv1.Position > 618500)) return;
  212. break;
  213. }
  214. }
  215. //找到当前所在站台的下一个取货点
  216. var beforeStation = Device.Where(v => v.Is(DF.涂布RGV取货设备组) && v.CODE != pickStation.Entity.CODE)
  217. .Select(v => v.Create<StationDeviceGroup>()).MinBy(v => pickStation.Distance(v));
  218. //写入移动任务
  219. rgvDevice.Move(beforeStation);
  220. return;
  221. }
  222. if (pickStation.Entity.CODE == "G11")
  223. {
  224. rgvDevice.Move(Device.Find("G2").Create<StationDeviceGroup>());
  225. return;
  226. }
  227. }
  228. //是否在一个放货站台
  229. if (pickStation.Entity.Is(DF.涂布RGV放货设备组))
  230. {
  231. pickStation = rgvDevice.BeforeStation();
  232. rgvDevice.Move(pickStation);
  233. return;
  234. }
  235. }
  236. //此时RGV即没有等待执行的放货任务,也不在任何一个取货点,因此需要调往最近的一个取货点
  237. //找到距离这个RGV最近的一个取货点
  238. //必须所有RGV都是空闲状态时才可以进行初始化
  239. if (rgvDevice.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲)) return;
  240. pickStation = rgvDevice.BeforeStation();
  241. rgvDevice.Move(pickStation);
  242. });
  243. }
  244. protected override bool SelectDevice(WCS_DEVICE dev)
  245. {
  246. return dev.Is(DF.涂布RGV);
  247. }
  248. }
  249. [WorkTitle(typeof(RGVHandler), "BOPP环穿")]
  250. public class BOPP环穿 : DeviceWork<RGVDevice>
  251. {
  252. protected override void Do(RGVDevice obj)
  253. {
  254. obj.EX(obj =>
  255. {
  256. if (obj.Data2.Trigger_1 != obj.Data.Trigger_1) throw new WarnException($"等待执行任务{obj.Data2.TaskID_1}--{obj.Data2.TaskID_2}");
  257. if (obj.Data2.WorkMode != RGVMode.自动) throw new DoException(obj.Data2.WorkMode.ToString());
  258. if (obj.Data2.SystemStatus != RGVRunStatus.空闲) throw new DoException(obj.Data2.SystemStatus.ToString());
  259. //RGV当前是否刚刚完成取货任务,等待放货
  260. if (obj.IsPut())
  261. {
  262. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV无光电,无法放货,请检查实际情况");
  263. //取小车上的任务
  264. DB.Do(db =>
  265. {
  266. List<int> taskids = new List<int>() { obj.Data2.TaskID_1, obj.Data2.TaskID_2 };
  267. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  268. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  269. throw new WarnException("任务组ID不一致");
  270. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  271. throw new WarnException("无任务组ID");
  272. var destStation = Device.Find(tasks.FirstOrDefault().ADDRNEXT).Create<StationDeviceGroup>();
  273. obj.Put(destStation, obj.Data2.TaskID_1, obj.Data2.TaskID_2);
  274. });
  275. return;
  276. }
  277. var pickStation = obj.CurrentStation();
  278. //RGV是否在任何一个站台
  279. if (pickStation != null)
  280. {
  281. //是否是取货站台
  282. if (pickStation.Entity.Is(DF.BOPPRGV取货设备组))
  283. {
  284. // 筛选出有任务号和起始及目标地址的设备
  285. var devs = pickStation.RGVGetTaskedDevice();
  286. //是否需要取货
  287. if (devs != null && devs.Count() > 0)
  288. {
  289. //开始下达取货任务
  290. DB.Do(db =>
  291. {
  292. if (obj.Data2.TaskType_1 != RGVTaskType.移动) throw new WarnException($"RGV正在执行{obj.Data2.TaskType_1}");
  293. if (obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法取货,请检查实际情况");
  294. if (!obj.Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) throw new WarnException("RGV无到站状态,请检查RGV实际状态");
  295. var taskids = devs.Select(v => v.Data2.Tasknum);
  296. var tasks = db.Default.Set<WCS_TASK>().Where(p => taskids.Contains(p.ID));
  297. if (tasks.GroupBy(p => p.TaskGroupKey).Count() > 1)
  298. throw new WarnException("任务组ID不一致");
  299. if (!tasks.GroupBy(p => p.TaskGroupKey).Any())
  300. throw new WarnException("无任务组ID");
  301. var gw1 = pickStation.Items.ToArray()[0];
  302. var gw2 = pickStation.Items.ToArray()[1];
  303. if (gw1.Data2.Tasknum != 0 && gw2.Data2.Tasknum != 0)
  304. {
  305. if (gw1.Data2.Goodsend != gw2.Data2.Goodsend) throw new WarnException($"{obj.Entity.CODE}目标地址不一致");
  306. }
  307. obj.Pick(pickStation, gw1.Data2.Tasknum, gw2.Data2.Tasknum);
  308. foreach (var task in tasks)
  309. {
  310. var msg = $"下达从{obj.Data.StartPosition_1}移动至{obj.Data.DestPosition_1}的RGV PLC指令。";
  311. msg += $"[{task.ID}][{obj.Data.StartPosition_1}][{obj.Data.DestPosition_1}[{obj.Data.Trigger_1}]";
  312. task.CreateStatusLog(db, msg, this.GetType());
  313. }
  314. });
  315. return;
  316. }
  317. ////找到自己的后一个小车
  318. //var afterRgv = obj.After();
  319. ////自己是否阻挡了该小车
  320. //if (afterRgv.Data2.WorkMode == RGVMode.自动 && afterRgv.Data2.SystemStatus != RGVRunStatus.空闲 && obj.StopedByMe(afterRgv))
  321. //{
  322. // //找到当前所在站台的下一个取货点
  323. // var beforeStation = Device.Where(v => v.Is(DF.BOPPRGV取货设备组) && v.CODE != pickStation.Entity.CODE)
  324. // .Select(v => v.Create<StationDeviceGroup>())
  325. // .OrderBy(v => pickStation.Distance(v))
  326. // .FirstOrDefault();
  327. // //写入移动任务
  328. // obj.Move(beforeStation);
  329. // return;
  330. //}
  331. //else
  332. //{
  333. // //取前一个取货点
  334. // pickStation = obj.BeforeStation();
  335. // //前一个取货点的小车
  336. // var rgv = pickStation.CurrentRGV();
  337. // //前一个取货点没有车 且没有非空闲目的地为前一个取货点的小车
  338. // if (rgv == null && !obj.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲 && v.Data2.DestPosition_1 == pickStation.Entity.CODE.Replace("G", "").ToShort()))
  339. // {
  340. // obj.Move(pickStation);
  341. // return;
  342. // }
  343. //}
  344. //计算当前RGV拦住小车的数量
  345. var max = obj.RGVList.Count(v => v.Data2.WorkMode == RGVMode.自动 && v.Data2.SystemStatus != RGVRunStatus.空闲 && obj.StopedByMe(v));
  346. if (pickStation.Entity.CODE == "G19" && max > 2)
  347. {
  348. obj.Move(Device.Find("G23").Create<StationDeviceGroup>());
  349. }
  350. //else if (pickStation.Entity.CODE == "G23" && max > 3)
  351. //{
  352. // obj.Move(Device.Find("G19").Create<StationDeviceGroup>());
  353. //}
  354. }
  355. //是否在一个放货站台
  356. if (pickStation.Entity.Is(DF.BOPPRGV放货设备组))
  357. {
  358. pickStation = obj.BeforeStation();
  359. obj.Move(Device.Find("G23").Create<StationDeviceGroup>());
  360. }
  361. return;
  362. }
  363. //此时RGV即没有等待执行的放货任务,也不在任何一个取货点,因此需要调往最近的一个取货点
  364. //找到距离这个RGV最近的一个取货点
  365. //必须所有RGV都是空闲状态时才可以进行初始化
  366. if (obj.RGVList.Any(v => v.Data2.SystemStatus != RGVRunStatus.空闲 || v.Data2.WorkMode != RGVMode.自动)) return;
  367. pickStation = obj.BeforeStation();
  368. obj.Move(Device.Find("G23").Create<StationDeviceGroup>());
  369. });
  370. }
  371. protected override bool SelectDevice(WCS_DEVICE dev)
  372. {
  373. return dev.Is(DF.BOPPRGV);
  374. }
  375. }
  376. }