jdk-24/test/jdk/java/net/ServerSocket/ImplAccept.java

294 lines
9.8 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2023, 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 8278326
* @modules java.base/java.net:+open
* @run junit ImplAccept
* @summary Test ServerSocket.implAccept with a Socket in different states.
*/
import java.io.FileDescriptor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.InetAddress;
import java.net.InetSocketAddress;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.SocketImpl;
import java.net.SocketOption;
import java.net.StandardSocketOptions;
import java.io.IOException;
import org.junit.*;
import static org.junit.jupiter.api.Assertions.*;
public class ImplAccept {
/**
* Test ServerSocket.implAccept with an unbound Socket.
*/
@Test
public void testUnbound() throws Exception {
try (Socket socket = new Socket()) {
// Socket.impl -> DelegatingSocketImpl
SocketImpl si = getSocketImpl(socket);
assertTrue(isDelegatingSocketImpl(si));
try (ServerSocket ss = serverSocketToAccept(socket);
Socket peer = new Socket(ss.getInetAddress(), ss.getLocalPort())) {
Socket s = ss.accept();
assertTrue(s == socket);
// Socket.impl should be replaced with a new PlatformSocketImpl
SocketImpl psi = getSocketImpl(socket);
assertTrue(isPlatformSocketImpl(psi));
// socket and peer should be connected to each other
pingPong(socket, peer);
}
}
}
/**
* Test ServerSocket.implAccept with a bound Socket. The usage is nonsensical
* but we can test that the accepted Socket is connected and the underlying
* socket from the original SocketImpl is closed.
*/
@Test
public void testBound() throws Exception {
try (Socket socket = new Socket()) {
// Socket.impl -> DelegatingSocketImpl -> PlatformSocketImpl
SocketImpl si = getSocketImpl(socket);
SocketImpl psi1 = getDelegate(si);
assertTrue(isPlatformSocketImpl(psi1));
// bind to local address
socket.bind(loopbackSocketAddress());
assertTrue(isSocketOpen(psi1));
try (ServerSocket ss = serverSocketToAccept(socket);
Socket peer = new Socket(ss.getInetAddress(), ss.getLocalPort())) {
Socket s = ss.accept();
assertTrue(s == socket);
// Socket.impl should be replaced with a new PlatformSocketImpl
SocketImpl psi2 = getSocketImpl(socket);
assertTrue(isPlatformSocketImpl(psi2));
// psi1 should be closed
assertFalse(isSocketOpen(psi1));
// socket and peer should be connected to each other
pingPong(socket, peer);
}
}
}
/**
* Test ServerSocket.implAccept with a connected Socket. The usage is nonsensical
* but we can test that the accepted Socket is connected and the underlying
* socket from the original SocketImpl is closed.
*/
@Test
public void testConnected() throws Exception {
Socket socket;
Socket peer1;
try (ServerSocket ss = new ServerSocket()) {
ss.bind(loopbackSocketAddress());
socket = new Socket(ss.getInetAddress(), ss.getLocalPort());
peer1 = ss.accept();
}
try {
// Socket.impl -> DelegatingSocketImpl -> PlatformSocketImpl
SocketImpl si = getSocketImpl(socket);
SocketImpl psi1 = getDelegate(si);
assertTrue(isPlatformSocketImpl(psi1));
try (ServerSocket ss = serverSocketToAccept(socket);
Socket peer2 = new Socket(ss.getInetAddress(), ss.getLocalPort())) {
Socket s = ss.accept();
assertTrue(s == socket);
// Socket.impl should be replaced with a new PlatformSocketImpl
SocketImpl psi2 = getSocketImpl(socket);
assertTrue(isPlatformSocketImpl(psi2));
// psi1 should be closed and peer1 should read EOF
assertFalse(isSocketOpen(psi1));
assertTrue(peer1.getInputStream().read() == -1);
// socket and peer2 should be connected to each other
pingPong(socket, peer2);
}
} finally {
socket.close();
peer1.close();
}
}
/**
* Test ServerSocket.implAccept with a closed Socket. The usage is nonsensical
* but we can test ServerSocket.accept throws and that it closes the connection
* to the peer.
*/
@Test
public void testClosed() throws Exception {
Socket socket = new Socket();
socket.close();
try (ServerSocket ss = serverSocketToAccept(socket);
Socket peer = new Socket(ss.getInetAddress(), ss.getLocalPort())) {
SocketImpl si = getSocketImpl(socket);
// accept should throw and peer should read EOF
assertThrows(IOException.class, ss::accept);
assertTrue(peer.getInputStream().read() == -1);
// the SocketImpl should have not changed
assertTrue(getSocketImpl(socket) == si);
}
}
/**
* Returns the socket's SocketImpl.
*/
private static SocketImpl getSocketImpl(Socket s) {
try {
Field f = Socket.class.getDeclaredField("impl");
f.setAccessible(true);
return (SocketImpl) f.get(s);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Returns the SocketImpl that the given SocketImpl delegates to.
*/
private static SocketImpl getDelegate(SocketImpl si) {
try {
Class<?> clazz = Class.forName("java.net.DelegatingSocketImpl");
Field f = clazz.getDeclaredField("delegate");
f.setAccessible(true);
return (SocketImpl) f.get(si);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Returns true if the SocketImpl is a DelegatingSocketImpl.
*/
private static boolean isDelegatingSocketImpl(SocketImpl si) {
try {
Class<?> clazz = Class.forName("java.net.DelegatingSocketImpl");
return clazz.isInstance(si);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Returns true if the SocketImpl is a PlatformSocketImpl.
*/
private static boolean isPlatformSocketImpl(SocketImpl si) {
try {
Class<?> clazz = Class.forName("sun.net.PlatformSocketImpl");
return clazz.isInstance(si);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
/**
* Returns true if the SocketImpl has an open socket.
*/
private static boolean isSocketOpen(SocketImpl si) throws Exception {
assertTrue(isPlatformSocketImpl(si));
// check if SocketImpl.fd is set
Field f = SocketImpl.class.getDeclaredField("fd");
f.setAccessible(true);
FileDescriptor fd = (FileDescriptor) f.get(si);
if (fd == null) {
return false; // not created
}
// call getOption to get the value of the SO_REUSEADDR socket option
Method m = SocketImpl.class.getDeclaredMethod("getOption", SocketOption.class);
m.setAccessible(true);
try {
m.invoke(si, StandardSocketOptions.SO_REUSEADDR);
return true; // socket is open
} catch (InvocationTargetException e) {
if (e.getCause() instanceof IOException) {
return false; // assume socket is closed
}
throw e;
}
}
/**
* Test that two sockets are connected to each other.
*/
private static void pingPong(Socket s1, Socket s2) throws Exception {
s1.getOutputStream().write(11);
s2.getOutputStream().write(22);
assertTrue(s1.getInputStream().read() == 22);
assertTrue(s2.getInputStream().read() == 11);
}
/**
* Creates a ServerSocket that returns the given Socket from accept.
*/
private static ServerSocket serverSocketToAccept(Socket s) throws IOException {
ServerSocket ss = new ServerSocket() {
@Override
public Socket accept() throws IOException {
implAccept(s);
return s;
}
};
ss.bind(loopbackSocketAddress());
return ss;
}
/**
* Returns a new InetSocketAddress with the loopback interface and port 0.
*/
private static InetSocketAddress loopbackSocketAddress() {
InetAddress loopback = InetAddress.getLoopbackAddress();
return new InetSocketAddress(loopback, 0);
}
}