1 // Copyright 2009 The Go Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
13 var tcpServerTests = []struct {
14 snet, saddr string // server endpoint
15 tnet, taddr string // target endpoint for client
17 {snet: "tcp", saddr: ":0", tnet: "tcp", taddr: "127.0.0.1"},
18 {snet: "tcp", saddr: "0.0.0.0:0", tnet: "tcp", taddr: "127.0.0.1"},
19 {snet: "tcp", saddr: "[::ffff:0.0.0.0]:0", tnet: "tcp", taddr: "127.0.0.1"},
20 {snet: "tcp", saddr: "[::]:0", tnet: "tcp", taddr: "::1"},
22 {snet: "tcp", saddr: ":0", tnet: "tcp", taddr: "::1"},
23 {snet: "tcp", saddr: "0.0.0.0:0", tnet: "tcp", taddr: "::1"},
24 {snet: "tcp", saddr: "[::ffff:0.0.0.0]:0", tnet: "tcp", taddr: "::1"},
25 {snet: "tcp", saddr: "[::]:0", tnet: "tcp", taddr: "127.0.0.1"},
27 {snet: "tcp", saddr: ":0", tnet: "tcp4", taddr: "127.0.0.1"},
28 {snet: "tcp", saddr: "0.0.0.0:0", tnet: "tcp4", taddr: "127.0.0.1"},
29 {snet: "tcp", saddr: "[::ffff:0.0.0.0]:0", tnet: "tcp4", taddr: "127.0.0.1"},
30 {snet: "tcp", saddr: "[::]:0", tnet: "tcp6", taddr: "::1"},
32 {snet: "tcp", saddr: ":0", tnet: "tcp6", taddr: "::1"},
33 {snet: "tcp", saddr: "0.0.0.0:0", tnet: "tcp6", taddr: "::1"},
34 {snet: "tcp", saddr: "[::ffff:0.0.0.0]:0", tnet: "tcp6", taddr: "::1"},
35 {snet: "tcp", saddr: "[::]:0", tnet: "tcp4", taddr: "127.0.0.1"},
37 {snet: "tcp", saddr: "127.0.0.1:0", tnet: "tcp", taddr: "127.0.0.1"},
38 {snet: "tcp", saddr: "[::ffff:127.0.0.1]:0", tnet: "tcp", taddr: "127.0.0.1"},
39 {snet: "tcp", saddr: "[::1]:0", tnet: "tcp", taddr: "::1"},
41 {snet: "tcp4", saddr: ":0", tnet: "tcp4", taddr: "127.0.0.1"},
42 {snet: "tcp4", saddr: "0.0.0.0:0", tnet: "tcp4", taddr: "127.0.0.1"},
43 {snet: "tcp4", saddr: "[::ffff:0.0.0.0]:0", tnet: "tcp4", taddr: "127.0.0.1"},
45 {snet: "tcp4", saddr: "127.0.0.1:0", tnet: "tcp4", taddr: "127.0.0.1"},
47 {snet: "tcp6", saddr: ":0", tnet: "tcp6", taddr: "::1"},
48 {snet: "tcp6", saddr: "[::]:0", tnet: "tcp6", taddr: "::1"},
50 {snet: "tcp6", saddr: "[::1]:0", tnet: "tcp6", taddr: "::1"},
53 // TestTCPServer tests concurrent accept-read-write servers.
54 func TestTCPServer(t *testing.T) {
57 for i, tt := range tcpServerTests {
58 t.Run(tt.snet+" "+tt.saddr+"<-"+tt.taddr, func(t *testing.T) {
59 if !testableListenArgs(tt.snet, tt.saddr, tt.taddr) {
60 t.Skip("not testable")
63 ln, err := Listen(tt.snet, tt.saddr)
65 if perr := parseDialError(err); perr != nil {
71 var lss []*localServer
72 var tpchs []chan error
74 for _, ls := range lss {
78 for i := 0; i < N; i++ {
79 ls := (&streamListener{Listener: ln}).newLocalServer()
81 tpchs = append(tpchs, make(chan error, 1))
83 for i := 0; i < N; i++ {
85 handler := func(ls *localServer, ln Listener) { ls.transponder(ln, ch) }
86 if err := lss[i].buildup(handler); err != nil {
91 var trchs []chan error
92 for i := 0; i < N; i++ {
93 _, port, err := SplitHostPort(lss[i].Listener.Addr().String())
97 d := Dialer{Timeout: someTimeout}
98 c, err := d.Dial(tt.tnet, JoinHostPort(tt.taddr, port))
100 if perr := parseDialError(err); perr != nil {
106 trchs = append(trchs, make(chan error, 1))
107 go transceiver(c, []byte("TCP SERVER TEST"), trchs[i])
110 for _, ch := range trchs {
111 for err := range ch {
112 t.Errorf("#%d: %v", i, err)
115 for _, ch := range tpchs {
116 for err := range ch {
117 t.Errorf("#%d: %v", i, err)
124 // TestUnixAndUnixpacketServer tests concurrent accept-read-write
126 func TestUnixAndUnixpacketServer(t *testing.T) {
127 var unixAndUnixpacketServerTests = []struct {
128 network, address string
130 {"unix", testUnixAddr(t)},
131 {"unix", "@nettest/go/unix"},
133 {"unixpacket", testUnixAddr(t)},
134 {"unixpacket", "@nettest/go/unixpacket"},
139 for i, tt := range unixAndUnixpacketServerTests {
140 if !testableListenArgs(tt.network, tt.address, "") {
141 t.Logf("skipping %s test", tt.network+" "+tt.address)
145 ln, err := Listen(tt.network, tt.address)
147 if perr := parseDialError(err); perr != nil {
153 var lss []*localServer
154 var tpchs []chan error
156 for _, ls := range lss {
160 for i := 0; i < N; i++ {
161 ls := (&streamListener{Listener: ln}).newLocalServer()
162 lss = append(lss, ls)
163 tpchs = append(tpchs, make(chan error, 1))
165 for i := 0; i < N; i++ {
167 handler := func(ls *localServer, ln Listener) { ls.transponder(ln, ch) }
168 if err := lss[i].buildup(handler); err != nil {
173 var trchs []chan error
174 for i := 0; i < N; i++ {
175 d := Dialer{Timeout: someTimeout}
176 c, err := d.Dial(lss[i].Listener.Addr().Network(), lss[i].Listener.Addr().String())
178 if perr := parseDialError(err); perr != nil {
184 if addr := c.LocalAddr(); addr != nil {
185 t.Logf("connected %s->%s", addr, lss[i].Listener.Addr())
189 trchs = append(trchs, make(chan error, 1))
190 go transceiver(c, []byte("UNIX AND UNIXPACKET SERVER TEST"), trchs[i])
193 for _, ch := range trchs {
194 for err := range ch {
195 t.Errorf("#%d: %v", i, err)
198 for _, ch := range tpchs {
199 for err := range ch {
200 t.Errorf("#%d: %v", i, err)
206 var udpServerTests = []struct {
207 snet, saddr string // server endpoint
208 tnet, taddr string // target endpoint for client
209 dial bool // test with Dial
211 {snet: "udp", saddr: ":0", tnet: "udp", taddr: "127.0.0.1"},
212 {snet: "udp", saddr: "0.0.0.0:0", tnet: "udp", taddr: "127.0.0.1"},
213 {snet: "udp", saddr: "[::ffff:0.0.0.0]:0", tnet: "udp", taddr: "127.0.0.1"},
214 {snet: "udp", saddr: "[::]:0", tnet: "udp", taddr: "::1"},
216 {snet: "udp", saddr: ":0", tnet: "udp", taddr: "::1"},
217 {snet: "udp", saddr: "0.0.0.0:0", tnet: "udp", taddr: "::1"},
218 {snet: "udp", saddr: "[::ffff:0.0.0.0]:0", tnet: "udp", taddr: "::1"},
219 {snet: "udp", saddr: "[::]:0", tnet: "udp", taddr: "127.0.0.1"},
221 {snet: "udp", saddr: ":0", tnet: "udp4", taddr: "127.0.0.1"},
222 {snet: "udp", saddr: "0.0.0.0:0", tnet: "udp4", taddr: "127.0.0.1"},
223 {snet: "udp", saddr: "[::ffff:0.0.0.0]:0", tnet: "udp4", taddr: "127.0.0.1"},
224 {snet: "udp", saddr: "[::]:0", tnet: "udp6", taddr: "::1"},
226 {snet: "udp", saddr: ":0", tnet: "udp6", taddr: "::1"},
227 {snet: "udp", saddr: "0.0.0.0:0", tnet: "udp6", taddr: "::1"},
228 {snet: "udp", saddr: "[::ffff:0.0.0.0]:0", tnet: "udp6", taddr: "::1"},
229 {snet: "udp", saddr: "[::]:0", tnet: "udp4", taddr: "127.0.0.1"},
231 {snet: "udp", saddr: "127.0.0.1:0", tnet: "udp", taddr: "127.0.0.1"},
232 {snet: "udp", saddr: "[::ffff:127.0.0.1]:0", tnet: "udp", taddr: "127.0.0.1"},
233 {snet: "udp", saddr: "[::1]:0", tnet: "udp", taddr: "::1"},
235 {snet: "udp4", saddr: ":0", tnet: "udp4", taddr: "127.0.0.1"},
236 {snet: "udp4", saddr: "0.0.0.0:0", tnet: "udp4", taddr: "127.0.0.1"},
237 {snet: "udp4", saddr: "[::ffff:0.0.0.0]:0", tnet: "udp4", taddr: "127.0.0.1"},
239 {snet: "udp4", saddr: "127.0.0.1:0", tnet: "udp4", taddr: "127.0.0.1"},
241 {snet: "udp6", saddr: ":0", tnet: "udp6", taddr: "::1"},
242 {snet: "udp6", saddr: "[::]:0", tnet: "udp6", taddr: "::1"},
244 {snet: "udp6", saddr: "[::1]:0", tnet: "udp6", taddr: "::1"},
246 {snet: "udp", saddr: "127.0.0.1:0", tnet: "udp", taddr: "127.0.0.1", dial: true},
248 {snet: "udp", saddr: "[::1]:0", tnet: "udp", taddr: "::1", dial: true},
251 func TestUDPServer(t *testing.T) {
252 for i, tt := range udpServerTests {
254 t.Run(fmt.Sprint(i), func(t *testing.T) {
255 if !testableListenArgs(tt.snet, tt.saddr, tt.taddr) {
256 t.Skipf("skipping %s %s<-%s test", tt.snet, tt.saddr, tt.taddr)
258 t.Logf("%s %s<-%s", tt.snet, tt.saddr, tt.taddr)
260 c1, err := ListenPacket(tt.snet, tt.saddr)
262 if perr := parseDialError(err); perr != nil {
268 ls := (&packetListener{PacketConn: c1}).newLocalServer()
270 tpch := make(chan error, 1)
271 handler := func(ls *localPacketServer, c PacketConn) { packetTransponder(c, tpch) }
272 if err := ls.buildup(handler); err != nil {
276 trch := make(chan error, 1)
277 _, port, err := SplitHostPort(ls.PacketConn.LocalAddr().String())
282 d := Dialer{Timeout: someTimeout}
283 c2, err := d.Dial(tt.tnet, JoinHostPort(tt.taddr, port))
285 if perr := parseDialError(err); perr != nil {
291 go transceiver(c2, []byte("UDP SERVER TEST"), trch)
293 c2, err := ListenPacket(tt.tnet, JoinHostPort(tt.taddr, "0"))
295 if perr := parseDialError(err); perr != nil {
301 dst, err := ResolveUDPAddr(tt.tnet, JoinHostPort(tt.taddr, port))
305 go packetTransceiver(c2, []byte("UDP SERVER TEST"), dst, trch)
308 for trch != nil || tpch != nil {
310 case err, ok := <-trch:
315 t.Errorf("client: %v", err)
317 case err, ok := <-tpch:
322 t.Errorf("server: %v", err)
330 func TestUnixgramServer(t *testing.T) {
331 var unixgramServerTests = []struct {
332 saddr string // server endpoint
333 caddr string // client endpoint
334 dial bool // test with Dial
336 {saddr: testUnixAddr(t), caddr: testUnixAddr(t)},
337 {saddr: testUnixAddr(t), caddr: testUnixAddr(t), dial: true},
339 {saddr: "@nettest/go/unixgram/server", caddr: "@nettest/go/unixgram/client"},
342 for i, tt := range unixgramServerTests {
344 t.Run(fmt.Sprint(i), func(t *testing.T) {
345 if !testableListenArgs("unixgram", tt.saddr, "") {
346 t.Skipf("skipping unixgram %s<-%s test", tt.saddr, tt.caddr)
348 t.Logf("unixgram %s<-%s", tt.saddr, tt.caddr)
350 c1, err := ListenPacket("unixgram", tt.saddr)
352 if perr := parseDialError(err); perr != nil {
358 ls := (&packetListener{PacketConn: c1}).newLocalServer()
360 tpch := make(chan error, 1)
361 handler := func(ls *localPacketServer, c PacketConn) { packetTransponder(c, tpch) }
362 if err := ls.buildup(handler); err != nil {
366 trch := make(chan error, 1)
368 d := Dialer{Timeout: someTimeout, LocalAddr: &UnixAddr{Net: "unixgram", Name: tt.caddr}}
369 c2, err := d.Dial("unixgram", ls.PacketConn.LocalAddr().String())
371 if perr := parseDialError(err); perr != nil {
376 defer os.Remove(c2.LocalAddr().String())
378 go transceiver(c2, []byte(c2.LocalAddr().String()), trch)
380 c2, err := ListenPacket("unixgram", tt.caddr)
382 if perr := parseDialError(err); perr != nil {
387 defer os.Remove(c2.LocalAddr().String())
389 go packetTransceiver(c2, []byte("UNIXGRAM SERVER TEST"), ls.PacketConn.LocalAddr(), trch)
392 for trch != nil || tpch != nil {
394 case err, ok := <-trch:
399 t.Errorf("client: %v", err)
401 case err, ok := <-tpch:
406 t.Errorf("server: %v", err)