From 69826637a8aef11dd3f1c5d5213ab4552c8e60d3 Mon Sep 17 00:00:00 2001 From: nmittler Date: Mon, 7 Apr 2014 14:00:34 -0700 Subject: [PATCH] Adding examples for HTTP/2 framing. Motivation: Provide some example code to show how to bootstrap client and server for use with HTTP/2 framing. Modifications: - Fixed Http2ConnectionHandler to allow headers after stream creation. Needed for response headers. - Added toString() to all frame classes to help with debugging/logging - Added example classes for HTTP/2 Result: HTTP/2 connections now properly support response headers. Examples for HTTP/2 provided with the distribution of examples module. After your change, what will change. --- .../connection/Http2ConnectionHandler.java | 27 ++-- .../draft10/frame/DefaultHttp2DataFrame.java | 12 +- .../frame/DefaultHttp2GoAwayFrame.java | 10 +- .../frame/DefaultHttp2HeadersFrame.java | 9 +- .../draft10/frame/DefaultHttp2PingFrame.java | 9 +- .../frame/DefaultHttp2PriorityFrame.java | 8 ++ .../frame/DefaultHttp2PushPromiseFrame.java | 8 +- .../frame/DefaultHttp2RstStreamFrame.java | 9 ++ .../frame/DefaultHttp2SettingsFrame.java | 20 +++ .../frame/DefaultHttp2WindowUpdateFrame.java | 9 ++ .../codec/http2/draft10/frame/Http2Flags.java | 29 ++++ .../Http2ConnectionHandlerTest.java | 10 +- example/pom.xml | 5 + .../example/http2/client/Http2Client.java | 131 ++++++++++++++++++ .../http2/client/Http2ClientInitializer.java | 45 ++++++ .../http2/client/Http2FrameLogger.java | 77 ++++++++++ .../client/Http2ResponseClientHandler.java | 77 ++++++++++ .../http2/server/HelloWorldHandler.java | 71 ++++++++++ .../example/http2/server/Http2Server.java | 69 +++++++++ .../http2/server/Http2ServerInitializer.java | 37 +++++ 20 files changed, 654 insertions(+), 18 deletions(-) create mode 100644 example/src/main/java/io/netty/example/http2/client/Http2Client.java create mode 100644 example/src/main/java/io/netty/example/http2/client/Http2ClientInitializer.java create mode 100644 example/src/main/java/io/netty/example/http2/client/Http2FrameLogger.java create mode 100644 example/src/main/java/io/netty/example/http2/client/Http2ResponseClientHandler.java create mode 100644 example/src/main/java/io/netty/example/http2/server/HelloWorldHandler.java create mode 100644 example/src/main/java/io/netty/example/http2/server/Http2Server.java create mode 100644 example/src/main/java/io/netty/example/http2/server/Http2ServerInitializer.java diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandler.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandler.java index 6bf6d8a35f..b707b1a951 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandler.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandler.java @@ -247,10 +247,16 @@ public class Http2ConnectionHandler extends ChannelHandlerAdapter { connection.remote().createStream(frame.getStreamId(), frame.getPriority(), frame.isEndOfStream()); } else { - // If the stream already exists, it must be a reserved push stream. If so, open - // it for push to the local endpoint. - stream.verifyState(PROTOCOL_ERROR, RESERVED_REMOTE); - stream.openForPush(); + if (stream.getState() == RESERVED_REMOTE) { + // Received headers for a reserved push stream ... open it for push to the + // local endpoint. + stream.verifyState(PROTOCOL_ERROR, RESERVED_REMOTE); + stream.openForPush(); + } else { + // Receiving headers on an existing stream. Make sure the stream is in an allowed + // state. + stream.verifyState(PROTOCOL_ERROR, OPEN, HALF_CLOSED_LOCAL); + } // If the headers completes this stream, close it. if (frame.isEndOfStream()) { @@ -450,17 +456,20 @@ public class Http2ConnectionHandler extends ChannelHandlerAdapter { stream = connection.local().createStream(frame.getStreamId(), frame.getPriority(), frame.isEndOfStream()); } else { - // If the stream already exists, it must be a reserved push stream. If so, open - // it for push to the remote endpoint. - stream.verifyState(PROTOCOL_ERROR, RESERVED_LOCAL); - stream.openForPush(); + if (stream.getState() == RESERVED_LOCAL) { + // Sending headers on a reserved push stream ... open it for push to the remote + // endpoint. + stream.openForPush(); + } else { + // The stream already exists, make sure it's in an allowed state. + stream.verifyState(PROTOCOL_ERROR, OPEN, HALF_CLOSED_REMOTE); + } // If the headers are the end of the stream, close it now. if (frame.isEndOfStream()) { stream.closeLocalSide(ctx, promise); } } - // Flush to send all of the frames. ctx.writeAndFlush(frame, promise); } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2DataFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2DataFrame.java index 561b910ce4..96257e34ae 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2DataFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2DataFrame.java @@ -17,7 +17,6 @@ package io.netty.handler.codec.http2.draft10.frame; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.MAX_FRAME_PAYLOAD_LENGTH; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.MAX_UNSIGNED_SHORT; - import io.netty.buffer.ByteBuf; import io.netty.buffer.DefaultByteBufHolder; import io.netty.buffer.Unpooled; @@ -121,6 +120,17 @@ public final class DefaultHttp2DataFrame extends DefaultByteBufHolder implements return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", endOfStream=").append(endOfStream); + builder.append(", paddingLength=").append(paddingLength); + builder.append(", contentLength=").append(content().readableBytes()); + builder.append("]"); + return builder.toString(); + } + private Builder copyBuilder() { return new Builder().setStreamId(streamId).setPaddingLength(paddingLength) .setEndOfStream(endOfStream); diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2GoAwayFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2GoAwayFrame.java index 1572a21a2a..36537ec3c0 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2GoAwayFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2GoAwayFrame.java @@ -17,7 +17,6 @@ package io.netty.handler.codec.http2.draft10.frame; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.MAX_FRAME_PAYLOAD_LENGTH; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.MAX_UNSIGNED_INT; - import io.netty.buffer.ByteBuf; import io.netty.buffer.DefaultByteBufHolder; import io.netty.buffer.Unpooled; @@ -110,6 +109,15 @@ public final class DefaultHttp2GoAwayFrame extends DefaultByteBufHolder implemen return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("lastStreamId=").append(lastStreamId); + builder.append(", errorCode=").append(errorCode); + builder.append("]"); + return builder.toString(); + } + private Builder copyBuilder() { return new Builder().setErrorCode(errorCode).setLastStreamId(lastStreamId); } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2HeadersFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2HeadersFrame.java index d5b5c0c759..4ac2d3bfea 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2HeadersFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2HeadersFrame.java @@ -97,8 +97,13 @@ public final class DefaultHttp2HeadersFrame implements Http2HeadersFrame { @Override public String toString() { - return "DefaultHttp2HeadersFrame [streamId=" + streamId + ", priority=" + priority - + ", endOfStream=" + endOfStream + ", headers=" + headers + ']'; + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", priority=").append(priority); + builder.append(", endOfStream=").append(endOfStream); + builder.append(", headers=").append(headers); + builder.append("]"); + return builder.toString(); } public static class Builder { diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PingFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PingFrame.java index bff0e49037..4f49523a81 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PingFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PingFrame.java @@ -16,7 +16,6 @@ package io.netty.handler.codec.http2.draft10.frame; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.PING_FRAME_PAYLOAD_LENGTH; - import io.netty.buffer.ByteBuf; import io.netty.buffer.DefaultByteBufHolder; @@ -100,6 +99,14 @@ public final class DefaultHttp2PingFrame extends DefaultByteBufHolder implements return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("ack=").append(ack); + builder.append("]"); + return builder.toString(); + } + /** * Builds instances of {@link DefaultHttp2PingFrame}. */ diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PriorityFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PriorityFrame.java index 53d643c019..3b77de0414 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PriorityFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PriorityFrame.java @@ -73,6 +73,14 @@ public final class DefaultHttp2PriorityFrame implements Http2PriorityFrame { return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", priority=").append(priority); + builder.append("]"); + return builder.toString(); + } /** * Builds instances of {@link DefaultHttp2PriorityFrame}. */ diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PushPromiseFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PushPromiseFrame.java index c498d46577..1316825de6 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PushPromiseFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2PushPromiseFrame.java @@ -89,8 +89,12 @@ public final class DefaultHttp2PushPromiseFrame implements Http2PushPromiseFrame @Override public String toString() { - return "DefaultHttp2PushPromiseFrame [streamId=" + streamId + ", promisedStreamId=" - + promisedStreamId + ", headers=" + headers + ']'; + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", promisedStreamId=").append(promisedStreamId); + builder.append(", headers=").append(headers); + builder.append("]"); + return builder.toString(); } public static class Builder { diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2RstStreamFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2RstStreamFrame.java index 63ff05cdb6..196a21bb41 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2RstStreamFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2RstStreamFrame.java @@ -74,6 +74,15 @@ public final class DefaultHttp2RstStreamFrame implements Http2RstStreamFrame { return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", errorCode=").append(errorCode); + builder.append("]"); + return builder.toString(); + } + /** * Builds instances of {@link DefaultHttp2RstStreamFrame}. */ diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2SettingsFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2SettingsFrame.java index 9e9d3bc68d..8ca9536cf9 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2SettingsFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2SettingsFrame.java @@ -118,6 +118,26 @@ public final class DefaultHttp2SettingsFrame implements Http2SettingsFrame { return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("ack=").append(ack); + if (headerTableSize != null) { + builder.append(", headerTableSize=").append(headerTableSize); + } + if (pushEnabled != null) { + builder.append(", pushEnabled=").append(pushEnabled); + } + if (maxConcurrentStreams != null) { + builder.append(", maxConcurrentStreams=").append(maxConcurrentStreams); + } + if (initialWindowSize != null) { + builder.append(", initialWindowSize=").append(initialWindowSize); + } + builder.append("]"); + return builder.toString(); + } + /** * Builds instances of {@link DefaultHttp2SettingsFrame}. */ diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2WindowUpdateFrame.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2WindowUpdateFrame.java index 4451e11765..40c32786fa 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2WindowUpdateFrame.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/DefaultHttp2WindowUpdateFrame.java @@ -73,6 +73,15 @@ public final class DefaultHttp2WindowUpdateFrame implements Http2WindowUpdateFra return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(getClass().getSimpleName()).append("["); + builder.append("streamId=").append(streamId); + builder.append(", windowSizeIncrement=").append(windowSizeIncrement); + builder.append("]"); + return builder.toString(); + } + /** * Builds instances of {@link DefaultHttp2WindowUpdateFrame}. */ diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/Http2Flags.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/Http2Flags.java index 94fdbdb294..ef522c38a6 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/Http2Flags.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/draft10/frame/Http2Flags.java @@ -132,6 +132,35 @@ public class Http2Flags { return true; } + @Override + public String toString() { + StringBuilder builder = new StringBuilder(); + builder.append("value = ").append(value).append(" ("); + if (isAck()) { + builder.append("ACK,"); + } + if (isEndOfHeaders()) { + builder.append("END_OF_HEADERS,"); + } + if (isEndOfStream()) { + builder.append("END_OF_STREAM,"); + } + if (isPriorityPresent()) { + builder.append("PRIORITY_PRESENT,"); + } + if (isEndOfSegment()) { + builder.append("END_OF_SEGMENT,"); + } + if (isPadHighPresent()) { + builder.append("PAD_HIGH,"); + } + if (isPadLowPresent()) { + builder.append("PAD_LOW,"); + } + builder.append(")"); + return builder.toString(); + } + private boolean isSet(short mask) { return (value & mask) != 0; } diff --git a/codec-http2/src/test/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandlerTest.java b/codec-http2/src/test/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandlerTest.java index 00c4d36d1b..1c6ec93e1b 100644 --- a/codec-http2/src/test/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandlerTest.java +++ b/codec-http2/src/test/java/io/netty/handler/codec/http2/draft10/connection/Http2ConnectionHandlerTest.java @@ -16,6 +16,8 @@ package io.netty.handler.codec.http2.draft10.connection; import static io.netty.handler.codec.http2.draft10.Http2Error.PROTOCOL_ERROR; +import static io.netty.handler.codec.http2.draft10.connection.Http2Stream.State.RESERVED_LOCAL; +import static io.netty.handler.codec.http2.draft10.connection.Http2Stream.State.RESERVED_REMOTE; import static io.netty.handler.codec.http2.draft10.frame.Http2FrameCodecUtil.PING_FRAME_PAYLOAD_LENGTH; import static io.netty.util.CharsetUtil.UTF_8; import static org.junit.Assert.assertEquals; @@ -227,7 +229,8 @@ public class Http2ConnectionHandlerTest { } @Test - public void inboundHeadersWithForPromisedStreamShouldHalfOpenStream() throws Exception { + public void inboundHeadersForPromisedStreamShouldHalfOpenStream() throws Exception { + when(stream.getState()).thenReturn(RESERVED_REMOTE); Http2Frame frame = new DefaultHttp2HeadersFrame.Builder().setStreamId(STREAM_ID).setPriority(1) .setHeaders(Http2Headers.EMPTY_HEADERS).build(); @@ -237,7 +240,8 @@ public class Http2ConnectionHandlerTest { } @Test - public void inboundHeadersWithForPromisedStreamShouldCloseStream() throws Exception { + public void inboundHeadersForPromisedStreamShouldCloseStream() throws Exception { + when(stream.getState()).thenReturn(RESERVED_REMOTE); Http2Frame frame = new DefaultHttp2HeadersFrame.Builder().setStreamId(STREAM_ID).setPriority(1) .setEndOfStream(true).setHeaders(Http2Headers.EMPTY_HEADERS) @@ -479,6 +483,7 @@ public class Http2ConnectionHandlerTest { @Test public void outboundHeadersShouldOpenStreamForPush() throws Exception { + when(stream.getState()).thenReturn(RESERVED_LOCAL); Http2Frame frame = new DefaultHttp2HeadersFrame.Builder().setStreamId(STREAM_ID).setPriority(1) .setHeaders(Http2Headers.EMPTY_HEADERS).build(); @@ -491,6 +496,7 @@ public class Http2ConnectionHandlerTest { @Test public void outboundHeadersShouldClosePushStream() throws Exception { + when(stream.getState()).thenReturn(RESERVED_LOCAL); Http2Frame frame = new DefaultHttp2HeadersFrame.Builder().setStreamId(STREAM_ID).setPriority(1) .setEndOfStream(true).setHeaders(Http2Headers.EMPTY_HEADERS) diff --git a/example/pom.xml b/example/pom.xml index 332211454a..c5c9fdce87 100644 --- a/example/pom.xml +++ b/example/pom.xml @@ -52,6 +52,11 @@ netty-codec-http ${project.version} + + ${project.groupId} + netty-codec-http2 + ${project.version} + ${project.groupId} netty-codec-socks diff --git a/example/src/main/java/io/netty/example/http2/client/Http2Client.java b/example/src/main/java/io/netty/example/http2/client/Http2Client.java new file mode 100644 index 0000000000..86306f5294 --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/client/Http2Client.java @@ -0,0 +1,131 @@ +/* + * 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.example.http2.client; + +import static java.util.concurrent.TimeUnit.SECONDS; +import io.netty.bootstrap.Bootstrap; +import io.netty.channel.Channel; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelOption; +import io.netty.channel.EventLoopGroup; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.nio.NioSocketChannel; +import io.netty.handler.codec.http.HttpMethod; +import io.netty.handler.codec.http2.draft10.DefaultHttp2Headers; +import io.netty.handler.codec.http2.draft10.Http2Headers; +import io.netty.handler.codec.http2.draft10.frame.DefaultHttp2HeadersFrame; +import io.netty.handler.codec.http2.draft10.frame.Http2Frame; + +import java.net.InetSocketAddress; +import java.util.concurrent.BlockingQueue; + +/** + * An HTTP2 client that allows you to send HTTP2 frames to a server. Inbound and outbound frames + * are logged. + */ +public class Http2Client { + + private final String host; + private final int port; + private final Http2ResponseClientHandler httpResponseHandler; + private Channel channel; + private EventLoopGroup workerGroup; + + public Http2Client(String host, int port) { + this.host = host; + this.port = port; + httpResponseHandler = new Http2ResponseClientHandler(); + } + + public void start() { + if (channel != null) { + System.out.println("Already running!"); + return; + } + + workerGroup = new NioEventLoopGroup(); + + Bootstrap b = new Bootstrap(); + b.group(workerGroup); + b.channel(NioSocketChannel.class); + b.option(ChannelOption.SO_KEEPALIVE, true); + b.remoteAddress(new InetSocketAddress(host, port)); + b.handler(new Http2ClientInitializer(httpResponseHandler)); + + // Start the client. + channel = b.connect().syncUninterruptibly().channel(); + System.out.println("Connected to [" + host + ':' + port + ']'); + } + + public void stop() { + try { + // Wait until the connection is closed. + channel.close().syncUninterruptibly(); + } finally { + if (workerGroup != null) { + workerGroup.shutdownGracefully(); + } + } + } + + public ChannelFuture send(Http2Frame request) { + // Sends the HTTP request. + return channel.writeAndFlush(request); + } + + public Http2Frame get() { + Http2Headers headers = + DefaultHttp2Headers.newBuilder().setAuthority(host) + .setMethod(HttpMethod.GET.name()).build(); + return new DefaultHttp2HeadersFrame.Builder().setHeaders(headers).setStreamId(3) + .setEndOfStream(true).build(); + } + + public BlockingQueue queue() { + return httpResponseHandler.queue(); + } + + public static void main(String[] args) throws Exception { + int port; + if (args.length > 0) { + port = Integer.parseInt(args[0]); + } else { + port = 8443; + } + + final Http2Client client = new Http2Client("localhost", port); + + try { + client.start(); + ChannelFuture requestFuture = client.send(client.get()).sync(); + + if (!requestFuture.isSuccess()) { + requestFuture.cause().printStackTrace(); + } + + // Waits for the complete response + ChannelFuture responseFuture = client.queue().poll(5, SECONDS); + + if (!responseFuture.isSuccess()) { + responseFuture.cause().printStackTrace(); + } + + System.out.println("Finished HTTP/2 request"); + } finally { + client.stop(); + } + } +} diff --git a/example/src/main/java/io/netty/example/http2/client/Http2ClientInitializer.java b/example/src/main/java/io/netty/example/http2/client/Http2ClientInitializer.java new file mode 100644 index 0000000000..3fff0c0d89 --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/client/Http2ClientInitializer.java @@ -0,0 +1,45 @@ +/* + * 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.example.http2.client; + +import static io.netty.util.internal.logging.InternalLogLevel.INFO; +import io.netty.channel.ChannelInitializer; +import io.netty.channel.ChannelPipeline; +import io.netty.channel.socket.SocketChannel; +import io.netty.handler.codec.http2.draft10.connection.Http2ConnectionHandler; +import io.netty.handler.codec.http2.draft10.frame.Http2FrameCodec; + +/** + * Configures the client pipeline to support HTTP/2 frames. + */ +public class Http2ClientInitializer extends ChannelInitializer { + + private final Http2ResponseClientHandler httpResponseHandler; + + public Http2ClientInitializer(Http2ResponseClientHandler httpResponseHandler) { + this.httpResponseHandler = httpResponseHandler; + } + + @Override + public void initChannel(SocketChannel ch) throws Exception { + ChannelPipeline pipeline = ch.pipeline(); + + pipeline.addLast("http2FrameCodec", new Http2FrameCodec()); + pipeline.addLast("spdyFrameLogger", new Http2FrameLogger(INFO)); + pipeline.addLast("http2ConnectionHandler", new Http2ConnectionHandler(false)); + pipeline.addLast("httpHandler", httpResponseHandler); + } +} diff --git a/example/src/main/java/io/netty/example/http2/client/Http2FrameLogger.java b/example/src/main/java/io/netty/example/http2/client/Http2FrameLogger.java new file mode 100644 index 0000000000..7932b3c15b --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/client/Http2FrameLogger.java @@ -0,0 +1,77 @@ +/* + * 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.example.http2.client; + +import io.netty.channel.ChannelHandlerAdapter; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.ChannelPromise; +import io.netty.handler.codec.http2.draft10.frame.Http2Frame; +import io.netty.util.internal.logging.InternalLogLevel; +import io.netty.util.internal.logging.InternalLogger; +import io.netty.util.internal.logging.InternalLoggerFactory; + +/** + * Logs HTTP2 frames for debugging purposes. + */ +public class Http2FrameLogger extends ChannelHandlerAdapter { + + private enum Direction { + INBOUND, OUTBOUND + } + + protected final InternalLogger logger; + private final InternalLogLevel level; + + public Http2FrameLogger(InternalLogLevel level) { + if (level == null) { + throw new NullPointerException("level"); + } + + logger = InternalLoggerFactory.getInstance(getClass()); + this.level = level; + } + + @Override + public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { + if (acceptMessage(msg)) { + log((Http2Frame) msg, Direction.INBOUND); + } + super.channelRead(ctx, msg); + } + + @Override + public void write(ChannelHandlerContext ctx, Object msg, ChannelPromise promise) throws Exception { + if (acceptMessage(msg)) { + log((Http2Frame) msg, Direction.OUTBOUND); + } + super.write(ctx, msg, promise); + } + + private static boolean acceptMessage(Object msg) throws Exception { + return msg instanceof Http2Frame; + } + + private void log(Http2Frame msg, Direction d) { + if (logger.isEnabled(level)) { + StringBuilder b = new StringBuilder("\n----------------"); + b.append(d.name()); + b.append("--------------------\n"); + b.append(msg); + b.append("\n------------------------------------"); + logger.log(level, b.toString()); + } + } +} diff --git a/example/src/main/java/io/netty/example/http2/client/Http2ResponseClientHandler.java b/example/src/main/java/io/netty/example/http2/client/Http2ResponseClientHandler.java new file mode 100644 index 0000000000..0816103127 --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/client/Http2ResponseClientHandler.java @@ -0,0 +1,77 @@ +/* + * 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.example.http2.client; + +import io.netty.buffer.ByteBuf; +import io.netty.channel.ChannelFuture; +import io.netty.channel.ChannelHandlerContext; +import io.netty.channel.SimpleChannelInboundHandler; +import io.netty.handler.codec.http2.draft10.frame.Http2DataFrame; +import io.netty.util.CharsetUtil; + +import java.util.concurrent.BlockingQueue; +import java.util.concurrent.LinkedBlockingQueue; + +/** + * A handler that interprets response messages as text and prints it out to the console. + */ +public class Http2ResponseClientHandler extends SimpleChannelInboundHandler { + private final BlockingQueue queue = new LinkedBlockingQueue(); + + private ByteBuf data; + + @Override + public void messageReceived(ChannelHandlerContext ctx, Http2DataFrame frame) throws Exception { + System.out.println("Received frame: " + frame); + + // Copy the data into the buffer. + int available = frame.content().readableBytes(); + if (data == null) { + data = ctx.alloc().buffer(available); + data.writeBytes(frame.content()); + } else { + // Expand the buffer + ByteBuf newBuffer = ctx.alloc().buffer(data.readableBytes() + available); + newBuffer.writeBytes(data); + newBuffer.writeBytes(frame.content()); + data.release(); + data = newBuffer; + } + + // If it's the last frame, print the complete message. + if (frame.isEndOfStream()) { + byte[] bytes = new byte[data.readableBytes()]; + data.readBytes(bytes); + System.out.println("Received message: " + new String(bytes, CharsetUtil.UTF_8)); + + // Free the data buffer. + data.release(); + data = null; + + queue.add(ctx.channel().newSucceededFuture()); + } + } + + @Override + public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { + queue.add(ctx.channel().newFailedFuture(cause)); + cause.printStackTrace(); + ctx.close(); + } + + public BlockingQueue queue() { + return queue; + } +} diff --git a/example/src/main/java/io/netty/example/http2/server/HelloWorldHandler.java b/example/src/main/java/io/netty/example/http2/server/HelloWorldHandler.java new file mode 100644 index 0000000000..11e45901ff --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/server/HelloWorldHandler.java @@ -0,0 +1,71 @@ +/* + * 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.example.http2.server; + +import io.netty.buffer.ByteBuf; +import io.netty.channel.ChannelHandlerAdapter; +import io.netty.channel.ChannelHandlerContext; +import io.netty.handler.codec.http2.draft10.DefaultHttp2Headers; +import io.netty.handler.codec.http2.draft10.Http2Headers; +import io.netty.handler.codec.http2.draft10.frame.DefaultHttp2DataFrame; +import io.netty.handler.codec.http2.draft10.frame.DefaultHttp2HeadersFrame; +import io.netty.handler.codec.http2.draft10.frame.Http2DataFrame; +import io.netty.handler.codec.http2.draft10.frame.Http2HeadersFrame; +import io.netty.handler.codec.http2.draft10.frame.Http2StreamFrame; +import io.netty.util.CharsetUtil; +import io.netty.util.ReferenceCountUtil; + +/** + * A simple handler that responds with the message "Hello World!". + */ +public class HelloWorldHandler extends ChannelHandlerAdapter { + private static final byte[] RESPONSE_BYTES = "Hello World".getBytes(CharsetUtil.UTF_8); + + @Override + public void channelRead(ChannelHandlerContext ctx, Object msg) throws Exception { + if (msg instanceof Http2StreamFrame) { + Http2StreamFrame frame = (Http2StreamFrame) msg; + if (frame.isEndOfStream()) { + sendResponse(ctx, frame.getStreamId()); + } + } + + ReferenceCountUtil.release(msg); + } + + @Override + public void exceptionCaught(ChannelHandlerContext ctx, Throwable cause) throws Exception { + cause.printStackTrace(); + } + + private void sendResponse(ChannelHandlerContext ctx, int streamId) { + ByteBuf content = ctx.alloc().buffer(); + content.writeBytes(RESPONSE_BYTES); + + // Send a frame for the response status + Http2Headers headers = DefaultHttp2Headers.newBuilder().setStatus("200").build(); + Http2HeadersFrame headersFrame = + new DefaultHttp2HeadersFrame.Builder().setStreamId(streamId).setHeaders(headers) + .build(); + ctx.write(headersFrame); + + // Send a data frame with the response message. + Http2DataFrame data = + new DefaultHttp2DataFrame.Builder().setStreamId(streamId).setEndOfStream(true) + .setContent(content).build(); + ctx.writeAndFlush(data); + } +} diff --git a/example/src/main/java/io/netty/example/http2/server/Http2Server.java b/example/src/main/java/io/netty/example/http2/server/Http2Server.java new file mode 100644 index 0000000000..705cb531a9 --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/server/Http2Server.java @@ -0,0 +1,69 @@ +/* + * 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.example.http2.server; + +import io.netty.bootstrap.ServerBootstrap; +import io.netty.channel.Channel; +import io.netty.channel.ChannelOption; +import io.netty.channel.EventLoopGroup; +import io.netty.channel.nio.NioEventLoopGroup; +import io.netty.channel.socket.nio.NioServerSocketChannel; + +/** + * A HTTP/2 Server that responds to requests with a Hello World. + *

+ * Once started, you can test the server with the example client. + */ +public class Http2Server { + + private final int port; + + public Http2Server(int port) { + this.port = port; + } + + public void run() throws Exception { + // Configure the server. + EventLoopGroup bossGroup = new NioEventLoopGroup(1); + EventLoopGroup workerGroup = new NioEventLoopGroup(); + try { + ServerBootstrap b = new ServerBootstrap(); + b.option(ChannelOption.SO_BACKLOG, 1024); + b.group(bossGroup, workerGroup).channel(NioServerSocketChannel.class) + .childHandler(new Http2ServerInitializer()); + + Channel ch = b.bind(port).sync().channel(); + ch.closeFuture().sync(); + } finally { + bossGroup.shutdownGracefully(); + workerGroup.shutdownGracefully(); + } + } + + public static void main(String[] args) throws Exception { + int port; + if (args.length > 0) { + port = Integer.parseInt(args[0]); + } else { + port = 8443; + } + + System.out.println("HTTP2 server started at port " + port + '.'); + + new Http2Server(port).run(); + } +} diff --git a/example/src/main/java/io/netty/example/http2/server/Http2ServerInitializer.java b/example/src/main/java/io/netty/example/http2/server/Http2ServerInitializer.java new file mode 100644 index 0000000000..aca941973c --- /dev/null +++ b/example/src/main/java/io/netty/example/http2/server/Http2ServerInitializer.java @@ -0,0 +1,37 @@ +/* + * 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.example.http2.server; + +import io.netty.channel.ChannelInitializer; +import io.netty.channel.ChannelPipeline; +import io.netty.channel.socket.SocketChannel; +import io.netty.handler.codec.http2.draft10.connection.Http2ConnectionHandler; +import io.netty.handler.codec.http2.draft10.frame.Http2FrameCodec; + +/** + * Sets up the Netty pipeline + */ +public class Http2ServerInitializer extends ChannelInitializer { + @Override + public void initChannel(SocketChannel ch) throws Exception { + ChannelPipeline p = ch.pipeline(); + + p.addLast("http2FrameCodec", new Http2FrameCodec()); + p.addLast("http2ConnectionHandler", new Http2ConnectionHandler(true)); + p.addLast("helloWorldHandler", new HelloWorldHandler()); + } +}