HTTP2FrameHelper.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385
  1. #if (!UNITY_WEBGL || UNITY_EDITOR) && !BESTHTTP_DISABLE_ALTERNATE_SSL && !BESTHTTP_DISABLE_HTTP2
  2. using BestHTTP.Extensions;
  3. using BestHTTP.PlatformSupport.Memory;
  4. using System;
  5. using System.Collections.Generic;
  6. using System.IO;
  7. namespace BestHTTP.Connections.HTTP2
  8. {
  9. // https://httpwg.org/specs/rfc7540.html#ErrorCodes
  10. public enum HTTP2ErrorCodes
  11. {
  12. NO_ERROR = 0x00,
  13. PROTOCOL_ERROR = 0x01,
  14. INTERNAL_ERROR = 0x02,
  15. FLOW_CONTROL_ERROR = 0x03,
  16. SETTINGS_TIMEOUT = 0x04,
  17. STREAM_CLOSED = 0x05,
  18. FRAME_SIZE_ERROR = 0x06,
  19. REFUSED_STREAM = 0x07,
  20. CANCEL = 0x08,
  21. COMPRESSION_ERROR = 0x09,
  22. CONNECT_ERROR = 0x0A,
  23. ENHANCE_YOUR_CALM = 0x0B,
  24. INADEQUATE_SECURITY = 0x0C,
  25. HTTP_1_1_REQUIRED = 0x0D
  26. }
  27. public static class HTTP2FrameHelper
  28. {
  29. public static HTTP2ContinuationFrame ReadContinuationFrame(HTTP2FrameHeaderAndPayload header)
  30. {
  31. // https://httpwg.org/specs/rfc7540.html#CONTINUATION
  32. HTTP2ContinuationFrame frame = new HTTP2ContinuationFrame(header);
  33. frame.HeaderBlockFragment = header.Payload;
  34. header.Payload = null;
  35. return frame;
  36. }
  37. public static HTTP2WindowUpdateFrame ReadWindowUpdateFrame(HTTP2FrameHeaderAndPayload header)
  38. {
  39. // https://httpwg.org/specs/rfc7540.html#WINDOW_UPDATE
  40. HTTP2WindowUpdateFrame frame = new HTTP2WindowUpdateFrame(header);
  41. frame.ReservedBit = BufferHelper.ReadBit(header.Payload[0], 0);
  42. frame.WindowSizeIncrement = BufferHelper.ReadUInt31(header.Payload, 0);
  43. return frame;
  44. }
  45. public static HTTP2GoAwayFrame ReadGoAwayFrame(HTTP2FrameHeaderAndPayload header)
  46. {
  47. // https://httpwg.org/specs/rfc7540.html#GOAWAY
  48. // str id error
  49. // | 0, 1, 2, 3 | 4, 5, 6, 7 | ...
  50. HTTP2GoAwayFrame frame = new HTTP2GoAwayFrame(header);
  51. frame.ReservedBit = BufferHelper.ReadBit(header.Payload[0], 0);
  52. frame.LastStreamId = BufferHelper.ReadUInt31(header.Payload, 0);
  53. frame.ErrorCode = BufferHelper.ReadUInt32(header.Payload, 4);
  54. frame.AdditionalDebugDataLength = header.PayloadLength - 8;
  55. if (frame.AdditionalDebugDataLength > 0)
  56. {
  57. frame.AdditionalDebugData = BufferPool.Get(frame.AdditionalDebugDataLength, true);
  58. Array.Copy(header.Payload, 8, frame.AdditionalDebugData, 0, frame.AdditionalDebugDataLength);
  59. }
  60. return frame;
  61. }
  62. public static HTTP2PingFrame ReadPingFrame(HTTP2FrameHeaderAndPayload header)
  63. {
  64. // https://httpwg.org/specs/rfc7540.html#PING
  65. HTTP2PingFrame frame = new HTTP2PingFrame(header);
  66. Array.Copy(header.Payload, 0, frame.OpaqueData, 0, frame.OpaqueDataLength);
  67. return frame;
  68. }
  69. public static HTTP2PushPromiseFrame ReadPush_PromiseFrame(HTTP2FrameHeaderAndPayload header)
  70. {
  71. // https://httpwg.org/specs/rfc7540.html#PUSH_PROMISE
  72. HTTP2PushPromiseFrame frame = new HTTP2PushPromiseFrame(header);
  73. frame.HeaderBlockFragmentLength = header.PayloadLength - 4; // PromisedStreamId
  74. bool isPadded = (frame.Flags & HTTP2PushPromiseFlags.PADDED) != 0;
  75. if (isPadded)
  76. {
  77. frame.PadLength = header.Payload[0];
  78. frame.HeaderBlockFragmentLength -= (uint)(1 + (frame.PadLength ?? 0));
  79. }
  80. frame.ReservedBit = BufferHelper.ReadBit(header.Payload[1], 0);
  81. frame.PromisedStreamId = BufferHelper.ReadUInt31(header.Payload, 1);
  82. frame.HeaderBlockFragmentIdx = (UInt32)(isPadded ? 5 : 4);
  83. frame.HeaderBlockFragment = header.Payload;
  84. header.Payload = null;
  85. return frame;
  86. }
  87. public static HTTP2RSTStreamFrame ReadRST_StreamFrame(HTTP2FrameHeaderAndPayload header)
  88. {
  89. // https://httpwg.org/specs/rfc7540.html#RST_STREAM
  90. HTTP2RSTStreamFrame frame = new HTTP2RSTStreamFrame(header);
  91. frame.ErrorCode = BufferHelper.ReadUInt32(header.Payload, 0);
  92. return frame;
  93. }
  94. public static HTTP2PriorityFrame ReadPriorityFrame(HTTP2FrameHeaderAndPayload header)
  95. {
  96. // https://httpwg.org/specs/rfc7540.html#PRIORITY
  97. if (header.PayloadLength != 5)
  98. {
  99. //throw FRAME_SIZE_ERROR
  100. }
  101. HTTP2PriorityFrame frame = new HTTP2PriorityFrame(header);
  102. frame.IsExclusive = BufferHelper.ReadBit(header.Payload[0], 0);
  103. frame.StreamDependency = BufferHelper.ReadUInt31(header.Payload, 0);
  104. frame.Weight = header.Payload[4];
  105. return frame;
  106. }
  107. public static HTTP2HeadersFrame ReadHeadersFrame(HTTP2FrameHeaderAndPayload header)
  108. {
  109. // https://httpwg.org/specs/rfc7540.html#HEADERS
  110. HTTP2HeadersFrame frame = new HTTP2HeadersFrame(header);
  111. frame.HeaderBlockFragmentLength = header.PayloadLength;
  112. bool isPadded = (frame.Flags & HTTP2HeadersFlags.PADDED) != 0;
  113. bool isPriority = (frame.Flags & HTTP2HeadersFlags.PRIORITY) != 0;
  114. int payloadIdx = 0;
  115. if (isPadded)
  116. {
  117. frame.PadLength = header.Payload[payloadIdx++];
  118. uint subLength = (uint)(1 + (frame.PadLength ?? 0));
  119. if (subLength <= frame.HeaderBlockFragmentLength)
  120. frame.HeaderBlockFragmentLength -= subLength;
  121. //else
  122. // throw PROTOCOL_ERROR;
  123. }
  124. if (isPriority)
  125. {
  126. frame.IsExclusive = BufferHelper.ReadBit(header.Payload[payloadIdx], 0);
  127. frame.StreamDependency = BufferHelper.ReadUInt31(header.Payload, payloadIdx);
  128. payloadIdx += 4;
  129. frame.Weight = header.Payload[payloadIdx++];
  130. uint subLength = 5;
  131. if (subLength <= frame.HeaderBlockFragmentLength)
  132. frame.HeaderBlockFragmentLength -= subLength;
  133. //else
  134. // throw PROTOCOL_ERROR;
  135. }
  136. frame.HeaderBlockFragmentIdx = (UInt32)payloadIdx;
  137. frame.HeaderBlockFragment = header.Payload;
  138. return frame;
  139. }
  140. public static HTTP2DataFrame ReadDataFrame(HTTP2FrameHeaderAndPayload header)
  141. {
  142. // https://httpwg.org/specs/rfc7540.html#DATA
  143. HTTP2DataFrame frame = new HTTP2DataFrame(header);
  144. frame.DataLength = header.PayloadLength;
  145. bool isPadded = (frame.Flags & HTTP2DataFlags.PADDED) != 0;
  146. if (isPadded)
  147. {
  148. frame.PadLength = header.Payload[0];
  149. uint subLength = (uint)(1 + (frame.PadLength ?? 0));
  150. if (subLength <= frame.DataLength)
  151. frame.DataLength -= subLength;
  152. //else
  153. // throw PROTOCOL_ERROR;
  154. }
  155. frame.DataIdx = (UInt32)(isPadded ? 1 : 0);
  156. frame.Data = header.Payload;
  157. header.Payload = null;
  158. return frame;
  159. }
  160. public static HTTP2AltSVCFrame ReadAltSvcFrame(HTTP2FrameHeaderAndPayload header)
  161. {
  162. HTTP2AltSVCFrame frame = new HTTP2AltSVCFrame(header);
  163. // Implement
  164. return frame;
  165. }
  166. public static void StreamRead(Stream stream, byte[] buffer, int offset, uint count)
  167. {
  168. if (count == 0)
  169. return;
  170. uint sumRead = 0;
  171. do
  172. {
  173. int readCount = (int)(count - sumRead);
  174. int streamReadCount = stream.Read(buffer, (int)(offset + sumRead), readCount);
  175. if (streamReadCount <= 0 && readCount > 0)
  176. throw new Exception("TCP Stream closed!");
  177. sumRead += (uint)streamReadCount;
  178. } while (sumRead < count);
  179. }
  180. public static PooledBuffer HeaderAsBinary(HTTP2FrameHeaderAndPayload header)
  181. {
  182. // https://httpwg.org/specs/rfc7540.html#FrameHeader
  183. var buffer = BufferPool.Get(9, true);
  184. BufferHelper.SetUInt24(buffer, 0, header.PayloadLength);
  185. buffer[3] = (byte)header.Type;
  186. buffer[4] = header.Flags;
  187. BufferHelper.SetUInt31(buffer, 5, header.StreamId);
  188. return new PooledBuffer { Data = buffer, Length = 9 };
  189. }
  190. public static HTTP2FrameHeaderAndPayload ReadHeader(Stream stream)
  191. {
  192. byte[] buffer = BufferPool.Get(9, true);
  193. StreamRead(stream, buffer, 0, 9);
  194. HTTP2FrameHeaderAndPayload header = new HTTP2FrameHeaderAndPayload();
  195. header.PayloadLength = BufferHelper.ReadUInt24(buffer, 0);
  196. header.Type = (HTTP2FrameTypes)buffer[3];
  197. header.Flags = buffer[4];
  198. header.StreamId = BufferHelper.ReadUInt31(buffer, 5);
  199. BufferPool.Release(buffer);
  200. header.Payload = BufferPool.Get(header.PayloadLength, true);
  201. StreamRead(stream, header.Payload, 0, header.PayloadLength);
  202. return header;
  203. }
  204. public static HTTP2SettingsFrame ReadSettings(HTTP2FrameHeaderAndPayload header)
  205. {
  206. HTTP2SettingsFrame frame = new HTTP2SettingsFrame(header);
  207. if (header.PayloadLength > 0)
  208. {
  209. int kvpCount = (int)(header.PayloadLength / 6);
  210. frame.Settings = new List<KeyValuePair<HTTP2Settings, uint>>(kvpCount);
  211. for (int i = 0; i < kvpCount; ++i)
  212. {
  213. HTTP2Settings key = (HTTP2Settings)BufferHelper.ReadUInt16(header.Payload, i * 6);
  214. UInt32 value = BufferHelper.ReadUInt32(header.Payload, (i * 6) + 2);
  215. frame.Settings.Add(new KeyValuePair<HTTP2Settings, uint>(key, value));
  216. }
  217. }
  218. return frame;
  219. }
  220. public static HTTP2FrameHeaderAndPayload CreateACKSettingsFrame()
  221. {
  222. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  223. frame.Type = HTTP2FrameTypes.SETTINGS;
  224. frame.Flags = (byte)HTTP2SettingsFlags.ACK;
  225. return frame;
  226. }
  227. public static HTTP2FrameHeaderAndPayload CreateSettingsFrame(List<KeyValuePair<HTTP2Settings, UInt32>> settings)
  228. {
  229. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  230. frame.Type = HTTP2FrameTypes.SETTINGS;
  231. frame.Flags = 0;
  232. frame.PayloadLength = (UInt32)settings.Count * 6;
  233. frame.Payload = BufferPool.Get(frame.PayloadLength, true);
  234. for (int i = 0; i < settings.Count; ++i)
  235. {
  236. BufferHelper.SetUInt16(frame.Payload, i * 6, (UInt16)settings[i].Key);
  237. BufferHelper.SetUInt32(frame.Payload, (i * 6) + 2, settings[i].Value);
  238. }
  239. return frame;
  240. }
  241. public static HTTP2FrameHeaderAndPayload CreatePingFrame(HTTP2PingFlags flags = HTTP2PingFlags.None)
  242. {
  243. // https://httpwg.org/specs/rfc7540.html#PING
  244. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  245. frame.Type = HTTP2FrameTypes.PING;
  246. frame.Flags = (byte)flags;
  247. frame.StreamId = 0;
  248. frame.Payload = BufferPool.Get(8, true);
  249. frame.PayloadLength = 8;
  250. return frame;
  251. }
  252. public static HTTP2FrameHeaderAndPayload CreateWindowUpdateFrame(UInt32 streamId, UInt32 windowSizeIncrement)
  253. {
  254. // https://httpwg.org/specs/rfc7540.html#WINDOW_UPDATE
  255. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  256. frame.Type = HTTP2FrameTypes.WINDOW_UPDATE;
  257. frame.Flags = 0;
  258. frame.StreamId = streamId;
  259. frame.Payload = BufferPool.Get(4, true);
  260. frame.PayloadLength = 4;
  261. BufferHelper.SetBit(0, 0, 0);
  262. BufferHelper.SetUInt31(frame.Payload, 0, windowSizeIncrement);
  263. return frame;
  264. }
  265. public static HTTP2FrameHeaderAndPayload CreateGoAwayFrame(UInt32 lastStreamId, HTTP2ErrorCodes error)
  266. {
  267. // https://httpwg.org/specs/rfc7540.html#GOAWAY
  268. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  269. frame.Type = HTTP2FrameTypes.GOAWAY;
  270. frame.Flags = 0;
  271. frame.StreamId = 0;
  272. frame.Payload = BufferPool.Get(8, true);
  273. frame.PayloadLength = 8;
  274. BufferHelper.SetUInt31(frame.Payload, 0, lastStreamId);
  275. BufferHelper.SetUInt31(frame.Payload, 4, (UInt32)error);
  276. return frame;
  277. }
  278. public static HTTP2FrameHeaderAndPayload CreateRSTFrame(UInt32 streamId, HTTP2ErrorCodes errorCode)
  279. {
  280. // https://httpwg.org/specs/rfc7540.html#RST_STREAM
  281. HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
  282. frame.Type = HTTP2FrameTypes.RST_STREAM;
  283. frame.Flags = 0;
  284. frame.StreamId = streamId;
  285. frame.Payload = BufferPool.Get(4, true);
  286. frame.PayloadLength = 4;
  287. BufferHelper.SetUInt32(frame.Payload, 0, (UInt32)errorCode);
  288. return frame;
  289. }
  290. }
  291. }
  292. #endif