Removed extra dispatches from HTTP client code.
This commit is contained in:
parent
d4dfc0762f
commit
71f6527cae
|
@ -28,6 +28,7 @@ import java.util.concurrent.atomic.AtomicInteger;
|
|||
import org.eclipse.jetty.client.api.Connection;
|
||||
import org.eclipse.jetty.client.api.Destination;
|
||||
import org.eclipse.jetty.util.BlockingArrayQueue;
|
||||
import org.eclipse.jetty.util.Callback;
|
||||
import org.eclipse.jetty.util.Promise;
|
||||
import org.eclipse.jetty.util.component.ContainerLifeCycle;
|
||||
import org.eclipse.jetty.util.component.Dumpable;
|
||||
|
@ -41,15 +42,15 @@ public class ConnectionPool implements Closeable, Dumpable
|
|||
private final AtomicInteger connectionCount = new AtomicInteger();
|
||||
private final Destination destination;
|
||||
private final int maxConnections;
|
||||
private final Promise<Connection> connectionPromise;
|
||||
private final Callback requester;
|
||||
private final BlockingDeque<Connection> idleConnections;
|
||||
private final BlockingQueue<Connection> activeConnections;
|
||||
|
||||
public ConnectionPool(Destination destination, int maxConnections, Promise<Connection> connectionPromise)
|
||||
public ConnectionPool(Destination destination, int maxConnections, Callback requester)
|
||||
{
|
||||
this.destination = destination;
|
||||
this.maxConnections = maxConnections;
|
||||
this.connectionPromise = connectionPromise;
|
||||
this.requester = requester;
|
||||
this.idleConnections = new LinkedBlockingDeque<>(maxConnections);
|
||||
this.activeConnections = new BlockingArrayQueue<>(maxConnections);
|
||||
}
|
||||
|
@ -104,10 +105,8 @@ public class ConnectionPool implements Closeable, Dumpable
|
|||
{
|
||||
if (LOG.isDebugEnabled())
|
||||
LOG.debug("Connection {}/{} creation succeeded {}", next, maxConnections, connection);
|
||||
if (activate(connection))
|
||||
connectionPromise.succeeded(connection);
|
||||
else
|
||||
connectionPromise.failed(new IllegalStateException("Active connection overflow"));
|
||||
idle(connection, true);
|
||||
requester.succeeded();
|
||||
}
|
||||
|
||||
@Override
|
||||
|
@ -116,7 +115,7 @@ public class ConnectionPool implements Closeable, Dumpable
|
|||
if (LOG.isDebugEnabled())
|
||||
LOG.debug("Connection " + next + "/" + maxConnections + " creation failed", x);
|
||||
connectionCount.decrementAndGet();
|
||||
connectionPromise.failed(x);
|
||||
requester.failed(x);
|
||||
}
|
||||
});
|
||||
|
||||
|
@ -160,9 +159,16 @@ public class ConnectionPool implements Closeable, Dumpable
|
|||
{
|
||||
released(connection);
|
||||
if (activeConnections.remove(connection))
|
||||
return idle(connection, false);
|
||||
return false;
|
||||
}
|
||||
|
||||
protected boolean idle(Connection connection, boolean created)
|
||||
{
|
||||
// Make sure we use "hot" connections first
|
||||
if (idleConnections.offerFirst(connection))
|
||||
// Make sure we use "hot" connections first.
|
||||
boolean idle = created ? idleConnections.offerLast(connection)
|
||||
: idleConnections.offerFirst(connection);
|
||||
if (idle)
|
||||
{
|
||||
if (LOG.isDebugEnabled())
|
||||
LOG.debug("Connection idle {}", connection);
|
||||
|
@ -173,10 +179,9 @@ public class ConnectionPool implements Closeable, Dumpable
|
|||
if (LOG.isDebugEnabled())
|
||||
LOG.debug("Connection idle overflow {}", connection);
|
||||
connection.close();
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
protected void released(Connection connection)
|
||||
{
|
||||
|
|
|
@ -205,7 +205,7 @@ public abstract class HttpDestination implements Destination, Closeable, Dumpabl
|
|||
return queue.offer(exchange);
|
||||
}
|
||||
|
||||
protected abstract void send();
|
||||
public abstract void send();
|
||||
|
||||
public void newConnection(Promise<Connection> promise)
|
||||
{
|
||||
|
|
|
@ -20,8 +20,8 @@ package org.eclipse.jetty.client;
|
|||
|
||||
import org.eclipse.jetty.client.api.Connection;
|
||||
import org.eclipse.jetty.client.api.Destination;
|
||||
import org.eclipse.jetty.util.Callback;
|
||||
import org.eclipse.jetty.util.LeakDetector;
|
||||
import org.eclipse.jetty.util.Promise;
|
||||
|
||||
public class LeakTrackingConnectionPool extends ConnectionPool
|
||||
{
|
||||
|
@ -34,9 +34,9 @@ public class LeakTrackingConnectionPool extends ConnectionPool
|
|||
}
|
||||
};
|
||||
|
||||
public LeakTrackingConnectionPool(Destination destination, int maxConnections, Promise<Connection> connectionPromise)
|
||||
public LeakTrackingConnectionPool(Destination destination, int maxConnections, Callback requester)
|
||||
{
|
||||
super(destination, maxConnections, connectionPromise);
|
||||
super(destination, maxConnections, requester);
|
||||
start();
|
||||
}
|
||||
|
||||
|
|
|
@ -35,7 +35,7 @@ public abstract class MultiplexHttpDestination<C extends Connection> extends Htt
|
|||
}
|
||||
|
||||
@Override
|
||||
protected void send()
|
||||
public void send()
|
||||
{
|
||||
while (true)
|
||||
{
|
||||
|
@ -56,7 +56,7 @@ public abstract class MultiplexHttpDestination<C extends Connection> extends Htt
|
|||
}
|
||||
case CONNECTED:
|
||||
{
|
||||
if (process(connection, false))
|
||||
if (process(connection))
|
||||
break;
|
||||
return;
|
||||
}
|
||||
|
@ -75,7 +75,7 @@ public abstract class MultiplexHttpDestination<C extends Connection> extends Htt
|
|||
C connection = this.connection = (C)result;
|
||||
if (connect.compareAndSet(ConnectState.CONNECTING, ConnectState.CONNECTED))
|
||||
{
|
||||
process(connection, true);
|
||||
process(connection);
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -91,7 +91,7 @@ public abstract class MultiplexHttpDestination<C extends Connection> extends Htt
|
|||
abort(x);
|
||||
}
|
||||
|
||||
protected boolean process(final C connection, boolean dispatch)
|
||||
protected boolean process(final C connection)
|
||||
{
|
||||
HttpClient client = getHttpClient();
|
||||
final HttpExchange exchange = getHttpExchanges().poll();
|
||||
|
@ -112,23 +112,9 @@ public abstract class MultiplexHttpDestination<C extends Connection> extends Htt
|
|||
exchange.abort(cause);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (dispatch)
|
||||
{
|
||||
client.getExecutor().execute(new Runnable()
|
||||
{
|
||||
@Override
|
||||
public void run()
|
||||
{
|
||||
send(connection, exchange);
|
||||
}
|
||||
});
|
||||
}
|
||||
else
|
||||
{
|
||||
send(connection, exchange);
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
|
|
|
@ -23,10 +23,10 @@ import java.util.Arrays;
|
|||
|
||||
import org.eclipse.jetty.client.api.Connection;
|
||||
import org.eclipse.jetty.client.api.Request;
|
||||
import org.eclipse.jetty.util.Promise;
|
||||
import org.eclipse.jetty.util.Callback;
|
||||
import org.eclipse.jetty.util.component.ContainerLifeCycle;
|
||||
|
||||
public abstract class PoolingHttpDestination<C extends Connection> extends HttpDestination implements Promise<Connection>
|
||||
public abstract class PoolingHttpDestination<C extends Connection> extends HttpDestination implements Callback
|
||||
{
|
||||
private final ConnectionPool connectionPool;
|
||||
|
||||
|
@ -47,30 +47,23 @@ public abstract class PoolingHttpDestination<C extends Connection> extends HttpD
|
|||
}
|
||||
|
||||
@Override
|
||||
@SuppressWarnings("unchecked")
|
||||
public void succeeded(Connection connection)
|
||||
public void succeeded()
|
||||
{
|
||||
process((C)connection, true);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void failed(final Throwable x)
|
||||
{
|
||||
getHttpClient().getExecutor().execute(new Runnable()
|
||||
{
|
||||
@Override
|
||||
public void run()
|
||||
{
|
||||
abort(x);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected void send()
|
||||
public void send()
|
||||
{
|
||||
if (getHttpExchanges().isEmpty())
|
||||
return;
|
||||
C connection = acquire();
|
||||
if (connection != null)
|
||||
process(connection, false);
|
||||
process(connection);
|
||||
}
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
|
@ -87,9 +80,8 @@ public abstract class PoolingHttpDestination<C extends Connection> extends HttpD
|
|||
* <p>If a request is waiting to be executed, it will be dequeued and executed by the new connection.</p>
|
||||
*
|
||||
* @param connection the new connection
|
||||
* @param dispatch whether to dispatch the processing to another thread
|
||||
*/
|
||||
public void process(final C connection, boolean dispatch)
|
||||
public void process(final C connection)
|
||||
{
|
||||
HttpClient client = getHttpClient();
|
||||
final HttpExchange exchange = getHttpExchanges().poll();
|
||||
|
@ -121,23 +113,9 @@ public abstract class PoolingHttpDestination<C extends Connection> extends HttpD
|
|||
exchange.abort(cause);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (dispatch)
|
||||
{
|
||||
client.getExecutor().execute(new Runnable()
|
||||
{
|
||||
@Override
|
||||
public void run()
|
||||
{
|
||||
send(connection, exchange);
|
||||
}
|
||||
});
|
||||
}
|
||||
else
|
||||
{
|
||||
send(connection, exchange);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -155,7 +133,7 @@ public abstract class PoolingHttpDestination<C extends Connection> extends HttpD
|
|||
{
|
||||
if (connectionPool.isActive(connection))
|
||||
{
|
||||
process(connection, false);
|
||||
process(connection);
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -198,7 +176,7 @@ public abstract class PoolingHttpDestination<C extends Connection> extends HttpD
|
|||
// idle timeout, so no worries.
|
||||
C newConnection = acquire();
|
||||
if (newConnection != null)
|
||||
process(newConnection, false);
|
||||
process(newConnection);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -18,7 +18,6 @@
|
|||
|
||||
package org.eclipse.jetty.client.http;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.nio.channels.AsynchronousCloseException;
|
||||
import java.util.concurrent.TimeoutException;
|
||||
import java.util.concurrent.atomic.AtomicBoolean;
|
||||
|
@ -76,6 +75,7 @@ public class HttpConnectionOverHTTP extends AbstractConnection implements Connec
|
|||
{
|
||||
super.onOpen();
|
||||
fillInterested();
|
||||
getHttpDestination().send();
|
||||
}
|
||||
|
||||
public boolean isClosed()
|
||||
|
|
|
@ -87,16 +87,11 @@ public class HttpRequestAbortTest extends AbstractHttpClientServerTest
|
|||
Assert.assertFalse(begin.get());
|
||||
}
|
||||
|
||||
// The request send triggered a connection creation
|
||||
// that is not awaited before failing the exchange.
|
||||
Thread.sleep(1000);
|
||||
|
||||
// However, the connection has not been used, so it's a good one.
|
||||
HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort());
|
||||
ConnectionPool connectionPool = destination.getConnectionPool();
|
||||
Assert.assertEquals(1, connectionPool.getConnectionCount());
|
||||
Assert.assertEquals(0, connectionPool.getConnectionCount());
|
||||
Assert.assertEquals(0, connectionPool.getActiveConnections().size());
|
||||
Assert.assertEquals(1, connectionPool.getIdleConnections().size());
|
||||
Assert.assertEquals(0, connectionPool.getIdleConnections().size());
|
||||
}
|
||||
|
||||
@Test
|
||||
|
|
|
@ -94,12 +94,12 @@ public class HttpDestinationOverHTTPTest extends AbstractHttpClientServerTest
|
|||
HttpDestinationOverHTTP destination = new HttpDestinationOverHTTP(client, new Origin("http", "localhost", connector.getLocalPort()))
|
||||
{
|
||||
@Override
|
||||
public void process(HttpConnectionOverHTTP connection, boolean dispatch)
|
||||
public void process(HttpConnectionOverHTTP connection)
|
||||
{
|
||||
try
|
||||
{
|
||||
latch.await(5, TimeUnit.SECONDS);
|
||||
super.process(connection, dispatch);
|
||||
super.process(connection);
|
||||
}
|
||||
catch (InterruptedException x)
|
||||
{
|
||||
|
@ -142,7 +142,7 @@ public class HttpDestinationOverHTTPTest extends AbstractHttpClientServerTest
|
|||
// Acquire the connection to make it active
|
||||
Assert.assertSame(connection1, destination.acquire());
|
||||
|
||||
destination.process(connection1, false);
|
||||
destination.process(connection1);
|
||||
destination.release(connection1);
|
||||
|
||||
Connection connection2 = destination.acquire();
|
||||
|
|
Loading…
Reference in New Issue