|
1 | 1 | /*
|
2 |
| - * Copyright (c) 2001, 2010, Oracle and/or its affiliates. All rights reserved. |
| 2 | + * Copyright (c) 2001, 2021, Oracle and/or its affiliates. All rights reserved. |
3 | 3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
4 | 4 | *
|
5 | 5 | * This code is free software; you can redistribute it and/or modify it
|
|
23 | 23 |
|
24 | 24 | /* @test
|
25 | 25 | * @summary Test DatagramChannel's send and receive methods
|
26 |
| - * @author Mike McCloskey |
| 26 | + * @run testng/othervm/timeout=20 SRTest |
27 | 27 | */
|
28 | 28 |
|
29 | 29 | import java.io.*;
|
30 | 30 | import java.net.*;
|
31 | 31 | import java.nio.*;
|
32 | 32 | import java.nio.channels.*;
|
33 |
| -import java.nio.charset.*; |
| 33 | +import java.util.concurrent.CompletableFuture; |
| 34 | +import java.util.concurrent.CompletionException; |
| 35 | +import java.util.concurrent.ExecutorService; |
| 36 | +import java.util.concurrent.Executors; |
| 37 | +import java.util.stream.Stream; |
| 38 | +import static java.nio.charset.StandardCharsets.US_ASCII; |
34 | 39 |
|
| 40 | +import org.testng.annotations.*; |
35 | 41 |
|
36 | 42 | public class SRTest {
|
37 | 43 |
|
| 44 | + ExecutorService executorService; |
38 | 45 | static PrintStream log = System.err;
|
39 | 46 |
|
40 |
| - public static void main(String[] args) throws Exception { |
41 |
| - test(); |
| 47 | + static final String DATA_STRING = "hello"; |
| 48 | + |
| 49 | + @BeforeClass |
| 50 | + public void beforeClass() { |
| 51 | + executorService = Executors.newCachedThreadPool(); |
42 | 52 | }
|
43 | 53 |
|
44 |
| - static void test() throws Exception { |
45 |
| - ClassicReader classicReader; |
46 |
| - NioReader nioReader; |
| 54 | + @AfterClass |
| 55 | + public void afterClass() { |
| 56 | + executorService.shutdown(); |
| 57 | + } |
47 | 58 |
|
48 |
| - classicReader = new ClassicReader(); |
49 |
| - invoke(classicReader, new ClassicWriter(classicReader.port())); |
| 59 | + @Test |
| 60 | + public void classicReaderClassicWriter() throws Exception { |
| 61 | + try (ClassicReader cr = new ClassicReader(); |
| 62 | + ClassicWriter cw = new ClassicWriter(cr.port())) { |
| 63 | + invoke(executorService, cr, cw); |
| 64 | + } |
50 | 65 | log.println("Classic RW: OK");
|
| 66 | + } |
51 | 67 |
|
52 |
| - classicReader = new ClassicReader(); |
53 |
| - invoke(classicReader, new NioWriter(classicReader.port())); |
| 68 | + @Test |
| 69 | + public void classicReaderNioWriter() throws Exception { |
| 70 | + try (ClassicReader cr = new ClassicReader(); |
| 71 | + NioWriter nw = new NioWriter(cr.port())) { |
| 72 | + invoke(executorService, cr, nw); |
| 73 | + } |
54 | 74 | log.println("Classic R, Nio W: OK");
|
| 75 | + } |
55 | 76 |
|
56 |
| - nioReader = new NioReader(); |
57 |
| - invoke(nioReader, new ClassicWriter(nioReader.port())); |
| 77 | + @Test |
| 78 | + public void nioReaderClassicWriter() throws Exception { |
| 79 | + try (NioReader nr = new NioReader(); |
| 80 | + ClassicWriter cw = new ClassicWriter(nr.port())) { |
| 81 | + invoke(executorService, nr, cw); |
| 82 | + } |
58 | 83 | log.println("Classic W, Nio R: OK");
|
| 84 | + } |
59 | 85 |
|
60 |
| - nioReader = new NioReader(); |
61 |
| - invoke(nioReader, new NioWriter(nioReader.port())); |
| 86 | + @Test |
| 87 | + public void nioReaderNioWriter() throws Exception { |
| 88 | + try (NioReader nr = new NioReader(); |
| 89 | + NioWriter nw = new NioWriter(nr.port())) { |
| 90 | + invoke(executorService, nr, nw); |
| 91 | + } |
62 | 92 | log.println("Nio RW: OK");
|
63 | 93 | }
|
64 | 94 |
|
65 |
| - static void invoke(Sprintable reader, Sprintable writer) throws Exception { |
66 |
| - Thread readerThread = new Thread(reader); |
67 |
| - readerThread.start(); |
68 |
| - Thread.sleep(50); |
69 |
| - |
70 |
| - Thread writerThread = new Thread(writer); |
71 |
| - writerThread.start(); |
72 |
| - |
73 |
| - writerThread.join(); |
74 |
| - readerThread.join(); |
75 |
| - |
76 |
| - reader.throwException(); |
77 |
| - writer.throwException(); |
| 95 | + private static void invoke(ExecutorService e, Runnable reader, Runnable writer) { |
| 96 | + CompletableFuture<Void> f1 = CompletableFuture.runAsync(writer, e); |
| 97 | + CompletableFuture<Void> f2 = CompletableFuture.runAsync(reader, e); |
| 98 | + wait(f1, f2); |
78 | 99 | }
|
79 | 100 |
|
80 |
| - public interface Sprintable extends Runnable { |
81 |
| - public void throwException() throws Exception; |
| 101 | + // Exit with CompletionException if any passed futures complete exceptionally |
| 102 | + private static void wait(CompletableFuture<?>... futures) throws CompletionException { |
| 103 | + CompletableFuture<?> future = CompletableFuture.allOf(futures); |
| 104 | + Stream.of(futures) |
| 105 | + .forEach(f -> f.exceptionally(ex -> { |
| 106 | + future.completeExceptionally(ex); |
| 107 | + return null; |
| 108 | + })); |
| 109 | + future.join(); |
82 | 110 | }
|
83 | 111 |
|
84 |
| - public static class ClassicWriter implements Sprintable { |
85 |
| - final int port; |
86 |
| - Exception e = null; |
87 |
| - |
88 |
| - ClassicWriter(int port) { |
89 |
| - this.port = port; |
90 |
| - } |
| 112 | + public static class ClassicWriter implements Runnable, AutoCloseable { |
| 113 | + final DatagramSocket ds; |
| 114 | + final int dstPort; |
91 | 115 |
|
92 |
| - public void throwException() throws Exception { |
93 |
| - if (e != null) |
94 |
| - throw e; |
| 116 | + ClassicWriter(int dstPort) throws SocketException { |
| 117 | + this.dstPort = dstPort; |
| 118 | + this.ds = new DatagramSocket(); |
95 | 119 | }
|
96 | 120 |
|
97 | 121 | public void run() {
|
98 | 122 | try {
|
99 |
| - DatagramSocket ds = new DatagramSocket(); |
100 |
| - String dataString = "hello"; |
101 |
| - byte[] data = dataString.getBytes(); |
102 |
| - InetAddress address = InetAddress.getLocalHost(); |
| 123 | + byte[] data = DATA_STRING.getBytes(US_ASCII); |
| 124 | + InetAddress address = InetAddress.getLoopbackAddress(); |
103 | 125 | DatagramPacket dp = new DatagramPacket(data, data.length,
|
104 |
| - address, port); |
| 126 | + address, dstPort); |
105 | 127 | ds.send(dp);
|
106 |
| - Thread.sleep(50); |
107 |
| - ds.send(dp); |
108 |
| - } catch (Exception ex) { |
109 |
| - e = ex; |
| 128 | + } catch (Exception e) { |
| 129 | + log.println("ClassicWriter [" + ds.getLocalAddress() + "]"); |
| 130 | + throw new RuntimeException("ClassicWriter threw exception: " + e); |
| 131 | + } finally { |
| 132 | + log.println("ClassicWriter finished"); |
110 | 133 | }
|
111 | 134 | }
|
112 |
| - } |
113 |
| - |
114 |
| - public static class NioWriter implements Sprintable { |
115 |
| - final int port; |
116 |
| - Exception e = null; |
117 | 135 |
|
118 |
| - NioWriter(int port) { |
119 |
| - this.port = port; |
| 136 | + @Override |
| 137 | + public void close() throws IOException { |
| 138 | + ds.close(); |
120 | 139 | }
|
| 140 | + } |
| 141 | + |
| 142 | + public static class NioWriter implements Runnable, AutoCloseable { |
| 143 | + final DatagramChannel dc; |
| 144 | + final int dstPort; |
121 | 145 |
|
122 |
| - public void throwException() throws Exception { |
123 |
| - if (e != null) |
124 |
| - throw e; |
| 146 | + NioWriter(int dstPort) throws IOException { |
| 147 | + this.dc = DatagramChannel.open(); |
| 148 | + this.dstPort = dstPort; |
125 | 149 | }
|
126 | 150 |
|
127 | 151 | public void run() {
|
128 | 152 | try {
|
129 |
| - DatagramChannel dc = DatagramChannel.open(); |
130 | 153 | ByteBuffer bb = ByteBuffer.allocateDirect(256);
|
131 |
| - bb.put("hello".getBytes()); |
| 154 | + bb.put(DATA_STRING.getBytes(US_ASCII)); |
132 | 155 | bb.flip();
|
133 |
| - InetAddress address = InetAddress.getLocalHost(); |
134 |
| - InetSocketAddress isa = new InetSocketAddress(address, port); |
135 |
| - dc.send(bb, isa); |
136 |
| - Thread.sleep(50); |
| 156 | + InetAddress address = InetAddress.getLoopbackAddress(); |
| 157 | + InetSocketAddress isa = new InetSocketAddress(address, dstPort); |
137 | 158 | dc.send(bb, isa);
|
138 | 159 | } catch (Exception ex) {
|
139 |
| - e = ex; |
| 160 | + log.println("NioWriter [" + dc.socket().getLocalAddress() + "]"); |
| 161 | + throw new RuntimeException("NioWriter threw exception: " + ex); |
| 162 | + } finally { |
| 163 | + log.println("NioWriter finished"); |
140 | 164 | }
|
141 | 165 | }
|
| 166 | + |
| 167 | + @Override |
| 168 | + public void close() throws IOException { |
| 169 | + dc.close(); |
| 170 | + } |
142 | 171 | }
|
143 | 172 |
|
144 |
| - public static class ClassicReader implements Sprintable { |
| 173 | + public static class ClassicReader implements Runnable, AutoCloseable { |
145 | 174 | final DatagramSocket ds;
|
146 |
| - Exception e = null; |
147 | 175 |
|
148 | 176 | ClassicReader() throws IOException {
|
149 |
| - this.ds = new DatagramSocket(); |
| 177 | + InetSocketAddress address = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0); |
| 178 | + this.ds = new DatagramSocket(address); |
150 | 179 | }
|
151 | 180 |
|
152 | 181 | int port() {
|
153 | 182 | return ds.getLocalPort();
|
154 | 183 | }
|
155 | 184 |
|
156 |
| - public void throwException() throws Exception { |
157 |
| - if (e != null) |
158 |
| - throw e; |
159 |
| - } |
160 |
| - |
161 | 185 | public void run() {
|
162 | 186 | try {
|
163 | 187 | byte[] buf = new byte[256];
|
164 | 188 | DatagramPacket dp = new DatagramPacket(buf, buf.length);
|
165 | 189 | ds.receive(dp);
|
166 |
| - String received = new String(dp.getData()); |
167 |
| - log.println(received); |
168 |
| - ds.close(); |
| 190 | + String received = new String(dp.getData(), dp.getOffset(), dp.getLength(), US_ASCII); |
| 191 | + log.println("ClassicReader received: " + received); |
169 | 192 | } catch (Exception ex) {
|
170 |
| - e = ex; |
| 193 | + log.println("ClassicReader [" + ds.getLocalAddress() +"]"); |
| 194 | + throw new RuntimeException("ClassicReader threw exception: " + ex); |
| 195 | + } finally { |
| 196 | + log.println("ClassicReader finished"); |
171 | 197 | }
|
172 | 198 | }
|
| 199 | + |
| 200 | + @Override |
| 201 | + public void close() throws IOException { |
| 202 | + ds.close(); |
| 203 | + } |
173 | 204 | }
|
174 | 205 |
|
175 |
| - public static class NioReader implements Sprintable { |
| 206 | + public static class NioReader implements Runnable, AutoCloseable { |
176 | 207 | final DatagramChannel dc;
|
177 |
| - Exception e = null; |
178 | 208 |
|
179 | 209 | NioReader() throws IOException {
|
180 |
| - this.dc = DatagramChannel.open().bind(new InetSocketAddress(0)); |
| 210 | + InetSocketAddress address = new InetSocketAddress(InetAddress.getLoopbackAddress(), 0); |
| 211 | + this.dc = DatagramChannel.open().bind(address); |
181 | 212 | }
|
182 | 213 |
|
183 | 214 | int port() {
|
184 | 215 | return dc.socket().getLocalPort();
|
185 | 216 | }
|
186 | 217 |
|
187 |
| - public void throwException() throws Exception { |
188 |
| - if (e != null) |
189 |
| - throw e; |
190 |
| - } |
191 |
| - |
192 | 218 | public void run() {
|
193 | 219 | try {
|
194 | 220 | ByteBuffer bb = ByteBuffer.allocateDirect(100);
|
195 |
| - SocketAddress sa = dc.receive(bb); |
| 221 | + dc.receive(bb); |
196 | 222 | bb.flip();
|
197 |
| - CharBuffer cb = Charset.forName("US-ASCII"). |
198 |
| - newDecoder().decode(bb); |
199 |
| - log.println("From: "+sa+ " said " +cb); |
200 |
| - dc.close(); |
| 223 | + CharBuffer cb = US_ASCII.newDecoder().decode(bb); |
| 224 | + log.println("NioReader received: " + cb); |
201 | 225 | } catch (Exception ex) {
|
202 |
| - e = ex; |
| 226 | + log.println("NioReader [" + dc.socket().getLocalAddress() +"]"); |
| 227 | + throw new RuntimeException("NioReader threw exception: " + ex); |
| 228 | + } finally { |
| 229 | + log.println("NioReader finished"); |
203 | 230 | }
|
204 | 231 | }
|
205 |
| - } |
206 | 232 |
|
| 233 | + @Override |
| 234 | + public void close() throws IOException { |
| 235 | + dc.close(); |
| 236 | + } |
| 237 | + } |
207 | 238 | }
|
0 commit comments