2016-05-18 16:39:08 +01:00
|
|
|
/*
|
2023-01-25 13:38:02 +00:00
|
|
|
* Copyright (c) 2016, 2023, Oracle and/or its affiliates. All rights reserved.
|
2016-05-18 16:39:08 +01:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.io.File;
|
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.InputStream;
|
|
|
|
import java.io.OutputStream;
|
|
|
|
import java.net.URI;
|
|
|
|
import java.net.URISyntaxException;
|
2018-01-19 11:24:39 -08:00
|
|
|
import java.security.Security;
|
2018-04-17 08:54:17 -07:00
|
|
|
import java.net.http.HttpClient;
|
|
|
|
import java.net.http.HttpRequest;
|
|
|
|
import java.net.http.HttpRequest.BodyPublishers;
|
|
|
|
import java.net.http.HttpResponse.BodyHandlers;
|
2018-07-04 16:16:24 +01:00
|
|
|
import javax.net.ssl.SSLContext;
|
2016-05-18 16:39:08 +01:00
|
|
|
import javax.net.ssl.SSLParameters;
|
|
|
|
import javax.net.ssl.SSLSession;
|
2023-01-25 13:38:02 +00:00
|
|
|
import jdk.httpclient.test.lib.http2.Http2TestServer;
|
|
|
|
import jdk.httpclient.test.lib.http2.Http2TestExchange;
|
|
|
|
import jdk.httpclient.test.lib.http2.Http2Handler;
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @bug 8150769 8157107
|
2023-01-25 13:38:02 +00:00
|
|
|
* @library /test/jdk/java/net/httpclient/lib
|
|
|
|
* @build jdk.httpclient.test.lib.http2.Http2TestServer
|
2016-05-18 16:39:08 +01:00
|
|
|
* @summary Checks that SSL parameters can be set for HTTP/2 connection
|
2018-04-17 08:54:17 -07:00
|
|
|
* @run main/othervm
|
|
|
|
* -Djdk.internal.httpclient.debug=true
|
|
|
|
* -Djdk.httpclient.HttpClient.log=all
|
|
|
|
* TLSConnection
|
2016-05-18 16:39:08 +01:00
|
|
|
*/
|
|
|
|
public class TLSConnection {
|
|
|
|
|
|
|
|
private static final String KEYSTORE = System.getProperty("test.src")
|
|
|
|
+ File.separator + "keystore.p12";
|
2018-04-17 08:54:17 -07:00
|
|
|
private static final String PASSWORD = "password";
|
|
|
|
|
|
|
|
private static final SSLParameters USE_DEFAULT_SSL_PARAMETERS = new SSLParameters();
|
2016-05-18 16:39:08 +01:00
|
|
|
|
2018-07-04 16:16:24 +01:00
|
|
|
// expect highest supported version we know about
|
|
|
|
static String expectedTLSVersion(SSLContext ctx) throws Exception {
|
|
|
|
if (ctx == null)
|
|
|
|
ctx = SSLContext.getDefault();
|
|
|
|
SSLParameters params = ctx.getSupportedSSLParameters();
|
|
|
|
String[] protocols = params.getProtocols();
|
|
|
|
for (String prot : protocols) {
|
|
|
|
if (prot.equals("TLSv1.3"))
|
|
|
|
return "TLSv1.3";
|
|
|
|
}
|
|
|
|
return "TLSv1.2";
|
|
|
|
}
|
|
|
|
|
2016-05-18 16:39:08 +01:00
|
|
|
public static void main(String[] args) throws Exception {
|
2018-01-19 11:24:39 -08:00
|
|
|
// re-enable 3DES
|
|
|
|
Security.setProperty("jdk.tls.disabledAlgorithms", "");
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
// enable all logging
|
2016-12-09 11:35:02 +00:00
|
|
|
System.setProperty("jdk.httpclient.HttpClient.log", "all,frames:all");
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
// initialize JSSE
|
|
|
|
System.setProperty("javax.net.ssl.keyStore", KEYSTORE);
|
|
|
|
System.setProperty("javax.net.ssl.keyStorePassword", PASSWORD);
|
|
|
|
System.setProperty("javax.net.ssl.trustStore", KEYSTORE);
|
|
|
|
System.setProperty("javax.net.ssl.trustStorePassword", PASSWORD);
|
|
|
|
|
|
|
|
Handler handler = new Handler();
|
|
|
|
|
2018-04-17 08:54:17 -07:00
|
|
|
try (Http2TestServer server = new Http2TestServer("localhost", true, 0)) {
|
2016-12-09 11:35:02 +00:00
|
|
|
server.addHandler(handler, "/");
|
2016-05-18 16:39:08 +01:00
|
|
|
server.start();
|
|
|
|
|
|
|
|
int port = server.getAddress().getPort();
|
2018-04-17 08:54:17 -07:00
|
|
|
String uriString = "https://localhost:" + Integer.toString(port);
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
// run test cases
|
|
|
|
boolean success = true;
|
|
|
|
|
|
|
|
SSLParameters parameters = null;
|
|
|
|
success &= expectFailure(
|
2018-04-17 08:54:17 -07:00
|
|
|
"---\nTest #1: SSL parameters is null, expect NPE",
|
2016-05-18 16:39:08 +01:00
|
|
|
() -> connect(uriString, parameters),
|
|
|
|
NullPointerException.class);
|
|
|
|
|
|
|
|
success &= expectSuccess(
|
2018-04-17 08:54:17 -07:00
|
|
|
"---\nTest #2: default SSL parameters, "
|
2016-05-18 16:39:08 +01:00
|
|
|
+ "expect successful connection",
|
2018-04-17 08:54:17 -07:00
|
|
|
() -> connect(uriString, USE_DEFAULT_SSL_PARAMETERS));
|
2018-07-04 16:16:24 +01:00
|
|
|
success &= checkProtocol(handler.getSSLSession(), expectedTLSVersion(null));
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
// set SSL_DHE_RSA_WITH_3DES_EDE_CBC_SHA cipher suite
|
|
|
|
// which has less priority in default cipher suite list
|
|
|
|
success &= expectSuccess(
|
2018-04-17 08:54:17 -07:00
|
|
|
"---\nTest #3: SSL parameters with "
|
2016-05-18 16:39:08 +01:00
|
|
|
+ "SSL_DHE_RSA_WITH_3DES_EDE_CBC_SHA cipher suite, "
|
|
|
|
+ "expect successful connection",
|
|
|
|
() -> connect(uriString, new SSLParameters(
|
2018-04-17 08:54:17 -07:00
|
|
|
new String[] { "SSL_DHE_RSA_WITH_3DES_EDE_CBC_SHA" },
|
|
|
|
new String[] { "TLSv1.2" })));
|
2016-05-18 16:39:08 +01:00
|
|
|
success &= checkProtocol(handler.getSSLSession(), "TLSv1.2");
|
|
|
|
success &= checkCipherSuite(handler.getSSLSession(),
|
|
|
|
"SSL_DHE_RSA_WITH_3DES_EDE_CBC_SHA");
|
|
|
|
|
|
|
|
// set TLS_RSA_WITH_AES_128_CBC_SHA cipher suite
|
|
|
|
// which has less priority in default cipher suite list
|
2018-04-17 08:54:17 -07:00
|
|
|
// also set TLSv1.2 protocol
|
2016-05-18 16:39:08 +01:00
|
|
|
success &= expectSuccess(
|
2018-04-17 08:54:17 -07:00
|
|
|
"---\nTest #4: SSL parameters with "
|
2016-05-18 16:39:08 +01:00
|
|
|
+ "TLS_RSA_WITH_AES_128_CBC_SHA cipher suite,"
|
|
|
|
+ " expect successful connection",
|
|
|
|
() -> connect(uriString, new SSLParameters(
|
|
|
|
new String[] { "TLS_RSA_WITH_AES_128_CBC_SHA" },
|
2018-04-17 08:54:17 -07:00
|
|
|
new String[] { "TLSv1.2" })));
|
|
|
|
success &= checkProtocol(handler.getSSLSession(), "TLSv1.2");
|
2016-05-18 16:39:08 +01:00
|
|
|
success &= checkCipherSuite(handler.getSSLSession(),
|
|
|
|
"TLS_RSA_WITH_AES_128_CBC_SHA");
|
|
|
|
|
|
|
|
if (success) {
|
|
|
|
System.out.println("Test passed");
|
|
|
|
} else {
|
|
|
|
throw new RuntimeException("At least one test case failed");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static interface Test {
|
|
|
|
|
|
|
|
public void run() throws Exception;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static class Handler implements Http2Handler {
|
|
|
|
|
|
|
|
private static final byte[] BODY = "Test response".getBytes();
|
|
|
|
|
|
|
|
private volatile SSLSession sslSession;
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void handle(Http2TestExchange t) throws IOException {
|
|
|
|
System.out.println("Handler: received request to "
|
|
|
|
+ t.getRequestURI());
|
|
|
|
|
|
|
|
try (InputStream is = t.getRequestBody()) {
|
|
|
|
byte[] body = is.readAllBytes();
|
|
|
|
System.out.println("Handler: read " + body.length
|
|
|
|
+ " bytes of body: ");
|
|
|
|
System.out.println(new String(body));
|
|
|
|
}
|
|
|
|
|
2022-04-15 09:07:09 +00:00
|
|
|
sslSession = t.getSSLSession();
|
|
|
|
|
2016-05-18 16:39:08 +01:00
|
|
|
try (OutputStream os = t.getResponseBody()) {
|
|
|
|
t.sendResponseHeaders(200, BODY.length);
|
|
|
|
os.write(BODY);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
SSLSession getSSLSession() {
|
|
|
|
return sslSession;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void connect(String uriString, SSLParameters sslParameters)
|
2018-04-17 08:54:17 -07:00
|
|
|
throws URISyntaxException, IOException, InterruptedException
|
2016-12-09 11:35:02 +00:00
|
|
|
{
|
2018-04-17 08:54:17 -07:00
|
|
|
HttpClient.Builder builder = HttpClient.newBuilder()
|
|
|
|
.version(HttpClient.Version.HTTP_2);
|
|
|
|
if (sslParameters != USE_DEFAULT_SSL_PARAMETERS)
|
|
|
|
builder.sslParameters(sslParameters);
|
|
|
|
HttpClient client = builder.build();
|
|
|
|
|
2016-12-09 11:35:02 +00:00
|
|
|
HttpRequest request = HttpRequest.newBuilder(new URI(uriString))
|
2018-04-17 08:54:17 -07:00
|
|
|
.POST(BodyPublishers.ofString("body"))
|
|
|
|
.build();
|
|
|
|
String body = client.send(request, BodyHandlers.ofString()).body();
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
System.out.println("Response: " + body);
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean checkProtocol(SSLSession session, String protocol) {
|
|
|
|
if (session == null) {
|
|
|
|
System.out.println("Check protocol: no session provided");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
System.out.println("Check protocol: negotiated protocol: "
|
|
|
|
+ session.getProtocol());
|
|
|
|
System.out.println("Check protocol: expected protocol: "
|
|
|
|
+ protocol);
|
|
|
|
if (!protocol.equals(session.getProtocol())) {
|
|
|
|
System.out.println("Check protocol: unexpected negotiated protocol");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean checkCipherSuite(SSLSession session, String ciphersuite) {
|
|
|
|
if (session == null) {
|
|
|
|
System.out.println("Check protocol: no session provided");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
System.out.println("Check protocol: negotiated ciphersuite: "
|
|
|
|
+ session.getCipherSuite());
|
|
|
|
System.out.println("Check protocol: expected ciphersuite: "
|
|
|
|
+ ciphersuite);
|
|
|
|
if (!ciphersuite.equals(session.getCipherSuite())) {
|
|
|
|
System.out.println("Check protocol: unexpected negotiated ciphersuite");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean expectSuccess(String message, Test test) {
|
|
|
|
System.out.println(message);
|
|
|
|
try {
|
|
|
|
test.run();
|
|
|
|
System.out.println("Passed");
|
|
|
|
return true;
|
|
|
|
} catch (Exception e) {
|
|
|
|
System.out.println("Failed: unexpected exception:");
|
|
|
|
e.printStackTrace(System.out);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static boolean expectFailure(String message, Test test,
|
2018-04-17 08:54:17 -07:00
|
|
|
Class<? extends Throwable> expectedException) {
|
2016-05-18 16:39:08 +01:00
|
|
|
|
|
|
|
System.out.println(message);
|
|
|
|
try {
|
|
|
|
test.run();
|
|
|
|
System.out.println("Failed: unexpected successful connection");
|
|
|
|
return false;
|
|
|
|
} catch (Exception e) {
|
|
|
|
System.out.println("Got an exception:");
|
|
|
|
e.printStackTrace(System.out);
|
|
|
|
if (expectedException != null
|
|
|
|
&& !expectedException.isAssignableFrom(e.getClass())) {
|
|
|
|
System.out.printf("Failed: expected %s, but got %s%n",
|
|
|
|
expectedException.getName(),
|
|
|
|
e.getClass().getName());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
System.out.println("Passed: expected exception");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|