// Copyright 2012 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // This file implements API tests across platforms and will never have a build // tag. package net import ( "bytes" "os" "runtime" "testing" "time" ) var condErrorf = func() func(*testing.T, string, ...interface{}) { // A few APIs are not implemented yet on both Plan 9 and Windows. switch runtime.GOOS { case "plan9", "windows": return (*testing.T).Logf } return (*testing.T).Errorf }() func TestTCPListenerSpecificMethods(t *testing.T) { la, err := ResolveTCPAddr("tcp4", "127.0.0.1:0") if err != nil { t.Fatalf("net.ResolveTCPAddr failed: %v", err) } ln, err := ListenTCP("tcp4", la) if err != nil { t.Fatalf("net.ListenTCP failed: %v", err) } ln.Addr() ln.SetDeadline(time.Now().Add(30 * time.Nanosecond)) defer ln.Close() if c, err := ln.Accept(); err != nil { if !err.(Error).Timeout() { t.Errorf("net.TCPListener.Accept failed: %v", err) return } } else { c.Close() } if c, err := ln.AcceptTCP(); err != nil { if !err.(Error).Timeout() { t.Errorf("net.TCPListener.AcceptTCP failed: %v", err) return } } else { c.Close() } if f, err := ln.File(); err != nil { condErrorf(t, "net.TCPListener.File failed: %v", err) return } else { f.Close() } } func TestTCPConnSpecificMethods(t *testing.T) { la, err := ResolveTCPAddr("tcp4", "127.0.0.1:0") if err != nil { t.Fatalf("net.ResolveTCPAddr failed: %v", err) } ln, err := ListenTCP("tcp4", la) if err != nil { t.Fatalf("net.ListenTCP failed: %v", err) } ln.Addr() defer ln.Close() done := make(chan int) go transponder(t, ln, done) ra, err := ResolveTCPAddr("tcp4", ln.Addr().String()) if err != nil { t.Errorf("net.ResolveTCPAddr failed: %v", err) return } c, err := DialTCP("tcp4", nil, ra) if err != nil { t.Errorf("net.DialTCP failed: %v", err) return } c.SetKeepAlive(false) c.SetLinger(0) c.SetNoDelay(false) c.LocalAddr() c.RemoteAddr() c.SetDeadline(time.Now().Add(100 * time.Millisecond)) c.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) defer c.Close() if _, err := c.Write([]byte("TCPCONN TEST")); err != nil { t.Errorf("net.TCPConn.Write failed: %v", err) return } rb := make([]byte, 128) if _, err := c.Read(rb); err != nil { t.Errorf("net.TCPConn.Read failed: %v", err) return } <-done } func TestUDPConnSpecificMethods(t *testing.T) { la, err := ResolveUDPAddr("udp4", "127.0.0.1:0") if err != nil { t.Fatalf("net.ResolveUDPAddr failed: %v", err) } c, err := ListenUDP("udp4", la) if err != nil { t.Fatalf("net.ListenUDP failed: %v", err) } c.LocalAddr() c.RemoteAddr() c.SetDeadline(time.Now().Add(100 * time.Millisecond)) c.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) c.SetReadBuffer(2048) c.SetWriteBuffer(2048) defer c.Close() wb := []byte("UDPCONN TEST") rb := make([]byte, 128) if _, err := c.WriteToUDP(wb, c.LocalAddr().(*UDPAddr)); err != nil { t.Errorf("net.UDPConn.WriteToUDP failed: %v", err) return } if _, _, err := c.ReadFromUDP(rb); err != nil { t.Errorf("net.UDPConn.ReadFromUDP failed: %v", err) return } if _, _, err := c.WriteMsgUDP(wb, nil, c.LocalAddr().(*UDPAddr)); err != nil { condErrorf(t, "net.UDPConn.WriteMsgUDP failed: %v", err) return } if _, _, _, _, err := c.ReadMsgUDP(rb, nil); err != nil { condErrorf(t, "net.UDPConn.ReadMsgUDP failed: %v", err) return } if f, err := c.File(); err != nil { condErrorf(t, "net.UDPConn.File failed: %v", err) return } else { f.Close() } } func TestIPConnSpecificMethods(t *testing.T) { switch runtime.GOOS { case "plan9": t.Skipf("skipping read test on %q", runtime.GOOS) } if os.Getuid() != 0 { t.Skipf("skipping test; must be root") } la, err := ResolveIPAddr("ip4", "127.0.0.1") if err != nil { t.Fatalf("net.ResolveIPAddr failed: %v", err) } c, err := ListenIP("ip4:icmp", la) if err != nil { t.Fatalf("net.ListenIP failed: %v", err) } c.LocalAddr() c.RemoteAddr() c.SetDeadline(time.Now().Add(100 * time.Millisecond)) c.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) c.SetReadBuffer(2048) c.SetWriteBuffer(2048) defer c.Close() id := os.Getpid() & 0xffff wb := newICMPEchoRequest(id, 1, 128, []byte("IPCONN TEST ")) rb := make([]byte, 20+128) if _, err := c.WriteToIP(wb, c.LocalAddr().(*IPAddr)); err != nil { t.Errorf("net.IPConn.WriteToIP failed: %v", err) return } if _, _, err := c.ReadFromIP(rb); err != nil { t.Errorf("net.IPConn.ReadFromIP failed: %v", err) return } if _, _, err := c.WriteMsgIP(wb, nil, c.LocalAddr().(*IPAddr)); err != nil { condErrorf(t, "net.UDPConn.WriteMsgIP failed: %v", err) return } if _, _, _, _, err := c.ReadMsgIP(rb, nil); err != nil { condErrorf(t, "net.UDPConn.ReadMsgIP failed: %v", err) return } if f, err := c.File(); err != nil { condErrorf(t, "net.IPConn.File failed: %v", err) return } else { f.Close() } } func TestUnixListenerSpecificMethods(t *testing.T) { switch runtime.GOOS { case "plan9", "windows": t.Skipf("skipping read test on %q", runtime.GOOS) } p := "/tmp/gotest.net" os.Remove(p) la, err := ResolveUnixAddr("unix", p) if err != nil { t.Fatalf("net.ResolveUnixAddr failed: %v", err) } ln, err := ListenUnix("unix", la) if err != nil { t.Fatalf("net.ListenUnix failed: %v", err) } ln.Addr() ln.SetDeadline(time.Now().Add(30 * time.Nanosecond)) defer ln.Close() defer os.Remove(p) if c, err := ln.Accept(); err != nil { if !err.(Error).Timeout() { t.Errorf("net.TCPListener.AcceptTCP failed: %v", err) return } } else { c.Close() } if c, err := ln.AcceptUnix(); err != nil { if !err.(Error).Timeout() { t.Errorf("net.TCPListener.AcceptTCP failed: %v", err) return } } else { c.Close() } if f, err := ln.File(); err != nil { t.Errorf("net.UnixListener.File failed: %v", err) return } else { f.Close() } } func TestUnixConnSpecificMethods(t *testing.T) { switch runtime.GOOS { case "plan9", "windows": t.Skipf("skipping test on %q", runtime.GOOS) } p1, p2, p3 := "/tmp/gotest.net1", "/tmp/gotest.net2", "/tmp/gotest.net3" os.Remove(p1) os.Remove(p2) os.Remove(p3) a1, err := ResolveUnixAddr("unixgram", p1) if err != nil { t.Fatalf("net.ResolveUnixAddr failed: %v", err) } c1, err := DialUnix("unixgram", a1, nil) if err != nil { t.Fatalf("net.DialUnix failed: %v", err) } c1.LocalAddr() c1.RemoteAddr() c1.SetDeadline(time.Now().Add(100 * time.Millisecond)) c1.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c1.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) c1.SetReadBuffer(2048) c1.SetWriteBuffer(2048) defer c1.Close() defer os.Remove(p1) a2, err := ResolveUnixAddr("unixgram", p2) if err != nil { t.Errorf("net.ResolveUnixAddr failed: %v", err) return } c2, err := DialUnix("unixgram", a2, nil) if err != nil { t.Errorf("net.DialUnix failed: %v", err) return } c2.LocalAddr() c2.RemoteAddr() c2.SetDeadline(time.Now().Add(100 * time.Millisecond)) c2.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c2.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) c2.SetReadBuffer(2048) c2.SetWriteBuffer(2048) defer c2.Close() defer os.Remove(p2) a3, err := ResolveUnixAddr("unixgram", p3) if err != nil { t.Errorf("net.ResolveUnixAddr failed: %v", err) return } c3, err := ListenUnixgram("unixgram", a3) if err != nil { t.Errorf("net.ListenUnixgram failed: %v", err) return } c3.LocalAddr() c3.RemoteAddr() c3.SetDeadline(time.Now().Add(100 * time.Millisecond)) c3.SetReadDeadline(time.Now().Add(100 * time.Millisecond)) c3.SetWriteDeadline(time.Now().Add(100 * time.Millisecond)) c3.SetReadBuffer(2048) c3.SetWriteBuffer(2048) defer c3.Close() defer os.Remove(p3) wb := []byte("UNIXCONN TEST") rb1 := make([]byte, 128) rb2 := make([]byte, 128) rb3 := make([]byte, 128) if _, _, err := c1.WriteMsgUnix(wb, nil, a2); err != nil { t.Errorf("net.UnixConn.WriteMsgUnix failed: %v", err) return } if _, _, _, _, err := c2.ReadMsgUnix(rb2, nil); err != nil { t.Errorf("net.UnixConn.ReadMsgUnix failed: %v", err) return } if _, err := c2.WriteToUnix(wb, a1); err != nil { t.Errorf("net.UnixConn.WriteToUnix failed: %v", err) return } if _, _, err := c1.ReadFromUnix(rb1); err != nil { t.Errorf("net.UnixConn.ReadFromUnix failed: %v", err) return } if _, err := c3.WriteToUnix(wb, a1); err != nil { t.Errorf("net.UnixConn.WriteToUnix failed: %v", err) return } if _, _, err := c1.ReadFromUnix(rb1); err != nil { t.Errorf("net.UnixConn.ReadFromUnix failed: %v", err) return } if _, err := c2.WriteToUnix(wb, a3); err != nil { t.Errorf("net.UnixConn.WriteToUnix failed: %v", err) return } if _, _, err := c3.ReadFromUnix(rb3); err != nil { t.Errorf("net.UnixConn.ReadFromUnix failed: %v", err) return } if f, err := c1.File(); err != nil { t.Errorf("net.UnixConn.File failed: %v", err) return } else { f.Close() } } func newICMPEchoRequest(id, seqnum, msglen int, filler []byte) []byte { b := newICMPInfoMessage(id, seqnum, msglen, filler) b[0] = 8 // calculate ICMP checksum cklen := len(b) s := uint32(0) for i := 0; i < cklen-1; i += 2 { s += uint32(b[i+1])<<8 | uint32(b[i]) } if cklen&1 == 1 { s += uint32(b[cklen-1]) } s = (s >> 16) + (s & 0xffff) s = s + (s >> 16) // place checksum back in header; using ^= avoids the // assumption the checksum bytes are zero b[2] ^= byte(^s & 0xff) b[3] ^= byte(^s >> 8) return b } func newICMPInfoMessage(id, seqnum, msglen int, filler []byte) []byte { b := make([]byte, msglen) copy(b[8:], bytes.Repeat(filler, (msglen-8)/len(filler)+1)) b[0] = 0 // type b[1] = 0 // code b[2] = 0 // checksum b[3] = 0 // checksum b[4] = byte(id >> 8) // identifier b[5] = byte(id & 0xff) // identifier b[6] = byte(seqnum >> 8) // sequence number b[7] = byte(seqnum & 0xff) // sequence number return b }