HTTPCLIENT-1284: HttpClient incorrectly generates Host header when physical connection route differs from the host name specified in the request URI

git-svn-id: https://svn.apache.org/repos/asf/httpcomponents/httpclient/trunk@1424448 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Oleg Kalnichevski 2012-12-20 13:05:41 +00:00
parent f492a000e7
commit e798d4947c
4 changed files with 198 additions and 17 deletions

View File

@ -1,6 +1,10 @@
Changes in trunk
-------------------
* [HTTPCLIENT-1284] HttpClient incorrectly generates Host header when physical connection
route differs from the host name specified in the request URI.
Contributed by Oleg Kalnichevski <olegk at apache.org>
* Kerberos and SPNego auth schemes use incorrect authorization header name when authenticating
with a proxy.
Contributed by Oleg Kalnichevski <olegk at apache.org>

View File

@ -456,19 +456,24 @@ public class DefaultRequestDirector implements RequestDirector {
new BasicScheme(), new UsernamePasswordCredentials(userinfo));
}
// Reset headers on the request wrapper
wrapper.resetHeaders();
// Re-write request URI if needed
rewriteRequestURI(wrapper, route);
// Use virtual host if set
target = virtualHost;
if (virtualHost != null) {
target = virtualHost;
} else {
URI requestURI = wrapper.getURI();
if (requestURI.isAbsolute()) {
target = new HttpHost(
requestURI.getHost(), requestURI.getPort(), requestURI.getScheme());
}
}
if (target == null) {
target = route.getTargetHost();
}
// Reset headers on the request wrapper
wrapper.resetHeaders();
// Re-write request URI if needed
rewriteRequestURI(wrapper, route);
// Populate the execution context
context.setAttribute(ExecutionContext.HTTP_TARGET_HOST, target);
context.setAttribute(ClientContext.ROUTE, route);

View File

@ -35,6 +35,7 @@ import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.http.HttpException;
import org.apache.http.HttpHost;
import org.apache.http.HttpRequest;
import org.apache.http.ProtocolException;
import org.apache.http.annotation.Immutable;
import org.apache.http.auth.AuthState;
@ -42,6 +43,7 @@ import org.apache.http.auth.UsernamePasswordCredentials;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpExecutionAware;
import org.apache.http.client.methods.HttpRequestWrapper;
import org.apache.http.client.methods.HttpUriRequest;
import org.apache.http.client.params.ClientPNames;
import org.apache.http.client.protocol.ClientContext;
import org.apache.http.client.protocol.HttpClientContext;
@ -113,15 +115,16 @@ public class ProtocolExec implements ClientExecChain {
Args.notNull(request, "HTTP request");
Args.notNull(context, "HTTP context");
HttpHost target = route.getTargetHost();
// Get user info from the URI
AuthState targetAuthState = context.getTargetAuthState();
if (targetAuthState != null) {
String userinfo = request.getURI().getUserInfo();
if (userinfo != null) {
targetAuthState.update(
new BasicScheme(), new UsernamePasswordCredentials(userinfo));
URI uri = request.getURI();
if (uri != null) {
String userinfo = uri.getUserInfo();
if (userinfo != null) {
targetAuthState.update(
new BasicScheme(), new UsernamePasswordCredentials(userinfo));
}
}
}
@ -132,7 +135,7 @@ public class ProtocolExec implements ClientExecChain {
HttpHost virtualHost = (HttpHost) params.getParameter(ClientPNames.VIRTUAL_HOST);
// HTTPCLIENT-1092 - add the port if necessary
if (virtualHost != null && virtualHost.getPort() == -1) {
int port = target.getPort();
int port = route.getTargetHost().getPort();
if (port != -1){
virtualHost = new HttpHost(virtualHost.getHostName(), port, virtualHost.getSchemeName());
}
@ -141,8 +144,24 @@ public class ProtocolExec implements ClientExecChain {
}
}
HttpHost target = null;
if (virtualHost != null) {
target = virtualHost;
} else {
HttpRequest original = request.getOriginal();
if (original instanceof HttpUriRequest) {
URI uri = ((HttpUriRequest) original).getURI();
if (uri.isAbsolute()) {
target = new HttpHost(uri.getHost(), uri.getPort(), uri.getScheme());
}
}
}
if (target == null) {
target = route.getTargetHost();
}
// Run request protocol interceptors
context.setAttribute(ExecutionContext.HTTP_TARGET_HOST, virtualHost != null ? virtualHost : target);
context.setAttribute(ExecutionContext.HTTP_TARGET_HOST, target);
context.setAttribute(ClientContext.ROUTE, route);
context.setAttribute(ExecutionContext.HTTP_REQUEST, request);

View File

@ -0,0 +1,153 @@
/*
* ====================================================================
*
* 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.impl.client.integration;
import java.io.IOException;
import java.net.URI;
import java.util.List;
import org.apache.http.HttpEntity;
import org.apache.http.HttpException;
import org.apache.http.HttpRequest;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.HttpVersion;
import org.apache.http.client.CookieStore;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.protocol.ClientContext;
import org.apache.http.cookie.Cookie;
import org.apache.http.impl.client.BasicCookieStore;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.localserver.LocalTestServer;
import org.apache.http.message.BasicHeader;
import org.apache.http.protocol.BasicHttpContext;
import org.apache.http.protocol.HttpContext;
import org.apache.http.protocol.HttpRequestHandler;
import org.apache.http.util.EntityUtils;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;
/**
* This class tests cookie matching when using Virtual Host.
*/
public class TestCookieVirtualHost extends IntegrationTestBase {
@Before
public void setUp() throws Exception {
this.localServer = new LocalTestServer(null, null);
this.localServer.registerDefaultHandlers();
this.localServer.start();
}
@Test
public void testCookieMatchingWithVirtualHosts() throws Exception {
this.localServer.register("*", new HttpRequestHandler() {
public void handle(
final HttpRequest request,
final HttpResponse response,
final HttpContext context) throws HttpException, IOException {
int n = Integer.parseInt(request.getFirstHeader("X-Request").getValue());
switch (n) {
case 1:
// Assert Host is forwarded from URI
Assert.assertEquals("app.mydomain.fr", request
.getFirstHeader("Host").getValue());
response.setStatusLine(HttpVersion.HTTP_1_1,
HttpStatus.SC_OK);
// Respond with Set-Cookie on virtual host domain. This
// should be valid.
response.addHeader(new BasicHeader("Set-Cookie",
"name1=value1; domain=mydomain.fr; path=/"));
break;
case 2:
// Assert Host is still forwarded from URI
Assert.assertEquals("app.mydomain.fr", request
.getFirstHeader("Host").getValue());
// We should get our cookie back.
Assert.assertNotNull("We must get a cookie header",
request.getFirstHeader("Cookie"));
response.setStatusLine(HttpVersion.HTTP_1_1,
HttpStatus.SC_OK);
break;
case 3:
// Assert Host is forwarded from URI
Assert.assertEquals("app.mydomain.fr", request
.getFirstHeader("Host").getValue());
response.setStatusLine(HttpVersion.HTTP_1_1,
HttpStatus.SC_OK);
break;
}
}
});
this.httpclient = HttpClients.createDefault();
CookieStore cookieStore = new BasicCookieStore();
HttpContext context = new BasicHttpContext();
context.setAttribute(ClientContext.COOKIE_STORE, cookieStore);
// First request : retrieve a domain cookie from remote server.
URI uri = new URI("http://app.mydomain.fr");
HttpRequest httpRequest = new HttpGet(uri);
httpRequest.addHeader("X-Request", "1");
HttpResponse response1 = this.httpclient.execute(getServerHttp(),
httpRequest, context);
HttpEntity e1 = response1.getEntity();
EntityUtils.consume(e1);
// We should have one cookie set on domain.
List<Cookie> cookies = cookieStore.getCookies();
Assert.assertNotNull(cookies);
Assert.assertEquals(1, cookies.size());
Assert.assertEquals("name1", cookies.get(0).getName());
// Second request : send the cookie back.
uri = new URI("http://app.mydomain.fr");
httpRequest = new HttpGet(uri);
httpRequest.addHeader("X-Request", "2");
HttpResponse response2 = this.httpclient.execute(getServerHttp(),
httpRequest, context);
HttpEntity e2 = response2.getEntity();
EntityUtils.consume(e2);
// Third request : Host header
uri = new URI("http://app.mydomain.fr");
httpRequest = new HttpGet(uri);
httpRequest.addHeader("X-Request", "3");
HttpResponse response3 = this.httpclient.execute(getServerHttp(),
httpRequest, context);
HttpEntity e3 = response3.getEntity();
EntityUtils.consume(e3);
}
}