/* * Copyright 2010 Ning, Inc. * * This program is licensed 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. */ package org.asynchttpclient; import static org.asynchttpclient.Dsl.*; import static org.asynchttpclient.util.DateUtils.unpreciseMillisTime; import static org.testng.Assert.*; import java.io.IOException; import java.util.concurrent.ExecutionException; import java.util.concurrent.Future; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.eclipse.jetty.continuation.Continuation; import org.eclipse.jetty.continuation.ContinuationSupport; import org.eclipse.jetty.server.Request; import org.eclipse.jetty.server.handler.AbstractHandler; import org.testng.annotations.Test; /** * Per request timeout configuration test. * * @author Hubert Iwaniuk */ public class PerRequestTimeoutTest extends AbstractBasicTest { private static final String MSG = "Enough is enough."; private void checkTimeoutMessage(String message, boolean requestTimeout) { if (requestTimeout) assertTrue(message.startsWith("Request timeout"), "error message indicates reason of error but got: " + message); else assertTrue(message.startsWith("Read timeout"), "error message indicates reason of error but got: " + message); assertTrue(message.contains("localhost"), "error message contains remote host address but got: " + message); assertTrue(message.contains("after 100 ms"), "error message contains timeout configuration value but got: " + message); } @Override public AbstractHandler configureHandler() throws Exception { return new SlowHandler(); } private class SlowHandler extends AbstractHandler { public void handle(String target, Request baseRequest, HttpServletRequest request, final HttpServletResponse response) throws IOException, ServletException { response.setStatus(HttpServletResponse.SC_OK); final Continuation continuation = ContinuationSupport.getContinuation(request); continuation.suspend(); new Thread(new Runnable() { public void run() { try { Thread.sleep(1500); response.getOutputStream().print(MSG); response.getOutputStream().flush(); } catch (InterruptedException e) { logger.error(e.getMessage(), e); } catch (IOException e) { logger.error(e.getMessage(), e); } } }).start(); new Thread(new Runnable() { public void run() { try { Thread.sleep(3000); response.getOutputStream().print(MSG); response.getOutputStream().flush(); continuation.complete(); } catch (InterruptedException e) { logger.error(e.getMessage(), e); } catch (IOException e) { logger.error(e.getMessage(), e); } } }).start(); baseRequest.setHandled(true); } } @Test(groups = "standalone") public void testRequestTimeout() throws IOException { try (AsyncHttpClient client = asyncHttpClient()) { Future<Response> responseFuture = client.prepareGet(getTargetUrl()).setRequestTimeout(100).execute(); Response response = responseFuture.get(2000, TimeUnit.MILLISECONDS); assertNull(response); } catch (InterruptedException e) { fail("Interrupted.", e); } catch (ExecutionException e) { assertTrue(e.getCause() instanceof TimeoutException); checkTimeoutMessage(e.getCause().getMessage(), true); } catch (TimeoutException e) { fail("Timeout.", e); } } @Test(groups = "standalone") public void testReadTimeout() throws IOException { try (AsyncHttpClient client = asyncHttpClient(config().setReadTimeout(100))) { Future<Response> responseFuture = client.prepareGet(getTargetUrl()).execute(); Response response = responseFuture.get(2000, TimeUnit.MILLISECONDS); assertNull(response); } catch (InterruptedException e) { fail("Interrupted.", e); } catch (ExecutionException e) { assertTrue(e.getCause() instanceof TimeoutException); checkTimeoutMessage(e.getCause().getMessage(), false); } catch (TimeoutException e) { fail("Timeout.", e); } } @Test(groups = "standalone") public void testGlobalDefaultPerRequestInfiniteTimeout() throws IOException { try (AsyncHttpClient client = asyncHttpClient(config().setRequestTimeout(100))) { Future<Response> responseFuture = client.prepareGet(getTargetUrl()).setRequestTimeout(-1).execute(); Response response = responseFuture.get(); assertNotNull(response); } catch (InterruptedException e) { fail("Interrupted.", e); } catch (ExecutionException e) { assertTrue(e.getCause() instanceof TimeoutException); checkTimeoutMessage(e.getCause().getMessage(), true); } } @Test(groups = "standalone") public void testGlobalRequestTimeout() throws IOException { try (AsyncHttpClient client = asyncHttpClient(config().setRequestTimeout(100))) { Future<Response> responseFuture = client.prepareGet(getTargetUrl()).execute(); Response response = responseFuture.get(2000, TimeUnit.MILLISECONDS); assertNull(response); } catch (InterruptedException e) { fail("Interrupted.", e); } catch (ExecutionException e) { assertTrue(e.getCause() instanceof TimeoutException); checkTimeoutMessage(e.getCause().getMessage(), true); } catch (TimeoutException e) { fail("Timeout.", e); } } @Test(groups = "standalone") public void testGlobalIdleTimeout() throws IOException { final long times[] = new long[] { -1, -1 }; try (AsyncHttpClient client = asyncHttpClient(config().setPooledConnectionIdleTimeout(2000))) { Future<Response> responseFuture = client.prepareGet(getTargetUrl()).execute(new AsyncCompletionHandler<Response>() { @Override public Response onCompleted(Response response) throws Exception { return response; } @Override public State onBodyPartReceived(HttpResponseBodyPart content) throws Exception { times[0] = unpreciseMillisTime(); return super.onBodyPartReceived(content); } @Override public void onThrowable(Throwable t) { times[1] = unpreciseMillisTime(); super.onThrowable(t); } }); Response response = responseFuture.get(); assertNotNull(response); assertEquals(response.getResponseBody(), MSG + MSG); } catch (InterruptedException e) { fail("Interrupted.", e); } catch (ExecutionException e) { logger.info(String.format("\n@%dms Last body part received\n@%dms Connection killed\n %dms difference.", times[0], times[1], (times[1] - times[0]))); fail("Timeouted on idle.", e); } } }