StateMachineMessageCutter.cs 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. using Edge.Core.Processor.Communicator;
  2. using System;
  3. using System.Collections.Generic;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. namespace HengshanPaymentTerminal
  8. {
  9. public class StateMachineMessageCutter : IMessageCutter<byte[]>
  10. {
  11. #region Internal State enum
  12. private enum State
  13. {
  14. Uninitialized,
  15. LengthReady,
  16. BodyReady,
  17. EtxReady,
  18. CrcReady
  19. }
  20. #endregion
  21. #region Fields
  22. public byte[] Message { get; private set; }
  23. public event EventHandler OnMessageCut;
  24. public event EventHandler<MessageCutterInvalidMessageReadEventArg> OnInvalidMessageRead;
  25. private string loggerAppendix = "HengshanPay Terminal";
  26. private readonly SizableWindow<byte> window;
  27. //private State nextState = State.Uninitialized;
  28. private const int STX = 0xFA;
  29. #endregion
  30. #region Logger
  31. static NLog.Logger innerLogger = NLog.LogManager.LoadConfiguration("nlog.config").GetLogger("Communicator");
  32. #endregion
  33. public StateMachineMessageCutter()
  34. {
  35. window = new SizableWindow<byte>();
  36. window.OnWindowFull += (data, localTcpServerListeningPort) =>
  37. {
  38. byte[] tempBytes = window.ToArray();
  39. string receiveStr = BitConverter.ToString(tempBytes).Replace("-", " ");
  40. innerLogger.Info($"receive data : ${receiveStr}");
  41. //查找包头位置
  42. int findIndex = -1;
  43. for (int index = 0;index < tempBytes.Length - 1;index++)
  44. {
  45. if (tempBytes[index] == STX && tempBytes[index + 1] != STX)
  46. {
  47. findIndex = index;
  48. break;
  49. }
  50. }
  51. innerLogger.Info($"find head index = ${findIndex}");
  52. //若没有找到包头,证明目前缓存中都是垃圾数据,清掉
  53. if (findIndex == -1)
  54. {
  55. window.RemoveRange(0, tempBytes.Length);
  56. return;
  57. }
  58. //长度不够表示有粘包,部分数据还未接收到,等待下一轮接收数据再处理
  59. //1byte FA + 1 byte 目标地址 + 1 byte 源地址 + 1 byte 帧号 + 2 byte 有效数据长度 + 2 byte 数据校验 = 8byte
  60. if(findIndex + 8 > tempBytes.Length)
  61. {
  62. innerLogger.Info($"Insufficient data length,at least ,need {findIndex + 8},but length is {tempBytes.Length}");
  63. return;
  64. }
  65. //获取有效数据长度
  66. int dataLen = Bcd2Int(tempBytes[4], tempBytes[5]);
  67. byte[] reduceFAData = Reduce0xFAPair(tempBytes);
  68. string reduceFADataStr = BitConverter.ToString(reduceFAData).Replace("-", " ");
  69. innerLogger.Info($"data length is ${dataLen},reduceFAData len is ${reduceFAData.Length},value is ${reduceFADataStr}");
  70. //去除多余FA后,实际数据不足有效数据长度,表示有粘包,仍非完整数据,等待下一轮接收数据再处理
  71. if(reduceFAData.Length < dataLen + 2)
  72. {
  73. innerLogger.Info("Insufficient data length");
  74. return;
  75. }
  76. //计算crc,若值对应不上,表示数据错误
  77. ushort calculatorValue = HengshanCRC16.ComputeChecksum(reduceFAData.AsSpan(1,reduceFAData.Length - 3).ToArray());
  78. byte highByte = reduceFAData[reduceFAData.Length - 2];
  79. byte lowByte = reduceFAData[reduceFAData.Length - 1];
  80. int crcValue = (highByte << 8) | lowByte;
  81. if (calculatorValue != crcValue)
  82. {
  83. innerLogger.Info($"crc value error,get value is {crcValue},calculator value is {calculatorValue}");
  84. window.RemoveRange(0, tempBytes.Length);
  85. return;
  86. }
  87. //获取到实际数据包,并将缓存数据删除
  88. Message = reduceFAData;
  89. var safe = OnMessageCut;
  90. safe?.Invoke(this, null);
  91. window.RemoveRange(0, tempBytes.Length);
  92. };
  93. }
  94. public void Feed(byte[] data,int localTcpServerListeningPort = 0)
  95. {
  96. List<byte> list = new(data);
  97. window.Add(list, localTcpServerListeningPort);
  98. //for (int i = 0; i < data.Length; i++)
  99. //{
  100. // window.Add(data[i]);
  101. //}
  102. }
  103. private int Get0xFAPairCountInWindow(IEnumerable<byte> data)
  104. {
  105. return (int)Math.Round(((double)(data.Count(w => w == 0xFA)) / 2), MidpointRounding.AwayFromZero);
  106. }
  107. /// <summary>
  108. /// 删除多余的FA,在接收到的一个完整数据包中,数据包头为FA,其余数据中每一个FA会替换为两个FA,这里要删除多余的fa来获取实际数据包
  109. /// </summary>
  110. /// <param name="data">原数据包</param>
  111. /// <returns></returns>
  112. public byte[] Reduce0xFAPair(byte[] data)
  113. {
  114. List<byte> list = new();
  115. for (int index = 0; index < data.Length; index++)
  116. {
  117. if (index < data.Length -1 && data[index] == STX && data[index + 1] == STX)
  118. {
  119. list.Add(STX);
  120. index++;
  121. }
  122. else
  123. {
  124. list.Add(data[index]);
  125. }
  126. }
  127. return list.ToArray();
  128. }
  129. public int Bcd2Int(byte byte1,byte byte2)
  130. {
  131. // 提取第一个字节的高四位和低四位
  132. int digit1 = (byte1 >> 4) & 0x0F; // 高四位
  133. int digit2 = byte1 & 0x0F; // 低四位
  134. // 提取第二个字节的高四位和低四位
  135. int digit3 = (byte2 >> 4) & 0x0F; // 高四位
  136. int digit4 = byte2 & 0x0F; // 低四位
  137. // 组合成一个整数
  138. int result = digit1 * 1000 + digit2 * 100 + digit3 * 10 + digit4;
  139. return result;
  140. }
  141. }
  142. }