-
-
Notifications
You must be signed in to change notification settings - Fork 407
/
base_test.go
152 lines (120 loc) · 3.23 KB
/
base_test.go
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
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
package pg
import (
"context"
"encoding/binary"
"net"
"testing"
"time"
"github.com/go-pg/pg/v10/internal/pool"
"github.com/stretchr/testify/assert"
)
/*
The test is for testing the case that sending a cancel request when the timeout from connection comes earlier than ctx.Done().
*/
func Test_baseDB_withConn(t *testing.T) {
b := mockBaseDB{}
b.init()
b.pool = &mockPooler{}
ctx, _ := context.WithDeadline(context.TODO(), time.Now().Add(1000*time.Second)) // Make a deadline in context further than the timeout of connection.
b.withConn(ctx, func(context.Context, *pool.Conn) error {
// Immediately returns the error, so it is faster than the ctx.Done() returns. The error code here according to the function `isBadConn`.
return &mockPGError{map[byte]string{byte('C'): "57014"}}
})
// In the new change, a cancel request is sent to db and that connection is removed from the connection pool.
// Check if the cancel request, its code int32(80877102), is sent.
assert.Equal(t, int32(80877102), b.pool.(*mockPooler).mockConn.cancelCode)
assert.True(t, b.pool.(*mockPooler).toRemove)
}
type mockBaseDB struct {
baseDB
}
func (m *mockBaseDB) init() {
m.opt = &Options{}
}
type mockPooler struct {
conn *pool.Conn
toRemove bool
mockConn mockConn
}
func (m *mockPooler) NewConn(ctx context.Context) (*pool.Conn, error) {
m.mockConn = mockConn{}
m.conn = pool.NewConn(&m.mockConn, pool.NewConnPool(&pool.Options{}))
m.conn.ProcessID = 123
m.conn.SecretKey = 234
return m.conn, nil
}
func (m *mockPooler) CloseConn(conn *pool.Conn) error {
return nil
}
func (m *mockPooler) Get(ctx context.Context) (*pool.Conn, error) {
return &pool.Conn{ProcessID: 123, SecretKey: 234, Inited: true}, nil
}
func (m *mockPooler) Put(ctx context.Context, conn *pool.Conn) {
return
}
func (m *mockPooler) Remove(ctx context.Context, conn *pool.Conn, err error) {
m.toRemove = true
return
}
func (m *mockPooler) Len() int {
return 1
}
func (m *mockPooler) IdleLen() int {
return 1
}
func (m *mockPooler) Stats() *pool.Stats {
return nil
}
func (m *mockPooler) Close() error {
return nil
}
func (m *mockPooler) GetWriteBuffer() *pool.WriteBuffer {
return pool.NewWriteBuffer(1024)
}
func (m *mockPooler) PutWriteBuffer(_ *pool.WriteBuffer) {
}
func (m *mockPooler) GetReaderContext() *pool.ReaderContext {
return pool.NewReaderContext(1024)
}
func (m *mockPooler) PutReaderContext(_ *pool.ReaderContext) {
}
type mockPGError struct {
M map[byte]string
}
func (m *mockPGError) Error() string {
return ""
}
func (m *mockPGError) Field(field byte) string {
return m.M[field]
}
func (m *mockPGError) IntegrityViolation() bool {
return false
}
type mockConn struct {
cancelCode int32
}
func (m *mockConn) Read(b []byte) (n int, err error) {
return 0, nil
}
func (m *mockConn) Write(b []byte) (n int, err error) {
m.cancelCode = int32(binary.BigEndian.Uint32(b[4:8]))
return 0, nil
}
func (m *mockConn) Close() error {
return nil
}
func (m *mockConn) LocalAddr() net.Addr {
return nil
}
func (m *mockConn) RemoteAddr() net.Addr {
return nil
}
func (m *mockConn) SetDeadline(t time.Time) error {
return nil
}
func (m *mockConn) SetReadDeadline(t time.Time) error {
return nil
}
func (m *mockConn) SetWriteDeadline(t time.Time) error {
return nil
}