TaskExtension.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434
  1. using ServiceCenter.Extensions;
  2. using ServiceCenter.Logs;
  3. using ServiceCenter.Redis;
  4. using ServiceCenter.SqlSugars;
  5. using SqlSugar;
  6. using WCS.Core;
  7. using WCS.Entity;
  8. using WCS.WorkEngineering.Model.WMS;
  9. using TaskStatus = WCS.Entity.TaskStatus;
  10. namespace WCS.WorkEngineering.Extensions
  11. {
  12. /// <summary>
  13. /// 任务扩展
  14. /// </summary>
  15. public static class TaskExtension
  16. {
  17. /// <summary>
  18. /// 更新任务执行记录
  19. /// 同步更新历史任务
  20. /// </summary>
  21. /// <param name="task">任务信息</param>
  22. /// <param name="db">数据库上下文</param>
  23. /// <param name="curPoint">当前地址</param>
  24. /// <param name="desc">描述</param>
  25. public static void AddWCS_TASK_DTL(this WCS_TaskInfo task, SqlSugarProvider db, string curPoint, string desc) => task.AddWCS_TASK_DTL(db, curPoint, "", desc);
  26. /// <summary>
  27. /// 更新任务执行记录
  28. /// 同步更新历史任务
  29. /// </summary>
  30. /// <param name="task">任务信息</param>
  31. /// <param name="db">数据库上下文</param>
  32. /// <param name="curPoint">当前地址</param>
  33. /// <param name="nextPoint">下一个地址</param>
  34. /// <param name="desc">描述</param>
  35. public static void AddWCS_TASK_DTL(this WCS_TaskInfo task, SqlSugarProvider db, string curPoint, string nextPoint, string desc)
  36. {
  37. db.InsertableRowLock(new WCS_TaskDtl
  38. {
  39. ID = Guid.NewGuid(),
  40. ParentTaskCode = task.ID,
  41. CurPoint = curPoint,
  42. NextPoint = nextPoint,
  43. Desc = desc,
  44. AddWho = "WCS"
  45. }).SplitTable().ExecuteCommand();
  46. task.UpdateableOldTask(db);
  47. }
  48. /// <summary>
  49. /// 更新任务执行记录
  50. /// 同步更新历史任务
  51. /// </summary>
  52. /// <param name="task">任务信息</param>
  53. /// <param name="db">数据库上下文</param>
  54. /// <param name="curPoint">当前地址</param>
  55. /// <param name="desc">描述</param>
  56. public static void AddWCS_TASK_DTL(this WCS_TaskOld task, SqlSugarProvider db, string curPoint, string desc) => task.AddWCS_TASK_DTL(db, curPoint, "", desc);
  57. /// <summary>
  58. /// 更新任务执行记录
  59. /// 同步更新历史任务
  60. /// </summary>
  61. /// <param name="task">任务信息</param>
  62. /// <param name="db">数据库上下文</param>
  63. /// <param name="curPoint">当前地址</param>
  64. /// <param name="nextPoint">下一个地址</param>
  65. /// <param name="desc">描述</param>
  66. public static void AddWCS_TASK_DTL(this WCS_TaskOld task, SqlSugarProvider db, string curPoint, string nextPoint, string desc)
  67. {
  68. db.InsertableRowLock(new WCS_TaskDtl
  69. {
  70. ID = Guid.NewGuid(),
  71. ParentTaskCode = task.Id,
  72. CurPoint = curPoint,
  73. NextPoint = nextPoint,
  74. Desc = desc,
  75. AddWho = "WCS"
  76. }).SplitTable().ExecuteCommand();
  77. }
  78. /// <summary>
  79. /// 更新历史表数据
  80. /// </summary>
  81. /// <param name="taskInfo"></param>
  82. /// <param name="db"></param>
  83. public static void UpdateableOldTask(this WCS_TaskInfo taskInfo, SqlSugarProvider db)
  84. {
  85. if (taskInfo.Status == TaskStatus.NewBuild) return;
  86. // 同步任务信息
  87. var taskOld = db.Queryable<WCS_TaskOld>().RowLock().Where(v => v.Id == taskInfo.ID).SplitTable(tabs => tabs.Take(2)).ToList().OrderByDescending(v => v.AddTime).First();
  88. if (taskOld is not null)
  89. {
  90. if (taskInfo.Status >= TaskStatus.Finish) taskInfo.CompleteOrCancelTasks(db);
  91. else
  92. {
  93. taskOld = taskInfo.Mapper<WCS_TaskOld, WCS_TaskInfo>();
  94. taskOld.Id = taskInfo.ID;
  95. db.UpdateableRowLock(taskOld).Where(x => x.Id == taskOld.Id).SplitTable(tabs => tabs.Take(2)).ExecuteCommand();
  96. }
  97. }
  98. else
  99. {
  100. throw new KnownException($"WCS_TaskOld表中不存在任务:{taskInfo.ID},无法执行WCS_TaskInfo与WCS_TaskOld同步动作", LogLevelEnum.Mid);
  101. }
  102. }
  103. /// <summary>
  104. /// 完成或取消任务
  105. /// </summary>
  106. /// <param name="taskInfo"></param>
  107. /// <param name="db"></param>
  108. public static void CompleteOrCancelTasks(this WCS_TaskInfo taskInfo, SqlSugarProvider db)
  109. {
  110. if (taskInfo.Status is not Entity.TaskStatus.Finish and not Entity.TaskStatus.Cancel) throw new KnownException("任务未完成或取消,无法执行WCS_TaskInfo与WCS_TaskOld同步动作", LogLevelEnum.Mid);
  111. // 任务完成或取消,进行相关同步动作
  112. var taskOld = db.Queryable<WCS_TaskOld>().RowLock().Where(v => v.Id == taskInfo.ID).SplitTable(tabs => tabs.Take(2)).ToList().OrderByDescending(v => v.AddTime).First();
  113. if (taskOld is not null)
  114. {
  115. taskOld = taskInfo.Mapper<WCS_TaskOld, WCS_TaskInfo>();
  116. taskOld.Id = taskInfo.ID;
  117. //更新任务历史表,删除任务当前表
  118. db.UpdateableRowLock(taskOld).Where(x => x.Id == taskOld.Id).SplitTable(tabs => tabs.Take(2)).ExecuteCommand();
  119. db.DeleteableRowLock(taskInfo).ExecuteCommand();
  120. }
  121. else
  122. {
  123. throw new KnownException($"WCS_TaskOld表中不存在任务:{taskInfo.ID},无法执行WCS_TaskInfo与WCS_TaskOld同步动作", LogLevelEnum.Mid);
  124. }
  125. }
  126. /// <summary>
  127. /// 更新表数据
  128. /// </summary>
  129. /// <param name="taskInfo"></param>
  130. /// <param name="db"></param>
  131. public static void Updateable(this WCS_TaskInfo taskInfo, SqlSugarProvider db)
  132. {
  133. }
  134. /// <summary>
  135. /// 添加异常信息
  136. /// </summary>
  137. /// <param name="db">db</param>
  138. /// <param name="error"></param>
  139. /// <returns></returns>
  140. public static void AddBaseErrorInfo(this BaseErrorInfo error, SqlSugarProvider db)
  141. {
  142. var errorinfoWcs = db.Queryable<BaseErrorInfo>().With(SqlWith.NoLock)
  143. .Where(v => v.BusName == error.BusName && v.Message == error.Message && v.BarCode == error.BarCode).First();
  144. if (errorinfoWcs != null && (error.AddTime - errorinfoWcs.AddTime).TotalMinutes < 60)
  145. {
  146. errorinfoWcs.Count += 1;
  147. db.Updateable(errorinfoWcs).ExecuteCommand();
  148. }
  149. else
  150. {
  151. db.Insertable(error).ExecuteCommand();
  152. }
  153. }
  154. /// <summary>
  155. /// 添加异常信息
  156. /// </summary>
  157. /// <param name="db">db</param>
  158. /// <param name="error"></param>
  159. /// <returns></returns>
  160. public static void AddBaseErrorInfo(this BaseErrorInfo error)
  161. {
  162. SqlSugarHelper.Do(db =>
  163. {
  164. var errorinfoWcs = db.Queryable<BaseErrorInfo>().With(SqlWith.NoLock)
  165. .Where(v => v.BusName == error.BusName && v.Message == error.Message && v.BarCode == error.BarCode).First();
  166. if (errorinfoWcs != null && (error.AddTime - errorinfoWcs.AddTime).TotalMinutes < 1)
  167. {
  168. errorinfoWcs.Count += 1;
  169. db.Updateable(errorinfoWcs).ExecuteCommand();
  170. }
  171. else
  172. {
  173. db.Insertable(error).ExecuteCommand();
  174. }
  175. });
  176. }
  177. ///// <summary>
  178. ///// 获取出库任务
  179. ///// </summary>
  180. ///// <param name="taskInfo">任务</param>
  181. ///// <param name="db">db</param>
  182. ///// <param name="allOutCode">可用出库站台</param>
  183. ///// <param name="floor">楼层</param>
  184. ///// <param name="obj">堆垛机</param>
  185. ///// <param name="index">递归次数</param>
  186. ///// <returns></returns>
  187. ///// <exception cref="KnownException"></exception>
  188. //public static WCS_TaskInfo GetOutTask(this WCS_TaskInfo taskInfo, SqlSugarHelper db, List<string> allOutCode, int floor, SRM obj, int index = 1)
  189. //{
  190. // return task;
  191. //}
  192. /// <summary>
  193. /// 获取AGV任务ID
  194. /// </summary>
  195. /// <param name="db">db</param>
  196. /// <returns></returns>
  197. public static int GetAgvTaskId(this SqlSugarProvider db)
  198. {
  199. //最多任务号不再连续
  200. var id = db.Queryable<WCS_AgvTaskInfo>().NoLock().SplitTable(v => v.Take(2)).Max(v => v.ID);
  201. return id + 1;
  202. }
  203. /// <summary>
  204. /// 更新任务执行记录
  205. /// </summary>
  206. /// <param name="task">任务信息</param>
  207. public static void UpdateRedisHash(this WCS_TaskInfo task)
  208. {
  209. var key = $"Hash:{task.ID}";
  210. if (task.Status >= TaskStatus.Finish)
  211. {
  212. RedisHub.WMS.Del(key);
  213. }
  214. else
  215. {
  216. RedisHub.WMS.HMSet(key, task.ToDic());
  217. }
  218. }
  219. public static int GetOutTime(this List<BaseMatinfo> matInfoList, string matCode, string defaultOutTime)
  220. {
  221. var outTime = Convert.ToInt32(defaultOutTime);
  222. if (matInfoList.Any(m => matCode.Contains(m.Code)))
  223. {
  224. outTime = matInfoList.Where(m => matCode.Contains(m.Code)).Min(m => m.TimeOut);
  225. }
  226. return outTime;
  227. }
  228. #region 工字轮支线分流
  229. /// <summary>
  230. /// 初始化码垛信息
  231. /// </summary>
  232. /// <param name="task"></param>
  233. public static void InitStackStructure(this WCS_TaskInfo task, SqlSugarProvider db, World world)
  234. {
  235. var billBomsetgrp = db.Queryable<BillBomsetgrp>().Single(x => x.IsStop == 0 && x.BomCode.Contains(task.MatCode));
  236. if (billBomsetgrp == null)
  237. {
  238. world.Log($"物料规格[{task.MatCode}]无可用码垛垛形");
  239. return;
  240. }
  241. ;
  242. var billBomsetinfos = db.Queryable<BillBomsetinfo>().Where(x => x.BomSetHdrId == billBomsetgrp.Id).ToList();
  243. var addTime = DateTime.Now;
  244. //开始构造垛形信息
  245. var palletizing = new WCS_Palletizing()
  246. {
  247. Code = billBomsetgrp.Code,
  248. ShortCode = billBomsetgrp.ShortCode,
  249. ProMaterCode = billBomsetgrp.ProMaterCode,
  250. TpTypeCode = billBomsetgrp.TpTypeCode,
  251. LayerCountQty = 2,
  252. StampType = billBomsetgrp.StampType,
  253. Finish = false,
  254. AddTime = addTime,
  255. EditTime = addTime,
  256. TaskId = task.ID,
  257. WarehouseCode = task.WarehouseCode,
  258. DeviceCode = task.Device
  259. };
  260. var matInfoList = db.Queryable<BaseMatinfo>().NoLock().Where(x => x.TimeOut != 0).ToList();
  261. var defaultOutTime = db.Queryable<fjSysConfig>().First(x => x.Code == "DefaultOutTime") ?? new fjSysConfig() { SContent = "12" };
  262. palletizing = db.InsertableRowLock(palletizing).ExecuteReturnEntity();
  263. foreach (var item in billBomsetinfos.Where(x => x.IsEmpty == 0).GroupBy(x => x.Row).OrderBy(x => x.Key))
  264. {
  265. var layerNo = item.Key <= 6 ? 1 : 2;
  266. //获取层信息
  267. var palletizingLayer = db.Queryable<WCS_PalletizingLayer>().Single(x => x.PalletizingId == palletizing.Id && x.LayerNo == layerNo);
  268. if (palletizingLayer == null)
  269. {
  270. palletizingLayer = new WCS_PalletizingLayer()
  271. {
  272. LayerNo = layerNo,
  273. PalletizingId = palletizing.Id,
  274. WarehouseCode = palletizing.WarehouseCode,
  275. };
  276. palletizingLayer = db.InsertableRowLock(palletizingLayer).ExecuteReturnEntity();
  277. }
  278. //获取行信息
  279. var palletizingRow = db.Queryable<WCS_PalletizingRow>().Single(x => x.PalletizingLayerId == palletizingLayer.Id && x.RowNo == item.Key);
  280. if (palletizingRow == null)
  281. {
  282. palletizingRow = new WCS_PalletizingRow()
  283. {
  284. RowNo = item.Key,
  285. PalletizingLayerId = palletizingLayer.Id,
  286. PalletizingId = palletizing.Id,
  287. WarehouseCode = palletizingLayer.WarehouseCode,
  288. AddTime = addTime,
  289. EditTime = addTime,
  290. };
  291. palletizingRow = db.InsertableRowLock(palletizingRow).ExecuteReturnEntity();
  292. }
  293. //重新查询最新的数据
  294. var layer = palletizingLayer;
  295. palletizingLayer = db.Queryable<WCS_PalletizingLayer>().Single(x => x.Id == layer.Id);
  296. var row = palletizingRow;
  297. palletizingRow = db.Queryable<WCS_PalletizingRow>().Single(x => x.Id == row.Id);
  298. //构造位信息
  299. foreach (var loc in item)
  300. {
  301. var palletizingLoc = db.Queryable<WCS_PalletizingLoc>().Single(x => x.PalletizingRowId == palletizingRow.Id && x.XYNo == loc.XYNo);
  302. if (palletizingLoc == null)
  303. {
  304. palletizingLoc = new WCS_PalletizingLoc()
  305. {
  306. IsEmpty = loc.IsEmpty != 0,
  307. XYNo = loc.XYNo,
  308. MatCode = loc.MatCode,
  309. SideNum = loc.SideNum,
  310. SpoolType = loc.SpoolType,
  311. TaskId = task.ID,
  312. PalletizingRowId = palletizingRow.Id,
  313. Finish = false,
  314. PalletizingId = palletizing.Id,
  315. WarehouseCode = palletizingRow.WarehouseCode,
  316. TimeOut = matInfoList.GetOutTime(loc.MatCode, defaultOutTime.SContent)
  317. };
  318. db.InsertableRowLock(palletizingLoc).ExecuteReturnEntity();
  319. }
  320. //同步是否混合料行
  321. palletizingRow.IsMixRow = loc.IsMixRow != 0;
  322. db.UpdateableRowLock(palletizingRow).UpdateColumns(x => new { x.IsMixRow }).ExecuteCommand();
  323. }
  324. //更新行信息
  325. palletizingRow = db.Queryable<WCS_PalletizingRow>().Includes(x => x.Locs).Single(x => x.Id == row.Id);
  326. palletizingRow.QtyMaxCount = palletizingRow.Locs.Count(x => !x.IsEmpty);
  327. palletizingRow.IsEmpty = palletizingRow.QtyMaxCount <= 0;
  328. palletizingRow.MatCodeList = palletizingRow.Locs.Select(x => x.MatCode).ToList().GetMatList();
  329. db.UpdateableRowLock(palletizingRow).UpdateColumns(x => new { x.QtyMaxCount, x.IsEmpty, x.MatCodeList }).ExecuteCommand();
  330. //更新层信息
  331. palletizingLayer = db.Queryable<WCS_PalletizingLayer>().Includes(x => x.Rows, l => l.Locs).Single(x => x.Id == layer.Id);
  332. var count = palletizingLayer.Rows.Count(x => !x.IsEmpty); //计算所有不空数量
  333. palletizingLayer.IsEmpty = count <= 0;
  334. palletizingLayer.RowCountQty = palletizingLayer.Rows.Count;
  335. palletizingLayer.Finish = false;
  336. palletizingLayer.MatCodeList = palletizingLayer.Rows.SelectMany(x => x.Locs).Select(x => x.MatCode).ToList().GetMatList();
  337. db.UpdateableRowLock(palletizingLayer).UpdateColumns(x => new { x.IsEmpty, x.RowCountQty, x.Finish, x.MatCodeList }).ExecuteCommand();
  338. }
  339. var palletizing1 = palletizing;
  340. palletizing = db.Queryable<WCS_Palletizing>().Includes(x => x.Layers, r => r.Rows, l => l.Locs).Single(x => x.Id == palletizing1.Id);
  341. //计算垛形信息
  342. var goods = palletizing.Layers.Select(x => x.Rows).SelectMany(x => x).Select(x => x.Locs).SelectMany(x => x).ToList();
  343. palletizing.CountQty = goods.Count(x => !x.IsEmpty);
  344. palletizing.MatCodeList = palletizing.Layers.SelectMany(x => x.Rows).SelectMany(x => x.Locs).Select(x => x.MatCode).ToList().GetMatList();
  345. db.UpdateableRowLock(palletizing).UpdateColumns(x => new { x.CountQty, x.MatCodeList }).ExecuteCommand();
  346. }
  347. public static string GetMatList(this List<string> matList)
  348. {
  349. return matList.Distinct().Aggregate("", (current, mat) => current + $"[{mat}]");
  350. }
  351. /// <summary>
  352. /// 去除转义字符
  353. /// </summary>
  354. /// <param name="value"></param>
  355. /// <returns></returns>
  356. public static string RemoveEscapeCharacters(this string? value)
  357. {
  358. return value.Trim('\0', '\a', '\b', '\f', '\n', '\r', '\t', '\v').Trim();
  359. }
  360. #endregion 工字轮支线分流
  361. }
  362. /// <summary>
  363. /// 垛形位信息
  364. /// </summary>
  365. public class StackPosInfo
  366. {
  367. /// <summary>
  368. /// 任务号
  369. /// </summary>
  370. public int TaskNumber { get; set; }
  371. /// <summary>
  372. /// 是否空置
  373. /// </summary>
  374. public bool IsEmpty { get; set; }
  375. /// <summary>
  376. /// 坐标号
  377. /// </summary>
  378. public string XYNo { get; set; }
  379. /// <summary>
  380. /// 物料编码
  381. /// </summary>
  382. public string MatCode { get; set; }
  383. /// <summary>
  384. /// 正反面
  385. /// </summary>
  386. public int SideNum { get; set; }
  387. /// <summary>
  388. /// 工字轮类型
  389. /// </summary>
  390. public string SpoolType { get; set; }
  391. /// <summary>
  392. /// 是否结束
  393. /// </summary>
  394. public bool Finish { get; set; }
  395. }
  396. }