Adding payload to the Http2Client example.
Modifications: When trying out the Http2Client example I noticed that adding a request payload would not cause a data frame to written. This seems to be because writeHeaders completes the promise, and then the writeData call ends up in FlowControlWriter.writeFrame, isDone is true and the data released and the call aborted and returned. Adding a new promise for the writeData method allows a data frame to be written. Result: A body/payload can now be sent to the server. The example was updated to simply echo the payload received back to the calling client.
This commit is contained in:
parent
9d08a34c34
commit
cbe20faf6c
@ -15,6 +15,7 @@
|
|||||||
package io.netty.example.http2.client;
|
package io.netty.example.http2.client;
|
||||||
|
|
||||||
import io.netty.bootstrap.Bootstrap;
|
import io.netty.bootstrap.Bootstrap;
|
||||||
|
import io.netty.buffer.Unpooled;
|
||||||
import io.netty.channel.Channel;
|
import io.netty.channel.Channel;
|
||||||
import io.netty.channel.ChannelFuture;
|
import io.netty.channel.ChannelFuture;
|
||||||
import io.netty.channel.ChannelOption;
|
import io.netty.channel.ChannelOption;
|
||||||
@ -27,6 +28,7 @@ import io.netty.handler.codec.http.HttpHeaders;
|
|||||||
import io.netty.handler.codec.http2.Http2OrHttpChooser.SelectedProtocol;
|
import io.netty.handler.codec.http2.Http2OrHttpChooser.SelectedProtocol;
|
||||||
import io.netty.handler.ssl.SslContext;
|
import io.netty.handler.ssl.SslContext;
|
||||||
import io.netty.handler.ssl.util.InsecureTrustManagerFactory;
|
import io.netty.handler.ssl.util.InsecureTrustManagerFactory;
|
||||||
|
import io.netty.util.CharsetUtil;
|
||||||
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
|
|
||||||
@ -76,8 +78,9 @@ public final class Http2Client {
|
|||||||
Http2ClientConnectionHandler http2ConnectionHandler = initializer.connectionHandler();
|
Http2ClientConnectionHandler http2ConnectionHandler = initializer.connectionHandler();
|
||||||
http2ConnectionHandler.awaitInitialization();
|
http2ConnectionHandler.awaitInitialization();
|
||||||
|
|
||||||
// Create a simple GET request with just headers.
|
// Create a simple POST request with just headers.
|
||||||
FullHttpRequest request = new DefaultFullHttpRequest(HTTP_1_1, GET, "/whatever");
|
FullHttpRequest request = new DefaultFullHttpRequest(HTTP_1_1, POST, "/whatever",
|
||||||
|
Unpooled.copiedBuffer("sample data".getBytes(CharsetUtil.UTF_8)));
|
||||||
request.headers().add(HttpHeaders.Names.HOST, HOST + ':' + PORT);
|
request.headers().add(HttpHeaders.Names.HOST, HOST + ':' + PORT);
|
||||||
|
|
||||||
// Send the request to the server.
|
// Send the request to the server.
|
||||||
|
@ -110,7 +110,7 @@ public class Http2ClientConnectionHandler extends AbstractHttp2ConnectionHandler
|
|||||||
int streamId = nextStreamId();
|
int streamId = nextStreamId();
|
||||||
writeHeaders(ctx, promise, streamId, headers.build(), 0, !hasData, false);
|
writeHeaders(ctx, promise, streamId, headers.build(), 0, !hasData, false);
|
||||||
if (hasData) {
|
if (hasData) {
|
||||||
writeData(ctx, promise, streamId, httpMsg.content(), 0, true, true, false);
|
writeData(ctx, ctx.newPromise(), streamId, httpMsg.content(), 0, true, true, false);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
ctx.write(msg, promise);
|
ctx.write(msg, promise);
|
||||||
|
@ -42,7 +42,7 @@ public class HelloWorldHttp1Handler extends SimpleChannelInboundHandler<HttpRequ
|
|||||||
boolean keepAlive = HttpHeaderUtil.isKeepAlive(req);
|
boolean keepAlive = HttpHeaderUtil.isKeepAlive(req);
|
||||||
|
|
||||||
ByteBuf content = ctx.alloc().buffer();
|
ByteBuf content = ctx.alloc().buffer();
|
||||||
content.writeBytes(HelloWorldHttp2Handler.RESPONSE_BYTES);
|
content.writeBytes(HelloWorldHttp2Handler.RESPONSE_BYTES.duplicate());
|
||||||
|
|
||||||
FullHttpResponse response = new DefaultFullHttpResponse(HTTP_1_1, OK, content);
|
FullHttpResponse response = new DefaultFullHttpResponse(HTTP_1_1, OK, content);
|
||||||
response.headers().set(CONTENT_TYPE, "text/plain; charset=UTF-8");
|
response.headers().set(CONTENT_TYPE, "text/plain; charset=UTF-8");
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
|
|
||||||
package io.netty.example.http2.server;
|
package io.netty.example.http2.server;
|
||||||
|
|
||||||
|
import static io.netty.buffer.Unpooled.copiedBuffer;
|
||||||
|
import static io.netty.buffer.Unpooled.unreleasableBuffer;
|
||||||
import static io.netty.example.http2.Http2ExampleUtil.UPGRADE_RESPONSE_HEADER;
|
import static io.netty.example.http2.Http2ExampleUtil.UPGRADE_RESPONSE_HEADER;
|
||||||
import static io.netty.util.internal.logging.InternalLogLevel.INFO;
|
import static io.netty.util.internal.logging.InternalLogLevel.INFO;
|
||||||
import io.netty.buffer.ByteBuf;
|
import io.netty.buffer.ByteBuf;
|
||||||
@ -44,7 +46,7 @@ public class HelloWorldHttp2Handler extends AbstractHttp2ConnectionHandler {
|
|||||||
|
|
||||||
private static final Http2FrameLogger logger = new Http2FrameLogger(INFO,
|
private static final Http2FrameLogger logger = new Http2FrameLogger(INFO,
|
||||||
InternalLoggerFactory.getInstance(Http2ClientConnectionHandler.class));
|
InternalLoggerFactory.getInstance(Http2ClientConnectionHandler.class));
|
||||||
static final byte[] RESPONSE_BYTES = "Hello World".getBytes(CharsetUtil.UTF_8);
|
static final ByteBuf RESPONSE_BYTES = unreleasableBuffer(copiedBuffer("Hello World", CharsetUtil.UTF_8));
|
||||||
|
|
||||||
public HelloWorldHttp2Handler() {
|
public HelloWorldHttp2Handler() {
|
||||||
this(new DefaultHttp2Connection(true));
|
this(new DefaultHttp2Connection(true));
|
||||||
@ -79,7 +81,7 @@ public class HelloWorldHttp2Handler extends AbstractHttp2ConnectionHandler {
|
|||||||
public void onDataRead(ChannelHandlerContext ctx, int streamId, ByteBuf data, int padding,
|
public void onDataRead(ChannelHandlerContext ctx, int streamId, ByteBuf data, int padding,
|
||||||
boolean endOfStream, boolean endOfSegment) throws Http2Exception {
|
boolean endOfStream, boolean endOfSegment) throws Http2Exception {
|
||||||
if (endOfStream) {
|
if (endOfStream) {
|
||||||
sendResponse(ctx(), streamId);
|
sendResponse(ctx(), streamId, data.retain());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -92,7 +94,7 @@ public class HelloWorldHttp2Handler extends AbstractHttp2ConnectionHandler {
|
|||||||
boolean exclusive, int padding, boolean endStream, boolean endSegment)
|
boolean exclusive, int padding, boolean endStream, boolean endSegment)
|
||||||
throws Http2Exception {
|
throws Http2Exception {
|
||||||
if (endStream) {
|
if (endStream) {
|
||||||
sendResponse(ctx(), streamId);
|
sendResponse(ctx(), streamId, RESPONSE_BYTES.duplicate());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -105,14 +107,11 @@ public class HelloWorldHttp2Handler extends AbstractHttp2ConnectionHandler {
|
|||||||
/**
|
/**
|
||||||
* Sends a "Hello World" DATA frame to the client.
|
* Sends a "Hello World" DATA frame to the client.
|
||||||
*/
|
*/
|
||||||
private void sendResponse(ChannelHandlerContext ctx, int streamId) {
|
private void sendResponse(ChannelHandlerContext ctx, int streamId, ByteBuf payload) {
|
||||||
// Send a frame for the response status
|
// Send a frame for the response status
|
||||||
Http2Headers headers = DefaultHttp2Headers.newBuilder().status("200").build();
|
Http2Headers headers = DefaultHttp2Headers.newBuilder().status("200").build();
|
||||||
writeHeaders(ctx(), ctx().newPromise(), streamId, headers, 0, false, false);
|
writeHeaders(ctx(), ctx().newPromise(), streamId, headers, 0, false, false);
|
||||||
|
|
||||||
// Send a data frame with the response message.
|
writeData(ctx(), ctx().newPromise(), streamId, payload, 0, true, true, false);
|
||||||
ByteBuf content = ctx.alloc().buffer();
|
|
||||||
content.writeBytes(RESPONSE_BYTES);
|
|
||||||
writeData(ctx(), ctx().newPromise(), streamId, content, 0, true, true, false);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user