wangpengfei
2023-06-02 064c0874e5fd041c4641ef873d1bf72ac98a184d
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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
/*
Copyright 2018 The Kubernetes Authors.
 
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
 
    http://www.apache.org/licenses/LICENSE-2.0
 
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
 
package connrotation
 
import (
    "context"
    "net"
    "sync"
    "sync/atomic"
    "testing"
    "time"
)
 
func TestCloseAll(t *testing.T) {
    closed := make(chan struct{})
    dialFn := func(ctx context.Context, network, address string) (net.Conn, error) {
        return closeOnlyConn{onClose: func() { closed <- struct{}{} }}, nil
    }
    dialer := NewDialer(dialFn)
 
    const numConns = 10
 
    // Outer loop to ensure Dialer is re-usable after CloseAll.
    for i := 0; i < 5; i++ {
        for j := 0; j < numConns; j++ {
            if _, err := dialer.Dial("", ""); err != nil {
                t.Fatal(err)
            }
        }
        dialer.CloseAll()
        for j := 0; j < numConns; j++ {
            select {
            case <-closed:
            case <-time.After(time.Second):
                t.Fatalf("iteration %d: 1s after CloseAll only %d/%d connections closed", i, j, numConns)
            }
        }
    }
}
 
// TestCloseAllRace ensures CloseAll works with connections being simultaneously dialed
func TestCloseAllRace(t *testing.T) {
    conns := int64(0)
    dialer := NewDialer(func(ctx context.Context, network, address string) (net.Conn, error) {
        return closeOnlyConn{onClose: func() { atomic.AddInt64(&conns, -1) }}, nil
    })
 
    done := make(chan struct{})
    wg := &sync.WaitGroup{}
 
    // Close all as fast as we can
    wg.Add(1)
    go func() {
        defer wg.Done()
        for {
            select {
            case <-done:
                return
            default:
                dialer.CloseAll()
            }
        }
    }()
 
    // Dial as fast as we can
    wg.Add(1)
    go func() {
        defer wg.Done()
        for {
            select {
            case <-done:
                return
            default:
                if _, err := dialer.Dial("", ""); err != nil {
                    t.Error(err)
                    return
                }
                atomic.AddInt64(&conns, 1)
            }
        }
    }()
 
    // Soak to ensure no races
    time.Sleep(time.Second)
 
    // Signal completion
    close(done)
    // Wait for goroutines
    wg.Wait()
    // Ensure CloseAll ran after all dials
    dialer.CloseAll()
 
    // Expect all connections to close within 5 seconds
    for start := time.Now(); time.Now().Sub(start) < 5*time.Second; time.Sleep(10 * time.Millisecond) {
        // Ensure all connections were closed
        if c := atomic.LoadInt64(&conns); c == 0 {
            break
        } else {
            t.Logf("got %d open connections, want 0, will retry", c)
        }
    }
    // Ensure all connections were closed
    if c := atomic.LoadInt64(&conns); c != 0 {
        t.Fatalf("got %d open connections, want 0", c)
    }
}
 
type closeOnlyConn struct {
    net.Conn
    onClose func()
}
 
func (c closeOnlyConn) Close() error {
    go c.onClose()
    return nil
}