NoInteractionSystems.cs 49 KB

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