StateMachineMessageCutter.cs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. using Edge.Core.Processor.Communicator;
  2. using System;
  3. using System.Linq;
  4. namespace GasConcentrations_Yt95h
  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 = "Yt95hEx_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.DumpWindowToQueue();
  49. this.nextState = State.BodyReady;
  50. this.window.NewSize += (data[2] + 2); // data[2] The number of data bytes to follow
  51. innerLogger.Debug($"{this.loggerAppendix} MsgBodyLen caculated with: {this.window.NewSize}");
  52. break;
  53. case State.DataAddressReady:
  54. this.nextState = State.BodyReady;
  55. this.window.NewSize += 4;
  56. innerLogger.Debug($"{this.loggerAppendix} MsgBodyLen caculated with: {this.window.NewSize}");
  57. break;
  58. case State.BodyReady:
  59. this.Message = this.window.ToArray();
  60. var safe = this.OnMessageCut;
  61. safe?.Invoke(this, null);
  62. this.nextState = State.Uninitialized;
  63. this.window.Clear();
  64. break;
  65. default:
  66. throw new ArgumentOutOfRangeException();
  67. }
  68. };
  69. }
  70. private enum State
  71. {
  72. Uninitialized,
  73. LengthReady,
  74. DataAddressReady,
  75. BodyReady,
  76. }
  77. private void ReadInvalidMessage()
  78. {
  79. this.Message = this.window.ToArray();
  80. var safeHandler = this.OnInvalidMessageRead;
  81. safeHandler?.Invoke(this, null);
  82. this.window.Clear();
  83. }
  84. public void Feed(byte[] next)
  85. {
  86. for (int i = 0; i < next.Length; i++)
  87. {
  88. window.Add(next[i]);
  89. }
  90. }
  91. }
  92. }