TaskExtension.cs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. using DBHelper;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol;
  8. using WCS.Service.Helpers;
  9. using WCS.Service.Log;
  10. namespace WCS.Service.Extensions
  11. {
  12. /// <summary>
  13. /// 任务扩展
  14. /// </summary>
  15. public static class TaskExtension
  16. {
  17. public static T Create<T>(this WCS_TASK source)
  18. {
  19. return (T)Activator.CreateInstance(typeof(T), source);
  20. }
  21. /// <summary>
  22. /// 获取可用的出库任务
  23. /// </summary>
  24. /// <param name="tasks"></param>
  25. /// <returns></returns>
  26. public static Task[] GetOutTask(this List<WCS_TASK> tasks)
  27. {
  28. var tasklist = tasks.Select(v => v.Create<Task>());
  29. var task = tasklist.FirstOrDefault() ?? throw new WarnException("无可用出库任务--GetOutTask");
  30. //AGV任务ID不为零表示为车间叫料任务
  31. if (task.AgvTask != 0)
  32. {
  33. //按照AGV任务ID分一次组
  34. tasklist = tasklist.OrderByDescending(v => v.Priority)
  35. .ThenBy(v => v.CREATETIME)
  36. .GroupBy(v => v.AgvTask)
  37. .FirstOrDefault() ?? throw new WarnException("无可用叫料任务--GetOutTask");
  38. //无论这个AGV任务绑定的货物相隔多远都必须要一起出出去
  39. return tasklist.OrderBy(v => v.Col).ToArray();
  40. }
  41. return tasklist.OrderByDescending(v => v.Priority)
  42. .GroupBy(v => v.MaterialCode)
  43. .OrderBy(v => v.Key).FirstOrDefault()
  44. .OrderByDescending(v => v.Priority)
  45. .ThenBy(v => v.Line)
  46. .ThenBy(v => v.Layer)
  47. .ThenBy(v => v.Col)
  48. .Take(2)
  49. .DistinctBy(v => v.Col)
  50. .OrderBy(v => v.Col)
  51. .ToArray();
  52. }
  53. /// <summary>
  54. /// 获取出库任务的站台号及下一个地址
  55. /// </summary>
  56. /// <param name="task">任务</param>
  57. /// <param name="srmFork">货叉</param>
  58. public static void GetSrmStationAndaddNext(this WCS_TASK task, SrmFork srmFork)
  59. {
  60. //TODO:确定在倒库任务重是否能生效
  61. //取任务巷道到达目标地址的下一个地址,即任务堆垛机的站台对应的设备组
  62. var stations = Device.Where(v => v.DEVICEGROUP.Any(p => p.MEMBER == Device.Find(task.TUNNEL).GetPath(task.ADDRTO.Replace("G", ""))))
  63. .Select(v => v.Create<StationDeviceGroup>())
  64. .FirstOrDefault()?.Items
  65. .OrderByDescending(v => v.Entity.CODE)
  66. .ToArray();
  67. var addNext = task.TYPE == TaskType.倒库 ? task.DEVICEDL : task.ADDRTO;
  68. //一工位放较大的站台号
  69. switch (srmFork)
  70. {
  71. case SrmFork.货叉1:
  72. task.SRMSTATION = stations?[0].Entity.CODE;
  73. task.ADDRNEXT = stations?[0].Entity.GetPath(addNext).CODE;
  74. break;
  75. case SrmFork.货叉2:
  76. task.SRMSTATION = stations?[1].Entity.CODE;
  77. task.ADDRNEXT = stations?[1].Entity.GetPath(addNext).CODE;
  78. break;
  79. }
  80. }
  81. /// <summary>
  82. /// 有效任务数是否符合任务组任务数
  83. /// </summary>
  84. /// <param name="tasks"></param>
  85. /// <param name="executable"></param>
  86. /// <param name="db"></param>
  87. public static void ValidTaskCheck(this List<WCS_TASK> tasks, int executable, DB db)
  88. {
  89. var task = tasks.FirstOrDefault();
  90. var taskCount = db.Default.Set<WCS_TASK>().Count(v => v.TaskGroupKey == task.TaskGroupKey && v.TYPE == TaskType.入库);
  91. //开始检查任务数是否匹配
  92. if (executable != taskCount) throw new WarnException($"可执行数{executable},任务组任务数{taskCount},数量不匹配,{task.ID}-{task.TaskGroupKey}");
  93. }
  94. /// <summary>
  95. /// 有效任务数是否符合任务组任务数 临时
  96. /// </summary>
  97. /// <param name="tasks"></param>
  98. /// <param name="executable"></param>
  99. /// <param name="db"></param>
  100. public static List<WCS_TASK> ValidTaskCheck(this List<FinishTaskList<string>> devs, DB db)
  101. {
  102. var taskIds = devs.Select(v => v.Station.Data2.Tasknum).ToList();
  103. var taskList = db.Default.Set<WCS_TASK>().Where(v => taskIds.Contains(v.ID)).ToList();
  104. var task = taskList.FirstOrDefault() ?? throw new WarnException($"ValidTaskCheck 无任务"); ;
  105. var taskCount = db.Default.Set<WCS_TASK>().Count(v => v.TaskGroupKey == task.TaskGroupKey && v.TYPE == TaskType.入库);
  106. //开始检查任务数是否匹配
  107. if (devs.Count != taskCount) throw new WarnException($"可执行数{devs.Count},任务组任务数{taskCount},数量不匹配,{task.ID}-{task.TaskGroupKey}");
  108. return taskList;
  109. }
  110. /// <summary>
  111. /// 用于任务创建时获取放货站台
  112. /// </summary>
  113. /// <param name="task"></param>
  114. public static void TaskGetSrmStation(this WCS_TASK task)
  115. {
  116. task.GetSrmStationAndaddNext(SrmFork.货叉1);
  117. task.SRMSTATION = Device.Where(v => v.IsDevGroup()).FirstOrDefault(v => v.DEVICEGROUP.Any(b => b.MEMBER.CODE == task.SRMSTATION))?.CODE;
  118. task.ADDRNEXT = string.Empty;
  119. }
  120. public static void AGVStatusChange(this WCS_AGVTask task, AGVTaskStatus status, string type = "同步")
  121. {
  122. InfoLog.INFO_AGV($"AGV状态更新Status:{task.Status},AGVStatus:{status},{type}");
  123. }
  124. }
  125. public enum SrmIndex
  126. {
  127. 工位一 = 0,
  128. 工位二 = 1,
  129. }
  130. public class Task : WCS_TASK
  131. {
  132. /// <summary>
  133. /// 行
  134. /// </summary>
  135. public short Line { get; set; }
  136. /// <summary>
  137. /// 列
  138. /// </summary>
  139. public short Col { get; set; }
  140. /// <summary>
  141. /// 层
  142. /// </summary>
  143. public short Layer { get; set; }
  144. public Task(WCS_TASK task)
  145. {
  146. var addrFrom = task.ADDRFROM.Split("-");
  147. ADDRTO = task.ADDRTO;
  148. TUNNEL = task.TUNNEL;
  149. DEVICE = task.DEVICE;
  150. Priority = task.Priority;
  151. AgvTask = task.AgvTask;
  152. CREATETIME = task.CREATETIME;
  153. MaterialCode = task.MaterialCode;
  154. SRMSTATION = task.SRMSTATION;
  155. ID = task.ID;
  156. Line = addrFrom[0].ToShort();
  157. Col = addrFrom[1].ToShort();
  158. Layer = addrFrom[2].ToShort();
  159. }
  160. }
  161. }