360665: Fixed an endless loop on https proxy requests + added tests

Signed-off-by: Simone Bordet <simone.bordet@gmail.com>
This commit is contained in:
Thomas Becker 2011-11-02 14:04:49 +01:00 committed by Simone Bordet
parent 5ca9dbb7d4
commit f01877e738
3 changed files with 164 additions and 11 deletions

View File

@ -65,7 +65,7 @@ public class HttpDestination implements Dumpable
private List<HttpCookie> _cookies; private List<HttpCookie> _cookies;
HttpDestination(HttpClient client, Address address, boolean ssl) HttpDestination(HttpClient client, Address address, boolean ssl)
{ {
_client = client; _client = client;
@ -524,7 +524,7 @@ public class HttpDestination implements Dumpable
// Add any known authorizations // Add any known authorizations
if (_authorizations != null) if (_authorizations != null)
{ {
Authentication auth = (Authentication)_authorizations.match(ex.getURI()); Authentication auth = (Authentication)_authorizations.match(ex.getRequestURI());
if (auth != null) if (auth != null)
(auth).setCredentials(ex); (auth).setCredentials(ex);
} }
@ -665,7 +665,7 @@ public class HttpDestination implements Dumpable
AggregateLifeCycle.dump(out,indent,_connections); AggregateLifeCycle.dump(out,indent,_connections);
} }
} }
private class ConnectExchange extends ContentExchange private class ConnectExchange extends ContentExchange
{ {
private final SelectConnector.ProxySelectChannelEndPoint proxyEndPoint; private final SelectConnector.ProxySelectChannelEndPoint proxyEndPoint;
@ -687,13 +687,17 @@ public class HttpDestination implements Dumpable
@Override @Override
protected void onResponseComplete() throws IOException protected void onResponseComplete() throws IOException
{ {
if (getResponseStatus() == HttpStatus.OK_200) int responseStatus = getResponseStatus();
if (responseStatus == HttpStatus.OK_200)
{ {
proxyEndPoint.upgrade(); proxyEndPoint.upgrade();
} }
else if(responseStatus == HttpStatus.GATEWAY_TIMEOUT_504){
onExpire();
}
else else
{ {
onConnectionFailed(new ConnectException(exchange.getAddress().toString())); onException(new ConnectException("Proxy: " + proxyEndPoint.getRemoteAddr() +":" + proxyEndPoint.getRemotePort() + " didn't return http return code 200, but " + responseStatus + " while trying to request: " + exchange.getAddress().toString()));
} }
} }
@ -702,5 +706,22 @@ public class HttpDestination implements Dumpable
{ {
HttpDestination.this.onConnectionFailed(x); HttpDestination.this.onConnectionFailed(x);
} }
@Override
protected void onException(Throwable x)
{
_queue.remove(exchange);
exchange.setStatus(STATUS_EXCEPTED);
exchange.getEventListener().onException(x);
}
@Override
protected void onExpire()
{
_queue.remove(exchange);
exchange.setStatus(STATUS_EXPIRED);
exchange.getEventListener().onExpire();
}
} }
} }

View File

@ -0,0 +1,131 @@
// ========================================================================
// Copyright (c) 2009-2009 Mort Bay Consulting Pty. Ltd.
// ------------------------------------------------------------------------
// All rights reserved. This program and the accompanying materials
// are made available under the terms of the Eclipse Public License v1.0
// and Apache License v2.0 which accompanies this distribution.
// The Eclipse Public License is available at
// http://www.eclipse.org/legal/epl-v10.html
// The Apache License v2.0 is available at
// http://www.opensource.org/licenses/apache2.0.php
// You may elect to redistribute this code under either of these licenses.
// ========================================================================
package org.eclipse.jetty.client;
import static org.junit.Assert.assertEquals;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.eclipse.jetty.http.HttpStatus;
import org.eclipse.jetty.server.HttpConnection;
import org.eclipse.jetty.server.Request;
import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.handler.ConnectHandler;
import org.eclipse.jetty.server.nio.SelectChannelConnector;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
/* ------------------------------------------------------------ */
/**
* This UnitTest class executes two tests. Both will send a http request to https://google.com through a misbehaving proxy server.
*
* The first test runs against a proxy which simply closes the connection (as nginx does) for a connect request. The second proxy server always responds with a
* 500 error.
*
* The expected result for both tests is an exception and the HttpExchange should have status HttpExchange.STATUS_EXCEPTED.
*/
public class HttpsViaBrokenHttpProxyTest
{
private Server _proxy = new Server();
private HttpClient _client = new HttpClient();
/* ------------------------------------------------------------ */
/**
* @throws java.lang.Exception
*/
@Before
public void setUpBeforeClass() throws Exception
{
// setup proxies with different behaviour
_proxy.addConnector(new SelectChannelConnector());
_proxy.setHandler(new BadBehavingConnectHandler());
_proxy.start();
int proxyClosingConnectionPort = _proxy.getConnectors()[0].getLocalPort();
_client.setProxy(new Address("localhost",proxyClosingConnectionPort));
_client.start();
}
/* ------------------------------------------------------------ */
/**
* @throws java.lang.Exception
*/
@After
public void tearDownAfterClass() throws Exception
{
_client.stop();
_proxy.stop();
}
private class BadBehavingConnectHandler extends ConnectHandler
{
@Override
protected void handleConnect(Request baseRequest, HttpServletRequest request, HttpServletResponse response, String serverAddress)
throws ServletException, IOException
{
if (serverAddress.contains("close"))
HttpConnection.getCurrentConnection().getEndPoint().close();
else if (serverAddress.contains("error500"))
{
response.setStatus(HttpStatus.INTERNAL_SERVER_ERROR_500);
}
else if (serverAddress.contains("error504"))
{
response.setStatus(HttpStatus.GATEWAY_TIMEOUT_504);
}
baseRequest.setHandled(true);
}
}
@Test
public void httpsViaProxyThatClosesConnectionOnConnectRequestTest()
{
sendRequestThroughProxy("close",9);
}
@Test
public void httpsViaProxyThatReturns500ErrorTest() throws Exception
{
sendRequestThroughProxy("error500",9);
}
@Test
public void httpsViaProxyThatReturns504ErrorTest() throws Exception
{
sendRequestThroughProxy("error504",8);
}
private void sendRequestThroughProxy(String desiredBehaviour, int exptectedStatus)
{
String url = "https://" + desiredBehaviour + ".com/";
try
{
ContentExchange exchange = new ContentExchange();
exchange.setURL(url);
exchange.addRequestHeader("behaviour",desiredBehaviour);
_client.send(exchange);
assertEquals(HttpExchange.toState(exptectedStatus) + " status awaited",exptectedStatus,exchange.waitForDone());
}
catch (Exception e)
{
System.out.println(e.getMessage());
}
}
}

View File

@ -1,9 +1,13 @@
package org.eclipse.jetty.client; package org.eclipse.jetty.client;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
import java.io.IOException; import java.io.IOException;
import java.net.URLEncoder; import java.net.URLEncoder;
import java.util.concurrent.CountDownLatch; import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import javax.servlet.ServletException; import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream; import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletRequest;
@ -26,9 +30,6 @@ import org.eclipse.jetty.toolchain.test.MavenTestingUtils;
import org.junit.After; import org.junit.After;
import org.junit.Test; import org.junit.Test;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertTrue;
public class ProxyTunnellingTest public class ProxyTunnellingTest
{ {
private Server server; private Server server;
@ -41,7 +42,7 @@ public class ProxyTunnellingTest
{ {
return proxyConnector.getLocalPort(); return proxyConnector.getLocalPort();
} }
protected void startSSLServer(Handler handler) throws Exception protected void startSSLServer(Handler handler) throws Exception
{ {
SslSelectChannelConnector connector = new SslSelectChannelConnector(); SslSelectChannelConnector connector = new SslSelectChannelConnector();
@ -217,11 +218,11 @@ public class ProxyTunnellingTest
ContentExchange exchange = new ContentExchange(true) ContentExchange exchange = new ContentExchange(true)
{ {
@Override @Override
protected void onConnectionFailed(Throwable x) protected void onException(Throwable x)
{ {
latch.countDown(); latch.countDown();
} }
}; };
exchange.setMethod(HttpMethods.GET); exchange.setMethod(HttpMethods.GET);
String body = "BODY"; String body = "BODY";