DeviceExtension.cs 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874
  1. using Logs;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using WCS.Core;
  6. using WCS.Entity;
  7. using WCS.Entity.Protocol;
  8. using WCS.Entity.Protocol.RGV;
  9. namespace WCS.Service.Extensions
  10. {
  11. /// <summary>
  12. /// 输送机设备组
  13. /// </summary>
  14. public class StationDeviceGroup : DeviceGroup<IStation520, IStation521, IStation523>
  15. {
  16. /// <summary>
  17. /// 当前设备可用的RGV
  18. /// </summary>
  19. private static readonly List<RGVDevice> AllRgvList;
  20. static StationDeviceGroup()
  21. {
  22. AllRgvList = Device.Where(v => v.IsRGV() && v.CODE != "RGV8").Select(v => v.Create<RGVDevice>()).ToList();
  23. }
  24. public StationDeviceGroup(WCS_DEVICE entity) : base(entity)
  25. {
  26. }
  27. /// <summary>
  28. /// 执行输送机设备组任务 单例锁
  29. /// </summary>
  30. /// <param name="act"></param>
  31. public void EX(Action<StationDeviceGroup> act)
  32. {
  33. var key = $"WCS:Lock:{Entity.CODE}";
  34. try
  35. {
  36. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  37. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  38. act(this);
  39. }
  40. catch (DoException ex)
  41. {
  42. ex.DoExceptionEX(Entity);
  43. }
  44. catch (WarnException ex)
  45. {
  46. ex.WarnExceptionEX(Entity);
  47. }
  48. catch (Exception ex)
  49. {
  50. ex.ExceptionEx(Entity);
  51. }
  52. finally
  53. {
  54. ProtocolProxy.Yg150Redis.Del(key);
  55. }
  56. }
  57. /// <summary>
  58. /// 当前设备可用的RGV
  59. /// </summary>
  60. public List<RGVDevice> RgvList
  61. {
  62. get
  63. {
  64. return AllRgvList.Where(v => v.LocationList.Any(p => p.Entity == Entity) && v.Data2.WorkMode != 0).ToList();
  65. }
  66. }
  67. /// <summary>
  68. /// 设备组自身的位置
  69. /// </summary>
  70. public float Position
  71. {
  72. get
  73. {
  74. return StationLocation.ALLlocations.FirstOrDefault(v => v.Station == Entity.CODE)!.Location;
  75. }
  76. }
  77. /// <summary>
  78. /// 设备组所在环穿的总长度
  79. /// </summary>
  80. public float Length
  81. {
  82. get
  83. {
  84. return StationLocation.ALLlocations.FirstOrDefault(v => v.Station == Entity.CODE)!.Length;
  85. }
  86. }
  87. /// <summary>
  88. /// 设备组是否满足任务执行条件
  89. /// </summary>
  90. /// <param name="type">给当前设备组下发任务时需要的请求</param>
  91. /// <returns>true:不满足执行条件需要进行停止执行 false:表示满足条件不需要停止执行 </returns>
  92. /// <exception cref="Exception"></exception>
  93. public void WhetherToExecute(IstationRequest type = IstationRequest.无)
  94. {
  95. foreach (var item in Items)
  96. {
  97. if (item.Data.VoucherNo != item.Data2.VoucherNo) throw new WarnException($"等待{item.Entity.CODE}执行任务{item.Data.Tasknum},凭证号不一致");
  98. if (item.Data3.Status.HasFlag(StationStatus.运行状态位)) throw new DoException($"{item.Entity.CODE}运行中");
  99. if (!item.Data2.Status.HasFlag(IstationStatus.光电状态)) throw new DoException($"[{item.Entity.CODE}]无光电");
  100. }
  101. }
  102. /// <summary>
  103. /// 获取设备组中需要取货的设备
  104. /// </summary>
  105. /// <param name="obj"></param>
  106. /// <returns></returns>
  107. public List<Device<IStation520, IStation521, IStation523>> RgvGetTaskedDevice()
  108. {
  109. var a = Items.Where(v => v.Data2.Status.HasFlag(IstationStatus.光电状态) && v.Data2.Tasknum > 10000)
  110. .Where(v => v.Entity.CODE.ToShort() != v.Data2.Goodsend && v.Data2.Goodsend != 0)
  111. .ToList();
  112. return a.Count == 0 ? null : a;
  113. }
  114. /// <summary>
  115. /// 最近的RGV
  116. /// </summary>
  117. /// <returns></returns>
  118. public RGVDevice RecentRgv()
  119. {
  120. return RgvList.MinBy(v => v.Distance(this));
  121. }
  122. /// <summary>
  123. /// 计算目标RGV与站台自身的距离
  124. /// </summary>
  125. /// <param name="rgv"></param>
  126. /// <returns></returns>
  127. public float Distance(RGVDevice rgv)
  128. {
  129. return DevEX.Distance(Position, rgv.Position, Length);
  130. }
  131. /// <summary>
  132. /// 计算两个站台之间的距离
  133. /// </summary>
  134. /// <param name="rgv"></param>
  135. /// <returns></returns>
  136. public float Distance(StationDeviceGroup dev)
  137. {
  138. return DevEX.Distance(Position, dev.Position, Length);
  139. }
  140. /// <summary>
  141. /// 当前RGV
  142. /// </summary>
  143. /// <returns></returns>
  144. public RGVDevice CurrentRGV()
  145. {
  146. //RGV与站台距离误差为 正负50
  147. var max = Position + 500;
  148. var min = Position - 500;
  149. return RgvList?.FirstOrDefault(v => v.Data2.Position < max && v.Data2.Position > min);
  150. }
  151. /// <summary>
  152. /// 是否需要RGV
  153. /// </summary>
  154. /// <returns>true:需要RGV false:不需要RGV</returns>
  155. public bool NeedRgv()
  156. {
  157. var rgvs = Device.Where(v => v.IsRGV()).Select(v => v.Device<IRGV521>());
  158. var code = Entity.CODE.Replace("G", "").ToShort();
  159. if (rgvs.Any(v => v.Data.DestPosition_1 == code && v.Data.SystemStatus != RGVRunStatus.空闲))
  160. throw new WarnException("已有RGV执行中");
  161. foreach (var item in Items)
  162. {
  163. if (item.Data3.Status.HasFlag(StationStatus.运行状态位)) return false;
  164. if (!item.Data2.Status.HasFlag(IstationStatus.光电状态)) return false;
  165. }
  166. return true;
  167. }
  168. /// <summary>
  169. /// BCR 站点是否被禁止
  170. /// </summary>
  171. /// <returns></returns>
  172. public void BcrStationIsForbid()
  173. {
  174. var config = ProtocolProxy.Ygwms150Redis.Get("ForbidTubuEnter").Split(",");
  175. if (config.Contains(Entity.CODE)) throw new WarnException("当前入库口已被禁用,请联系机修人员了解具体情况");
  176. }
  177. }
  178. /// <summary>
  179. /// 输送机设备
  180. /// </summary>
  181. public class StationDevice : Device<IStation520, IStation521, IStation523>
  182. {
  183. public StationDevice(WCS_DEVICE entity) : base(entity)
  184. {
  185. }
  186. /// <summary>
  187. /// 设备组是否满足任务执行条件
  188. /// </summary>
  189. /// <param name="type">给当前设备组下发任务时需要的请求</param>
  190. /// <returns>true:不满足执行条件需要进行停止执行 false:表示满足条件不需要停止执行 </returns>
  191. /// <exception cref="Exception"></exception>
  192. public void WhetherToExecute(IstationRequest type = IstationRequest.无)
  193. {
  194. //正在运行
  195. if (Data3.Status.HasFlag(StationStatus.运行状态位)) throw new DoException("运行中");
  196. //上一次的任务还未执行
  197. if (Data.VoucherNo != Data2.VoucherNo)
  198. throw new WarnException($"等待任务[{Data2.Tasknum}]执行");
  199. //没有光电
  200. if (!Data2.Status.HasFlag(IstationStatus.光电状态)) throw new DoException("无光电"); ;
  201. //没有任务号
  202. switch (type)
  203. {
  204. case IstationRequest.无:
  205. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  206. throw new WarnException($"设备无任务");
  207. break;
  208. case IstationRequest.扫码入库:
  209. if (Data2.Tasknum > 10000 && Data.Tasknum > 10000)
  210. throw new WarnException($"设备已有任务任务");
  211. break;
  212. case IstationRequest.堆垛机放货完成请求目标地址:
  213. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  214. throw new WarnException($"设备无任务信息");
  215. break;
  216. case IstationRequest.请求分配目标地址:
  217. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  218. throw new WarnException($"设备无任务信息");
  219. break;
  220. }
  221. //没有请求
  222. if (type != IstationRequest.无 && Data2.Request != type)
  223. throw new WarnException($"有光电无{type}请求");
  224. }
  225. /// <summary>
  226. /// 执行输送机任务 单例锁
  227. /// </summary>
  228. /// <param name="act"></param>
  229. public void EX(Action<StationDevice> act)
  230. {
  231. var key = $"WCS:Lock:{Entity.CODE}";
  232. try
  233. {
  234. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  235. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  236. act(this);
  237. }
  238. catch (DoException ex)
  239. {
  240. ex.DoExceptionEX(Entity);
  241. }
  242. catch (WarnException ex)
  243. {
  244. ex.WarnExceptionEX(Entity);
  245. }
  246. catch (Exception ex)
  247. {
  248. ex.ExceptionEx(Entity);
  249. }
  250. finally
  251. {
  252. ProtocolProxy.Yg150Redis.Del(key);
  253. }
  254. }
  255. }
  256. /// <summary>
  257. /// RGV设备
  258. /// </summary>
  259. public class RGVDevice : Device<IRGV520, IRGV521, IRGV523>
  260. {
  261. static RGVDevice()
  262. {
  263. AllRGVList = Device.Where(v => v.IsRGV() && v.CODE != "RGV8").Select(v => v.Create<RGVDevice>()).ToList();
  264. }
  265. public RGVDevice(WCS_DEVICE entity) : base(entity)
  266. {
  267. }
  268. /// <summary>
  269. /// 所有环穿RGV
  270. /// </summary>
  271. private static List<RGVDevice> AllRGVList { get; set; }
  272. /// <summary>
  273. /// 与当前RGV处于同一环穿的RGV
  274. /// </summary>
  275. public List<RGVDevice> RGVList
  276. {
  277. get
  278. {
  279. return AllRGVList.Where(v => v.Entity.DEVICEPROTOCOLS.Any(d => Entity.DEVICEPROTOCOLS.Any(e => e.DB.PLC.IP == d.DB.PLC.IP)))
  280. .Where(v => v.Entity.CODE != Entity.CODE && v.Data2.WorkMode != 0).ToList();
  281. }
  282. }
  283. /// <summary>
  284. /// RGV当前位置
  285. /// </summary>
  286. public float Position
  287. {
  288. get
  289. {
  290. return Data2.Position;
  291. }
  292. }
  293. /// <summary>
  294. /// 与当前RGV处于同一环穿的站台
  295. /// </summary>
  296. public List<StationDeviceGroup> LocationList
  297. {
  298. get
  299. {
  300. return StationLocation.ALLlocations.Where(v => Entity.DEVICEPROTOCOLS.Any(p => p.DB.PLC.CODE == v.PLC))
  301. .Select(v => Device.Find(v.Station).Create<StationDeviceGroup>()).ToList();
  302. }
  303. }
  304. /// <summary>
  305. /// 总长度
  306. /// </summary>
  307. public float Length
  308. {
  309. get
  310. {
  311. return LocationList.FirstOrDefault().Length;
  312. }
  313. }
  314. /// <summary>
  315. /// 执行RGV任务 单例锁
  316. /// </summary>
  317. /// <param name="act"></param>
  318. public void EX(Action<RGVDevice> act)
  319. {
  320. var key = $"WCS:Lock:{Entity.CODE}";
  321. try
  322. {
  323. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  324. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  325. act(this);
  326. }
  327. catch (DoException ex)
  328. {
  329. ex.DoExceptionEX(Entity);
  330. }
  331. catch (WarnException ex)
  332. {
  333. ex.WarnExceptionEX(Entity);
  334. }
  335. catch (Exception ex)
  336. {
  337. ex.ExceptionEx(Entity);
  338. }
  339. finally
  340. {
  341. ProtocolProxy.Yg150Redis.Del(key);
  342. }
  343. }
  344. /// <summary>
  345. /// 获取前一个取货点
  346. /// </summary>
  347. /// <returns></returns>
  348. public StationDeviceGroup BeforeStation()
  349. {
  350. var a = LocationList.Where(v => v.Entity.Is(DF.涂布RGV取货设备组) || v.Entity.Is(DF.BOPPRGV取货设备组));
  351. return LocationList.Where(v => v.Entity.Is(DF.涂布RGV取货设备组) || v.Entity.Is(DF.BOPPRGV取货设备组) && v.Entity.CODE != this.CurrentStation().Entity.CODE).OrderBy(v => Distance(v)).FirstOrDefault();
  352. }
  353. /// <summary>
  354. /// 前一个RGV
  355. /// </summary>
  356. /// <returns></returns>
  357. public RGVDevice Before()
  358. {
  359. //按照位置排序
  360. var arr = RGVList.OrderBy(v => v.Position);
  361. var rgv = arr.FirstOrDefault(v => v.Position > Position);
  362. if (rgv == null)
  363. rgv = arr.LastOrDefault(v => v.Position < Position);
  364. return rgv;
  365. }
  366. /// <summary>
  367. /// 后一个RGV
  368. /// </summary>
  369. /// <returns></returns>
  370. public RGVDevice After()
  371. {
  372. //到当前RGV最近的一个RGV
  373. return RGVList.OrderBy(v => v.Distance(this)).FirstOrDefault();
  374. }
  375. /// <summary>
  376. /// 获取当前所在的取货站台
  377. /// </summary>
  378. /// <returns></returns>
  379. public StationDeviceGroup CurrentStation()
  380. {
  381. return LocationList.Where(v => v.Entity.Is(DF.涂布RGV取货设备组) || v.Entity.Is(DF.涂布RGV放货设备组) || v.Entity.Is(DF.BOPPRGV取货设备组) || v.Entity.Is(DF.BOPPRGV放货设备组)).Where(v =>
  382. {
  383. //RGV与站台距离误差为 正负50500
  384. var max = v.Position + 500;
  385. var min = v.Position - 500;
  386. return Data2.Position < max && Data2.Position > min;
  387. }).FirstOrDefault();
  388. }
  389. /// <summary>
  390. /// 计算当前RGV与指定RGV之间的距离
  391. /// </summary>
  392. /// <param name="rgv"></param>
  393. /// <returns></returns>
  394. public float Distance(RGVDevice rgv)
  395. {
  396. //return Math.Abs((Position - rgv.Position + Length) % Length);
  397. return DevEX.Distance(Position, rgv.Position, Length);
  398. }
  399. /// <summary>
  400. /// 计算当前RGV与指定站台之间的距离
  401. /// </summary>
  402. /// <param name="after"></param>
  403. /// <returns></returns>
  404. public float Distance(StationDeviceGroup after)
  405. {
  406. if (after == null) throw new WarnException($"不是一个有效的StationDeviceGroup,{Entity.CODE}");
  407. return DevEX.Distance(Position, after.Position, Length);
  408. }
  409. /// <summary>
  410. /// 是否需要执行放货任务
  411. /// </summary>
  412. /// <returns></returns>
  413. public bool IsPut()
  414. {
  415. if (Data2.TaskType_1 != RGVTaskType.取货) return false;
  416. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) return false;
  417. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.任务完成)) return false;
  418. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) return false;
  419. return true;
  420. }
  421. /// <summary>
  422. /// 写入移动任务
  423. /// </summary>
  424. /// <param name="addr">目标地址</param>
  425. public void Move(StationDeviceGroup addr)
  426. {
  427. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV移动任务-开始:{Data.TaskID_1},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  428. if (Data2.WorkMode != RGVMode.自动) throw new WarnException($"RGV状态{Data2.WorkMode},无法执行移动任务");
  429. if (Data2.SystemStatus != RGVRunStatus.空闲) throw new WarnException($"rgv状态为{Data2.SystemStatus},无法执行移动任务");
  430. if (Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法执行移动任务");
  431. Data.TaskID_1 = addr.Entity.CODE.GetShortCode();
  432. Data.TaskType_1 = RGVTaskType.移动;
  433. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  434. Data.Trigger_1++;
  435. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV移动任务-结束:{Data.TaskID_1},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  436. }
  437. /// <summary>
  438. /// 写入取货任务
  439. /// </summary>
  440. /// <param name="addr">目标地址</param>
  441. public void Pick(StationDeviceGroup addr, int task1 = 0, int task2 = 0)
  442. {
  443. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV取货任务-开始:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  444. Data.TaskType_1 = RGVTaskType.取货;
  445. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  446. if (task1 != 0) Data.TaskID_1 = task1;
  447. if (task2 != 0) Data.TaskID_2 = task2;
  448. Data.Trigger_1++;
  449. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV取货任务-结束:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  450. }
  451. /// <summary>
  452. /// 写入放货任务
  453. /// </summary>
  454. /// <param name="addr">目标地址</param>
  455. public void Put(StationDeviceGroup addr, int task1 = 0, int task2 = 0)
  456. {
  457. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV放货任务-开始:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  458. Data.TaskType_1 = RGVTaskType.放货;
  459. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  460. if (task1 != 0) Data.TaskID_1 = task1;
  461. if (task2 != 0) Data.TaskID_2 = task2;
  462. Data.Trigger_1++;
  463. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV放货任务-结束:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  464. }
  465. /// <summary>
  466. /// 筛选出所有与当前RGV距离小于指定长度的RGV
  467. /// </summary>
  468. /// <param name="distance">指定长度</param>
  469. /// <returns></returns>
  470. public RGVDevice[] RgvAfter(float distance)
  471. {
  472. return RGVList.Where(v => Distance(v) < distance).ToArray();
  473. }
  474. /// <summary>
  475. /// 当前RGV是否有拦住指定RGV
  476. /// </summary>
  477. /// <param name="rgv">RGV</param>
  478. /// <returns></returns>
  479. public bool StopedByMe(RGVDevice rgv)
  480. {
  481. //目标站台
  482. var target = rgv.Data2.DestPosition_1;
  483. //获取目标站台的设备组信息
  484. var station = Device.Find($"G{target}").Create<StationDeviceGroup>();
  485. if (station.Distance(rgv) < 5000) return false;
  486. //当前RGV与目标站台的距离小于传入RGV到达目标站台的距离
  487. return (this.Distance(station) < rgv.Distance(station)) || station.CurrentRGV()?.Entity.CODE == this.Entity.CODE;
  488. }
  489. /// <summary>
  490. /// 获取当前RGV的下一个站台,即距离最近的一个站台
  491. /// </summary>
  492. /// <returns></returns>
  493. public StationDeviceGroup NextStation()
  494. {
  495. //先取当前RGV与所有站台的距离
  496. var dev = LocationList.OrderBy(v => v.Distance(this)).FirstOrDefault();
  497. return dev;
  498. }
  499. }
  500. /// <summary>
  501. /// 堆垛机设备
  502. /// </summary>
  503. public class SRMDevice : Device<ISRM520, ISRM521, ISRM537>
  504. {
  505. public SRMDevice(WCS_DEVICE entity) : base(entity)
  506. {
  507. }
  508. /// <summary>
  509. /// 获取放货点
  510. /// </summary>
  511. public List<StationDevice> GetDeliveryPoint()
  512. {
  513. return Entity.ROUTES.Select(v => v.NEXT) //巷道
  514. .SelectMany(v => v.ROUTES.Select(d => d.NEXT)) //放货点
  515. .Where(v => v.IsConv()) //必须是输送线
  516. .Select(v => v.Create<StationDevice>()).ToList();
  517. }
  518. /// <summary>
  519. /// 获取取货点
  520. /// </summary>
  521. public List<StationDevice> GetPickPoint()
  522. {
  523. return Device.Where(v => v.Is(DF.SRM二级品取货) || v.Is(DF.SRM涂布取货) || v.Is(DF.SRMBOPP取货))
  524. .Where(v => v.ROUTES.Any(p => p.NEXT.ROUTES.Any(d => d.NEXT == Entity)))
  525. .Select(v => v.Create<StationDevice>())
  526. .ToList();
  527. }
  528. /// <summary>
  529. /// 处理完成任务
  530. /// </summary>
  531. public void FinishedTaskHandle()
  532. {
  533. WCS_TASK task = new WCS_TASK();
  534. }
  535. /// <summary>
  536. /// 执行堆垛机任务 单例锁
  537. /// </summary>
  538. /// <param name="act"></param>
  539. public void EX(Action<SRMDevice> act)
  540. {
  541. var key = $"WCS:Lock:{Entity.CODE}";
  542. try
  543. {
  544. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  545. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  546. act(this);
  547. }
  548. catch (DoException ex)
  549. {
  550. ex.DoExceptionEX(Entity);
  551. }
  552. catch (WarnException ex)
  553. {
  554. ex.WarnExceptionEX(Entity);
  555. }
  556. catch (Exception ex)
  557. {
  558. ex.ExceptionEx(Entity);
  559. }
  560. finally
  561. {
  562. ProtocolProxy.Yg150Redis.Del(key);
  563. }
  564. }
  565. /// <summary>
  566. /// 执行出库任务 出库单例锁
  567. /// </summary>
  568. /// <param name="act"></param>
  569. public void EXOutStock(Action<SRMDevice> act)
  570. {
  571. var key = "WCS:Lock:";
  572. try
  573. {
  574. if (Entity.CODE == "SRM3" || Entity.CODE == "SRM4")
  575. {
  576. key += "SRM3-SRM4-Out";
  577. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  578. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  579. }
  580. if (Entity.CODE == "SRM5" || Entity.CODE == "SRM6")
  581. {
  582. key += "SRM5-SRM6-Out";
  583. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  584. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  585. }
  586. if (Entity.CODE == "SRM7" || Entity.CODE == "SRM8")
  587. {
  588. key += "SRM7-SRM8-Out";
  589. if (ProtocolProxy.Yg150Redis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  590. ProtocolProxy.Yg150Redis.Set(key, Entity.CODE);
  591. }
  592. act(this);
  593. }
  594. finally
  595. {
  596. if (Entity.CODE == "SRM3" || Entity.CODE == "SRM4") ProtocolProxy.Yg150Redis.Del($"{key}SRM3-SRM4-Out");
  597. if (Entity.CODE == "SRM5" || Entity.CODE == "SRM6") ProtocolProxy.Yg150Redis.Del($"{key}SRM5-SRM6-Out");
  598. if (Entity.CODE == "SRM7" || Entity.CODE == "SRM8") ProtocolProxy.Yg150Redis.Del($"{key}SRM7-SRM8-Out");
  599. }
  600. }
  601. /// <summary>
  602. /// 一工位写任务
  603. /// </summary>
  604. /// <param name="task"></param>
  605. /// <param name="goodsnum">货物数量</param>
  606. public void WriteTask1(Task task, short goodsnum)
  607. {
  608. InfoLog.INFO_SRMINFO($"出库--写入堆垛机[{Entity.CODE}]1工位-开始:[{Data.TaskID_1}][{Data.SLine_1}][{Data.SCol_1}][{Data.SLayer_1}][{Data.ELine_1}][{Data.VoucherNo_1}]--[{Data.RES1_1}]");
  609. Data.TaskID_1 = task.ID;
  610. Data.SLine_1 = task.Line;
  611. Data.SCol_1 = task.Col;
  612. Data.SLayer_1 = task.Layer;
  613. Data.ELine_1 = task.SRMSTATION.ToShort();
  614. Data.ECol_1 = 0;
  615. Data.ELayer_1 = 0;
  616. Data.RES1_1 = goodsnum;
  617. Data.VoucherNo_1++;
  618. InfoLog.INFO_SRMINFO($"出库--写入堆垛机[{Entity.CODE}]1工位-结束:[{Data.TaskID_1}][{Data.SLine_1}][{Data.SCol_1}][{Data.SLayer_1}][{Data.ELine_1}][{Data.VoucherNo_1}]--[{Data.RES1_1}]");
  619. }
  620. /// <summary>
  621. /// 二工位写任务
  622. /// </summary>
  623. /// <param name="task"></param>
  624. /// <param name="goodsnum">货物数量</param>
  625. public void WriteTask2(Task task, short goodsnum)
  626. {
  627. InfoLog.INFO_SRMINFO($"出库--写入堆垛机[{Entity.CODE}]2工位-开始:[{Data.TaskID_2}][{Data.SLine_2}][{Data.SCol_2}][{Data.SLayer_2}][{Data.ELine_2}][{Data.VoucherNo_2}]--[{Data.RES1_2}]");
  628. Data.TaskID_2 = task.ID;
  629. Data.SLine_2 = task.Line;
  630. Data.SCol_2 = task.Col;
  631. Data.SLayer_2 = task.Layer;
  632. Data.ELine_2 = task.SRMSTATION.ToShort();
  633. Data.ECol_2 = 0;
  634. Data.ELayer_2 = 0;
  635. Data.RES1_2 = goodsnum;
  636. Data.VoucherNo_2++;
  637. InfoLog.INFO_SRMINFO($"出库--写入堆垛机[{Entity.CODE}]2工位-结束:[{Data.TaskID_2}][{Data.SLine_2}][{Data.SCol_2}][{Data.SLayer_2}][{Data.ELine_2}][{Data.VoucherNo_2}]--[{Data.RES1_2}]");
  638. }
  639. /// <summary>
  640. /// 获取任务对应的货叉
  641. /// </summary>
  642. /// <param name="task">任务信息</param>
  643. /// <param name="index">任务在下发任务集合中的索引</param>
  644. /// <returns></returns>
  645. public SrmFork GetFork(Task task, int index)
  646. {
  647. if (index > 1) throw new WarnException("一次最多下发两个任务");
  648. //如果索引是1,直接返回货叉2
  649. if (index == 1) return SrmFork.货叉2;
  650. //判断任务列是多少
  651. return task.Col switch
  652. {
  653. 102 => Entity.CODE switch
  654. {
  655. "SRM1" => SrmFork.货叉1,
  656. _ => SrmFork.货叉2,
  657. },
  658. 112 => SrmFork.货叉2,
  659. _ => SrmFork.货叉1,
  660. };
  661. }
  662. /// <summary>
  663. /// 检查同组堆垛机是否有出库任务正在执行
  664. /// </summary>
  665. public void CheckOutTask()
  666. {
  667. }
  668. }
  669. /// <summary>
  670. /// 异常处理
  671. /// </summary>
  672. public static class DevEX
  673. {
  674. /// <summary>
  675. /// 计算两点距离
  676. /// </summary>
  677. /// <param name="start">起始点</param>
  678. /// <param name="end">结束点</param>
  679. /// <param name="total">总长</param>
  680. /// <returns></returns>
  681. public static float Distance(float start, float end, float total)
  682. {
  683. float distance = 0;
  684. if (start > end) distance = (total - start) + end;
  685. else distance = end - start;
  686. return distance;
  687. }
  688. public static void DoExceptionEX(this DoException ex, WCS_DEVICE Entity)
  689. {
  690. InfoLog.INFO_INFO($"[{Entity.CODE}]--{ex.Message}");
  691. }
  692. /// <summary>
  693. /// 警报执行记录
  694. /// </summary>
  695. /// <param name="ex">警报信息</param>
  696. /// <param name="Entity">发生设备</param>
  697. /// <param name="reportMonitor">是否上报监控</param>
  698. /// <exception cref="Exception"></exception>
  699. public static void WarnExceptionEX(this WarnException ex, WCS_DEVICE Entity, bool reportMonitor = true)
  700. {
  701. InfoLog.INFO_WARN($"[{Entity.CODE}]--{ex.Message}");
  702. if (ex.Message.Contains("The database operation was expected")) return;
  703. Helpers.LogHelper.AddWCS_EXCEPTION(ex.Message, Entity.CODE, WCS_EXCEPTIONTYPE.无.ToString());
  704. //排除部分频繁触发的异常上报
  705. if (ex.Message.Contains("触发并发管控")) return;
  706. if (reportMonitor)
  707. {
  708. Ltc.Log(ex.GetBaseException().Message);
  709. throw new Exception($"[{Entity.CODE}]--{ex.Message}");
  710. }
  711. }
  712. public static void ExceptionEx(this Exception ex, WCS_DEVICE Entity)
  713. {
  714. InfoLog.INFO_ERROR($"[{Entity.CODE}]--{ex.Message}--{ex.StackTrace}");
  715. //排除部分频繁触发的异常上报
  716. if (ex.Message.Contains("Collection was modified; enumeration operation may not execute.")) return;
  717. Ltc.Log(ex.GetBaseException().Message);
  718. }
  719. }
  720. /// <summary>
  721. /// 堆垛机货叉/工位
  722. /// </summary>
  723. public enum SrmFork
  724. {
  725. 货叉1 = 0,
  726. 货叉2 = 1,
  727. }
  728. /// <summary>
  729. /// 站台位置信息
  730. /// </summary>
  731. public class StationLocation
  732. {
  733. /// <summary>
  734. /// 所有环穿站台的信息
  735. /// </summary>
  736. public static List<StationLocation> ALLlocations { get; set; } = new List<StationLocation>();
  737. static StationLocation()
  738. {
  739. ALLlocations.AddRange(new List<StationLocation>() {
  740. new StationLocation("G1",486326,"RGV3",1567770),
  741. new StationLocation("G2",693631,"RGV3",1567770),
  742. new StationLocation("G3",789931,"RGV3",1567770),
  743. new StationLocation("G4",961595,"RGV3",1567770),
  744. new StationLocation("G5",1013350,"RGV3",1567770),
  745. new StationLocation("G6",1069938,"RGV3",1567770),
  746. new StationLocation("G7",1126338,"RGV3",1567770),
  747. new StationLocation("G8",1178355,"RGV3",1567770),
  748. new StationLocation("G9",1256875,"RGV3",1567770),
  749. new StationLocation("G10",1313239,"RGV3",1567770),
  750. new StationLocation("G11",1369970,"RGV3",1567770),
  751. new StationLocation("G12",636770,"RGV1",3719290),
  752. new StationLocation("G13",749520,"RGV1",3719290),
  753. new StationLocation("G14",879930,"RGV1",3719290),
  754. new StationLocation("G15",936310,"RGV1",3719290),
  755. new StationLocation("G16",988000,"RGV1",3719290),
  756. //new StationLocation("G17",1607000,"RGV1",3719290),
  757. //new StationLocation("G18",1667000,"RGV1",3719290),
  758. new StationLocation("G19",1785000,"RGV1",3719290),
  759. //new StationLocation("G20",2548012,"RGV1",3719290),
  760. //new StationLocation("G21",2606033,"RGV1",3719290),
  761. //new StationLocation("G22",2660833,"RGV1",3719290),
  762. new StationLocation("G23",2714350,"RGV1",3719290),
  763. });
  764. }
  765. public StationLocation(string station, int location, string plc, int length)
  766. {
  767. Station = station;
  768. Location = location;
  769. PLC = plc;
  770. Length = length;
  771. }
  772. /// <summary>
  773. /// 输送机设备组编号
  774. /// </summary>
  775. public string Station { get; set; }
  776. /// <summary>
  777. /// 输送机在环轨中的位置
  778. /// </summary>
  779. public int Location { get; set; }
  780. /// <summary>
  781. /// 所属RGV组 PLC名称
  782. /// </summary>
  783. public string PLC { get; set; }
  784. /// <summary>
  785. /// 所属环穿轨道的长度
  786. /// </summary>
  787. public int Length { get; set; }
  788. }
  789. /// <summary>
  790. /// 巷道信息
  791. /// </summary>
  792. public class TunnelInfo
  793. {
  794. public WCS_DEVICE Tunnel;
  795. public WCS_DEVICE taskIN;
  796. public Device<ISRM520, ISRM521, ISRM537> SRM;
  797. }
  798. }