liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
// Copyright 2016 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 aix darwin dragonfly freebsd linux netbsd openbsd solaris
 
package net
 
import (
   "context"
   "syscall"
   "testing"
   "time"
)
 
// Issue 16523
func TestDialContextCancelRace(t *testing.T) {
   oldConnectFunc := connectFunc
   oldGetsockoptIntFunc := getsockoptIntFunc
   oldTestHookCanceledDial := testHookCanceledDial
   defer func() {
       connectFunc = oldConnectFunc
       getsockoptIntFunc = oldGetsockoptIntFunc
       testHookCanceledDial = oldTestHookCanceledDial
   }()
 
   ln, err := newLocalListener("tcp")
   if err != nil {
       t.Fatal(err)
   }
   listenerDone := make(chan struct{})
   go func() {
       defer close(listenerDone)
       c, err := ln.Accept()
       if err == nil {
           c.Close()
       }
   }()
   defer func() { <-listenerDone }()
   defer ln.Close()
 
   sawCancel := make(chan bool, 1)
   testHookCanceledDial = func() {
       sawCancel <- true
   }
 
   ctx, cancelCtx := context.WithCancel(context.Background())
 
   connectFunc = func(fd int, addr syscall.Sockaddr) error {
       err := oldConnectFunc(fd, addr)
       t.Logf("connect(%d, addr) = %v", fd, err)
       if err == nil {
           // On some operating systems, localhost
           // connects _sometimes_ succeed immediately.
           // Prevent that, so we exercise the code path
           // we're interested in testing. This seems
           // harmless. It makes FreeBSD 10.10 work when
           // run with many iterations. It failed about
           // half the time previously.
           return syscall.EINPROGRESS
       }
       return err
   }
 
   getsockoptIntFunc = func(fd, level, opt int) (val int, err error) {
       val, err = oldGetsockoptIntFunc(fd, level, opt)
       t.Logf("getsockoptIntFunc(%d, %d, %d) = (%v, %v)", fd, level, opt, val, err)
       if level == syscall.SOL_SOCKET && opt == syscall.SO_ERROR && err == nil && val == 0 {
           t.Logf("canceling context")
 
           // Cancel the context at just the moment which
           // caused the race in issue 16523.
           cancelCtx()
 
           // And wait for the "interrupter" goroutine to
           // cancel the dial by messing with its write
           // timeout before returning.
           select {
           case <-sawCancel:
               t.Logf("saw cancel")
           case <-time.After(5 * time.Second):
               t.Errorf("didn't see cancel after 5 seconds")
           }
       }
       return
   }
 
   var d Dialer
   c, err := d.DialContext(ctx, "tcp", ln.Addr().String())
   if err == nil {
       c.Close()
       t.Fatal("unexpected successful dial; want context canceled error")
   }
 
   select {
   case <-ctx.Done():
   case <-time.After(5 * time.Second):
       t.Fatal("expected context to be canceled")
   }
 
   oe, ok := err.(*OpError)
   if !ok || oe.Op != "dial" {
       t.Fatalf("Dial error = %#v; want dial *OpError", err)
   }
 
   if oe.Err != errCanceled {
       t.Errorf("DialContext = (%v, %v); want OpError with error %v", c, err, errCanceled)
   }
}