/// <summary>Constructs an incoming request frame.</summary> /// <param name="protocol">The Ice protocol.</param> /// <param name="data">The frame data as an array segment.</param> /// <param name="maxSize">The maximum payload size, checked during decompression.</param> /// <param name="socketStream">The optional socket stream. The stream is non-null if there's still data to /// read on the stream after the reading the request frame.</param> internal IncomingRequestFrame( Protocol protocol, ArraySegment <byte> data, int maxSize, SocketStream?socketStream) : base(protocol, maxSize) { SocketStream = socketStream; var istr = new InputStream(data, Protocol.GetEncoding()); if (Protocol == Protocol.Ice1) { var requestHeader = new Ice1RequestHeader(istr); Identity = requestHeader.Identity; Facet = Ice1Definitions.GetFacet(requestHeader.FacetPath); Location = Array.Empty <string>(); Operation = requestHeader.Operation; IsIdempotent = requestHeader.OperationMode != OperationMode.Normal; Context = requestHeader.Context; Priority = default; Deadline = DateTime.MaxValue; } else { int headerSize = istr.ReadSize(); int startPos = istr.Pos; // We use the generated code for the header body and read the rest of the header "by hand". var requestHeaderBody = new Ice2RequestHeaderBody(istr); Identity = requestHeaderBody.Identity; Facet = requestHeaderBody.Facet ?? ""; Location = requestHeaderBody.Location ?? Array.Empty <string>(); Operation = requestHeaderBody.Operation; IsIdempotent = requestHeaderBody.Idempotent ?? false; Priority = requestHeaderBody.Priority ?? default; if (requestHeaderBody.Deadline < -1 || requestHeaderBody.Deadline == 0) { throw new InvalidDataException($"received invalid deadline value {requestHeaderBody.Deadline}"); } // The infinite deadline is encoded as -1 and converted to DateTime.MaxValue Deadline = requestHeaderBody.Deadline == -1 ? DateTime.MaxValue : DateTime.UnixEpoch + TimeSpan.FromMilliseconds(requestHeaderBody.Deadline); Context = requestHeaderBody.Context ?? new SortedDictionary <string, string>(); BinaryContext = istr.ReadBinaryContext(); if (istr.Pos - startPos != headerSize) { throw new InvalidDataException( @$ "received invalid request header: expected {headerSize} bytes but read {istr.Pos - startPos } bytes"); } if (Location.Any(segment => segment.Length == 0)) { throw new InvalidDataException("received request with an empty location segment"); } }
/// <summary>Constructs an incoming request frame.</summary> /// <param name="protocol">The Ice protocol.</param> /// <param name="data">The frame data as an array segment.</param> /// <param name="sizeMax">The maximum payload size, checked during decompression.</param> public IncomingRequestFrame(Protocol protocol, ArraySegment <byte> data, int sizeMax) : base(data, protocol, sizeMax) { var istr = new InputStream(Data, Protocol.GetEncoding()); if (Protocol == Protocol.Ice1) { var requestHeaderBody = new Ice1RequestHeaderBody(istr); Identity = requestHeaderBody.Identity; Facet = Ice1Definitions.GetFacet(requestHeaderBody.FacetPath); Location = Array.Empty <string>(); Operation = requestHeaderBody.Operation; IsIdempotent = requestHeaderBody.OperationMode != OperationMode.Normal; Context = requestHeaderBody.Context; Priority = default; Deadline = DateTime.MaxValue; } else { var requestHeaderBody = new Ice2RequestHeaderBody(istr); Identity = requestHeaderBody.Identity; Facet = requestHeaderBody.Facet ?? ""; Location = requestHeaderBody.Location ?? Array.Empty <string>(); Operation = requestHeaderBody.Operation; IsIdempotent = requestHeaderBody.Idempotent ?? false; Priority = requestHeaderBody.Priority ?? default; if (requestHeaderBody.Deadline < -1 || requestHeaderBody.Deadline == 0) { throw new InvalidDataException($"received invalid deadline value {requestHeaderBody.Deadline}"); } // The infinite deadline is encoded as -1 and converted to DateTime.MaxValue Deadline = requestHeaderBody.Deadline == -1 ? DateTime.MaxValue : DateTime.UnixEpoch + TimeSpan.FromMilliseconds(requestHeaderBody.Deadline); Context = null !; // initialized below if (Location.Any(segment => segment.Length == 0)) { throw new InvalidDataException("received request with empty location segment"); } } if (Identity.Name.Length == 0) { throw new InvalidDataException("received request with null identity"); } if (Operation.Length == 0) { throw new InvalidDataException("received request with empty operation name"); } (int size, int sizeLength, Encoding encoding) = Data.Slice(istr.Pos).AsReadOnlySpan().ReadEncapsulationHeader(Protocol.GetEncoding()); Payload = Data.Slice(istr.Pos, size + sizeLength); // the payload is the encapsulation if (Protocol == Protocol.Ice2) { // BinaryContext is a computed property that depends on Payload. if (BinaryContext.TryGetValue(0, out ReadOnlyMemory <byte> value)) { Context = value.Read(istr => istr.ReadDictionary(minKeySize: 1, minValueSize: 1, InputStream.IceReaderIntoString, InputStream.IceReaderIntoString)); } else { Context = new Dictionary <string, string>(); } } if (protocol == Protocol.Ice1 && size + 4 + istr.Pos != data.Count) { // The payload holds an encapsulation and the encapsulation must use up the full buffer with ice1. // "4" corresponds to fixed-length size with the 1.1 encoding. throw new InvalidDataException($"invalid request encapsulation size: {size}"); } Encoding = encoding; HasCompressedPayload = Encoding == Encoding.V20 && Payload[sizeLength + 2] != 0; }