2016-12-22 10:27:03 +01: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.proxy;
|
|
|
|
|
2019-02-02 07:16:36 +01:00
|
|
|
import io.netty.bootstrap.Bootstrap;
|
|
|
|
import io.netty.bootstrap.ServerBootstrap;
|
|
|
|
import io.netty.channel.Channel;
|
|
|
|
import io.netty.channel.ChannelFuture;
|
2016-12-22 10:27:03 +01:00
|
|
|
import io.netty.channel.ChannelHandlerContext;
|
2019-03-13 09:46:10 +01:00
|
|
|
import io.netty.channel.ChannelInboundHandler;
|
2019-02-02 07:16:36 +01:00
|
|
|
import io.netty.channel.ChannelInitializer;
|
2016-12-22 10:27:03 +01:00
|
|
|
import io.netty.channel.ChannelPromise;
|
2019-02-02 07:16:36 +01:00
|
|
|
import io.netty.channel.EventLoopGroup;
|
|
|
|
import io.netty.channel.MultithreadEventLoopGroup;
|
|
|
|
import io.netty.channel.local.LocalAddress;
|
|
|
|
import io.netty.channel.local.LocalChannel;
|
|
|
|
import io.netty.channel.local.LocalHandler;
|
|
|
|
import io.netty.channel.local.LocalServerChannel;
|
|
|
|
import io.netty.handler.codec.http.DefaultFullHttpResponse;
|
2017-05-13 01:30:38 +02:00
|
|
|
import io.netty.handler.codec.http.DefaultHttpHeaders;
|
2016-12-22 10:27:03 +01:00
|
|
|
import io.netty.handler.codec.http.FullHttpRequest;
|
|
|
|
import io.netty.handler.codec.http.HttpHeaderNames;
|
2017-05-13 01:30:38 +02:00
|
|
|
import io.netty.handler.codec.http.HttpHeaders;
|
2019-02-02 07:16:36 +01:00
|
|
|
import io.netty.handler.codec.http.HttpResponseEncoder;
|
|
|
|
import io.netty.handler.codec.http.HttpResponseStatus;
|
2016-12-22 10:27:03 +01:00
|
|
|
import io.netty.handler.codec.http.HttpVersion;
|
2019-02-02 07:16:36 +01:00
|
|
|
import io.netty.handler.proxy.HttpProxyHandler.HttpProxyConnectException;
|
2016-12-22 10:27:03 +01:00
|
|
|
import io.netty.util.NetUtil;
|
2019-02-02 07:16:36 +01:00
|
|
|
import java.util.concurrent.atomic.AtomicReference;
|
2016-12-22 10:27:03 +01:00
|
|
|
import org.junit.Test;
|
|
|
|
|
|
|
|
import java.net.InetAddress;
|
|
|
|
import java.net.InetSocketAddress;
|
|
|
|
|
2019-02-02 07:16:36 +01:00
|
|
|
import static org.junit.Assert.*;
|
2016-12-22 10:27:03 +01:00
|
|
|
import static org.mockito.Mockito.*;
|
|
|
|
|
|
|
|
public class HttpProxyHandlerTest {
|
|
|
|
|
|
|
|
@Test
|
2018-03-23 11:58:43 +01:00
|
|
|
public void testHostname() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("localhost"), 8080);
|
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"localhost:8080",
|
|
|
|
"localhost:8080",
|
|
|
|
null,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testHostnameUnresolved() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("localhost", 8080);
|
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"localhost:8080",
|
|
|
|
"localhost:8080",
|
|
|
|
null,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testHostHeaderWithHttpDefaultPort() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("localhost"), 80);
|
|
|
|
testInitialMessage(socketAddress,
|
|
|
|
"localhost:80",
|
|
|
|
"localhost:80", null,
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testHostHeaderWithHttpDefaultPortIgnored() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("localhost", 80);
|
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"localhost:80",
|
|
|
|
"localhost",
|
|
|
|
null,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testHostHeaderWithHttpsDefaultPort() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("localhost"), 443);
|
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"localhost:443",
|
|
|
|
"localhost:443",
|
|
|
|
null,
|
|
|
|
false);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testHostHeaderWithHttpsDefaultPortIgnored() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("localhost", 443);
|
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"localhost:443",
|
|
|
|
"localhost",
|
|
|
|
null,
|
|
|
|
true);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testIpv6() throws Exception {
|
2016-12-22 10:27:03 +01:00
|
|
|
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("::1"), 8080);
|
2018-03-23 11:58:43 +01:00
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"[::1]:8080",
|
|
|
|
"[::1]:8080",
|
|
|
|
null,
|
|
|
|
true);
|
2016-12-22 10:27:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2018-03-23 11:58:43 +01:00
|
|
|
public void testIpv6Unresolved() throws Exception {
|
2016-12-22 10:27:03 +01:00
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("::1", 8080);
|
2018-03-23 11:58:43 +01:00
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"[::1]:8080",
|
|
|
|
"[::1]:8080",
|
|
|
|
null,
|
|
|
|
true);
|
2016-12-22 10:27:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2018-03-23 11:58:43 +01:00
|
|
|
public void testIpv4() throws Exception {
|
2016-12-22 10:27:03 +01:00
|
|
|
InetSocketAddress socketAddress = new InetSocketAddress(InetAddress.getByName("10.0.0.1"), 8080);
|
2018-03-23 11:58:43 +01:00
|
|
|
testInitialMessage(socketAddress,
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
null,
|
|
|
|
true);
|
2016-12-22 10:27:03 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2018-03-23 11:58:43 +01:00
|
|
|
public void testIpv4Unresolved() throws Exception {
|
2016-12-22 10:27:03 +01:00
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("10.0.0.1", 8080);
|
2018-03-23 11:58:43 +01:00
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
null,
|
|
|
|
true);
|
2016-12-22 10:27:03 +01:00
|
|
|
}
|
|
|
|
|
2017-05-13 01:30:38 +02:00
|
|
|
@Test
|
|
|
|
public void testCustomHeaders() throws Exception {
|
|
|
|
InetSocketAddress socketAddress = InetSocketAddress.createUnresolved("10.0.0.1", 8080);
|
2018-03-23 11:58:43 +01:00
|
|
|
testInitialMessage(
|
|
|
|
socketAddress,
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
"10.0.0.1:8080",
|
|
|
|
new DefaultHttpHeaders()
|
|
|
|
.add("CUSTOM_HEADER", "CUSTOM_VALUE1")
|
|
|
|
.add("CUSTOM_HEADER", "CUSTOM_VALUE2"),
|
|
|
|
true);
|
2017-05-13 01:30:38 +02:00
|
|
|
}
|
|
|
|
|
2019-02-02 07:16:36 +01:00
|
|
|
@Test
|
|
|
|
public void testExceptionDuringConnect() throws Exception {
|
|
|
|
EventLoopGroup group = null;
|
|
|
|
Channel serverChannel = null;
|
|
|
|
Channel clientChannel = null;
|
|
|
|
try {
|
|
|
|
group = new MultithreadEventLoopGroup(1, LocalHandler.newFactory());
|
|
|
|
final LocalAddress addr = new LocalAddress("a");
|
|
|
|
final AtomicReference<Throwable> exception = new AtomicReference<Throwable>();
|
|
|
|
ChannelFuture sf =
|
|
|
|
new ServerBootstrap().channel(LocalServerChannel.class).group(group).childHandler(
|
|
|
|
new ChannelInitializer<Channel>() {
|
|
|
|
|
|
|
|
@Override
|
|
|
|
protected void initChannel(Channel ch) {
|
|
|
|
ch.pipeline().addFirst(new HttpResponseEncoder());
|
2019-03-13 09:46:10 +01:00
|
|
|
ch.pipeline().addFirst(new ChannelInboundHandler() {
|
2019-02-02 07:16:36 +01:00
|
|
|
@Override
|
|
|
|
public void channelActive(ChannelHandlerContext ctx) {
|
|
|
|
DefaultFullHttpResponse response = new DefaultFullHttpResponse(
|
|
|
|
HttpVersion.HTTP_1_1,
|
|
|
|
HttpResponseStatus.BAD_GATEWAY);
|
|
|
|
response.headers().add("name", "value");
|
|
|
|
response.headers().add(HttpHeaderNames.CONTENT_LENGTH, "0");
|
|
|
|
ctx.writeAndFlush(response);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}).bind(addr);
|
|
|
|
serverChannel = sf.sync().channel();
|
|
|
|
ChannelFuture cf = new Bootstrap().channel(LocalChannel.class).group(group).handler(
|
|
|
|
new ChannelInitializer<Channel>() {
|
|
|
|
@Override
|
|
|
|
protected void initChannel(Channel ch) {
|
|
|
|
ch.pipeline().addFirst(new HttpProxyHandler(addr));
|
2019-03-13 09:46:10 +01:00
|
|
|
ch.pipeline().addLast(new ChannelInboundHandler() {
|
2019-02-02 07:16:36 +01:00
|
|
|
@Override
|
|
|
|
public void exceptionCaught(ChannelHandlerContext ctx,
|
|
|
|
Throwable cause) {
|
|
|
|
exception.set(cause);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}).connect(new InetSocketAddress("localhost", 1234));
|
|
|
|
clientChannel = cf.sync().channel();
|
|
|
|
clientChannel.close().sync();
|
|
|
|
|
|
|
|
assertTrue(exception.get() instanceof HttpProxyConnectException);
|
|
|
|
HttpProxyConnectException actual = (HttpProxyConnectException) exception.get();
|
|
|
|
assertNotNull(actual.headers());
|
|
|
|
assertEquals("value", actual.headers().get("name"));
|
|
|
|
} finally {
|
|
|
|
if (clientChannel != null) {
|
|
|
|
clientChannel.close();
|
|
|
|
}
|
|
|
|
if (serverChannel != null) {
|
|
|
|
serverChannel.close();
|
|
|
|
}
|
|
|
|
if (group != null) {
|
|
|
|
group.shutdownGracefully();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-23 11:58:43 +01:00
|
|
|
private static void testInitialMessage(InetSocketAddress socketAddress,
|
|
|
|
String expectedUrl,
|
|
|
|
String expectedHostHeader,
|
|
|
|
HttpHeaders headers,
|
|
|
|
boolean ignoreDefaultPortsInConnectHostHeader) throws Exception {
|
2016-12-22 10:27:03 +01:00
|
|
|
InetSocketAddress proxyAddress = new InetSocketAddress(NetUtil.LOCALHOST, 8080);
|
|
|
|
|
|
|
|
ChannelPromise promise = mock(ChannelPromise.class);
|
|
|
|
verifyNoMoreInteractions(promise);
|
|
|
|
|
|
|
|
ChannelHandlerContext ctx = mock(ChannelHandlerContext.class);
|
|
|
|
when(ctx.connect(same(proxyAddress), isNull(InetSocketAddress.class), same(promise))).thenReturn(promise);
|
|
|
|
|
2018-03-23 11:58:43 +01:00
|
|
|
HttpProxyHandler handler = new HttpProxyHandler(
|
|
|
|
new InetSocketAddress(NetUtil.LOCALHOST, 8080),
|
|
|
|
headers,
|
|
|
|
ignoreDefaultPortsInConnectHostHeader);
|
2016-12-22 10:27:03 +01:00
|
|
|
handler.connect(ctx, socketAddress, null, promise);
|
|
|
|
|
|
|
|
FullHttpRequest request = (FullHttpRequest) handler.newInitialMessage(ctx);
|
|
|
|
try {
|
|
|
|
assertEquals(HttpVersion.HTTP_1_1, request.protocolVersion());
|
2018-03-23 11:58:43 +01:00
|
|
|
assertEquals(expectedUrl, request.uri());
|
2017-05-13 01:30:38 +02:00
|
|
|
HttpHeaders actualHeaders = request.headers();
|
2018-03-23 11:58:43 +01:00
|
|
|
assertEquals(expectedHostHeader, actualHeaders.get(HttpHeaderNames.HOST));
|
2017-05-13 01:30:38 +02:00
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
|
|
|
}
|
2016-12-22 10:27:03 +01:00
|
|
|
} finally {
|
|
|
|
request.release();
|
|
|
|
}
|
|
|
|
verify(ctx).connect(proxyAddress, null, promise);
|
|
|
|
}
|
|
|
|
}
|