All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.netty.handler.codec.http2.Http2MultiplexCodec Maven / Gradle / Ivy

Go to download

This artifact provides a single jar that contains all classes required to use remote Jakarta Enterprise Beans and Jakarta Messaging, including all dependencies. It is intended for use by those not using maven, maven users should just import the Jakarta Enterprise Beans and Jakarta Messaging BOM's instead (shaded JAR's cause lots of problems with maven, as it is very easy to inadvertently end up with different versions on classes on the class path).

There is a newer version: 35.0.0.Beta1
Show newest version
/*
 * Copyright 2016 The Netty Project
 *
 * The Netty 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 io.netty.handler.codec.http2;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.netty.channel.Channel;
import io.netty.channel.ChannelConfig;
import io.netty.channel.ChannelFuture;
import io.netty.channel.ChannelFutureListener;
import io.netty.channel.ChannelHandler;
import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelId;
import io.netty.channel.ChannelMetadata;
import io.netty.channel.ChannelOutboundBuffer;
import io.netty.channel.ChannelPipeline;
import io.netty.channel.ChannelProgressivePromise;
import io.netty.channel.ChannelPromise;
import io.netty.channel.DefaultChannelConfig;
import io.netty.channel.DefaultChannelPipeline;
import io.netty.channel.EventLoop;
import io.netty.channel.MessageSizeEstimator;
import io.netty.channel.RecvByteBufAllocator;
import io.netty.channel.RecvByteBufAllocator.Handle;
import io.netty.channel.VoidChannelPromise;
import io.netty.channel.WriteBufferWaterMark;
import io.netty.util.DefaultAttributeMap;
import io.netty.util.ReferenceCountUtil;
import io.netty.util.ReferenceCounted;
import io.netty.util.internal.StringUtil;
import io.netty.util.internal.ThrowableUtil;
import io.netty.util.internal.UnstableApi;
import io.netty.util.internal.logging.InternalLogger;
import io.netty.util.internal.logging.InternalLoggerFactory;

import java.net.SocketAddress;
import java.nio.channels.ClosedChannelException;
import java.util.ArrayDeque;
import java.util.Queue;
import java.util.concurrent.RejectedExecutionException;

import static io.netty.handler.codec.http2.Http2CodecUtil.HTTP_UPGRADE_STREAM_ID;
import static io.netty.handler.codec.http2.Http2CodecUtil.isStreamIdValid;
import static io.netty.handler.codec.http2.Http2Error.INTERNAL_ERROR;
import static io.netty.handler.codec.http2.Http2Exception.connectionError;
import static java.lang.Math.min;

/**
 * An HTTP/2 handler that creates child channels for each stream.
 *
 * 

When a new stream is created, a new {@link Channel} is created for it. Applications send and * receive {@link Http2StreamFrame}s on the created channel. {@link ByteBuf}s cannot be processed by the channel; * all writes that reach the head of the pipeline must be an instance of {@link Http2StreamFrame}. Writes that reach * the head of the pipeline are processed directly by this handler and cannot be intercepted. * *

The child channel will be notified of user events that impact the stream, such as {@link * Http2GoAwayFrame} and {@link Http2ResetFrame}, as soon as they occur. Although {@code * Http2GoAwayFrame} and {@code Http2ResetFrame} signify that the remote is ignoring further * communication, closing of the channel is delayed until any inbound queue is drained with {@link * Channel#read()}, which follows the default behavior of channels in Netty. Applications are * free to close the channel in response to such events if they don't have use for any queued * messages. Any connection level events like {@link Http2SettingsFrame} and {@link Http2GoAwayFrame} * will be processed internally and also propagated down the pipeline for other handlers to act on. * *

Outbound streams are supported via the {@link Http2StreamChannelBootstrap}. * *

{@link ChannelConfig#setMaxMessagesPerRead(int)} and {@link ChannelConfig#setAutoRead(boolean)} are supported. * *

Reference Counting

* * Some {@link Http2StreamFrame}s implement the {@link ReferenceCounted} interface, as they carry * reference counted objects (e.g. {@link ByteBuf}s). The multiplex codec will call {@link ReferenceCounted#retain()} * before propagating a reference counted object through the pipeline, and thus an application handler needs to release * such an object after having consumed it. For more information on reference counting take a look at * http://netty.io/wiki/reference-counted-objects.html * *

Channel Events

* * A child channel becomes active as soon as it is registered to an {@link EventLoop}. Therefore, an active channel * does not map to an active HTTP/2 stream immediately. Only once a {@link Http2HeadersFrame} has been successfully sent * or received, does the channel map to an active HTTP/2 stream. In case it is not possible to open a new HTTP/2 stream * (i.e. due to the maximum number of active streams being exceeded), the child channel receives an exception * indicating the cause and is closed immediately thereafter. * *

Writability and Flow Control

* * A child channel observes outbound/remote flow control via the channel's writability. A channel only becomes writable * when it maps to an active HTTP/2 stream and the stream's flow control window is greater than zero. A child channel * does not know about the connection-level flow control window. {@link ChannelHandler}s are free to ignore the * channel's writability, in which case the excessive writes will be buffered by the parent channel. It's important to * note that only {@link Http2DataFrame}s are subject to HTTP/2 flow control. */ @UnstableApi public class Http2MultiplexCodec extends Http2FrameCodec { private static final InternalLogger logger = InternalLoggerFactory.getInstance(DefaultHttp2StreamChannel.class); private static final ChannelFutureListener CHILD_CHANNEL_REGISTRATION_LISTENER = new ChannelFutureListener() { @Override public void operationComplete(ChannelFuture future) { registerDone(future); } }; private static final ChannelMetadata METADATA = new ChannelMetadata(false, 16); private static final ClosedChannelException CLOSED_CHANNEL_EXCEPTION = ThrowableUtil.unknownStackTrace( new ClosedChannelException(), DefaultHttp2StreamChannel.Http2ChannelUnsafe.class, "write(...)"); /** * Number of bytes to consider non-payload messages. 9 is arbitrary, but also the minimum size of an HTTP/2 frame. * Primarily is non-zero. */ private static final int MIN_HTTP2_FRAME_SIZE = 9; /** * Returns the flow-control size for DATA frames, and 0 for all other frames. */ private static final class FlowControlledFrameSizeEstimator implements MessageSizeEstimator { static final FlowControlledFrameSizeEstimator INSTANCE = new FlowControlledFrameSizeEstimator(); static final MessageSizeEstimator.Handle HANDLE_INSTANCE = new MessageSizeEstimator.Handle() { @Override public int size(Object msg) { return msg instanceof Http2DataFrame ? // Guard against overflow. (int) min(Integer.MAX_VALUE, ((Http2DataFrame) msg).initialFlowControlledBytes() + (long) MIN_HTTP2_FRAME_SIZE) : MIN_HTTP2_FRAME_SIZE; } }; @Override public Handle newHandle() { return HANDLE_INSTANCE; } } private final ChannelHandler inboundStreamHandler; private final ChannelHandler upgradeStreamHandler; private int initialOutboundStreamWindow = Http2CodecUtil.DEFAULT_WINDOW_SIZE; private boolean parentReadInProgress; private int idCount; // Linked-List for DefaultHttp2StreamChannel instances that need to be processed by channelReadComplete(...) private DefaultHttp2StreamChannel head; private DefaultHttp2StreamChannel tail; // Need to be volatile as accessed from within the DefaultHttp2StreamChannel in a multi-threaded fashion. volatile ChannelHandlerContext ctx; Http2MultiplexCodec(Http2ConnectionEncoder encoder, Http2ConnectionDecoder decoder, Http2Settings initialSettings, ChannelHandler inboundStreamHandler, ChannelHandler upgradeStreamHandler) { super(encoder, decoder, initialSettings); this.inboundStreamHandler = inboundStreamHandler; this.upgradeStreamHandler = upgradeStreamHandler; } @Override public void onHttpClientUpgrade() throws Http2Exception { // We must have an upgrade handler or else we can't handle the stream if (upgradeStreamHandler == null) { throw connectionError(INTERNAL_ERROR, "Client is misconfigured for upgrade requests"); } // Creates the Http2Stream in the Connection. super.onHttpClientUpgrade(); // Now make a new FrameStream, set it's underlying Http2Stream, and initialize it. Http2MultiplexCodecStream codecStream = newStream(); codecStream.setStreamAndProperty(streamKey, connection().stream(HTTP_UPGRADE_STREAM_ID)); onHttp2UpgradeStreamInitialized(ctx, codecStream); } private static void registerDone(ChannelFuture future) { // Handle any errors that occurred on the local thread while registering. Even though // failures can happen after this point, they will be handled by the channel by closing the // childChannel. if (!future.isSuccess()) { Channel childChannel = future.channel(); if (childChannel.isRegistered()) { childChannel.close(); } else { childChannel.unsafe().closeForcibly(); } } } @Override public final void handlerAdded0(ChannelHandlerContext ctx) throws Exception { if (ctx.executor() != ctx.channel().eventLoop()) { throw new IllegalStateException("EventExecutor must be EventLoop of Channel"); } this.ctx = ctx; } @Override public final void handlerRemoved0(ChannelHandlerContext ctx) throws Exception { super.handlerRemoved0(ctx); // Unlink the linked list to guard against GC nepotism. DefaultHttp2StreamChannel ch = head; while (ch != null) { DefaultHttp2StreamChannel curr = ch; ch = curr.next; curr.next = curr.previous = null; } head = tail = null; } @Override Http2MultiplexCodecStream newStream() { return new Http2MultiplexCodecStream(); } @Override final void onHttp2Frame(ChannelHandlerContext ctx, Http2Frame frame) { if (frame instanceof Http2StreamFrame) { Http2StreamFrame streamFrame = (Http2StreamFrame) frame; ((Http2MultiplexCodecStream) streamFrame.stream()).channel.fireChildRead(streamFrame); } else if (frame instanceof Http2GoAwayFrame) { onHttp2GoAwayFrame(ctx, (Http2GoAwayFrame) frame); // Allow other handlers to act on GOAWAY frame ctx.fireChannelRead(frame); } else if (frame instanceof Http2SettingsFrame) { Http2Settings settings = ((Http2SettingsFrame) frame).settings(); if (settings.initialWindowSize() != null) { initialOutboundStreamWindow = settings.initialWindowSize(); } // Allow other handlers to act on SETTINGS frame ctx.fireChannelRead(frame); } else { // Send any other frames down the pipeline ctx.fireChannelRead(frame); } } private void onHttp2UpgradeStreamInitialized(ChannelHandlerContext ctx, Http2MultiplexCodecStream stream) { assert stream.state() == Http2Stream.State.HALF_CLOSED_LOCAL; DefaultHttp2StreamChannel ch = new DefaultHttp2StreamChannel(stream, true); ch.outboundClosed = true; // Add our upgrade handler to the channel and then register the channel. // The register call fires the channelActive, etc. ch.pipeline().addLast(upgradeStreamHandler); ChannelFuture future = ctx.channel().eventLoop().register(ch); if (future.isDone()) { registerDone(future); } else { future.addListener(CHILD_CHANNEL_REGISTRATION_LISTENER); } } @Override final void onHttp2StreamStateChanged(ChannelHandlerContext ctx, Http2FrameStream stream) { Http2MultiplexCodecStream s = (Http2MultiplexCodecStream) stream; switch (stream.state()) { case HALF_CLOSED_REMOTE: case OPEN: if (s.channel != null) { // ignore if child channel was already created. break; } // fall-trough ChannelFuture future = ctx.channel().eventLoop().register(new DefaultHttp2StreamChannel(s, false)); if (future.isDone()) { registerDone(future); } else { future.addListener(CHILD_CHANNEL_REGISTRATION_LISTENER); } break; case CLOSED: DefaultHttp2StreamChannel channel = s.channel; if (channel != null) { channel.streamClosed(); } break; default: // ignore for now break; } } @Override final void onHttp2StreamWritabilityChanged(ChannelHandlerContext ctx, Http2FrameStream stream, boolean writable) { (((Http2MultiplexCodecStream) stream).channel).writabilityChanged(writable); } // TODO: This is most likely not the best way to expose this, need to think more about it. final Http2StreamChannel newOutboundStream() { return new DefaultHttp2StreamChannel(newStream(), true); } @Override final void onHttp2FrameStreamException(ChannelHandlerContext ctx, Http2FrameStreamException cause) { Http2FrameStream stream = cause.stream(); DefaultHttp2StreamChannel childChannel = ((Http2MultiplexCodecStream) stream).channel; try { childChannel.pipeline().fireExceptionCaught(cause.getCause()); } finally { childChannel.unsafe().closeForcibly(); } } private boolean isChildChannelInReadPendingQueue(DefaultHttp2StreamChannel childChannel) { return childChannel.previous != null || childChannel.next != null || head == childChannel; } final void tryAddChildChannelToReadPendingQueue(DefaultHttp2StreamChannel childChannel) { if (!isChildChannelInReadPendingQueue(childChannel)) { addChildChannelToReadPendingQueue(childChannel); } } final void addChildChannelToReadPendingQueue(DefaultHttp2StreamChannel childChannel) { if (tail == null) { assert head == null; tail = head = childChannel; } else { childChannel.previous = tail; tail.next = childChannel; tail = childChannel; } } private void tryRemoveChildChannelFromReadPendingQueue(DefaultHttp2StreamChannel childChannel) { if (isChildChannelInReadPendingQueue(childChannel)) { removeChildChannelFromReadPendingQueue(childChannel); } } private void removeChildChannelFromReadPendingQueue(DefaultHttp2StreamChannel childChannel) { DefaultHttp2StreamChannel previous = childChannel.previous; if (childChannel.next != null) { childChannel.next.previous = previous; } else { tail = tail.previous; // If there is no next, this childChannel is the tail, so move the tail back. } if (previous != null) { previous.next = childChannel.next; } else { head = head.next; // If there is no previous, this childChannel is the head, so move the tail forward. } childChannel.next = childChannel.previous = null; } private void onHttp2GoAwayFrame(ChannelHandlerContext ctx, final Http2GoAwayFrame goAwayFrame) { try { forEachActiveStream(new Http2FrameStreamVisitor() { @Override public boolean visit(Http2FrameStream stream) { final int streamId = stream.id(); final DefaultHttp2StreamChannel childChannel = ((Http2MultiplexCodecStream) stream).channel; if (streamId > goAwayFrame.lastStreamId() && connection().local().isValidStreamId(streamId)) { childChannel.pipeline().fireUserEventTriggered(goAwayFrame.retainedDuplicate()); } return true; } }); } catch (Http2Exception e) { ctx.fireExceptionCaught(e); ctx.close(); } } /** * Notifies any child streams of the read completion. */ @Override public final void channelReadComplete(ChannelHandlerContext ctx) throws Exception { try { onChannelReadComplete(ctx); } finally { parentReadInProgress = false; tail = head = null; // We always flush as this is what Http2ConnectionHandler does for now. flush0(ctx); } channelReadComplete0(ctx); } @Override public final void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { parentReadInProgress = true; super.channelRead(ctx, msg); } final void onChannelReadComplete(ChannelHandlerContext ctx) { // If we have many child channel we can optimize for the case when multiple call flush() in // channelReadComplete(...) callbacks and only do it once as otherwise we will end-up with multiple // write calls on the socket which is expensive. DefaultHttp2StreamChannel current = head; while (current != null) { DefaultHttp2StreamChannel childChannel = current; // Clear early in case fireChildReadComplete() causes it to need to be re-processed current = current.next; childChannel.next = childChannel.previous = null; childChannel.fireChildReadComplete(); } } // Allow to override for testing void flush0(ChannelHandlerContext ctx) { flush(ctx); } /** * Return bytes to flow control. *

* Package private to allow to override for testing * @param ctx The {@link ChannelHandlerContext} associated with the parent channel. * @param stream The object representing the HTTP/2 stream. * @param bytes The number of bytes to return to flow control. * @return {@code true} if a frame has been written as a result of this method call. * @throws Http2Exception If this operation violates the flow control limits. */ boolean onBytesConsumed(@SuppressWarnings("unused") ChannelHandlerContext ctx, Http2FrameStream stream, int bytes) throws Http2Exception { return consumeBytes(stream.id(), bytes); } // Allow to extend for testing static class Http2MultiplexCodecStream extends DefaultHttp2FrameStream { DefaultHttp2StreamChannel channel; } private boolean initialWritability(DefaultHttp2FrameStream stream) { // If the stream id is not valid yet we will just mark the channel as writable as we will be notified // about non-writability state as soon as the first Http2HeaderFrame is written (if needed). // This should be good enough and simplify things a lot. return !isStreamIdValid(stream.id()) || isWritable(stream); } // TODO: Handle writability changes due writing from outside the eventloop. private final class DefaultHttp2StreamChannel extends DefaultAttributeMap implements Http2StreamChannel { private final Http2StreamChannelConfig config = new Http2StreamChannelConfig(this); private final Http2ChannelUnsafe unsafe = new Http2ChannelUnsafe(); private final ChannelId channelId; private final ChannelPipeline pipeline; private final DefaultHttp2FrameStream stream; private final ChannelPromise closePromise; private final boolean outbound; private volatile boolean registered; // We start with the writability of the channel when creating the StreamChannel. private volatile boolean writable; private boolean outboundClosed; /** * This variable represents if a read is in progress for the current channel. Note that depending upon the * {@link RecvByteBufAllocator} behavior a read may extend beyond the {@link Http2ChannelUnsafe#beginRead()} * method scope. The {@link Http2ChannelUnsafe#beginRead()} loop may drain all pending data, and then if the * parent channel is reading this channel may still accept frames. */ private boolean readInProgress; private Queue inboundBuffer; /** {@code true} after the first HEADERS frame has been written **/ private boolean firstFrameWritten; // Currently the child channel and parent channel are always on the same EventLoop thread. This allows us to // extend the read loop of a child channel if the child channel drains its queued data during read, and the // parent channel is still in its read loop. The next/previous links build a doubly linked list that the parent // channel will iterate in its channelReadComplete to end the read cycle for each child channel in the list. DefaultHttp2StreamChannel next; DefaultHttp2StreamChannel previous; DefaultHttp2StreamChannel(DefaultHttp2FrameStream stream, boolean outbound) { this.stream = stream; this.outbound = outbound; writable = initialWritability(stream); ((Http2MultiplexCodecStream) stream).channel = this; pipeline = new DefaultChannelPipeline(this) { @Override protected void incrementPendingOutboundBytes(long size) { // Do thing for now } @Override protected void decrementPendingOutboundBytes(long size) { // Do thing for now } }; closePromise = pipeline.newPromise(); channelId = new Http2StreamChannelId(parent().id(), ++idCount); } @Override public Http2FrameStream stream() { return stream; } void streamClosed() { unsafe.readEOS(); // Attempt to drain any queued data from the queue and deliver it to the application before closing this // channel. unsafe.doBeginRead(); } @Override public ChannelMetadata metadata() { return METADATA; } @Override public ChannelConfig config() { return config; } @Override public boolean isOpen() { return !closePromise.isDone(); } @Override public boolean isActive() { return isOpen(); } @Override public boolean isWritable() { return writable; } @Override public ChannelId id() { return channelId; } @Override public EventLoop eventLoop() { return parent().eventLoop(); } @Override public Channel parent() { return ctx.channel(); } @Override public boolean isRegistered() { return registered; } @Override public SocketAddress localAddress() { return parent().localAddress(); } @Override public SocketAddress remoteAddress() { return parent().remoteAddress(); } @Override public ChannelFuture closeFuture() { return closePromise; } @Override public long bytesBeforeUnwritable() { // TODO: Do a proper impl return config().getWriteBufferHighWaterMark(); } @Override public long bytesBeforeWritable() { // TODO: Do a proper impl return 0; } @Override public Unsafe unsafe() { return unsafe; } @Override public ChannelPipeline pipeline() { return pipeline; } @Override public ByteBufAllocator alloc() { return config().getAllocator(); } @Override public Channel read() { pipeline().read(); return this; } @Override public Channel flush() { pipeline().flush(); return this; } @Override public ChannelFuture bind(SocketAddress localAddress) { return pipeline().bind(localAddress); } @Override public ChannelFuture connect(SocketAddress remoteAddress) { return pipeline().connect(remoteAddress); } @Override public ChannelFuture connect(SocketAddress remoteAddress, SocketAddress localAddress) { return pipeline().connect(remoteAddress, localAddress); } @Override public ChannelFuture disconnect() { return pipeline().disconnect(); } @Override public ChannelFuture close() { return pipeline().close(); } @Override public ChannelFuture deregister() { return pipeline().deregister(); } @Override public ChannelFuture bind(SocketAddress localAddress, ChannelPromise promise) { return pipeline().bind(localAddress, promise); } @Override public ChannelFuture connect(SocketAddress remoteAddress, ChannelPromise promise) { return pipeline().connect(remoteAddress, promise); } @Override public ChannelFuture connect(SocketAddress remoteAddress, SocketAddress localAddress, ChannelPromise promise) { return pipeline().connect(remoteAddress, localAddress, promise); } @Override public ChannelFuture disconnect(ChannelPromise promise) { return pipeline().disconnect(promise); } @Override public ChannelFuture close(ChannelPromise promise) { return pipeline().close(promise); } @Override public ChannelFuture deregister(ChannelPromise promise) { return pipeline().deregister(promise); } @Override public ChannelFuture write(Object msg) { return pipeline().write(msg); } @Override public ChannelFuture write(Object msg, ChannelPromise promise) { return pipeline().write(msg, promise); } @Override public ChannelFuture writeAndFlush(Object msg, ChannelPromise promise) { return pipeline().writeAndFlush(msg, promise); } @Override public ChannelFuture writeAndFlush(Object msg) { return pipeline().writeAndFlush(msg); } @Override public ChannelPromise newPromise() { return pipeline().newPromise(); } @Override public ChannelProgressivePromise newProgressivePromise() { return pipeline().newProgressivePromise(); } @Override public ChannelFuture newSucceededFuture() { return pipeline().newSucceededFuture(); } @Override public ChannelFuture newFailedFuture(Throwable cause) { return pipeline().newFailedFuture(cause); } @Override public ChannelPromise voidPromise() { return pipeline().voidPromise(); } @Override public int hashCode() { return id().hashCode(); } @Override public boolean equals(Object o) { return this == o; } @Override public int compareTo(Channel o) { if (this == o) { return 0; } return id().compareTo(o.id()); } @Override public String toString() { return parent().toString() + "(H2 - " + stream + ')'; } void writabilityChanged(boolean writable) { assert eventLoop().inEventLoop(); if (writable != this.writable && isActive()) { // Only notify if we received a state change. this.writable = writable; pipeline().fireChannelWritabilityChanged(); } } /** * Receive a read message. This does not notify handlers unless a read is in progress on the * channel. */ void fireChildRead(Http2Frame frame) { assert eventLoop().inEventLoop(); if (!isActive()) { ReferenceCountUtil.release(frame); } else if (readInProgress) { // If readInProgress there cannot be anything in the queue, otherwise we would have drained it from the // queue and processed it during the read cycle. assert inboundBuffer == null || inboundBuffer.isEmpty(); final Handle allocHandle = unsafe.recvBufAllocHandle(); unsafe.doRead0(frame, allocHandle); // We currently don't need to check for readEOS because the parent channel and child channel are limited // to the same EventLoop thread. There are a limited number of frame types that may come after EOS is // read (unknown, reset) and the trade off is less conditionals for the hot path (headers/data) at the // cost of additional readComplete notifications on the rare path. if (allocHandle.continueReading()) { tryAddChildChannelToReadPendingQueue(this); } else { tryRemoveChildChannelFromReadPendingQueue(this); unsafe.notifyReadComplete(allocHandle); } } else { if (inboundBuffer == null) { inboundBuffer = new ArrayDeque(4); } inboundBuffer.add(frame); } } void fireChildReadComplete() { assert eventLoop().inEventLoop(); assert readInProgress; unsafe.notifyReadComplete(unsafe.recvBufAllocHandle()); } private final class Http2ChannelUnsafe implements Unsafe { private final VoidChannelPromise unsafeVoidPromise = new VoidChannelPromise(DefaultHttp2StreamChannel.this, false); @SuppressWarnings("deprecation") private Handle recvHandle; private boolean writeDoneAndNoFlush; private boolean closeInitiated; private boolean readEOS; @Override public void connect(final SocketAddress remoteAddress, SocketAddress localAddress, final ChannelPromise promise) { if (!promise.setUncancellable()) { return; } promise.setFailure(new UnsupportedOperationException()); } @Override public Handle recvBufAllocHandle() { if (recvHandle == null) { recvHandle = config().getRecvByteBufAllocator().newHandle(); recvHandle.reset(config()); } return recvHandle; } @Override public SocketAddress localAddress() { return parent().unsafe().localAddress(); } @Override public SocketAddress remoteAddress() { return parent().unsafe().remoteAddress(); } @Override public void register(EventLoop eventLoop, ChannelPromise promise) { if (!promise.setUncancellable()) { return; } if (registered) { throw new UnsupportedOperationException("Re-register is not supported"); } registered = true; if (!outbound) { // Add the handler to the pipeline now that we are registered. pipeline().addLast(inboundStreamHandler); } promise.setSuccess(); pipeline().fireChannelRegistered(); if (isActive()) { pipeline().fireChannelActive(); } } @Override public void bind(SocketAddress localAddress, ChannelPromise promise) { if (!promise.setUncancellable()) { return; } promise.setFailure(new UnsupportedOperationException()); } @Override public void disconnect(ChannelPromise promise) { close(promise); } @Override public void close(final ChannelPromise promise) { if (!promise.setUncancellable()) { return; } if (closeInitiated) { if (closePromise.isDone()) { // Closed already. promise.setSuccess(); } else if (!(promise instanceof VoidChannelPromise)) { // Only needed if no VoidChannelPromise. // This means close() was called before so we just register a listener and return closePromise.addListener(new ChannelFutureListener() { @Override public void operationComplete(ChannelFuture future) { promise.setSuccess(); } }); } return; } closeInitiated = true; tryRemoveChildChannelFromReadPendingQueue(DefaultHttp2StreamChannel.this); final boolean wasActive = isActive(); // Only ever send a reset frame if the connection is still alive and if the stream may have existed // as otherwise we may send a RST on a stream in an invalid state and cause a connection error. if (parent().isActive() && !readEOS && connection().streamMayHaveExisted(stream().id())) { Http2StreamFrame resetFrame = new DefaultHttp2ResetFrame(Http2Error.CANCEL).stream(stream()); write(resetFrame, unsafe().voidPromise()); flush(); } if (inboundBuffer != null) { for (;;) { Object msg = inboundBuffer.poll(); if (msg == null) { break; } ReferenceCountUtil.release(msg); } } // The promise should be notified before we call fireChannelInactive(). outboundClosed = true; closePromise.setSuccess(); promise.setSuccess(); fireChannelInactiveAndDeregister(voidPromise(), wasActive); } @Override public void closeForcibly() { close(unsafe().voidPromise()); } @Override public void deregister(ChannelPromise promise) { fireChannelInactiveAndDeregister(promise, false); } private void fireChannelInactiveAndDeregister(final ChannelPromise promise, final boolean fireChannelInactive) { if (!promise.setUncancellable()) { return; } if (!registered) { promise.setSuccess(); return; } // As a user may call deregister() from within any method while doing processing in the ChannelPipeline, // we need to ensure we do the actual deregister operation later. This is necessary to preserve the // behavior of the AbstractChannel, which always invokes channelUnregistered and channelInactive // events 'later' to ensure the current events in the handler are completed before these events. // // See: // https://github.com/netty/netty/issues/4435 invokeLater(new Runnable() { @Override public void run() { if (fireChannelInactive) { pipeline.fireChannelInactive(); } // The user can fire `deregister` events multiple times but we only want to fire the pipeline // event if the channel was actually registered. if (registered) { registered = false; pipeline.fireChannelUnregistered(); } safeSetSuccess(promise); } }); } private void safeSetSuccess(ChannelPromise promise) { if (!(promise instanceof VoidChannelPromise) && !promise.trySuccess()) { logger.warn("Failed to mark a promise as success because it is done already: {}", promise); } } private void invokeLater(Runnable task) { try { // This method is used by outbound operation implementations to trigger an inbound event later. // They do not trigger an inbound event immediately because an outbound operation might have been // triggered by another inbound event handler method. If fired immediately, the call stack // will look like this for example: // // handlerA.inboundBufferUpdated() - (1) an inbound handler method closes a connection. // -> handlerA.ctx.close() // -> channel.unsafe.close() // -> handlerA.channelInactive() - (2) another inbound handler method called while in (1) yet // // which means the execution of two inbound handler methods of the same handler overlap undesirably. eventLoop().execute(task); } catch (RejectedExecutionException e) { logger.warn("Can't invoke task later as EventLoop rejected it", e); } } @Override public void beginRead() { if (readInProgress || !isActive()) { return; } readInProgress = true; doBeginRead(); } void doBeginRead() { Object message; if (inboundBuffer == null || (message = inboundBuffer.poll()) == null) { if (readEOS) { unsafe.closeForcibly(); } } else { final Handle allocHandle = recvBufAllocHandle(); allocHandle.reset(config()); boolean continueReading = false; do { doRead0((Http2Frame) message, allocHandle); } while ((readEOS || (continueReading = allocHandle.continueReading())) && (message = inboundBuffer.poll()) != null); if (continueReading && parentReadInProgress && !readEOS) { // Currently the parent and child channel are on the same EventLoop thread. If the parent is // currently reading it is possile that more frames will be delivered to this child channel. In // the case that this child channel still wants to read we delay the channelReadComplete on this // child channel until the parent is done reading. assert !isChildChannelInReadPendingQueue(DefaultHttp2StreamChannel.this); addChildChannelToReadPendingQueue(DefaultHttp2StreamChannel.this); } else { notifyReadComplete(allocHandle); } } } void readEOS() { readEOS = true; } void notifyReadComplete(Handle allocHandle) { assert next == null && previous == null; readInProgress = false; allocHandle.readComplete(); pipeline().fireChannelReadComplete(); // Reading data may result in frames being written (e.g. WINDOW_UPDATE, RST, etc..). If the parent // channel is not currently reading we need to force a flush at the child channel, because we cannot // rely upon flush occurring in channelReadComplete on the parent channel. flush(); if (readEOS) { unsafe.closeForcibly(); } } @SuppressWarnings("deprecation") void doRead0(Http2Frame frame, Handle allocHandle) { pipeline().fireChannelRead(frame); allocHandle.incMessagesRead(1); if (frame instanceof Http2DataFrame) { final int numBytesToBeConsumed = ((Http2DataFrame) frame).initialFlowControlledBytes(); allocHandle.attemptedBytesRead(numBytesToBeConsumed); allocHandle.lastBytesRead(numBytesToBeConsumed); if (numBytesToBeConsumed != 0) { try { writeDoneAndNoFlush |= onBytesConsumed(ctx, stream, numBytesToBeConsumed); } catch (Http2Exception e) { pipeline().fireExceptionCaught(e); } } } else { allocHandle.attemptedBytesRead(MIN_HTTP2_FRAME_SIZE); allocHandle.lastBytesRead(MIN_HTTP2_FRAME_SIZE); } } @Override public void write(Object msg, final ChannelPromise promise) { // After this point its not possible to cancel a write anymore. if (!promise.setUncancellable()) { ReferenceCountUtil.release(msg); return; } if (!isActive() || // Once the outbound side was closed we should not allow header / data frames outboundClosed && (msg instanceof Http2HeadersFrame || msg instanceof Http2DataFrame)) { ReferenceCountUtil.release(msg); promise.setFailure(CLOSED_CHANNEL_EXCEPTION); return; } try { if (msg instanceof Http2StreamFrame) { Http2StreamFrame frame = validateStreamFrame((Http2StreamFrame) msg).stream(stream()); if (!firstFrameWritten && !isStreamIdValid(stream().id())) { if (!(frame instanceof Http2HeadersFrame)) { ReferenceCountUtil.release(frame); promise.setFailure( new IllegalArgumentException("The first frame must be a headers frame. Was: " + frame.name())); return; } firstFrameWritten = true; ChannelFuture future = write0(frame); if (future.isDone()) { firstWriteComplete(future, promise); } else { future.addListener(new ChannelFutureListener() { @Override public void operationComplete(ChannelFuture future) { firstWriteComplete(future, promise); } }); } return; } } else { String msgStr = msg.toString(); ReferenceCountUtil.release(msg); promise.setFailure(new IllegalArgumentException( "Message must be an " + StringUtil.simpleClassName(Http2StreamFrame.class) + ": " + msgStr)); return; } ChannelFuture future = write0(msg); if (future.isDone()) { writeComplete(future, promise); } else { future.addListener(new ChannelFutureListener() { @Override public void operationComplete(ChannelFuture future) { writeComplete(future, promise); } }); } } catch (Throwable t) { promise.tryFailure(t); } finally { writeDoneAndNoFlush = true; } } private void firstWriteComplete(ChannelFuture future, ChannelPromise promise) { Throwable cause = future.cause(); if (cause == null) { // As we just finished our first write which made the stream-id valid we need to re-evaluate // the writability of the channel. writabilityChanged(Http2MultiplexCodec.this.isWritable(stream)); promise.setSuccess(); } else { // If the first write fails there is not much we can do, just close closeForcibly(); promise.setFailure(wrapStreamClosedError(cause)); } } private void writeComplete(ChannelFuture future, ChannelPromise promise) { Throwable cause = future.cause(); if (cause == null) { promise.setSuccess(); } else { Throwable error = wrapStreamClosedError(cause); if (error instanceof ClosedChannelException) { if (config.isAutoClose()) { // Close channel if needed. closeForcibly(); } else { outboundClosed = true; } } promise.setFailure(error); } } private Throwable wrapStreamClosedError(Throwable cause) { // If the error was caused by STREAM_CLOSED we should use a ClosedChannelException to better // mimic other transports and make it easier to reason about what exceptions to expect. if (cause instanceof Http2Exception && ((Http2Exception) cause).error() == Http2Error.STREAM_CLOSED) { return new ClosedChannelException().initCause(cause); } return cause; } private Http2StreamFrame validateStreamFrame(Http2StreamFrame frame) { if (frame.stream() != null && frame.stream() != stream) { String msgString = frame.toString(); ReferenceCountUtil.release(frame); throw new IllegalArgumentException( "Stream " + frame.stream() + " must not be set on the frame: " + msgString); } return frame; } private ChannelFuture write0(Object msg) { ChannelPromise promise = ctx.newPromise(); Http2MultiplexCodec.this.write(ctx, msg, promise); return promise; } @Override public void flush() { // If we are currently in the parent channel's read loop we should just ignore the flush. // We will ensure we trigger ctx.flush() after we processed all Channels later on and // so aggregate the flushes. This is done as ctx.flush() is expensive when as it may trigger an // write(...) or writev(...) operation on the socket. if (!writeDoneAndNoFlush || parentReadInProgress) { // There is nothing to flush so this is a NOOP. return; } try { flush0(ctx); } finally { writeDoneAndNoFlush = false; } } @Override public ChannelPromise voidPromise() { return unsafeVoidPromise; } @Override public ChannelOutboundBuffer outboundBuffer() { // Always return null as we not use the ChannelOutboundBuffer and not even support it. return null; } } /** * {@link ChannelConfig} so that the high and low writebuffer watermarks can reflect the outbound flow control * window, without having to create a new {@link WriteBufferWaterMark} object whenever the flow control window * changes. */ private final class Http2StreamChannelConfig extends DefaultChannelConfig { Http2StreamChannelConfig(Channel channel) { super(channel); } @Override public int getWriteBufferHighWaterMark() { return min(parent().config().getWriteBufferHighWaterMark(), initialOutboundStreamWindow); } @Override public int getWriteBufferLowWaterMark() { return min(parent().config().getWriteBufferLowWaterMark(), initialOutboundStreamWindow); } @Override public MessageSizeEstimator getMessageSizeEstimator() { return FlowControlledFrameSizeEstimator.INSTANCE; } @Override public WriteBufferWaterMark getWriteBufferWaterMark() { int mark = getWriteBufferHighWaterMark(); return new WriteBufferWaterMark(mark, mark); } @Override public ChannelConfig setMessageSizeEstimator(MessageSizeEstimator estimator) { throw new UnsupportedOperationException(); } @Override @Deprecated public ChannelConfig setWriteBufferHighWaterMark(int writeBufferHighWaterMark) { throw new UnsupportedOperationException(); } @Override @Deprecated public ChannelConfig setWriteBufferLowWaterMark(int writeBufferLowWaterMark) { throw new UnsupportedOperationException(); } @Override public ChannelConfig setWriteBufferWaterMark(WriteBufferWaterMark writeBufferWaterMark) { throw new UnsupportedOperationException(); } @Override public ChannelConfig setRecvByteBufAllocator(RecvByteBufAllocator allocator) { if (!(allocator.newHandle() instanceof RecvByteBufAllocator.ExtendedHandle)) { throw new IllegalArgumentException("allocator.newHandle() must return an object of type: " + RecvByteBufAllocator.ExtendedHandle.class); } super.setRecvByteBufAllocator(allocator); return this; } } } }