You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

MessageReader.cs 4.2 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134
  1. using Google.Protobuf;
  2. using Protobuf;
  3. using System;
  4. using System.IO;
  5. using System.IO.Compression;
  6. namespace Playback
  7. {
  8. public class FileFormatNotLegalException : Exception
  9. {
  10. private readonly string fileName;
  11. public FileFormatNotLegalException(string fileName)
  12. {
  13. this.fileName = fileName;
  14. }
  15. public override string Message => $"The file: " + this.fileName + " is not a legal playback file for THUAI6.";
  16. }
  17. public class MessageReader : IDisposable
  18. {
  19. private FileStream? fs;
  20. private CodedInputStream cos;
  21. private GZipStream gzs;
  22. private byte[] buffer;
  23. public bool Finished { get; private set; } = false;
  24. public readonly uint teamCount;
  25. public readonly uint playerCount;
  26. const int bufferMaxSize = 1024 * 1024; // 1M
  27. public MessageReader(string fileName)
  28. {
  29. if (!fileName.EndsWith(PlayBackConstant.ExtendedName))
  30. {
  31. fileName += PlayBackConstant.ExtendedName;
  32. }
  33. fs = new FileStream(fileName, FileMode.Open, FileAccess.Read);
  34. try
  35. {
  36. var prefixLen = PlayBackConstant.Prefix.Length;
  37. byte[] bt = new byte[prefixLen + sizeof(UInt32) * 2];
  38. fs.Read(bt, 0, bt.Length);
  39. for (int i = 0; i < prefixLen; ++i)
  40. {
  41. if (bt[i] != PlayBackConstant.Prefix[i]) throw new FileFormatNotLegalException(fileName);
  42. }
  43. teamCount = BitConverter.ToUInt32(bt, prefixLen);
  44. playerCount = BitConverter.ToUInt32(bt, prefixLen + sizeof(UInt32));
  45. }
  46. catch
  47. {
  48. throw new FileFormatNotLegalException(fileName);
  49. }
  50. gzs = new GZipStream(fs, CompressionMode.Decompress);
  51. var tmpBuffer = new byte[bufferMaxSize];
  52. var bufferSize = gzs.Read(tmpBuffer);
  53. if (bufferSize == 0)
  54. {
  55. buffer = tmpBuffer;
  56. Finished = true;
  57. }
  58. else if (bufferSize != bufferMaxSize) // 不留空位,防止 CodedInputStream 获取信息错误
  59. {
  60. if (bufferSize == 0)
  61. {
  62. Finished = true;
  63. }
  64. buffer = new byte[bufferSize];
  65. Array.Copy(tmpBuffer, buffer, bufferSize);
  66. }
  67. else
  68. {
  69. buffer = tmpBuffer;
  70. }
  71. cos = new CodedInputStream(buffer);
  72. }
  73. public MessageToClient? ReadOne()
  74. {
  75. beginRead:
  76. if (Finished) return null;
  77. var pos = cos.Position;
  78. try
  79. {
  80. MessageToClient? msg = new MessageToClient();
  81. cos.ReadMessage(msg);
  82. return msg;
  83. }
  84. catch (InvalidProtocolBufferException)
  85. {
  86. var leftByte = buffer.Length - pos; // 上次读取剩余的字节
  87. for (int i = 0; i < leftByte; ++i)
  88. {
  89. buffer[i] = buffer[pos + i];
  90. }
  91. var bufferSize = gzs.Read(buffer, (int)leftByte, (int)(buffer.Length - leftByte)) + leftByte;
  92. if (bufferSize == leftByte)
  93. {
  94. Finished = true;
  95. return null;
  96. }
  97. if (bufferSize != buffer.Length) // 不留空位,防止 CodedInputStream 获取信息错误
  98. {
  99. var tmpBuffer = new byte[bufferSize];
  100. Array.Copy(buffer, tmpBuffer, bufferSize);
  101. buffer = tmpBuffer;
  102. }
  103. cos = new CodedInputStream(buffer);
  104. goto beginRead;
  105. }
  106. }
  107. public void Dispose()
  108. {
  109. Finished = true;
  110. if (fs == null)
  111. return;
  112. if (fs.CanRead)
  113. {
  114. fs.Close();
  115. }
  116. }
  117. ~MessageReader()
  118. {
  119. Dispose();
  120. }
  121. }
  122. }