| // 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. |
| |
| // +build darwin freebsd linux netbsd openbsd |
| |
| package net |
| |
| import ( |
| "io" |
| "syscall" |
| "testing" |
| ) |
| |
| // Issue 3590. netFd.AddFD should return an error |
| // from the underlying pollster rather than panicing. |
| func TestAddFDReturnsError(t *testing.T) { |
| ln := newLocalListener(t).(*TCPListener) |
| defer ln.Close() |
| connected := make(chan bool) |
| go func() { |
| for { |
| c, err := ln.Accept() |
| if err != nil { |
| return |
| } |
| connected <- true |
| defer c.Close() |
| } |
| }() |
| |
| c, err := DialTCP("tcp", nil, ln.Addr().(*TCPAddr)) |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer c.Close() |
| <-connected |
| |
| // replace c's pollServer with a closed version. |
| ps, err := newPollServer() |
| if err != nil { |
| t.Fatal(err) |
| } |
| ps.poll.Close() |
| c.conn.fd.pollServer = ps |
| |
| var b [1]byte |
| _, err = c.Read(b[:]) |
| if err, ok := err.(*OpError); ok { |
| if err.Op == "addfd" { |
| return |
| } |
| if err, ok := err.Err.(*OpError); ok { |
| // the err is sometimes wrapped by another OpError |
| if err.Op == "addfd" { |
| return |
| } |
| } |
| } |
| t.Error("unexpected error:", err) |
| } |
| |
| var chkReadErrTests = []struct { |
| n int |
| err error |
| fd *netFD |
| expected error |
| }{ |
| |
| {100, nil, &netFD{sotype: syscall.SOCK_STREAM}, nil}, |
| {100, io.EOF, &netFD{sotype: syscall.SOCK_STREAM}, io.EOF}, |
| {100, errClosing, &netFD{sotype: syscall.SOCK_STREAM}, errClosing}, |
| {0, nil, &netFD{sotype: syscall.SOCK_STREAM}, io.EOF}, |
| {0, io.EOF, &netFD{sotype: syscall.SOCK_STREAM}, io.EOF}, |
| {0, errClosing, &netFD{sotype: syscall.SOCK_STREAM}, errClosing}, |
| |
| {100, nil, &netFD{sotype: syscall.SOCK_DGRAM}, nil}, |
| {100, io.EOF, &netFD{sotype: syscall.SOCK_DGRAM}, io.EOF}, |
| {100, errClosing, &netFD{sotype: syscall.SOCK_DGRAM}, errClosing}, |
| {0, nil, &netFD{sotype: syscall.SOCK_DGRAM}, nil}, |
| {0, io.EOF, &netFD{sotype: syscall.SOCK_DGRAM}, io.EOF}, |
| {0, errClosing, &netFD{sotype: syscall.SOCK_DGRAM}, errClosing}, |
| |
| {100, nil, &netFD{sotype: syscall.SOCK_SEQPACKET}, nil}, |
| {100, io.EOF, &netFD{sotype: syscall.SOCK_SEQPACKET}, io.EOF}, |
| {100, errClosing, &netFD{sotype: syscall.SOCK_SEQPACKET}, errClosing}, |
| {0, nil, &netFD{sotype: syscall.SOCK_SEQPACKET}, io.EOF}, |
| {0, io.EOF, &netFD{sotype: syscall.SOCK_SEQPACKET}, io.EOF}, |
| {0, errClosing, &netFD{sotype: syscall.SOCK_SEQPACKET}, errClosing}, |
| |
| {100, nil, &netFD{sotype: syscall.SOCK_RAW}, nil}, |
| {100, io.EOF, &netFD{sotype: syscall.SOCK_RAW}, io.EOF}, |
| {100, errClosing, &netFD{sotype: syscall.SOCK_RAW}, errClosing}, |
| {0, nil, &netFD{sotype: syscall.SOCK_RAW}, nil}, |
| {0, io.EOF, &netFD{sotype: syscall.SOCK_RAW}, io.EOF}, |
| {0, errClosing, &netFD{sotype: syscall.SOCK_RAW}, errClosing}, |
| } |
| |
| func TestChkReadErr(t *testing.T) { |
| for _, tt := range chkReadErrTests { |
| actual := chkReadErr(tt.n, tt.err, tt.fd) |
| if actual != tt.expected { |
| t.Errorf("chkReadError(%v, %v, %v): expected %v, actual %v", tt.n, tt.err, tt.fd.sotype, tt.expected, actual) |
| } |
| } |
| } |