// // ======================================================================== // 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.fcgi.server; import java.io.EOFException; import java.io.IOException; import java.net.URI; import java.net.URLEncoder; import java.nio.ByteBuffer; import java.util.Arrays; import java.util.Random; import java.util.concurrent.CountDownLatch; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicReference; import java.util.zip.GZIPOutputStream; import javax.servlet.ServletException; import javax.servlet.ServletOutputStream; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.eclipse.jetty.client.api.ContentResponse; import org.eclipse.jetty.client.api.Request; import org.eclipse.jetty.client.api.Response; import org.eclipse.jetty.client.api.Result; import org.eclipse.jetty.client.util.BytesContentProvider; import org.eclipse.jetty.client.util.DeferredContentProvider; import org.eclipse.jetty.client.util.FutureResponseListener; import org.eclipse.jetty.http.HttpMethod; import org.eclipse.jetty.server.handler.AbstractHandler; import org.eclipse.jetty.toolchain.test.IO; import org.eclipse.jetty.toolchain.test.annotation.Slow; import org.eclipse.jetty.util.Callback; import org.eclipse.jetty.util.log.StacklessLogging; import org.junit.Assert; import org.junit.Test; public class HttpClientTest extends AbstractHttpClientServerTest { @Test public void testGETResponseWithoutContent() throws Exception { start(new EmptyServerHandler()); for (int i = 0; i < 2; ++i) { Response response = client.GET(scheme + "://localhost:" + connector.getLocalPort()); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); } } @Test public void testGETResponseWithContent() throws Exception { final byte[] data = new byte[]{0, 1, 2, 3, 4, 5, 6, 7}; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { response.getOutputStream().write(data); baseRequest.setHandled(true); } }); int maxConnections = 256; client.setMaxConnectionsPerDestination(maxConnections); for (int i = 0; i < maxConnections + 1; ++i) { ContentResponse response = client.GET(scheme + "://localhost:" + connector.getLocalPort()); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); byte[] content = response.getContent(); Assert.assertArrayEquals(data, content); } } @Test public void testGETResponseWithBigContent() throws Exception { final byte[] data = new byte[16 * 1024 * 1024]; new Random().nextBytes(data); start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { // Setting the Content-Length triggers the HTTP // content mode for response content parsing, // otherwise the RAW content mode is used. response.setContentLength(data.length); response.getOutputStream().write(data); baseRequest.setHandled(true); } }); Request request = client.newRequest(scheme + "://localhost:" + connector.getLocalPort()); FutureResponseListener listener = new FutureResponseListener(request, data.length); request.send(listener); ContentResponse response = listener.get(15, TimeUnit.SECONDS); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); byte[] content = response.getContent(); Assert.assertArrayEquals(data, content); } @Test public void testGETWithParametersResponseWithContent() throws Exception { final String paramName1 = "a"; final String paramName2 = "b"; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { response.setCharacterEncoding("UTF-8"); ServletOutputStream output = response.getOutputStream(); String paramValue1 = request.getParameter(paramName1); output.write(paramValue1.getBytes("UTF-8")); String paramValue2 = request.getParameter(paramName2); Assert.assertEquals("", paramValue2); output.write("empty".getBytes("UTF-8")); baseRequest.setHandled(true); } }); String value1 = "\u20AC"; String paramValue1 = URLEncoder.encode(value1, "UTF-8"); String query = paramName1 + "=" + paramValue1 + "&" + paramName2; ContentResponse response = client.GET(scheme + "://localhost:" + connector.getLocalPort() + "/?" + query); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); String content = new String(response.getContent(), "UTF-8"); Assert.assertEquals(value1 + "empty", content); } @Test public void testGETWithParametersMultiValuedResponseWithContent() throws Exception { final String paramName1 = "a"; final String paramName2 = "b"; start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { response.setCharacterEncoding("UTF-8"); ServletOutputStream output = response.getOutputStream(); String[] paramValues1 = request.getParameterValues(paramName1); for (String paramValue : paramValues1) output.write(paramValue.getBytes("UTF-8")); String paramValue2 = request.getParameter(paramName2); output.write(paramValue2.getBytes("UTF-8")); baseRequest.setHandled(true); } }); String value11 = "\u20AC"; String value12 = "\u20AA"; String value2 = "&"; String paramValue11 = URLEncoder.encode(value11, "UTF-8"); String paramValue12 = URLEncoder.encode(value12, "UTF-8"); String paramValue2 = URLEncoder.encode(value2, "UTF-8"); String query = paramName1 + "=" + paramValue11 + "&" + paramName1 + "=" + paramValue12 + "&" + paramName2 + "=" + paramValue2; ContentResponse response = client.GET(scheme + "://localhost:" + connector.getLocalPort() + "/?" + query); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); String content = new String(response.getContent(), "UTF-8"); Assert.assertEquals(value11 + value12 + value2, content); } @Test public void testPOSTWithParameters() throws Exception { final String paramName = "a"; final String paramValue = "\u20AC"; 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); String value = request.getParameter(paramName); if (paramValue.equals(value)) { response.setCharacterEncoding("UTF-8"); response.setContentType("text/plain"); response.getOutputStream().print(value); } } }); ContentResponse response = client.POST(scheme + "://localhost:" + connector.getLocalPort()) .param(paramName, paramValue) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(paramValue, new String(response.getContent(), "UTF-8")); } @Test public void testPOSTWithQueryString() throws Exception { final String paramName = "a"; final String paramValue = "\u20AC"; 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); String value = request.getParameter(paramName); if (paramValue.equals(value)) { response.setCharacterEncoding("UTF-8"); response.setContentType("text/plain"); response.getOutputStream().print(value); } } }); String uri = scheme + "://localhost:" + connector.getLocalPort() + "/?" + paramName + "=" + URLEncoder.encode(paramValue, "UTF-8"); ContentResponse response = client.POST(uri) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(paramValue, new String(response.getContent(), "UTF-8")); } @Test public void testPUTWithParameters() throws Exception { final String paramName = "a"; final String paramValue = "\u20AC"; 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); String value = request.getParameter(paramName); if (paramValue.equals(value)) { response.setCharacterEncoding("UTF-8"); response.setContentType("text/plain"); response.getOutputStream().print(value); } } }); URI uri = URI.create(scheme + "://localhost:" + connector.getLocalPort() + "/path?" + paramName + "=" + paramValue); ContentResponse response = client.newRequest(uri) .method(HttpMethod.PUT) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(paramValue, new String(response.getContent(), "UTF-8")); } @Test public void testPOSTWithParametersWithContent() throws Exception { final byte[] content = {0, 1, 2, 3}; final String paramName = "a"; final String paramValue = "\u20AC"; 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); String value = request.getParameter(paramName); if (paramValue.equals(value)) { response.setCharacterEncoding("UTF-8"); response.setContentType("application/octet-stream"); IO.copy(request.getInputStream(), response.getOutputStream()); } } }); for (int i = 0; i < 256; ++i) { ContentResponse response = client.POST(scheme + "://localhost:" + connector.getLocalPort() + "/?b=1") .param(paramName, paramValue) .content(new BytesContentProvider(content)) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertArrayEquals(content, response.getContent()); } } @Test public void testPOSTWithContentNotifiesRequestContentListener() throws Exception { final byte[] content = {0, 1, 2, 3}; start(new EmptyServerHandler()); ContentResponse response = client.POST(scheme + "://localhost:" + connector.getLocalPort()) .onRequestContent(new Request.ContentListener() { @Override public void onContent(Request request, ByteBuffer buffer) { byte[] bytes = new byte[buffer.remaining()]; buffer.get(bytes); if (!Arrays.equals(content, bytes)) request.abort(new Exception()); } }) .content(new BytesContentProvider(content)) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); } @Test public void testPOSTWithContentTracksProgress() throws Exception { start(new EmptyServerHandler()); final AtomicInteger progress = new AtomicInteger(); ContentResponse response = client.POST(scheme + "://localhost:" + connector.getLocalPort()) .onRequestContent(new Request.ContentListener() { @Override public void onContent(Request request, ByteBuffer buffer) { byte[] bytes = new byte[buffer.remaining()]; Assert.assertEquals(1, bytes.length); buffer.get(bytes); Assert.assertEquals(bytes[0], progress.getAndIncrement()); } }) .content(new BytesContentProvider(new byte[]{0}, new byte[]{1}, new byte[]{2}, new byte[]{3}, new byte[]{4})) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(5, progress.get()); } @Test public void testGZIPContentEncoding() throws Exception { final byte[] data = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; 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); response.setHeader("Content-Encoding", "gzip"); GZIPOutputStream gzipOutput = new GZIPOutputStream(response.getOutputStream()); gzipOutput.write(data); gzipOutput.finish(); } }); ContentResponse response = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertEquals(200, response.getStatus()); Assert.assertArrayEquals(data, response.getContent()); } @Slow @Test public void testRequestIdleTimeout() throws Exception { final long idleTimeout = 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); TimeUnit.MILLISECONDS.sleep(2 * idleTimeout); } catch (InterruptedException x) { throw new ServletException(x); } } }); final String host = "localhost"; final int port = connector.getLocalPort(); try { client.newRequest(host, port) .scheme(scheme) .idleTimeout(idleTimeout, TimeUnit.MILLISECONDS) .timeout(3 * idleTimeout, TimeUnit.MILLISECONDS) .send(); Assert.fail(); } catch (ExecutionException expected) { Assert.assertTrue(expected.getCause() instanceof TimeoutException); } // Make another request without specifying the idle timeout, should not fail ContentResponse response = client.newRequest(host, port) .scheme(scheme) .timeout(3 * idleTimeout, TimeUnit.MILLISECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); } @Test public void testConnectionIdleTimeout() throws Exception { final long idleTimeout = 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); TimeUnit.MILLISECONDS.sleep(2 * idleTimeout); } catch (InterruptedException x) { throw new ServletException(x); } } }); connector.setIdleTimeout(idleTimeout); try { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .idleTimeout(4 * idleTimeout, TimeUnit.MILLISECONDS) .timeout(3 * idleTimeout, TimeUnit.MILLISECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { Assert.assertTrue(x.getCause() instanceof EOFException); } connector.setIdleTimeout(5 * idleTimeout); // Make another request to be sure the connection is recreated ContentResponse response = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .idleTimeout(4 * idleTimeout, TimeUnit.MILLISECONDS) .timeout(3 * idleTimeout, TimeUnit.MILLISECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); } @Test public void testSendToIPv6Address() throws Exception { start(new EmptyServerHandler()); ContentResponse response = client.newRequest("[::1]", connector.getLocalPort()) .scheme(scheme) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); } @Test public void testHEADWithResponseContentLength() throws Exception { final int length = 1024; 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); response.getOutputStream().write(new byte[length]); } }); // HEAD requests receive a Content-Length header, but do not // receive the content so they must handle this case properly ContentResponse response = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .method(HttpMethod.HEAD) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(0, response.getContent().length); // Perform a normal GET request to be sure the content is now read response = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .timeout(5, TimeUnit.SECONDS) .send(); Assert.assertNotNull(response); Assert.assertEquals(200, response.getStatus()); Assert.assertEquals(length, response.getContent().length); } @Test public void testLongPollIsAbortedWhenClientIsStopped() throws Exception { final CountDownLatch latch = new CountDownLatch(1); 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); request.startAsync(); latch.countDown(); } }); final CountDownLatch completeLatch = new CountDownLatch(1); client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .send(new Response.CompleteListener() { @Override public void onComplete(Result result) { if (result.isFailed()) completeLatch.countDown(); } }); Assert.assertTrue(latch.await(5, TimeUnit.SECONDS)); // Stop the client, the complete listener must be invoked. client.stop(); Assert.assertTrue(completeLatch.await(5, TimeUnit.SECONDS)); } @Test public void testEarlyEOF() 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); // Promise some content, then flush the headers, then fail to send the content. response.setContentLength(16); response.flushBuffer(); throw new NullPointerException("Explicitly thrown by test"); } }); try (StacklessLogging stackless = new StacklessLogging(org.eclipse.jetty.server.HttpChannel.class)) { client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .timeout(60, TimeUnit.SECONDS) .send(); Assert.fail(); } catch (ExecutionException x) { // Expected. } } @Test public void testSmallContentDelimitedByEOFWithSlowRequest() throws Exception { testContentDelimitedByEOFWithSlowRequest(1024); } @Test public void testBigContentDelimitedByEOFWithSlowRequest() throws Exception { testContentDelimitedByEOFWithSlowRequest(128 * 1024); } private void testContentDelimitedByEOFWithSlowRequest(int length) throws Exception { final byte[] data = new byte[length]; new Random().nextBytes(data); 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); response.setHeader("Connection", "close"); response.getOutputStream().write(data); } }); DeferredContentProvider content = new DeferredContentProvider(ByteBuffer.wrap(new byte[]{0})); Request request = client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .content(content); FutureResponseListener listener = new FutureResponseListener(request); request.send(listener); // Wait some time to simulate a slow request. Thread.sleep(1000); content.close(); ContentResponse response = listener.get(5, TimeUnit.SECONDS); Assert.assertEquals(200, response.getStatus()); Assert.assertArrayEquals(data, response.getContent()); } @Test public void testSmallAsyncContent() throws Exception { start(new AbstractHandler() { @Override public void handle(String target, org.eclipse.jetty.server.Request baseRequest, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException { ServletOutputStream output = response.getOutputStream(); output.write(65); output.flush(); output.write(66); } }); final AtomicInteger contentCount = new AtomicInteger(); final AtomicReference<Callback> callbackRef = new AtomicReference<>(); final AtomicReference<CountDownLatch> contentLatch = new AtomicReference<>(new CountDownLatch(1)); final CountDownLatch completeLatch = new CountDownLatch(1); client.newRequest("localhost", connector.getLocalPort()) .scheme(scheme) .onResponseContentAsync(new Response.AsyncContentListener() { @Override public void onContent(Response response, ByteBuffer content, Callback callback) { contentCount.incrementAndGet(); callbackRef.set(callback); contentLatch.get().countDown(); } }) .send(new Response.CompleteListener() { @Override public void onComplete(Result result) { completeLatch.countDown(); } }); Assert.assertTrue(contentLatch.get().await(5, TimeUnit.SECONDS)); Callback callback = callbackRef.get(); // Wait a while to be sure that the parsing does not proceed. TimeUnit.MILLISECONDS.sleep(1000); Assert.assertEquals(1, contentCount.get()); // Succeed the content callback to proceed with parsing. callbackRef.set(null); contentLatch.set(new CountDownLatch(1)); callback.succeeded(); Assert.assertTrue(contentLatch.get().await(5, TimeUnit.SECONDS)); callback = callbackRef.get(); // Wait a while to be sure that the parsing does not proceed. TimeUnit.MILLISECONDS.sleep(1000); Assert.assertEquals(2, contentCount.get()); Assert.assertEquals(1, completeLatch.getCount()); // Succeed the content callback to proceed with parsing. callbackRef.set(null); contentLatch.set(new CountDownLatch(1)); callback.succeeded(); Assert.assertTrue(completeLatch.await(5, TimeUnit.SECONDS)); Assert.assertEquals(2, contentCount.get()); } }