RGVSystems.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. using ServiceCenter.Extensions;
  2. using ServiceCenter.Logs;
  3. using ServiceCenter.SqlSugars;
  4. using System.ComponentModel;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol.BCR;
  8. using WCS.Entity.Protocol.RGV;
  9. using WCS.Entity.Protocol.Robot;
  10. using WCS.Entity.Protocol.Station;
  11. using WCS.Entity.Protocol.Truss;
  12. using WCS.WorkEngineering.Extensions;
  13. using WCS.WorkEngineering.Worlds;
  14. using DeviceFlags = WCS.WorkEngineering.Extensions.DeviceFlags;
  15. using TaskStatus = WCS.Entity.TaskStatus;
  16. namespace WCS.WorkEngineering.Systems
  17. {
  18. /// <summary>
  19. /// Agv交互系统
  20. /// </summary>
  21. [BelongTo(typeof(RgvWorld))]
  22. [Description("RGV交互系统")]
  23. public class RGVSystems : DeviceSystem<Device<IRGV520, IRGV521, IBCR81>>
  24. {
  25. protected override bool ParallelDo => true;
  26. protected override bool SaveLogsToFile => true;
  27. /// <summary>
  28. /// 取货点设备集合
  29. /// </summary>
  30. private readonly Dictionary<string, List<Device<IStation520, IStation521, IStation523>>> _pickUpDevices = new();
  31. public RGVSystems()
  32. {
  33. //获取所有的巷道集合
  34. var rgvList = Device.All.Where(v => v.HasFlag(DeviceFlags.RGV));
  35. foreach (var rgv in rgvList)
  36. {
  37. _pickUpDevices.Add(rgv.Code, rgv.Sources.Where(v => v.HasFlag(DeviceFlags.输送机)).Select(v => new Device<IStation520, IStation521, IStation523>(v, this.World)).ToList());
  38. }
  39. }
  40. public override void Do(Device<IRGV520, IRGV521, IBCR81> obj)
  41. {
  42. try
  43. {
  44. if (obj.Data.VoucherNo != obj.Data2.VoucherNo) throw new KnownException($"凭证号不一致,DB520:{obj.Data.VoucherNo}-DB521:{obj.Data2.VoucherNo}", LogLevelEnum.High);
  45. if (obj.Data2.WorkMode != RGVWorkMode.Automatic) return;
  46. if (obj.Data.RES1 == 1) return; //wcs任务完成确认信号未清除
  47. if (obj.Data2.Status.HasFlag(RGVStatus.Taskfinishi))
  48. {
  49. switch (obj.Data2.CmdType)
  50. {
  51. case RGVCmdType.PickGoods: //单独取货任务完成,默认只有空托盘才会下发单独取货任务
  52. //开始申请读码信息
  53. var bcrCode = obj.Data3.GetBCRCode();
  54. var taskNumber = 0;
  55. SqlSugarHelper.Do(_db =>
  56. {
  57. var db = _db.Default;
  58. var dev = Device.All.First(x => x.Code == obj.Data2.StartPosition.ToString());
  59. if (dev.HasFlag(DeviceFlags.拆盘机09))
  60. {
  61. //开始绑定任务,并下发新的任务信息到小车
  62. var palletizingInfo = db.Queryable<WCS_Palletizing>().Single(x => x.Id == obj.Data2.TaskNumber);
  63. if (palletizingInfo == null) return;
  64. palletizingInfo.PalleCode = bcrCode;
  65. db.Updateable(palletizingInfo).ExecuteCommand();
  66. taskNumber = palletizingInfo.Id;
  67. }
  68. else if (dev.HasFlag(DeviceFlags.拆盘机非09))
  69. {
  70. //开始处理对应的搬运任务信息
  71. var task = db.Queryable<WCS_TaskInfo>().First(x => x.Type == TaskType.Delivery && x.ID == obj.Data2.TaskNumber && x.AddrTo == obj.Data2.DestPosition.ToString());
  72. if (task == null) throw new KnownException($"未找到对应的任务{obj.Data2.TaskNumber}", LogLevelEnum.Mid);
  73. task.BarCode = bcrCode;
  74. db.Updateable(task).ExecuteCommand();
  75. task.AddWCS_TASK_DTL(db, obj.Entity.Code, obj.Data2.DestPosition.ToString(), $"环形库码垛位{obj.Data2.DestPosition}搬运任务绑定条码信息{bcrCode}");
  76. taskNumber = task.ID;
  77. }
  78. });
  79. if (taskNumber == 0) return;
  80. //清空起始点信息
  81. var staDev = new Device<IStation520>(Device.All.FirstOrDefault(x => x.Code == obj.Data2.DestPosition.ToString())!, World);
  82. staDev.Data.TaskNumber = 0;
  83. staDev.Data.GoodsStart = 0;
  84. staDev.Data.GoodsEnd = 0;
  85. obj.Data2.TaskNumber = taskNumber;
  86. obj.Data.RES1 = 1;
  87. break;
  88. case RGVCmdType.PutGoods:
  89. WCS_TaskInfo finishiTask = null;
  90. var startPosition = Device.All.Where(x => x.Code == obj.Data2.StartPosition.ToString()).Select(x => new Device<IStation520, IStation521, IStation523>(x, World)).FirstOrDefault();
  91. var destPosition = Device.All.FirstOrDefault(x => x.Code == obj.Data2.DestPosition.ToString());
  92. var isPalletizing = destPosition!.HasFlag(DeviceFlags.桁架码垛位, DeviceFlags.环形库码垛工位);
  93. short countQty = 0;
  94. short shortCode = 0;
  95. SqlSugarHelper.Do(_db =>
  96. {
  97. var db = _db.Default;
  98. if (isPalletizing)
  99. {
  100. if (destPosition.HasFlag(DeviceFlags.桁架码垛位))
  101. {
  102. var palletizingInfo = db.Queryable<WCS_Palletizing>().First(x => x.Id == obj.Data.TaskNumber);
  103. countQty = palletizingInfo.CountQty.ToShort();
  104. shortCode = palletizingInfo.ShortCode;
  105. }
  106. else if (destPosition.HasFlag(DeviceFlags.环形库码垛工位))
  107. {
  108. var deliveryTask = db.Queryable<WCS_TaskInfo>().First(x => x.ID == obj.Data.TaskNumber);
  109. countQty = deliveryTask.FullQty;
  110. shortCode = deliveryTask.PalletType;
  111. deliveryTask.Status = TaskStatus.RgvCompleted;
  112. deliveryTask.EditTime = DateTime.Now;
  113. deliveryTask.LastInteractionPoint = obj.Entity.Code;
  114. db.Updateable(deliveryTask).ExecuteCommand();
  115. deliveryTask.AddWCS_TASK_DTL(db, deliveryTask.AddrTo, $"RGV任务执行结束");
  116. }
  117. }
  118. });
  119. if (startPosition.Data.TaskNumber == obj.Data.TaskNumber) //初始化起始点信息
  120. {
  121. startPosition.Data.TaskNumber = 0;
  122. startPosition.Data.GoodsEnd = 0;
  123. }
  124. //目标地址是码垛工位
  125. if (isPalletizing)
  126. {
  127. if (destPosition.HasFlag(DeviceFlags.桁架码垛位))
  128. {
  129. var dest = new Device<ITruss530>(destPosition!, World);
  130. dest.Data.MaxQuantity = countQty;
  131. dest.Data.Type = shortCode;
  132. dest.Data.VoucherNo++;
  133. }
  134. else if (destPosition.HasFlag(DeviceFlags.环形库码垛工位))
  135. {
  136. var dest = new Device<IRobot530>(destPosition!, World);
  137. dest.Data.MaxQuantity = countQty;
  138. dest.Data.Type = shortCode;
  139. dest.Data.VoucherNo++;
  140. }
  141. }
  142. obj.Data.RES1 = 1;
  143. break;
  144. case RGVCmdType.Move:
  145. obj.Data.RES1 = 1;
  146. break;
  147. case RGVCmdType.ChangePutGoods:
  148. break;
  149. case RGVCmdType.ChangePickGoods:
  150. break;
  151. case RGVCmdType.PickPutGoods:
  152. staDev = new Device<IStation520, IStation521>(Device.All.FirstOrDefault(x => x.Code == obj.Data.DestPosition.ToString())!, World);
  153. staDev.Data.TaskNumber = 0;
  154. staDev.Data.GoodsStart = 0;
  155. staDev.Data.GoodsEnd = 0;
  156. obj.Data.RES1 = 1;
  157. break;
  158. default:
  159. throw new ArgumentOutOfRangeException();
  160. }
  161. return;
  162. }
  163. if (obj.Data2.SystemStatus != RGVSystemStatus.空闲) return;
  164. if (obj.Data2.Status.HasFlag(RGVStatus.RES1)) //离开非安全区域
  165. {
  166. obj.Data.TaskNumber = obj.Data.TaskNumber;
  167. obj.Data.CmdType = RGVCmdType.Move;
  168. obj.Data.DestPosition = obj.Entity.Code switch
  169. {
  170. "RGV1" => 1668,
  171. "RGV2" => 1683,
  172. "RGV3" => 1698,
  173. "RGV4" => 1713,
  174. "RGV5" => 1728,
  175. "RGV6" => 1743,
  176. _ => throw new ArgumentOutOfRangeException()
  177. };
  178. obj.Data.VoucherNo++;
  179. return;
  180. }
  181. if (obj.Data2.CmdType == RGVCmdType.PickGoods && !obj.Data2.Status.HasFlag(RGVStatus.Taskfinishi))
  182. {
  183. if (obj.Data2.Status.HasFlag(RGVStatus.PH_Status))
  184. {
  185. obj.Data.TaskNumber = obj.Data.TaskNumber;
  186. obj.Data.CmdType = RGVCmdType.PutGoods;
  187. obj.Data.StartPosition = obj.Data2.StartPosition;
  188. obj.Data.DestPosition = obj.Data2.DestPosition;
  189. obj.Data.VoucherNo++;
  190. return;
  191. }
  192. }
  193. var pickUpDevices = _pickUpDevices.FirstOrDefault(x => x.Key == obj.Entity.Code).Value;
  194. //有货且需要搬运货物的站台
  195. var devs = pickUpDevices.Where(v => v.Data3.Status.HasFlag(StationStatus.PH_Status) && v.Data.TaskNumber != 0)
  196. .Where(v => v.Entity.Code.ToShort() != v.Data.GoodsEnd && v.Data.GoodsEnd != 0)
  197. .ToList();
  198. //筛选出目标站台无货的站台
  199. var putDev = obj.Entity.Targets.Where(x => x.HasFlag(DeviceFlags.输送机))
  200. .Select(x => new Device<IStation520, IStation521, IStation523>(x, World))
  201. .Where(x => !x.Data3.Status.HasFlag(StationStatus.PH_Status))
  202. .Select(x => x.Entity.Code.ToShort());
  203. var devList = devs.OrderBy(x => x.Entity.Code).Where(x => putDev.Contains(x.Data.GoodsEnd));
  204. foreach (var dev in devList)
  205. {
  206. //区分任务是拆盘机到码垛工位,还是码垛工位到拆盘机
  207. if (dev.Entity.HasFlag(DeviceFlags.拆盘机09, DeviceFlags.拆盘机非09))
  208. {
  209. obj.Data.TaskNumber = dev.Data.TaskNumber;
  210. obj.Data.CmdType = RGVCmdType.PickGoods;
  211. obj.Data.StartPosition = dev.Entity.Code.ToShort();
  212. obj.Data.DestPosition = dev.Data.GoodsEnd;
  213. obj.Data.VoucherNo++;
  214. return;
  215. }
  216. //非拆盘机起始任务
  217. //站台中的任务号
  218. WCS_TaskInfo task = null;
  219. SqlSugarHelper.Do(_db =>
  220. {
  221. var db = _db.Default;
  222. var taskInfo = db.Queryable<WCS_TaskInfo>().First(p => p.ID == dev.Data.TaskNumber && p.Status == TaskStatus.ConveyorExecution);
  223. if (taskInfo == null) return;
  224. taskInfo.Status = TaskStatus.RgvExecution;
  225. taskInfo.AddrNext = obj.Entity.Code;
  226. taskInfo.EditWho = "WCS";
  227. taskInfo.EditTime = DateTime.Now; ;
  228. db.Updateable(taskInfo).ExecuteCommand();
  229. taskInfo.AddWCS_TASK_DTL(db, dev.Entity.Code, obj.Entity.Code, $"任务分配至{obj.Entity.Code}");
  230. task = taskInfo;
  231. });
  232. if (task == null) continue;
  233. obj.Data.TaskNumber = task.ID;
  234. obj.Data.CmdType = RGVCmdType.PickPutGoods;
  235. obj.Data.StartPosition = dev.Entity.Code.ToShort();
  236. obj.Data.DestPosition = dev.Data.GoodsEnd;
  237. obj.Data.VoucherNo++;
  238. return;
  239. }
  240. }
  241. catch (Exception ex)
  242. {
  243. if (ex.Message == "Destination array was not long enough. Check the destination index, length, and the array's lower bounds")
  244. {
  245. World.Log($"未知异常:{ex.StackTrace}");
  246. }
  247. else if (ex.Message == "Number was less than the array's lower bound in the first dimension")
  248. {
  249. World.Log($"未知异常:{ex.StackTrace}");
  250. }
  251. else throw new KnownException(ex.Message, LogLevelEnum.Mid);
  252. }
  253. }
  254. public override bool Select(Device dev)
  255. {
  256. return dev.Code is /*"RGV1" or "RGV2" or*/ "RGV3" or "RGV4" or "RGV5" or "RGV6";
  257. //return dev.HasFlag(Extensions.DeviceFlags.RGV);
  258. }
  259. }
  260. }