AgvSystems.cs 11 KB

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