org.reaktivity.nukleus.http2.internal.Http2Connection Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of nukleus-http2 Show documentation
Show all versions of nukleus-http2 Show documentation
HTTP/2 Nukleus Implementation
/**
* Copyright 2016-2018 The Reaktivity Project
*
* The Reaktivity Project licenses this file to you under the Apache License,
* version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package org.reaktivity.nukleus.http2.internal;
import static org.reaktivity.nukleus.buffer.BufferPool.NO_SLOT;
import static org.reaktivity.nukleus.http2.internal.Http2StreamState.CLOSED;
import static org.reaktivity.nukleus.http2.internal.Http2StreamState.HALF_CLOSED_REMOTE;
import static org.reaktivity.nukleus.http2.internal.Http2StreamState.OPEN;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.CONNECTION;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.DEFAULT_ACCESS_CONTROL_ALLOW_ORIGIN;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.KEEP_ALIVE;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.PROXY_CONNECTION;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.TE;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.TRAILERS;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackContext.UPGRADE;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackHeaderFieldFW.HeaderFieldType.UNKNOWN;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackLiteralHeaderFieldFW.LiteralType.INCREMENTAL_INDEXING;
import static org.reaktivity.nukleus.http2.internal.types.stream.HpackLiteralHeaderFieldFW.LiteralType.WITHOUT_INDEXING;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import org.agrona.DirectBuffer;
import org.agrona.MutableDirectBuffer;
import org.agrona.collections.Int2ObjectHashMap;
import org.agrona.concurrent.UnsafeBuffer;
import org.reaktivity.nukleus.function.MessageConsumer;
import org.reaktivity.nukleus.function.MessageFunction;
import org.reaktivity.nukleus.function.MessagePredicate;
import org.reaktivity.nukleus.http2.internal.types.HttpHeaderFW;
import org.reaktivity.nukleus.http2.internal.types.ListFW;
import org.reaktivity.nukleus.http2.internal.types.OctetsFW;
import org.reaktivity.nukleus.http2.internal.types.String16FW;
import org.reaktivity.nukleus.http2.internal.types.StringFW;
import org.reaktivity.nukleus.http2.internal.types.control.HttpRouteExFW;
import org.reaktivity.nukleus.http2.internal.types.control.RouteFW;
import org.reaktivity.nukleus.http2.internal.types.stream.AbortFW;
import org.reaktivity.nukleus.http2.internal.types.stream.BeginFW;
import org.reaktivity.nukleus.http2.internal.types.stream.DataFW;
import org.reaktivity.nukleus.http2.internal.types.stream.EndFW;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackContext;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackHeaderBlockFW;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackHeaderFieldFW;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackHuffman;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackLiteralHeaderFieldFW;
import org.reaktivity.nukleus.http2.internal.types.stream.HpackStringFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2ContinuationFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2DataExFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2DataFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2ErrorCode;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2Flags;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2FrameFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2FrameHeaderFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2FrameType;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2HeadersFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2PingFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2PrefaceFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2PriorityFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2RstStreamFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2SettingsFW;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2SettingsId;
import org.reaktivity.nukleus.http2.internal.types.stream.Http2WindowUpdateFW;
import org.reaktivity.nukleus.http2.internal.types.stream.HttpBeginExFW;
import org.reaktivity.nukleus.http2.internal.types.stream.ResetFW;
import org.reaktivity.nukleus.http2.internal.types.stream.WindowFW;
import org.reaktivity.nukleus.route.RouteManager;
final class Http2Connection
{
private static final Map EMPTY_HEADERS = Collections.emptyMap();
ServerStreamFactory factory;
private DecoderState decoderState;
// slab to assemble a complete HTTP2 frame
// no need for separate slab per HTTP2 stream as the frames are not fragmented
private int frameSlot = NO_SLOT;
int frameSlotLimit;
// slab to assemble a complete HTTP2 headers frame(including its continuation frames)
// no need for separate slab per HTTP2 stream as no interleaved frames of any other type
// or from any other stream
private int headersSlotIndex = NO_SLOT;
private int headersSlotOffset;
final long networkRouteId;
final long networkId;
long authorization;
int lastStreamId;
int networkReplyBudget;
int networkReplyPadding;
int outWindowThreshold = -1;
final WriteScheduler writeScheduler;
final MessageConsumer network;
final long networkReplyId;
private final HpackContext decodeContext;
private final HpackContext encodeContext;
private final MessageFunction wrapRoute;
final long networkReplyGroupId;
final Int2ObjectHashMap http2Streams; // HTTP2 stream-id --> Http2Stream
private int clientStreamCount;
private int promisedStreamCount;
private int maxClientStreamId;
private int maxPushPromiseStreamId;
private boolean goaway;
Settings initialSettings;
Settings localSettings;
Settings remoteSettings;
private boolean expectContinuation;
private int expectContinuationStreamId;
private boolean expectDynamicTableSizeUpdate = true;
long http2OutWindow;
long http2InWindow;
private final Consumer headerFieldConsumer;
private final Consumer trailerFieldConsumer;
private final HeadersContext headersContext = new HeadersContext();
private final EncodeHeadersContext encodeHeadersContext = new EncodeHeadersContext();
final Http2Writer http2Writer;
final MessageConsumer networkReply;
RouteManager router;
long traceId;
private Http2ErrorCode decodeError;
Http2Connection(
ServerStreamFactory factory,
RouteManager router,
MessageConsumer network,
long networkRouteId,
long networkId,
MessageConsumer networkReply,
long networkReplyId,
MessageFunction wrapRoute)
{
this.factory = factory;
this.router = router;
this.wrapRoute = wrapRoute;
this.network = network;
this.networkId = networkId;
this.networkRouteId = networkRouteId;
this.networkReplyId = networkReplyId;
this.http2Streams = new Int2ObjectHashMap<>();
this.localSettings = new Settings();
this.remoteSettings = new Settings();
this.decodeContext = new HpackContext(localSettings.headerTableSize, false);
this.encodeContext = new HpackContext(remoteSettings.headerTableSize, true);
this.http2Writer = factory.http2Writer;
this.writeScheduler = new Http2WriteScheduler(this, networkReply, http2Writer, networkRouteId, networkReplyId);
this.http2InWindow = localSettings.initialWindowSize;
this.http2OutWindow = remoteSettings.initialWindowSize;
this.networkReply = networkReply;
this.networkReplyGroupId = factory.supplyGroupId.getAsLong();
BiConsumer nameValue =
((BiConsumer)this::collectHeaders)
.andThen(this::validatePseudoHeaders)
.andThen(this::uppercaseHeaders)
.andThen(this::connectionHeaders)
.andThen(this::contentLengthHeader)
.andThen(this::teHeader)
.andThen(this::mapToHttp);
Consumer consumer = this::validateHeaderFieldType;
consumer = consumer.andThen(this::dynamicTableSizeUpdate);
this.headerFieldConsumer = consumer.andThen(h -> decodeHeaderField(h, nameValue));
this.trailerFieldConsumer = h -> decodeHeaderField(h, this::validateTrailerFieldName);
}
void handleBegin(
BeginFW beginRO)
{
this.authorization = beginRO.authorization();
this.decoderState = this::decodePreface;
this.initialSettings = new Settings(factory.config.serverConcurrentStreams(), 0);
writeScheduler.settings(initialSettings.maxConcurrentStreams, initialSettings.initialWindowSize);
factory.counters.settingsFramesWritten.getAsLong();
}
void handleData(
DataFW data)
{
final long traceId = data.trace();
final OctetsFW payload = data.payload();
final DirectBuffer buffer = payload.buffer();
final int offset = payload.offset();
final int limit = payload.limit();
DirectBuffer decodeBuffer = buffer;
int decodeOffset = offset;
int decodeLimit = limit;
if (frameSlot != NO_SLOT)
{
final MutableDirectBuffer frameBuffer = factory.framePool.buffer(frameSlot);
frameBuffer.putBytes(frameSlotLimit, buffer, offset, limit - offset);
frameSlotLimit += limit - offset;
decodeBuffer = frameBuffer;
decodeOffset = 0;
decodeLimit = frameSlotLimit;
}
this.traceId = traceId;
int decodeProgress = 0;
while (decodeOffset < decodeLimit && decodeError == null)
{
decodeProgress = decoderState.decode(decodeBuffer, decodeOffset, decodeLimit);
if (decodeProgress <= 0)
{
// incomplete frame
break;
}
decodeOffset += decodeProgress;
}
this.traceId = 0;
final int decodeRemaining = decodeLimit - decodeOffset;
if (decodeProgress >= 0 && decodeRemaining > 0)
{
if (frameSlot == NO_SLOT)
{
assert frameSlotLimit == 0;
final int newFrameSlot = factory.framePool.acquire(networkId);
if (newFrameSlot != NO_SLOT)
{
frameSlot = newFrameSlot;
}
else
{
decodeProgress = -1; // error
}
}
if (frameSlot != NO_SLOT)
{
final MutableDirectBuffer frameBuffer = factory.framePool.buffer(frameSlot);
frameBuffer.putBytes(0, decodeBuffer, decodeOffset, decodeRemaining);
frameSlotLimit = decodeRemaining;
}
}
else
{
releaseFrameSlot();
}
if (decodeError != null)
{
error(decodeError);
}
if (decodeProgress < 0 ||
decodeError != null && decodeError != Http2ErrorCode.NO_ERROR)
{
// TODO: use traceId ??
http2Streams.forEach((i, s) -> s.onAbort(traceId));
doResetNetworkAndCleanup();
}
}
void handleAbort(
long traceId)
{
http2Streams.forEach((i, s) -> s.onAbort(traceId));
doCleanup();
}
void handleReset(
ResetFW reset)
{
http2Streams.forEach((i, s) -> s.onReset(reset.trace()));
doCleanup();
}
void handleEnd(
EndFW end)
{
decoderState = (b, o, l) -> o;
http2Streams.forEach((i, s) -> s.onEnd());
writeScheduler.doEnd();
doCleanup();
}
private int decodePreface(
final DirectBuffer buffer,
final int offset,
final int limit)
{
final Http2PrefaceFW preface = factory.prefaceRO.tryWrap(buffer, offset, limit);
int decodeProgress = 0;
if (preface != null)
{
if (preface.error())
{
decodeProgress = -1;
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
}
else
{
decodeProgress = preface.sizeof();
this.decoderState = this::decodeFrame;
}
}
return decodeProgress;
}
private int decodeFrame(
final DirectBuffer buffer,
final int offset,
final int limit)
{
final Http2FrameFW http2Frame = factory.http2RO.tryWrap(buffer, offset, limit);
if (http2Frame != null)
{
if (http2Frame.payloadLength() > localSettings.maxFrameSize)
{
this.decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
}
else if (http2Frame.streamId() == 0)
{
onConnectionFrame(http2Frame);
}
else
{
onStreamFrame(http2Frame);
}
}
else
{
final Http2FrameHeaderFW http2FrameHeader = factory.http2HeaderRO.tryWrap(buffer, offset, limit);
if (http2FrameHeader != null)
{
if (http2FrameHeader.payloadLength() > localSettings.maxFrameSize)
{
this.decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
}
}
}
return http2Frame != null ? http2Frame.sizeof() : 0;
}
private void onConnectionFrame(
final Http2FrameFW http2Frame)
{
switch (http2Frame.type())
{
case SETTINGS:
factory.counters.settingsFramesRead.getAsLong();
onConnectionSettings(http2Frame);
break;
case PING:
factory.counters.pingFramesRead.getAsLong();
onConnectionPing(http2Frame);
break;
case GO_AWAY:
factory.counters.goawayFramesRead.getAsLong();
onConnectionGoAway(http2Frame);
break;
case WINDOW_UPDATE:
factory.counters.windowUpdateFramesRead.getAsLong();
onConnectionWindowUpdate(http2Frame);
break;
case UNKNOWN:
break;
default:
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
break;
}
}
private void onStreamFrame(
final Http2FrameFW http2Frame)
{
int streamId = http2Frame.streamId();
Http2FrameType type = http2Frame.type();
if ((streamId & 0x01) != 0x01 &&
type != Http2FrameType.WINDOW_UPDATE &&
type != Http2FrameType.RST_STREAM &&
type != Http2FrameType.PRIORITY)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
if (expectContinuation &&
(type != Http2FrameType.CONTINUATION || streamId != expectContinuationStreamId))
{
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
if ((streamId & 0x01) == 0x01 &&
streamId > maxClientStreamId &&
type != Http2FrameType.HEADERS &&
type != Http2FrameType.PRIORITY)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
Http2Stream stream = http2Streams.get(streamId);
if (stream == null)
{
switch (type)
{
case HEADERS:
factory.counters.headersFramesRead.getAsLong();
onStreamHeaders(http2Frame);
break;
case CONTINUATION:
factory.counters.continuationFramesRead.getAsLong();
onStreamContinuation(http2Frame);
break;
case WINDOW_UPDATE:
// "half-closed (remote)" or "closed" stream MUST NOT be treated as error
factory.counters.windowUpdateFramesRead.getAsLong();
onStreamWindowUpdate(stream, http2Frame);
break;
case PRIORITY:
// "half-closed (remote)" or "closed" stream MUST NOT be treated as error
factory.counters.priorityFramesRead.getAsLong();
onStreamPriority(stream, http2Frame);
break;
case RST_STREAM:
onStreamRst(stream, http2Frame);
break;
case UNKNOWN:
break;
default:
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
break;
}
}
else
{
switch (type)
{
case DATA:
factory.counters.dataFramesRead.getAsLong();
onStreamData(stream, http2Frame);
break;
case HEADERS:
factory.counters.headersFramesRead.getAsLong();
onStreamHeaders(stream, http2Frame);
break;
case PRIORITY:
factory.counters.priorityFramesRead.getAsLong();
onStreamPriority(stream, http2Frame);
break;
case RST_STREAM:
factory.counters.resetStreamFramesRead.getAsLong();
onStreamRst(stream, http2Frame);
break;
case WINDOW_UPDATE:
factory.counters.windowUpdateFramesRead.getAsLong();
onStreamWindowUpdate(stream, http2Frame);
break;
case UNKNOWN:
break;
default:
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
break;
}
}
}
private void onConnectionSettings(
final Http2FrameFW http2Frame)
{
final Http2SettingsFW settings =
factory.settingsRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (settings == null || settings.ack() && settings.payloadLength() != 0)
{
decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
}
else if (!settings.ack())
{
settings.forEach(this::applySetting);
writeScheduler.settingsAck();
factory.counters.settingsFramesWritten.getAsLong();
}
else
{
http2Streams.values().forEach(this::applyInitialWindowDelta);
// now that peer acked our initial settings, can use them as our local settings
localSettings = initialSettings;
}
}
private void onConnectionPing(
Http2FrameFW http2Frame)
{
final Http2PingFW ping = factory.pingRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (ping == null)
{
this.decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
return;
}
if (!ping.ack())
{
writeScheduler.pingAck(ping.payload(), 0, ping.payload().capacity());
factory.counters.pingFramesWritten.getAsLong();
}
}
private void onConnectionGoAway(
Http2FrameFW http2Frame)
{
// TODO: ConnectionState
if (!goaway)
{
goaway = true;
remoteSettings.enablePush = false; // no new streams
this.decodeError = Http2ErrorCode.NO_ERROR;
}
}
private void onConnectionWindowUpdate(
Http2FrameFW http2Frame)
{
final Http2WindowUpdateFW http2Window =
factory.http2WindowRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (http2Window == null)
{
decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
return;
}
// 6.9 WINDOW_UPDATE - legal range for flow-control window increment is 1 to 2^31-1 octets.
if (http2Window.size() < 1)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
http2OutWindow += http2Window.size();
// 6.9.1 A sender MUST NOT allow a flow-control window to exceed 2^31-1 octets.
if (http2OutWindow > Integer.MAX_VALUE)
{
decodeError = Http2ErrorCode.FLOW_CONTROL_ERROR;
return;
}
writeScheduler.onHttp2Window();
}
private void onStreamHeaders(
Http2FrameFW http2Frame)
{
int streamId = http2Frame.streamId();
Http2HeadersFW http2Headers = factory.headersRO.wrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
int parentStreamId = http2Headers.parentStream();
if (parentStreamId == streamId)
{
// 5.3.1 A stream cannot depend on itself
doRstStream(streamId, Http2ErrorCode.PROTOCOL_ERROR);
}
if (http2Headers.dataLength() < 0)
{
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
if (streamId <= maxClientStreamId)
{
this.decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
maxClientStreamId = streamId;
if (clientStreamCount >= localSettings.maxConcurrentStreams)
{
doRstStream(streamId, Http2ErrorCode.REFUSED_STREAM);
return;
}
if (!http2Headers.endHeaders())
{
assert headersSlotIndex == NO_SLOT;
assert headersSlotOffset == 0;
headersSlotIndex = factory.headersPool.acquire(networkId);
if (headersSlotIndex == NO_SLOT)
{
// all slots are in use, just reset the connection
factory.doReset(network, networkRouteId, networkId, factory.supplyTrace.getAsLong());
handleAbort(factory.supplyTrace.getAsLong());
return;
}
MutableDirectBuffer headersBuffer = factory.headersPool.buffer(headersSlotIndex);
headersBuffer.putBytes(headersSlotOffset, http2Headers.buffer(),
http2Headers.dataOffset(), http2Headers.dataLength());
headersSlotOffset = http2Headers.dataLength();
expectContinuation = true;
expectContinuationStreamId = streamId;
return;
}
onStreamHeadersEnd(http2Frame, http2Headers.buffer(), http2Headers.dataOffset(),
http2Headers.dataOffset() + http2Headers.dataLength());
}
private void onStreamHeaders(
Http2Stream stream,
Http2FrameFW http2Frame)
{
if (stream.state == Http2StreamState.HALF_CLOSED_REMOTE)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
Http2HeadersFW http2Trailers = factory.headersRO.wrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
HpackHeaderBlockFW headerBlock = factory.blockRO.wrap(http2Trailers.buffer(), http2Trailers.dataOffset(),
http2Trailers.dataOffset() + http2Trailers.dataLength());
headerBlock.forEach(trailerFieldConsumer);
if (headersContext.error())
{
if (headersContext.streamError != null)
{
doRstStream(stream.http2StreamId, headersContext.streamError);
return;
}
if (headersContext.connectionError != null)
{
decodeError = headersContext.connectionError;
return;
}
}
}
private void onStreamContinuation(
Http2FrameFW http2Frame)
{
if (!expectContinuation)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
assert headersSlotIndex != NO_SLOT;
assert headersSlotOffset != 0;
Http2ContinuationFW http2Continuation =
factory.continationRO.wrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
DirectBuffer payload = http2Continuation.payload();
MutableDirectBuffer headersBuffer = factory.headersPool.buffer(headersSlotIndex);
headersBuffer.putBytes(headersSlotOffset, payload, 0, payload.capacity());
headersSlotOffset += payload.capacity();
if (!http2Continuation.endHeaders())
{
assert expectContinuation;
assert expectContinuationStreamId == http2Continuation.streamId();
return;
}
onStreamHeadersEnd(http2Frame, headersBuffer, 0, headersSlotOffset);
releaseHeadersSlot();
expectContinuation = false;
expectContinuationStreamId = 0;
}
private void onStreamHeadersEnd(
Http2FrameFW http2Frame,
DirectBuffer headersBuffer,
int headersOffset,
int headersLimit)
{
int streamId = http2Frame.streamId();
headersContext.reset();
factory.httpBeginExRW.wrap(factory.scratch, 0, factory.scratch.capacity());
HpackHeaderBlockFW headerBlock = factory.blockRO.wrap(headersBuffer, headersOffset, headersLimit);
headerBlock.forEach(headerFieldConsumer);
// All HTTP/2 requests MUST include exactly one valid value for the
// ":method", ":scheme", and ":path" pseudo-header fields, unless it is
// a CONNECT request (Section 8.3). An HTTP request that omits
// mandatory pseudo-header fields is malformed
if (!headersContext.error() &&
(headersContext.method != 1 || headersContext.scheme != 1 || headersContext.path != 1))
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
}
if (headersContext.error())
{
if (headersContext.streamError != null)
{
doRstStream(streamId, headersContext.streamError);
return;
}
if (headersContext.connectionError != null)
{
decodeError = headersContext.connectionError;
return;
}
}
addDefaultPortToAuthorityIfNeeded();
RouteFW route = resolveTarget(headersContext.headers);
if (route == null)
{
noRoute(streamId);
}
else
{
Http2StreamState nextState = http2Frame.endStream() ? HALF_CLOSED_REMOTE : OPEN;
followRoute(streamId, nextState, route);
}
}
private void onStreamData(
Http2Stream stream,
Http2FrameFW http2Frame)
{
if (stream.state == HALF_CLOSED_REMOTE)
{
decodeError = Http2ErrorCode.STREAM_CLOSED;
closeStream(stream);
return;
}
// handle invalid padding length
Http2DataFW http2Data = factory.http2DataRO.wrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (http2Data.dataLength() < 0)
{
decodeError = Http2ErrorCode.STREAM_CLOSED;
closeStream(stream);
return;
}
final int payloadLength = http2Frame.payloadLength();
if (stream.http2InWindow < payloadLength || http2InWindow < payloadLength)
{
doRstByUs(stream, Http2ErrorCode.FLOW_CONTROL_ERROR);
return;
}
http2InWindow -= payloadLength;
stream.http2InWindow -= payloadLength;
stream.totalData += payloadLength;
if (http2Data.endStream())
{
// 8.1.2.6 A request is malformed if the value of a content-length header field does
// not equal the sum of the DATA frame payload lengths
if (stream.contentLength != -1 && stream.totalData != stream.contentLength)
{
doRstByUs(stream, Http2ErrorCode.PROTOCOL_ERROR);
//stream.httpWriteScheduler.doEnd(stream.targetId);
return;
}
stream.state = Http2StreamState.HALF_CLOSED_REMOTE;
}
stream.onData(traceId, http2Data);
}
private void onStreamPriority(
Http2Stream stream,
Http2FrameFW http2Frame)
{
Http2PriorityFW http2Priority = factory.priorityRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (http2Priority == null)
{
decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
return;
}
if (stream != null)
{
if (http2Priority.parentStream() == stream.http2StreamId)
{
// 5.3.1 A stream cannot depend on itself
doRstByUs(stream, Http2ErrorCode.PROTOCOL_ERROR);
return;
}
}
}
private void onStreamWindowUpdate(
Http2Stream stream,
Http2FrameFW http2Frame)
{
Http2WindowUpdateFW http2Window =
factory.http2WindowRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (http2Window == null)
{
decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
return;
}
// 6.9 WINDOW_UPDATE - legal range for flow-control window increment is 1 to 2^31-1 octets.
if (http2Window.size() < 1)
{
doRstByUs(stream, Http2ErrorCode.PROTOCOL_ERROR);
return;
}
if (stream != null)
{
// 6.9.1 A sender MUST NOT allow a flow-control window to exceed 2^31-1 octets.
stream.http2OutWindow += http2Window.size();
if (stream.http2OutWindow > Integer.MAX_VALUE)
{
doRstByUs(stream, Http2ErrorCode.FLOW_CONTROL_ERROR);
return;
}
writeScheduler.onHttp2Window(stream.http2StreamId);
}
}
private void onStreamRst(
Http2Stream stream,
Http2FrameFW http2Frame)
{
Http2RstStreamFW http2RstStream =
factory.http2RstStreamRO.tryWrap(http2Frame.buffer(), http2Frame.offset(), http2Frame.limit());
if (http2RstStream == null)
{
decodeError = Http2ErrorCode.FRAME_SIZE_ERROR;
return;
}
if (stream != null)
{
stream.onReset(factory.supplyTrace.getAsLong());
closeStream(stream);
}
}
private void applySetting(
Http2SettingsId id,
Long value)
{
switch (id)
{
case HEADER_TABLE_SIZE:
remoteSettings.headerTableSize = value.intValue();
break;
case ENABLE_PUSH:
if (!(value == 0L || value == 1L))
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
remoteSettings.enablePush = (value == 1L);
break;
case MAX_CONCURRENT_STREAMS:
remoteSettings.maxConcurrentStreams = value.intValue();
break;
case INITIAL_WINDOW_SIZE:
if (value > Integer.MAX_VALUE)
{
decodeError = Http2ErrorCode.FLOW_CONTROL_ERROR;
return;
}
int old = remoteSettings.initialWindowSize;
remoteSettings.initialWindowSize = value.intValue();
int update = value.intValue() - old;
// 6.9.2. Initial Flow-Control Window Size
// SETTINGS frame can alter the initial flow-control
// window size for streams with active flow-control windows
for(Http2Stream http2Stream: http2Streams.values())
{
http2Stream.http2OutWindow += update; // http2OutWindow can become negative
if (http2Stream.http2OutWindow > Integer.MAX_VALUE)
{
// 6.9.2. Initial Flow-Control Window Size
// An endpoint MUST treat a change to SETTINGS_INITIAL_WINDOW_SIZE that
// causes any flow-control window to exceed the maximum size as a
// connection error of type FLOW_CONTROL_ERROR.
decodeError = Http2ErrorCode.FLOW_CONTROL_ERROR;
return;
}
}
break;
case MAX_FRAME_SIZE:
if (value < Math.pow(2, 14) || value > Math.pow(2, 24) -1)
{
decodeError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
remoteSettings.maxFrameSize = value.intValue();
break;
case MAX_HEADER_LIST_SIZE:
remoteSettings.maxHeaderListSize = value.intValue();
break;
default:
// Ignore the unkonwn setting
break;
}
}
private void applyInitialWindowDelta(
Http2Stream http2Stream)
{
final int initialWindowDelta = initialSettings.initialWindowSize - localSettings.initialWindowSize;
http2Stream.http2InWindow += initialWindowDelta; // http2InWindow can become negative
}
private void releaseFrameSlot()
{
if (frameSlot != NO_SLOT)
{
factory.framePool.release(frameSlot);
frameSlot = NO_SLOT;
frameSlotLimit = 0;
}
}
private void releaseHeadersSlot()
{
if (headersSlotIndex != NO_SLOT)
{
factory.headersPool.release(headersSlotIndex);
headersSlotIndex = NO_SLOT;
headersSlotOffset = 0;
}
}
private void doResetNetworkAndCleanup()
{
factory.doReset(networkReply, networkRouteId, networkId, factory.supplyTrace.getAsLong());
doCleanup();
}
private void doCleanup()
{
releaseFrameSlot();
releaseHeadersSlot();
http2Streams.values().forEach(this::closeStream);
http2Streams.clear();
}
private void followRoute(
int streamId,
Http2StreamState state,
RouteFW route)
{
final long applicationRouteId = route.correlationId();
HttpWriter httpWriter = factory.httpWriter;
Http2Stream stream = newStream(streamId, state, applicationRouteId, httpWriter);
stream.contentLength = headersContext.contentLength;
HttpBeginExFW beginEx = factory.httpBeginExRW.build();
httpWriter.doHttpBegin(stream.applicationTarget, applicationRouteId, stream.applicationId, traceId, stream.correlationId,
beginEx.buffer(), beginEx.offset(), beginEx.sizeof());
router.setThrottle(stream.applicationId, stream::onThrottle);
if (state == HALF_CLOSED_REMOTE)
{
// Deferring until WINDOW is received
stream.endDeferred = true;
}
}
// No route for the HTTP2 request, send 404 on the corresponding HTTP2 stream
private void noRoute(
int streamId)
{
send404(streamId);
}
// No route for the HTTP2 request, send 404 on the corresponding HTTP2 stream
void send404(
int streamId)
{
ListFW headers =
factory.headersRW.wrap(factory.errorBuf, 0, factory.errorBuf.capacity())
.item(b -> b.name(":status").value("404"))
.build();
writeScheduler.headers(0, streamId, Http2Flags.END_STREAM, headers);
if ((streamId & 0x01L) == 0x00L)
{
factory.counters.pushHeadersFramesWritten.getAsLong();
}
else
{
factory.counters.headersFramesWritten.getAsLong();
}
}
void closeStream(
Http2Stream stream)
{
if (stream.state != CLOSED)
{
stream.state = CLOSED;
if (stream.isClientInitiated())
{
clientStreamCount--;
}
else
{
promisedStreamCount--;
}
factory.correlations.remove(stream.correlationId);
http2Streams.remove(stream.http2StreamId);
stream.close();
}
}
private RouteFW resolveTarget(
Map headers)
{
MessagePredicate filter = (t, b, o, l) ->
{
RouteFW route = factory.routeRO.wrap(b, o, o + l);
OctetsFW extension = route.extension();
Map routeHeaders;
if (extension.sizeof() == 0)
{
routeHeaders = EMPTY_HEADERS;
}
else
{
final HttpRouteExFW routeEx = extension.get(factory.httpRouteExRO::wrap);
routeHeaders = new LinkedHashMap<>();
routeEx.headers().forEach(h -> routeHeaders.put(h.name().asString(), h.value().asString()));
}
return headers.entrySet().containsAll(routeHeaders.entrySet());
};
return router.resolve(networkRouteId, authorization, filter, wrapRoute);
}
void handleWindow(
WindowFW windowRO)
{
writeScheduler.onWindow(windowRO.trace());
}
void error(
Http2ErrorCode errorCode)
{
writeScheduler.goaway(lastStreamId, errorCode);
factory.counters.goawayFramesWritten.getAsLong();
factory.doReset(network, networkRouteId, networkId, factory.supplyTrace.getAsLong());
factory.doAbort(networkReply, networkRouteId, networkReplyId, factory.supplyTrace.getAsLong());
http2Streams.forEach((i, s) -> s.onError(traceId));
doCleanup();
}
void doRstByUs(
Http2Stream stream,
Http2ErrorCode errorCode)
{
stream.onReset(factory.supplyTrace.getAsLong());
doRstStream(stream.http2StreamId, errorCode);
closeStream(stream);
}
private void doRstStream(
int streamId,
Http2ErrorCode errorCode)
{
writeScheduler.rst(streamId, errorCode);
factory.counters.resetStreamFramesWritten.getAsLong();
}
private int nextPromisedId()
{
maxPushPromiseStreamId += 2;
return maxPushPromiseStreamId;
}
/*
* @param streamId corresponding http2 stream-id on which service response
* will be sent
* @return a stream id on which PUSH_PROMISE can be sent
* -1 otherwise
*/
private int findPushId(
int streamId)
{
if (remoteSettings.enablePush && promisedStreamCount +1 < remoteSettings.maxConcurrentStreams)
{
// PUSH_PROMISE frames MUST only be sent on a peer-initiated stream
if (streamId%2 == 0)
{
// Find a stream on which PUSH_PROMISE can be sent
return http2Streams.entrySet()
.stream()
.map(Map.Entry::getValue)
.filter(s -> (s.http2StreamId & 0x01) == 1) // client-initiated stream
.filter(s -> s.state == OPEN || s.state == HALF_CLOSED_REMOTE)
.mapToInt(s -> s.http2StreamId)
.findAny()
.orElse(-1);
}
else
{
return streamId; // client-initiated stream
}
}
return -1;
}
private void doPromisedRequest(
int http2StreamId,
long authorization,
ListFW headers)
{
Map headersMap = new HashMap<>();
headers.forEach(
httpHeader -> headersMap.put(httpHeader.name().asString(), httpHeader.value().asString()));
RouteFW route = resolveTarget(headersMap);
final long applicationRouteId = route.correlationId();
HttpWriter httpWriter = factory.httpWriter;
Http2Stream http2Stream = newStream(http2StreamId, HALF_CLOSED_REMOTE, applicationRouteId, httpWriter);
final MessageConsumer applicationTarget = http2Stream.applicationTarget;
long targetId = http2Stream.applicationId;
httpWriter.doHttpBegin(applicationTarget, applicationRouteId, targetId, factory.supplyTrace.getAsLong(), authorization,
http2Stream.correlationId, hs -> headers.forEach(h -> hs.item(b -> b.name(h.name())
.value(h.value()))));
router.setThrottle(targetId, http2Stream::onThrottle);
http2Stream.endDeferred = true;
}
private Http2Stream newStream(
int http2StreamId,
Http2StreamState state,
long applicationRouteId,
HttpWriter httpWriter)
{
assert http2StreamId != 0;
Http2Stream http2Stream = new Http2Stream(factory, this, http2StreamId, state, applicationRouteId, httpWriter);
http2Streams.put(http2StreamId, http2Stream);
Correlation correlation = new Correlation(http2Stream.correlationId, networkReplyId, writeScheduler,
this::doPromisedRequest, this, http2StreamId, encodeContext, this::nextPromisedId, this::findPushId);
factory.correlations.put(http2Stream.correlationId, correlation);
if (http2Stream.isClientInitiated())
{
clientStreamCount++;
}
else
{
promisedStreamCount++;
}
return http2Stream;
}
private void validateHeaderFieldType(
HpackHeaderFieldFW hf)
{
if (!headersContext.error() && hf.type() == UNKNOWN)
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
}
}
private void dynamicTableSizeUpdate(
HpackHeaderFieldFW hf)
{
if (!headersContext.error())
{
switch (hf.type())
{
case INDEXED:
case LITERAL:
expectDynamicTableSizeUpdate = false;
break;
case UPDATE:
if (!expectDynamicTableSizeUpdate)
{
// dynamic table size update MUST occur at the beginning of the first header block
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
int maxTableSize = hf.tableSize();
if (maxTableSize > localSettings.headerTableSize)
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
decodeContext.updateSize(hf.tableSize());
break;
default:
break;
}
}
}
private void validatePseudoHeaders(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error())
{
if (name.capacity() > 0 && name.getByte(0) == ':')
{
// All pseudo-header fields MUST appear in the header block before regular header fields
if (headersContext.regularHeader)
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
// request pseudo-header fields MUST be one of :authority, :method, :path, :scheme,
int index = decodeContext.index(name);
switch (index)
{
case 1: // :authority
break;
case 2: // :method
headersContext.method++;
break;
case 4: // :path
if (value.capacity() > 0) // :path MUST not be empty
{
headersContext.path++;
}
break;
case 6: // :scheme
headersContext.scheme++;
break;
default:
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
}
else
{
headersContext.regularHeader = true;
}
}
}
private void validateTrailerFieldName(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error())
{
if (name.capacity() > 0 && name.getByte(0) == ':')
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
return;
}
}
}
private void connectionHeaders(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error() && name.equals(HpackContext.CONNECTION))
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
}
}
private void contentLengthHeader(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error() && name.equals(decodeContext.nameBuffer(28)))
{
String contentLength = value.getStringWithoutLengthUtf8(0, value.capacity());
headersContext.contentLength = Long.parseLong(contentLength);
}
}
// 8.1.2.2 TE header MUST NOT contain any value other than "trailers".
private void teHeader(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error() && name.equals(TE) && !value.equals(TRAILERS))
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
}
}
private void uppercaseHeaders(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error())
{
for(int i=0; i < name.capacity(); i++)
{
if (name.getByte(i) >= 'A' && name.getByte(i) <= 'Z')
{
headersContext.streamError = Http2ErrorCode.PROTOCOL_ERROR;
}
}
}
}
// Collect headers into map to resolve target
// TODO avoid this
private void collectHeaders(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error())
{
String nameStr = name.getStringWithoutLengthUtf8(0, name.capacity());
String valueStr = value.getStringWithoutLengthUtf8(0, value.capacity());
// TODO cookie needs to be appended with ';'
headersContext.headers.merge(nameStr, valueStr, (o, n) -> String.format("%s, %s", o, n));
}
}
// Writes HPACK header field to http representation in a buffer
private void mapToHttp(
DirectBuffer name,
DirectBuffer value)
{
if (!headersContext.error())
{
factory.httpBeginExRW.headersItem(item -> item.name(name, 0, name.capacity())
.value(value, 0, value.capacity()));
}
}
private void addDefaultPortToAuthorityIfNeeded()
{
String authority = headersContext.headers.get(":authority");
if (authority.indexOf(':') == -1)
{
String scheme = headersContext.headers.get(":scheme");
String defaultPort = scheme.equals("https") ? ":443" : ":80";
headersContext.headers.put(":authority", authority + defaultPort);
// rebuild http request as :authority header is modified
factory.httpBeginExRW.wrap(factory.scratch, 0, factory.scratch.capacity());
for(Map.Entry e : headersContext.headers.entrySet())
{
factory.httpBeginExRW.headersItem(item -> item.name(e.getKey())
.value(e.getValue()));
}
}
}
private void decodeHeaderField(
HpackHeaderFieldFW hf,
BiConsumer nameValue)
{
if (!headersContext.error())
{
decodeHF(hf, nameValue);
}
}
private void decodeHF(
HpackHeaderFieldFW hf,
BiConsumer nameValue)
{
int index;
DirectBuffer name = null;
DirectBuffer value = null;
switch (hf.type())
{
case INDEXED :
index = hf.index();
if (!decodeContext.valid(index))
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
name = decodeContext.nameBuffer(index);
value = decodeContext.valueBuffer(index);
nameValue.accept(name, value);
break;
case LITERAL :
HpackLiteralHeaderFieldFW literalRO = hf.literal();
if (literalRO.error())
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
switch (literalRO.nameType())
{
case INDEXED:
{
index = literalRO.nameIndex();
if (!decodeContext.valid(index))
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
name = decodeContext.nameBuffer(index);
HpackStringFW valueRO = literalRO.valueLiteral();
value = valueRO.payload();
if (valueRO.huffman())
{
MutableDirectBuffer dst = new UnsafeBuffer(new byte[4096]); // TODO
int length = HpackHuffman.decode(value, dst);
if (length == -1)
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
value = new UnsafeBuffer(dst, 0, length);
}
nameValue.accept(name, value);
}
break;
case NEW:
{
HpackStringFW nameRO = literalRO.nameLiteral();
name = nameRO.payload();
if (nameRO.huffman())
{
MutableDirectBuffer dst = new UnsafeBuffer(new byte[4096]); // TODO
int length = HpackHuffman.decode(name, dst);
if (length == -1)
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
name = new UnsafeBuffer(dst, 0, length);
}
HpackStringFW valueRO = literalRO.valueLiteral();
value = valueRO.payload();
if (valueRO.huffman())
{
MutableDirectBuffer dst = new UnsafeBuffer(new byte[4096]); // TODO
int length = HpackHuffman.decode(value, dst);
if (length == -1)
{
headersContext.connectionError = Http2ErrorCode.COMPRESSION_ERROR;
return;
}
value = new UnsafeBuffer(dst, 0, length);
}
nameValue.accept(name, value);
}
break;
}
if (literalRO.literalType() == INCREMENTAL_INDEXING)
{
// make a copy for name and value as they go into dynamic table (outlives current frame)
MutableDirectBuffer nameCopy = new UnsafeBuffer(new byte[name.capacity()]);
nameCopy.putBytes(0, name, 0, name.capacity());
MutableDirectBuffer valueCopy = new UnsafeBuffer(new byte[value.capacity()]);
valueCopy.putBytes(0, value, 0, value.capacity());
decodeContext.add(nameCopy, valueCopy);
}
break;
default:
break;
}
}
void mapPushPromise(
ListFW httpHeaders,
HpackHeaderBlockFW.Builder builder)
{
httpHeaders.forEach(h -> builder.header(b -> mapHeader(h, b)));
}
void mapHeaders(
ListFW httpHeaders,
HpackHeaderBlockFW.Builder builder)
{
encodeHeadersContext.reset();
httpHeaders.forEach(this::status) // checks if there is :status
.forEach(this::accessControlAllowOrigin) // checks if there is access-control-allow-origin
.forEach(this::serverHeader) // checks if there is server
.forEach(this::connectionHeaders); // collects all connection headers
if (!encodeHeadersContext.status)
{
builder.header(b -> b.indexed(8)); // no mandatory :status header, add :status: 200
}
httpHeaders.forEach(h ->
{
if (validHeader(h))
{
builder.header(b -> mapHeader(h, b));
}
});
if (factory.config.accessControlAllowOrigin() && !encodeHeadersContext.accessControlAllowOrigin)
{
builder.header(b -> b.literal(l -> l.type(WITHOUT_INDEXING).name(20).value(DEFAULT_ACCESS_CONTROL_ALLOW_ORIGIN)));
}
// add configured Server header if there is no Server header in response
if (factory.config.serverHeader() != null && !encodeHeadersContext.serverHeader)
{
DirectBuffer server = factory.config.serverHeader();
builder.header(b -> b.literal(l -> l.type(WITHOUT_INDEXING).name(54).value(server)));
}
}
private void status(
HttpHeaderFW httpHeader)
{
if (!encodeHeadersContext.status)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
factory.valueRO.wrap(value.buffer(), value.offset() + 2, value.sizeof() - 2);
if (factory.nameRO.equals(encodeContext.nameBuffer(8)))
{
encodeHeadersContext.status = true;
}
}
}
// Checks if response has access-control-allow-origin header
private void accessControlAllowOrigin(
HttpHeaderFW httpHeader)
{
if (factory.config.accessControlAllowOrigin() && !encodeHeadersContext.accessControlAllowOrigin)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
factory.valueRO.wrap(value.buffer(), value.offset() + 2, value.sizeof() - 2);
if (factory.nameRO.equals(encodeContext.nameBuffer(20)))
{
encodeHeadersContext.accessControlAllowOrigin = true;
}
}
}
// Checks if response has server header
private void serverHeader(
HttpHeaderFW httpHeader)
{
if (factory.config.serverHeader() != null && !encodeHeadersContext.serverHeader)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
factory.valueRO.wrap(value.buffer(), value.offset() + 2, value.sizeof() - 2);
if (factory.nameRO.equals(encodeContext.nameBuffer(54)))
{
encodeHeadersContext.serverHeader = true;
}
}
}
private void connectionHeaders(
HttpHeaderFW httpHeader)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
if (factory.nameRO.equals(CONNECTION))
{
String[] headers = value.asString().split(",");
for (String header : headers)
{
encodeHeadersContext.connectionHeaders.add(header.trim());
}
}
}
private boolean validHeader(
HttpHeaderFW httpHeader)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
factory.valueRO.wrap(value.buffer(), value.offset() + 2, value.sizeof() - 2);
// Removing 8.1.2.1 Pseudo-Header Fields
// Not sending error as it will allow requests to loop back
if (factory.nameRO.equals(encodeContext.nameBuffer(1)) || // :authority
factory.nameRO.equals(encodeContext.nameBuffer(2)) || // :method
factory.nameRO.equals(encodeContext.nameBuffer(4)) || // :path
factory.nameRO.equals(encodeContext.nameBuffer(6))) // :scheme
{
return false;
}
// Removing 8.1.2.2 connection-specific header fields from response
if (factory.nameRO.equals(encodeContext.nameBuffer(57)) || // transfer-encoding
factory.nameRO.equals(CONNECTION) ||
factory.nameRO.equals(KEEP_ALIVE) ||
factory.nameRO.equals(PROXY_CONNECTION) ||
factory.nameRO.equals(UPGRADE))
{
return false;
}
// Removing any header that is nominated by Connection header field
for(String connectionHeader: encodeHeadersContext.connectionHeaders)
{
if (name.asString().equals(connectionHeader))
{
return false;
}
}
return true;
}
// Map http1.1 header to http2 header field in HEADERS, PUSH_PROMISE request
private void mapHeader(
HttpHeaderFW httpHeader,
HpackHeaderFieldFW.Builder builder)
{
StringFW name = httpHeader.name();
String16FW value = httpHeader.value();
factory.nameRO.wrap(name.buffer(), name.offset() + 1, name.sizeof() - 1); // +1, -1 for length-prefixed buffer
factory.valueRO.wrap(value.buffer(), value.offset() + 2, value.sizeof() - 2);
int index = encodeContext.index(factory.nameRO, factory.valueRO);
if (index != -1)
{
// Indexed
builder.indexed(index);
}
else
{
// Literal
builder.literal(literalBuilder -> buildLiteral(literalBuilder, encodeContext));
}
}
// Building Literal representation of header field
// TODO dynamic table, huffman, never indexed
private void buildLiteral(
HpackLiteralHeaderFieldFW.Builder builder,
HpackContext hpackContext)
{
int nameIndex = hpackContext.index(factory.nameRO);
builder.type(WITHOUT_INDEXING);
if (nameIndex != -1)
{
builder.name(nameIndex);
}
else
{
builder.name(factory.nameRO, 0, factory.nameRO.capacity());
}
builder.value(factory.valueRO, 0, factory.valueRO.capacity());
}
void handleHttpBegin(
BeginFW begin,
MessageConsumer applicationReplyThrottle,
long applicationRouteId,
long applicationReplyId,
Correlation correlation)
{
OctetsFW extension = begin.extension();
Http2Stream stream = http2Streams.get(correlation.http2StreamId);
if (stream == null)
{
factory.doReset(applicationReplyThrottle, applicationRouteId, applicationReplyId, factory.supplyTrace.getAsLong());
}
else
{
stream.applicationReplyThrottle = applicationReplyThrottle;
stream.applicationReplyId = applicationReplyId;
stream.sendHttpWindow(factory.supplyTrace.getAsLong());
if (extension.sizeof() > 0)
{
HttpBeginExFW beginEx = extension.get(factory.beginExRO::wrap);
writeScheduler.headers(begin.trace(), correlation.http2StreamId, Http2Flags.NONE, beginEx.headers());
if ((correlation.http2StreamId & 0x01L) == 0x00L)
{
factory.counters.pushHeadersFramesWritten.getAsLong();
}
else
{
factory.counters.headersFramesWritten.getAsLong();
}
}
}
}
void handleHttpData(
DataFW dataRO,
Correlation correlation)
{
OctetsFW extension = dataRO.extension();
OctetsFW payload = dataRO.payload();
long traceId = dataRO.trace();
if (extension.sizeof() > 0)
{
int pushStreamId = correlation.pushStreamIds.applyAsInt(correlation.http2StreamId);
if (pushStreamId != -1)
{
int promisedStreamId = correlation.promisedStreamIds.getAsInt();
Http2DataExFW dataEx = extension.get(factory.dataExRO::wrap);
writeScheduler.pushPromise(traceId, pushStreamId, promisedStreamId, dataEx.headers());
correlation.pushHandler.accept(promisedStreamId, dataRO.authorization(), dataEx.headers());
factory.counters.pushPromiseFramesWritten.getAsLong();
}
else
{
factory.counters.pushPromiseFramesSkipped.getAsLong();
}
}
if (payload != null)
{
Http2Stream stream = http2Streams.get(correlation.http2StreamId);
if (stream != null)
{
stream.applicationReplyBudget -= dataRO.length() + dataRO.padding();
if (stream.applicationReplyBudget < 0)
{
doRstByUs(stream, Http2ErrorCode.INTERNAL_ERROR);
return;
}
}
writeScheduler.data(traceId, correlation.http2StreamId, payload.buffer(), payload.offset(), payload.sizeof());
factory.counters.dataFramesWritten.getAsLong();
}
}
void handleHttpEnd(
EndFW end,
Correlation correlation)
{
Http2Stream stream = http2Streams.get(correlation.http2StreamId);
if (stream != null)
{
stream.onHttpEnd(end.trace());
}
}
void handleHttpAbort(
AbortFW abort,
Correlation correlation)
{
Http2Stream stream = http2Streams.get(correlation.http2StreamId);
if (stream != null)
{
stream.onHttpAbort();
}
}
private static final class HeadersContext
{
Http2ErrorCode connectionError;
Map headers = new LinkedHashMap<>();
int method;
int scheme;
int path;
boolean regularHeader;
Http2ErrorCode streamError;
long contentLength = -1;
void reset()
{
connectionError = null;
headers.clear();
method = 0;
scheme = 0;
path = 0;
regularHeader = false;
streamError = null;
contentLength = -1;
}
boolean error()
{
return streamError != null || connectionError != null;
}
}
private static final class EncodeHeadersContext
{
boolean status;
boolean accessControlAllowOrigin;
boolean serverHeader;
final List connectionHeaders = new ArrayList<>();
void reset()
{
status = false;
accessControlAllowOrigin = false;
serverHeader = false;
connectionHeaders.clear();
}
}
@FunctionalInterface
private interface DecoderState
{
int decode(DirectBuffer buffer, int offset, int length);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy