AgvSystems.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214
  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.SRM;
  8. using WCS.Entity.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(x=>x.Take(2)).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(x=>x.Take(2)).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. agv.Status = AGVTaskStatus.LeaveGet;
  78. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  79. break;
  80. //完成任务
  81. case AGVTaskStatus.MissionCompleted when agv.Status != AGVTaskStatus.MissionCompleted:
  82. {
  83. if (agv.TaskType is AGVTaskType.CallForMaterial or AGVTaskType.ForkliftFilling or AGVTaskType.CallMaterial)
  84. {
  85. var taskInfo = db.Default.Queryable<WCS_TaskInfo>().First(v => v.AgvTaskID == agv.ID);
  86. if (taskInfo == null) throw new Exception($"未找到AGV任务{agv.ID}对应WCS任务");
  87. //更新AGV任务状态
  88. agv.Status = AGVTaskStatus.MissionCompleted;
  89. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  90. //更新WCS任务状态
  91. taskInfo.Status = Entity.TaskStatus.Finish;
  92. taskInfo.EndTime = DateTime.Now;
  93. db.Default.Updateable(taskInfo).ExecuteCommand();
  94. taskInfo.AddWCS_TASK_DTL(db.Default, "agv", "任务完成");
  95. taskInfos.Add(taskInfo);
  96. }
  97. else
  98. {
  99. agv.Status = AGVTaskStatus.MissionCompleted;
  100. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  101. }
  102. break;
  103. }
  104. }
  105. });
  106. }
  107. catch (Exception ex)
  108. {
  109. World.Log(ex.Message, LogLevelEnum.Mid);
  110. this.ExRecord(obj.Entity.Code, ex.Message);
  111. }
  112. }
  113. foreach (var taskInfo in taskInfos.Where(taskInfo => taskInfo.Status == TaskStatus.Finish))
  114. {
  115. WmsApi.CompleteTask(taskInfo.ID);
  116. }
  117. }
  118. }
  119. else if (obj.Entity.Code == "入库AGV")
  120. {
  121. List<WCS_AgvTaskInfo> agvTaskInfos = new List<WCS_AgvTaskInfo>();
  122. //获取所有未结束的入库AGV任务
  123. SqlSugarHelper.Do(db =>
  124. {
  125. agvTaskInfos = db.Default.Queryable<WCS_AgvTaskInfo>()
  126. .Where(v => v.Status < AGVTaskStatus.MissionCompleted && v.TaskType == AGVTaskType.EnterDepot)
  127. .SplitTable(tabs => tabs.Take(2)).OrderBy(v => v.EditTime).ToList();
  128. });
  129. if (agvTaskInfos.Any())
  130. {
  131. this.ExRecord(obj.Entity.Code, "可用入库AGV任务列表", agvTaskInfos.Select(v => v.ID).ToList());
  132. foreach (var agv in agvTaskInfos)
  133. {
  134. try
  135. {
  136. SqlSugarHelper.Do(db =>
  137. {
  138. #region 开始跟据AGV状态做出处理
  139. switch (agv.AgvStatus)
  140. {
  141. case AGVTaskStatus.NewBuild when agv.Status == AGVTaskStatus.NewBuild:
  142. AgvApi.托盘回库(agv.Position, agv.ID.ToString());
  143. agv.AgvID = agv.ID.ToString();
  144. agv.Status = AGVTaskStatus.Confirm;
  145. agv.AgvStatus = AGVTaskStatus.Confirm;
  146. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  147. break;
  148. //巷道分配
  149. case AGVTaskStatus.RequestOrPermission1 when agv.Status != AGVTaskStatus.Complete1:
  150. {
  151. agv.Status = AGVTaskStatus.Complete1;
  152. agv.Position = "2501";
  153. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  154. //task.Status = TaskStatus.Finish;
  155. //db.Default.Updateable(task).ExecuteCommand();
  156. //task.AddWCS_TASK_DTL(db.Default, "AGV搬运任务分配放货点", agv.Position, $"任务分配至:{agv.Position}");
  157. //调继续执行任务接口
  158. AgvApi.ContinueTask(agv.AgvID, agv.Position);
  159. //}
  160. break;
  161. }
  162. //取货站点安全交互
  163. case AGVTaskStatus.RequestOrPermission2 when agv.Status != AGVTaskStatus.RequestOrPermission2:
  164. {
  165. if (agv.Position.IsNullOrEmpty()) throw new Exception($"无有效放货地址");
  166. var dev = Device.All.First(x => x.Code == agv.Position);
  167. //调继续执行任务接口
  168. AgvApi.ContinueTask(agv.AgvID, dev.Code);
  169. break;
  170. }
  171. case AGVTaskStatus.LeaveGet when agv.Status != AGVTaskStatus.LeaveGet:
  172. var devinfo = new Device<IStation520, IStation521>(Device.All.First(x => x.Code == agv.Position), World);
  173. devinfo.Data.CmdType = StationCmd.Res3;
  174. break;
  175. //完成任务
  176. case AGVTaskStatus.MissionCompleted when agv.Status != AGVTaskStatus.MissionCompleted:
  177. if (agv.TaskType == AGVTaskType.EnterDepot)
  178. {
  179. agv.Status = AGVTaskStatus.MissionCompleted;
  180. db.Default.Updateable(agv).SplitTable(x=>x.Take(2)).ExecuteCommand();
  181. }
  182. break;
  183. }
  184. #endregion 开始跟据AGV状态做出处理
  185. });
  186. }
  187. catch (Exception ex)
  188. {
  189. World.Log(ex.Message, LogLevelEnum.Mid);
  190. this.ExRecord(obj.Entity.Code, ex.Message);
  191. }
  192. }
  193. }
  194. }
  195. }
  196. public override bool Select(Device dev)
  197. {
  198. return dev.Code is "出库AGV" or "入库AGV";
  199. }
  200. }
  201. }