StateMachineMessageCutter.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293
  1. using System;
  2. using System.Linq;
  3. using Edge.Core.Processor;using Edge.Core.IndustryStandardInterface.Pump;
  4. using Edge.Core.Processor.Communicator;
  5. namespace Dfs.WayneChina.HengshanTerminalWrapper
  6. {
  7. public class StateMachineMessageCutter : IMessageCutter<byte[]>
  8. {
  9. public byte[] Message { get; private set; }
  10. public event EventHandler OnMessageCut;
  11. public event EventHandler<MessageCutterInvalidMessageReadEventArg> OnInvalidMessageRead;
  12. static NLog.Logger innerLogger = NLog.LogManager.LoadConfiguration("nlog.config").GetLogger("Communicator");
  13. private string loggerAppendix = "HengshanICTerm msgCutter ";
  14. private readonly SizableWindow<byte> window;
  15. private State nextState = State.Uninitialized;
  16. /// <summary>
  17. /// Constructor
  18. /// </summary>
  19. public StateMachineMessageCutter()
  20. {
  21. //this.initWindowSize = initWindowSize;
  22. this.window = new SizableWindow<byte>();
  23. this.window.OnWindowFull += (data) =>
  24. {
  25. switch (nextState)
  26. {
  27. case State.Uninitialized:
  28. if (data.First() == 0xFF)
  29. {
  30. //前导码(0FFH) +长度(0-20H)
  31. this.window.NewSize = 2;
  32. this.nextState = State.LengthReady;
  33. //innerLogger.Debug(this.loggerAppendix + " state is State.Uninitialized and next is 0xFF, switch to LengthReady");
  34. }
  35. else
  36. {
  37. this.OnInvalidMessageRead?.Invoke(this, new MessageCutterInvalidMessageReadEventArg()
  38. {
  39. Message = "invalid byte[0]: 0x" + data.First().ToString("x2") + ", will skip"
  40. });
  41. this.window.Clear();
  42. }
  43. break;
  44. case State.LengthReady:
  45. var lengthByte = this.window.Skip(1).First();
  46. if (lengthByte > 0x20)
  47. {
  48. this.OnInvalidMessageRead?.Invoke(this, new MessageCutterInvalidMessageReadEventArg()
  49. {
  50. Message = "invalid lengthByte: 0x" + lengthByte.ToString("x2") + ", should <=0x20, still keep it"
  51. });
  52. }
  53. // 1 is header 0xFF, another 1 is XRL
  54. this.window.NewSize = lengthByte + 2;
  55. this.nextState = State.BodyReady;
  56. break;
  57. case State.BodyReady:
  58. //innerLogger.Debug(this.loggerAppendix + " Fire OnMessageConstructed with innerQueue: " + this.buffer.ToHexLogString());
  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. //this.window.NewSize = this.initWindowSize;
  65. break;
  66. default:
  67. throw new ArgumentOutOfRangeException();
  68. }
  69. };
  70. }
  71. private enum State
  72. {
  73. Uninitialized,
  74. //HeaderSeeking,
  75. //HeaderReady,
  76. LengthReady,
  77. BodyReady,
  78. }
  79. public void Feed(byte[] next)
  80. {
  81. //innerLogger.Debug(this.loggerAppendix + " " + next.ToHexLogString() + " is feed in Window in state: " + nextState);
  82. for (int i = 0; i < next.Length; i++)
  83. this.window.Add(next[i]);
  84. }
  85. }
  86. }