StateMachineMessageCutter.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. using Edge.Core.Processor.Communicator;
  2. using System;
  3. using System.Linq;
  4. namespace PressureGage_3051
  5. {
  6. public class StateMachineMessageCutter : IMessageCutter<byte[]>
  7. {
  8. public byte[] Message { get; private set; }
  9. public event EventHandler OnMessageCut;
  10. public event EventHandler<MessageCutterInvalidMessageReadEventArg> OnInvalidMessageRead;
  11. static NLog.Logger innerLogger = NLog.LogManager.LoadConfiguration("nlog.config").GetLogger("StateMachineMessageCutter");
  12. private string loggerAppendix = "Sensor_Handler msgCutter";
  13. private readonly SizableWindow<byte> window;
  14. private State nextState = State.Uninitialized;
  15. public StateMachineMessageCutter()
  16. {
  17. this.window = new SizableWindow<byte>(2);
  18. this.window.OnWindowFull += (data) =>
  19. {
  20. switch (nextState)
  21. {
  22. case State.Uninitialized:
  23. if (0 < data[0] && data[0] <= 255)
  24. {
  25. if (data[1] == 3 || data[1] == 4)
  26. {
  27. this.nextState = State.LengthReady;
  28. this.window.NewSize += 1;
  29. innerLogger.Debug($"{this.loggerAppendix} state is State.Uninitialized, will switch to LengthReady");
  30. }
  31. else if (data[1] == 6 || data[1] == 16)
  32. {
  33. this.nextState = State.DataAddressReady;
  34. this.window.NewSize += 2;
  35. innerLogger.Debug($"{this.loggerAppendix} state is State.Uninitialized, will switch to DataAddressReady");
  36. }
  37. else
  38. {
  39. ReadInvalidMessage();
  40. }
  41. }
  42. else
  43. {
  44. ReadInvalidMessage();
  45. }
  46. break;
  47. case State.LengthReady:
  48. this.nextState = State.BodyReady;
  49. this.window.NewSize += (data[2] + 2); // data[2] The number of data bytes to follow
  50. innerLogger.Debug($"{this.loggerAppendix} MsgBodyLen caculated with: {this.window.NewSize}");
  51. break;
  52. case State.DataAddressReady:
  53. this.nextState = State.BodyReady;
  54. this.window.NewSize += 4;
  55. innerLogger.Debug($"{this.loggerAppendix} MsgBodyLen caculated with: {this.window.NewSize}");
  56. break;
  57. case State.BodyReady:
  58. this.Message = this.window.ToArray();
  59. var safe = this.OnMessageCut;
  60. safe?.Invoke(this, null);
  61. this.nextState = State.Uninitialized;
  62. this.window.Clear();
  63. break;
  64. default:
  65. throw new ArgumentOutOfRangeException();
  66. }
  67. };
  68. }
  69. private enum State
  70. {
  71. Uninitialized,
  72. LengthReady,
  73. DataAddressReady,
  74. BodyReady,
  75. }
  76. private void ReadInvalidMessage()
  77. {
  78. this.Message = this.window.ToArray();
  79. var safeHandler = this.OnInvalidMessageRead;
  80. safeHandler?.Invoke(this, null);
  81. this.window.Clear();
  82. }
  83. public void Feed(byte[] next)
  84. {
  85. for (int i = 0; i < next.Length; i++)
  86. {
  87. window.Add(next[i]);
  88. }
  89. }
  90. }
  91. }