f98dd1a338
Reviewed-on: https://go-review.googlesource.com/19200 From-SVN: r233110
784 lines
20 KiB
Go
784 lines
20 KiB
Go
// Copyright 2011 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.
|
|
|
|
package net
|
|
|
|
import (
|
|
"internal/testenv"
|
|
"io"
|
|
"net/internal/socktest"
|
|
"runtime"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
var prohibitionaryDialArgTests = []struct {
|
|
network string
|
|
address string
|
|
}{
|
|
{"tcp6", "127.0.0.1"},
|
|
{"tcp6", "::ffff:127.0.0.1"},
|
|
}
|
|
|
|
func TestProhibitionaryDialArg(t *testing.T) {
|
|
switch runtime.GOOS {
|
|
case "plan9":
|
|
t.Skipf("not supported on %s", runtime.GOOS)
|
|
}
|
|
if testing.Short() || !*testExternal {
|
|
t.Skip("avoid external network")
|
|
}
|
|
if !supportsIPv4map {
|
|
t.Skip("mapping ipv4 address inside ipv6 address not supported")
|
|
}
|
|
|
|
ln, err := Listen("tcp", "[::]:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer ln.Close()
|
|
|
|
_, port, err := SplitHostPort(ln.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
for i, tt := range prohibitionaryDialArgTests {
|
|
c, err := Dial(tt.network, JoinHostPort(tt.address, port))
|
|
if err == nil {
|
|
c.Close()
|
|
t.Errorf("#%d: %v", i, err)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestSelfConnect(t *testing.T) {
|
|
if runtime.GOOS == "windows" {
|
|
// TODO(brainman): do not know why it hangs.
|
|
t.Skip("known-broken test on windows")
|
|
}
|
|
|
|
// Test that Dial does not honor self-connects.
|
|
// See the comment in DialTCP.
|
|
|
|
// Find a port that would be used as a local address.
|
|
l, err := Listen("tcp", "127.0.0.1:0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
c, err := Dial("tcp", l.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
addr := c.LocalAddr().String()
|
|
c.Close()
|
|
l.Close()
|
|
|
|
// Try to connect to that address repeatedly.
|
|
n := 100000
|
|
if testing.Short() {
|
|
n = 1000
|
|
}
|
|
switch runtime.GOOS {
|
|
case "darwin", "dragonfly", "freebsd", "netbsd", "openbsd", "plan9", "solaris", "windows":
|
|
// Non-Linux systems take a long time to figure
|
|
// out that there is nothing listening on localhost.
|
|
n = 100
|
|
}
|
|
for i := 0; i < n; i++ {
|
|
c, err := DialTimeout("tcp", addr, time.Millisecond)
|
|
if err == nil {
|
|
if c.LocalAddr().String() == addr {
|
|
t.Errorf("#%d: Dial %q self-connect", i, addr)
|
|
} else {
|
|
t.Logf("#%d: Dial %q succeeded - possibly racing with other listener", i, addr)
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDialTimeoutFDLeak(t *testing.T) {
|
|
switch runtime.GOOS {
|
|
case "plan9":
|
|
t.Skipf("%s does not have full support of socktest", runtime.GOOS)
|
|
}
|
|
|
|
const T = 100 * time.Millisecond
|
|
|
|
switch runtime.GOOS {
|
|
case "plan9", "windows":
|
|
origTestHookDialChannel := testHookDialChannel
|
|
testHookDialChannel = func() { time.Sleep(2 * T) }
|
|
defer func() { testHookDialChannel = origTestHookDialChannel }()
|
|
if runtime.GOOS == "plan9" {
|
|
break
|
|
}
|
|
fallthrough
|
|
default:
|
|
sw.Set(socktest.FilterConnect, func(so *socktest.Status) (socktest.AfterFilter, error) {
|
|
time.Sleep(2 * T)
|
|
return nil, errTimeout
|
|
})
|
|
defer sw.Set(socktest.FilterConnect, nil)
|
|
}
|
|
|
|
// Avoid tracking open-close jitterbugs between netFD and
|
|
// socket that leads to confusion of information inside
|
|
// socktest.Switch.
|
|
// It may happen when the Dial call bumps against TCP
|
|
// simultaneous open. See selfConnect in tcpsock_posix.go.
|
|
defer func() {
|
|
sw.Set(socktest.FilterClose, nil)
|
|
forceCloseSockets()
|
|
}()
|
|
var mu sync.Mutex
|
|
var attempts int
|
|
sw.Set(socktest.FilterClose, func(so *socktest.Status) (socktest.AfterFilter, error) {
|
|
mu.Lock()
|
|
attempts++
|
|
mu.Unlock()
|
|
return nil, errTimedout
|
|
})
|
|
|
|
const N = 100
|
|
var wg sync.WaitGroup
|
|
wg.Add(N)
|
|
for i := 0; i < N; i++ {
|
|
go func() {
|
|
defer wg.Done()
|
|
// This dial never starts to send any SYN
|
|
// segment because of above socket filter and
|
|
// test hook.
|
|
c, err := DialTimeout("tcp", "127.0.0.1:0", T)
|
|
if err == nil {
|
|
t.Errorf("unexpectedly established: tcp:%s->%s", c.LocalAddr(), c.RemoteAddr())
|
|
c.Close()
|
|
}
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
if attempts < N {
|
|
t.Errorf("got %d; want >= %d", attempts, N)
|
|
}
|
|
}
|
|
|
|
func TestDialerDualStackFDLeak(t *testing.T) {
|
|
switch runtime.GOOS {
|
|
case "plan9":
|
|
t.Skipf("%s does not have full support of socktest", runtime.GOOS)
|
|
case "windows":
|
|
t.Skipf("not implemented a way to cancel dial racers in TCP SYN-SENT state on %s", runtime.GOOS)
|
|
}
|
|
if !supportsIPv4 || !supportsIPv6 {
|
|
t.Skip("both IPv4 and IPv6 are required")
|
|
}
|
|
|
|
origTestHookLookupIP := testHookLookupIP
|
|
defer func() { testHookLookupIP = origTestHookLookupIP }()
|
|
testHookLookupIP = lookupLocalhost
|
|
handler := func(dss *dualStackServer, ln Listener) {
|
|
for {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
dss, err := newDualStackServer([]streamListener{
|
|
{network: "tcp4", address: "127.0.0.1"},
|
|
{network: "tcp6", address: "::1"},
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer dss.teardown()
|
|
if err := dss.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
before := sw.Sockets()
|
|
const T = 100 * time.Millisecond
|
|
const N = 10
|
|
var wg sync.WaitGroup
|
|
wg.Add(N)
|
|
d := &Dialer{DualStack: true, Timeout: T}
|
|
for i := 0; i < N; i++ {
|
|
go func() {
|
|
defer wg.Done()
|
|
c, err := d.Dial("tcp", JoinHostPort("localhost", dss.port))
|
|
if err != nil {
|
|
t.Error(err)
|
|
return
|
|
}
|
|
c.Close()
|
|
}()
|
|
}
|
|
wg.Wait()
|
|
time.Sleep(2 * T) // wait for the dial racers to stop
|
|
after := sw.Sockets()
|
|
if len(after) != len(before) {
|
|
t.Errorf("got %d; want %d", len(after), len(before))
|
|
}
|
|
}
|
|
|
|
// Define a pair of blackholed (IPv4, IPv6) addresses, for which dialTCP is
|
|
// expected to hang until the timeout elapses. These addresses are reserved
|
|
// for benchmarking by RFC 6890.
|
|
const (
|
|
slowDst4 = "192.18.0.254"
|
|
slowDst6 = "2001:2::254"
|
|
slowTimeout = 1 * time.Second
|
|
)
|
|
|
|
// In some environments, the slow IPs may be explicitly unreachable, and fail
|
|
// more quickly than expected. This test hook prevents dialTCP from returning
|
|
// before the deadline.
|
|
func slowDialTCP(net string, laddr, raddr *TCPAddr, deadline time.Time, cancel <-chan struct{}) (*TCPConn, error) {
|
|
c, err := dialTCP(net, laddr, raddr, deadline, cancel)
|
|
if ParseIP(slowDst4).Equal(raddr.IP) || ParseIP(slowDst6).Equal(raddr.IP) {
|
|
time.Sleep(deadline.Sub(time.Now()))
|
|
}
|
|
return c, err
|
|
}
|
|
|
|
func dialClosedPort() (actual, expected time.Duration) {
|
|
// Estimate the expected time for this platform.
|
|
// On Windows, dialing a closed port takes roughly 1 second,
|
|
// but other platforms should be instantaneous.
|
|
if runtime.GOOS == "windows" {
|
|
expected = 1500 * time.Millisecond
|
|
} else {
|
|
expected = 95 * time.Millisecond
|
|
}
|
|
|
|
l, err := Listen("tcp", "127.0.0.1:0")
|
|
if err != nil {
|
|
return 999 * time.Hour, expected
|
|
}
|
|
addr := l.Addr().String()
|
|
l.Close()
|
|
// On OpenBSD, interference from TestSelfConnect is mysteriously
|
|
// causing the first attempt to hang for a few seconds, so we throw
|
|
// away the first result and keep the second.
|
|
for i := 1; ; i++ {
|
|
startTime := time.Now()
|
|
c, err := Dial("tcp", addr)
|
|
if err == nil {
|
|
c.Close()
|
|
}
|
|
elapsed := time.Now().Sub(startTime)
|
|
if i == 2 {
|
|
return elapsed, expected
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDialParallel(t *testing.T) {
|
|
if testing.Short() || !*testExternal {
|
|
t.Skip("avoid external network")
|
|
}
|
|
if !supportsIPv4 || !supportsIPv6 {
|
|
t.Skip("both IPv4 and IPv6 are required")
|
|
}
|
|
|
|
closedPortDelay, expectClosedPortDelay := dialClosedPort()
|
|
if closedPortDelay > expectClosedPortDelay {
|
|
t.Errorf("got %v; want <= %v", closedPortDelay, expectClosedPortDelay)
|
|
}
|
|
|
|
const instant time.Duration = 0
|
|
const fallbackDelay = 200 * time.Millisecond
|
|
|
|
// Some cases will run quickly when "connection refused" is fast,
|
|
// or trigger the fallbackDelay on Windows. This value holds the
|
|
// lesser of the two delays.
|
|
var closedPortOrFallbackDelay time.Duration
|
|
if closedPortDelay < fallbackDelay {
|
|
closedPortOrFallbackDelay = closedPortDelay
|
|
} else {
|
|
closedPortOrFallbackDelay = fallbackDelay
|
|
}
|
|
|
|
origTestHookDialTCP := testHookDialTCP
|
|
defer func() { testHookDialTCP = origTestHookDialTCP }()
|
|
testHookDialTCP = slowDialTCP
|
|
|
|
nCopies := func(s string, n int) []string {
|
|
out := make([]string, n)
|
|
for i := 0; i < n; i++ {
|
|
out[i] = s
|
|
}
|
|
return out
|
|
}
|
|
|
|
var testCases = []struct {
|
|
primaries []string
|
|
fallbacks []string
|
|
teardownNetwork string
|
|
expectOk bool
|
|
expectElapsed time.Duration
|
|
}{
|
|
// These should just work on the first try.
|
|
{[]string{"127.0.0.1"}, []string{}, "", true, instant},
|
|
{[]string{"::1"}, []string{}, "", true, instant},
|
|
{[]string{"127.0.0.1", "::1"}, []string{slowDst6}, "tcp6", true, instant},
|
|
{[]string{"::1", "127.0.0.1"}, []string{slowDst4}, "tcp4", true, instant},
|
|
// Primary is slow; fallback should kick in.
|
|
{[]string{slowDst4}, []string{"::1"}, "", true, fallbackDelay},
|
|
// Skip a "connection refused" in the primary thread.
|
|
{[]string{"127.0.0.1", "::1"}, []string{}, "tcp4", true, closedPortDelay},
|
|
{[]string{"::1", "127.0.0.1"}, []string{}, "tcp6", true, closedPortDelay},
|
|
// Skip a "connection refused" in the fallback thread.
|
|
{[]string{slowDst4, slowDst6}, []string{"::1", "127.0.0.1"}, "tcp6", true, fallbackDelay + closedPortDelay},
|
|
// Primary refused, fallback without delay.
|
|
{[]string{"127.0.0.1"}, []string{"::1"}, "tcp4", true, closedPortOrFallbackDelay},
|
|
{[]string{"::1"}, []string{"127.0.0.1"}, "tcp6", true, closedPortOrFallbackDelay},
|
|
// Everything is refused.
|
|
{[]string{"127.0.0.1"}, []string{}, "tcp4", false, closedPortDelay},
|
|
// Nothing to do; fail instantly.
|
|
{[]string{}, []string{}, "", false, instant},
|
|
// Connecting to tons of addresses should not trip the deadline.
|
|
{nCopies("::1", 1000), []string{}, "", true, instant},
|
|
}
|
|
|
|
handler := func(dss *dualStackServer, ln Listener) {
|
|
for {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
|
|
// Convert a list of IP strings into TCPAddrs.
|
|
makeAddrs := func(ips []string, port string) addrList {
|
|
var out addrList
|
|
for _, ip := range ips {
|
|
addr, err := ResolveTCPAddr("tcp", JoinHostPort(ip, port))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
out = append(out, addr)
|
|
}
|
|
return out
|
|
}
|
|
|
|
for i, tt := range testCases {
|
|
dss, err := newDualStackServer([]streamListener{
|
|
{network: "tcp4", address: "127.0.0.1"},
|
|
{network: "tcp6", address: "::1"},
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer dss.teardown()
|
|
if err := dss.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if tt.teardownNetwork != "" {
|
|
// Destroy one of the listening sockets, creating an unreachable port.
|
|
dss.teardownNetwork(tt.teardownNetwork)
|
|
}
|
|
|
|
primaries := makeAddrs(tt.primaries, dss.port)
|
|
fallbacks := makeAddrs(tt.fallbacks, dss.port)
|
|
d := Dialer{
|
|
FallbackDelay: fallbackDelay,
|
|
Timeout: slowTimeout,
|
|
}
|
|
ctx := &dialContext{
|
|
Dialer: d,
|
|
network: "tcp",
|
|
address: "?",
|
|
finalDeadline: d.deadline(time.Now()),
|
|
}
|
|
startTime := time.Now()
|
|
c, err := dialParallel(ctx, primaries, fallbacks)
|
|
elapsed := time.Now().Sub(startTime)
|
|
|
|
if c != nil {
|
|
c.Close()
|
|
}
|
|
|
|
if tt.expectOk && err != nil {
|
|
t.Errorf("#%d: got %v; want nil", i, err)
|
|
} else if !tt.expectOk && err == nil {
|
|
t.Errorf("#%d: got nil; want non-nil", i)
|
|
}
|
|
|
|
expectElapsedMin := tt.expectElapsed - 95*time.Millisecond
|
|
expectElapsedMax := tt.expectElapsed + 95*time.Millisecond
|
|
if !(elapsed >= expectElapsedMin) {
|
|
t.Errorf("#%d: got %v; want >= %v", i, elapsed, expectElapsedMin)
|
|
} else if !(elapsed <= expectElapsedMax) {
|
|
t.Errorf("#%d: got %v; want <= %v", i, elapsed, expectElapsedMax)
|
|
}
|
|
}
|
|
// Wait for any slowDst4/slowDst6 connections to timeout.
|
|
time.Sleep(slowTimeout * 3 / 2)
|
|
}
|
|
|
|
func lookupSlowFast(fn func(string) ([]IPAddr, error), host string) ([]IPAddr, error) {
|
|
switch host {
|
|
case "slow6loopback4":
|
|
// Returns a slow IPv6 address, and a local IPv4 address.
|
|
return []IPAddr{
|
|
{IP: ParseIP(slowDst6)},
|
|
{IP: ParseIP("127.0.0.1")},
|
|
}, nil
|
|
default:
|
|
return fn(host)
|
|
}
|
|
}
|
|
|
|
func TestDialerFallbackDelay(t *testing.T) {
|
|
if testing.Short() || !*testExternal {
|
|
t.Skip("avoid external network")
|
|
}
|
|
if !supportsIPv4 || !supportsIPv6 {
|
|
t.Skip("both IPv4 and IPv6 are required")
|
|
}
|
|
|
|
origTestHookLookupIP := testHookLookupIP
|
|
defer func() { testHookLookupIP = origTestHookLookupIP }()
|
|
testHookLookupIP = lookupSlowFast
|
|
|
|
origTestHookDialTCP := testHookDialTCP
|
|
defer func() { testHookDialTCP = origTestHookDialTCP }()
|
|
testHookDialTCP = slowDialTCP
|
|
|
|
var testCases = []struct {
|
|
dualstack bool
|
|
delay time.Duration
|
|
expectElapsed time.Duration
|
|
}{
|
|
// Use a very brief delay, which should fallback immediately.
|
|
{true, 1 * time.Nanosecond, 0},
|
|
// Use a 200ms explicit timeout.
|
|
{true, 200 * time.Millisecond, 200 * time.Millisecond},
|
|
// The default is 300ms.
|
|
{true, 0, 300 * time.Millisecond},
|
|
// This case is last, in order to wait for hanging slowDst6 connections.
|
|
{false, 0, slowTimeout},
|
|
}
|
|
|
|
handler := func(dss *dualStackServer, ln Listener) {
|
|
for {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
dss, err := newDualStackServer([]streamListener{
|
|
{network: "tcp", address: "127.0.0.1"},
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer dss.teardown()
|
|
if err := dss.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
for i, tt := range testCases {
|
|
d := &Dialer{DualStack: tt.dualstack, FallbackDelay: tt.delay, Timeout: slowTimeout}
|
|
|
|
startTime := time.Now()
|
|
c, err := d.Dial("tcp", JoinHostPort("slow6loopback4", dss.port))
|
|
elapsed := time.Now().Sub(startTime)
|
|
if err == nil {
|
|
c.Close()
|
|
} else if tt.dualstack {
|
|
t.Error(err)
|
|
}
|
|
expectMin := tt.expectElapsed - 1*time.Millisecond
|
|
expectMax := tt.expectElapsed + 95*time.Millisecond
|
|
if !(elapsed >= expectMin) {
|
|
t.Errorf("#%d: got %v; want >= %v", i, elapsed, expectMin)
|
|
}
|
|
if !(elapsed <= expectMax) {
|
|
t.Errorf("#%d: got %v; want <= %v", i, elapsed, expectMax)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDialSerialAsyncSpuriousConnection(t *testing.T) {
|
|
if runtime.GOOS == "plan9" {
|
|
t.Skip("skipping on plan9; no deadline support, golang.org/issue/11932")
|
|
}
|
|
ln, err := newLocalListener("tcp")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer ln.Close()
|
|
|
|
d := Dialer{}
|
|
ctx := &dialContext{
|
|
Dialer: d,
|
|
network: "tcp",
|
|
address: "?",
|
|
finalDeadline: d.deadline(time.Now()),
|
|
}
|
|
|
|
results := make(chan dialResult)
|
|
cancel := make(chan struct{})
|
|
|
|
// Spawn a connection in the background.
|
|
go dialSerialAsync(ctx, addrList{ln.Addr()}, nil, cancel, results)
|
|
|
|
// Receive it at the server.
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer c.Close()
|
|
|
|
// Tell dialSerialAsync that someone else won the race.
|
|
close(cancel)
|
|
|
|
// The connection should close itself, without sending data.
|
|
c.SetReadDeadline(time.Now().Add(1 * time.Second))
|
|
var b [1]byte
|
|
if _, err := c.Read(b[:]); err != io.EOF {
|
|
t.Errorf("got %v; want %v", err, io.EOF)
|
|
}
|
|
}
|
|
|
|
func TestDialerPartialDeadline(t *testing.T) {
|
|
now := time.Date(2000, time.January, 1, 0, 0, 0, 0, time.UTC)
|
|
var testCases = []struct {
|
|
now time.Time
|
|
deadline time.Time
|
|
addrs int
|
|
expectDeadline time.Time
|
|
expectErr error
|
|
}{
|
|
// Regular division.
|
|
{now, now.Add(12 * time.Second), 1, now.Add(12 * time.Second), nil},
|
|
{now, now.Add(12 * time.Second), 2, now.Add(6 * time.Second), nil},
|
|
{now, now.Add(12 * time.Second), 3, now.Add(4 * time.Second), nil},
|
|
// Bump against the 2-second sane minimum.
|
|
{now, now.Add(12 * time.Second), 999, now.Add(2 * time.Second), nil},
|
|
// Total available is now below the sane minimum.
|
|
{now, now.Add(1900 * time.Millisecond), 999, now.Add(1900 * time.Millisecond), nil},
|
|
// Null deadline.
|
|
{now, noDeadline, 1, noDeadline, nil},
|
|
// Step the clock forward and cross the deadline.
|
|
{now.Add(-1 * time.Millisecond), now, 1, now, nil},
|
|
{now.Add(0 * time.Millisecond), now, 1, noDeadline, errTimeout},
|
|
{now.Add(1 * time.Millisecond), now, 1, noDeadline, errTimeout},
|
|
}
|
|
for i, tt := range testCases {
|
|
deadline, err := partialDeadline(tt.now, tt.deadline, tt.addrs)
|
|
if err != tt.expectErr {
|
|
t.Errorf("#%d: got %v; want %v", i, err, tt.expectErr)
|
|
}
|
|
if deadline != tt.expectDeadline {
|
|
t.Errorf("#%d: got %v; want %v", i, deadline, tt.expectDeadline)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDialerLocalAddr(t *testing.T) {
|
|
ch := make(chan error, 1)
|
|
handler := func(ls *localServer, ln Listener) {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
ch <- err
|
|
return
|
|
}
|
|
defer c.Close()
|
|
ch <- nil
|
|
}
|
|
ls, err := newLocalServer("tcp")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer ls.teardown()
|
|
if err := ls.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
laddr, err := ResolveTCPAddr(ls.Listener.Addr().Network(), ls.Listener.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
laddr.Port = 0
|
|
d := &Dialer{LocalAddr: laddr}
|
|
c, err := d.Dial(ls.Listener.Addr().Network(), ls.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer c.Close()
|
|
c.Read(make([]byte, 1))
|
|
err = <-ch
|
|
if err != nil {
|
|
t.Error(err)
|
|
}
|
|
}
|
|
|
|
func TestDialerDualStack(t *testing.T) {
|
|
if !supportsIPv4 || !supportsIPv6 {
|
|
t.Skip("both IPv4 and IPv6 are required")
|
|
}
|
|
|
|
closedPortDelay, expectClosedPortDelay := dialClosedPort()
|
|
if closedPortDelay > expectClosedPortDelay {
|
|
t.Errorf("got %v; want <= %v", closedPortDelay, expectClosedPortDelay)
|
|
}
|
|
|
|
origTestHookLookupIP := testHookLookupIP
|
|
defer func() { testHookLookupIP = origTestHookLookupIP }()
|
|
testHookLookupIP = lookupLocalhost
|
|
handler := func(dss *dualStackServer, ln Listener) {
|
|
for {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
|
|
var timeout = 150*time.Millisecond + closedPortDelay
|
|
for _, dualstack := range []bool{false, true} {
|
|
dss, err := newDualStackServer([]streamListener{
|
|
{network: "tcp4", address: "127.0.0.1"},
|
|
{network: "tcp6", address: "::1"},
|
|
})
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer dss.teardown()
|
|
if err := dss.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
d := &Dialer{DualStack: dualstack, Timeout: timeout}
|
|
for range dss.lns {
|
|
c, err := d.Dial("tcp", JoinHostPort("localhost", dss.port))
|
|
if err != nil {
|
|
t.Error(err)
|
|
continue
|
|
}
|
|
switch addr := c.LocalAddr().(*TCPAddr); {
|
|
case addr.IP.To4() != nil:
|
|
dss.teardownNetwork("tcp4")
|
|
case addr.IP.To16() != nil && addr.IP.To4() == nil:
|
|
dss.teardownNetwork("tcp6")
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
time.Sleep(timeout * 3 / 2) // wait for the dial racers to stop
|
|
}
|
|
|
|
func TestDialerKeepAlive(t *testing.T) {
|
|
handler := func(ls *localServer, ln Listener) {
|
|
for {
|
|
c, err := ln.Accept()
|
|
if err != nil {
|
|
return
|
|
}
|
|
c.Close()
|
|
}
|
|
}
|
|
ls, err := newLocalServer("tcp")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer ls.teardown()
|
|
if err := ls.buildup(handler); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer func() { testHookSetKeepAlive = func() {} }()
|
|
|
|
for _, keepAlive := range []bool{false, true} {
|
|
got := false
|
|
testHookSetKeepAlive = func() { got = true }
|
|
var d Dialer
|
|
if keepAlive {
|
|
d.KeepAlive = 30 * time.Second
|
|
}
|
|
c, err := d.Dial("tcp", ls.Listener.Addr().String())
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
c.Close()
|
|
if got != keepAlive {
|
|
t.Errorf("Dialer.KeepAlive = %v: SetKeepAlive called = %v, want %v", d.KeepAlive, got, !got)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDialCancel(t *testing.T) {
|
|
if runtime.GOOS == "plan9" || runtime.GOOS == "nacl" {
|
|
// plan9 is not implemented and nacl doesn't have
|
|
// external network access.
|
|
t.Skipf("skipping on %s", runtime.GOOS)
|
|
}
|
|
onGoBuildFarm := testenv.Builder() != ""
|
|
if testing.Short() && !onGoBuildFarm {
|
|
t.Skip("skipping in short mode")
|
|
}
|
|
|
|
blackholeIPPort := JoinHostPort(slowDst4, "1234")
|
|
if !supportsIPv4 {
|
|
blackholeIPPort = JoinHostPort(slowDst6, "1234")
|
|
}
|
|
|
|
ticker := time.NewTicker(10 * time.Millisecond)
|
|
defer ticker.Stop()
|
|
|
|
const cancelTick = 5 // the timer tick we cancel the dial at
|
|
const timeoutTick = 100
|
|
|
|
var d Dialer
|
|
cancel := make(chan struct{})
|
|
d.Cancel = cancel
|
|
errc := make(chan error, 1)
|
|
connc := make(chan Conn, 1)
|
|
go func() {
|
|
if c, err := d.Dial("tcp", blackholeIPPort); err != nil {
|
|
errc <- err
|
|
} else {
|
|
connc <- c
|
|
}
|
|
}()
|
|
ticks := 0
|
|
for {
|
|
select {
|
|
case <-ticker.C:
|
|
ticks++
|
|
if ticks == cancelTick {
|
|
close(cancel)
|
|
}
|
|
if ticks == timeoutTick {
|
|
t.Fatal("timeout waiting for dial to fail")
|
|
}
|
|
case c := <-connc:
|
|
c.Close()
|
|
t.Fatal("unexpected successful connection")
|
|
case err := <-errc:
|
|
if perr := parseDialError(err); perr != nil {
|
|
t.Error(perr)
|
|
}
|
|
if ticks < cancelTick {
|
|
t.Fatalf("dial error after %d ticks (%d before cancel sent): %v",
|
|
ticks, cancelTick-ticks, err)
|
|
}
|
|
if oe, ok := err.(*OpError); !ok || oe.Err != errCanceled {
|
|
t.Fatalf("dial error = %v (%T); want OpError with Err == errCanceled", err, err)
|
|
}
|
|
return // success.
|
|
}
|
|
}
|
|
}
|