CTaskEventBase.cs 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. using DbCommon.BusinessCore.BaseCore;
  2. using Pms.DataLibrary.Models;
  3. using Pms.DataLibrary.Order;
  4. using Pms.Models;
  5. using ProjectManagementSystem.Common.Config;
  6. using ProjectManagementSystem.Common.Extenions;
  7. using ProjectManagementSystem.Common.Logger;
  8. using ProjectManagementSystem.Common.WebApi;
  9. using System;
  10. using System.Collections.Generic;
  11. using System.Linq;
  12. using System.Text;
  13. using System.Threading;
  14. using System.Threading.Tasks;
  15. namespace ProjectManagementSystem.Common.Core
  16. {
  17. public abstract class CTaskEventBase
  18. {
  19. protected LocationPropertyManager m_locationManager = new LocationPropertyManager();
  20. protected MaterialBindingManager materialBindingManager = new MaterialBindingManager();
  21. private Dictionary<string, TaskData> TaskDataDict { get; set; } = new Dictionary<string, TaskData>();
  22. private List<ICustomProcess> taskProcesses = new List<ICustomProcess>();
  23. private Dictionary<int, ResultAgvData> agvDataDict = new Dictionary<int, ResultAgvData>();
  24. private StringBuilder stringBuilder = new StringBuilder();
  25. public CTaskEventBase()
  26. {
  27. var assemblyName = this.GetType().Assembly.GetName().Name;
  28. var nameSpace = this.GetType().Namespace;
  29. var processArray = Common.Function.AppSetting.TryGetValue<string>("TaskProcess").ToValueArray<string>();
  30. foreach (var item in processArray)
  31. {
  32. string fullClassName = $"{nameSpace}.{item}";
  33. var processObj = Common.Function.InstanceConstructor.GetInstance<ICustomProcess>(assemblyName, fullClassName);
  34. if (processObj != null)
  35. {
  36. taskProcesses.Add(processObj);
  37. }
  38. else
  39. {
  40. CLog.Instance.SystemLog.WriteInfo($"创建对象失败 {assemblyName} {fullClassName}");
  41. }
  42. }
  43. Task.Factory.StartNew(TaskThread);
  44. Task.Factory.StartNew(TaskThread2);
  45. }
  46. private void TaskEventProc()
  47. {
  48. DateTime start = DateTime.Now;
  49. var taskDataList = PmsApi.GetTaskDataList();
  50. if (taskDataList == null) return;
  51. var agvDataList = PmsApi.GetAllCarrier();
  52. if (agvDataList == null) return;
  53. var taskApiTimeSpan = (DateTime.Now - start).TotalMilliseconds;
  54. DateTime start2 = DateTime.Now;
  55. AgvDataProc(taskDataList, agvDataList);
  56. TaskStateProc(taskDataList, agvDataList);
  57. TaskDynamicRely(taskDataList);
  58. TaskCustomProc(taskDataList);
  59. var operatorTimeSpan = (DateTime.Now - start2).TotalMilliseconds;
  60. //DateTime start3 = DateTime.Now;
  61. //stringBuilder.Clear();
  62. //for (int i = 0; i < taskProcesses.Count; i++)
  63. //{
  64. // DateTime temp = DateTime.Now;
  65. // taskProcesses[i].CustomProcess(taskDataList);
  66. // var tempTimeSpan = (DateTime.Now - temp).TotalMilliseconds;
  67. // stringBuilder.Append($"{tempTimeSpan:F1}ms ");
  68. //}
  69. //var customProcessTimeSpan = (DateTime.Now - start3).TotalMilliseconds;
  70. CLog.Instance.TaskLog.WriteDebug($"{taskApiTimeSpan:F1}ms {operatorTimeSpan:F1}ms");
  71. }
  72. private void TaskEventProc2()
  73. {
  74. DateTime start = DateTime.Now;
  75. var taskDataList = PmsApi.GetTaskDataList();
  76. if (taskDataList == null) return;
  77. var taskApiTimeSpan = (DateTime.Now - start).TotalMilliseconds;
  78. //DateTime start2 = DateTime.Now;
  79. //AgvDataProc(taskDataList, agvDataList);
  80. //TaskStateProc(taskDataList, agvDataList);
  81. //TaskDynamicRely(taskDataList);
  82. //TaskCustomProc(taskDataList);
  83. //var operatorTimeSpan = (DateTime.Now - start2).TotalMilliseconds;
  84. DateTime start3 = DateTime.Now;
  85. stringBuilder.Clear();
  86. for (int i = 0; i < taskProcesses.Count; i++)
  87. {
  88. DateTime temp = DateTime.Now;
  89. taskProcesses[i].CustomProcess(taskDataList);
  90. var tempTimeSpan = (DateTime.Now - temp).TotalMilliseconds;
  91. stringBuilder.Append($"{tempTimeSpan:F1}ms ");
  92. }
  93. var customProcessTimeSpan = (DateTime.Now - start3).TotalMilliseconds;
  94. CLog.Instance.TaskLog.WriteDebug($"{taskApiTimeSpan:F1}ms {customProcessTimeSpan:F1}ms 【{stringBuilder.ToString()}】");
  95. }
  96. private void AgvDataProc(List<TaskData> taskDataList, List<ResultAgvData> agvDataList)
  97. {
  98. //AGV数据的记录
  99. for (int i = 0; i < agvDataList.Count; i++)
  100. {
  101. var agvData = agvDataList[i];
  102. bool record = false;
  103. if (!agvDataDict.ContainsKey(agvData.AgvID))
  104. {
  105. agvDataDict.Add(agvData.AgvID, agvData);
  106. record = true;
  107. }
  108. else
  109. {
  110. var lastAgvData = agvDataDict[agvData.AgvID];
  111. if (agvData.Online != lastAgvData.Online
  112. || agvData.GraphEdge != lastAgvData.GraphEdge
  113. || agvData.GraphVertex != lastAgvData.GraphVertex
  114. || agvData.LogicBits != lastAgvData.LogicBits)
  115. {
  116. record = true;
  117. }
  118. agvDataDict[agvData.AgvID] = agvData;
  119. }
  120. if (record)
  121. {
  122. string stepString = taskDataList.FirstOrDefault(d => d.TaskState > ETaskState.None && d.Carrier == agvData.AgvID)?.GetCurrentStepInfo()?.ToSimpleString();
  123. CLog.Instance.GetAgvLog(agvData.AgvID).WriteDebug($"{stepString}【{agvData.ToSimpleString()}】");
  124. }
  125. }
  126. }
  127. private void TaskStateProc(List<TaskData> taskDataList, List<ResultAgvData> agvDataList)
  128. {
  129. // 当前任务更新到缓存
  130. foreach (var taskData in taskDataList)
  131. {
  132. int agvID = taskData.Carrier > 0 ? taskData.Carrier : taskData.BindingAGVNumber;
  133. bool recordStep = false;
  134. if (!TaskDataDict.ContainsKey(taskData.ID))
  135. {
  136. if (taskData.TaskState > ETaskState.None)
  137. {
  138. TaskDataDict.Add(taskData.ID, taskData);
  139. CLog.Instance.GetAgvLog(agvID).WriteDebug($"{taskData.TaskID} 任务状态{taskData.TaskState}(添加记录) {taskData.ID} {taskData.ToSimpleString()}");
  140. recordStep = true;
  141. }
  142. }
  143. else
  144. {
  145. var lastTaskInfo = TaskDataDict[taskData.ID];
  146. if (lastTaskInfo.TaskState != taskData.TaskState)
  147. {
  148. //CLog.Instance.GetAgvLog(agvID).WriteDebug($"{taskData.TaskID} 运行状态改变 {lastTaskInfo.TaskState} => {taskData.TaskState}");
  149. TaskStateChanged(taskData, lastTaskInfo);
  150. }
  151. var lastStepInfo = lastTaskInfo.GetCurrentStepInfo();
  152. var stepInfo = taskData.GetCurrentStepInfo();
  153. if (stepInfo != null
  154. && (lastStepInfo?.TemplateStepID != stepInfo.TemplateStepID
  155. || lastStepInfo?.SystemStepState != stepInfo.SystemStepState))
  156. {
  157. recordStep = true;
  158. }
  159. TaskDataDict[taskData.ID] = taskData;
  160. }
  161. if (recordStep)
  162. {
  163. string stepString = taskData.GetCurrentStepInfo()?.ToSimpleString();
  164. var agvData = agvDataDict.ContainsKey(taskData.Carrier) ? agvDataDict[taskData.Carrier] : null;
  165. string agvString = agvData?.ToSimpleString();
  166. if (!string.IsNullOrEmpty(stepString)
  167. ||!string.IsNullOrEmpty(agvString))
  168. {
  169. CLog.Instance.GetAgvLog(agvID).WriteDebug($"{stepString}【{agvString}】");
  170. }
  171. }
  172. }
  173. //获取历史任务对比
  174. if (TaskDataDict.Count == 0) { return; }
  175. string[] taskIdArray = TaskDataDict.Keys.ToArray();
  176. foreach (string id in taskIdArray)
  177. {
  178. if (taskDataList.Exists(d => d.ID == id)) continue;
  179. var taskData = PmsApi.GetHistoryTaskData(TaskDataDict[id].TaskID);
  180. if (taskData != null
  181. && taskData.ID == id
  182. && TaskDataDict.ContainsKey(id))
  183. {
  184. CLog.Instance.GetAgvLog(taskData.Carrier).WriteDebug($"{TaskDataDict[id].TaskID} 任务状态{taskData.TaskState}(移除记录) {id}");
  185. TaskDataDict.Remove(id);
  186. switch (taskData.TaskState)
  187. {
  188. case ETaskState.Finished:
  189. TaskFinished(taskData);
  190. break;
  191. case ETaskState.Canceled:
  192. TaskCanceled(taskData);
  193. break;
  194. case ETaskState.TerminateTask:
  195. TaskTerminated(taskData);
  196. break;
  197. case ETaskState.Exceptioned:
  198. TaskExceptioned(taskData);
  199. break;
  200. case ETaskState.RecoveryException:
  201. break;
  202. default:
  203. TaskBecomeHistory(taskData);
  204. break;
  205. }
  206. }
  207. }
  208. }
  209. /// <summary>
  210. /// 动态自动续接任务(通过Excel配置的任务依赖)
  211. /// </summary>
  212. private void TaskDynamicRely(List<TaskData> taskDataList)
  213. {
  214. if (ExcelConfig.Instance.TaskRelyList.Count == 0) return;
  215. var taskIds = taskDataList.Where(d => d.TaskState == ETaskState.Add)
  216. .Select(d => d.TaskID)
  217. .ToArray();
  218. for (int i = 0; i < taskIds.Length; i++)
  219. {
  220. string taskId = taskIds[i];
  221. //重新获取当前所有任务
  222. var taskDataListTemp = PmsApi.GetTaskDataList();
  223. if (taskDataListTemp == null) continue;
  224. var taskData = taskDataListTemp.FirstOrDefault(d => d.TaskID == taskId);
  225. if (taskData == null) continue;
  226. string parentId = GetParentTaskId(taskData, taskDataListTemp, out string currentWhouse, out string relayLastWhouse, out int carrier);
  227. if (parentId != taskData.ParentTaskID)
  228. {
  229. bool result = PmsApi.TaskChangeRely(taskData.TaskID, parentId);
  230. CLog.Instance.GetAgvLog(taskData.Carrier).WriteInfo($"{taskData.TaskID} 更改父任务{result.ToChineseString()} {taskData.ParentTaskID} => {parentId}" +
  231. (string.IsNullOrEmpty(parentId) ? string.Empty : $"({carrier}# {relayLastWhouse} 前往续接 {currentWhouse})"));
  232. }
  233. }
  234. }
  235. private string GetParentTaskId(TaskData taskData, List<TaskData> taskDataList, out string currentWhouse, out string relayLastWhouse, out int carrier)
  236. {
  237. currentWhouse = null;
  238. relayLastWhouse = null;
  239. carrier = 0;
  240. //从配置表中获取当前任务的第一个点可以续接哪些库位
  241. string firstPos = taskData.StepList.OrderBy(d => d.StepID).FirstOrDefault()?.WareHouseID;
  242. string firstWhouse = m_locationManager.QueryWareHouse(firstPos);
  243. var taskRely = Common.Config.ExcelConfig.Instance.TaskRelyList.FirstOrDefault(f => f.StartWareHouse == firstWhouse);
  244. if (taskRely == null) return string.Empty;
  245. //获取已经被续接的任务号
  246. var parentTaskList = taskDataList.Where(d => !string.IsNullOrEmpty(d.ParentTaskID)).ToList();
  247. var task = taskDataList
  248. .Where(d =>
  249. {
  250. //已经续接过的任务不能用
  251. var parentTask = parentTaskList.FirstOrDefault(f => f.ParentTaskID == d.TaskID);
  252. if (parentTask != null)
  253. {
  254. if (parentTask.TaskID != taskData.TaskID)
  255. {
  256. return false;
  257. }
  258. }
  259. //查找已开始任务
  260. if (d.TaskState <= ETaskState.Add) return false;
  261. //AGV类型需要一致
  262. if (d.CarrierType != taskData.CarrierType) return false;
  263. ////临近任务结束
  264. //if (d.StepList.Count - d.CurrentStepID > 3) return false;
  265. string lastPos = d.StepList.OrderByDescending(f => f.StepID).FirstOrDefault()?.WareHouseID;
  266. string lastWhouse = m_locationManager.QueryWareHouse(lastPos);
  267. d.reserved2 = lastWhouse;
  268. return taskRely.EndWareHouse.Contains(lastWhouse);
  269. })
  270. .OrderBy(d =>
  271. {
  272. return d.StepList.Count - d.CurrentStepID;//依据任务离最后一步的步骤数量进行排序
  273. })
  274. .ThenBy(d => d.StartTime)
  275. .FirstOrDefault();
  276. if (task != null)
  277. {
  278. currentWhouse = firstWhouse;
  279. relayLastWhouse = task.reserved2;
  280. carrier = task.Carrier;
  281. }
  282. return (task == null) ? string.Empty : task.TaskID;
  283. }
  284. public void TaskThread()
  285. {
  286. CLog.Instance.SystemLog.WriteDebug($"CTaskEvent已启动");
  287. while (true)
  288. {
  289. try
  290. {
  291. TaskEventProc();
  292. }
  293. catch (Exception ex)
  294. {
  295. CLog.Instance.TaskLog.WriteException("TaskEvent", ex);
  296. Thread.Sleep(5000);
  297. }
  298. Thread.Sleep(1000);
  299. }
  300. }
  301. public void TaskThread2()
  302. {
  303. CLog.Instance.SystemLog.WriteDebug($"CTaskEvent2已启动");
  304. while (true)
  305. {
  306. try
  307. {
  308. TaskEventProc2();
  309. }
  310. catch (Exception ex)
  311. {
  312. CLog.Instance.TaskLog.WriteException("TaskEvent2", ex);
  313. Thread.Sleep(5000);
  314. }
  315. Thread.Sleep(1000);
  316. }
  317. }
  318. public abstract void TaskCustomProc(List<TaskData> taskDataList);
  319. public abstract void TaskStateChanged(TaskData currentTaskInfo, TaskData lastTaskInfo);
  320. public abstract void TaskFinished(TaskData taskInfo);
  321. public abstract void TaskExceptioned(TaskData taskInfo);
  322. public abstract void TaskCanceled(TaskData taskInfo);
  323. public abstract void TaskTerminated(TaskData taskInfo);
  324. public abstract void TaskBecomeHistory(TaskData taskInfo);
  325. }
  326. public interface ICustomProcess
  327. {
  328. void CustomProcess(List<TaskData> taskDataList);
  329. }
  330. }