// // ======================================================================== // Copyright (c) 1995-2017 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 java.io.IOException; import java.nio.ByteBuffer; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicBoolean; import java.util.concurrent.atomic.AtomicReference; import javax.servlet.DispatcherType; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.eclipse.jetty.client.api.Request; import org.eclipse.jetty.client.api.Result; import org.eclipse.jetty.client.http.HttpDestinationOverHTTP; import org.eclipse.jetty.client.util.ByteBufferContentProvider; import org.eclipse.jetty.server.handler.AbstractHandler; import org.eclipse.jetty.util.IO; import org.eclipse.jetty.util.log.StacklessLogging; import org.eclipse.jetty.util.ssl.SslContextFactory; import org.junit.Assert; import org.junit.Test; public class HttpRequestAbortTest extends AbstractHttpClientServerTest { public HttpRequestAbortTest(SslContextFactory sslContextFactory) { super(sslContextFactory); } @Test public void testAbortBeforeQueued() throws Exception { start(new EmptyServerHandler()); Exception failure = new Exception("oops"); try { Request request = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .timeout(5, TimeUnit.SECONDS); request.abort(failure); request.send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertSame(failure, x.getCause()); // Make sure the pool is in a sane state. HttpDestination destination = (HttpDestination)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(1, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(1, connectionPool.getIdleConnections().size()); } } @Test public void testAbortOnQueued() throws Exception { start(new EmptyServerHandler()); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); final AtomicBoolean begin = new AtomicBoolean(); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .listener(new Request.Listener.Adapter() { @Override public void onQueued(Request request) { aborted.set(request.abort(cause)); latch.countDown(); } @Override public void onBegin(Request request) { begin.set(true); } }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); Assert.assertFalse(begin.get()); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortOnBegin() throws Exception { start(new EmptyServerHandler()); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); final CountDownLatch committed = new CountDownLatch(1); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .listener(new Request.Listener.Adapter() { @Override public void onBegin(Request request) { aborted.set(request.abort(cause)); latch.countDown(); } @Override public void onCommit(Request request) { committed.countDown(); } }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); Assert.assertFalse(committed.await(1, TimeUnit.SECONDS)); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortOnHeaders() throws Exception { start(new EmptyServerHandler()); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); final CountDownLatch committed = new CountDownLatch(1); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .listener(new Request.Listener.Adapter() { @Override public void onHeaders(Request request) { aborted.set(request.abort(cause)); latch.countDown(); } @Override public void onCommit(Request request) { committed.countDown(); } }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); Assert.assertFalse(committed.await(1, TimeUnit.SECONDS)); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortOnCommit() throws Exception { start(new EmptyServerHandler()); // Test can behave in 2 ways: // A) the request is failed before the response arrived // B) the request is failed after the response arrived final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .onRequestCommit(request -> { aborted.set(request.abort(cause)); latch.countDown(); }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortOnCommitWithContent() throws Exception { final AtomicReference<IOException> failure = new AtomicReference<>(); start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { try { baseRequest.setHandled(true); if (request.getDispatcherType() != DispatcherType.ERROR) IO.copy(request.getInputStream(), response.getOutputStream()); } catch (IOException x) { failure.set(x); throw x; } } }); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .onRequestCommit(request -> { aborted.set(request.abort(cause)); latch.countDown(); }) .content(new ByteBufferContentProvider(ByteBuffer.wrap(new byte[]{0}), ByteBuffer.wrap(new byte[]{1})) { @Override public long getLength() { return -1; } }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortOnContent() throws Exception { try (StacklessLogging suppressor = new StacklessLogging(org.eclipse.jetty.server.HttpChannel.class)) { CountDownLatch serverLatch = new CountDownLatch(1); start(new EmptyServerHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { try { super.handle(target, baseRequest, request, response); if (request.getDispatcherType() != DispatcherType.ERROR) IO.copy(request.getInputStream(), response.getOutputStream()); } finally { serverLatch.countDown(); } } }); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .onRequestContent((request, content) -> { aborted.set(request.abort(cause)); latch.countDown(); }) .content(new ByteBufferContentProvider(ByteBuffer.wrap(new byte[]{0}), ByteBuffer.wrap(new byte[]{1})) { @Override public long getLength() { return -1; } }) .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); } Assert.assertTrue(serverLatch.await(5, TimeUnit.SECONDS)); HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } } @Test(expected = InterruptedException.class) public void testInterrupt() throws Exception { final long delay = 1000; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { try { baseRequest.setHandled(true); if (request.getDispatcherType() != DispatcherType.ERROR) TimeUnit.MILLISECONDS.sleep(2 * delay); } catch (InterruptedException x) { throw new ServletException(x); } } }); Request request = client.newRequest("localhost", connector.getLocalPort()) .timeout(3 * delay, TimeUnit.MILLISECONDS) .scheme(scheme); final Thread thread = Thread.currentThread(); new Thread(() -> { try { TimeUnit.MILLISECONDS.sleep(delay); thread.interrupt(); } catch (InterruptedException x) { throw new RuntimeException(x); } }).start(); request.send(); } @Test public void testAbortLongPoll() throws Exception { final long delay = 1000; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { try { baseRequest.setHandled(true); if (request.getDispatcherType() != DispatcherType.ERROR) TimeUnit.MILLISECONDS.sleep(2 * delay); } catch (InterruptedException x) { throw new ServletException(x); } } }); final Request request = client.newRequest("localhost", connector.getLocalPort()) .timeout(3 * delay, TimeUnit.MILLISECONDS) .scheme(scheme); final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); new Thread(() -> { try { TimeUnit.MILLISECONDS.sleep(delay); aborted.set(request.abort(cause)); latch.countDown(); } catch (InterruptedException x) { throw new RuntimeException(x); } }).start(); try { request.send(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); } HttpDestinationOverHTTP destination = (HttpDestinationOverHTTP)client.getDestination(scheme, "localhost", connector.getLocalPort()); DuplexConnectionPool connectionPool = (DuplexConnectionPool)destination.getConnectionPool(); Assert.assertEquals(0, connectionPool.getConnectionCount()); Assert.assertEquals(0, connectionPool.getActiveConnections().size()); Assert.assertEquals(0, connectionPool.getIdleConnections().size()); } @Test public void testAbortLongPollAsync() throws Exception { final long delay = 1000; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { try { baseRequest.setHandled(true); if (request.getDispatcherType() != DispatcherType.ERROR) TimeUnit.MILLISECONDS.sleep(2 * delay); } catch (InterruptedException x) { throw new ServletException(x); } } }); final Throwable cause = new Exception(); final CountDownLatch latch = new CountDownLatch(1); Request request = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .timeout(3 * delay, TimeUnit.MILLISECONDS); request.send(result -> { Assert.assertTrue(result.isFailed()); Assert.assertSame(cause, result.getFailure()); latch.countDown(); }); TimeUnit.MILLISECONDS.sleep(delay); request.abort(cause); Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); } @Test public void testAbortConversation() throws Exception { start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { baseRequest.setHandled(true); if (!"/done".equals(request.getRequestURI())) response.sendRedirect("/done"); } }); // The test may fail to abort the request in this way: // T1 aborts the request, which aborts the sender, which shuts down the output; // server reads -1 and closes; T2 reads -1 and the receiver fails the response with an EOFException; // T1 tries to abort the receiver, but it's already failed. final Throwable cause = new Exception(); final AtomicBoolean aborted = new AtomicBoolean(); final CountDownLatch latch = new CountDownLatch(1); client.getProtocolHandlers().clear(); client.getProtocolHandlers().put(new RedirectProtocolHandler(client) { @Override public void onComplete(Result result) { // Abort the request after the 3xx response but before issuing the next request if (!result.isFailed()) { aborted.set(result.getRequest().abort(cause)); latch.countDown(); } super.onComplete(result); } }); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .path("/redirect") .timeout(5, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); if (aborted.get()) Assert.assertSame(cause, x.getCause()); } } }