NoInteractionSystems.cs 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815
  1. using Newtonsoft.Json;
  2. using PlcSiemens.Core.Extension;
  3. using ServiceCenter.Extensions;
  4. using ServiceCenter.Logs;
  5. using ServiceCenter.SqlSugars;
  6. using SqlSugar;
  7. using System.ComponentModel;
  8. using WCS.Core;
  9. using WCS.Entity;
  10. using WCS.Entity.Protocol.Station;
  11. using WCS.WorkEngineering.Extensions;
  12. using WCS.WorkEngineering.Model.WMS;
  13. using WCS.WorkEngineering.WebApi.Controllers;
  14. using WCS.WorkEngineering.WebApi.Models.WMS.Request;
  15. using WCS.WorkEngineering.Worlds;
  16. using DeviceFlags = WCS.WorkEngineering.Extensions.DeviceFlags;
  17. using TaskStatus = WCS.Entity.TaskStatus;
  18. namespace WCS.WorkEngineering.Systems
  19. {
  20. /// <summary>
  21. /// 无交互系统
  22. /// </summary>
  23. [BelongTo(typeof(NoInteractionWorld))]
  24. [Description("无交互系统")]
  25. public class NoInteractionSystems : DeviceSystem<Device<IStation520>>
  26. {
  27. protected override bool ParallelDo => true;
  28. public override void Do(Device<IStation520> obj)
  29. {
  30. var taskInfos = new List<int>();
  31. var sqlSugar = new SqlSugarHelper();
  32. SqlSugarHelper.Do(db =>
  33. {
  34. //获取所有的新建任务,组盘任务不需要
  35. taskInfos = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status == 0).ToList()
  36. .Where(x => x.Type != TaskType.SetPlate || (x.Type == TaskType.SetPlate && x.AddrFrom != "Robot") && x.BusType!="芯股用空托盘入库" && !x.BarCode.Contains("Error")).Select(x => x.ID).ToList();
  37. var time = DateTime.Now.AddMinutes(-20);
  38. var time1 = DateTime.Now.AddMinutes(-40);
  39. var timeOut = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "车间叫料" && x.Status < TaskStatus.StackerExecution && x.AddTime < time);
  40. if (timeOut > 0) World.Log($"共有{timeOut}个车间叫料任务超过二十分钟未执行,请检查相关相关任务对应的堆垛机是否正常、堆垛机放货点是否卡任务等", LogLevelEnum.High);
  41. var timeOut1 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "车间叫料" && x.Status < TaskStatus.StackerExecution && x.AddTime < time1);
  42. if (timeOut1 > 0) World.Log($"共有{timeOut1}个车间叫料任务超过四十分钟未执行,请检查相关相关任务对应的堆垛机是否正常、堆垛机放货点是否卡任务等", LogLevelEnum.High);
  43. var timeOut2 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "皮盘入库" && x.Status < TaskStatus.StackerExecution && x.AddTime < time);
  44. if (timeOut2 > 0) World.Log($"共有{timeOut2}个皮盘入库任务超过二十分钟未执行,请检查相关相关任务及对应agv设备", LogLevelEnum.High);
  45. var timeOut3 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Count(x => x.BusType == "皮盘入库" && x.Status < TaskStatus.StackerExecution && x.AddTime < time1);
  46. if (timeOut3 > 0) World.Log($"共有{timeOut3}个皮盘入库任务超过四十分钟未执行,请检查相关相关任务及对应agv设备", LogLevelEnum.High);
  47. var time3 = DateTime.Now.AddHours(-24);
  48. var timeOut4 = db.Default.Queryable<BillInvnow>().NoLock().Count(x => x.AddTime < time3 && x.ContGrpBarCode.Contains("TPB"));
  49. if (timeOut4 > 0) World.Log($"共有{timeOut4}个库存信息超过24小时未执行,请检查相关库存信息并处理", LogLevelEnum.High);
  50. var time4 = DateTime.Now.AddHours(-12);
  51. var timeOut5 = db.Default.Queryable<WCS_Palletizing>().NoLock().Where(x => x.AddTime < time4 && !x.Finish).ToList();
  52. if (timeOut5.Count > 0) World.Log($"共有{timeOut5.Count}个桁架码垛信息超过12小时未结束,请检查相关码垛信息并处理,具体在{JsonConvert.SerializeObject(timeOut5.Select(x=>x.PalletizingStation).ToList())}", LogLevelEnum.High);
  53. });
  54. if (!taskInfos.Any())
  55. {
  56. return;
  57. }
  58. var isEnd = false; //每个周期只处理一个任务
  59. foreach (var item in taskInfos)
  60. {
  61. if (isEnd) return;
  62. try
  63. {
  64. SqlSugarHelper.Do(db =>
  65. {
  66. var task = db.Default.Queryable<WCS_TaskInfo>().RowLock().Where(t => t.ID == item && t.Status== TaskStatus.NewBuild).First();
  67. switch (task.Type)
  68. {
  69. case TaskType.SetPlate:
  70. if (task.AddrFrom != "Robot")
  71. {
  72. //更新任务状态
  73. task.Status = TaskStatus.WaitingToExecute;
  74. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status }).ExecuteCommand();
  75. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  76. isEnd = true;
  77. }
  78. break;
  79. case TaskType.EnterDepot:
  80. if (task.LastInteractionPoint == "2")
  81. {
  82. var sta = GetAgvStation(task, db);
  83. var agv = new WCS_AgvTaskInfo()
  84. {
  85. ID = db.GetAgvTaskId(),
  86. TaskType = AGVTaskType.EnterDepot,
  87. Status = AGVTaskStatus.NewBuild,
  88. TaskId = task.ID,
  89. Position = task.WorkBench,
  90. Station = sta,
  91. AddWho = "WCS",
  92. AddTime = DateTime.Now
  93. };
  94. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  95. task.AgvTaskID = agv.ID;
  96. task.Status = Entity.TaskStatus.WaitingToExecute;
  97. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.AgvTaskID }).ExecuteCommand();
  98. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化单独返皮盘任务");
  99. }
  100. else if (task.BusType == TaskBusType.芯股用空托盘入库.GetDescription()) return;
  101. else if (task.BusType == TaskBusType.人工满托入库.GetDescription())
  102. {
  103. var agv = new WCS_AgvTaskInfo()
  104. {
  105. ID = db.GetAgvTaskId(),
  106. TaskType = AGVTaskType.EnterDepot,
  107. Status = AGVTaskStatus.NewBuild,
  108. TaskId = task.ID,
  109. Position = task.WorkBench,
  110. Station = task.WarehouseCode switch
  111. {
  112. "1N" => "2535",
  113. "1S" => "2735",
  114. "2N" => "2935",
  115. "2S" => "3135",
  116. _ => "2535"
  117. },
  118. AddWho = "WCS",
  119. AddTime = DateTime.Now
  120. };
  121. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  122. task.AgvTaskID = agv.ID;
  123. task.Device = task.WarehouseCode switch
  124. {
  125. "1N" => "TY1",
  126. "1S" => "TY2",
  127. "2N" => "TY3",
  128. "2S" => "TY4",
  129. _ => "TY1"
  130. };
  131. task.Height = 1;
  132. task.Status = Entity.TaskStatus.WaitingToExecute;
  133. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.AgvTaskID, x.Device, x.Height }).ExecuteCommand();
  134. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化人工满托入库任务");
  135. }
  136. else if (task.BusType == TaskBusType.UT人工满托入库.GetDescription())
  137. {
  138. var agv = new WCS_AgvTaskInfo()
  139. {
  140. ID = db.GetAgvTaskId(),
  141. TaskType = AGVTaskType.EnterDepot,
  142. Status = AGVTaskStatus.NewBuild,
  143. TaskId = task.ID,
  144. Position = task.WorkBench,
  145. Station = task.WarehouseCode switch
  146. {
  147. "1N" => "2533",
  148. "1S" => "2733",
  149. "2N" => "2933",
  150. "2S" => "3133",
  151. _ => "2533"
  152. },
  153. AddWho = "WCS",
  154. AddTime = DateTime.Now
  155. };
  156. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  157. task.AgvTaskID = agv.ID;
  158. task.Device = task.WarehouseCode switch
  159. {
  160. "1N" => "TY1",
  161. "1S" => "TY2",
  162. "2N" => "TY3",
  163. "2S" => "TY4",
  164. _ => "TY1"
  165. };
  166. task.Height = 1;
  167. task.Status = Entity.TaskStatus.WaitingToExecute;
  168. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.AgvTaskID, x.Device, x.Height }).ExecuteCommand();
  169. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化UT人工满托入库任务");
  170. }
  171. else if (task.BusType == TaskBusType.重绕满托入库.GetDescription())
  172. {
  173. task.Status = Entity.TaskStatus.WaitingToExecute;
  174. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status }).ExecuteCommand();
  175. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  176. }
  177. else if (task.BusType == TaskBusType.帘线退料重绕.GetDescription())
  178. {
  179. var agv = new WCS_AgvTaskInfo()
  180. {
  181. ID = db.GetAgvTaskId(),
  182. TaskType = AGVTaskType.EnterDepot,
  183. Status = AGVTaskStatus.NewBuild,
  184. TaskId = task.ID,
  185. Position = task.WorkBench,
  186. WorkShop = 222,
  187. Station = "9001",
  188. AddWho = "WCS",
  189. AddTime = DateTime.Now
  190. };
  191. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  192. task.Status = Entity.TaskStatus.WaitingToExecute;
  193. task.AgvTaskID = agv.ID;
  194. task.AddrTo = agv.Station;
  195. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status,x.AgvTaskID,x.AddrTo }).ExecuteCommand();
  196. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  197. }
  198. else
  199. {
  200. //更新任务状态
  201. task.Status = Entity.TaskStatus.WaitingToExecute;
  202. task.Device = task.WarehouseCode switch
  203. {
  204. "1N" => "SRM1",
  205. "1S" => "SRM2",
  206. "2N" => "SRM3",
  207. "2S" => "SRM4",
  208. "3N" => "SRM5",
  209. "3S" => "SRM6",
  210. _ => task.Device
  211. };
  212. task.Device = task.AddrFrom switch
  213. {
  214. "1666" or "1661" => "SRM1",
  215. "1681" or "1676" => "SRM2",
  216. "1696" or "1691" => "SRM3",
  217. "1711" or "1706" => "SRM4",
  218. "1726" or "1721" => "SRM5",
  219. "1741" or "1736" => "SRM6",
  220. _ => task.Device
  221. };
  222. //计算下一个地址
  223. var path1 = DevicePath.GetPath(task.AddrFrom, task.Device);
  224. task.AddrNext = path1.Points[1].Code;
  225. var des1 = task.AddrNext;
  226. var des2 = "1";
  227. //分拣一
  228. if (task.AddrNext == "1604")
  229. {
  230. des2 = "1606";
  231. }
  232. //else if (task.AddrNext == "1614")
  233. //{
  234. // des2 = "1616";
  235. //}
  236. var num1 = db.Default.Queryable<WCS_TaskInfo>().RowLock().Where(t => t.WorkBench == des1 && t.Status < TaskStatus.Finish && t.Type == TaskType.EnterDepot).Count();
  237. var num2 = db.Default.Queryable<WCS_TaskInfo>().RowLock().Where(t => t.WorkBench == des2 && t.Status < TaskStatus.Finish && t.Type == TaskType.EnterDepot).Count();
  238. if (task.AddrNext == "1604")
  239. {
  240. task.AddrNext = num1 <= num2 ? des2 : des1;
  241. }
  242. task.WorkBench = task.AddrNext;
  243. task.SrmStation = task.BarCode.Contains("TPA") || task.BarCode.Contains("TPB") ? task.AddrFrom : path1.Points[2].Code;
  244. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.Device, x.AddrNext, x.SrmStation, x.WorkBench }).ExecuteCommand();
  245. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化入库任务信息");
  246. }
  247. isEnd = true;
  248. break;
  249. case TaskType.OutDepot:
  250. {
  251. if (task.Device.Contains("Robot")) //机器人任务
  252. {
  253. var lastTask = db.Default.Queryable<WCS_TaskInfo>().NoLock().First(x => x.Type == TaskType.OutDepot && x.AddrTo == task.AddrTo && x.Status > TaskStatus.WaitingToExecute && x.Status < TaskStatus.Finish);
  254. if (lastTask != null)
  255. {
  256. World.Log($"等待任务结束:{task.AddrTo}上一组码垛任务未结束,等待{lastTask.ID}任务完成后开始初始化当前任务{task.ID}");
  257. return;
  258. }
  259. var pos = task.AddrFrom.Split("-");
  260. task.Status = Entity.TaskStatus.WaitingToExecute;
  261. task.Line = pos[0].ToShort();
  262. task.Col = pos[1].ToShort();
  263. task.Layer = pos[2].ToShort();
  264. task.Depth = pos[3].ToShort();
  265. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.Status, x.Line, x.Col, x.Layer, x.Depth }).ExecuteCommand();
  266. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,放货站台:{task.SrmStation}");
  267. isEnd = true;
  268. }
  269. else if (task.Device == "FJXG") //芯股仓库的任务
  270. {
  271. if (task.LastInteractionPoint != "3") //车间叫料任务,此值为3时是单独取满任务
  272. {
  273. var sta = GetAgvStation(task, db); //芯股任务,取货站台就是芯股站台
  274. var agv = new WCS_AgvTaskInfo()
  275. {
  276. ID = db.GetAgvTaskId(),
  277. TaskType = AGVTaskType.EnterDepot,
  278. Status = AGVTaskStatus.NewBuild,
  279. TaskId = task.ID,
  280. Position = task.WorkBench,
  281. Station = sta,
  282. AddWho = "WCS",
  283. AddTime = DateTime.Now
  284. };
  285. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  286. }
  287. var agvId = DateTime.Now.GetFormat(GetFormatterEnum.only) + task.ID;
  288. var agvTask = new WCS_AgvTaskInfo()
  289. {
  290. ID = db.GetAgvTaskId(),
  291. TaskType = AGVTaskType.CallMaterial,
  292. Status = AGVTaskStatus.NewBuild,
  293. TaskId = task.ID,
  294. Position = task.WorkBench,
  295. Station = task.AddrFrom,
  296. AddWho = "WCS",
  297. AddTime = DateTime.Now,
  298. AgvID = agvId
  299. };
  300. //开始处理车间叫料AGV任务任务
  301. db.Default.InsertableRowLock(agvTask).SplitTable().ExecuteCommand();
  302. agvTask = db.Default.Queryable<WCS_AgvTaskInfo>().RowLock().SplitTable(x => x.Take(2)).Single(x => x.TaskId == task.ID && x.TaskType == AGVTaskType.CallMaterial);
  303. if (agvTask.Status != AGVTaskStatus.NewBuild)
  304. {
  305. World.Log($"AGV任务{agvTask.ID}状态不是新建", LogLevelEnum.High);
  306. return;
  307. }
  308. agvId = agvTask.AgvID;
  309. agvTask.Status = AGVTaskStatus.Confirm;
  310. agvTask.AgvStatus = AGVTaskStatus.Confirm;
  311. db.Default.UpdateableRowLock(agvTask).UpdateColumns(x => new { x.AgvID, x.Status, x.AgvStatus }).SplitTable(x => x.Take(2)).ExecuteCommand();
  312. task.Status = TaskStatus.AGVExecution;
  313. task.AgvTaskID = agvTask.ID;
  314. task.EditTime = DateTime.Now;
  315. task.EditWho = "WCS";
  316. db.Default.Updateable(task).UpdateColumns(x => new { x.Status, x.AgvTaskID, x.EditTime, x.EditWho }).ExecuteCommand();
  317. task.AddWCS_TASK_DTL(db.Default, task.AddrFrom, "AGV", $"任务下发至AGV{agvId}");
  318. var res = AgvApi.机台叫料(task.AddrFrom, task.WorkBench, agvId);
  319. isEnd = true;
  320. }
  321. else if (task.BusType == TaskBusType.芯股站台送空托.GetDescription())
  322. {
  323. if (task.SrmStation == "1")
  324. {
  325. var srmStation = new string[2];
  326. switch (task.Device)
  327. {
  328. case "SRM1":
  329. srmStation[0] = "2534";
  330. srmStation[1] = "2533";
  331. break;
  332. case "SRM2":
  333. srmStation[0] = "2734";
  334. srmStation[1] = "2733";
  335. break;
  336. case "SRM3":
  337. srmStation[0] = "2934";
  338. srmStation[1] = "2933";
  339. break;
  340. case "SRM4":
  341. srmStation[0] = "3134";
  342. srmStation[1] = "3133";
  343. break;
  344. case "SRM5":
  345. srmStation[0] = "3334";
  346. srmStation[1] = "3333";
  347. break;
  348. case "SRM6":
  349. srmStation[0] = "3534";
  350. srmStation[1] = "3533";
  351. break;
  352. }
  353. //开始计算当前这个任务要从哪个站台出
  354. //计算两个站台小于取货完成数量的AGV任务
  355. var agv1 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[0]).Count();
  356. var agv2 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[1]).Count();
  357. task.SrmStation = "";
  358. //if (srmStation[0] == "2534"/* || srmStation[0] == "2934"*/)
  359. //{
  360. // task.AddrTo = srmStation[0];
  361. //}
  362. //else
  363. {
  364. task.AddrTo = agv1 > agv2 ? srmStation[1] : srmStation[0];
  365. }
  366. }
  367. if (task.SrmStation.IsNullOrEmpty()) //如果没有指定放货站台
  368. {
  369. if (task.Device.IsNullOrEmpty())
  370. {
  371. task.Device = "SRM" + task.Tunnel.GetLastDigit();
  372. }
  373. //获取堆垛机到目标地址的路径信息
  374. var path = DevicePath.GetPath(task.Device, task.AddrTo);
  375. task.SrmStation = path.Points[1].Code;
  376. }
  377. //更新任务状态
  378. task.Status = Entity.TaskStatus.WaitingToExecute;
  379. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.SrmStation, x.AddrTo, x.Status }).ExecuteCommand();
  380. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,放货站台:{task.SrmStation}");
  381. isEnd = true;
  382. }
  383. else if (task.Device == "CR")
  384. {
  385. if (task.LastInteractionPoint != "3") //车间叫料任务,此值为3时是单独取满任务
  386. {
  387. //var sta = GetAgvStation(task, db);
  388. var agv = new WCS_AgvTaskInfo()
  389. {
  390. ID = db.GetAgvTaskId(),
  391. TaskType = AGVTaskType.EnterDepot,
  392. Status = AGVTaskStatus.NewBuild,
  393. TaskId = task.ID,
  394. Position = task.WorkBench,
  395. Station = "2501",//默认去分拣一
  396. AddWho = "WCS",
  397. AddTime = DateTime.Now
  398. };
  399. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  400. }
  401. //更新任务状态
  402. task.Status = Entity.TaskStatus.WaitingToExecute;
  403. task.AddrTo = task.WorkBench;
  404. db.Default.UpdateableRowLock(task).UpdateColumns(x => new {x.Status , x.AddrTo }).ExecuteCommand();
  405. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,重绕组盘区{task.AddrFrom}");
  406. isEnd = true;
  407. break;
  408. }
  409. else //立库出库任务
  410. {
  411. if (task.SrmStation == "1")
  412. {
  413. var srmStation = new string[2];
  414. switch (task.Device)
  415. {
  416. case "SRM1":
  417. srmStation[0] = "2534";
  418. srmStation[1] = "2533";
  419. break;
  420. case "SRM2":
  421. srmStation[0] = "2734";
  422. srmStation[1] = "2733";
  423. break;
  424. case "SRM3":
  425. srmStation[0] = "2934";
  426. srmStation[1] = "2933";
  427. break;
  428. case "SRM4":
  429. srmStation[0] = "3134";
  430. srmStation[1] = "3133";
  431. break;
  432. case "SRM5":
  433. srmStation[0] = "3334";
  434. srmStation[1] = "3333";
  435. break;
  436. case "SRM6":
  437. srmStation[0] = "3534";
  438. srmStation[1] = "3533";
  439. break;
  440. }
  441. //开始计算当前这个任务要从哪个站台出
  442. //计算两个站台小于取货完成数量的AGV任务
  443. var agv1 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[0]).Count();
  444. var agv2 = db.Default.Queryable<WCS_TaskInfo>().NoLock().Where(x => x.Status <= TaskStatus.StackerExecution && x.AddrTo == srmStation[1]).Count();
  445. task.SrmStation = "";
  446. //if (srmStation[0] == "2534"/* || srmStation[0] == "2934"*/)
  447. //{
  448. // task.AddrTo = srmStation[0];
  449. //}
  450. //else
  451. {
  452. task.AddrTo = agv1 > agv2 ? srmStation[1] : srmStation[0];
  453. }
  454. }
  455. if (task.SrmStation.IsNullOrEmpty()) //如果没有指定放货站台
  456. {
  457. if (task.Device.IsNullOrEmpty())
  458. {
  459. task.Device = "SRM" + task.Tunnel.GetLastDigit();
  460. }
  461. //获取堆垛机到目标地址的路径信息
  462. var path = DevicePath.GetPath(task.Device, task.AddrTo);
  463. task.SrmStation = path.Points[1].Code;
  464. }
  465. var devs = Device.All.Where(x => x.HasFlag(DeviceFlags.AGV取货站台口)).Select(x => x.Code);
  466. if (devs.Contains(task.SrmStation) && task.LastInteractionPoint != "3") //车间叫料任务,此值为3时是单独取满任务
  467. {
  468. var sta = GetAgvStation(task, db);
  469. var agv = new WCS_AgvTaskInfo()
  470. {
  471. ID = db.GetAgvTaskId(),
  472. TaskType = AGVTaskType.EnterDepot,
  473. Status = AGVTaskStatus.NewBuild,
  474. TaskId = task.ID,
  475. Position = task.WorkBench,
  476. Station = sta,
  477. AddWho = "WCS",
  478. AddTime = DateTime.Now
  479. };
  480. db.Default.InsertableRowLock(agv).SplitTable().ExecuteCommand();
  481. }
  482. //更新任务状态
  483. task.Status = Entity.TaskStatus.WaitingToExecute;
  484. db.Default.UpdateableRowLock(task).UpdateColumns(x => new { x.SrmStation, x.AddrTo, x.Status }).ExecuteCommand();
  485. task.AddWCS_TASK_DTL(db.Default, task.Device, $"初始化出库任务信息,放货站台:{task.SrmStation}");
  486. isEnd = true;
  487. }
  488. break;
  489. }
  490. case TaskType.Delivery:
  491. break;
  492. case TaskType.EmptyInit:
  493. break;
  494. }
  495. });
  496. }
  497. catch (Exception ex)
  498. {
  499. if (ex.Message.Contains("Index")) World.Log($"{ex.Message}:{ex.StackTrace}");
  500. if (ex.Message.Contains("SqlTransaction")) World.Log($"{ex.Message}:{ex.StackTrace}");
  501. World.Log(ex.Message, LogLevelEnum.Mid);
  502. }
  503. }
  504. }
  505. public override bool Select(Device dev)
  506. {
  507. return dev.Code == nameof(NoInteractionSystems);
  508. }
  509. /// <summary>
  510. /// 获取AGV的目标站台
  511. /// </summary>
  512. /// <param name="task">任务信息</param>
  513. /// <param name="db">数据库操作对象</param>
  514. /// <returns>目标站台</returns>
  515. private string GetAgvStation(WCS_TaskInfo task, SqlSugarHelper db)
  516. {
  517. // 默认站台
  518. var defaultStation = "2701";
  519. var sta = defaultStation;
  520. // 定义仓库与站台的映射关系
  521. var stationMap = new Dictionary<string, (string N, string S)>
  522. {
  523. ["1"] = ("2501", "2701"),
  524. ["2"] = ("2901", "3101"),
  525. ["3"] = ("3301", "3501")
  526. };
  527. var oldWarehouseCode = task.WarehouseCode;
  528. // 检查是否需要更新 warehouseCode
  529. if (task.BusType == TaskBusType.皮盘入库.GetDescription() || task.BusType == TaskBusType.车间叫料.GetDescription())
  530. {
  531. task.WarehouseCode = task.MaterialCode switch
  532. {
  533. "分拣库1北" => "1N",
  534. "分拣库1南" => "1S",
  535. "分拣库2北" => "2N",
  536. "分拣库2南" => "2S",
  537. "分拣库3北" => "3N",
  538. "分拣库3南" => "3S",
  539. _ => task.WarehouseCode
  540. };
  541. }
  542. // 确定站台初始值
  543. if (stationMap.TryGetValue(task.WarehouseCode[0].ToString(), out var stationPair))
  544. sta = task.WarehouseCode.Contains("N") ? stationPair.N : stationPair.S;
  545. // 配置值获取
  546. var num = db.Default.Queryable<fjSysConfig>()
  547. .NoLock()
  548. .First(x => x.Code == "SameSideTaskNum")?.SContent.ToInt() ?? 5;
  549. // 判断当前侧任务数量是否超出限制
  550. var currentTaskCount = GetStationTaskCount(db, sta);
  551. if (currentTaskCount > num)
  552. {
  553. // 同一分拣库的两侧均分任务
  554. if (task.WarehouseCode.StartsWith("1") || task.WarehouseCode.StartsWith("2") ||
  555. task.WarehouseCode.StartsWith("3"))
  556. {
  557. var (northCount, southCount) = GetNorthAndSouthTaskCounts(db, stationPair);
  558. sta = southCount > northCount ? stationPair.N : stationPair.S;
  559. }
  560. else
  561. {
  562. return defaultStation;
  563. }
  564. }
  565. // 配置值获取
  566. num = db.Default.Queryable<fjSysConfig>()
  567. .NoLock()
  568. .First(x => x.Code == "TasksInSameWarehouse")?.SContent.ToInt() ?? 10;
  569. // 判断是否需要在三个分拣中心中选取任务数量较少的一侧进行分配
  570. currentTaskCount = GetStationTaskCount(db, sta);
  571. var otherSide = stationMap.First(x => x.Value.N == sta || x.Value.S == sta);
  572. currentTaskCount += GetStationTaskCount(db, otherSide.Value.N==sta ? otherSide.Value.S : otherSide.Value.N);
  573. if (currentTaskCount > num)
  574. {
  575. sta = db.Default.Queryable<WCS_AgvTaskInfo>()
  576. .NoLock()
  577. .Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 &&
  578. v.TaskType == AGVTaskType.EnterDepot &&
  579. (v.Station != "2535" && v.Station != "2735" && v.Station != "2935" && v.Station != "3135" && !v.Station.StartsWith("9")))
  580. .SplitTable(v => v.Take(2))
  581. .ToList()
  582. .GroupBy(x => x.Station)
  583. .Select(g => new { Station = g.Key, Count = g.Count() })
  584. .OrderBy(x => x.Count)
  585. .FirstOrDefault()?.Station ?? defaultStation;
  586. }
  587. task.WarehouseCode = oldWarehouseCode;
  588. return sta;
  589. }
  590. /// <summary>
  591. /// 获取指定站台的任务数量
  592. /// </summary>
  593. private int GetStationTaskCount(SqlSugarHelper db, string station)
  594. {
  595. return db.Default.Queryable<WCS_AgvTaskInfo>()
  596. .NoLock()
  597. .Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 &&
  598. v.TaskType == AGVTaskType.EnterDepot && v.Station == station)
  599. .SplitTable(v => v.Take(2))
  600. .Count();
  601. }
  602. /// <summary>
  603. /// 获取北侧和南侧的任务数量
  604. /// </summary>
  605. private (int NorthCount, int SouthCount) GetNorthAndSouthTaskCounts(SqlSugarHelper db,
  606. (string N, string S) stationPair)
  607. {
  608. var northCount = GetStationTaskCount(db, stationPair.N);
  609. var southCount = GetStationTaskCount(db, stationPair.S);
  610. return (northCount, southCount);
  611. }
  612. /// <summary>
  613. /// 获取AGV的目标站台
  614. /// </summary>
  615. /// <param name="task"></param>
  616. /// <param name="db"></param>
  617. /// <returns></returns>
  618. private string GetAgvStation1(WCS_TaskInfo task, SqlSugarHelper db)
  619. {
  620. string sta = "";
  621. var n = 0;
  622. var s = 0;
  623. var warehouseCode = task.WarehouseCode;
  624. if (task.BusType == TaskBusType.皮盘入库.GetDescription() || task.BusType == TaskBusType.车间叫料.GetDescription())
  625. {
  626. warehouseCode = task.MaterialCode switch
  627. {
  628. "分拣库1北" => "1N",
  629. "分拣库1南" => "1S",
  630. "分拣库2北" => "2N",
  631. "分拣库2南" => "2S",
  632. "分拣库3北" => "3N",
  633. "分拣库3南" => "3S",
  634. _ => task.WarehouseCode
  635. };
  636. }
  637. #region 判断是否需要在同一分拣库的两侧进行均分
  638. //获取配置值
  639. var num = 5;
  640. var config = db.Default.Queryable<fjSysConfig>().NoLock().First(x => x.Code == "SameSideTaskNum");
  641. if (config != null)
  642. {
  643. num = config.SContent.ToInt();
  644. }
  645. sta = warehouseCode switch
  646. {
  647. "1N" => "2501",
  648. "1S" => "2701",
  649. "2N" => "2901",
  650. "2S" => "3101",
  651. "3N" => "3301",
  652. "3S" => "3501",
  653. _ => "2501",
  654. };
  655. //获取当前侧的已有任务数量
  656. var staNumber = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v =>
  657. v.Status >= AGVTaskStatus.Confirm &&
  658. v.Status < AGVTaskStatus.Complete1 &&
  659. v.TaskType == AGVTaskType.EnterDepot && v.Station == sta)
  660. .SplitTable(v => v.Take(2)).Count();
  661. //当前侧任务数量已大于配置任务数量
  662. if (staNumber > num)
  663. {
  664. //同侧之间均分
  665. if (warehouseCode.Contains("1"))
  666. {
  667. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="2501")
  668. .SplitTable(v => v.Take(2)).Count();
  669. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="2701")
  670. .SplitTable(v => v.Take(2)).Count();
  671. sta = n > s ? "2701" : "2501";
  672. }
  673. else if (warehouseCode.Contains("2"))
  674. {
  675. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="2901")
  676. .SplitTable(v => v.Take(2)).Count();
  677. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="3101")
  678. .SplitTable(v => v.Take(2)).Count();
  679. sta = n > s ? "3101" : "2901";
  680. }
  681. else if (warehouseCode.Contains("3"))
  682. {
  683. n = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="3301")
  684. .SplitTable(v => v.Take(2)).Count();
  685. s = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot && v.Station=="3501")
  686. .SplitTable(v => v.Take(2)).Count();
  687. sta = n > s ? "3501" : "3301";
  688. }
  689. else
  690. {
  691. return default;
  692. }
  693. }
  694. #endregion
  695. #region 判断是否需要在三个分拣中心中选取任务数量较少的一侧进行分配
  696. staNumber = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v =>
  697. v.Status >= AGVTaskStatus.Confirm &&
  698. v.Status < AGVTaskStatus.Complete1 &&
  699. v.TaskType == AGVTaskType.EnterDepot && v.Station == sta)
  700. .SplitTable(v => v.Take(2)).Count();
  701. num = 10;
  702. config = db.Default.Queryable<fjSysConfig>().NoLock().First(x => x.Code == "TasksInSameWarehouse");
  703. if (config != null)
  704. {
  705. num = config.SContent.ToInt();
  706. }
  707. if (staNumber > num)
  708. {
  709. sta = db.Default.Queryable<WCS_AgvTaskInfo>().NoLock().Where(v => v.Status >= AGVTaskStatus.Confirm && v.Status < AGVTaskStatus.Complete1 && v.TaskType == AGVTaskType.EnterDepot
  710. && (v.Station != "2535" && v.Station != "2735" && v.Station != "2935" && v.Station != "3135" && !v.Station.StartsWith("9")))//过滤掉人工满托入库和重绕区任务
  711. .SplitTable(v => v.Take(2)).ToList().GroupBy(x => x.Station).Select(x => new { x.Key, Count = x.Count() }).MinBy(x => x.Count).Key;
  712. }
  713. #endregion
  714. return sta;
  715. }
  716. /// <summary>
  717. /// 获取AGV返空可用目标站台
  718. /// </summary>
  719. /// <param name="warehouseCode">仓库</param>
  720. /// <param name="direction">需要获取的方向</param>
  721. /// <returns>目标站台列表</returns>
  722. public List<string> GetAgvStation(string warehouseCode, string direction)
  723. {
  724. // 定义方向和仓库与站台的映射关系
  725. var stationMap = new Dictionary<string, Dictionary<string, List<string>>>
  726. {
  727. ["N"] = new()
  728. {
  729. ["1"] = new List<string> { "2501", "2505", "2509", "2513" },
  730. ["2"] = new List<string> { "2901", "2905", "2909", "2913" },
  731. ["3"] = new List<string> { "3301", "3305", "3309", "3313" }
  732. },
  733. ["S"] = new()
  734. {
  735. ["1"] = new List<string> { "2701", "2705", "2709", "2713" },
  736. ["2"] = new List<string> { "3101", "3105", "3109", "3113" },
  737. ["3"] = new List<string> { "3501", "3505", "3509", "3513" }
  738. }
  739. };
  740. // 查找方向和仓库对应的站台
  741. if (stationMap.TryGetValue(direction, out var warehouseStations))
  742. {
  743. var warehouseKey = warehouseStations.Keys.FirstOrDefault(warehouseCode.Contains);
  744. if (warehouseKey != null) return warehouseStations[warehouseKey];
  745. }
  746. // 如果没有找到匹配的站台,则返回空列表
  747. return new List<string>();
  748. }
  749. }
  750. }