123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434 |
- #if (!UNITY_WEBGL || UNITY_EDITOR) && !BESTHTTP_DISABLE_ALTERNATE_SSL
- using Best.HTTP.Shared.Extensions;
- using Best.HTTP.Shared.Streams;
- using Best.HTTP.Shared.Logger;
- using Best.HTTP.Shared.PlatformSupport.Memory;
- using System;
- using System.Collections.Generic;
- using System.IO;
- namespace Best.HTTP.Hosts.Connections.HTTP2
- {
- // https://httpwg.org/specs/rfc7540.html#ErrorCodes
- public enum HTTP2ErrorCodes
- {
- NO_ERROR = 0x00,
- PROTOCOL_ERROR = 0x01,
- INTERNAL_ERROR = 0x02,
- FLOW_CONTROL_ERROR = 0x03,
- SETTINGS_TIMEOUT = 0x04,
- STREAM_CLOSED = 0x05,
- FRAME_SIZE_ERROR = 0x06,
- REFUSED_STREAM = 0x07,
- CANCEL = 0x08,
- COMPRESSION_ERROR = 0x09,
- CONNECT_ERROR = 0x0A,
- ENHANCE_YOUR_CALM = 0x0B,
- INADEQUATE_SECURITY = 0x0C,
- HTTP_1_1_REQUIRED = 0x0D
- }
- public static class HTTP2FrameHelper
- {
- public static HTTP2ContinuationFrame ReadContinuationFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#CONTINUATION
- HTTP2ContinuationFrame frame = new HTTP2ContinuationFrame(header);
- frame.HeaderBlockFragment = header.Payload;
- header.Payload = BufferSegment.Empty;
- return frame;
- }
- public static HTTP2WindowUpdateFrame ReadWindowUpdateFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#WINDOW_UPDATE
- HTTP2WindowUpdateFrame frame = new HTTP2WindowUpdateFrame(header);
- frame.ReservedBit = BufferHelper.ReadBit(header.Payload.Data[0], 0);
- frame.WindowSizeIncrement = BufferHelper.ReadUInt31(header.Payload, 0);
- return frame;
- }
- public static HTTP2GoAwayFrame ReadGoAwayFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#GOAWAY
- // str id error
- // | 0, 1, 2, 3 | 4, 5, 6, 7 | ...
- HTTP2GoAwayFrame frame = new HTTP2GoAwayFrame(header);
- frame.ReservedBit = BufferHelper.ReadBit(header.Payload.Data[0], 0);
- frame.LastStreamId = BufferHelper.ReadUInt31(header.Payload, 0);
- frame.ErrorCode = BufferHelper.ReadUInt32(header.Payload, 4);
- var additionalDebugDataLength = header.Payload.Count - 8;
- if (additionalDebugDataLength > 0)
- {
- var buff = BufferPool.Get(additionalDebugDataLength, true);
- Array.Copy(header.Payload.Data, 8, buff, 0, additionalDebugDataLength);
- frame.AdditionalDebugData = buff.AsBuffer(additionalDebugDataLength);
- }
- return frame;
- }
- public static HTTP2PingFrame ReadPingFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#PING
- HTTP2PingFrame frame = new HTTP2PingFrame(header);
- Array.Copy(header.Payload.Data, 0, frame.OpaqueData.Data, 0, frame.OpaqueData.Count);
- return frame;
- }
- public static HTTP2PushPromiseFrame ReadPush_PromiseFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#PUSH_PROMISE
- HTTP2PushPromiseFrame frame = new HTTP2PushPromiseFrame(header);
- int HeaderBlockFragmentLength = header.Payload.Count - 4; // PromisedStreamId
- bool isPadded = (frame.Flags & HTTP2PushPromiseFlags.PADDED) != 0;
- if (isPadded)
- {
- frame.PadLength = header.Payload.Data[0];
- HeaderBlockFragmentLength -= 1 + (frame.PadLength ?? 0);
- }
- frame.ReservedBit = BufferHelper.ReadBit(header.Payload.Data[1], 0);
- frame.PromisedStreamId = BufferHelper.ReadUInt31(header.Payload, 1);
- var HeaderBlockFragmentIdx = isPadded ? 5 : 4;
- frame.HeaderBlockFragment = header.Payload.Slice(HeaderBlockFragmentIdx, HeaderBlockFragmentLength);
- header.Payload = BufferSegment.Empty;
- return frame;
- }
- public static HTTP2RSTStreamFrame ReadRST_StreamFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#RST_STREAM
- HTTP2RSTStreamFrame frame = new HTTP2RSTStreamFrame(header);
- frame.ErrorCode = BufferHelper.ReadUInt32(header.Payload, 0);
- return frame;
- }
- public static HTTP2PriorityFrame ReadPriorityFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#PRIORITY
- if (header.Payload.Count != 5)
- {
- //throw FRAME_SIZE_ERROR
- }
- HTTP2PriorityFrame frame = new HTTP2PriorityFrame(header);
- frame.IsExclusive = BufferHelper.ReadBit(header.Payload.Data[0], 0);
- frame.StreamDependency = BufferHelper.ReadUInt31(header.Payload, 0);
- frame.Weight = header.Payload.Data[4];
- return frame;
- }
- public static HTTP2HeadersFrame ReadHeadersFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#HEADERS
- HTTP2HeadersFrame frame = new HTTP2HeadersFrame(header);
- var HeaderBlockFragmentLength = header.Payload.Count;
- bool isPadded = (frame.Flags & HTTP2HeadersFlags.PADDED) != 0;
- bool isPriority = (frame.Flags & HTTP2HeadersFlags.PRIORITY) != 0;
- int payloadIdx = 0;
- if (isPadded)
- {
- frame.PadLength = header.Payload.Data[payloadIdx++];
- int subLength = 1 + (frame.PadLength ?? 0);
- if (subLength <= HeaderBlockFragmentLength)
- HeaderBlockFragmentLength -= subLength;
- //else
- // throw PROTOCOL_ERROR;
- }
- if (isPriority)
- {
- frame.IsExclusive = BufferHelper.ReadBit(header.Payload.Data[payloadIdx], 0);
- frame.StreamDependency = BufferHelper.ReadUInt31(header.Payload, payloadIdx);
- payloadIdx += 4;
- frame.Weight = header.Payload.Data[payloadIdx++];
- int subLength = 5;
- if (subLength <= HeaderBlockFragmentLength)
- HeaderBlockFragmentLength -= subLength;
- //else
- // throw PROTOCOL_ERROR;
- }
- var HeaderBlockFragmentIdx = payloadIdx;
- frame.HeaderBlockFragment = header.Payload.Slice(HeaderBlockFragmentIdx, HeaderBlockFragmentLength);
- header.Payload = BufferSegment.Empty;
- return frame;
- }
- public static HTTP2DataFrame ReadDataFrame(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#DATA
- HTTP2DataFrame frame = new HTTP2DataFrame(header);
- var DataLength = header.Payload.Count;
- bool isPadded = (frame.Flags & HTTP2DataFlags.PADDED) != 0;
- if (isPadded)
- {
- frame.PadLength = header.Payload.Data[0];
- int subLength = 1 + (frame.PadLength ?? 0);
- if (subLength <= DataLength)
- DataLength -= subLength;
- //else
- // throw PROTOCOL_ERROR;
- }
- var DataIdx = isPadded ? 1 : 0;
- frame.Data = header.Payload.Slice(DataIdx, DataLength);
- header.Payload = BufferSegment.Empty;
- return frame;
- }
- public static HTTP2AltSVCFrame ReadAltSvcFrame(HTTP2FrameHeaderAndPayload header)
- {
- HTTP2AltSVCFrame frame = new HTTP2AltSVCFrame(header);
-
- // Implement
- return frame;
- }
- public static void StreamRead(Stream stream, byte[] buffer, int offset, uint count)
- {
- if (count == 0)
- return;
- uint sumRead = 0;
- do
- {
- int readCount = (int)(count - sumRead);
- int streamReadCount = stream.Read(buffer, (int)(offset + sumRead), readCount);
- if (streamReadCount <= 0 && readCount > 0)
- throw new Exception("TCP Stream closed!");
- sumRead += (uint)streamReadCount;
- } while (sumRead < count);
- }
- public static void StreamRead(Stream stream, BufferSegment buffer)
- {
- if (buffer.Count == 0)
- return;
- uint sumRead = 0;
- do
- {
- int readCount = (int)(buffer.Count - sumRead);
- int streamReadCount = stream.Read(buffer.Data, (int)(buffer.Offset + sumRead), readCount);
- if (streamReadCount <= 0 && readCount > 0)
- throw new Exception("TCP Stream closed!");
- sumRead += (uint)streamReadCount;
- } while (sumRead < buffer.Count);
- }
- public static AutoReleaseBuffer HeaderAsBinary(HTTP2FrameHeaderAndPayload header)
- {
- // https://httpwg.org/specs/rfc7540.html#FrameHeader
- var buffer = BufferPool.Get(9, true);
- BufferHelper.SetUInt24(buffer, 0, (uint)header.Payload.Count);
- buffer[3] = (byte)header.Type;
- buffer[4] = header.Flags;
- BufferHelper.SetUInt31(buffer, 5, header.StreamId);
- return new AutoReleaseBuffer { Data = buffer, Count = 9 };
- }
- public unsafe static bool CanReadFullFrame(PeekableStream stream)
- {
- // https://httpwg.org/specs/rfc7540.html#FrameHeader
- // A frame without any payload is 9 bytes
- if (stream.Length < 9)
- return false;
- stream.BeginPeek();
- // First 3 bytes are the payload length
- var rawLength = stackalloc byte[3];
- rawLength[0] = (byte)stream.PeekByte();
- rawLength[1] = (byte)stream.PeekByte();
- rawLength[2] = (byte)stream.PeekByte();
- var payloadLength = (UInt32)(rawLength[2] | rawLength[1] << 8 | rawLength[0] << 16);
- return stream.Length >= (9 + payloadLength);
- }
- public static HTTP2FrameHeaderAndPayload ReadHeader(Stream stream, LoggingContext context)
- {
- byte[] buffer = BufferPool.Get(9, true, context);
- using var _ = buffer.AsAutoRelease();
- StreamRead(stream, buffer, 0, 9);
- HTTP2FrameHeaderAndPayload header = new HTTP2FrameHeaderAndPayload();
- var PayloadLength = (int)BufferHelper.ReadUInt24(buffer, 0);
- header.Type = (HTTP2FrameTypes)buffer[3];
- header.Flags = buffer[4];
- header.StreamId = BufferHelper.ReadUInt31(buffer, 5);
- header.Payload = BufferPool.Get(PayloadLength, true, context).AsBuffer(PayloadLength);
- try
- {
- StreamRead(stream, header.Payload);
- }
- catch
- {
- BufferPool.Release(header.Payload);
- throw;
- }
- return header;
- }
- public static HTTP2SettingsFrame ReadSettings(HTTP2FrameHeaderAndPayload header)
- {
- HTTP2SettingsFrame frame = new HTTP2SettingsFrame(header);
- if (header.Payload.Count > 0)
- {
- int kvpCount = (int)(header.Payload.Count / 6);
- frame.Settings = new List<KeyValuePair<HTTP2Settings, uint>>(kvpCount);
- for (int i = 0; i < kvpCount; ++i)
- {
- HTTP2Settings key = (HTTP2Settings)BufferHelper.ReadUInt16(header.Payload.Data, i * 6);
- UInt32 value = BufferHelper.ReadUInt32(header.Payload, (i * 6) + 2);
- frame.Settings.Add(new KeyValuePair<HTTP2Settings, uint>(key, value));
- }
- }
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreateACKSettingsFrame()
- {
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.SETTINGS;
- frame.Flags = (byte)HTTP2SettingsFlags.ACK;
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreateSettingsFrame(List<KeyValuePair<HTTP2Settings, UInt32>> settings, LoggingContext context)
- {
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.SETTINGS;
- frame.Flags = 0;
- var PayloadLength = settings.Count * 6;
- frame.Payload = BufferPool.Get(PayloadLength, true, context).AsBuffer(PayloadLength);
- for (int i = 0; i < settings.Count; ++i)
- {
- BufferHelper.SetUInt16(frame.Payload.Data, i * 6, (UInt16)settings[i].Key);
- BufferHelper.SetUInt32(frame.Payload.Data, (i * 6) + 2, settings[i].Value);
- }
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreatePingFrame(HTTP2PingFlags flags, LoggingContext context)
- {
- // https://httpwg.org/specs/rfc7540.html#PING
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.PING;
- frame.Flags = (byte)flags;
- frame.StreamId = 0;
- frame.Payload = BufferPool.Get(8, true, context).AsBuffer(8);
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreateWindowUpdateFrame(UInt32 streamId, UInt32 windowSizeIncrement, LoggingContext context)
- {
- // https://httpwg.org/specs/rfc7540.html#WINDOW_UPDATE
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.WINDOW_UPDATE;
- frame.Flags = 0;
- frame.StreamId = streamId;
- frame.Payload = BufferPool.Get(4, true, context).AsBuffer(4);
- BufferHelper.SetBit(0, 0, 0);
- BufferHelper.SetUInt31(frame.Payload.Data, 0, windowSizeIncrement);
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreateGoAwayFrame(UInt32 lastStreamId, HTTP2ErrorCodes error, LoggingContext context)
- {
- // https://httpwg.org/specs/rfc7540.html#GOAWAY
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.GOAWAY;
- frame.Flags = 0;
- frame.StreamId = 0;
- frame.Payload = BufferPool.Get(8, true, context).AsBuffer(8);
- BufferHelper.SetUInt31(frame.Payload.Data, 0, lastStreamId);
- BufferHelper.SetUInt31(frame.Payload.Data, 4, (UInt32)error);
- return frame;
- }
- public static HTTP2FrameHeaderAndPayload CreateRSTFrame(UInt32 streamId, HTTP2ErrorCodes errorCode, LoggingContext context)
- {
- // https://httpwg.org/specs/rfc7540.html#RST_STREAM
- HTTP2FrameHeaderAndPayload frame = new HTTP2FrameHeaderAndPayload();
- frame.Type = HTTP2FrameTypes.RST_STREAM;
- frame.Flags = 0;
- frame.StreamId = streamId;
- frame.Payload = BufferPool.Get(4, true, context).AsBuffer(4);
- BufferHelper.SetUInt32(frame.Payload.Data, 0, (UInt32)errorCode);
- return frame;
- }
- }
- }
- #endif
|