MesController.cs 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657
  1. using LocalhostMES.Core;
  2. using LocalhostMES.DataBase;
  3. using LocalhostMES.Helpers;
  4. using LocalhostMES.Models;
  5. using Newtonsoft.Json;
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Threading;
  10. using System.Threading.Tasks;
  11. using System.Web.Http;
  12. namespace LocalhostMES.Api.Controllers
  13. {
  14. [RoutePrefix("api/mes")]
  15. public class MesController : ApiController
  16. {
  17. private static int _snCounter = 1;
  18. private Management management;
  19. public MesController()
  20. {
  21. management = Management.GetManagement();
  22. }
  23. #region 2.1 厂级MES下发工单信息
  24. [HttpPost]
  25. [Route("receiveWorkOrder")]
  26. public IHttpActionResult ReceiveWorkOrder([FromBody] WorkOrderRequest request)
  27. {
  28. try
  29. {
  30. LogHelper.WriteLogInfo("厂级MES下发工单信息");
  31. // 验证必填字段
  32. if ( string.IsNullOrEmpty(request.WorkOrderNo) ||
  33. string.IsNullOrEmpty(request.MaterialCode) ||
  34. string.IsNullOrEmpty(request.MaterialName) ||
  35. string.IsNullOrEmpty(request.LineCode) )
  36. {
  37. LogHelper.WriteLogInfo("必填字段不能为空");
  38. return BadRequest("必填字段不能为空");
  39. }
  40. // 工单状态验证
  41. if ( !IsValidStatus(request.Status) )
  42. {
  43. LogHelper.WriteLogInfo("无效的工单状态");
  44. return BadRequest($"无效的工单状态: {request.Status}");
  45. }
  46. // 创建或更新工单信息
  47. var workOrder = new WorkOrderInfo
  48. {
  49. WorkOrderNo = request.WorkOrderNo,
  50. MaterialCode = request.MaterialCode,
  51. OrderNo = request.OrderNo,
  52. MaterialName = request.MaterialName,
  53. PlannedQuantity = int.TryParse(request.WorkOrderNum, out var qty) ? qty : 0,
  54. CompletedQuantity = 0,
  55. Status = request.Status,
  56. LineCode = request.LineCode,
  57. CreateTime = DateTime.Now,
  58. StartTime = DateTime.Now.AddSeconds(2),
  59. EndTime = DateTime.Now.AddSeconds(5)
  60. };
  61. if ( DatabaseHelper.InsertWorkOrderInfo(workOrder) )
  62. {
  63. var response = new ApiResponse<bool>
  64. {
  65. code = "200",
  66. success = true,
  67. msg = "工单接收成功",
  68. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  69. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  70. data = true
  71. };
  72. LogHelper.WriteLogInfo("工单接收成功");
  73. return Ok(response);
  74. }
  75. else
  76. {
  77. var errorResponse = new ErrorResponse
  78. {
  79. code = "55000",
  80. success = false,
  81. msg = $"工单接收失败:数据库插入异常",
  82. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  83. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  84. };
  85. LogHelper.WriteLogInfo("数据库插入异常");
  86. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  87. }
  88. }
  89. catch ( Exception ex )
  90. {
  91. var errorResponse = new ErrorResponse
  92. {
  93. code = "55000",
  94. success = false,
  95. msg = $"工单接收失败: {ex.Message}",
  96. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  97. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  98. };
  99. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  100. }
  101. }
  102. [HttpGet]
  103. [Route("getWorkOrders")]
  104. public IHttpActionResult GetWorkOrders()
  105. {
  106. try
  107. {
  108. LogHelper.WriteLogInfo("接口getWorkOrders收到消息");
  109. var workOrders = DatabaseHelper.SelectWorkOrderInfo(null, management.IsLocalhostMode);
  110. var response = new ApiResponse<List<WorkOrderInfo>>
  111. {
  112. code = "200",
  113. success = true,
  114. msg = "获取成功",
  115. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  116. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  117. data = workOrders
  118. };
  119. return Ok(response);
  120. }
  121. catch ( Exception ex )
  122. {
  123. var errorResponse = new ErrorResponse
  124. {
  125. code = "55000",
  126. success = false,
  127. msg = $"获取工单失败: {ex.Message}",
  128. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  129. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  130. };
  131. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  132. }
  133. }
  134. #endregion
  135. #region 2.2 SN打印请求
  136. [HttpPost]
  137. [Route("requestSnPrint")]
  138. public async Task<IHttpActionResult> RequestSnPrint([FromBody] SnPrintRequest request)
  139. {
  140. try
  141. {
  142. // 验证请求头
  143. if ( !ValidateHeaders() )
  144. {
  145. return Unauthorized();
  146. }
  147. // 验证必填字段
  148. if ( string.IsNullOrEmpty(request.workOrderNo) || request.count <= 0 )
  149. {
  150. return BadRequest("工单号和数量为必填项且数量必须大于0");
  151. }
  152. var workOrder = DatabaseHelper.SelectWorkOrderInfo(request.workOrderNo);
  153. // 验证工单是否存在
  154. if ( workOrder.Count == 0 )
  155. {
  156. return BadRequest($"工单 {request.workOrderNo} 不存在");
  157. }
  158. // 验证打印类型
  159. if ( request.printType < 1 || request.printType > 4 )
  160. {
  161. return BadRequest($"无效的打印类型: {request.printType}");
  162. }
  163. LogHelper.WriteLogInfo("请求参数:");
  164. LogHelper.WriteLogInfo(JsonConvert.SerializeObject(request, Formatting.Indented));
  165. if ( management.IsLocalhostMode )
  166. {
  167. var Manualsninfo = DatabaseHelper.SelectSnInfo(request.workOrderNo);
  168. DatabaseHelper.InsertSnInfo(new SnInfo()
  169. {
  170. Sn = "000001" + DateCodeConverter.GetYearCode(DateTime.Now.Year) +
  171. DateCodeConverter.GetMonthCode(DateTime.Now.Month) +
  172. DateCodeConverter.GetDayCode(DateTime.Now.Day) +
  173. "00000" + Manualsninfo.Count,
  174. IsUsed = false,
  175. WorkOrderNo = request.workOrderNo,
  176. GenerateTime = DateTime.Now,
  177. PrintType = "1",
  178. });
  179. }
  180. else
  181. {
  182. ApiResponse<SnPrintResponseData> mesres = await management.ApiClient.RequestSnPrintAsync(request);
  183. if ( mesres.success && mesres.data != null && mesres.data.Sn.Count != 0 )
  184. {
  185. for ( int i = 0; i < mesres.data.Sn.Count; i++ )
  186. {
  187. if ( !DatabaseHelper.InsertSnInfo(new SnInfo()
  188. {
  189. WorkOrderNo = mesres.data.WorkOrderNo,
  190. Sn = mesres.data.Sn[ i ].Sn,
  191. GenerateTime = DateTime.Now,
  192. PrintType = "1",
  193. IsUsed = false,
  194. }) )
  195. {
  196. mesres.msg = "插入SN到数据库出错";
  197. mesres.success = false;
  198. return Content(System.Net.HttpStatusCode.InternalServerError, mesres);
  199. }
  200. }
  201. return Ok(mesres);
  202. }
  203. return Content(System.Net.HttpStatusCode.InternalServerError, mesres);
  204. }
  205. var sninfo = DatabaseHelper.SelectSnInfo(request.workOrderNo, true, false);
  206. // 生成SN列表
  207. var snList = new List<SnItem>();
  208. if ( sninfo.Count != 0 )
  209. {
  210. snList.Add(new SnItem { Sn = sninfo[ 0 ].Sn });
  211. var responseData = new SnPrintResponseData
  212. {
  213. Sn = snList,
  214. OrderNo = request.orderNo ?? "",
  215. WorkOrderNo = request.workOrderNo,
  216. Line = request.line ?? "",
  217. VehicleCode = workOrder[0].MaterialCode,
  218. MaterialDesc = workOrder[0].MaterialName
  219. };
  220. var response = new ApiResponse<SnPrintResponseData>
  221. {
  222. code = "200",
  223. success = true,
  224. msg = "SN生成成功",
  225. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  226. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  227. data = responseData
  228. };
  229. return Ok(response);
  230. }
  231. var errorResponse = new ErrorResponse
  232. {
  233. code = "55000",
  234. success = false,
  235. msg = $"SN获取失败",
  236. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  237. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  238. };
  239. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  240. }
  241. catch ( Exception ex )
  242. {
  243. var errorResponse = new ErrorResponse
  244. {
  245. code = "55000",
  246. success = false,
  247. msg = $"SN生成失败: {ex.Message}",
  248. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  249. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  250. };
  251. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  252. }
  253. }
  254. [HttpGet]
  255. [Route("getSns/{workOrderNo}")]
  256. public IHttpActionResult GetSns(string workOrderNo)
  257. {
  258. try
  259. {
  260. var sninfo = DatabaseHelper.SelectSnInfo(workOrderNo, true, false);
  261. if ( sninfo.Count != 0 )
  262. {
  263. var response = new ApiResponse<List<SnInfo>>
  264. {
  265. code = "200",
  266. success = true,
  267. msg = "获取成功",
  268. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  269. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  270. data = sninfo
  271. };
  272. return Ok(response);
  273. }
  274. var errorResponse = new ErrorResponse
  275. {
  276. code = "55001",
  277. success = false,
  278. msg = $"工单 {workOrderNo} 没有SN记录",
  279. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  280. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  281. };
  282. return Content(System.Net.HttpStatusCode.NotFound, errorResponse);
  283. }
  284. catch ( Exception ex )
  285. {
  286. var errorResponse = new ErrorResponse
  287. {
  288. code = "55000",
  289. success = false,
  290. msg = $"获取SN失败: {ex.Message}",
  291. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  292. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  293. };
  294. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  295. }
  296. }
  297. #endregion
  298. #region 2.3 接收SN和关键件
  299. [HttpPost]
  300. [Route("receiveSnComponent")]
  301. public async Task<IHttpActionResult> ReceiveSnComponent([FromBody] SnKeyComponentRequest request)
  302. {
  303. try
  304. {
  305. LogHelper.WriteLogInfo($"收到请求: 接收SN和关键件");
  306. if ( management.IsLocalhostMode )
  307. {
  308. return Ok(new ApiResponse<bool?>() { data = true });
  309. }
  310. // 验证请求头
  311. if ( !ValidateHeaders() )
  312. {
  313. return Unauthorized();
  314. }
  315. // 验证必填字段
  316. if ( string.IsNullOrEmpty(request.plant) ||
  317. string.IsNullOrEmpty(request.stationCode) ||
  318. string.IsNullOrEmpty(request.positionCode) ||
  319. string.IsNullOrEmpty(request.sn) ||
  320. string.IsNullOrEmpty(request.scanTime) )
  321. {
  322. return BadRequest("必填字段不能为空");
  323. }
  324. // 处理关键件绑定
  325. if ( request.part != null && request.part.Count > 0 )
  326. {
  327. foreach ( var part in request.part )
  328. {
  329. if ( string.IsNullOrEmpty(part.partNum) )
  330. {
  331. return BadRequest("关键件条码不能为空");
  332. }
  333. var bindRecord = new BindRecord
  334. {
  335. Sn = request.sn,
  336. PartNum = part.partNum,
  337. StationCode = request.stationCode,
  338. BindTime = DateTime.TryParse(request.scanTime, out var scanTime) ? scanTime : DateTime.Now,
  339. Employee = request.employee ?? "",
  340. ScanType = request.scanTpye
  341. };
  342. DatabaseHelper.InsertBindRecord(bindRecord);
  343. if ( management.IsLocalhostMode )
  344. {
  345. //if (DatabaseHelper.GetProductProductionRecords(request.sn) == 0)
  346. //{
  347. var response = new ApiResponse<bool>
  348. {
  349. code = "200",
  350. success = true,
  351. msg = "已存在记录",
  352. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  353. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  354. data = true
  355. };
  356. return Ok(response);
  357. }
  358. }
  359. }
  360. else
  361. {
  362. var bindRecord = new BindRecord
  363. {
  364. Sn = request.sn,
  365. PartNum ="",
  366. StationCode = request.stationCode,
  367. BindTime = DateTime.TryParse(request.scanTime, out var scanTime) ? scanTime : DateTime.Now,
  368. Employee = request.employee ?? "",
  369. ScanType = request.scanTpye
  370. };
  371. DatabaseHelper.InsertBindRecord(bindRecord);
  372. }
  373. if ( management.IsLocalhostMode )
  374. {
  375. LogHelper.WriteLogInfo($"收到请求: 接收SN和关键件 成功");
  376. return Ok(new ApiResponse<bool?>() { data = true });
  377. }
  378. var okresponse = await management.ApiClient.SendSnComponentAsync(request);
  379. if ( okresponse.success && okresponse.code == "200" )
  380. {
  381. LogHelper.WriteLogInfo($"收到请求: 接收SN和关键件 成功");
  382. return Ok(okresponse);
  383. }
  384. else
  385. {
  386. LogHelper.WriteLogInfo($"收到请求: 接收SN和关键件 失败\r\n" + okresponse.ToString());
  387. okresponse.success = false;
  388. return Content(System.Net.HttpStatusCode.InternalServerError, okresponse);
  389. }
  390. }
  391. catch ( Exception ex )
  392. {
  393. var errorResponse = new ErrorResponse
  394. {
  395. code = "55000",
  396. success = false,
  397. msg = $"SN和关键件接收失败: {ex.Message}",
  398. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  399. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  400. };
  401. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  402. }
  403. }
  404. [HttpGet]
  405. [Route("getBindRecords")]
  406. public IHttpActionResult GetBindRecords()
  407. {
  408. try
  409. {
  410. var records = DatabaseHelper.SelectBindRecord();
  411. var response = new ApiResponse<List<BindRecord>>
  412. {
  413. code = "200",
  414. success = true,
  415. msg = "获取成功",
  416. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  417. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  418. data = records
  419. };
  420. return Ok(response);
  421. }
  422. catch ( Exception ex )
  423. {
  424. var errorResponse = new ErrorResponse
  425. {
  426. code = "55000",
  427. success = false,
  428. msg = $"获取绑定记录失败: {ex.Message}",
  429. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  430. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  431. };
  432. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  433. }
  434. }
  435. #endregion
  436. #region 2.4 接收工件加工参数
  437. [HttpPost]
  438. [Route("receiveProcessParameters")]
  439. public async Task<IHttpActionResult> ReceiveProcessParameters([FromBody] ProcessParameterRequest request)
  440. {
  441. try
  442. {
  443. LogHelper.WriteLogInfo($"收到请求:接收工件加工参数 ");
  444. // 验证请求头
  445. if ( !ValidateHeaders() )
  446. {
  447. return Unauthorized();
  448. }
  449. // 验证必填字段
  450. if ( string.IsNullOrEmpty(request.plant) ||
  451. string.IsNullOrEmpty(request.station) ||
  452. string.IsNullOrEmpty(request.sn) ||
  453. string.IsNullOrEmpty(request.overallResult) )
  454. {
  455. return BadRequest("必填字段不能为空");
  456. }
  457. // 验证总结果
  458. if ( request.overallResult != "OK" && request.overallResult != "NG" )
  459. {
  460. return BadRequest($"无效的总结果: {request.overallResult}");
  461. }
  462. // 保存加工记录
  463. var processRecord = new ProcessRecord
  464. {
  465. Sn = request.sn,
  466. Station = request.station,
  467. OverallResult = request.overallResult,
  468. ProcessTime = DateTime.Now,
  469. Equipment = request.equipment ?? ""
  470. };
  471. DatabaseHelper.InsertProcessRecord(processRecord);
  472. if ( management.IsLocalhostMode )
  473. {
  474. var response = new ApiResponse<bool>
  475. {
  476. code = "200",
  477. success = true,
  478. msg = "加工参数接收成功",
  479. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  480. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  481. data = true
  482. };
  483. LogHelper.WriteLogInfo($"收到请求:接收工件加工参数 成功");
  484. return Ok(response);
  485. }
  486. var okresponse = await management.ApiClient.SendProcessParametersAsync(request);
  487. if ( okresponse.success && okresponse.code == "200" )
  488. {
  489. LogHelper.WriteLogInfo($"收到请求: 接收工件加工参数 成功");
  490. return Ok(okresponse);
  491. }
  492. else
  493. {
  494. LogHelper.WriteLogInfo($"收到请求: 接收工件加工参数 失败\r\n" + okresponse.ToString());
  495. okresponse.success = false;
  496. return Content(System.Net.HttpStatusCode.InternalServerError, okresponse);
  497. }
  498. }
  499. catch ( Exception ex )
  500. {
  501. LogHelper.WriteLogInfo("出错");
  502. var errorResponse = new ErrorResponse
  503. {
  504. code = "55000",
  505. success = false,
  506. msg = $"加工参数接收失败: {ex.Message}",
  507. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  508. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  509. };
  510. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  511. }
  512. }
  513. [HttpGet]
  514. [Route("getProcessRecords")]
  515. public IHttpActionResult GetProcessRecords()
  516. {
  517. try
  518. {
  519. var records = DatabaseHelper.SelectProcessRecord();
  520. var response = new ApiResponse<List<ProcessRecord>>
  521. {
  522. code = "200",
  523. success = true,
  524. msg = "获取成功",
  525. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  526. traceId = Guid.NewGuid().ToString("N").Substring(0, 24),
  527. data = records
  528. };
  529. return Ok(response);
  530. }
  531. catch ( Exception ex )
  532. {
  533. var errorResponse = new ErrorResponse
  534. {
  535. code = "55000",
  536. success = false,
  537. msg = $"获取加工记录失败: {ex.Message}",
  538. msgTime = DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),
  539. traceId = Guid.NewGuid().ToString("N").Substring(0, 24)
  540. };
  541. return Content(System.Net.HttpStatusCode.InternalServerError, errorResponse);
  542. }
  543. }
  544. #endregion
  545. #region 辅助方法
  546. private bool ValidateHeaders()
  547. {
  548. // 简单的token验证
  549. if ( Request.Headers.TryGetValues("app-key", out var appKeyValues) &&
  550. Request.Headers.TryGetValues("token", out var tokenValues) )
  551. {
  552. var appKey = appKeyValues.FirstOrDefault();
  553. var token = tokenValues.FirstOrDefault();
  554. // 这里使用简单的验证
  555. return appKey == "test-app-key" && token == "test-token-123";
  556. }
  557. return false;
  558. }
  559. private bool IsValidStatus(string status)
  560. {
  561. var validStatuses = new[] { "0", "1", "2", "3", "4", "5", "6", "7" };
  562. return validStatuses.Contains(status);
  563. }
  564. private string GenerateSn(WorkOrderInfo workOrder, int printType)
  565. {
  566. var prefix = GetPrintTypePrefix(printType);
  567. var timestamp = DateTime.Now.ToString("yyMMddHHmm");
  568. var randomStr = Guid.NewGuid().ToString("N").Substring(0, 6).ToUpper();
  569. var counter = Interlocked.Increment(ref _snCounter);
  570. return $"{prefix}-{workOrder.MaterialCode}-{timestamp}-{counter:D4}-{randomStr}";
  571. }
  572. private string GetPrintTypePrefix(int printType)
  573. {
  574. switch ( printType )
  575. {
  576. case 1:
  577. return "P";
  578. case 2:
  579. return "C";
  580. case 3:
  581. return "Q";
  582. case 4:
  583. return "B";
  584. default:
  585. return "U";
  586. }
  587. }
  588. private string GetPrintTypeName(int printType)
  589. {
  590. switch ( printType )
  591. {
  592. case 1:
  593. return "产品码";
  594. case 2:
  595. return "零件码";
  596. case 3:
  597. return "合格码";
  598. case 4:
  599. return "装箱码";
  600. default:
  601. return "未知";
  602. }
  603. }
  604. #endregion
  605. }
  606. }