2015-01-20 01:48:11 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2014 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;
|
|
|
|
|
2015-06-17 00:11:28 +02:00
|
|
|
import static io.netty.buffer.ByteBufUtil.hexDump;
|
2015-01-20 01:48:11 +01:00
|
|
|
import static io.netty.handler.codec.http2.Http2CodecUtil.HTTP_UPGRADE_STREAM_ID;
|
|
|
|
import static io.netty.handler.codec.http2.Http2CodecUtil.connectionPrefaceBuf;
|
|
|
|
import static io.netty.handler.codec.http2.Http2CodecUtil.getEmbeddedHttp2Exception;
|
|
|
|
import static io.netty.handler.codec.http2.Http2Error.INTERNAL_ERROR;
|
|
|
|
import static io.netty.handler.codec.http2.Http2Error.NO_ERROR;
|
|
|
|
import static io.netty.handler.codec.http2.Http2Error.PROTOCOL_ERROR;
|
|
|
|
import static io.netty.handler.codec.http2.Http2Exception.connectionError;
|
|
|
|
import static io.netty.handler.codec.http2.Http2Exception.isStreamError;
|
2015-06-17 00:11:28 +02:00
|
|
|
import static io.netty.handler.codec.http2.Http2FrameTypes.SETTINGS;
|
2015-05-06 21:04:55 +02:00
|
|
|
import static io.netty.util.CharsetUtil.UTF_8;
|
2015-01-20 01:48:11 +01:00
|
|
|
import static io.netty.util.internal.ObjectUtil.checkNotNull;
|
2015-06-17 00:11:28 +02:00
|
|
|
import static java.lang.Math.min;
|
2015-03-20 02:36:24 +01:00
|
|
|
import static java.lang.String.format;
|
2015-01-20 01:48:11 +01:00
|
|
|
import io.netty.buffer.ByteBuf;
|
2015-03-23 18:52:11 +01:00
|
|
|
import io.netty.buffer.ByteBufUtil;
|
2015-01-20 01:48:11 +01:00
|
|
|
import io.netty.channel.ChannelFuture;
|
|
|
|
import io.netty.channel.ChannelFutureListener;
|
|
|
|
import io.netty.channel.ChannelHandlerContext;
|
|
|
|
import io.netty.channel.ChannelOutboundHandler;
|
|
|
|
import io.netty.channel.ChannelPromise;
|
|
|
|
import io.netty.handler.codec.ByteToMessageDecoder;
|
|
|
|
import io.netty.handler.codec.http2.Http2Exception.CompositeStreamException;
|
|
|
|
import io.netty.handler.codec.http2.Http2Exception.StreamException;
|
2015-03-20 02:36:24 +01:00
|
|
|
import io.netty.util.internal.logging.InternalLogger;
|
|
|
|
import io.netty.util.internal.logging.InternalLoggerFactory;
|
2015-01-20 01:48:11 +01:00
|
|
|
|
|
|
|
import java.net.SocketAddress;
|
|
|
|
import java.util.List;
|
|
|
|
|
|
|
|
/**
|
2015-02-02 19:20:56 +01:00
|
|
|
* Provides the default implementation for processing inbound frame events and delegates to a
|
|
|
|
* {@link Http2FrameListener}
|
2015-01-20 01:48:11 +01:00
|
|
|
* <p>
|
|
|
|
* This class will read HTTP/2 frames and delegate the events to a {@link Http2FrameListener}
|
|
|
|
* <p>
|
2015-02-02 19:20:56 +01:00
|
|
|
* This interface enforces inbound flow control functionality through
|
|
|
|
* {@link Http2LocalFlowController}
|
2015-01-20 01:48:11 +01:00
|
|
|
*/
|
|
|
|
public class Http2ConnectionHandler extends ByteToMessageDecoder implements Http2LifecycleManager,
|
|
|
|
ChannelOutboundHandler {
|
2015-04-09 21:30:16 +02:00
|
|
|
private static final InternalLogger logger = InternalLoggerFactory.getInstance(Http2ConnectionHandler.class);
|
2015-01-20 01:48:11 +01:00
|
|
|
private final Http2ConnectionDecoder decoder;
|
|
|
|
private final Http2ConnectionEncoder encoder;
|
2015-05-21 21:12:29 +02:00
|
|
|
private final Http2Settings initialSettings;
|
2015-01-20 01:48:11 +01:00
|
|
|
private ChannelFutureListener closeListener;
|
2015-03-23 18:52:11 +01:00
|
|
|
private BaseDecoder byteDecoder;
|
2015-01-20 01:48:11 +01:00
|
|
|
|
|
|
|
public Http2ConnectionHandler(boolean server, Http2FrameListener listener) {
|
|
|
|
this(new DefaultHttp2Connection(server), listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionHandler(Http2Connection connection, Http2FrameListener listener) {
|
|
|
|
this(connection, new DefaultHttp2FrameReader(), new DefaultHttp2FrameWriter(), listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionHandler(Http2Connection connection, Http2FrameReader frameReader,
|
2015-03-27 23:37:20 +01:00
|
|
|
Http2FrameWriter frameWriter, Http2FrameListener listener) {
|
2015-05-21 21:12:29 +02:00
|
|
|
initialSettings = null;
|
2015-03-27 23:37:20 +01:00
|
|
|
encoder = new DefaultHttp2ConnectionEncoder(connection, frameWriter);
|
|
|
|
decoder = new DefaultHttp2ConnectionDecoder(connection, encoder, frameReader, listener);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-27 23:37:20 +01:00
|
|
|
* Constructor for pre-configured encoder and decoder. Just sets the {@code this} as the
|
2015-01-20 01:48:11 +01:00
|
|
|
* {@link Http2LifecycleManager} and builds them.
|
|
|
|
*/
|
2015-03-27 23:37:20 +01:00
|
|
|
public Http2ConnectionHandler(Http2ConnectionDecoder decoder,
|
|
|
|
Http2ConnectionEncoder encoder) {
|
2015-05-21 21:12:29 +02:00
|
|
|
this.initialSettings = null;
|
|
|
|
this.decoder = checkNotNull(decoder, "decoder");
|
|
|
|
this.encoder = checkNotNull(encoder, "encoder");
|
|
|
|
if (encoder.connection() != decoder.connection()) {
|
|
|
|
throw new IllegalArgumentException("Encoder and Decoder do not share the same connection object");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionHandler(Http2Connection connection, Http2FrameListener listener,
|
|
|
|
Http2Settings initialSettings) {
|
|
|
|
this(connection, new DefaultHttp2FrameReader(), new DefaultHttp2FrameWriter(), listener,
|
|
|
|
initialSettings);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionHandler(Http2Connection connection, Http2FrameReader frameReader,
|
|
|
|
Http2FrameWriter frameWriter, Http2FrameListener listener,
|
|
|
|
Http2Settings initialSettings) {
|
|
|
|
this.initialSettings = initialSettings;
|
|
|
|
encoder = new DefaultHttp2ConnectionEncoder(connection, frameWriter);
|
|
|
|
decoder = new DefaultHttp2ConnectionDecoder(connection, encoder, frameReader, listener);
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionHandler(Http2ConnectionDecoder decoder,
|
|
|
|
Http2ConnectionEncoder encoder,
|
|
|
|
Http2Settings initialSettings) {
|
|
|
|
this.initialSettings = initialSettings;
|
2015-03-27 23:37:20 +01:00
|
|
|
this.decoder = checkNotNull(decoder, "decoder");
|
|
|
|
this.encoder = checkNotNull(encoder, "encoder");
|
2015-01-20 01:48:11 +01:00
|
|
|
if (encoder.connection() != decoder.connection()) {
|
|
|
|
throw new IllegalArgumentException("Encoder and Decoder do not share the same connection object");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2Connection connection() {
|
|
|
|
return encoder.connection();
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionDecoder decoder() {
|
|
|
|
return decoder;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Http2ConnectionEncoder encoder() {
|
|
|
|
return encoder;
|
|
|
|
}
|
|
|
|
|
2015-03-23 18:52:11 +01:00
|
|
|
private boolean prefaceSent() {
|
|
|
|
return byteDecoder != null && byteDecoder.prefaceSent();
|
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
/**
|
|
|
|
* Handles the client-side (cleartext) upgrade from HTTP to HTTP/2.
|
|
|
|
* Reserves local stream 1 for the HTTP/2 response.
|
|
|
|
*/
|
|
|
|
public void onHttpClientUpgrade() throws Http2Exception {
|
|
|
|
if (connection().isServer()) {
|
|
|
|
throw connectionError(PROTOCOL_ERROR, "Client-side HTTP upgrade requested for a server");
|
|
|
|
}
|
2015-03-23 18:52:11 +01:00
|
|
|
if (prefaceSent() || decoder.prefaceReceived()) {
|
2015-01-20 01:48:11 +01:00
|
|
|
throw connectionError(PROTOCOL_ERROR, "HTTP upgrade must occur before HTTP/2 preface is sent or received");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a local stream used for the HTTP cleartext upgrade.
|
2015-04-22 23:35:31 +02:00
|
|
|
connection().local().createStream(HTTP_UPGRADE_STREAM_ID, true);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles the server-side (cleartext) upgrade from HTTP to HTTP/2.
|
|
|
|
* @param settings the settings for the remote endpoint.
|
|
|
|
*/
|
|
|
|
public void onHttpServerUpgrade(Http2Settings settings) throws Http2Exception {
|
|
|
|
if (!connection().isServer()) {
|
|
|
|
throw connectionError(PROTOCOL_ERROR, "Server-side HTTP upgrade requested for a client");
|
|
|
|
}
|
2015-03-23 18:52:11 +01:00
|
|
|
if (prefaceSent() || decoder.prefaceReceived()) {
|
2015-01-20 01:48:11 +01:00
|
|
|
throw connectionError(PROTOCOL_ERROR, "HTTP upgrade must occur before HTTP/2 preface is sent or received");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply the settings but no ACK is necessary.
|
|
|
|
encoder.remoteSettings(settings);
|
|
|
|
|
|
|
|
// Create a stream in the half-closed state.
|
2015-04-22 23:35:31 +02:00
|
|
|
connection().remote().createStream(HTTP_UPGRADE_STREAM_ID, true);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
2015-06-04 20:55:18 +02:00
|
|
|
@Override
|
|
|
|
public void flush(ChannelHandlerContext ctx) throws Http2Exception {
|
|
|
|
// Trigger pending writes in the remote flow controller.
|
|
|
|
connection().remote().flowController().writePendingBytes();
|
|
|
|
try {
|
|
|
|
super.flush(ctx);
|
|
|
|
} catch (Throwable t) {
|
|
|
|
throw new Http2Exception(INTERNAL_ERROR, "Error flushing" , t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-23 18:52:11 +01:00
|
|
|
private abstract class BaseDecoder {
|
|
|
|
public abstract void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception;
|
|
|
|
public void handlerRemoved(ChannelHandlerContext ctx) throws Exception { }
|
|
|
|
public void channelActive(ChannelHandlerContext ctx) throws Exception { }
|
|
|
|
|
|
|
|
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
try {
|
2015-03-28 21:32:19 +01:00
|
|
|
final Http2Connection connection = connection();
|
|
|
|
// Check if there are streams to avoid the overhead of creating the ChannelFuture.
|
|
|
|
if (connection.numActiveStreams() > 0) {
|
|
|
|
final ChannelFuture future = ctx.newSucceededFuture();
|
2015-04-09 18:48:58 +02:00
|
|
|
connection.forEachActiveStream(new Http2StreamVisitor() {
|
2015-03-28 21:32:19 +01:00
|
|
|
@Override
|
|
|
|
public boolean visit(Http2Stream stream) throws Http2Exception {
|
|
|
|
closeStream(stream, future);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
});
|
2015-03-23 18:52:11 +01:00
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
try {
|
|
|
|
encoder().close();
|
|
|
|
} finally {
|
|
|
|
decoder().close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if the HTTP/2 connection preface been sent.
|
|
|
|
*/
|
|
|
|
public boolean prefaceSent() {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private final class PrefaceDecoder extends BaseDecoder {
|
|
|
|
private ByteBuf clientPrefaceString;
|
|
|
|
private boolean prefaceSent;
|
|
|
|
|
|
|
|
public PrefaceDecoder(ChannelHandlerContext ctx) {
|
|
|
|
clientPrefaceString = clientPrefaceString(encoder.connection());
|
|
|
|
// This handler was just added to the context. In case it was handled after
|
|
|
|
// the connection became active, send the connection preface now.
|
|
|
|
sendPreface(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public boolean prefaceSent() {
|
|
|
|
return prefaceSent;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {
|
|
|
|
try {
|
2015-06-17 00:11:28 +02:00
|
|
|
if (readClientPrefaceString(in) && verifyFirstFrameIsSettings(in)) {
|
2015-03-23 18:52:11 +01:00
|
|
|
// After the preface is read, it is time to hand over control to the post initialized decoder.
|
2015-06-17 00:11:28 +02:00
|
|
|
byteDecoder = new FrameDecoder();
|
|
|
|
byteDecoder.decode(ctx, in, out);
|
2015-03-23 18:52:11 +01:00
|
|
|
}
|
|
|
|
} catch (Throwable e) {
|
|
|
|
onException(ctx, e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void channelActive(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
// The channel just became active - send the connection preface to the remote endpoint.
|
|
|
|
sendPreface(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
cleanup();
|
|
|
|
super.channelInactive(ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Releases the {@code clientPrefaceString}. Any active streams will be left in the open.
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public void handlerRemoved(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Releases the {@code clientPrefaceString}. Any active streams will be left in the open.
|
|
|
|
*/
|
|
|
|
private void cleanup() {
|
|
|
|
if (clientPrefaceString != null) {
|
|
|
|
clientPrefaceString.release();
|
|
|
|
clientPrefaceString = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decodes the client connection preface string from the input buffer.
|
|
|
|
*
|
|
|
|
* @return {@code true} if processing of the client preface string is complete. Since client preface strings can
|
|
|
|
* only be received by servers, returns true immediately for client endpoints.
|
|
|
|
*/
|
|
|
|
private boolean readClientPrefaceString(ByteBuf in) throws Http2Exception {
|
|
|
|
if (clientPrefaceString == null) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int prefaceRemaining = clientPrefaceString.readableBytes();
|
2015-06-17 00:11:28 +02:00
|
|
|
int bytesRead = min(in.readableBytes(), prefaceRemaining);
|
2015-03-23 18:52:11 +01:00
|
|
|
|
|
|
|
// If the input so far doesn't match the preface, break the connection.
|
|
|
|
if (bytesRead == 0 || !ByteBufUtil.equals(in, in.readerIndex(),
|
|
|
|
clientPrefaceString, clientPrefaceString.readerIndex(), bytesRead)) {
|
2015-06-17 00:11:28 +02:00
|
|
|
String receivedBytes = hexDump(in, in.readerIndex(),
|
|
|
|
min(in.readableBytes(), clientPrefaceString.readableBytes()));
|
|
|
|
throw connectionError(PROTOCOL_ERROR, "HTTP/2 client preface string missing or corrupt. " +
|
|
|
|
"Hex dump for received bytes: %s", receivedBytes);
|
2015-03-23 18:52:11 +01:00
|
|
|
}
|
|
|
|
in.skipBytes(bytesRead);
|
|
|
|
clientPrefaceString.skipBytes(bytesRead);
|
|
|
|
|
|
|
|
if (!clientPrefaceString.isReadable()) {
|
|
|
|
// Entire preface has been read.
|
|
|
|
clientPrefaceString.release();
|
|
|
|
clientPrefaceString = null;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-06-17 00:11:28 +02:00
|
|
|
/**
|
|
|
|
* Peeks at that the next frame in the buffer and verifies that it is a {@code SETTINGS} frame.
|
|
|
|
*
|
|
|
|
* @param in the inbound buffer.
|
|
|
|
* @return {@code} true if the next frame is a {@code SETTINGS} frame, {@code false} if more
|
|
|
|
* data is required before we can determine the next frame type.
|
|
|
|
* @throws Http2Exception thrown if the next frame is NOT a {@code SETTINGS} frame.
|
|
|
|
*/
|
|
|
|
private boolean verifyFirstFrameIsSettings(ByteBuf in) throws Http2Exception {
|
|
|
|
if (in.readableBytes() < 4) {
|
|
|
|
// Need more data before we can see the frame type for the first frame.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
byte frameType = in.getByte(in.readerIndex() + 3);
|
|
|
|
if (frameType != SETTINGS) {
|
|
|
|
throw connectionError(PROTOCOL_ERROR, "First received frame was not SETTINGS. " +
|
|
|
|
"Hex dump for first 4 bytes: %s", hexDump(in, in.readerIndex(), 4));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-23 18:52:11 +01:00
|
|
|
/**
|
|
|
|
* Sends the HTTP/2 connection preface upon establishment of the connection, if not already sent.
|
|
|
|
*/
|
|
|
|
private void sendPreface(ChannelHandlerContext ctx) {
|
|
|
|
if (prefaceSent || !ctx.channel().isActive()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
prefaceSent = true;
|
|
|
|
|
|
|
|
if (!connection().isServer()) {
|
|
|
|
// Clients must send the preface string as the first bytes on the connection.
|
|
|
|
ctx.write(connectionPrefaceBuf()).addListener(ChannelFutureListener.CLOSE_ON_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Both client and server must send their initial settings.
|
2015-05-21 21:12:29 +02:00
|
|
|
encoder.writeSettings(ctx, initialSettings(), ctx.newPromise()).addListener(
|
2015-03-23 18:52:11 +01:00
|
|
|
ChannelFutureListener.CLOSE_ON_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private final class FrameDecoder extends BaseDecoder {
|
|
|
|
@Override
|
|
|
|
public void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {
|
|
|
|
try {
|
|
|
|
decoder.decodeFrame(ctx, in, out);
|
|
|
|
} catch (Throwable e) {
|
|
|
|
onException(ctx, e);
|
|
|
|
}
|
|
|
|
}
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void handlerAdded(ChannelHandlerContext ctx) throws Exception {
|
2015-03-27 23:37:20 +01:00
|
|
|
// Initialize the encoder and decoder.
|
|
|
|
encoder.lifecycleManager(this);
|
|
|
|
decoder.lifecycleManager(this);
|
2015-03-23 18:52:11 +01:00
|
|
|
byteDecoder = new PrefaceDecoder(ctx);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void handlerRemoved0(ChannelHandlerContext ctx) throws Exception {
|
2015-03-23 18:52:11 +01:00
|
|
|
if (byteDecoder != null) {
|
|
|
|
byteDecoder.handlerRemoved(ctx);
|
|
|
|
byteDecoder = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void channelActive(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
if (byteDecoder == null) {
|
|
|
|
byteDecoder = new PrefaceDecoder(ctx);
|
2015-03-18 22:09:19 +01:00
|
|
|
}
|
2015-03-23 18:52:11 +01:00
|
|
|
byteDecoder.channelActive(ctx);
|
2015-04-29 01:27:54 +02:00
|
|
|
super.channelActive(ctx);
|
2015-03-23 18:52:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void channelInactive(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
if (byteDecoder != null) {
|
|
|
|
byteDecoder.channelInactive(ctx);
|
2015-04-29 01:27:54 +02:00
|
|
|
super.channelInactive(ctx);
|
2015-03-23 18:52:11 +01:00
|
|
|
byteDecoder = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void decode(ChannelHandlerContext ctx, ByteBuf in, List<Object> out) throws Exception {
|
|
|
|
byteDecoder.decode(ctx, in, out);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void bind(ChannelHandlerContext ctx, SocketAddress localAddress, ChannelPromise promise) throws Exception {
|
|
|
|
ctx.bind(localAddress, promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void connect(ChannelHandlerContext ctx, SocketAddress remoteAddress, SocketAddress localAddress,
|
|
|
|
ChannelPromise promise) throws Exception {
|
|
|
|
ctx.connect(remoteAddress, localAddress, promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void disconnect(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
|
|
|
|
ctx.disconnect(promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void close(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
|
|
|
|
// Avoid NotYetConnectedException
|
|
|
|
if (!ctx.channel().isActive()) {
|
|
|
|
ctx.close(promise);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-04-02 23:39:46 +02:00
|
|
|
ChannelFuture future = goAway(ctx, null);
|
2015-04-23 23:23:23 +02:00
|
|
|
ctx.flush();
|
2015-01-20 01:48:11 +01:00
|
|
|
|
|
|
|
// If there are no active streams, close immediately after the send is complete.
|
|
|
|
// Otherwise wait until all streams are inactive.
|
2015-05-06 21:04:55 +02:00
|
|
|
if (isGracefulShutdownComplete()) {
|
2015-01-20 01:48:11 +01:00
|
|
|
future.addListener(new ClosingChannelFutureListener(ctx, promise));
|
|
|
|
} else {
|
|
|
|
closeListener = new ClosingChannelFutureListener(ctx, promise);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-29 07:51:32 +02:00
|
|
|
@Override
|
|
|
|
public void deregister(ChannelHandlerContext ctx, ChannelPromise promise) throws Exception {
|
|
|
|
ctx.deregister(promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void read(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
ctx.read();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception {
|
|
|
|
ctx.write(msg, promise);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void flush(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
ctx.flush();
|
|
|
|
}
|
|
|
|
|
2015-04-23 23:23:23 +02:00
|
|
|
@Override
|
|
|
|
public void channelReadComplete(ChannelHandlerContext ctx) throws Exception {
|
|
|
|
// Trigger flush after read on the assumption that flush is cheap if there is nothing to write and that
|
|
|
|
// for flow-control the read may release window that causes data to be written that can now be flushed.
|
2015-06-04 20:55:18 +02:00
|
|
|
flush(ctx);
|
2015-04-23 23:23:23 +02:00
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
/**
|
|
|
|
* Handles {@link Http2Exception} objects that were thrown from other handlers. Ignores all other exceptions.
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception {
|
|
|
|
if (getEmbeddedHttp2Exception(cause) != null) {
|
|
|
|
// Some exception in the causality chain is an Http2Exception - handle it.
|
|
|
|
onException(ctx, cause);
|
|
|
|
} else {
|
|
|
|
super.exceptionCaught(ctx, cause);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Closes the local side of the given stream. If this causes the stream to be closed, adds a
|
|
|
|
* hook to close the channel after the given future completes.
|
|
|
|
*
|
|
|
|
* @param stream the stream to be half closed.
|
|
|
|
* @param future If closing, the future after which to close the channel.
|
|
|
|
*/
|
|
|
|
@Override
|
2015-04-02 23:39:46 +02:00
|
|
|
public void closeStreamLocal(Http2Stream stream, ChannelFuture future) {
|
2015-01-20 01:48:11 +01:00
|
|
|
switch (stream.state()) {
|
|
|
|
case HALF_CLOSED_LOCAL:
|
|
|
|
case OPEN:
|
|
|
|
stream.closeLocalSide();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
closeStream(stream, future);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Closes the remote side of the given stream. If this causes the stream to be closed, adds a
|
|
|
|
* hook to close the channel after the given future completes.
|
|
|
|
*
|
|
|
|
* @param stream the stream to be half closed.
|
|
|
|
* @param future If closing, the future after which to close the channel.
|
|
|
|
*/
|
|
|
|
@Override
|
2015-04-02 23:39:46 +02:00
|
|
|
public void closeStreamRemote(Http2Stream stream, ChannelFuture future) {
|
2015-01-20 01:48:11 +01:00
|
|
|
switch (stream.state()) {
|
|
|
|
case HALF_CLOSED_REMOTE:
|
|
|
|
case OPEN:
|
|
|
|
stream.closeRemoteSide();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
closeStream(stream, future);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2015-02-02 19:20:56 +01:00
|
|
|
public void closeStream(final Http2Stream stream, ChannelFuture future) {
|
2015-01-20 01:48:11 +01:00
|
|
|
stream.close();
|
|
|
|
|
2015-05-30 01:27:59 +02:00
|
|
|
if (future.isDone()) {
|
|
|
|
checkCloseConnection(future);
|
|
|
|
} else {
|
|
|
|
future.addListener(new ChannelFutureListener() {
|
|
|
|
@Override
|
|
|
|
public void operationComplete(ChannelFuture future) throws Exception {
|
|
|
|
checkCloseConnection(future);
|
2015-05-06 21:04:55 +02:00
|
|
|
}
|
2015-05-30 01:27:59 +02:00
|
|
|
});
|
|
|
|
}
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Central handler for all exceptions caught during HTTP/2 processing.
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public void onException(ChannelHandlerContext ctx, Throwable cause) {
|
|
|
|
Http2Exception embedded = getEmbeddedHttp2Exception(cause);
|
|
|
|
if (isStreamError(embedded)) {
|
|
|
|
onStreamError(ctx, cause, (StreamException) embedded);
|
|
|
|
} else if (embedded instanceof CompositeStreamException) {
|
|
|
|
CompositeStreamException compositException = (CompositeStreamException) embedded;
|
|
|
|
for (StreamException streamException : compositException) {
|
|
|
|
onStreamError(ctx, cause, streamException);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
onConnectionError(ctx, cause, embedded);
|
|
|
|
}
|
2015-04-23 23:23:23 +02:00
|
|
|
ctx.flush();
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
2015-05-06 21:04:55 +02:00
|
|
|
/**
|
|
|
|
* Called by the graceful shutdown logic to determine when it is safe to close the connection. Returns {@code true}
|
|
|
|
* if the graceful shutdown has completed and the connection can be safely closed. This implementation just
|
|
|
|
* guarantees that there are no active streams. Subclasses may override to provide additional checks.
|
|
|
|
*/
|
|
|
|
protected boolean isGracefulShutdownComplete() {
|
|
|
|
return connection().numActiveStreams() == 0;
|
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
/**
|
|
|
|
* Handler for a connection error. Sends a GO_AWAY frame to the remote endpoint. Once all
|
|
|
|
* streams are closed, the connection is shut down.
|
|
|
|
*
|
|
|
|
* @param ctx the channel context
|
|
|
|
* @param cause the exception that was caught
|
|
|
|
* @param http2Ex the {@link Http2Exception} that is embedded in the causality chain. This may
|
|
|
|
* be {@code null} if it's an unknown exception.
|
|
|
|
*/
|
|
|
|
protected void onConnectionError(ChannelHandlerContext ctx, Throwable cause, Http2Exception http2Ex) {
|
|
|
|
if (http2Ex == null) {
|
|
|
|
http2Ex = new Http2Exception(INTERNAL_ERROR, cause.getMessage(), cause);
|
|
|
|
}
|
2015-04-02 23:39:46 +02:00
|
|
|
goAway(ctx, http2Ex).addListener(new ClosingChannelFutureListener(ctx, ctx.newPromise()));
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handler for a stream error. Sends a {@code RST_STREAM} frame to the remote endpoint and closes the
|
|
|
|
* stream.
|
|
|
|
*
|
|
|
|
* @param ctx the channel context
|
|
|
|
* @param cause the exception that was caught
|
|
|
|
* @param http2Ex the {@link StreamException} that is embedded in the causality chain.
|
|
|
|
*/
|
|
|
|
protected void onStreamError(ChannelHandlerContext ctx, Throwable cause, StreamException http2Ex) {
|
2015-04-02 23:39:46 +02:00
|
|
|
resetStream(ctx, http2Ex.streamId(), http2Ex.error().code(), ctx.newPromise());
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
protected Http2FrameWriter frameWriter() {
|
|
|
|
return encoder().frameWriter();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2015-04-02 23:39:46 +02:00
|
|
|
public ChannelFuture resetStream(final ChannelHandlerContext ctx, int streamId, long errorCode,
|
2015-03-14 20:58:30 +01:00
|
|
|
final ChannelPromise promise) {
|
|
|
|
final Http2Stream stream = connection().stream(streamId);
|
|
|
|
if (stream == null || stream.isResetSent()) {
|
|
|
|
// Don't write a RST_STREAM frame if we are not aware of the stream, or if we have already written one.
|
|
|
|
return promise.setSuccess();
|
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
ChannelFuture future = frameWriter().writeRstStream(ctx, streamId, errorCode, promise);
|
|
|
|
|
2015-03-14 20:58:30 +01:00
|
|
|
// Synchronously set the resetSent flag to prevent any subsequent calls
|
|
|
|
// from resulting in multiple reset frames being sent.
|
|
|
|
stream.resetSent();
|
|
|
|
|
|
|
|
future.addListener(new ChannelFutureListener() {
|
|
|
|
@Override
|
|
|
|
public void operationComplete(ChannelFuture future) throws Exception {
|
|
|
|
if (future.isSuccess()) {
|
|
|
|
closeStream(stream, promise);
|
|
|
|
} else {
|
|
|
|
// The connection will be closed and so no need to change the resetSent flag to false.
|
|
|
|
onConnectionError(ctx, future.cause(), null);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2015-01-20 01:48:11 +01:00
|
|
|
|
|
|
|
return future;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
2015-03-20 02:36:24 +01:00
|
|
|
public ChannelFuture goAway(final ChannelHandlerContext ctx, final int lastStreamId, final long errorCode,
|
|
|
|
final ByteBuf debugData, ChannelPromise promise) {
|
|
|
|
try {
|
|
|
|
final Http2Connection connection = connection();
|
2015-05-03 17:49:49 +02:00
|
|
|
if (connection.goAwaySent() && lastStreamId > connection.remote().lastStreamKnownByPeer()) {
|
2015-03-20 02:36:24 +01:00
|
|
|
throw connectionError(PROTOCOL_ERROR, "Last stream identifier must not increase between " +
|
|
|
|
"sending multiple GOAWAY frames (was '%d', is '%d').",
|
2015-05-03 17:49:49 +02:00
|
|
|
connection.remote().lastStreamKnownByPeer(),
|
2015-03-20 02:36:24 +01:00
|
|
|
lastStreamId);
|
|
|
|
}
|
|
|
|
connection.goAwaySent(lastStreamId, errorCode, debugData);
|
|
|
|
|
2015-05-11 21:10:23 +02:00
|
|
|
// Need to retain before we write the buffer because if we do it after the refCnt could already be 0 and
|
|
|
|
// result in an IllegalRefCountException.
|
|
|
|
debugData.retain();
|
2015-03-20 02:36:24 +01:00
|
|
|
ChannelFuture future = frameWriter().writeGoAway(ctx, lastStreamId, errorCode, debugData, promise);
|
|
|
|
|
2015-05-11 21:10:23 +02:00
|
|
|
if (future.isDone()) {
|
|
|
|
processGoAwayWriteResult(ctx, lastStreamId, errorCode, debugData, future);
|
|
|
|
} else {
|
|
|
|
future.addListener(new ChannelFutureListener() {
|
|
|
|
@Override
|
|
|
|
public void operationComplete(ChannelFuture future) throws Exception {
|
|
|
|
processGoAwayWriteResult(ctx, lastStreamId, errorCode, debugData, future);
|
2015-03-20 02:36:24 +01:00
|
|
|
}
|
2015-05-11 21:10:23 +02:00
|
|
|
});
|
|
|
|
}
|
2015-03-20 02:36:24 +01:00
|
|
|
|
|
|
|
return future;
|
2015-05-11 21:10:23 +02:00
|
|
|
} catch (Throwable cause) { // Make sure to catch Throwable because we are doing a retain() in this method.
|
2015-01-20 01:48:11 +01:00
|
|
|
debugData.release();
|
2015-05-11 21:10:23 +02:00
|
|
|
return promise.setFailure(cause);
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-30 01:27:59 +02:00
|
|
|
/**
|
|
|
|
* Closes the connection if the graceful shutdown process has completed.
|
|
|
|
* @param future Represents the status that will be passed to the {@link #closeListener}.
|
|
|
|
*/
|
|
|
|
private void checkCloseConnection(ChannelFuture future) {
|
|
|
|
// If this connection is closing and the graceful shutdown has completed, close the connection
|
|
|
|
// once this operation completes.
|
|
|
|
if (closeListener != null && isGracefulShutdownComplete()) {
|
|
|
|
ChannelFutureListener closeListener = Http2ConnectionHandler.this.closeListener;
|
|
|
|
// This method could be called multiple times
|
|
|
|
// and we don't want to notify the closeListener multiple times.
|
|
|
|
Http2ConnectionHandler.this.closeListener = null;
|
|
|
|
try {
|
|
|
|
closeListener.operationComplete(future);
|
|
|
|
} catch (Exception e) {
|
|
|
|
throw new IllegalStateException("Close listener threw an unexpected exception", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 21:12:29 +02:00
|
|
|
/**
|
|
|
|
* Gets the initial settings to be sent to the remote endpoint.
|
|
|
|
*/
|
|
|
|
private Http2Settings initialSettings() {
|
|
|
|
return initialSettings != null ? initialSettings : decoder.localSettings();
|
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
/**
|
2015-04-23 23:23:23 +02:00
|
|
|
* Close the remote endpoint with with a {@code GO_AWAY} frame. Does <strong>not</strong> flush
|
|
|
|
* immediately, this is the responsibility of the caller.
|
2015-01-20 01:48:11 +01:00
|
|
|
*/
|
2015-04-02 23:39:46 +02:00
|
|
|
private ChannelFuture goAway(ChannelHandlerContext ctx, Http2Exception cause) {
|
2015-01-20 01:48:11 +01:00
|
|
|
long errorCode = cause != null ? cause.error().code() : NO_ERROR.code();
|
|
|
|
ByteBuf debugData = Http2CodecUtil.toByteBuf(ctx, cause);
|
2015-03-20 02:36:24 +01:00
|
|
|
int lastKnownStream = connection().remote().lastStreamCreated();
|
2015-04-02 23:39:46 +02:00
|
|
|
return goAway(ctx, lastKnownStream, errorCode, debugData, ctx.newPromise());
|
2015-01-20 01:48:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the client preface string if this is a client connection, otherwise returns {@code null}.
|
|
|
|
*/
|
|
|
|
private static ByteBuf clientPrefaceString(Http2Connection connection) {
|
|
|
|
return connection.isServer() ? connectionPrefaceBuf() : null;
|
|
|
|
}
|
|
|
|
|
2015-05-11 21:10:23 +02:00
|
|
|
private static void processGoAwayWriteResult(final ChannelHandlerContext ctx, final int lastStreamId,
|
|
|
|
final long errorCode, final ByteBuf debugData, ChannelFuture future) {
|
|
|
|
try {
|
|
|
|
if (future.isSuccess()) {
|
|
|
|
if (errorCode != NO_ERROR.code()) {
|
|
|
|
if (logger.isDebugEnabled()) {
|
|
|
|
logger.debug(
|
|
|
|
format("Sent GOAWAY: lastStreamId '%d', errorCode '%d', " +
|
|
|
|
"debugData '%s'. Forcing shutdown of the connection.",
|
|
|
|
lastStreamId, errorCode, debugData.toString(UTF_8)),
|
|
|
|
future.cause());
|
|
|
|
}
|
|
|
|
ctx.close();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (logger.isErrorEnabled()) {
|
|
|
|
logger.error(
|
|
|
|
format("Sending GOAWAY failed: lastStreamId '%d', errorCode '%d', " +
|
|
|
|
"debugData '%s'. Forcing shutdown of the connection.",
|
|
|
|
lastStreamId, errorCode, debugData.toString(UTF_8)), future.cause());
|
|
|
|
}
|
|
|
|
ctx.close();
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
// We're done with the debug data now.
|
|
|
|
debugData.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-20 01:48:11 +01:00
|
|
|
/**
|
|
|
|
* Closes the channel when the future completes.
|
|
|
|
*/
|
|
|
|
private static final class ClosingChannelFutureListener implements ChannelFutureListener {
|
|
|
|
private final ChannelHandlerContext ctx;
|
|
|
|
private final ChannelPromise promise;
|
|
|
|
|
|
|
|
ClosingChannelFutureListener(ChannelHandlerContext ctx, ChannelPromise promise) {
|
|
|
|
this.ctx = ctx;
|
|
|
|
this.promise = promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void operationComplete(ChannelFuture sentGoAwayFuture) throws Exception {
|
|
|
|
ctx.close(promise);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|