netty5/codec-http2/src/main/java/io/netty/handler/codec/http2/Http2FrameCodecBuilder.java

175 lines
6.7 KiB
Java
Raw Normal View History

HTTP/2 Child Channel and FrameCodec Feature Parity. Motivation: This PR (unfortunately) does 4 things: 1) Add outbound flow control to the Http2MultiplexCodec: The HTTP/2 child channel API should interact with HTTP/2 outbound/remote flow control. That is, if a H2 stream used up all its flow control window, the corresponding child channel should be marked unwritable and a writability-changed event should be fired. Similarly, a unwritable child channel should be marked writable and a writability-event should be fired, once a WINDOW_UPDATE frame has been received. The changes are (mostly) contained in ChannelOutboundBuffer, AbstractHttp2StreamChannel and Http2MultiplexCodec. 2) Introduce a Http2Stream2 object, that is used instead of stream identifiers on stream frames. A Http2Stream2 object allows an application to attach state to it, and so a application handler no longer needs to maintain stream state (i.e. in a map(id -> state)) himself. 3) Remove stream state events, which are no longer necessary due to the introduction of Http2Stream2. Also those stream state events have been found hard and complex to work with, when porting gRPC to the Http2FrameCodec. 4) Add support for HTTP/2 frames that have not yet been implemented, like PING and SETTINGS. Also add a Http2FrameCodecBuilder that exposes options from the Http2ConnectionHandler API that couldn't else be used with the frame codec, like buffering outbound streams, window update ratio, frame logger, etc. Modifications: 1) A child channel's writability and a H2 stream's outbound flow control window interact, as described in the motivation. A channel handler is free to ignore the channel's writability, in which case the parent channel is reponsible for buffering writes until a WINDOW_UPDATE is received. The connection-level flow control window is ignored for now. That is, a child channel's writability is only affected by the stream-level flow control window. So a child channel could be marked writable, even though the connection-level flow control window is zero. 2) Modify Http2StreamFrame and the Http2FrameCodec to take a Http2Stream2 object intstead of a primitive integer. Introduce a special Http2ChannelDuplexHandler that has newStream() and forEachActiveStream() methods. It's recommended for a user to extend from this handler, to use those advanced features. 3) As explained in the documentation, a new inbound stream active can be detected by checking if the Http2Stream2.managedState() of a Http2HeadersFrame is null. An outbound stream active can be detected by adding a listener to the ChannelPromise of the write of the first Http2HeadersFrame. A stream closed event can be listened to by adding a listener to the Http2Stream2.closeFuture(). 4) Add a simple Http2FrameCodecBuilder and implement the missing frame types. Result: 1) The Http2MultiplexCodec supports outbound flow control. 2) The Http2FrameCodec API makes it easy for a user to manage custom stream specific state and to create new outbound streams. 3) The Http2FrameCodec API is much cleaner and easier to work with. Hacks like the ChannelCarryingHeadersFrame are no longer necessary. 4) The Http2FrameCodec now also supports PING and SETTINGS frames. The Http2FrameCodecBuilder allows the Http2FrameCodec to use some of the rich features of the Http2ConnectionHandler API.
2016-08-23 13:03:39 +02:00
/*
* 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.handler.codec.http2.StreamBufferingEncoder.Http2ChannelClosedException;
import io.netty.handler.codec.http2.StreamBufferingEncoder.Http2GoAwayException;
import io.netty.util.internal.UnstableApi;
import java.util.concurrent.TimeUnit;
import static io.netty.handler.codec.http2.DefaultHttp2LocalFlowController.DEFAULT_WINDOW_UPDATE_RATIO;
import static io.netty.util.internal.ObjectUtil.checkNotNull;
import static io.netty.util.internal.ObjectUtil.checkPositiveOrZero;
/**
* Builder for the {@link Http2FrameCodec}.
*/
@UnstableApi
public final class Http2FrameCodecBuilder {
private final boolean server;
private Http2FrameWriter frameWriter;
private Http2FrameReader frameReader;
private Http2Settings initialSettings;
private long gracefulShutdownTimeoutMillis;
private float windowUpdateRatio;
private Http2FrameLogger frameLogger;
private boolean bufferOutboundStreams;
private Http2FrameCodecBuilder(boolean server) {
this.server = server;
frameWriter = new DefaultHttp2FrameWriter();
frameReader = new DefaultHttp2FrameReader();
initialSettings = new Http2Settings();
gracefulShutdownTimeoutMillis = Http2CodecUtil.DEFAULT_GRACEFUL_SHUTDOWN_TIMEOUT_MILLIS;
windowUpdateRatio = DEFAULT_WINDOW_UPDATE_RATIO;
}
/**
* Creates a builder for a HTTP/2 client.
*/
public static Http2FrameCodecBuilder forClient() {
return new Http2FrameCodecBuilder(false);
}
/**
* Creates a builder for a HTTP/2 server.
*/
public static Http2FrameCodecBuilder forServer() {
return new Http2FrameCodecBuilder(true);
}
/**
* Specify the {@link Http2FrameWriter} to use.
*
* <p>If not set, the {@link DefaultHttp2FrameWriter} is used.
*/
public Http2FrameCodecBuilder frameWriter(Http2FrameWriter frameWriter) {
this.frameWriter = checkNotNull(frameWriter, "frameWriter");
return this;
}
/**
* Specify the {@link Http2FrameWriter} to use.
*
* <p>If not set, the {@link DefaultHttp2FrameReader} is used.
*/
public Http2FrameCodecBuilder frameReader(Http2FrameReader frameReader) {
this.frameReader = checkNotNull(frameReader, "frameReader");
return this;
}
/**
* Specify the initial {@link Http2Settings} to send to the remote endpoint.
*
* <p>If not set, the default values of {@link Http2Settings} are used.
*/
public Http2FrameCodecBuilder initialSettings(Http2Settings initialSettings) {
this.initialSettings = checkNotNull(initialSettings, "initialSettings");
return this;
}
/**
* The amount of time to wait for all active streams to be closed, before the connection is closed.
*
* <p>The default value is {@link Http2CodecUtil#DEFAULT_GRACEFUL_SHUTDOWN_TIMEOUT_MILLIS}.
*/
public Http2FrameCodecBuilder gracefulShutdownTimeout(long timeout, TimeUnit unit) {
gracefulShutdownTimeoutMillis =
checkNotNull(unit, "unit").toMillis(checkPositiveOrZero(timeout, "timeout"));
return this;
}
/**
* Specify the HTTP/2 flow control window update ratio for both the connection and stream window.
*/
public Http2FrameCodecBuilder windowUpdateRatio(float windowUpdateRatio) {
if (Float.compare(windowUpdateRatio, 0) < 1 || Float.compare(windowUpdateRatio, 1) > -1) {
throw new IllegalArgumentException("windowUpdateRatio must be (0,1). Was: " + windowUpdateRatio);
}
this.windowUpdateRatio = windowUpdateRatio;
return this;
}
/**
* Specify the {@link Http2FrameLogger} to use.
*
* <p>By default no frame logger is used.
*/
public Http2FrameCodecBuilder frameLogger(Http2FrameLogger frameLogger) {
this.frameLogger = checkNotNull(frameLogger, "frameLogger");
return this;
}
/**
* Whether to buffer new outbound HTTP/2 streams when the {@code MAX_CONCURRENT_STREAMS} limit is reached.
*
* <p>When this limit is hit, instead of rejecting any new streams, newly created streams and their corresponding
* frames are buffered. Once an active stream gets closed or the maximum number of concurrent streams is increased,
* the codec will automatically try to empty its buffer and create as many new streams as possible.
*
* <p>If a {@code GOAWAY} frame is received from the remote endpoint, all buffered writes for streams with an ID
* less than the specified {@code lastStreamId} will immediately fail with a {@link Http2GoAwayException}.
*
* <p>If the channel gets closed, all new and buffered writes will immediately fail with a
* {@link Http2ChannelClosedException}.
*
* <p>This implementation makes the buffering mostly transparent and does not enforce an upper bound as to how many
* streams/frames can be buffered.
*/
public Http2FrameCodecBuilder bufferOutboundStreams(boolean bufferOutboundStreams) {
this.bufferOutboundStreams = bufferOutboundStreams;
return this;
}
/**
* Build a {@link Http2FrameCodec} object.
*/
public Http2FrameCodec build() {
Http2Connection connection = new DefaultHttp2Connection(server);
if (frameLogger != null) {
frameWriter = new Http2OutboundFrameLogger(frameWriter, frameLogger);
frameReader = new Http2InboundFrameLogger(frameReader, frameLogger);
}
Http2ConnectionEncoder encoder = new DefaultHttp2ConnectionEncoder(connection, frameWriter);
if (bufferOutboundStreams) {
encoder = new StreamBufferingEncoder(encoder);
}
connection.local().flowController(new DefaultHttp2LocalFlowController(connection, windowUpdateRatio,
true /* auto refill conn window */));
Http2ConnectionDecoder decoder = new DefaultHttp2ConnectionDecoder(connection, encoder, frameReader);
return new Http2FrameCodec(encoder, decoder, initialSettings, gracefulShutdownTimeoutMillis);
}
}