6e04e7659a
Uses a longer linger time and avoids closing socket too early. Reviewed-by: chegar
203 lines
7.8 KiB
Java
203 lines
7.8 KiB
Java
/*
|
|
* Copyright (c) 2018, 2019, 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.
|
|
*/
|
|
import java.io.InputStream;
|
|
import java.io.OutputStream;
|
|
import java.net.InetSocketAddress;
|
|
import java.net.ServerSocket;
|
|
import java.net.Socket;
|
|
import java.net.URI;
|
|
import java.net.SocketTimeoutException;
|
|
import java.time.Duration;
|
|
import javax.net.ssl.SSLServerSocketFactory;
|
|
import javax.net.ServerSocketFactory;
|
|
import javax.net.ssl.SSLContext;
|
|
import javax.net.ssl.SSLParameters;
|
|
import java.net.http.HttpClient;
|
|
import java.net.http.HttpRequest;
|
|
import java.net.http.HttpResponse;
|
|
import java.util.List;
|
|
import java.util.concurrent.CopyOnWriteArrayList;
|
|
|
|
import jdk.test.lib.net.SimpleSSLContext;
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8207966
|
|
* @library /test/lib
|
|
* @build jdk.test.lib.net.SimpleSSLContext
|
|
* @run main/othervm -Djdk.httpclient.enableAllMethodRetry
|
|
* -Djdk.tls.acknowledgeCloseNotify=true UnknownBodyLengthTest plain false
|
|
* @run main/othervm -Djdk.httpclient.enableAllMethodRetry
|
|
* -Djdk.tls.acknowledgeCloseNotify=true UnknownBodyLengthTest SSL false
|
|
* @run main/othervm -Djdk.httpclient.enableAllMethodRetry
|
|
* -Djdk.tls.acknowledgeCloseNotify=true UnknownBodyLengthTest plain true
|
|
* @run main/othervm -Djdk.httpclient.enableAllMethodRetry
|
|
* -Djdk.tls.acknowledgeCloseNotify=true UnknownBodyLengthTest SSL true
|
|
*/
|
|
|
|
public class UnknownBodyLengthTest {
|
|
static final byte[] BUF = new byte[32 * 10234 + 2];
|
|
|
|
volatile SSLContext ctx;
|
|
volatile ServerSocketFactory factory;
|
|
volatile String clientURL;
|
|
volatile int port;
|
|
final ServerSocket ss;
|
|
final List<Socket> acceptedList = new CopyOnWriteArrayList<>();
|
|
|
|
UnknownBodyLengthTest(boolean useSSL) throws Exception {
|
|
ctx = new SimpleSSLContext().get();
|
|
SSLContext.setDefault(ctx);
|
|
factory = useSSL ? SSLServerSocketFactory.getDefault()
|
|
: ServerSocketFactory.getDefault();
|
|
ss = factory.createServerSocket();
|
|
ss.setReuseAddress(true);
|
|
ss.bind(new InetSocketAddress("127.0.0.1", 0));
|
|
System.out.println("ServerSocket = " + ss.getClass() + " " + ss);
|
|
port = ss.getLocalPort();
|
|
clientURL = (useSSL ? "https" : "http") + "://localhost:"
|
|
+ Integer.toString(port) + "/test";
|
|
}
|
|
|
|
static void fillBuf(byte[] buf) {
|
|
for (int i=0; i<buf.length; i++)
|
|
buf[i] = (byte)i;
|
|
}
|
|
|
|
static void checkBuf(byte[] buf) {
|
|
if (buf.length != BUF.length)
|
|
throw new RuntimeException("buffer lengths not the same");
|
|
for (int i=0; i<buf.length; i++)
|
|
if (buf[i] != BUF[i])
|
|
throw new RuntimeException("error at position " + i);
|
|
}
|
|
|
|
volatile boolean stopped;
|
|
|
|
void server(final boolean withContentLength) {
|
|
fillBuf(BUF);
|
|
try {
|
|
while (!stopped) {
|
|
try {
|
|
Socket s = ss.accept();
|
|
acceptedList.add(s);
|
|
s.setTcpNoDelay(true);
|
|
// if we use linger=1 we still see some
|
|
// intermittent failures caused by IOException
|
|
// "Connection reset by peer".
|
|
// The client side is expecting EOF, but gets reset instead.
|
|
// 30 is a 'magic' value that may need to be adjusted again.
|
|
s.setSoLinger(true, 30);
|
|
System.out.println("Accepted: " + s.getRemoteSocketAddress());
|
|
System.out.println("Accepted: " + s);
|
|
OutputStream os = s.getOutputStream();
|
|
InputStream is = s.getInputStream();
|
|
boolean done = false;
|
|
byte[] buf = new byte[1024];
|
|
String rsp = "";
|
|
while (!done) {
|
|
int c = is.read(buf);
|
|
if (c < 0) break;
|
|
String s1 = new String(buf, 0, c, "ISO-8859-1");
|
|
rsp += s1;
|
|
done = rsp.endsWith("!#!#");
|
|
}
|
|
String r = "HTTP/1.0 200 OK\r\nConnection: close\r\nContent-Type:" +
|
|
" text/xml; charset=UTF-8\r\n";
|
|
os.write(r.getBytes());
|
|
String chdr = "Content-Length: " + Integer.toString(BUF.length) +
|
|
"\r\n";
|
|
System.out.println(chdr);
|
|
if (withContentLength)
|
|
os.write(chdr.getBytes());
|
|
os.write("\r\n".getBytes());
|
|
os.write(BUF);
|
|
if (is.available() > 0) {
|
|
System.out.println("Draining input: " + s);
|
|
is.read(buf);
|
|
}
|
|
os.flush();
|
|
s.shutdownOutput();
|
|
System.out.println("Closed output: " + s);
|
|
} catch(Exception e) {
|
|
if (!stopped) {
|
|
System.out.println("Unexpected server exception: " + e);
|
|
e.printStackTrace();
|
|
}
|
|
}
|
|
}
|
|
} catch(final Throwable t) {
|
|
if (!stopped) t.printStackTrace();
|
|
} finally {
|
|
stop();
|
|
}
|
|
}
|
|
|
|
void client(boolean useSSL) throws Exception {
|
|
SSLContext ctx = SSLContext.getDefault();
|
|
HttpClient.Builder clientB = HttpClient.newBuilder()
|
|
.version(HttpClient.Version.HTTP_2);
|
|
if (useSSL) {
|
|
clientB = clientB.sslContext(ctx)
|
|
.sslParameters(ctx.getSupportedSSLParameters());
|
|
}
|
|
final HttpClient client = clientB.build();
|
|
|
|
System.out.println("URL: " + clientURL);
|
|
final HttpResponse<byte[]> response = client
|
|
.send(HttpRequest
|
|
.newBuilder(new URI(clientURL))
|
|
.timeout(Duration.ofMillis(120_000))
|
|
.POST(HttpRequest.BodyPublishers.ofString("body!#!#"))
|
|
.build(), HttpResponse.BodyHandlers.ofByteArray());
|
|
|
|
System.out.println("Received reply: " + response.statusCode());
|
|
byte[] bb = response.body();
|
|
checkBuf(bb);
|
|
}
|
|
|
|
public static void main(final String[] args) throws Exception {
|
|
boolean ssl = args[0].equals("SSL");
|
|
boolean fixedlen = args[1].equals("true");
|
|
UnknownBodyLengthTest test = new UnknownBodyLengthTest(ssl);
|
|
try {
|
|
test.run(ssl, fixedlen);
|
|
} finally {
|
|
test.stop();
|
|
}
|
|
}
|
|
|
|
public void run(boolean ssl, boolean fixedlen) throws Exception {
|
|
new Thread(()->server(fixedlen)).start();
|
|
client(ssl);
|
|
}
|
|
|
|
public void stop() {
|
|
stopped = true;
|
|
try { ss.close(); } catch (Throwable t) { }
|
|
for (Socket s : acceptedList) {
|
|
try { s.close(); } catch (Throwable t) { }
|
|
}
|
|
}
|
|
}
|