blob: d5a68e0ef503b412b4ddb970fb3b538e8afc7fb5 [file] [log] [blame]
/*
* ====================================================================
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF 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.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*
*/
package org.apache.http.nio.client.integration;
import java.io.IOException;
import java.net.InetSocketAddress;
import java.net.URISyntaxException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Queue;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import org.apache.http.Header;
import org.apache.http.impl.client.DefaultRedirectStrategy;
import org.apache.http.localserver.HttpAsyncTestBase;
import org.apache.http.HttpException;
import org.apache.http.HttpHost;
import org.apache.http.HttpInetConnection;
import org.apache.http.HttpRequest;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.ProtocolException;
import org.apache.http.ProtocolVersion;
import org.apache.http.client.CircularRedirectException;
import org.apache.http.client.CookieStore;
import org.apache.http.client.RedirectException;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.protocol.HttpClientContext;
import org.apache.http.client.utils.URIBuilder;
import org.apache.http.cookie.SM;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.BasicCookieStore;
import org.apache.http.impl.cookie.BasicClientCookie;
import org.apache.http.impl.nio.bootstrap.HttpServer;
import org.apache.http.localserver.RandomHandler;
import org.apache.http.message.BasicHeader;
import org.apache.http.nio.entity.NStringEntity;
import org.apache.http.nio.protocol.BasicAsyncRequestHandler;
import org.apache.http.nio.reactor.ListenerEndpoint;
import org.apache.http.protocol.HTTP;
import org.apache.http.protocol.HttpContext;
import org.apache.http.protocol.HttpCoreContext;
import org.apache.http.protocol.HttpRequestHandler;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
/**
* Redirection test cases.
*/
@RunWith(Parameterized.class)
public class TestRedirects extends HttpAsyncTestBase {
@Parameterized.Parameters(name = "{0}")
public static Collection<Object[]> protocols() {
return Arrays.asList(new Object[][]{
{ProtocolScheme.http},
{ProtocolScheme.https},
});
}
public TestRedirects(final ProtocolScheme scheme) {
super(scheme);
}
static class BasicRedirectService implements HttpRequestHandler {
private final String schemeName;
private final int statuscode;
public BasicRedirectService(final String schemeName, final int statuscode) {
super();
this.schemeName = schemeName;
this.statuscode = statuscode;
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final HttpInetConnection conn = (HttpInetConnection) context.getAttribute(
HttpCoreContext.HTTP_CONNECTION);
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.equals("/oldlocation/")) {
final String redirectUrl = this.schemeName + "://localhost:" + conn.getLocalPort() + "/newlocation/";
response.setStatusLine(ver, this.statuscode);
response.addHeader(new BasicHeader("Location", redirectUrl));
response.addHeader(new BasicHeader("Connection", "close"));
} else if (uri.equals("/newlocation/")) {
response.setStatusLine(ver, HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
}
static class CircularRedirectService implements HttpRequestHandler {
public CircularRedirectService() {
super();
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.startsWith("/circular-oldlocation")) {
response.setStatusLine(ver, HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", "/circular-location2"));
} else if (uri.startsWith("/circular-location2")) {
response.setStatusLine(ver, HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", "/circular-oldlocation"));
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
}
static class RelativeRedirectService implements HttpRequestHandler {
public RelativeRedirectService() {
super();
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.equals("/oldlocation/")) {
response.setStatusLine(ver, HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", "/relativelocation/"));
} else if (uri.equals("/relativelocation/")) {
response.setStatusLine(ver, HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
}
static class RelativeRedirectService2 implements HttpRequestHandler {
public RelativeRedirectService2() {
super();
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.equals("/test/oldlocation")) {
response.setStatusLine(ver, HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", "relativelocation"));
} else if (uri.equals("/test/relativelocation")) {
response.setStatusLine(ver, HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
}
static class BogusRedirectService implements HttpRequestHandler {
private final String schemeName;
private final String url;
private final boolean absolute;
public BogusRedirectService(final String schemeName, final String url, final boolean absolute) {
super();
this.schemeName = schemeName;
this.url = url;
this.absolute = absolute;
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final HttpInetConnection conn = (HttpInetConnection) context.getAttribute(
HttpCoreContext.HTTP_CONNECTION);
String redirectUrl = this.url;
if (!this.absolute) {
redirectUrl = this.schemeName + "://localhost:" + conn.getLocalPort() + redirectUrl;
}
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.equals("/oldlocation/")) {
response.setStatusLine(ver, HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", redirectUrl));
} else if (uri.equals("/relativelocation/")) {
response.setStatusLine(ver, HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
}
private static class RomeRedirectService implements HttpRequestHandler {
public RomeRedirectService() {
super();
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final String uri = request.getRequestLine().getUri();
if (uri.equals("/rome")) {
response.setStatusCode(HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusCode(HttpStatus.SC_MOVED_TEMPORARILY);
response.addHeader(new BasicHeader("Location", "/rome"));
}
}
}
private static class DifferentHostRedirectService implements HttpRequestHandler {
private final String schemeName;
private final int statusCode;
private int targetHostPort;
public DifferentHostRedirectService(final String schemeName, final int statusCode) {
this.schemeName = schemeName;
this.statusCode = statusCode;
}
@Override
public void handle(final HttpRequest request, final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String uri = request.getRequestLine().getUri();
if (uri.equals("/oldlocation/")) {
final String redirectUrl =
this.schemeName + "://localhost:" + targetHostPort + "/newlocation/";
response.setStatusLine(ver, this.statusCode);
response.addHeader(new BasicHeader("Location", redirectUrl));
response.addHeader(new BasicHeader("Connection", "close"));
} else if (uri.equals("/newlocation/")) {
final String hostHeaderValue = request.getFirstHeader("Host").getValue();
if (hostHeaderValue.equals("localhost:" + targetHostPort)) {
response.setStatusLine(ver, HttpStatus.SC_OK);
final StringEntity entity = new StringEntity("Successful redirect");
response.setEntity(entity);
} else {
response.setStatusLine(ver, 421, "Misdirected Request");
}
} else {
response.setStatusLine(ver, HttpStatus.SC_NOT_FOUND);
}
}
public void setTargetHostPort(final int targetHostPort) {
this.targetHostPort = targetHostPort;
}
}
@Test
public void testBasicRedirect300() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MULTIPLE_CHOICES)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_MULTIPLE_CHOICES, response.getStatusLine().getStatusCode());
Assert.assertEquals("/oldlocation/", reqWrapper.getRequestLine().getUri());
}
@Test
public void testBasicRedirect301() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MOVED_PERMANENTLY)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test
public void testBasicRedirect302() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MOVED_TEMPORARILY)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test
public void testBasicRedirect302NoLocation() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new HttpRequestHandler() {
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
response.setStatusCode(HttpStatus.SC_MOVED_TEMPORARILY);
}
}));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_MOVED_TEMPORARILY, response.getStatusLine().getStatusCode());
Assert.assertEquals("/oldlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test
public void testBasicRedirect303() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_SEE_OTHER)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test
public void testBasicRedirect304() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_NOT_MODIFIED)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_NOT_MODIFIED, response.getStatusLine().getStatusCode());
Assert.assertEquals("/oldlocation/", reqWrapper.getRequestLine().getUri());
}
@Test
public void testBasicRedirect305() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_USE_PROXY)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_USE_PROXY, response.getStatusLine().getStatusCode());
Assert.assertEquals("/oldlocation/", reqWrapper.getRequestLine().getUri());
}
@Test
public void testBasicRedirect307() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_TEMPORARY_REDIRECT)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test(expected=ExecutionException.class)
public void testMaxRedirectCheck() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new CircularRedirectService()));
final HttpHost target = start();
final RequestConfig config = RequestConfig.custom()
.setCircularRedirectsAllowed(true)
.setMaxRedirects(5).build();
final HttpGet httpget = new HttpGet("/circular-oldlocation/");
httpget.setConfig(config);
try {
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, null);
future.get();
} catch (final ExecutionException e) {
Assert.assertTrue(e.getCause() instanceof RedirectException);
throw e;
}
}
@Test(expected=ExecutionException.class)
public void testCircularRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new CircularRedirectService()));
final HttpHost target = start();
final RequestConfig config = RequestConfig.custom()
.setCircularRedirectsAllowed(false)
.setRelativeRedirectsAllowed(true)
.build();
final HttpGet httpget = new HttpGet("/circular-oldlocation/");
httpget.setConfig(config);
try {
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, null);
future.get();
} catch (final ExecutionException e) {
Assert.assertTrue(e.getCause() instanceof CircularRedirectException);
throw e;
}
}
@Test
public void testPostNoRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MOVED_TEMPORARILY)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpPost httppost = new HttpPost("/oldlocation/");
httppost.setEntity(new NStringEntity("stuff"));
final Future<HttpResponse> future = this.httpclient.execute(target, httppost, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_MOVED_TEMPORARILY, response.getStatusLine().getStatusCode());
Assert.assertEquals("/oldlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals("POST", reqWrapper.getRequestLine().getMethod());
}
@Test
public void testPostRedirectSeeOther() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_SEE_OTHER)));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpPost httppost = new HttpPost("/oldlocation/");
httppost.setEntity(new NStringEntity("stuff"));
final Future<HttpResponse> future = this.httpclient.execute(target, httppost, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals("GET", reqWrapper.getRequestLine().getMethod());
}
@Test
public void testRelativeRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RelativeRedirectService()));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final RequestConfig config = RequestConfig.custom()
.setRelativeRedirectsAllowed(true)
.build();
final HttpGet httpget = new HttpGet("/oldlocation/");
httpget.setConfig(config);
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/relativelocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test
public void testRelativeRedirect2() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RelativeRedirectService2()));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final RequestConfig config = RequestConfig.custom()
.setRelativeRedirectsAllowed(true)
.build();
final HttpGet httpget = new HttpGet("/test/oldlocation");
httpget.setConfig(config);
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/test/relativelocation", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(target, host);
}
@Test(expected=ExecutionException.class)
public void testRejectRelativeRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RelativeRedirectService()));
final HttpHost target = start();
final RequestConfig config = RequestConfig.custom()
.setRelativeRedirectsAllowed(false)
.build();
final HttpGet httpget = new HttpGet("/oldlocation/");
httpget.setConfig(config);
try {
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, null);
future.get();
} catch (final ExecutionException e) {
Assert.assertTrue(e.getCause() instanceof ProtocolException);
throw e;
}
}
@Test(expected=ExecutionException.class)
public void testRejectBogusRedirectLocation() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BogusRedirectService(getSchemeName(), "xxx://bogus", true)));
final HttpHost target = start();
final HttpGet httpget = new HttpGet("/oldlocation/");
try {
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, null);
future.get();
} catch (final ExecutionException ex) {
Assert.assertTrue(ex.getCause() instanceof HttpException);
throw ex;
}
}
@Test(expected=ExecutionException.class)
public void testRejectInvalidRedirectLocation() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BogusRedirectService(getSchemeName(), "/newlocation/?p=I have spaces", false)));
final HttpHost target = start();
final HttpGet httpget = new HttpGet("/oldlocation/");
try {
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, null);
future.get();
} catch (final ExecutionException e) {
Assert.assertTrue(e.getCause() instanceof ProtocolException);
throw e;
}
}
@Test
public void testRedirectWithCookie() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MOVED_TEMPORARILY)));
final HttpHost target = start();
final CookieStore cookieStore = new BasicCookieStore();
final HttpClientContext context = HttpClientContext.create();
context.setCookieStore(cookieStore);
final BasicClientCookie cookie = new BasicClientCookie("name", "value");
cookie.setDomain(target.getHostName());
cookie.setPath("/");
cookieStore.addCookie(cookie);
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
final Header[] headers = reqWrapper.getHeaders(SM.COOKIE);
Assert.assertEquals("There can only be one (cookie)", 1, headers.length);
}
@Test
public void testDefaultHeadersRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(
new BasicRedirectService(getSchemeName(), HttpStatus.SC_MOVED_TEMPORARILY)));
final List<Header> defaultHeaders = new ArrayList<Header>(1);
defaultHeaders.add(new BasicHeader(HTTP.USER_AGENT, "my-test-client"));
this.clientBuilder.setDefaultHeaders(defaultHeaders);
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(target, httpget, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
final Header header = reqWrapper.getFirstHeader(HTTP.USER_AGENT);
Assert.assertEquals("my-test-client", header.getValue());
}
static class CrossSiteRedirectService implements HttpRequestHandler {
private final HttpHost host;
public CrossSiteRedirectService(final HttpHost host) {
super();
this.host = host;
}
@Override
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
final ProtocolVersion ver = request.getRequestLine().getProtocolVersion();
final String location;
try {
final URIBuilder uribuilder = new URIBuilder(request.getRequestLine().getUri());
uribuilder.setScheme(this.host.getSchemeName());
uribuilder.setHost(this.host.getHostName());
uribuilder.setPort(this.host.getPort());
uribuilder.setPath("/random/1024");
location = uribuilder.build().toASCIIString();
} catch (final URISyntaxException ex) {
throw new ProtocolException("Invalid request URI", ex);
}
response.setStatusLine(ver, HttpStatus.SC_TEMPORARY_REDIRECT);
response.addHeader(new BasicHeader("Location", location));
}
}
@Test
public void testCrossSiteRedirect() throws Exception {
this.serverBootstrap.registerHandler("/random/*", new BasicAsyncRequestHandler(
new RandomHandler()));
final HttpHost redirectTarget = start();
this.serverBootstrap.registerHandler("/redirect/*", new BasicAsyncRequestHandler(
new CrossSiteRedirectService(redirectTarget)));
final HttpServer secondServer = this.serverBootstrap.create();
try {
secondServer.start();
final ListenerEndpoint endpoint2 = secondServer.getEndpoint();
endpoint2.waitFor();
final InetSocketAddress address2 = (InetSocketAddress) endpoint2.getAddress();
final HttpHost initialTarget = new HttpHost("localhost", address2.getPort(), getSchemeName());
final Queue<Future<HttpResponse>> queue = new ConcurrentLinkedQueue<Future<HttpResponse>>();
for (int i = 0; i < 4; i++) {
final HttpClientContext context = HttpClientContext.create();
final HttpGet httpget = new HttpGet("/redirect/anywhere");
queue.add(this.httpclient.execute(initialTarget, httpget, context, null));
}
while (!queue.isEmpty()) {
final Future<HttpResponse> future = queue.remove();
final HttpResponse response = future.get();
Assert.assertNotNull(response);
Assert.assertEquals(200, response.getStatusLine().getStatusCode());
}
} finally {
this.server.shutdown(10, TimeUnit.SECONDS);
}
}
@Test
public void testRepeatRequest() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RomeRedirectService()));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final RequestConfig config = RequestConfig.custom().setRelativeRedirectsAllowed(true).build();
final HttpGet first = new HttpGet("/rome");
first.setConfig(config);
final Future<HttpResponse> future1 = this.httpclient.execute(target, first, context, null);
final HttpResponse response1 = future1.get();
Assert.assertNotNull(response1);
final HttpGet second = new HttpGet("/rome");
second.setConfig(config);
final Future<HttpResponse> future2 = this.httpclient.execute(target, second, context, null);
final HttpResponse response2 = future2.get();
Assert.assertNotNull(response2);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response2.getStatusLine().getStatusCode());
Assert.assertEquals("/rome", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(host, target);
}
@Test
public void testRepeatRequestRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RomeRedirectService()));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final RequestConfig config = RequestConfig.custom().setRelativeRedirectsAllowed(true).build();
final HttpGet first = new HttpGet("/lille");
first.setConfig(config);
final Future<HttpResponse> future1 = this.httpclient.execute(target, first, context, null);
final HttpResponse response1 = future1.get();
Assert.assertNotNull(response1);
final HttpGet second = new HttpGet("/lille");
second.setConfig(config);
final Future<HttpResponse> future2 = this.httpclient.execute(target, second, context, null);
final HttpResponse response2 = future2.get();
Assert.assertNotNull(response2);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response2.getStatusLine().getStatusCode());
Assert.assertEquals("/rome", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(host, target);
}
@Test
public void testDifferentRequestSameRedirect() throws Exception {
this.serverBootstrap.registerHandler("*", new BasicAsyncRequestHandler(new RomeRedirectService()));
final HttpHost target = start();
final HttpClientContext context = HttpClientContext.create();
final RequestConfig config = RequestConfig.custom().setRelativeRedirectsAllowed(true).build();
final HttpGet first = new HttpGet("/alian");
first.setConfig(config);
final Future<HttpResponse> future1 = this.httpclient.execute(target, first, context, null);
final HttpResponse response1 = future1.get();
Assert.assertNotNull(response1);
final HttpGet second = new HttpGet("/lille");
second.setConfig(config);
final Future<HttpResponse> future2 = this.httpclient.execute(target, second, context, null);
final HttpResponse response2 = future2.get();
Assert.assertNotNull(response2);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response2.getStatusLine().getStatusCode());
Assert.assertEquals("/rome", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(host, target);
}
@Test
public void testPostRedirectWithDifferentHost() throws Exception {
// do redirect for post requests
this.clientBuilder.setRedirectStrategy(new DefaultRedirectStrategy() {
@Override
public boolean isRedirected(final HttpRequest request, final HttpResponse response,
final HttpContext context)
throws ProtocolException {
// allow 307 redirect for all methods
return super.isRedirected(request, response, context)
|| response.getStatusLine().getStatusCode() == HttpStatus.SC_TEMPORARY_REDIRECT;
}
});
final DifferentHostRedirectService differentHostRequestHandler = new DifferentHostRedirectService(
getSchemeName(), HttpStatus.SC_TEMPORARY_REDIRECT);
this.serverBootstrap.registerHandler("*",
new BasicAsyncRequestHandler(differentHostRequestHandler));
final HttpHost originalHost = start(); // to start the original host and build the client
final HttpHost targetHost = startServer(); // to start the target host
differentHostRequestHandler.setTargetHostPort(targetHost.getPort());
final HttpClientContext context = HttpClientContext.create();
final HttpPost httpPost = new HttpPost("/oldlocation/");
final Future<HttpResponse> future = this.httpclient.execute(originalHost, httpPost, context, null);
final HttpResponse response = future.get();
Assert.assertNotNull(response);
final HttpRequest reqWrapper = context.getRequest();
final HttpHost host = context.getTargetHost();
Assert.assertEquals(HttpStatus.SC_OK, response.getStatusLine().getStatusCode());
Assert.assertEquals("/newlocation/", reqWrapper.getRequestLine().getUri());
Assert.assertEquals(targetHost, host);
}
}