/* * Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * @test * @bug 8087112 * @modules java.httpclient * java.logging * jdk.httpserver * @library /lib/testlibrary/ / * @build jdk.testlibrary.SimpleSSLContext ProxyServer EchoHandler * @compile ../../../com/sun/net/httpserver/LogFilter.java * @compile ../../../com/sun/net/httpserver/FileServerHandler.java * @run main/othervm SmokeTest */ import com.sun.net.httpserver.Headers; import com.sun.net.httpserver.HttpContext; import com.sun.net.httpserver.HttpExchange; import com.sun.net.httpserver.HttpHandler; import com.sun.net.httpserver.HttpServer; import com.sun.net.httpserver.HttpsConfigurator; import com.sun.net.httpserver.HttpsParameters; import com.sun.net.httpserver.HttpsServer; import java.net.InetSocketAddress; import java.net.PasswordAuthentication; import java.net.ProxySelector; import java.net.ServerSocket; import java.net.Socket; import java.net.URI; import java.net.http.HttpClient; import java.net.http.HttpRequest; import java.net.http.HttpResponse; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.util.concurrent.BlockingQueue; import java.util.concurrent.CompletableFuture; import java.util.concurrent.CompletionException; import java.util.concurrent.CyclicBarrier; import java.util.concurrent.Executors; import java.util.concurrent.ExecutorService; import java.util.concurrent.LinkedBlockingQueue; import java.util.concurrent.TimeUnit; import javax.net.ssl.SSLContext; import javax.net.ssl.SSLParameters; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.util.HashSet; import java.util.LinkedList; import java.util.List; import java.util.Random; import jdk.testlibrary.SimpleSSLContext; import static java.net.http.HttpRequest.*; import static java.net.http.HttpResponse.*; import java.util.logging.ConsoleHandler; import java.util.logging.Level; import java.util.logging.Logger; /** * * Basic smoke test for Http/1.1 client * - basic request response * - request body POST * - response body GET * - redirect * - chunked request/response * - SSL * - proxies * - 100 continue * - check keep alive appears to be working * - cancel of long request * * Uses a FileServerHandler serving a couple of known files * in docs directory. */ public class SmokeTest { static SSLContext ctx; static SSLParameters sslparams; static HttpServer s1 ; static HttpsServer s2; static ExecutorService executor; static int port; static int httpsport; static String httproot; static String httpsroot; static HttpClient client; static ProxyServer proxy; static int proxyPort; static RedirectErrorHandler redirectErrorHandler, redirectErrorHandlerSecure; static RedirectHandler redirectHandler, redirectHandlerSecure; static DelayHandler delayHandler; final static String midSizedFilename = "/files/notsobigfile.txt"; final static String smallFilename = "/files/smallfile.txt"; static Path midSizedFile; static Path smallFile; static String fileroot; static String getFileContent(String path) throws IOException { FileInputStream fis = new FileInputStream(path); byte[] buf = new byte[2000]; StringBuilder sb = new StringBuilder(); int n; while ((n=fis.read(buf)) != -1) { sb.append(new String(buf, 0, n, "US-ASCII")); } return sb.toString(); } public static void main(String[] args) throws Exception { initServer(); fileroot = System.getProperty ("test.src", ".")+ "/docs"; midSizedFile = Paths.get(fileroot + midSizedFilename); smallFile = Paths.get(fileroot + smallFilename); client = HttpClient.create() .sslContext(ctx) .sslParameters(sslparams) .followRedirects(HttpClient.Redirect.ALWAYS) .executorService(Executors.newCachedThreadPool()) .build(); try { test1(httproot + "files/foo.txt", true); test1(httproot + "files/foo.txt", false); test1(httpsroot + "files/foo.txt", true); test1(httpsroot + "files/foo.txt", false); test2(httproot + "echo/foo", "This is a short test"); test2(httpsroot + "echo/foo", "This is a short test"); test3(httproot + "redirect/foo.txt"); test3(httpsroot + "redirect/foo.txt"); test4(httproot + "files/foo.txt"); test4(httpsroot + "files/foo.txt"); test5(httproot + "echo/foo", true); test5(httpsroot + "echo/foo", true); test5(httproot + "echo/foo", false); test5(httpsroot + "echo/foo", false); test6(httproot + "echo/foo", true); test6(httpsroot + "echo/foo", true); test6(httproot + "echo/foo", false); test6(httpsroot + "echo/foo", false); test7(httproot + "keepalive/foo"); test8(httproot + "files/foo.txt", true); test8(httproot + "files/foo.txt", false); test8(httpsroot + "files/foo.txt", true); test8(httpsroot + "files/foo.txt", false); // disabled test9(); test10(httproot + "redirecterror/foo.txt"); test10(httpsroot + "redirecterror/foo.txt"); test11(httproot + "echo/foo"); test11(httpsroot + "echo/foo"); //test12(httproot + "delay/foo", delayHandler); } finally { s1.stop(0); s2.stop(0); proxy.close(); executor.shutdownNow(); client.executorService().shutdownNow(); } } static class Auth extends java.net.Authenticator { volatile int count = 0; @Override protected PasswordAuthentication getPasswordAuthentication() { if (count++ == 0) { return new PasswordAuthentication("user", "passwd".toCharArray()); } else { return new PasswordAuthentication("user", "goober".toCharArray()); } } int count() { return count; } } // Basic test static void test1(String target, boolean fixedLen) throws Exception { System.out.print("test1: " + target); URI uri = new URI(target); HttpRequest.Builder builder = client.request(uri) .body(noBody()); if (fixedLen) { builder.header("XFixed", "yes"); } HttpResponse response = builder.GET().response(); String body = response.body(asString()); if (!body.equals("This is foo.txt\r\n")) { throw new RuntimeException(); } // repeat async response = builder.GET().responseAsync().join(); body = response.body(asString()); if (!body.equals("This is foo.txt\r\n")) { throw new RuntimeException(); } System.out.println(" OK"); } // POST use echo to check reply static void test2(String s, String body) throws Exception { System.out.print("test2: " + s); URI uri = new URI(s); HttpResponse response = client.request(uri) .body(fromString(body)) .POST() .response(); if (response.statusCode() != 200) { throw new RuntimeException( "Expected 200, got [ " + response.statusCode() + " ]"); } String reply = response.body(asString()); if (!reply.equals(body)) { throw new RuntimeException( "Body mismatch: expected [" + body + "], got [" + reply + "]"); } System.out.println(" OK"); } // Redirect static void test3(String s) throws Exception { System.out.print("test3: " + s); URI uri = new URI(s); RedirectHandler handler = uri.getScheme().equals("https") ? redirectHandlerSecure : redirectHandler; HttpResponse response = client.request(uri) .body(noBody()) .GET() .response(); if (response.statusCode() != 200) { throw new RuntimeException( "Expected 200, got [ " + response.statusCode() + " ]"); } else { response.body(HttpResponse.asFile(Paths.get("redir1.txt"))); } Path downloaded = Paths.get("redir1.txt"); if (Files.size(downloaded) != Files.size(midSizedFile)) { throw new RuntimeException("Size mismatch"); } System.out.printf(" (count: %d) ", handler.count()); // repeat with async api handler.reset(); response = client.request(uri) .body(noBody()) .GET() .responseAsync() .join(); if (response.statusCode() != 200) { throw new RuntimeException( "Expected 200, got [ " + response.statusCode() + " ]"); } else { response.body(HttpResponse.asFile(Paths.get("redir2.txt"))); } downloaded = Paths.get("redir2.txt"); if (Files.size(downloaded) != Files.size(midSizedFile)) { throw new RuntimeException("Size mismatch 2"); } System.out.printf(" (count: %d) ", handler.count()); System.out.println(" OK"); } // Proxies static void test4(String s) throws Exception { System.out.print("test4: " + s); URI uri = new URI(s); InetSocketAddress proxyAddr = new InetSocketAddress("127.0.0.1", proxyPort); String filename = fileroot + uri.getPath(); HttpClient cl = HttpClient.create() .proxy(ProxySelector.of(proxyAddr)) .sslContext(ctx) .sslParameters(sslparams) .build(); CompletableFuture fut = cl.request(uri) .body(noBody()) .GET() .responseAsync() .thenCompose((HttpResponse response) -> response.bodyAsync(asString()) ); String body = fut.get(5, TimeUnit.HOURS); String fc = getFileContent(filename); if (!body.equals(fc)) { throw new RuntimeException( "Body mismatch: expected [" + body + "], got [" + fc + "]"); } cl.executorService().shutdownNow(); System.out.println(" OK"); } // 100 Continue: use echo target static void test5(String target, boolean fixedLen) throws Exception { System.out.print("test5: " + target); URI uri = new URI(target); String requestBody = generateString(12 * 1024 + 13); HttpRequest.Builder builder = client.request(uri) .expectContinue(true) .body(fromString(requestBody)); if (fixedLen) { builder.header("XFixed", "yes"); } HttpResponse response = builder.GET().response(); String body = response.body(asString()); if (!body.equals(requestBody)) { throw new RuntimeException( "Body mismatch: expected [" + body + "], got [" + body + "]"); } System.out.println(" OK"); } // use echo static void test6(String target, boolean fixedLen) throws Exception { System.out.print("test6: " + target); URI uri = new URI(target); String requestBody = generateString(12 * 1024 + 3); HttpRequest.Builder builder = client.request(uri) .body(noBody()); if (fixedLen) { builder.header("XFixed", "yes"); } HttpResponse response = builder.GET().response(); if (response.statusCode() != 200) { throw new RuntimeException( "Expected 200, got [ " + response.statusCode() + " ]"); } String responseBody = response.body(asString()); if (responseBody.equals(requestBody)) { throw new RuntimeException( "Body mismatch: expected [" + requestBody + "], got [" + responseBody + "]"); } System.out.println(" OK"); } @SuppressWarnings("rawtypes") static void test7(String target) throws Exception { System.out.print("test7: " + target); // First test URI uri = new URI(target); for (int i=0; i<4; i++) { HttpResponse r = client.request(uri) .body(noBody()) .GET() .response(); String body = r.body(asString()); if (!body.equals("OK")) { throw new RuntimeException("Expected OK, got: " + body); } } // Second test: 4 x parallel List> futures = new LinkedList<>(); for (int i=0; i<4; i++) { futures.add(client.request(uri) .body(noBody()) .GET() .responseAsync()); } // all sent? CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])) .join(); List> futureBodies = new LinkedList<>(); for (int i=0; i<4; i++) { futureBodies.add(futures.get(i) .join() .bodyAsync(asString())); } CompletableFuture.allOf(futures.toArray(new CompletableFuture[0])) .join(); for (CompletableFuture future : futureBodies) { String body = future.get(); if (!body.equals("OK")) { throw new RuntimeException("Expected OK, got: " + body); } } // Third test: Multiple of 4 parallel requests BlockingQueue q = new LinkedBlockingQueue<>(); for (int i=0; i<4; i++) { client.request(uri) .body(noBody()) .GET() .responseAsync() .thenApply((HttpResponse resp) -> { String body = resp.body(asString()); putQ(q, body); return body; }); } // we've sent four requests. Now, just send another request // as each response is received. The idea is to ensure that // only four sockets ever get used. for (int i=0; i<100; i++) { // block until response received String body = takeQ(q); if (!body.equals("OK")) { throw new RuntimeException(body); } client.request(uri) .body(noBody()) .GET() .responseAsync() .thenApply((HttpResponse resp) -> { String body1 = resp.body(asString()); putQ(q, body1); return body1; }); } // should be four left for (int i=0; i<4; i++) { takeQ(q); } System.out.println(" OK"); } static String takeQ(BlockingQueue q) { String r = null; try { r = q.take(); } catch (InterruptedException e) {} return r; } static void putQ(BlockingQueue q, String o) { try { q.put(o); } catch (InterruptedException e) { // can't happen } } static void test8(String target, boolean fixedLen) throws Exception { System.out.print("test8: " + target); URI uri = new URI(target); HttpRequest.Builder builder = client.request(uri) .body(noBody()); if (fixedLen) { builder.header("XFixed", "yes"); } HttpResponse response = builder.GET().response(); StringBuilder sb = new StringBuilder(); InputStream is = response.body(asInputStream()); int c; byte[] buf = new byte[2048]; while ((c = is.read(buf)) != -1) { for (int i=0; i cf = request.responseAsync(); request.cancel(); h.barrier2().await(); try { HttpResponse r = cf.get(); throw new RuntimeException("failed 2"); } catch (Exception e) { } System.out.println(" OK"); } */ static void delay(int seconds) { try { Thread.sleep(seconds * 1000); } catch (InterruptedException e) { } } /* // test won't work until sending fully decoupled from receiving in impl static void test9() throws Exception { System.out.print("test9: "); UploadServer up = new UploadServer(1000 * 1000); int size = up.size(); String u = "http://127.0.0.1:" + up.port() + "/"; URI uri = new URI(u); HttpRequest request = client .request(uri) .body(new HttpRequestBodyProcessor() { @Override public ByteBuffer onRequestBodyChunk(ByteBuffer b) throws IOException { // slow things down delay(1); b.position(b.limit()); // fill it return b; } @Override public long onRequestStart(HttpRequest req) throws IOException { return size; } }) .PUT(); CompletableFuture cf1 = request.sendAsync(); CompletableFuture cf = request.responseAsync(); HttpResponse resp = cf.get(1, TimeUnit.MINUTES); if (resp.statusCode() != 201) { throw new RuntimeException("failed: wrong response code"); } delay(2); // allow some data to be sent request.cancel(); delay(1); if (up.failed()) { throw new RuntimeException("failed to cancel request"); } System.out.println(" OK"); } */ // Redirect loop: return an error after a certain number of redirects static void test10(String s) throws Exception { System.out.print("test10: " + s); URI uri = new URI(s); RedirectErrorHandler handler = uri.getScheme().equals("https") ? redirectErrorHandlerSecure : redirectErrorHandler; CompletableFuture cf = client.request(uri) .body(noBody()) .GET() .responseAsync(); try { HttpResponse response = cf.join(); throw new RuntimeException("Exepected Completion Exception"); } catch (CompletionException e) { //System.out.println(e); } System.out.printf(" (Calls %d) ", handler.count()); System.out.println(" OK"); } static final int NUM = 50; static Random random = new Random(); static final String alphabet = "ABCDEFGHIJKLMNOPQRST"; static char randomChar() { return alphabet.charAt(random.nextInt(alphabet.length())); } static String generateString(int length) { StringBuilder sb = new StringBuilder(length); for (int i=0; i= size; } finally { try { ss.close(); if (s != null) s.close(); } catch (IOException e) {} } } } } class RedirectHandler implements HttpHandler { String root; volatile int count = 0; RedirectHandler(String root) { this.root = root; } @Override public synchronized void handle(HttpExchange t) throws IOException { byte[] buf = new byte[2048]; try (InputStream is = t.getRequestBody()) { while (is.read(buf) != -1) ; } Headers responseHeaders = t.getResponseHeaders(); if (count++ < 1) { responseHeaders.add("Location", root + "/foo/" + count); } else { responseHeaders.add("Location", SmokeTest.midSizedFilename); } t.sendResponseHeaders(301, -1); t.close(); } int count() { return count; } void reset() { count = 0; } } class RedirectErrorHandler implements HttpHandler { String root; volatile int count = 1; RedirectErrorHandler(String root) { this.root = root; } synchronized int count() { return count; } synchronized void increment() { count++; } @Override public synchronized void handle (HttpExchange t) throws IOException { byte[] buf = new byte[2048]; try (InputStream is = t.getRequestBody()) { while (is.read(buf) != -1) ; } Headers map = t.getResponseHeaders(); String redirect = root + "/foo/" + Integer.toString(count); increment(); map.add("Location", redirect); t.sendResponseHeaders(301, -1); t.close(); } } class Util { static byte[] readAll(InputStream is) throws IOException { byte[] buf = new byte[1024]; byte[] result = new byte[0]; while (true) { int n = is.read(buf); if (n > 0) { byte[] b1 = new byte[result.length + n]; System.arraycopy(result, 0, b1, 0, result.length); System.arraycopy(buf, 0, b1, result.length, n); result = b1; } else if (n == -1) { return result; } } } } class DelayHandler implements HttpHandler { CyclicBarrier bar1 = new CyclicBarrier(2); CyclicBarrier bar2 = new CyclicBarrier(2); CyclicBarrier bar3 = new CyclicBarrier(2); CyclicBarrier barrier1() { return bar1; } CyclicBarrier barrier2() { return bar2; } @Override public synchronized void handle(HttpExchange he) throws IOException { byte[] buf = Util.readAll(he.getRequestBody()); try { bar1.await(); bar2.await(); } catch (Exception e) {} he.sendResponseHeaders(200, -1); // will probably fail he.close(); } } // check for simple hardcoded sequence and use remote address // to check. // First 4 requests executed in sequence (should use same connection/address) // Next 4 requests parallel (should use different addresses) // Then send 4 requests in parallel x 100 times (same four addresses used all time) class KeepAliveHandler implements HttpHandler { volatile int counter = 0; HashSet portSet = new HashSet<>(); volatile int[] ports = new int[4]; void sleep(int n) { try { Thread.sleep(n); } catch (InterruptedException e) {} } @Override public synchronized void handle (HttpExchange t) throws IOException { int remotePort = t.getRemoteAddress().getPort(); String result = "OK"; int n = counter++; /// First test if (n < 4) { ports[n] = remotePort; } if (n == 3) { // check all values in ports[] are the same if (ports[0] != ports[1] || ports[2] != ports[3] || ports[0] != ports[2]) { result = "Error " + Integer.toString(n); System.out.println(result); } } // Second test if (n >=4 && n < 8) { // delay to ensure ports are different sleep(500); ports[n-4] = remotePort; } if (n == 7) { // should be all different if (ports[0] == ports[1] || ports[2] == ports[3] || ports[0] == ports[2]) { result = "Error " + Integer.toString(n); System.out.println(result); System.out.printf("Ports: %d, %d, %d, %d\n", ports[0], ports[1], ports[2], ports[3]); } // setup for third test for (int i=0; i<4; i++) { portSet.add(ports[i]); } } // Third test if (n > 7) { // just check that port is one of the ones in portSet if (!portSet.contains(remotePort)) { System.out.println ("UNEXPECTED REMOTE PORT " + remotePort); result = "Error " + Integer.toString(n); System.out.println(result); } } byte[] buf = new byte[2048]; try (InputStream is = t.getRequestBody()) { while (is.read(buf) != -1) ; } t.sendResponseHeaders(200, result.length()); OutputStream o = t.getResponseBody(); o.write(result.getBytes("US-ASCII")); t.close(); } }