PLCModel.cs 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732
  1. 
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Text;
  5. using System.Text.RegularExpressions;
  6. namespace Curtain.Net.Sockets.PLC.Model
  7. {
  8. /// <summary>
  9. /// SocketModel
  10. /// </summary>
  11. public interface ISocketModel
  12. {
  13. /// <summary>
  14. /// 接收通信超时(毫秒)
  15. /// </summary>
  16. int ReceiveTimeout
  17. {
  18. get;
  19. }
  20. /// <summary>
  21. /// 发送通信超时(毫秒)
  22. /// </summary>
  23. int SendTimeout
  24. {
  25. get;
  26. }
  27. /// <summary>
  28. /// 响应报文-头长度
  29. /// </summary>
  30. int HeadLength
  31. {
  32. get;
  33. }
  34. }
  35. /// <summary>
  36. /// PLC通信模型
  37. /// </summary>
  38. public interface IClientModel : ISocketModel
  39. {
  40. #region
  41. /// <summary>
  42. /// 初始化Socket连接
  43. /// </summary>
  44. /// <param name="socket"></param>
  45. void OnSocketConnect(PLCSocket socket);
  46. /// <summary>
  47. /// 根据响应报文-头,计算响应报文-文本长度
  48. /// </summary>
  49. /// <param name="m">报文</param>
  50. /// <returns>响应报文-文本长度</returns>
  51. int GetContentLength(PLCMessage m);
  52. /// <summary>
  53. /// 验证响应报文-头
  54. /// </summary>
  55. /// <param name="m">报文</param>
  56. /// <returns>是否通过</returns>
  57. bool CheckHead(PLCMessage m);
  58. /// <summary>
  59. /// 命令报文转换为发送字节
  60. /// </summary>
  61. /// <param name="command">命令报文</param>
  62. /// <returns>发送字节</returns>
  63. byte[] ToSendFromCommand(string command);
  64. /// <summary>
  65. /// 接收字节转换为响应报文
  66. /// </summary>
  67. /// <param name="receive">接收字节</param>
  68. /// <returns>响应报文</returns>
  69. string ToResponseFromReceive(byte[] receive);
  70. #endregion
  71. #region Value
  72. /// <summary>
  73. /// 设定读取的数据值
  74. /// </summary>
  75. /// <typeparam name="TValue"></typeparam>
  76. /// <param name="plc_m"></param>
  77. void SetReadValueFromBytes<TValue>(PLCMessage<TValue> plc_m);
  78. #endregion
  79. #region ReadMessage
  80. /// <summary>
  81. /// 获取读取命令报文
  82. /// </summary>
  83. /// <param name="code"></param>
  84. /// <param name="number"></param>
  85. /// <param name="length"></param>
  86. /// <returns></returns>
  87. PLCMessage<TValue> GetReadMessage<TValue>(char code, int number, int length = 1);
  88. /// <summary>
  89. /// 获取读取命令报文
  90. /// </summary>
  91. /// <param name="code"></param>
  92. /// <param name="number"></param>
  93. /// <param name="length"></param>
  94. /// <returns></returns>
  95. PLCMessage<TValue> GetReadMessage<TValue>(string code, string number, int length = 1);
  96. /// <summary>
  97. /// 获取读取命令报文
  98. /// </summary>
  99. /// <param name="code"></param>
  100. /// <param name="length"></param>
  101. /// <returns></returns>
  102. PLCMessage<TValue> GetReadMessage<TValue>(string code, int length = 1);
  103. #endregion
  104. #region WriteMessage
  105. /// <summary>
  106. /// 获取写入命令报文
  107. /// </summary>
  108. /// <typeparam name="TValue"></typeparam>
  109. /// <param name="code"></param>
  110. /// <param name="number"></param>
  111. /// <param name="value"></param>
  112. /// <param name="length"></param>
  113. /// <returns></returns>
  114. PLCMessage<TValue> GetWriteMessage<TValue>(char code, int number, TValue value, int length = 0);
  115. /// <summary>
  116. /// 获取写入命令报文
  117. /// </summary>
  118. /// <typeparam name="TValue"></typeparam>
  119. /// <param name="code"></param>
  120. /// <param name="number"></param>
  121. /// <param name="value"></param>
  122. /// <param name="length"></param>
  123. /// <returns></returns>
  124. PLCMessage<TValue> GetWriteMessage<TValue>(string code, string number, TValue value, int length = 0);
  125. /// <summary>
  126. /// 获取写入命令报文
  127. /// </summary>
  128. /// <typeparam name="TValue"></typeparam>
  129. /// <param name="code"></param>
  130. /// <param name="value"></param>
  131. /// <param name="length"></param>
  132. /// <returns></returns>
  133. PLCMessage<TValue> GetWriteMessage<TValue>(string code, TValue value, int length = 0);
  134. /// <summary>
  135. /// 获取写入命令报文
  136. /// </summary>
  137. /// <param name="code"></param>
  138. /// <param name="number"></param>
  139. /// <param name="value"></param>
  140. /// <param name="encoding"></param>
  141. /// <param name="length"></param>
  142. /// <returns></returns>
  143. PLCMessage<string> GetWriteMessage(char code, int number, string value, Encoding encoding, int length = 0);
  144. /// <summary>
  145. /// 获取写入命令报文
  146. /// </summary>
  147. /// <param name="code"></param>
  148. /// <param name="number"></param>
  149. /// <param name="value"></param>
  150. /// <param name="encoding"></param>
  151. /// <param name="length"></param>
  152. /// <returns></returns>
  153. PLCMessage<string> GetWriteMessage(string code, string number, string value, Encoding encoding, int length = 0);
  154. /// <summary>
  155. /// 获取写入命令报文
  156. /// </summary>
  157. /// <param name="code"></param>
  158. /// <param name="value"></param>
  159. /// <param name="encoding"></param>
  160. /// <param name="length"></param>
  161. /// <returns></returns>
  162. PLCMessage<string> GetWriteMessage(string code, string value, Encoding encoding, int length = 0);
  163. #endregion
  164. #region SendMessage
  165. /// <summary>
  166. /// 获取发送命令报文
  167. /// </summary>
  168. /// <param name="value"></param>
  169. /// <returns></returns>
  170. PLCMessage<string> GetSendMessage(string value);
  171. /// <summary>
  172. /// 获取发送命令报文
  173. /// </summary>
  174. /// <param name="value"></param>
  175. /// <param name="encoding"></param>
  176. /// <returns></returns>
  177. PLCMessage<string> GetSendMessage(string value, Encoding encoding);
  178. #endregion
  179. }
  180. /// <summary>
  181. /// PLC通信模型
  182. /// </summary>
  183. public abstract class ClientModel : IClientModel
  184. {
  185. #region factory
  186. private static readonly Dictionary<Type, IClientModel> _factory = new Dictionary<Type, IClientModel>();
  187. /// <summary>
  188. /// 创建PLC通信模型
  189. /// </summary>
  190. /// <typeparam name="T"></typeparam>
  191. /// <returns></returns>
  192. public static T CreateModel<T>()
  193. where T: IClientModel, new()
  194. {
  195. System.Type t = typeof(T);
  196. if (_factory.ContainsKey(t))
  197. {
  198. return (T)_factory[t];
  199. }
  200. T tt = new T();
  201. _factory.Add(t, tt);
  202. return tt;
  203. }
  204. #endregion
  205. #region client
  206. /// <summary>
  207. /// 接收通信超时(毫秒)
  208. /// </summary>
  209. public virtual int ReceiveTimeout => 3000;
  210. /// <summary>
  211. /// 发送通信超时(毫秒)
  212. /// </summary>
  213. public virtual int SendTimeout => 3000;
  214. /// <summary>
  215. /// 响应报文-头长度
  216. /// </summary>
  217. public virtual int HeadLength => 0;
  218. /// <summary>
  219. /// 初始化Socket连接
  220. /// </summary>
  221. /// <param name="socket"></param>
  222. public virtual void OnSocketConnect(PLCSocket socket)
  223. {
  224. }
  225. /// <summary>
  226. /// 根据响应报文-头,计算响应报文-文本长度
  227. /// </summary>
  228. /// <param name="m">报文</param>
  229. /// <returns>响应报文-文本长度</returns>
  230. public virtual int GetContentLength(PLCMessage m)
  231. {
  232. return 0;
  233. }
  234. /// <summary>
  235. /// 验证响应报文-头
  236. /// </summary>
  237. /// <param name="m">报文</param>
  238. /// <returns>是否通过</returns>
  239. public virtual bool CheckHead(PLCMessage m)
  240. {
  241. return false;
  242. }
  243. /// <summary>
  244. /// 命令报文转换为发送字节
  245. /// </summary>
  246. /// <param name="command">命令报文</param>
  247. /// <returns>发送字节</returns>
  248. public virtual byte[] ToSendFromCommand(string command)
  249. {
  250. return null;
  251. }
  252. /// <summary>
  253. /// 接收字节转换为响应报文
  254. /// </summary>
  255. /// <param name="receive">接收字节</param>
  256. /// <returns>响应报文</returns>
  257. public virtual string ToResponseFromReceive(byte[] receive)
  258. {
  259. return null;
  260. }
  261. #endregion
  262. #region Value
  263. /// <summary>
  264. /// 设定读取的数据值
  265. /// </summary>
  266. /// <typeparam name="TValue"></typeparam>
  267. /// <param name="plc_m"></param>
  268. public virtual void SetReadValueFromBytes<TValue>(PLCMessage<TValue> plc_m)
  269. {
  270. throw new NotSupportedException(typeof(TValue).Name);
  271. }
  272. #endregion
  273. #region ReadMessage
  274. /// <summary>
  275. /// 获取读取命令报文
  276. /// </summary>
  277. /// <param name="code"></param>
  278. /// <param name="number"></param>
  279. /// <param name="length"></param>
  280. /// <returns></returns>
  281. public virtual PLCMessage<TValue> GetReadMessage<TValue>(char code, int number, int length = 1)
  282. {
  283. return GetReadMessage<TValue>(code.ToString(), number.ToString(), length);
  284. }
  285. /// <summary>
  286. /// 获取读取命令报文
  287. /// </summary>
  288. /// <param name="code"></param>
  289. /// <param name="number"></param>
  290. /// <param name="length"></param>
  291. /// <returns></returns>
  292. public virtual PLCMessage<TValue> GetReadMessage<TValue>(string code, string number, int length = 1)
  293. {
  294. throw new NotSupportedException($"[{typeof(TValue).Name}][{code}{number}][{length}]");
  295. }
  296. /// <summary>
  297. /// 获取读取命令报文
  298. /// </summary>
  299. /// <param name="code"></param>
  300. /// <param name="length"></param>
  301. /// <returns></returns>
  302. public virtual PLCMessage<TValue> GetReadMessage<TValue>(string code, int length = 1)
  303. {
  304. string strPatten = @"([a-zA-Z]+)([~a-zA-Z]+)";
  305. Regex rex = new Regex(strPatten);
  306. MatchCollection matches = rex.Matches(code);
  307. if (matches.Count > 0)
  308. {
  309. return GetReadMessage<TValue>(matches[0].Groups[0].Value, matches[0].Groups[1].Value, length);
  310. }
  311. throw new NotSupportedException($"[{typeof(TValue).Name}][{code}][{length}]");
  312. }
  313. #endregion
  314. #region WriteMessage
  315. /// <summary>
  316. /// 获取写入命令报文
  317. /// </summary>
  318. /// <typeparam name="TValue"></typeparam>
  319. /// <param name="code"></param>
  320. /// <param name="number"></param>
  321. /// <param name="value"></param>
  322. /// <param name="length"></param>
  323. /// <returns></returns>
  324. public virtual PLCMessage<TValue> GetWriteMessage<TValue>(char code, int number, TValue value, int length = 0)
  325. {
  326. return GetWriteMessage(code.ToString(), number.ToString(), value);
  327. }
  328. /// <summary>
  329. /// 获取写入命令报文
  330. /// </summary>
  331. /// <typeparam name="TValue"></typeparam>
  332. /// <param name="code"></param>
  333. /// <param name="number"></param>
  334. /// <param name="value"></param>
  335. /// <param name="length"></param>
  336. /// <returns></returns>
  337. public virtual PLCMessage<TValue> GetWriteMessage<TValue>(string code, string number, TValue value, int length = 0)
  338. {
  339. throw new NotSupportedException($"[{typeof(TValue).Name}][{code}{number}][{value}]");
  340. }
  341. /// <summary>
  342. /// 获取写入命令报文
  343. /// </summary>
  344. /// <typeparam name="TValue"></typeparam>
  345. /// <param name="code"></param>
  346. /// <param name="value"></param>
  347. /// <param name="length"></param>
  348. /// <returns></returns>
  349. public virtual PLCMessage<TValue> GetWriteMessage<TValue>(string code, TValue value, int length = 0)
  350. {
  351. string strPatten = @"([a-zA-Z]+)([~a-zA-Z]+)";
  352. Regex rex = new Regex(strPatten);
  353. MatchCollection matches = rex.Matches(code);
  354. if (matches.Count > 0)
  355. {
  356. return GetWriteMessage<TValue>(matches[0].Groups[0].Value, matches[0].Groups[1].Value, value);
  357. }
  358. throw new NotSupportedException($"{typeof(TValue).Name}[{code}]");
  359. }
  360. /// <summary>
  361. /// 获取写入命令报文
  362. /// </summary>
  363. /// <param name="code"></param>
  364. /// <param name="number"></param>
  365. /// <param name="value"></param>
  366. /// <param name="encoding"></param>
  367. /// <param name="length"></param>
  368. /// <returns></returns>
  369. public virtual PLCMessage<string> GetWriteMessage(char code, int number, string value, Encoding encoding, int length = 0)
  370. {
  371. return GetWriteMessage(code.ToString(), number.ToString(), value, encoding);
  372. }
  373. /// <summary>
  374. /// 获取写入命令报文
  375. /// </summary>
  376. /// <param name="code"></param>
  377. /// <param name="number"></param>
  378. /// <param name="value"></param>
  379. /// <param name="encoding"></param>
  380. /// <param name="length"></param>
  381. /// <returns></returns>
  382. public virtual PLCMessage<string> GetWriteMessage(string code, string number, string value, Encoding encoding, int length = 0)
  383. {
  384. throw new NotSupportedException($"[{encoding.EncodingName}][{code}{number}][{value}]");
  385. }
  386. /// <summary>
  387. /// 获取写入命令报文
  388. /// </summary>
  389. /// <param name="code"></param>
  390. /// <param name="value"></param>
  391. /// <param name="encoding"></param>
  392. /// <param name="length"></param>
  393. /// <returns></returns>
  394. public virtual PLCMessage<string> GetWriteMessage(string code, string value, Encoding encoding, int length = 0)
  395. {
  396. string strPatten = @"([a-zA-Z]+)([~a-zA-Z]+)";
  397. Regex rex = new Regex(strPatten);
  398. MatchCollection matches = rex.Matches(code);
  399. if (matches.Count > 0)
  400. {
  401. return GetWriteMessage(matches[0].Groups[0].Value, matches[0].Groups[1].Value, value, encoding);
  402. }
  403. throw new NotSupportedException($"[{encoding.EncodingName}][{code}][{value}]");
  404. }
  405. #endregion
  406. #region SendMessage
  407. /// <summary>
  408. /// 获取发送命令报文
  409. /// </summary>
  410. /// <param name="value"></param>
  411. /// <returns></returns>
  412. public virtual PLCMessage<string> GetSendMessage(string value)
  413. {
  414. throw new NotSupportedException();
  415. }
  416. /// <summary>
  417. /// 获取发送命令报文
  418. /// </summary>
  419. /// <param name="value"></param>
  420. /// <param name="encoding"></param>
  421. /// <returns></returns>
  422. public virtual PLCMessage<string> GetSendMessage(string value, Encoding encoding)
  423. {
  424. throw new NotSupportedException();
  425. }
  426. #endregion
  427. }
  428. ///// <summary>
  429. /////
  430. ///// </summary>
  431. ///// <typeparam name="TModel"></typeparam>
  432. //public abstract class PLCModel<TModel> : PLCModel
  433. // where TModel : IPLCModel, new()
  434. //{
  435. // /// <summary>
  436. // ///
  437. // /// </summary>
  438. // public ClientSocket<TModel> ClientSocket
  439. // {
  440. // get;
  441. // internal set;
  442. // }
  443. //}
  444. /// <summary>
  445. /// 命令格式
  446. /// </summary>
  447. public enum CommandFormatType
  448. {
  449. /// <summary>
  450. /// PLC命令格式
  451. /// </summary>
  452. PLCCommand = 1,
  453. /// <summary>
  454. /// 起止符格式
  455. /// </summary>
  456. StartStopChar = 2,
  457. }
  458. /// <summary>
  459. /// 服务端模型
  460. /// </summary>
  461. public interface IServerModel : ISocketModel
  462. {
  463. /// <summary>
  464. /// 命令格式
  465. /// </summary>
  466. CommandFormatType FormatType
  467. {
  468. get;
  469. set;
  470. }
  471. /// <summary>
  472. /// 开始字符
  473. /// </summary>
  474. char StartChar
  475. {
  476. get;
  477. set;
  478. }
  479. /// <summary>
  480. /// 结束字符
  481. /// </summary>
  482. char StopChar
  483. {
  484. get;
  485. set;
  486. }
  487. /// <summary>
  488. /// 转义字符
  489. /// </summary>
  490. char EscapeChar
  491. {
  492. get;
  493. set;
  494. }
  495. /// <summary>
  496. /// 根据响应报文-头,计算响应报文-文本长度
  497. /// </summary>
  498. /// <param name="rs">报文</param>
  499. /// <returns>响应报文-文本长度</returns>
  500. int GetContentLength(ReceiveSession rs);
  501. /// <summary>
  502. /// 验证响应报文-头
  503. /// </summary>
  504. /// <param name="rs">报文</param>
  505. /// <returns>是否通过</returns>
  506. bool CheckHead(ReceiveSession rs);
  507. /// <summary>
  508. /// 接收字节转换为命令
  509. /// </summary>
  510. /// <param name="receive">接收字节</param>
  511. /// <returns>响应报文</returns>
  512. string ToCommandFromReceive(byte[] receive);
  513. /// <summary>
  514. /// 接收字节转换为报文头
  515. /// </summary>
  516. /// <param name="receive">接收字节</param>
  517. /// <returns>响应报文</returns>
  518. string ToHeadFromReceive(byte[] receive);
  519. /// <summary>
  520. /// 接收字节转换为报文正文
  521. /// </summary>
  522. /// <param name="receive">接收字节</param>
  523. /// <returns>响应报文</returns>
  524. string ToContentFromReceive(byte[] receive);
  525. /// <summary>
  526. /// 命令报文转换为发送字节
  527. /// </summary>
  528. /// <param name="command">命令报文</param>
  529. /// <returns>发送字节</returns>
  530. byte[] ToSendFromCommand(string command);
  531. }
  532. /// <summary>
  533. /// 服务端模型
  534. /// </summary>
  535. public abstract class ServerModel : IServerModel
  536. {
  537. #region factory
  538. private static readonly Dictionary<Type, IServerModel> _factory = new Dictionary<Type, IServerModel>();
  539. /// <summary>
  540. /// 创建服务端模型
  541. /// </summary>
  542. /// <typeparam name="T"></typeparam>
  543. /// <returns></returns>
  544. public static T CreateModel<T>()
  545. where T : IServerModel, new()
  546. {
  547. System.Type t = typeof(T);
  548. if (_factory.ContainsKey(t))
  549. {
  550. return (T)_factory[t];
  551. }
  552. T tt = new T();
  553. _factory.Add(t, tt);
  554. return tt;
  555. }
  556. #endregion
  557. /// <summary>
  558. /// 命令格式
  559. /// </summary>
  560. public virtual CommandFormatType FormatType
  561. {
  562. get;
  563. set;
  564. } = CommandFormatType.PLCCommand;
  565. /// <summary>
  566. /// 开始字符
  567. /// </summary>
  568. public virtual char StartChar
  569. {
  570. get;
  571. set;
  572. } = '@';
  573. /// <summary>
  574. /// 结束字符
  575. /// </summary>
  576. public virtual char StopChar
  577. {
  578. get;
  579. set;
  580. } = '$';
  581. /// <summary>
  582. /// 转义字符
  583. /// </summary>
  584. public virtual char EscapeChar
  585. {
  586. get;
  587. set;
  588. } = '\\';
  589. #region server
  590. /// <summary>
  591. /// 接收通信超时(毫秒)
  592. /// </summary>
  593. public virtual int ReceiveTimeout => 3000;
  594. /// <summary>
  595. /// 发送通信超时(毫秒)
  596. /// </summary>
  597. public virtual int SendTimeout => 3000;
  598. /// <summary>
  599. /// 响应报文-头长度
  600. /// </summary>
  601. public abstract int HeadLength
  602. {
  603. get;
  604. }
  605. /// <summary>
  606. /// 根据响应报文-头,计算响应报文-文本长度
  607. /// </summary>
  608. /// <param name="rs">报文</param>
  609. /// <returns>响应报文-文本长度</returns>
  610. public virtual int GetContentLength(ReceiveSession rs)
  611. {
  612. return -1;
  613. }
  614. /// <summary>
  615. /// 验证响应报文-头
  616. /// </summary>
  617. /// <param name="rs">报文</param>
  618. /// <returns>是否通过</returns>
  619. public virtual bool CheckHead(ReceiveSession rs)
  620. {
  621. return false;
  622. }
  623. /// <summary>
  624. /// 验证命令头
  625. /// </summary>
  626. /// <param name="headBytes"></param>
  627. /// <returns></returns>
  628. public virtual bool CheckHeadChar(byte[] headBytes)
  629. {
  630. char[] s = System.Text.Encoding.ASCII.GetChars(headBytes);
  631. return (s[0] == this.StartChar);
  632. }
  633. /// <summary>
  634. /// 接收字节转换为命令
  635. /// </summary>
  636. /// <param name="receive">接收字节</param>
  637. /// <returns>响应报文</returns>
  638. public virtual string ToCommandFromReceive(byte[] receive)
  639. {
  640. return null;
  641. }
  642. /// <summary>
  643. /// 接收字节转换为报文头
  644. /// </summary>
  645. /// <param name="receive">接收字节</param>
  646. /// <returns>响应报文</returns>
  647. public virtual string ToHeadFromReceive(byte[] receive)
  648. {
  649. return null;
  650. }
  651. /// <summary>
  652. /// 接收字节转换为报文正文
  653. /// </summary>
  654. /// <param name="receive">接收字节</param>
  655. /// <returns>响应报文</returns>
  656. public virtual string ToContentFromReceive(byte[] receive)
  657. {
  658. return null;
  659. }
  660. /// <summary>
  661. /// 命令报文转换为发送字节
  662. /// </summary>
  663. /// <param name="command">命令报文</param>
  664. /// <returns>发送字节</returns>
  665. public virtual byte[] ToSendFromCommand(string command)
  666. {
  667. return null;
  668. }
  669. #endregion
  670. }
  671. /// <summary>
  672. /// PLC地址信息
  673. /// </summary>
  674. public interface IAddressData
  675. {
  676. }
  677. }