| // 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 ( |
| "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) (*TCPConn, error) { |
| c, err := dialTCP(net, laddr, raddr, deadline) |
| 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) { |
| 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 = 100*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) |
| } |
| } |
| } |