22
22
import static org .junit .jupiter .api .Assertions .assertTrue ;
23
23
24
24
import java .io .IOException ;
25
+ import java .net .BindException ;
25
26
import java .net .ServerSocket ;
26
27
import java .net .Socket ;
27
28
import java .net .SocketAddress ;
33
34
import java .nio .channels .SocketChannel ;
34
35
import java .time .Duration ;
35
36
import java .time .temporal .ChronoUnit ;
37
+ import java .util .ArrayList ;
36
38
import java .util .Arrays ;
37
39
import java .util .List ;
38
40
import java .util .concurrent .CompletableFuture ;
@@ -73,28 +75,45 @@ public abstract class InterruptIssue158Test<A extends SocketAddress> extends Soc
73
75
private static boolean DELAY_CLOSE = SystemPropertyUtil .getBooleanSystemProperty (
74
76
"selftest.issue.158.delay-close" , true );
75
77
76
- private final A address ;
78
+ private A address = newAddress () ;
77
79
private TestInfo testInfo ;
80
+ private List <AutoCloseable > closeables = new ArrayList <>();
78
81
79
- @ SuppressWarnings ("unchecked" )
80
82
protected InterruptIssue158Test (AddressSpecifics <A > asp ) {
81
83
super (asp );
84
+ }
85
+
86
+ @ BeforeEach
87
+ public void beforeEach (TestInfo info ) {
88
+ this .testInfo = info ;
89
+ this .address = newAddress ();
90
+ }
82
91
92
+ @ SuppressWarnings ("unchecked" )
93
+ private A newAddress () {
83
94
try {
84
- address = (A ) newTempAddress ();
95
+ return (A ) newTempAddress ();
85
96
} catch (IOException e ) {
86
97
throw new RuntimeException (e );
87
98
}
88
99
}
89
100
90
- @ BeforeEach
91
- public void beforeEach (TestInfo info ) {
92
- this .testInfo = info ;
101
+ private void closeAfterTest () {
102
+ deleteSocketFile (address );
103
+
104
+ for (AutoCloseable cl : closeables ) {
105
+ try {
106
+ cl .close ();
107
+ } catch (Exception e ) {
108
+ // ignore
109
+ }
110
+ }
111
+ closeables .clear ();
93
112
}
94
113
95
114
@ AfterEach
96
115
public void afterEach () {
97
- deleteSocketFile ( address );
116
+ closeAfterTest ( );
98
117
}
99
118
100
119
protected abstract void deleteSocketFile (A sa );
@@ -119,7 +138,7 @@ public List<Arguments> clientProvider() {
119
138
120
139
public List <Arguments > serverProvider () {
121
140
return Arrays .asList ( //
122
- serverSocket (() -> newServerSocketBindOn (address ), ServerSocket ::accept ,
141
+ serverSocket (() -> registerCloseable ( newServerSocketBindOn (address ) ), ServerSocket ::accept ,
123
142
SocketException .class , ServerSocket ::isClosed ), //
124
143
serverSocket (this ::bindServerSocketChannel , ServerSocketChannel ::accept ,
125
144
ClosedByInterruptException .class , s -> !s .isOpen ())//
@@ -175,17 +194,30 @@ public <T extends AutoCloseable> void testSocketInterruption(boolean delay, IOSu
175
194
t .interrupt ();
176
195
t .join (Duration .of (1 , ChronoUnit .SECONDS ).toMillis ());
177
196
if (t .isAlive ()) {
178
- throw new RuntimeException ("Thread failed to terminate after interrupt" );
197
+ // Thread.interrupt is not guaranteed to succeed
198
+ // observed with graalvm-jdk-17.0.9+11.1 when building with agent for junixsocket-native-graalvm
199
+ // What we need to do here is to close all socket-related resources and try again
200
+ closeAfterTest ();
201
+ t .interrupt ();
202
+ t .join (Duration .of (1 , ChronoUnit .SECONDS ).toMillis ());
203
+ if (t .isAlive ()) {
204
+ throw new RuntimeException ("Thread failed to terminate after interrupt" );
205
+ }
179
206
}
180
207
Throwable thrownException = exceptionHolder .get ();
181
208
if (thrownException != null ) {
182
209
throw thrownException ;
183
210
}
184
211
}
185
212
213
+ private <C extends AutoCloseable > C registerCloseable (C closeable ) {
214
+ closeables .add (closeable );
215
+ return closeable ;
216
+ }
217
+
186
218
private void withServer (boolean acceptConnections , ThrowingRunnable func ) throws Throwable {
187
219
Semaphore done = new Semaphore (0 );
188
- try (ServerSocketChannel serverSocket = newServerSocketChannel ()) {
220
+ try (ServerSocketChannel serverSocket = registerCloseable ( newServerSocketChannel () )) {
189
221
serverSocket .bind (address );
190
222
Thread serverThread = null ;
191
223
if (acceptConnections ) {
@@ -249,7 +281,7 @@ <T extends AutoCloseable> Throwable runOperation(CountDownLatch ready, IOSupplie
249
281
Exception exc = null ;
250
282
try {
251
283
@ SuppressWarnings ({"resource" })
252
- T sock = socket .get ();
284
+ T sock = registerCloseable ( socket .get () );
253
285
ready .countDown ();
254
286
255
287
supported = true ;
@@ -314,14 +346,25 @@ private static <T> Arguments serverSocket(IOSupplier<T> socket, IOConsumer<T> bl
314
346
}
315
347
316
348
private SocketChannel connectSocketChannel () throws IOException {
317
- SocketChannel socket = newSocketChannel ();
349
+ SocketChannel socket = registerCloseable ( newSocketChannel () );
318
350
socket .connect (address );
319
351
return socket ;
320
352
}
321
353
322
354
private ServerSocketChannel bindServerSocketChannel () throws IOException {
323
- ServerSocketChannel socket = newServerSocketChannel ();
324
- socket .bind (address );
355
+ ServerSocketChannel socket = registerCloseable (newServerSocketChannel ());
356
+ try {
357
+ try {
358
+ socket .bind (address );
359
+ } catch (BindException e ) {
360
+ // With Inet sockets, our reserved address may just have been taken by another process
361
+ // so let's try again
362
+ address = newAddress ();
363
+ socket .bind (address );
364
+ }
365
+ } catch (BindException e ) {
366
+ throw (BindException ) new BindException (e .getMessage () + ": " + address ).initCause (e );
367
+ }
325
368
return socket ;
326
369
}
327
370
0 commit comments