HttpProxyHandler: allow setting headers

Motivation:

In some environments, the HTTP CONNECT handshake requires special headers to work.

Modification:

Update HttpProxyHandler to accept a HttpHeaders argument.

Result:

The header is passed along in the HTTP CONNECT request, and the proxy request can be successfully completed.
This commit is contained in:
Spencer Fang 2017-05-12 16:30:38 -07:00 committed by Norman Maurer
parent 2a376eeb1b
commit ec490b2a88
2 changed files with 45 additions and 8 deletions

View File

@ -25,6 +25,7 @@ import io.netty.handler.codec.http.DefaultFullHttpRequest;
import io.netty.handler.codec.http.FullHttpRequest; import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpClientCodec; import io.netty.handler.codec.http.HttpClientCodec;
import io.netty.handler.codec.http.HttpHeaderNames; import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpMethod; import io.netty.handler.codec.http.HttpMethod;
import io.netty.handler.codec.http.HttpResponse; import io.netty.handler.codec.http.HttpResponse;
import io.netty.handler.codec.http.HttpResponseStatus; import io.netty.handler.codec.http.HttpResponseStatus;
@ -47,15 +48,26 @@ public final class HttpProxyHandler extends ProxyHandler {
private final String password; private final String password;
private final CharSequence authorization; private final CharSequence authorization;
private HttpResponseStatus status; private HttpResponseStatus status;
private HttpHeaders headers;
public HttpProxyHandler(SocketAddress proxyAddress) { public HttpProxyHandler(SocketAddress proxyAddress) {
this(proxyAddress, null);
}
public HttpProxyHandler(SocketAddress proxyAddress, HttpHeaders headers) {
super(proxyAddress); super(proxyAddress);
username = null; username = null;
password = null; password = null;
authorization = null; authorization = null;
this.headers = headers;
} }
public HttpProxyHandler(SocketAddress proxyAddress, String username, String password) { public HttpProxyHandler(SocketAddress proxyAddress, String username, String password) {
this(proxyAddress, username, password, null);
}
public HttpProxyHandler(SocketAddress proxyAddress, String username, String password,
HttpHeaders headers) {
super(proxyAddress); super(proxyAddress);
if (username == null) { if (username == null) {
throw new NullPointerException("username"); throw new NullPointerException("username");
@ -73,6 +85,8 @@ public final class HttpProxyHandler extends ProxyHandler {
authz.release(); authz.release();
authzBase64.release(); authzBase64.release();
this.headers = headers;
} }
@Override @Override
@ -125,6 +139,10 @@ public final class HttpProxyHandler extends ProxyHandler {
req.headers().set(HttpHeaderNames.PROXY_AUTHORIZATION, authorization); req.headers().set(HttpHeaderNames.PROXY_AUTHORIZATION, authorization);
} }
if (headers != null) {
req.headers().add(headers);
}
return req; return req;
} }

View File

@ -17,8 +17,10 @@ package io.netty.handler.proxy;
import io.netty.channel.ChannelHandlerContext; import io.netty.channel.ChannelHandlerContext;
import io.netty.channel.ChannelPromise; import io.netty.channel.ChannelPromise;
import io.netty.handler.codec.http.DefaultHttpHeaders;
import io.netty.handler.codec.http.FullHttpRequest; import io.netty.handler.codec.http.FullHttpRequest;
import io.netty.handler.codec.http.HttpHeaderNames; import io.netty.handler.codec.http.HttpHeaderNames;
import io.netty.handler.codec.http.HttpHeaders;
import io.netty.handler.codec.http.HttpVersion; import io.netty.handler.codec.http.HttpVersion;
import io.netty.util.NetUtil; import io.netty.util.NetUtil;
import org.junit.Test; import org.junit.Test;
@ -34,28 +36,37 @@ public class HttpProxyHandlerTest {
@Test @Test
public void testIpv6() throws Exception { public void testIpv6() throws Exception {
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("::1"), 8080); InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("::1"), 8080);
testInitialMessage(socketAddress, "[::1]:8080"); testInitialMessage(socketAddress, "[::1]:8080", null);
} }
@Test @Test
public void testIpv6Unresolved() throws Exception { public void testIpv6Unresolved() throws Exception {
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("::1", 8080); InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("::1", 8080);
testInitialMessage(socketAddress, "[::1]:8080"); testInitialMessage(socketAddress, "[::1]:8080", null);
} }
@Test @Test
public void testIpv4() throws Exception { public void testIpv4() throws Exception {
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("10.0.0.1"), 8080); InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("10.0.0.1"), 8080);
testInitialMessage(socketAddress, "10.0.0.1:8080"); testInitialMessage(socketAddress, "10.0.0.1:8080", null);
} }
@Test @Test
public void testIpv4Unresolved() throws Exception { public void testIpv4Unresolved() throws Exception {
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("10.0.0.1", 8080); InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("10.0.0.1", 8080);
testInitialMessage(socketAddress, "10.0.0.1:8080"); testInitialMessage(socketAddress, "10.0.0.1:8080", null);
} }
private static void testInitialMessage(InetSocketAddress socketAddress, String expected) throws Exception { @Test
public void testCustomHeaders() throws Exception {
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("10.0.0.1", 8080);
testInitialMessage(socketAddress, "10.0.0.1:8080",
new DefaultHttpHeaders().add("CUSTOM_HEADER", "CUSTOM_VALUE1")
.add("CUSTOM_HEADER", "CUSTOM_VALUE2"));
}
private static void testInitialMessage(InetSocketAddress socketAddress, String hostPort,
HttpHeaders headers) throws Exception {
InetSocketAddress proxyAddress = new InetSocketAddress(NetUtil.LOCALHOST, 8080); InetSocketAddress proxyAddress = new InetSocketAddress(NetUtil.LOCALHOST, 8080);
ChannelPromise promise = mock(ChannelPromise.class); ChannelPromise promise = mock(ChannelPromise.class);
@ -64,14 +75,22 @@ public class HttpProxyHandlerTest {
ChannelHandlerContext ctx = mock(ChannelHandlerContext.class); ChannelHandlerContext ctx = mock(ChannelHandlerContext.class);
when(ctx.connect(same(proxyAddress), isNull(InetSocketAddress.class), same(promise))).thenReturn(promise); when(ctx.connect(same(proxyAddress), isNull(InetSocketAddress.class), same(promise))).thenReturn(promise);
HttpProxyHandler handler = new HttpProxyHandler(new InetSocketAddress(NetUtil.LOCALHOST, 8080)); HttpProxyHandler handler = new HttpProxyHandler(new InetSocketAddress(NetUtil.LOCALHOST, 8080), headers);
handler.connect(ctx, socketAddress, null, promise); handler.connect(ctx, socketAddress, null, promise);
FullHttpRequest request = (FullHttpRequest) handler.newInitialMessage(ctx); FullHttpRequest request = (FullHttpRequest) handler.newInitialMessage(ctx);
try { try {
assertEquals(HttpVersion.HTTP_1_1, request.protocolVersion()); assertEquals(HttpVersion.HTTP_1_1, request.protocolVersion());
assertEquals(expected, request.uri()); assertEquals(hostPort, request.uri());
assertEquals(expected, request.headers().get(HttpHeaderNames.HOST)); HttpHeaders actualHeaders = request.headers();
assertEquals(hostPort, actualHeaders.get(HttpHeaderNames.HOST));
if (headers != null) {
// The actual request header is a strict superset of the custom header
for (String name : headers.names()) {
assertEquals(headers.getAll(name), actualHeaders.getAll(name));
}
}
} finally { } finally {
request.release(); request.release();
} }