NoInteractionSystems.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. using PlcSiemens.Core.Extension;
  2. using ServiceCenter.Extensions;
  3. using ServiceCenter.Logs;
  4. using ServiceCenter.SqlSugars;
  5. using SqlSugar;
  6. using System.ComponentModel;
  7. using WCS.Core;
  8. using WCS.Entity;
  9. using WCS.Entity.Protocol.Station;
  10. using WCS.WorkEngineering.Extensions;
  11. using WCS.WorkEngineering.Worlds;
  12. using DeviceFlags = WCS.WorkEngineering.Extensions.DeviceFlags;
  13. using TaskStatus = WCS.Entity.TaskStatus;
  14. namespace WCS.WorkEngineering.Systems
  15. {
  16. /// <summary>
  17. /// 无交互系统
  18. /// </summary>
  19. [BelongTo(typeof(NoInteractionWorld))]
  20. [Description("无交互系统")]
  21. public class NoInteractionSystems : DeviceSystem<Device<IStation520>>
  22. {
  23. protected override bool ParallelDo => true;
  24. public override void Do(Device<IStation520> obj)
  25. {
  26. var taskInfos = new List<int>();
  27. var sqlSugar = new SqlSugarHelper();
  28. SqlSugarHelper.Do(db =>
  29. {
  30. //获取所有的新建任务,组盘任务不需要
  31. taskInfos = db.Default.Queryable<WCS_TaskInfo>().ReadPastUpdLock().Where(x => x.Status == 0).ToList()
  32. .Where(x => x.Type != TaskType.SetPlate || (x.Type == TaskType.SetPlate && x.AddrFrom != "Robot")).Where(x => x.BusType != "人工满托入库").Select(x => x.ID).ToList();
  33. var time = DateTime.Now.AddMinutes(-20);
  34. var time1 = DateTime.Now.AddMinutes(-40);
  35. var timeOut = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "车间叫料" && x.Status < TaskStatus.StackerExecution && x.AddTime < time);
  36. if (timeOut > 0) World.Log($"共有{timeOut}个车间叫料任务超过二十分钟未执行,请检查相关相关任务对应的堆垛机是否正常、堆垛机放货点是否卡任务等", LogLevelEnum.High);
  37. var timeOut1 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "车间叫料" && x.Status < TaskStatus.StackerExecution && x.AddTime < time1);
  38. if (timeOut1 > 0) World.Log($"共有{timeOut1}个车间叫料任务超过四十分钟未执行,请检查相关相关任务对应的堆垛机是否正常、堆垛机放货点是否卡任务等", LogLevelEnum.High);
  39. var timeOut2 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "皮盘入库" && x.Status < TaskStatus.StackerExecution && x.AddTime < time);
  40. if (timeOut2 > 0) World.Log($"共有{timeOut2}个皮盘入库任务超过二十分钟未执行,请检查相关相关任务及对应agv设备", LogLevelEnum.High);
  41. var timeOut3 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "皮盘入库" && x.Status < TaskStatus.StackerExecution && x.AddTime < time1);
  42. if (timeOut3 > 0) World.Log($"共有{timeOut3}个皮盘入库任务超过四十分钟未执行,请检查相关相关任务及对应agv设备", LogLevelEnum.High);
  43. var time3 = DateTime.Now.AddHours(-24);
  44. var timeOut4 = db.Default.Queryable<BillInvnow>().NoLock().Count(x => x.AddTime < time3);
  45. if (timeOut4 > 0) World.Log($"共有{timeOut4}个库存信息超过24小时未执行,请检查相关库存信息并处理", LogLevelEnum.High);
  46. });
  47. if (!taskInfos.Any())
  48. {
  49. return;
  50. }
  51. var isEnd = false; //每个周期只处理一个任务
  52. foreach (var item in taskInfos)
  53. {
  54. if (isEnd) return;
  55. try
  56. {
  57. SqlSugarHelper.Do(db =>
  58. {
  59. var task = db.Default.Queryable<WCS_TaskInfo>().Where(t => t.ID == item).First();
  60. switch (task.Type)
  61. {
  62. case TaskType.SetPlate:
  63. if (task.AddrFrom != "Robot")
  64. {
  65. //更新任务状态
  66. task.Status = TaskStatus.WaitingToExecute;
  67. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status }).ExecuteCommand();
  68. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  69. isEnd = true;
  70. }
  71. break;
  72. case TaskType.EnterDepot:
  73. if (task.LastInteractionPoint == "2")
  74. {
  75. var sta = GetAgvStation(task, db);
  76. var agv = new WCS_AgvTaskInfo()
  77. {
  78. ID = db.GetAgvTaskId(),
  79. TaskType = AGVTaskType.EnterDepot,
  80. Status = AGVTaskStatus.NewBuild,
  81. TaskId = task.ID,
  82. Position = task.WorkBench,
  83. Station = sta,
  84. AddWho = "WCS",
  85. AddTime = DateTime.Now
  86. };
  87. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  88. task.AgvTaskID = agv.ID;
  89. task.Status = Entity.TaskStatus.WaitingToExecute;
  90. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.AgvTaskID }).ExecuteCommand();
  91. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化单独返皮盘任务");
  92. }
  93. else
  94. {
  95. //更新任务状态
  96. task.Status = Entity.TaskStatus.WaitingToExecute;
  97. task.Device = task.WarehouseCode switch
  98. {
  99. "1N" => "SRM1",
  100. "1S" => "SRM2",
  101. "2N" => "SRM3",
  102. "2S" => "SRM4",
  103. "3N" => "SRM5",
  104. "3S" => "SRM6",
  105. _ => task.Device
  106. };
  107. task.Device = task.AddrFrom switch
  108. {
  109. "1666" or "1661" => "SRM1",
  110. "1681" or "1676" => "SRM2",
  111. "1696" or "1691" => "SRM3",
  112. "1711" or "1706" => "SRM4",
  113. "1726" or "1721" => "SRM5",
  114. "1741" or "1736" => "SRM6",
  115. _ => task.Device
  116. };
  117. //计算下一个地址
  118. var path1 = DevicePath.GetPath(task.AddrFrom, task.Device);
  119. task.AddrNext = path1.Points[1].Code;
  120. task.SrmStation = task.BarCode.Contains("TPA") || task.BarCode.Contains("TPB") ? task.AddrFrom : path1.Points[2].Code;
  121. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.Device, x.AddrNext, x.SrmStation }).ExecuteCommand();
  122. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  123. }
  124. isEnd = true;
  125. break;
  126. case TaskType.OutDepot:
  127. {
  128. if (task.Device.Contains("Robot"))
  129. {
  130. var lastTask = db.Default.Queryable<WCS_TaskInfo>().NoLock().First(x => x.Type == TaskType.OutDepot && x.AddrTo == task.AddrTo && x.Status > TaskStatus.WaitingToExecute && x.Status < TaskStatus.Finish);
  131. if (lastTask != null)
  132. {
  133. World.Log($"等待任务结束:{task.AddrTo}上一组码垛任务未结束,等待{lastTask.ID}任务完成后开始初始化当前任务{task.ID}");
  134. return;
  135. }
  136. var pos = task.AddrFrom.Split("-");
  137. task.Status = Entity.TaskStatus.WaitingToExecute;
  138. task.Line = pos[0].ToShort();
  139. task.Col = pos[1].ToShort();
  140. task.Layer = pos[2].ToShort();
  141. task.Depth = pos[3].ToShort();
  142. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.Line, x.Col, x.Layer, x.Depth }).ExecuteCommand();
  143. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,放货站台:{task.SrmStation}");
  144. isEnd = true;
  145. }
  146. else
  147. {
  148. if (task.SrmStation == "1")
  149. {
  150. var srmStation = new string[2];
  151. switch (task.Device)
  152. {
  153. case "SRM1":
  154. srmStation[0] = "2534";
  155. srmStation[1] = "2533";
  156. break;
  157. case "SRM2":
  158. srmStation[0] = "2734";
  159. srmStation[1] = "2733";
  160. break;
  161. case "SRM3":
  162. srmStation[0] = "2934";
  163. srmStation[1] = "2933";
  164. break;
  165. case "SRM4":
  166. srmStation[0] = "3134";
  167. srmStation[1] = "3133";
  168. break;
  169. case "SRM5":
  170. srmStation[0] = "3334";
  171. srmStation[1] = "3333";
  172. break;
  173. case "SRM6":
  174. srmStation[0] = "3534";
  175. srmStation[1] = "3533";
  176. break;
  177. }
  178. //开始计算当前这个任务要从哪个站台出
  179. //计算两个站台小于取货完成数量的AGV任务
  180. var agv1 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[0]).Count();
  181. var agv2 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[1]).Count();
  182. task.SrmStation = "";
  183. task.AddrTo = agv1 > agv2 ? srmStation[1] : srmStation[0];
  184. }
  185. if (task.SrmStation.IsNullOrEmpty()) //如果没有指定放货站台
  186. {
  187. if (task.Device.IsNullOrEmpty())
  188. {
  189. task.Device = "SRM" + task.Tunnel.GetLastDigit();
  190. }
  191. //获取堆垛机到目标地址的路径信息
  192. var path = DevicePath.GetPath(task.Device, task.AddrTo);
  193. task.SrmStation = path.Points[1].Code;
  194. }
  195. var devs = Device.All.Where(x => x.HasFlag(DeviceFlags.AGV取货站台口)).Select(x => x.Code);
  196. if (devs.Contains(task.SrmStation) && task.LastInteractionPoint != "3") //车间叫料任务,此值为3时是单独取满任务
  197. {
  198. var sta = GetAgvStation(task, db);
  199. var agv = new WCS_AgvTaskInfo()
  200. {
  201. ID = db.GetAgvTaskId(),
  202. TaskType = AGVTaskType.EnterDepot,
  203. Status = AGVTaskStatus.NewBuild,
  204. TaskId = task.ID,
  205. Position = task.WorkBench,
  206. Station = sta,
  207. AddWho = "WCS",
  208. AddTime = DateTime.Now
  209. };
  210. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  211. }
  212. //更新任务状态
  213. task.Status = Entity.TaskStatus.WaitingToExecute;
  214. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.SrmStation, x.AddrTo, x.Status }).ExecuteCommand();
  215. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,放货站台:{task.SrmStation}");
  216. isEnd = true;
  217. }
  218. break;
  219. }
  220. case TaskType.Delivery:
  221. break;
  222. case TaskType.EmptyInit:
  223. break;
  224. }
  225. });
  226. }
  227. catch (Exception ex)
  228. {
  229. if (ex.Message.Contains("Index")) World.Log($"{ex.Message}:{ex.StackTrace}");
  230. if (ex.Message.Contains("SqlTransaction")) World.Log($"{ex.Message}:{ex.StackTrace}");
  231. World.Log(ex.Message, LogLevelEnum.Mid);
  232. }
  233. }
  234. }
  235. public override bool Select(Device dev)
  236. {
  237. return dev.Code == nameof(NoInteractionSystems);
  238. }
  239. /// <summary>
  240. /// 获取AGV的目标站台
  241. /// </summary>
  242. /// <param name="task"></param>
  243. /// <param name="db"></param>
  244. /// <returns></returns>
  245. public string GetAgvStation(WCS_TaskInfo task, SqlSugarHelper db)
  246. {
  247. string sta = "";
  248. var n = 0;
  249. var s = 0;
  250. //同侧之间均分
  251. if (task.WarehouseCode.Contains("1"))
  252. {
  253. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("25"))
  254. .SplitTable(v => v.Take(2)).Count();
  255. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("27"))
  256. .SplitTable(v => v.Take(2)).Count();
  257. sta = n > s ? "2701" : "2501";
  258. }
  259. else if (task.WarehouseCode.Contains("2"))
  260. {
  261. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("29"))
  262. .SplitTable(v => v.Take(2)).Count();
  263. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("31"))
  264. .SplitTable(v => v.Take(2)).Count();
  265. sta = n > s ? "3101" : "2901";
  266. }
  267. else if (task.WarehouseCode.Contains("3"))
  268. {
  269. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("33"))
  270. .SplitTable(v => v.Take(2)).Count();
  271. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station.Contains("35"))
  272. .SplitTable(v => v.Take(2)).Count();
  273. sta = n > s ? "3501" : "3301";
  274. }
  275. else
  276. {
  277. return default;
  278. }
  279. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v =>
  280. v.Status >= AGVTaskStatus.Confirm &&
  281. v.Status < AGVTaskStatus.Complete1 &&
  282. v.TaskType == AGVTaskType.EnterDepot && v.Station == sta)
  283. .SplitTable(v => v.Take(2)).Count();
  284. if (n > 5)
  285. {
  286. sta = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot)
  287. .SplitTable(v => v.Take(2)).ToList().GroupBy(x => x.Station).Select(x => new { x.Key, Count = x.Count() }).MinBy(x => x.Count).Key;
  288. }
  289. return sta;
  290. }
  291. }
  292. }