Skip to content

Commit a690d3d

Browse files
committed
common: Allow ServerSocketChannel.bind to null/temp address if possible
JEP380 allows binding to null address, which creates a temporary address with a /tmp/socket_NNNNNNN address. For junixsocket AF_UNIX sockets, use a regular temporary address in this case.
1 parent 262827d commit a690d3d

File tree

9 files changed

+136
-9
lines changed

9 files changed

+136
-9
lines changed

junixsocket-common/src/main/java/org/newsclub/net/unix/AFAddressFamily.java

+11
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919

2020
import java.io.IOException;
2121
import java.lang.reflect.InvocationTargetException;
22+
import java.net.SocketAddress;
2223
import java.net.SocketException;
2324
import java.net.URI;
2425
import java.nio.channels.ServerSocketChannel;
@@ -367,4 +368,14 @@ public synchronized SelectorProvider getSelectorProvider() {
367368
}
368369
return selectorProvider;
369370
}
371+
372+
/**
373+
* Returns an appropriate SocketAddress to be used when calling bind with a null argument.
374+
*
375+
* @return The new socket address, or {@code null}.
376+
* @throws IOException on error.
377+
*/
378+
public SocketAddress nullBindAddress() throws IOException {
379+
return addressConfig.nullBindAddress();
380+
}
370381
}

junixsocket-common/src/main/java/org/newsclub/net/unix/AFServerSocket.java

+5-5
Original file line numberDiff line numberDiff line change
@@ -240,6 +240,11 @@ public final AFServerSocket<A> forceBindAddress(SocketAddress endpoint) {
240240
});
241241
}
242242

243+
@Override
244+
public final void bind(SocketAddress endpoint) throws IOException {
245+
bind(endpoint, 50);
246+
}
247+
243248
@SuppressWarnings("unchecked")
244249
@Override
245250
public final void bind(SocketAddress endpoint, int backlog) throws IOException {
@@ -550,11 +555,6 @@ public final AFServerSocket<A> bindHook(SocketAddressFilter hook) {
550555
return this;
551556
}
552557

553-
@Override
554-
public void bind(SocketAddress endpoint) throws IOException {
555-
bind(endpoint, 50);
556-
}
557-
558558
@Override
559559
public InetAddress getInetAddress() {
560560
if (!isBound()) {

junixsocket-common/src/main/java/org/newsclub/net/unix/AFSocketAddressConfig.java

+12
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717
*/
1818
package org.newsclub.net.unix;
1919

20+
import java.io.IOException;
21+
import java.net.SocketAddress;
2022
import java.net.SocketException;
2123
import java.net.URI;
2224
import java.util.Set;
@@ -70,4 +72,14 @@ protected AFSocketAddressConfig() {
7072
* @return The set of supported URI schemes.
7173
*/
7274
protected abstract Set<String> uriSchemes();
75+
76+
/**
77+
* Returns an appropriate SocketAddress to be used when calling bind with a null argument.
78+
*
79+
* @return The new socket address, or {@code null}.
80+
* @throws IOException on error.
81+
*/
82+
protected SocketAddress nullBindAddress() throws IOException {
83+
return null;
84+
}
7385
}

junixsocket-common/src/main/java/org/newsclub/net/unix/AFSocketImpl.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -386,7 +386,10 @@ protected final int available() throws IOException {
386386

387387
final void bind(SocketAddress addr, int options) throws IOException {
388388
if (addr == null) {
389-
throw new IllegalArgumentException("Cannot bind to null address");
389+
addr = addressFamily.nullBindAddress();
390+
if (addr == null) {
391+
throw new UnsupportedOperationException("Cannot bind to null address");
392+
}
390393
}
391394

392395
if (addr == AFSocketAddress.INTERNAL_DUMMY_BIND) { // NOPMD

junixsocket-common/src/main/java/org/newsclub/net/unix/AFUNIXSocketAddress.java

+5
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,11 @@ protected String selectorProviderClassname() {
8181
protected Set<String> uriSchemes() {
8282
return new HashSet<>(Arrays.asList("unix", "http+unix", "https+unix"));
8383
}
84+
85+
@Override
86+
protected SocketAddress nullBindAddress() throws IOException {
87+
return AFUNIXSocketAddress.ofNewTempFile();
88+
}
8489
});
8590

8691
private AFUNIXSocketAddress(int port, final byte[] socketAddress, Lease<ByteBuffer> nativeAddress)

junixsocket-common/src/test/java/org/newsclub/net/unix/ServerSocketTest.java

+5-3
Original file line numberDiff line numberDiff line change
@@ -80,10 +80,12 @@ public void close() throws IOException {
8080
public void testBindBadArguments() throws Exception {
8181
try (ServerSocket sock = newServerSocket()) {
8282
assertFalse(sock.isBound());
83-
assertThrows(IllegalArgumentException.class, () -> {
83+
84+
try {
8485
sock.bind(null);
85-
});
86-
assertFalse(sock.isBound());
86+
} catch (UnsupportedOperationException e) {
87+
assertFalse(sock.isBound());
88+
}
8789
}
8890
try (ServerSocket sock = newServerSocket()) {
8991
assertFalse(sock.isBound());

junixsocket-common/src/test/java/org/newsclub/net/unix/SocketChannelTest.java

+47
Original file line numberDiff line numberDiff line change
@@ -19,11 +19,13 @@
1919

2020
import static org.junit.jupiter.api.Assertions.assertEquals;
2121
import static org.junit.jupiter.api.Assertions.assertFalse;
22+
import static org.junit.jupiter.api.Assertions.assertNotNull;
2223
import static org.junit.jupiter.api.Assertions.assertThrows;
2324
import static org.junit.jupiter.api.Assertions.assertTrue;
2425
import static org.junit.jupiter.api.Assertions.fail;
2526

2627
import java.io.IOException;
28+
import java.net.ServerSocket;
2729
import java.net.SocketAddress;
2830
import java.net.SocketException;
2931
import java.net.SocketTimeoutException;
@@ -432,4 +434,49 @@ public void testAcceptNotBoundYet() throws Exception {
432434
ServerSocketChannel sc = newServerSocketChannel();
433435
assertThrows(NotYetBoundException.class, sc::accept);
434436
}
437+
438+
protected boolean mayTestBindNullThrowUnsupportedOperationException() {
439+
return true;
440+
}
441+
442+
protected boolean mayTestBindNullHaveNullLocalSocketAddress() {
443+
return true;
444+
}
445+
446+
protected void cleanupTestBindNull(ServerSocketChannel sc, SocketAddress addr) throws Exception {
447+
}
448+
449+
protected ServerSocket socketIfPossible(ServerSocketChannel channel) {
450+
try {
451+
return channel.socket();
452+
} catch (UnsupportedOperationException e) {
453+
return null;
454+
}
455+
}
456+
457+
@Test
458+
public void testBindNull() throws Exception {
459+
try (ServerSocketChannel sc = newServerSocketChannel()) {
460+
ServerSocket s = socketIfPossible(sc);
461+
assertTrue(s == null || !s.isBound());
462+
try {
463+
sc.bind(null);
464+
} catch (UnsupportedOperationException e) {
465+
if (mayTestBindNullThrowUnsupportedOperationException()) {
466+
// OK
467+
return;
468+
} else {
469+
throw e;
470+
}
471+
}
472+
assertTrue(s == null || s.isBound());
473+
474+
SocketAddress addr = sc.getLocalAddress();
475+
if (!mayTestBindNullHaveNullLocalSocketAddress()) {
476+
assertNotNull(addr);
477+
}
478+
479+
cleanupTestBindNull(sc, addr);
480+
}
481+
}
435482
}

junixsocket-common/src/test/java/org/newsclub/net/unix/domain/SocketChannelTest.java

+16
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@
2020
import static org.junit.jupiter.api.Assertions.assertEquals;
2121

2222
import java.net.ProtocolFamily;
23+
import java.net.SocketAddress;
2324
import java.nio.channels.DatagramChannel;
2425
import java.nio.channels.ServerSocketChannel;
2526
import java.nio.channels.SocketChannel;
@@ -75,4 +76,19 @@ public void testUnixDomainProtocolFamily() throws Exception {
7576
assertEquals(AFUNIXDatagramChannel.class, ch.getClass());
7677
}
7778
}
79+
80+
@Override
81+
protected boolean mayTestBindNullThrowUnsupportedOperationException() {
82+
return false;
83+
}
84+
85+
@Override
86+
protected boolean mayTestBindNullHaveNullLocalSocketAddress() {
87+
return false;
88+
}
89+
90+
@Override
91+
protected void cleanupTestBindNull(ServerSocketChannel sc, SocketAddress addr) throws Exception {
92+
// nothing to do, as -- unlike JEP380 -- junixsocket cleans up its mess
93+
}
7894
}

junixsocket-common/src/test/java/org/newsclub/net/unix/jep380/SocketChannelTest.java

+31
Original file line numberDiff line numberDiff line change
@@ -17,19 +17,50 @@
1717
*/
1818
package org.newsclub.net.unix.jep380;
1919

20+
import java.io.IOException;
2021
import java.net.SocketAddress;
22+
import java.nio.channels.ServerSocketChannel;
23+
import java.nio.file.Files;
24+
import java.nio.file.Path;
25+
import java.nio.file.Paths;
2126

2227
import org.newsclub.net.unix.AFSocketCapability;
2328
import org.newsclub.net.unix.AFSocketCapabilityRequirement;
2429

2530
import com.kohlschutter.annotations.compiletime.SuppressFBWarnings;
31+
import com.kohlschutter.testutil.AvailabilityRequirement;
2632

2733
@AFSocketCapabilityRequirement(AFSocketCapability.CAPABILITY_UNIX_DOMAIN)
34+
@AvailabilityRequirement(classes = "java.net.UnixDomainSocketAddress", //
35+
message = "This test requires Java 16 or later")
2836
@SuppressFBWarnings("NM_SAME_SIMPLE_NAME_AS_SUPERCLASS")
2937
public final class SocketChannelTest extends
3038
org.newsclub.net.unix.SocketChannelTest<SocketAddress> {
3139

3240
public SocketChannelTest() {
3341
super(JEP380AddressSpecifics.INSTANCE);
3442
}
43+
44+
@Override
45+
protected boolean mayTestBindNullThrowUnsupportedOperationException() {
46+
return false;
47+
}
48+
49+
@Override
50+
protected boolean mayTestBindNullHaveNullLocalSocketAddress() {
51+
return false;
52+
}
53+
54+
@Override
55+
protected void cleanupTestBindNull(ServerSocketChannel sc, SocketAddress addr)
56+
throws ClassNotFoundException, IOException {
57+
if (!Class.forName("java.net.UnixDomainSocketAddress").isAssignableFrom(addr.getClass())) {
58+
return;
59+
}
60+
61+
// JEP380 doesn't clean up socket files
62+
Path p = Paths.get(addr.toString());
63+
Files.delete(p);
64+
}
65+
3566
}

0 commit comments

Comments
 (0)