HTTP2Frames.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  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. namespace BestHTTP.Connections.HTTP2
  7. {
  8. // https://httpwg.org/specs/rfc7540.html#iana-frames
  9. public enum HTTP2FrameTypes : byte
  10. {
  11. DATA = 0x00,
  12. HEADERS = 0x01,
  13. PRIORITY = 0x02,
  14. RST_STREAM = 0x03,
  15. SETTINGS = 0x04,
  16. PUSH_PROMISE = 0x05,
  17. PING = 0x06,
  18. GOAWAY = 0x07,
  19. WINDOW_UPDATE = 0x08,
  20. CONTINUATION = 0x09,
  21. // https://tools.ietf.org/html/rfc7838#section-4
  22. ALT_SVC = 0x0A
  23. }
  24. [Flags]
  25. public enum HTTP2DataFlags : byte
  26. {
  27. None = 0x00,
  28. END_STREAM = 0x01,
  29. PADDED = 0x08,
  30. }
  31. [Flags]
  32. public enum HTTP2HeadersFlags : byte
  33. {
  34. None = 0x00,
  35. END_STREAM = 0x01,
  36. END_HEADERS = 0x04,
  37. PADDED = 0x08,
  38. PRIORITY = 0x20,
  39. }
  40. [Flags]
  41. public enum HTTP2SettingsFlags : byte
  42. {
  43. None = 0x00,
  44. ACK = 0x01,
  45. }
  46. [Flags]
  47. public enum HTTP2PushPromiseFlags : byte
  48. {
  49. None = 0x00,
  50. END_HEADERS = 0x04,
  51. PADDED = 0x08,
  52. }
  53. [Flags]
  54. public enum HTTP2PingFlags : byte
  55. {
  56. None = 0x00,
  57. ACK = 0x01,
  58. }
  59. [Flags]
  60. public enum HTTP2ContinuationFlags : byte
  61. {
  62. None = 0x00,
  63. END_HEADERS = 0x04,
  64. }
  65. public struct HTTP2FrameHeaderAndPayload
  66. {
  67. public UInt32 PayloadLength;
  68. public HTTP2FrameTypes Type;
  69. public byte Flags;
  70. public UInt32 StreamId;
  71. public byte[] Payload;
  72. public UInt32 PayloadOffset;
  73. public bool DontUseMemPool;
  74. public override string ToString()
  75. {
  76. return string.Format("[HTTP2FrameHeaderAndPayload Length: {0}, Type: {1}, Flags: {2}, StreamId: {3}, PayloadOffset: {4}, DontUseMemPool: {5}]",
  77. this.PayloadLength, this.Type, this.Flags.ToBinaryStr(), this.StreamId, this.PayloadOffset, this.DontUseMemPool);
  78. }
  79. public string PayloadAsHex()
  80. {
  81. System.Text.StringBuilder sb = new System.Text.StringBuilder("[", (int)this.PayloadLength + 1);
  82. if (this.Payload != null && this.PayloadLength > 0)
  83. {
  84. uint idx = this.PayloadOffset;
  85. sb.Append(this.Payload[idx++]);
  86. for (int i = 1; i < this.PayloadLength; i++)
  87. sb.AppendFormat(", {0:X2}", this.Payload[idx++]);
  88. }
  89. sb.Append("]");
  90. return sb.ToString();
  91. }
  92. }
  93. public struct HTTP2SettingsFrame
  94. {
  95. public readonly HTTP2FrameHeaderAndPayload Header;
  96. public HTTP2SettingsFlags Flags { get { return (HTTP2SettingsFlags)this.Header.Flags; } }
  97. public List<KeyValuePair<HTTP2Settings, UInt32>> Settings;
  98. public HTTP2SettingsFrame(HTTP2FrameHeaderAndPayload header)
  99. {
  100. this.Header = header;
  101. this.Settings = null;
  102. }
  103. public override string ToString()
  104. {
  105. string settings = null;
  106. if (this.Settings != null)
  107. {
  108. System.Text.StringBuilder sb = new System.Text.StringBuilder("[");
  109. foreach (var kvp in this.Settings)
  110. sb.AppendFormat("[{0}: {1}]", kvp.Key, kvp.Value);
  111. sb.Append("]");
  112. settings = sb.ToString();
  113. }
  114. return string.Format("[HTTP2SettingsFrame Header: {0}, Flags: {1}, Settings: {2}]", this.Header.ToString(), this.Flags, settings ?? "Empty");
  115. }
  116. }
  117. public struct HTTP2DataFrame
  118. {
  119. public readonly HTTP2FrameHeaderAndPayload Header;
  120. public HTTP2DataFlags Flags { get { return (HTTP2DataFlags)this.Header.Flags; } }
  121. public byte? PadLength;
  122. public UInt32 DataIdx;
  123. public byte[] Data;
  124. public uint DataLength;
  125. public HTTP2DataFrame(HTTP2FrameHeaderAndPayload header)
  126. {
  127. this.Header = header;
  128. this.PadLength = null;
  129. this.DataIdx = 0;
  130. this.Data = null;
  131. this.DataLength = 0;
  132. }
  133. public override string ToString()
  134. {
  135. return string.Format("[HTTP2DataFrame Header: {0}, Flags: {1}, PadLength: {2}, DataLength: {3}]",
  136. this.Header.ToString(),
  137. this.Flags,
  138. this.PadLength == null ? ":Empty" : this.PadLength.Value.ToString(),
  139. this.DataLength);
  140. }
  141. }
  142. public struct HTTP2HeadersFrame
  143. {
  144. public readonly HTTP2FrameHeaderAndPayload Header;
  145. public HTTP2HeadersFlags Flags { get { return (HTTP2HeadersFlags)this.Header.Flags; } }
  146. public byte? PadLength;
  147. public byte? IsExclusive;
  148. public UInt32? StreamDependency;
  149. public byte? Weight;
  150. public UInt32 HeaderBlockFragmentIdx;
  151. public byte[] HeaderBlockFragment;
  152. public UInt32 HeaderBlockFragmentLength;
  153. public HTTP2HeadersFrame(HTTP2FrameHeaderAndPayload header)
  154. {
  155. this.Header = header;
  156. this.PadLength = null;
  157. this.IsExclusive = null;
  158. this.StreamDependency = null;
  159. this.Weight = null;
  160. this.HeaderBlockFragmentIdx = 0;
  161. this.HeaderBlockFragment = null;
  162. this.HeaderBlockFragmentLength = 0;
  163. }
  164. public override string ToString()
  165. {
  166. return string.Format("[HTTP2HeadersFrame Header: {0}, Flags: {1}, PadLength: {2}, IsExclusive: {3}, StreamDependency: {4}, Weight: {5}, HeaderBlockFragmentLength: {6}]",
  167. this.Header.ToString(),
  168. this.Flags,
  169. this.PadLength == null ? ":Empty" : this.PadLength.Value.ToString(),
  170. this.IsExclusive == null ? "Empty" : this.IsExclusive.Value.ToString(),
  171. this.StreamDependency == null ? "Empty" : this.StreamDependency.Value.ToString(),
  172. this.Weight == null ? "Empty" : this.Weight.Value.ToString(),
  173. this.HeaderBlockFragmentLength);
  174. }
  175. }
  176. public struct HTTP2PriorityFrame
  177. {
  178. public readonly HTTP2FrameHeaderAndPayload Header;
  179. public byte IsExclusive;
  180. public UInt32 StreamDependency;
  181. public byte Weight;
  182. public HTTP2PriorityFrame(HTTP2FrameHeaderAndPayload header)
  183. {
  184. this.Header = header;
  185. this.IsExclusive = 0;
  186. this.StreamDependency = 0;
  187. this.Weight = 0;
  188. }
  189. public override string ToString()
  190. {
  191. return string.Format("[HTTP2PriorityFrame Header: {0}, IsExclusive: {1}, StreamDependency: {2}, Weight: {3}]",
  192. this.Header.ToString(), this.IsExclusive, this.StreamDependency, this.Weight);
  193. }
  194. }
  195. public struct HTTP2RSTStreamFrame
  196. {
  197. public readonly HTTP2FrameHeaderAndPayload Header;
  198. public UInt32 ErrorCode;
  199. public HTTP2ErrorCodes Error { get { return (HTTP2ErrorCodes)this.ErrorCode; } }
  200. public HTTP2RSTStreamFrame(HTTP2FrameHeaderAndPayload header)
  201. {
  202. this.Header = header;
  203. this.ErrorCode = 0;
  204. }
  205. public override string ToString()
  206. {
  207. return string.Format("[HTTP2RST_StreamFrame Header: {0}, Error: {1}({2})]", this.Header.ToString(), this.Error, this.ErrorCode);
  208. }
  209. }
  210. public struct HTTP2PushPromiseFrame
  211. {
  212. public readonly HTTP2FrameHeaderAndPayload Header;
  213. public HTTP2PushPromiseFlags Flags { get { return (HTTP2PushPromiseFlags)this.Header.Flags; } }
  214. public byte? PadLength;
  215. public byte ReservedBit;
  216. public UInt32 PromisedStreamId;
  217. public UInt32 HeaderBlockFragmentIdx;
  218. public byte[] HeaderBlockFragment;
  219. public UInt32 HeaderBlockFragmentLength;
  220. public HTTP2PushPromiseFrame(HTTP2FrameHeaderAndPayload header)
  221. {
  222. this.Header = header;
  223. this.PadLength = null;
  224. this.ReservedBit = 0;
  225. this.PromisedStreamId = 0;
  226. this.HeaderBlockFragmentIdx = 0;
  227. this.HeaderBlockFragment = null;
  228. this.HeaderBlockFragmentLength = 0;
  229. }
  230. public override string ToString()
  231. {
  232. return string.Format("[HTTP2Push_PromiseFrame Header: {0}, Flags: {1}, PadLength: {2}, ReservedBit: {3}, PromisedStreamId: {4}, HeaderBlockFragmentLength: {5}]",
  233. this.Header.ToString(),
  234. this.Flags,
  235. this.PadLength == null ? "Empty" : this.PadLength.Value.ToString(),
  236. this.ReservedBit,
  237. this.PromisedStreamId,
  238. this.HeaderBlockFragmentLength);
  239. }
  240. }
  241. public struct HTTP2PingFrame
  242. {
  243. public readonly HTTP2FrameHeaderAndPayload Header;
  244. public HTTP2PingFlags Flags { get { return (HTTP2PingFlags)this.Header.Flags; } }
  245. public readonly byte[] OpaqueData;
  246. public readonly byte OpaqueDataLength;
  247. public HTTP2PingFrame(HTTP2FrameHeaderAndPayload header)
  248. {
  249. this.Header = header;
  250. this.OpaqueData = BufferPool.Get(8, true);
  251. this.OpaqueDataLength = 8;
  252. }
  253. public override string ToString()
  254. {
  255. return string.Format("[HTTP2PingFrame Header: {0}, Flags: {1}, OpaqueData: {2}]",
  256. this.Header.ToString(),
  257. this.Flags,
  258. SecureProtocol.Org.BouncyCastle.Utilities.Encoders.Hex.ToHexString(this.OpaqueData, 0, this.OpaqueDataLength));
  259. }
  260. }
  261. public struct HTTP2GoAwayFrame
  262. {
  263. public readonly HTTP2FrameHeaderAndPayload Header;
  264. public HTTP2ErrorCodes Error { get { return (HTTP2ErrorCodes)this.ErrorCode; } }
  265. public byte ReservedBit;
  266. public UInt32 LastStreamId;
  267. public UInt32 ErrorCode;
  268. public byte[] AdditionalDebugData;
  269. public UInt32 AdditionalDebugDataLength;
  270. public HTTP2GoAwayFrame(HTTP2FrameHeaderAndPayload header)
  271. {
  272. this.Header = header;
  273. this.ReservedBit = 0;
  274. this.LastStreamId = 0;
  275. this.ErrorCode = 0;
  276. this.AdditionalDebugData = null;
  277. this.AdditionalDebugDataLength = 0;
  278. }
  279. public override string ToString()
  280. {
  281. return string.Format("[HTTP2GoAwayFrame Header: {0}, ReservedBit: {1}, LastStreamId: {2}, Error: {3}({4}), AdditionalDebugData({5}): {6}]",
  282. this.Header.ToString(),
  283. this.ReservedBit,
  284. this.LastStreamId,
  285. this.Error,
  286. this.ErrorCode,
  287. this.AdditionalDebugDataLength,
  288. this.AdditionalDebugData == null ? "Empty" : SecureProtocol.Org.BouncyCastle.Utilities.Encoders.Hex.ToHexString(this.AdditionalDebugData, 0, (int)this.AdditionalDebugDataLength));
  289. }
  290. }
  291. public struct HTTP2WindowUpdateFrame
  292. {
  293. public readonly HTTP2FrameHeaderAndPayload Header;
  294. public byte ReservedBit;
  295. public UInt32 WindowSizeIncrement;
  296. public HTTP2WindowUpdateFrame(HTTP2FrameHeaderAndPayload header)
  297. {
  298. this.Header = header;
  299. this.ReservedBit = 0;
  300. this.WindowSizeIncrement = 0;
  301. }
  302. public override string ToString()
  303. {
  304. return string.Format("[HTTP2WindowUpdateFrame Header: {0}, ReservedBit: {1}, WindowSizeIncrement: {2}]",
  305. this.Header.ToString(), this.ReservedBit, this.WindowSizeIncrement);
  306. }
  307. }
  308. public struct HTTP2ContinuationFrame
  309. {
  310. public readonly HTTP2FrameHeaderAndPayload Header;
  311. public HTTP2ContinuationFlags Flags { get { return (HTTP2ContinuationFlags)this.Header.Flags; } }
  312. public byte[] HeaderBlockFragment;
  313. public UInt32 HeaderBlockFragmentLength { get { return this.Header.PayloadLength; } }
  314. public HTTP2ContinuationFrame(HTTP2FrameHeaderAndPayload header)
  315. {
  316. this.Header = header;
  317. this.HeaderBlockFragment = null;
  318. }
  319. public override string ToString()
  320. {
  321. return string.Format("[HTTP2ContinuationFrame Header: {0}, Flags: {1}, HeaderBlockFragmentLength: {2}]",
  322. this.Header.ToString(),
  323. this.Flags,
  324. this.HeaderBlockFragmentLength);
  325. }
  326. }
  327. /// <summary>
  328. /// https://tools.ietf.org/html/rfc7838#section-4
  329. /// </summary>
  330. public struct HTTP2AltSVCFrame
  331. {
  332. public readonly HTTP2FrameHeaderAndPayload Header;
  333. public string Origin;
  334. public string AltSvcFieldValue;
  335. public HTTP2AltSVCFrame(HTTP2FrameHeaderAndPayload header)
  336. {
  337. this.Header = header;
  338. this.Origin = null;
  339. this.AltSvcFieldValue = null;
  340. }
  341. }
  342. }
  343. #endif