-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathdriver.go
191 lines (170 loc) · 6.53 KB
/
driver.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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
// Copyright 2020 The Cockroach 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 copyist
import (
"context"
"database/sql"
"database/sql/driver"
"errors"
)
// recordArgs is an untyped list of arguments and/or return values to/from a SQL
// driver method that was called during the recording process. It is stored as
// part of the Record struct that is stored, and is checked or returned during
// playback.
type recordArgs []interface{}
// record stores information for one SQL driver method that is called during the
// recording process. During playback, this record provides the information
// needed to verify the expected method is called, to check the right arguments
// are passed, and to provide the return value(s) from the method.
type record struct {
// Typ is the driver method that was called during the recording process.
Typ recordType
// Args are driver method arguments and/or return values that are needed for
// playback.
Args recordArgs
}
// recording is a list of records that need to be played back in sequence during
// one recording session.
type recording []*record
// proxyDriver records and plays back calls to driver.Driver methods.
//
// proxyDriver and proxyConn work together to take over connection pooling from
// the `sql` package. Connection pooling at any layer above the copyist driver
// is problematic, because it introduces non-determinism into recording
// sessions. Depending on whether a connection already exists, Driver.Open may
// or may not be called, with different recordings produced in each case.
//
// copyist disables `sql` package connection pooling by always returning
// driver.ErrBadConn from the driver.SessionResetter.ResetSession method, and
// instead pooling the connection in proxyDriver. In effect, copyist has a
// simple connection pool of size 1. That "pool" is cleared when copyist.Open is
// called, in order to ensure determinism. In addition, the global state
// maintains a monotonically increasing sequence number that identifies the
// current copyist session. Each time copyist.Open is called, the session is
// incremented. Connections created by earlier sessions cannot be reused. This
// ensures that copyist sessions are deterministic with regards to connection
// pooling - each session starts fresh.
type proxyDriver struct {
// Driver is the interface that must be implemented by a database
// driver.
//
// Database drivers may implement DriverContext for access
// to contexts and to parse the name only once for a pool of connections,
// instead of once per connection.
driver.Driver
// wrapped is the underlying driver that is being "recorded". This is nil
// if in playback mode.
wrapped driver.Driver
// driverName is the name of the wrapped driver.
driverName string
// pooled caches a copyist connection for reuse. For more information, see
// the proxyDriver comment regarding connection pooling.
pooled *proxyConn
}
// Open returns a new connection to the database.
// The name is a string in a driver-specific format.
//
// Open may return a cached connection (one previously
// closed), but doing so is unnecessary; the sql package
// maintains a pool of idle connections for efficient re-use.
//
// The returned connection is only used by one goroutine at a
// time.
func (d *proxyDriver) Open(name string) (driver.Conn, error) {
// Notify session that Open has been called so that it can do any needed
// per-session initialization.
if currentSession == nil {
panic(errors.New("copyist.Open was never called"))
}
currentSession.OnDriverOpen(d)
// Reuse pooled connection, if available and matching.
if conn := d.tryReuseConnection(name); conn != nil {
return conn, nil
}
if IsRecording() {
// Lazily get the wrapped driver.
if d.wrapped == nil {
// Open the database in order to get the sql.Driver object to wrap.
db, err := sql.Open(d.driverName, name)
if err != nil {
return nil, err
}
d.wrapped = db.Driver()
db.Close()
}
conn, err := d.wrapped.Open(name)
currentSession.AddRecord(&record{Typ: DriverOpen, Args: recordArgs{err}})
if err != nil {
return nil, err
}
return &proxyConn{driver: d, conn: conn, name: name, session: currentSession}, nil
}
rec, err := currentSession.VerifyRecord(DriverOpen)
if err != nil {
return nil, err
}
err, _ = rec.Args[0].(error)
if err != nil {
return nil, err
}
return &proxyConn{driver: d, name: name, session: currentSession}, nil
}
// tryPoolConnection puts the given connection into the pool if:
// 1. There is no connection in the pool already.
// 2. The connection was created by the current copyist session, not by a
// previous session. This check is necessary to ensure that connections are
// always re-opened for each session.
// 3. ResetSession on the underlying connection succeeds (or if the underlying
// connection is nil, or doesn't implement the driver.SessionResetter
// interface).
func (d *proxyDriver) tryPoolConnection(c *proxyConn) bool {
if d.pooled != nil {
// Already another connection in the pool.
return false
}
if c.session != currentSession {
// Connection was opened during a previous copyist session, so can't
// pool it.
return false
}
// Call ResetSession on the underlying connection, if it is implemented.
if resetter, ok := c.conn.(driver.SessionResetter); ok {
// TODO(andyk): Should we try to save and then use the context
// passed to ResetSession?
if resetter.ResetSession(context.Background()) != nil {
// Failed to reset.
return false
}
}
// Pool the connection for reuse.
c.driver.pooled = c
return true
}
// tryReuseConnection returns the pooled connection if it exists and if its name
// matches the given name, or nil if not.
func (d *proxyDriver) tryReuseConnection(name string) *proxyConn {
if d.pooled != nil && d.pooled.name == name {
pooled := d.pooled
d.pooled = nil
return pooled
}
return nil
}
// clearPooledConnection closes and clears the pooled connection, if it exists.
func (d *proxyDriver) clearPooledConnection() {
if d.pooled != nil {
d.pooled.Close()
d.pooled = nil
}
}