DeviceExtension.cs 48 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277
  1. using DBHelper_SqlSugar;
  2. using Logs;
  3. using System;
  4. using System.Collections.Concurrent;
  5. using System.Collections.Generic;
  6. using System.Linq;
  7. using System.Threading.Tasks;
  8. using WCS.Core;
  9. using WCS.Entity;
  10. using WCS.Entity.Protocol;
  11. using WCS.Entity.Protocol.RGV;
  12. using WCS.Service.Helpers;
  13. using LogHelper = WCS.Service.Helpers.LogHelper;
  14. using TaskStatus = WCS.Entity.TaskStatus;
  15. namespace WCS.Service.Extensions
  16. {
  17. public static class WCS_DEVICEExtension
  18. {
  19. private static ConcurrentDictionary<string, object> DeviceValues = new ConcurrentDictionary<string, object>();
  20. public static void AddFlag(this WCS_DEVICE source, DF flag)
  21. {
  22. var df = source.Get<DF>("DeviceFlag");
  23. df = df | flag;
  24. source.Set("DeviceFlag", df);
  25. }
  26. public static bool Is(this WCS_DEVICE source, DF flag)
  27. {
  28. var df = source.Get<DF>("DeviceFlag");
  29. return (df & flag) == flag;
  30. }
  31. public static void Set<T>(this WCS_DEVICE source, string key, T value)
  32. {
  33. DeviceValues[source.CODE + key] = value;
  34. }
  35. public static T Get<T>(this WCS_DEVICE source, string key)
  36. {
  37. if (!DeviceValues.ContainsKey(source.CODE + key))
  38. return default(T);
  39. return (T)DeviceValues[source.CODE + key];
  40. }
  41. public static short Code(this WCS_DEVICE source)
  42. {
  43. return short.Parse(source.CODE);
  44. }
  45. public static string Tunnel(this WCS_DEVICE source)
  46. {
  47. return source.Get<string>("Tunnel");
  48. }
  49. public static int TunnelNum(this WCS_DEVICE source)
  50. {
  51. return int.Parse(source.Tunnel().Last().ToString());
  52. }
  53. public static int Floor(this WCS_DEVICE source)
  54. {
  55. return source.Get<int>("Floor");
  56. }
  57. public static WCS_DEVICE SC(this WCS_DEVICE source)
  58. {
  59. return source.Get<WCS_DEVICE>("SC");
  60. }
  61. public static WCS_DEVICE RGV(this WCS_DEVICE source)
  62. {
  63. return source.Get<WCS_DEVICE>("RGV");
  64. }
  65. public static LocInfo LocInfo(this WCS_DEVICE source)
  66. {
  67. return source.Get<LocInfo>("LocInfo");
  68. }
  69. public static bool WakeupOn(this WCS_DEVICE source, int sec, string key)
  70. {
  71. var str = "WakeupOn" + key;
  72. var last = source.Get<DateTime>(str);
  73. if ((DateTime.Now - last).TotalMilliseconds > sec)
  74. {
  75. source.Set(str, DateTime.Now);
  76. return true;
  77. }
  78. else
  79. {
  80. Ltc.Log("OnSleep");
  81. return false;
  82. }
  83. }
  84. public static bool WakeupOn(this WCS_DEVICE source, int sec)
  85. {
  86. return source.WakeupOn(sec, "");
  87. }
  88. public static void AddFlag(DF flag, params string[] devices)
  89. {
  90. var arr = LogicHandler.AllObjects.OfType<WCS_DEVICE>().Where(v => devices.Contains(v.CODE)).ToArray();
  91. Parallel.ForEach(arr, v =>
  92. {
  93. v.AddFlag(flag);
  94. });
  95. }
  96. public static void AddFlag(DF flag, Action<WCS_DEVICE> callbck, params string[] devices)
  97. {
  98. var arr = LogicHandler.AllObjects.OfType<WCS_DEVICE>().Where(v => devices.Contains(v.CODE)).ToArray();
  99. Parallel.ForEach(arr, v =>
  100. {
  101. v.AddFlag(flag);
  102. callbck?.Invoke(v);
  103. });
  104. }
  105. public static void Set<T>(string key, T value, params string[] devices)
  106. {
  107. var arr = LogicHandler.AllObjects.OfType<WCS_DEVICE>().Where(v => devices.Contains(v.CODE)).ToArray();
  108. Parallel.ForEach(arr, v =>
  109. {
  110. v.Set(key, value);
  111. });
  112. }
  113. public static void Set<T>(string key, T value, Func<WCS_DEVICE, bool> func)
  114. {
  115. var arr = LogicHandler.AllObjects.OfType<WCS_DEVICE>().Where(func).ToArray();
  116. Parallel.ForEach(arr, v =>
  117. {
  118. v.Set(key, value);
  119. });
  120. }
  121. }
  122. /// <summary>
  123. /// 输送机设备组
  124. /// </summary>
  125. public class StationDeviceGroup : DeviceGroup<IStation520, IStation521, IStation523>
  126. {
  127. /// <summary>
  128. /// 当前设备可用的RGV
  129. /// </summary>
  130. private static List<RGVDevice> AllRGVList;
  131. static StationDeviceGroup()
  132. {
  133. AllRGVList = Device.Where(v => v.IsRGV() && v.CODE != "RGV8").Select(v => v.Create<RGVDevice>()).ToList();
  134. }
  135. public StationDeviceGroup(WCS_DEVICE entity) : base(entity)
  136. {
  137. }
  138. /// <summary>
  139. /// 执行输送机设备组任务 单例锁
  140. /// </summary>
  141. /// <param name="act"></param>
  142. public void EX(Action<StationDeviceGroup> act)
  143. {
  144. var key = $"WCS:Lock:{Entity.CODE}";
  145. try
  146. {
  147. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  148. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  149. act(this);
  150. }
  151. catch (DoException ex)
  152. {
  153. ex.DoExceptionEX(Entity);
  154. }
  155. catch (WarnException ex)
  156. {
  157. ex.WarnExceptionEX(Entity);
  158. }
  159. catch (Exception ex)
  160. {
  161. ex.ExceptionEx(Entity);
  162. }
  163. finally
  164. {
  165. ProtocolProxy.ConcurrencyControlRedis.Del(key);
  166. }
  167. }
  168. /// <summary>
  169. /// 当前设备可用的RGV
  170. /// </summary>
  171. public List<RGVDevice> RgvList
  172. {
  173. get
  174. {
  175. return AllRGVList.Where(v => v.LocationList.Any(p => p.Entity == Entity) && v.Data2.WorkMode != 0).ToList();
  176. }
  177. }
  178. /// <summary>
  179. /// 当前设备环穿组
  180. /// </summary>
  181. private List<StationLocation> LoncationList
  182. {
  183. get
  184. {
  185. var dev = StationLocation.ALLlocations.FirstOrDefault(v => v.Station == Entity.CODE);
  186. return StationLocation.ALLlocations.Where(v => v.PLC == dev.PLC).ToList();
  187. }
  188. }
  189. /// <summary>
  190. /// 设备组自身的位置
  191. /// </summary>
  192. public float Position
  193. {
  194. get
  195. {
  196. return StationLocation.ALLlocations.FirstOrDefault(v => v.Station == Entity.CODE).Location;
  197. }
  198. }
  199. /// <summary>
  200. /// 设备组所在环穿的总长度
  201. /// </summary>
  202. public float Length
  203. {
  204. get
  205. {
  206. return StationLocation.ALLlocations.FirstOrDefault(v => v.Station == Entity.CODE).Length;
  207. }
  208. }
  209. /// <summary>
  210. /// 设备组是否满足任务执行条件
  211. /// </summary>
  212. /// <param name="type">给当前设备组下发任务时需要的请求</param>
  213. /// <returns>true:不满足执行条件需要进行停止执行 false:表示满足条件不需要停止执行 </returns>
  214. /// <exception cref="Exception"></exception>
  215. public void WhetherToExecute(IstationRequest type = IstationRequest.无)
  216. {
  217. foreach (var item in Items)
  218. {
  219. if (item.Data.VoucherNo != item.Data2.VoucherNo) throw new WarnException($"等待{item.Entity.CODE}执行任务{item.Data.Tasknum},凭证号不一致");
  220. if (item.Data3.Status.HasFlag(StationStatus.运行状态位)) throw new DoException($"{item.Entity.CODE}运行中");
  221. if (!item.Data2.Status.HasFlag(IstationStatus.光电状态)) throw new DoException($"[{item.Entity.CODE}]无光电");
  222. }
  223. }
  224. /// <summary>
  225. /// 获取设备组中需要取货的设备
  226. /// </summary>
  227. /// <returns></returns>
  228. public List<Device<IStation520, IStation521, IStation523>> RGVGetTaskedDevice()
  229. {
  230. var a = Items.Where(v => v.Data2.Status.HasFlag(IstationStatus.光电状态) && v.Data2.Tasknum > 10000)
  231. .Where(v => v.Entity.CODE.ToShort() != v.Data2.Goodsend && v.Data2.Goodsend != 0)
  232. .ToList();
  233. return a.Count == 0 ? null : a;
  234. }
  235. /// <summary>
  236. /// 是否可以取货
  237. /// </summary>
  238. /// <returns></returns>
  239. public bool IsPickUp(RGVDevice rgvDevice)
  240. {
  241. if (Entity.CODE is "G2" or "G3") return true;
  242. var dCount = Device.Where(v => v.CODE is "G1" or "G1340" or "G1337").Select(v => v.Create<StationDeviceGroup>()).Count(v =>
  243. {
  244. var count = v.Items.Count(v => !v.Data3.Status.HasFlag(StationStatus.运行状态位) && !v.Data2.Status.HasFlag(IstationStatus.光电状态) && v.Data2.Tasknum < 10000
  245. && v.Data3.Status.HasFlag(StationStatus.自动));
  246. return count == 2 ? true : false;
  247. });
  248. var rCount = rgvDevice.RGVList.Count(v => v.Data.DestPosition_1 == 1);
  249. return rCount < dCount;
  250. }
  251. /// <summary>
  252. /// 入库位置获取需要生产任务的设备及条码信息
  253. /// </summary>
  254. /// <returns></returns>
  255. public List<FinishTaskList<string>> GetBcrValid()
  256. {
  257. var list = new List<FinishTaskList<string>>();
  258. //获取需要执行的设备信息
  259. foreach (var dev in Items)
  260. {
  261. if (!dev.Data2.Status.HasFlag(IstationStatus.光电状态))
  262. {
  263. InfoLog.INFO_INFO($"{dev.Entity.CODE}--没有光电");
  264. continue;
  265. }
  266. if (dev.Data2.Request != IstationRequest.扫码入库)
  267. {
  268. InfoLog.INFO_WarnDb($"{dev.Entity.CODE}--有光电没有扫码入库请求--1", Entity.CODE);
  269. continue;
  270. };
  271. if (dev.Data2.Tasknum > 10000)
  272. {
  273. InfoLog.INFO_WarnDb($"{dev.Entity.CODE}--有光电有请求,但已有任务号", Entity.CODE);
  274. continue;
  275. }
  276. var bcr = dev.Entity.BCR();
  277. var barcode = bcr.Content.Trim('\r');
  278. if (barcode == "")
  279. {
  280. InfoLog.INFO_WarnDb($"{dev.Entity.CODE}--扫码失败,内容为空", Entity.CODE);
  281. continue;
  282. };
  283. list.Add(new FinishTaskList<string>(barcode, dev.Entity.Create<StationDevice>()));
  284. }
  285. return list;
  286. }
  287. /// <summary>
  288. /// 获取下一个地址的有效设备
  289. /// </summary>
  290. /// <returns></returns>
  291. public List<FinishTaskList<string>> GetAddressValid()
  292. {
  293. var list = new List<FinishTaskList<string>>();
  294. //获取需要执行的设备信息
  295. foreach (var dev in Items)
  296. {
  297. if (!dev.Data2.Status.HasFlag(IstationStatus.光电状态))
  298. {
  299. InfoLog.INFO_INFO($"{dev.Entity.CODE}--没有光电");
  300. continue;
  301. }
  302. if (dev.Data2.Request != IstationRequest.请求分配目标地址)
  303. {
  304. InfoLog.INFO_WarnDb($"{dev.Entity.CODE}--有光电没有分配目标地址请求--2", Entity.CODE);
  305. continue;
  306. };
  307. if (dev.Data2.Tasknum < 10000)
  308. {
  309. InfoLog.INFO_WarnDb($"{dev.Entity.CODE}--有光电有请求没有任务号", Entity.CODE);
  310. continue;
  311. }
  312. list.Add(new FinishTaskList<string>(dev.Entity.CODE, dev.Entity.Create<StationDevice>()));
  313. }
  314. return list;
  315. }
  316. /// <summary>
  317. /// 最近的RGV
  318. /// </summary>
  319. /// <returns></returns>
  320. public RGVDevice RecentRGV()
  321. {
  322. return RgvList.OrderBy(v => v.Distance(this)).FirstOrDefault();
  323. }
  324. /// <summary>
  325. /// 计算目标RGV与站台自身的距离
  326. /// </summary>
  327. /// <param name="rgv"></param>
  328. /// <returns></returns>
  329. public float Distance(RGVDevice rgv)
  330. {
  331. return DevEX.Distance(Position, rgv.Position, Length);
  332. }
  333. /// <summary>
  334. /// 计算两个站台之间的距离
  335. /// </summary>
  336. /// <param name="rgv"></param>
  337. /// <returns></returns>
  338. public float Distance(StationDeviceGroup dev)
  339. {
  340. return DevEX.Distance(Position, dev.Position, Length);
  341. }
  342. /// <summary>
  343. /// 当前RGV
  344. /// </summary>
  345. /// <returns></returns>
  346. public RGVDevice CurrentRGV()
  347. {
  348. //RGV与站台距离误差为 正负50
  349. var max = Position + 500;
  350. var min = Position - 500;
  351. return RgvList?.FirstOrDefault(v => v.Data2.Position < max && v.Data2.Position > min);
  352. }
  353. /// <summary>
  354. /// 是否需要RGV
  355. /// </summary>
  356. /// <returns>true:需要RGV false:不需要RGV</returns>
  357. public bool NeedRgv()
  358. {
  359. var rgvs = Device.Where(v => v.IsRGV()).Select(v => v.Device<IRGV521>());
  360. var code = Entity.CODE.Replace("G", "").ToShort();
  361. if (rgvs.Any(v => v.Data.DestPosition_1 == code && v.Data.SystemStatus != RGVRunStatus.空闲))
  362. throw new WarnException("已有RGV执行中");
  363. foreach (var item in Items)
  364. {
  365. if (item.Data3.Status.HasFlag(StationStatus.运行状态位)) return false;
  366. if (!item.Data2.Status.HasFlag(IstationStatus.光电状态)) return false;
  367. }
  368. return true;
  369. }
  370. /// <summary>
  371. /// BCR 站点是否被禁止
  372. /// </summary>
  373. /// <returns></returns>
  374. public void BcrStationIsForbid()
  375. {
  376. var config = ProtocolProxy.ConcurrencyControlRedis.Get("ForbidTubuEnter").Split(",");
  377. if (config.Contains(Entity.CODE)) throw new WarnException("当前入库口已被禁用,请联系机修人员了解具体情况");
  378. }
  379. }
  380. /// <summary>
  381. /// 输送机设备
  382. /// </summary>
  383. public class StationDevice : Device<IStation520, IStation521, IStation523>
  384. {
  385. public StationDevice(WCS_DEVICE entity) : base(entity)
  386. {
  387. }
  388. /// <summary>
  389. /// 设备组是否满足任务执行条件
  390. /// </summary>
  391. /// <param name="type">给当前设备组下发任务时需要的请求</param>
  392. /// <returns>true:不满足执行条件需要进行停止执行 false:表示满足条件不需要停止执行 </returns>
  393. /// <exception cref="Exception"></exception>
  394. public void WhetherToExecute(IstationRequest type = IstationRequest.无)
  395. {
  396. //正在运行
  397. if (Data3.Status.HasFlag(StationStatus.运行状态位)) throw new DoException("运行中");
  398. //上一次的任务还未执行
  399. if (Data.VoucherNo != Data2.VoucherNo)
  400. throw new WarnException($"等待任务[{Data2.Tasknum}]执行");
  401. //没有光电
  402. if (!Data2.Status.HasFlag(IstationStatus.光电状态)) throw new DoException("无光电"); ;
  403. //没有任务号
  404. switch (type)
  405. {
  406. case IstationRequest.无:
  407. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  408. throw new WarnException($"设备无任务");
  409. break;
  410. case IstationRequest.扫码入库:
  411. if (Data2.Tasknum > 10000 && Data.Tasknum > 10000)
  412. throw new WarnException($"设备已有任务任务");
  413. break;
  414. case IstationRequest.堆垛机放货完成请求目标地址:
  415. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  416. throw new WarnException($"设备无任务信息");
  417. break;
  418. case IstationRequest.请求分配目标地址:
  419. if (Data2.Tasknum < 10000 && Data.Tasknum < 10000)
  420. throw new WarnException($"设备无任务信息");
  421. break;
  422. }
  423. //没有请求
  424. if (type != IstationRequest.无 && Data2.Request != type)
  425. throw new WarnException($"有光电无{type}请求");
  426. }
  427. /// <summary>
  428. /// 执行输送机任务 单例锁
  429. /// </summary>
  430. /// <param name="act"></param>
  431. public void EX(Action<StationDevice> act)
  432. {
  433. var key = $"WCS:Lock:{Entity.CODE}";
  434. try
  435. {
  436. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  437. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  438. act(this);
  439. }
  440. catch (DoException ex)
  441. {
  442. ex.DoExceptionEX(Entity);
  443. }
  444. catch (WarnException ex)
  445. {
  446. ex.WarnExceptionEX(Entity);
  447. }
  448. catch (Exception ex)
  449. {
  450. ex.ExceptionEx(Entity);
  451. }
  452. finally
  453. {
  454. ProtocolProxy.ConcurrencyControlRedis.Del(key);
  455. }
  456. }
  457. }
  458. /// <summary>
  459. /// RGV设备
  460. /// </summary>
  461. public class RGVDevice : Device<IRGV520, IRGV521, IRGV523>
  462. {
  463. static RGVDevice()
  464. {
  465. AllRGVList = Device.Where(v => v.IsRGV() && v.CODE != "RGV8").Select(v => v.Create<RGVDevice>()).ToList();
  466. }
  467. public RGVDevice(WCS_DEVICE entity) : base(entity)
  468. {
  469. }
  470. /// <summary>
  471. /// 所有环穿RGV
  472. /// </summary>
  473. private static List<RGVDevice> AllRGVList { get; set; }
  474. /// <summary>
  475. /// 与当前RGV处于同一环穿的RGV
  476. /// </summary>
  477. public List<RGVDevice> RGVList
  478. {
  479. get
  480. {
  481. //利用WorkMode来排除的环穿维护设备
  482. return AllRGVList.Where(v => v.Entity.DEVICEPROTOCOLS.Any(d => Entity.DEVICEPROTOCOLS.Any(e => e.DB.PLC.IP == d.DB.PLC.IP)))
  483. .Where(v => v.Data2.WorkMode != 0)
  484. .Where(v => v.Entity.CODE != Entity.CODE).ToList();
  485. }
  486. }
  487. /// <summary>
  488. /// RGV当前位置
  489. /// </summary>
  490. public float Position
  491. {
  492. get
  493. {
  494. return Data2.Position;
  495. }
  496. }
  497. /// <summary>
  498. /// 与当前RGV处于同一环穿的站台
  499. /// </summary>
  500. public List<StationDeviceGroup> LocationList
  501. {
  502. get
  503. {
  504. return StationLocation.ALLlocations.Where(v => Entity.DEVICEPROTOCOLS.Any(p => p.DB.PLC.CODE == v.PLC))
  505. .Select(v => Device.Find(v.Station).Create<StationDeviceGroup>()).ToList();
  506. }
  507. }
  508. /// <summary>
  509. /// 总长度
  510. /// </summary>
  511. public float Length
  512. {
  513. get
  514. {
  515. return LocationList.FirstOrDefault().Length;
  516. }
  517. }
  518. /// <summary>
  519. /// 执行RGV任务 单例锁
  520. /// </summary>
  521. /// <param name="act"></param>
  522. public void EX(Action<RGVDevice> act)
  523. {
  524. var key = $"WCS:Lock:{Entity.CODE}";
  525. try
  526. {
  527. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  528. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  529. act(this);
  530. }
  531. catch (DoException ex)
  532. {
  533. ex.DoExceptionEX(Entity);
  534. }
  535. catch (WarnException ex)
  536. {
  537. ex.WarnExceptionEX(Entity);
  538. }
  539. catch (Exception ex)
  540. {
  541. ex.ExceptionEx(Entity);
  542. }
  543. finally
  544. {
  545. ProtocolProxy.ConcurrencyControlRedis.Del(key);
  546. }
  547. }
  548. /// <summary>
  549. /// 获取前一个取货点
  550. /// </summary>
  551. /// <returns></returns>
  552. public StationDeviceGroup BeforeStation()
  553. {
  554. var a = LocationList.Where(v => v.Entity.Is(DF.涂布RGV取货设备组) || v.Entity.Is(DF.BOPPRGV取货设备组));
  555. 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();
  556. }
  557. /// <summary>
  558. /// 前一个RGV
  559. /// </summary>
  560. /// <returns></returns>
  561. public RGVDevice Before()
  562. {
  563. //按照位置排序
  564. var arr = RGVList.OrderBy(v => v.Position);
  565. var rgv = arr.FirstOrDefault(v => v.Position > Position);
  566. if (rgv == null)
  567. rgv = arr.LastOrDefault(v => v.Position < Position);
  568. return rgv;
  569. }
  570. /// <summary>
  571. /// 后一个RGV
  572. /// </summary>
  573. /// <returns></returns>
  574. public RGVDevice After()
  575. {
  576. //到当前RGV最近的一个RGV
  577. return RGVList.OrderBy(v => v.Distance(this)).FirstOrDefault();
  578. }
  579. /// <summary>
  580. /// 获取当前所在的取货站台
  581. /// </summary>
  582. /// <returns></returns>
  583. public StationDeviceGroup CurrentStation()
  584. {
  585. 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 =>
  586. {
  587. //RGV与站台距离误差为 正负50500
  588. var max = v.Position + 500;
  589. var min = v.Position - 500;
  590. return Data2.Position < max && Data2.Position > min;
  591. }).FirstOrDefault();
  592. }
  593. /// <summary>
  594. /// 计算当前RGV与指定RGV之间的距离
  595. /// </summary>
  596. /// <param name="rgv"></param>
  597. /// <returns></returns>
  598. public float Distance(RGVDevice rgv)
  599. {
  600. //return Math.Abs((Position - rgv.Position + Length) % Length);
  601. return DevEX.Distance(Position, rgv.Position, Length);
  602. }
  603. /// <summary>
  604. /// 计算当前RGV与指定站台之间的距离
  605. /// </summary>
  606. /// <param name="after"></param>
  607. /// <returns></returns>
  608. public float Distance(StationDeviceGroup after)
  609. {
  610. if (after == null) throw new WarnException($"不是一个有效的StationDeviceGroup,{Entity.CODE}");
  611. return DevEX.Distance(Position, after.Position, Length);
  612. }
  613. /// <summary>
  614. /// 是否需要执行放货任务
  615. /// </summary>
  616. /// <returns></returns>
  617. public bool IsPut()
  618. {
  619. if (Data2.TaskType_1 != RGVTaskType.取货) return false;
  620. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.RGV到站)) return false;
  621. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.任务完成)) return false;
  622. if (!Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) return false;
  623. return true;
  624. }
  625. /// <summary>
  626. /// 写入移动任务
  627. /// </summary>
  628. /// <param name="addr">目标地址</param>
  629. public void Move(StationDeviceGroup addr)
  630. {
  631. if (Data.TaskType_1 == RGVTaskType.取货) throw new WarnException($"当前有{Data.TaskType_1}任务,无法执行移动任务");
  632. if (Data2.WorkMode != RGVMode.自动) throw new WarnException($"RGV状态{Data2.WorkMode},无法执行移动任务");
  633. if (Data2.SystemStatus != RGVRunStatus.空闲) throw new WarnException($"rgv状态为{Data2.SystemStatus},无法执行移动任务");
  634. if (Data2.Status_1.HasFlag(WCS.Entity.Protocol.RGVStatus.光电)) throw new WarnException("RGV有光电,无法执行移动任务");
  635. if (Data2.TaskID_1 == addr.Entity.CODE.GetShortCode() && Data.TaskType_1 == RGVTaskType.移动)
  636. {
  637. InfoLog.INFO_RGVINFO($"{Entity.CODE}]--已有目标地址相同的移动任务");
  638. return;
  639. }
  640. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV移动任务-开始:{Data.TaskID_1},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  641. Data.TaskID_1 = addr.Entity.CODE.GetShortCode();
  642. Data.TaskType_1 = RGVTaskType.移动;
  643. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  644. Data.Trigger_1++;
  645. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV移动任务-结束:{Data.TaskID_1},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  646. }
  647. /// <summary>
  648. /// 写入取货任务
  649. /// </summary>
  650. /// <param name="addr">目标地址</param>
  651. public void Pick(StationDeviceGroup addr, int task1 = 0, int task2 = 0)
  652. {
  653. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV取货任务-开始:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  654. Data.TaskType_1 = RGVTaskType.取货;
  655. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  656. if (task1 != 0) Data.TaskID_1 = task1;
  657. if (task2 != 0) Data.TaskID_2 = task2;
  658. Data.Trigger_1++;
  659. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV取货任务-结束:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  660. }
  661. /// <summary>
  662. /// 写入放货任务
  663. /// </summary>
  664. /// <param name="addr">目标地址</param>
  665. public void Put(StationDeviceGroup addr, int task1 = 0, int task2 = 0)
  666. {
  667. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV放货任务-开始:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  668. Data.TaskType_1 = RGVTaskType.放货;
  669. Data.DestPosition_1 = addr.Entity.CODE.GetShortCode();
  670. if (task1 != 0) Data.TaskID_1 = task1;
  671. if (task2 != 0) Data.TaskID_2 = task2;
  672. Data.Trigger_1++;
  673. InfoLog.INFO_RGVINFO($"[{Entity.CODE}]--写入RGV放货任务-结束:{Data.TaskID_1},{Data.TaskID_2},{Data.TaskType_1},{Data.DestPosition_1},{Data.Trigger_1}");
  674. }
  675. /// <summary>
  676. /// 筛选出所有与当前RGV距离小于指定长度的RGV
  677. /// </summary>
  678. /// <param name="distance">指定长度</param>
  679. /// <returns></returns>
  680. public RGVDevice[] RgvAfter(float distance)
  681. {
  682. return RGVList.Where(v => Distance(v) < distance).ToArray();
  683. }
  684. /// <summary>
  685. /// 当前RGV是否有拦住指定RGV
  686. /// </summary>
  687. /// <param name="rgv">RGV</param>
  688. /// <returns></returns>
  689. public bool StopedByMe(RGVDevice rgv)
  690. {
  691. //目标站台
  692. var target = rgv.Data2.DestPosition_1;
  693. //获取目标站台的设备组信息
  694. var station = Device.Find($"G{target}").Create<StationDeviceGroup>();
  695. if (station.Distance(rgv) < 5000) return false;
  696. //当前RGV与目标站台的距离小于传入RGV到达目标站台的距离
  697. return (this.Distance(station) < rgv.Distance(station)) || station.CurrentRGV()?.Entity.CODE == this.Entity.CODE;
  698. }
  699. /// <summary>
  700. /// 获取当前RGV的下一个站台,即距离最近的一个站台
  701. /// </summary>
  702. /// <returns></returns>
  703. public StationDeviceGroup NextStation()
  704. {
  705. //先取当前RGV与所有站台的距离
  706. var dev = LocationList.OrderBy(v => v.Distance(this)).FirstOrDefault();
  707. return dev;
  708. }
  709. }
  710. /// <summary>
  711. /// 堆垛机设备
  712. /// </summary>
  713. public class SRMDevice : Device<ISRM520, ISRM521, ISRM537>
  714. {
  715. public SRMDevice(WCS_DEVICE entity) : base(entity)
  716. {
  717. }
  718. /// <summary>
  719. /// 获取放货点
  720. /// </summary>
  721. public List<StationDevice> GetDeliveryPoint()
  722. {
  723. return Entity.ROUTES.Select(v => v.NEXT) //巷道
  724. .SelectMany(v => v.ROUTES.Select(d => d.NEXT)) //放货点
  725. .Where(v => v.IsConv()) //必须是输送线
  726. .Select(v => v.Create<StationDevice>()).ToList();
  727. }
  728. /// <summary>
  729. /// 获取取货点
  730. /// </summary>
  731. public List<StationDevice> GetPickPoint()
  732. {
  733. return Device.Where(v => v.Is(DF.SRM二级品取货) || v.Is(DF.SRM涂布取货) || v.Is(DF.SRMBOPP取货))
  734. .Where(v => v.ROUTES.Any(p => p.NEXT.ROUTES.Any(d => d.NEXT == Entity)))
  735. .Select(v => v.Create<StationDevice>())
  736. .ToList();
  737. }
  738. /// <summary>
  739. /// 处理完成任务
  740. /// </summary>
  741. public void FinishedTaskHandle()
  742. {
  743. WCS_TASK task = new WCS_TASK();
  744. Db.Do(db =>
  745. {
  746. var taskIds = new List<int>() { Data2.FinishedTask_1, Data2.FinishedTask_2 }.ToArray();
  747. for (int i = 0; i < taskIds.Length; i++)
  748. {
  749. //判断当前工位是否有完成任务
  750. if (taskIds[i] == 0) continue;
  751. task = db.Default.Queryable<WCS_TASK>().Single(v => taskIds[i] == v.ID);
  752. if (task.STATUS != TaskStatus.堆垛机执行 && task.STATUS != TaskStatus.堆垛机完成)
  753. {
  754. InfoLog.INFO_WarnDb($"任务{task.ID},状态位{task.STATUS}", Entity.CODE);
  755. continue;
  756. };
  757. if (task.STATUS == TaskStatus.堆垛机完成)
  758. {
  759. if (i == 0)
  760. {
  761. Data.FinishedACK_1 = 1;
  762. Data.TaskID_1 = 0;
  763. }
  764. else
  765. {
  766. Data.FinishedACK_2 = 1;
  767. Data.TaskID_2 = 0;
  768. }
  769. throw new DoException("二次处理堆垛机完成任务");
  770. }
  771. switch (task.TYPE)
  772. {
  773. case TaskType.入库:
  774. task.ENDTIME = DateTime.Now;
  775. task.STATUS = WCS.Entity.TaskStatus.已完成;
  776. task.UPDATETIME = DateTime.Now;
  777. break;
  778. case TaskType.出库:
  779. task.STATUS = TaskStatus.堆垛机完成;
  780. task.UPDATETIME = DateTime.Now;
  781. break;
  782. case TaskType.移库:
  783. {
  784. if (task.STATUS == TaskStatus.堆垛机执行)
  785. {
  786. task.STATUS = TaskStatus.已完成;
  787. task.UPDATETIME = DateTime.Now;
  788. }
  789. break;
  790. }
  791. default:
  792. throw new Exception($"[{Entity.CODE}]任务类型错误,{task.ID}");
  793. }
  794. task.CreateStatusLog(db, $"状态由[{TaskStatus.堆垛机执行}]变更为[{task.STATUS}]", this.GetType());
  795. db.Default.Updateable(task).AddQueue();
  796. }
  797. db.Default.SaveQueues();
  798. });
  799. Db.Do(db =>
  800. {
  801. var taskIds = new List<int>() { Data2.FinishedTask_1, Data2.FinishedTask_2 }.ToArray();
  802. for (int i = 0; i < taskIds.Length; i++)
  803. {
  804. //判断当前工位是否有完成任务
  805. if (taskIds[i] == 0) continue;
  806. //获取当前工位的目标地址
  807. task = db.Default.Queryable<WCS_TASK>().Single(v => taskIds[i] == v.ID);
  808. if (i == 0)
  809. {
  810. Data.FinishedACK_1 = 1;
  811. Data.TaskID_1 = 0;
  812. }
  813. else
  814. {
  815. Data.FinishedACK_2 = 1;
  816. Data.TaskID_2 = 0;
  817. }
  818. }
  819. });
  820. }
  821. /// <summary>
  822. /// 执行堆垛机任务 单例锁
  823. /// </summary>
  824. /// <param name="act"></param>
  825. public void EX(Action<SRMDevice> act)
  826. {
  827. var key = $"WCS:Lock:{Entity.CODE}";
  828. try
  829. {
  830. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"[{Entity.CODE}]--触发并发管控");
  831. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  832. act(this);
  833. }
  834. catch (DoException ex)
  835. {
  836. ex.DoExceptionEX(Entity);
  837. }
  838. catch (WarnException ex)
  839. {
  840. ex.WarnExceptionEX(Entity);
  841. }
  842. catch (Exception ex)
  843. {
  844. ex.ExceptionEx(Entity);
  845. }
  846. finally
  847. {
  848. ProtocolProxy.ConcurrencyControlRedis.Del(key);
  849. }
  850. }
  851. /// <summary>
  852. /// 执行出库任务 出库单例锁
  853. /// </summary>
  854. /// <param name="act"></param>
  855. public void EXOutStock(Action<SRMDevice> act)
  856. {
  857. var key = "WCS:Lock:";
  858. try
  859. {
  860. if (Entity.CODE == "SRM3" || Entity.CODE == "SRM4")
  861. {
  862. key += "SRM3-SRM4-Out";
  863. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  864. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  865. }
  866. if (Entity.CODE == "SRM5" || Entity.CODE == "SRM6")
  867. {
  868. key += "SRM5-SRM6-Out";
  869. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  870. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  871. }
  872. if (Entity.CODE == "SRM7" || Entity.CODE == "SRM8")
  873. {
  874. key += "SRM7-SRM8-Out";
  875. if (ProtocolProxy.ConcurrencyControlRedis.Get(key) != null) throw new WarnException($"触发出库并发管控--[{Entity.CODE}]");
  876. ProtocolProxy.ConcurrencyControlRedis.Set(key, Entity.CODE);
  877. }
  878. act(this);
  879. }
  880. finally
  881. {
  882. if (Entity.CODE == "SRM3" || Entity.CODE == "SRM4") ProtocolProxy.ConcurrencyControlRedis.Del($"{key}SRM3-SRM4-Out");
  883. if (Entity.CODE == "SRM5" || Entity.CODE == "SRM6") ProtocolProxy.ConcurrencyControlRedis.Del($"{key}SRM5-SRM6-Out");
  884. if (Entity.CODE == "SRM7" || Entity.CODE == "SRM8") ProtocolProxy.ConcurrencyControlRedis.Del($"{key}SRM7-SRM8-Out");
  885. }
  886. }
  887. /// <summary>
  888. /// 一工位写任务
  889. /// </summary>
  890. /// <param name="task"></param>
  891. /// <param name="goodsnum">货物数量</param>
  892. public void WriteTask1(Task task, short goodsnum)
  893. {
  894. 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}]");
  895. Data.TaskID_1 = task.ID;
  896. Data.SLine_1 = task.Line;
  897. Data.SCol_1 = task.Col;
  898. Data.SLayer_1 = task.Layer;
  899. Data.ELine_1 = task.SRMSTATION.ToShort();
  900. Data.ECol_1 = 0;
  901. Data.ELayer_1 = 0;
  902. Data.RES1_1 = goodsnum;
  903. Data.VoucherNo_1++;
  904. 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}]");
  905. }
  906. /// <summary>
  907. /// 二工位写任务
  908. /// </summary>
  909. /// <param name="task"></param>
  910. /// <param name="goodsnum">货物数量</param>
  911. public void WriteTask2(Task task, short goodsnum)
  912. {
  913. 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}]");
  914. Data.TaskID_2 = task.ID;
  915. Data.SLine_2 = task.Line;
  916. Data.SCol_2 = task.Col;
  917. Data.SLayer_2 = task.Layer;
  918. Data.ELine_2 = task.SRMSTATION.ToShort();
  919. Data.ECol_2 = 0;
  920. Data.ELayer_2 = 0;
  921. Data.RES1_2 = goodsnum;
  922. Data.VoucherNo_2++;
  923. 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}]");
  924. }
  925. /// <summary>
  926. /// 获取任务对应的货叉
  927. /// </summary>
  928. /// <param name="task">任务信息</param>
  929. /// <param name="index">任务在下发任务集合中的索引</param>
  930. /// <returns></returns>
  931. public SrmFork GetFork(Task task, int index)
  932. {
  933. return index switch
  934. {
  935. > 1 => throw new WarnException("一次最多下发两个任务"),
  936. //如果索引是1,直接返回货叉2
  937. 1 => SrmFork.货叉2,
  938. _ => task.Col switch
  939. {
  940. 102 => Entity.CODE switch
  941. {
  942. "SRM1" => SrmFork.货叉1,
  943. _ => SrmFork.货叉2,
  944. },
  945. 112 => SrmFork.货叉2,
  946. _ => SrmFork.货叉1,
  947. }
  948. };
  949. }
  950. /// <summary>
  951. /// 检查同组堆垛机是否有出库任务正在执行
  952. /// </summary>
  953. public void CheckOutTask()
  954. {
  955. //检查同组堆垛机是否有正在执行出库任务的
  956. Db.Do(db =>
  957. {
  958. try
  959. {
  960. var srm = Device.Find("SRM4").Create<SRMDevice>();
  961. var task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM4" && v.TYPE == TaskType.出库);
  962. switch (Entity.CODE)
  963. {
  964. case "SRM3":
  965. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  966. throw new DoException("SRM4正在执行出库任务");
  967. break;
  968. case "SRM4":
  969. srm = Device.Find("SRM3").Create<SRMDevice>();
  970. task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM3" && v.TYPE == TaskType.出库);
  971. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  972. throw new DoException("SRM3正在执行出库任务");
  973. break;
  974. case "SRM5":
  975. srm = Device.Find("SRM6").Create<SRMDevice>();
  976. task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM6" && v.TYPE == TaskType.出库);
  977. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  978. throw new DoException("SRM6正在执行出库任务");
  979. break;
  980. case "SRM6":
  981. srm = Device.Find("SRM5").Create<SRMDevice>();
  982. task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM5" && v.TYPE == TaskType.出库);
  983. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  984. throw new DoException("SRM5正在执行出库任务");
  985. break;
  986. case "SRM7":
  987. srm = Device.Find("SRM8").Create<SRMDevice>();
  988. task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM8" && v.TYPE == TaskType.出库);
  989. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  990. throw new DoException("SRM8正在执行出库任务");
  991. break;
  992. case "SRM8":
  993. srm = Device.Find("SRM7").Create<SRMDevice>();
  994. task = db.Default.Queryable<WCS_TASK>().Any(v => v.STATUS == TaskStatus.堆垛机执行 && v.DEVICE == "SRM7" && v.TYPE == TaskType.出库);
  995. if (srm.Data3.SCAlarm == 0 && srm.Data2.SRMMode == WCS.Entity.Protocol.SRM.SCMode.远程 && task)
  996. throw new DoException("SRM7正在执行出库任务");
  997. break;
  998. }
  999. }
  1000. catch (Exception e)
  1001. {
  1002. InfoLog.INFO_WarnDb(e.Message, Entity.CODE);
  1003. }
  1004. });
  1005. }
  1006. }
  1007. /// <summary>
  1008. /// 异常处理
  1009. /// </summary>
  1010. public static class DevEX
  1011. {
  1012. /// <summary>
  1013. /// 计算两点距离
  1014. /// </summary>
  1015. /// <param name="start">起始点</param>
  1016. /// <param name="end">结束点</param>
  1017. /// <param name="total">总长</param>
  1018. /// <returns></returns>
  1019. public static float Distance(float start, float end, float total)
  1020. {
  1021. float distance = 0;
  1022. if (start > end) distance = (total - start) + end;
  1023. else distance = end - start;
  1024. return distance;
  1025. }
  1026. public static void DoExceptionEX(this DoException ex, WCS_DEVICE Entity)
  1027. {
  1028. InfoLog.INFO_INFO($"[{Entity.CODE}]--{ex.Message}");
  1029. }
  1030. /// <summary>
  1031. /// 警报执行记录
  1032. /// </summary>
  1033. /// <param name="ex">警报信息</param>
  1034. /// <param name="Entity">发生设备</param>
  1035. /// <param name="reportMonitor">是否上报监控</param>
  1036. /// <exception cref="Exception"></exception>
  1037. public static void WarnExceptionEX(this WarnException ex, WCS_DEVICE Entity, bool reportMonitor = true)
  1038. {
  1039. InfoLog.INFO_WARN($"[{Entity.CODE}]--{ex.Message}");
  1040. if (ex.Message.Contains("The database operation was expected")) return;
  1041. LogHelper.AddWCS_EXCEPTION(ex.Message, Entity.CODE, WCS_EXCEPTIONTYPE.无.ToString());
  1042. //排除部分频繁触发的异常上报
  1043. if (ex.Message.Contains("触发并发管控")) return;
  1044. if (reportMonitor)
  1045. {
  1046. Ltc.Log(ex.GetBaseException().Message);
  1047. throw new Exception($"[{Entity.CODE}]--{ex.Message}");
  1048. }
  1049. }
  1050. public static void ExceptionEx(this Exception ex, WCS_DEVICE Entity)
  1051. {
  1052. InfoLog.INFO_ERROR($"[{Entity.CODE}]--{ex.Message}--{ex.StackTrace}");
  1053. //排除部分频繁触发的异常上报
  1054. if (ex.Message.Contains("Collection was modified; enumeration operation may not execute.")) return;
  1055. Ltc.Log(ex.GetBaseException().Message);
  1056. }
  1057. }
  1058. /// <summary>
  1059. /// 堆垛机货叉/工位
  1060. /// </summary>
  1061. public enum SrmFork
  1062. {
  1063. 货叉1 = 0,
  1064. 货叉2 = 1,
  1065. }
  1066. /// <summary>
  1067. /// 站台位置信息
  1068. /// </summary>
  1069. public class StationLocation
  1070. {
  1071. /// <summary>
  1072. /// 所有环穿站台的信息
  1073. /// </summary>
  1074. public static List<StationLocation> ALLlocations { get; set; } = new List<StationLocation>();
  1075. static StationLocation()
  1076. {
  1077. ALLlocations.AddRange(new List<StationLocation>() {
  1078. new StationLocation("G1",486326,"RGV3",1567770),
  1079. new StationLocation("G2",693631,"RGV3",1567770),
  1080. new StationLocation("G3",789931,"RGV3",1567770),
  1081. new StationLocation("G4",961595,"RGV3",1567770),
  1082. new StationLocation("G5",1013350,"RGV3",1567770),
  1083. new StationLocation("G6",1069938,"RGV3",1567770),
  1084. new StationLocation("G7",1126338,"RGV3",1567770),
  1085. new StationLocation("G8",1178355,"RGV3",1567770),
  1086. new StationLocation("G9",1256875,"RGV3",1567770),
  1087. new StationLocation("G10",1313239,"RGV3",1567770),
  1088. new StationLocation("G11",1369970,"RGV3",1567770),
  1089. new StationLocation("G12",636770,"RGV1",3719290),
  1090. new StationLocation("G13",749520,"RGV1",3719290),
  1091. new StationLocation("G14",879930,"RGV1",3719290),
  1092. new StationLocation("G15",936310,"RGV1",3719290),
  1093. new StationLocation("G16",988000,"RGV1",3719290),
  1094. new StationLocation("G19",1785000,"RGV1",3719290),
  1095. new StationLocation("G23",2714350,"RGV1",3719290),
  1096. });
  1097. }
  1098. public StationLocation(string station, int location, string plc, int length)
  1099. {
  1100. Station = station;
  1101. Location = location;
  1102. PLC = plc;
  1103. Length = length;
  1104. }
  1105. /// <summary>
  1106. /// 输送机设备组编号
  1107. /// </summary>
  1108. public string Station { get; set; }
  1109. /// <summary>
  1110. /// 输送机在环轨中的位置
  1111. /// </summary>
  1112. public int Location { get; set; }
  1113. /// <summary>
  1114. /// 所属RGV组 PLC名称
  1115. /// </summary>
  1116. public string PLC { get; set; }
  1117. /// <summary>
  1118. /// 所属环穿轨道的长度
  1119. /// </summary>
  1120. public int Length { get; set; }
  1121. }
  1122. /// <summary>
  1123. /// 巷道信息
  1124. /// </summary>
  1125. public class TunnelInfo
  1126. {
  1127. public WCS_DEVICE Tunnel;
  1128. public WCS_DEVICE taskIN;
  1129. public Device<ISRM520, ISRM521, ISRM537> SRM;
  1130. }
  1131. /// <summary>
  1132. /// 设备配置
  1133. /// </summary>
  1134. [Flags]
  1135. public enum DF
  1136. {
  1137. 无 = 0,
  1138. SRM = 1 << 0,
  1139. SRM二级品取货 = 1 << 1,
  1140. SRM涂布取货 = 1 << 2,
  1141. SRM月台放货 = 1 << 3,
  1142. 一楼RGV放货 = 1 << 4,
  1143. 月台 = 1 << 5,
  1144. 涂布RGV = 1 << 6,
  1145. BOPPRGV = 1 << 7,
  1146. 涂布RGV取货设备组 = 1 << 8,
  1147. 涂布RGV放货设备组 = 1 << 9,
  1148. 涂布出库RGV取货站台 = 1 << 10,
  1149. 涂布入库RGV取货站台 = 1 << 11,
  1150. SRM涂布放货 = 1 << 12,
  1151. 涂布RGV取货站台 = 1 << 13,
  1152. BOPPRGV取货设备组 = 1 << 14,
  1153. BOPPRGV放货设备组 = 1 << 15,
  1154. SRMBOPP取货 = 1 << 16,
  1155. }
  1156. }