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.7 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  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 = 10 * 1024 * 1024; // 10M
  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)
  77. return null;
  78. var pos = cos.Position;
  79. try
  80. {
  81. MessageToClient? msg = new MessageToClient();
  82. cos.ReadMessage(msg);
  83. return msg;
  84. }
  85. catch (InvalidProtocolBufferException)
  86. {
  87. var leftByte = buffer.Length - pos; // 上次读取剩余的字节
  88. if (buffer.Length < bufferMaxSize / 2)
  89. {
  90. var newBuffer = new byte[bufferMaxSize];
  91. for (int i = 0; i < leftByte; i++)
  92. {
  93. newBuffer[i] = buffer[pos + i];
  94. }
  95. buffer = newBuffer;
  96. }
  97. else
  98. {
  99. for (int i = 0; i < leftByte; ++i)
  100. {
  101. buffer[i] = buffer[pos + i];
  102. }
  103. }
  104. var bufferSize = gzs.Read(buffer, (int)leftByte, (int)(buffer.Length - leftByte)) + leftByte;
  105. if (bufferSize == leftByte)
  106. {
  107. Finished = true;
  108. return null;
  109. }
  110. if (bufferSize != buffer.Length) // 不留空位,防止 CodedInputStream 获取信息错误
  111. {
  112. var tmpBuffer = new byte[bufferSize];
  113. Array.Copy(buffer, tmpBuffer, bufferSize);
  114. buffer = tmpBuffer;
  115. }
  116. cos = new CodedInputStream(buffer);
  117. goto beginRead;
  118. }
  119. }
  120. public void Dispose()
  121. {
  122. Finished = true;
  123. if (fs == null)
  124. return;
  125. if (fs.CanRead)
  126. {
  127. fs.Close();
  128. }
  129. }
  130. ~MessageReader()
  131. {
  132. Dispose();
  133. }
  134. }
  135. }