AgvSystems.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. using PlcSiemens.Core.Extension;
  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.Protocol.SRM;
  8. using WCS.Entity.Protocol.Protocol.Station;
  9. using WCS.WorkEngineering.Extensions;
  10. using WCS.WorkEngineering.WebApi.Controllers;
  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. /// Agv交互系统
  18. /// </summary>
  19. [BelongTo(typeof(MainWorld))]
  20. [Description("Agv交互系统")]
  21. public class AgvSystems : DeviceSystem<Device<IStation520>>
  22. {
  23. protected override bool ParallelDo => true;
  24. protected override bool SaveLogsToFile => true;
  25. private List<Station> devs = new List<Station>();
  26. private List<SRM> srms = new List<SRM>();
  27. public AgvSystems()
  28. {
  29. devs = Device.All.Where(v => v.HasFlag(DeviceFlags.入库, DeviceFlags.巷道口)).Select(v => new Station(v, this.World)).ToList();
  30. srms = Device.All.Where(v => v.HasProtocol(typeof(ISRM520))).Select(v => new SRM(v, this.World)).ToList();
  31. }
  32. public override void Do(Device<IStation520> obj)
  33. {
  34. if (obj.Entity.Code == "出库AGV")
  35. {
  36. var agvTaskInfos = new List<WCS_AgvTaskInfo>();
  37. //获取所有未结束的叫料及背负式补空AGV任务
  38. SqlSugarHelper.Do(db =>
  39. {
  40. agvTaskInfos = db.Default.Queryable<WCS_AgvTaskInfo>().Where(v => v.TaskType == AGVTaskType.CallMaterial)
  41. .Where(v => v.Status < AGVTaskStatus.MissionCompleted)
  42. .SplitTable(tabs => tabs.Take(2)).OrderBy(v => v.AddTime).ToList();
  43. });
  44. //有需要处理的AGV任务
  45. if (agvTaskInfos.Any())
  46. {
  47. this.ExRecord(obj.Entity.Code, "可用出库AGV任务列表", agvTaskInfos.Select(v => v.ID).ToList());
  48. var taskInfos = new List<WCS_TaskInfo>();
  49. foreach (var agv in agvTaskInfos)
  50. {
  51. try
  52. {
  53. SqlSugarHelper.Do(db =>
  54. {
  55. switch (agv.AgvStatus)
  56. {
  57. //取货点安全交互
  58. case AGVTaskStatus.RequestOrPermission2 when agv.Status != AGVTaskStatus.RequestOrPermission2:
  59. {
  60. var taskInfo = db.Default.Queryable<WCS_TaskInfo>().First(v => v.ID == agv.TaskId);
  61. agv.Status = AGVTaskStatus.RequestOrPermission2;
  62. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  63. taskInfo.AddWCS_TASK_DTL(db.Default, "agv", $"允许AGV任务{agv.ID}在站台{agv.Station}取货");
  64. AgvApi.ContinueTask(agv.AgvID, agv.Station);
  65. break;
  66. }
  67. case AGVTaskStatus.PutRequestOrPermission when agv.Status != AGVTaskStatus.PutRequestOrPermission:
  68. {
  69. agv.Status = AGVTaskStatus.PutRequestOrPermission;
  70. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  71. AgvApi.ContinueTask(agv.AgvID, agv.Station);
  72. break;
  73. }
  74. case AGVTaskStatus.LeaveGet when agv.Status != AGVTaskStatus.LeaveGet:
  75. var devinfo = new Device<IStation520, IStation521>(Device.All.First(x => x.Code == agv.Position), World);
  76. devinfo.Data.CmdType = StationCmd.Res3;
  77. break;
  78. //完成任务
  79. case AGVTaskStatus.MissionCompleted when agv.Status != AGVTaskStatus.MissionCompleted:
  80. {
  81. if (agv.TaskType is AGVTaskType.CallForMaterial or AGVTaskType.ForkliftFilling or AGVTaskType.CallMaterial)
  82. {
  83. var taskInfo = db.Default.Queryable<WCS_TaskInfo>().First(v => v.AgvTaskID == agv.ID);
  84. if (taskInfo == null) throw new Exception($"未找到AGV任务{agv.ID}对应WCS任务");
  85. //更新AGV任务状态
  86. agv.Status = AGVTaskStatus.MissionCompleted;
  87. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  88. //更新WCS任务状态
  89. taskInfo.Status = Entity.TaskStatus.Finish;
  90. taskInfo.EedTime = DateTime.Now;
  91. db.Default.Updateable(taskInfo).ExecuteCommand();
  92. taskInfo.AddWCS_TASK_DTL(db.Default, "agv", "任务完成");
  93. taskInfos.Add(taskInfo);
  94. }
  95. else
  96. {
  97. agv.Status = AGVTaskStatus.MissionCompleted;
  98. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  99. }
  100. break;
  101. }
  102. }
  103. });
  104. }
  105. catch (Exception ex)
  106. {
  107. World.Log(ex.Message, LogLevelEnum.Mid);
  108. this.ExRecord(obj.Entity.Code, ex.Message);
  109. }
  110. }
  111. foreach (var taskInfo in taskInfos.Where(taskInfo => taskInfo.Status == TaskStatus.Finish))
  112. {
  113. WmsApi.CompleteTask(taskInfo.ID);
  114. }
  115. }
  116. }
  117. else if (obj.Entity.Code == "入库AGV")
  118. {
  119. List<WCS_AgvTaskInfo> agvTaskInfos = new List<WCS_AgvTaskInfo>();
  120. //获取所有未结束的入库AGV任务
  121. SqlSugarHelper.Do(db =>
  122. {
  123. agvTaskInfos = db.Default.Queryable<WCS_AgvTaskInfo>()
  124. .Where(v => v.Status < AGVTaskStatus.MissionCompleted && v.TaskType == AGVTaskType.EnterDepot)
  125. .SplitTable(tabs => tabs.Take(2)).OrderBy(v => v.EditTime).ToList();
  126. });
  127. if (agvTaskInfos.Any())
  128. {
  129. this.ExRecord(obj.Entity.Code, "可用入库AGV任务列表", agvTaskInfos.Select(v => v.ID).ToList());
  130. foreach (var agv in agvTaskInfos)
  131. {
  132. try
  133. {
  134. SqlSugarHelper.Do(db =>
  135. {
  136. #region 开始跟据AGV状态做出处理
  137. switch (agv.AgvStatus)
  138. {
  139. case AGVTaskStatus.NewBuild when agv.Status == AGVTaskStatus.NewBuild:
  140. AgvApi.托盘回库(agv.Position, agv.ID.ToString());
  141. agv.AgvID = agv.ID.ToString();
  142. agv.Status = AGVTaskStatus.Confirm;
  143. agv.AgvStatus = AGVTaskStatus.Confirm;
  144. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  145. break;
  146. //巷道分配
  147. case AGVTaskStatus.RequestOrPermission1 when agv.Status != AGVTaskStatus.Complete1:
  148. {
  149. agv.Status = AGVTaskStatus.Complete1;
  150. agv.Position = "2501";
  151. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  152. //task.Status = TaskStatus.Finish;
  153. //db.Default.Updateable(task).ExecuteCommand();
  154. //task.AddWCS_TASK_DTL(db.Default, "AGV搬运任务分配放货点", agv.Position, $"任务分配至:{agv.Position}");
  155. //调继续执行任务接口
  156. AgvApi.ContinueTask(agv.AgvID, agv.Position);
  157. //}
  158. break;
  159. }
  160. //取货站点安全交互
  161. case AGVTaskStatus.RequestOrPermission2 when agv.Status != AGVTaskStatus.RequestOrPermission2:
  162. {
  163. if (agv.Position.IsNullOrEmpty()) throw new Exception($"无有效放货地址");
  164. var dev = Device.All.First(x => x.Code == agv.Position);
  165. //调继续执行任务接口
  166. AgvApi.ContinueTask(agv.AgvID, dev.Code);
  167. break;
  168. }
  169. case AGVTaskStatus.LeaveGet when agv.Status != AGVTaskStatus.LeaveGet:
  170. var devinfo = new Device<IStation520, IStation521>(Device.All.First(x => x.Code == agv.Position), World);
  171. devinfo.Data.CmdType = StationCmd.Res3;
  172. break;
  173. //完成任务
  174. case AGVTaskStatus.MissionCompleted when agv.Status != AGVTaskStatus.MissionCompleted:
  175. if (agv.TaskType == AGVTaskType.EnterDepot)
  176. {
  177. agv.Status = AGVTaskStatus.MissionCompleted;
  178. db.Default.Updateable(agv).SplitTable().ExecuteCommand();
  179. }
  180. break;
  181. }
  182. #endregion 开始跟据AGV状态做出处理
  183. });
  184. }
  185. catch (Exception ex)
  186. {
  187. World.Log(ex.Message, LogLevelEnum.Mid);
  188. this.ExRecord(obj.Entity.Code, ex.Message);
  189. }
  190. }
  191. }
  192. }
  193. }
  194. public override bool Select(Device dev)
  195. {
  196. return dev.Code is "出库AGV" or "入库AGV";
  197. }
  198. }
  199. }