/* * Copyright (c) 2010-2012 Sonatype, Inc. All rights reserved. * * This program is licensed to you under the Apache License Version 2.0, * and you may not use this file except in compliance with the Apache License Version 2.0. * You may obtain a copy of the Apache License Version 2.0 at http://www.apache.org/licenses/LICENSE-2.0. * * Unless required by applicable law or agreed to in writing, * software distributed under the Apache License Version 2.0 is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the Apache License Version 2.0 for the specific language governing permissions and limitations there under. */ package org.asynchttpclient.netty; import static org.asynchttpclient.Dsl.*; import static org.testng.Assert.*; import java.io.IOException; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.concurrent.CountDownLatch; import java.util.concurrent.Future; import java.util.concurrent.Semaphore; import java.util.concurrent.TimeUnit; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.asynchttpclient.AbstractBasicTest; import org.asynchttpclient.AsyncCompletionHandler; import org.asynchttpclient.AsyncHttpClient; import org.asynchttpclient.Response; 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; public class NettyRequestThrottleTimeoutTest extends AbstractBasicTest { private static final String MSG = "Enough is enough."; private static final int SLEEPTIME_MS = 1000; @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(SLEEPTIME_MS); 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 { final Semaphore requestThrottle = new Semaphore(1); int samples = 10; try (AsyncHttpClient client = asyncHttpClient(config().setMaxConnections(1))) { final CountDownLatch latch = new CountDownLatch(samples); final List<Exception> tooManyConnections = Collections.synchronizedList(new ArrayList<>(2)); for (int i = 0; i < samples; i++) { new Thread(new Runnable() { public void run() { try { requestThrottle.acquire(); Future<Response> responseFuture = null; try { responseFuture = client.prepareGet(getTargetUrl()).setRequestTimeout(SLEEPTIME_MS / 2) .execute(new AsyncCompletionHandler<Response>() { @Override public Response onCompleted(Response response) throws Exception { return response; } @Override public void onThrowable(Throwable t) { logger.error("onThrowable got an error", t); try { Thread.sleep(100); } catch (InterruptedException e) { } requestThrottle.release(); } }); } catch (Exception e) { tooManyConnections.add(e); } if (responseFuture != null) responseFuture.get(); } catch (Exception e) { } finally { latch.countDown(); } } }).start(); } try { latch.await(30, TimeUnit.SECONDS); } catch (Exception e) { fail("failed to wait for requests to complete"); } for (Exception e : tooManyConnections) logger.error("Exception while calling execute", e); assertTrue(tooManyConnections.isEmpty(), "Should not have any connection errors where too many connections have been attempted"); } } }