-
-
Notifications
You must be signed in to change notification settings - Fork 50
/
expectations.go
455 lines (397 loc) · 13.1 KB
/
expectations.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
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
package pgxmock
import (
"context"
"errors"
"fmt"
"reflect"
"strings"
"sync"
"time"
pgx "github.com/jackc/pgx/v5"
pgconn "github.com/jackc/pgx/v5/pgconn"
)
// an expectation interface
type expectation interface {
error() error
required() bool
fulfilled() bool
fulfill()
sync.Locker
fmt.Stringer
}
// CallModifier interface represents common interface for all expectations supported
type CallModifier interface {
// Maybe allows the expected method call to be optional.
// Not calling an optional method will not cause an error while asserting expectations
Maybe() CallModifier
// Times indicates that that the expected method should only fire the indicated number of times.
// Zero value is ignored and means the same as one.
Times(n uint) CallModifier
// WillDelayFor allows to specify duration for which it will delay
// result. May be used together with Context
WillDelayFor(duration time.Duration) CallModifier
// WillReturnError allows to set an error for the expected method
WillReturnError(err error)
// WillPanic allows to force the expected method to panic
WillPanic(v any)
}
// common expectation struct
// satisfies the expectation interface
type commonExpectation struct {
sync.Mutex
triggered uint // how many times method was called
err error // should method return error
optional bool // can method be skipped
panicArgument any // panic value to return for recovery
plannedDelay time.Duration // should method delay before return
plannedCalls uint // how many sequentional calls should be made
}
func (e *commonExpectation) error() error {
return e.err
}
func (e *commonExpectation) fulfill() {
e.triggered++
}
func (e *commonExpectation) fulfilled() bool {
return e.triggered >= max(e.plannedCalls, 1)
}
func (e *commonExpectation) required() bool {
return !e.optional
}
func (e *commonExpectation) waitForDelay(ctx context.Context) (err error) {
select {
case <-time.After(e.plannedDelay):
err = e.error()
case <-ctx.Done():
err = ctx.Err()
}
if e.panicArgument != nil {
panic(e.panicArgument)
}
return err
}
func (e *commonExpectation) Maybe() CallModifier {
e.optional = true
return e
}
func (e *commonExpectation) Times(n uint) CallModifier {
e.plannedCalls = n
return e
}
func (e *commonExpectation) WillDelayFor(duration time.Duration) CallModifier {
e.plannedDelay = duration
return e
}
func (e *commonExpectation) WillReturnError(err error) {
e.err = err
}
var errPanic = errors.New("pgxmock panic")
func (e *commonExpectation) WillPanic(v any) {
e.err = errPanic
e.panicArgument = v
}
// String returns string representation
func (e *commonExpectation) String() string {
w := new(strings.Builder)
if e.err != nil {
if e.err != errPanic {
fmt.Fprintf(w, "\t- returns error: %v\n", e.err)
} else {
fmt.Fprintf(w, "\t- panics with: %v\n", e.panicArgument)
}
}
if e.plannedDelay > 0 {
fmt.Fprintf(w, "\t- delayed execution for: %v\n", e.plannedDelay)
}
if e.optional {
fmt.Fprint(w, "\t- execution is optional\n")
}
if e.plannedCalls > 0 {
fmt.Fprintf(w, "\t- execution calls awaited: %d\n", e.plannedCalls)
}
return w.String()
}
// queryBasedExpectation is a base class that adds a query matching logic
type queryBasedExpectation struct {
expectSQL string
expectRewrittenSQL string
args []interface{}
}
func (e *queryBasedExpectation) argsMatches(sql string, args []interface{}) (rewrittenSQL string, err error) {
eargs := e.args
// check for any QueryRewriter arguments: only supported as the first argument
if len(args) == 1 {
if qrw, ok := args[0].(pgx.QueryRewriter); ok {
// note: pgx.Conn is not currently used by the query rewriter
if rewrittenSQL, args, err = qrw.RewriteQuery(context.Background(), nil, sql, args); err != nil {
return rewrittenSQL, fmt.Errorf("error rewriting query: %w", err)
}
}
// also do rewriting on the expected args if a QueryRewriter is present
if len(eargs) == 1 {
if qrw, ok := eargs[0].(pgx.QueryRewriter); ok {
if _, eargs, err = qrw.RewriteQuery(context.Background(), nil, sql, eargs); err != nil {
return "", fmt.Errorf("error rewriting query expectation: %w", err)
}
}
}
}
if len(args) != len(eargs) {
return rewrittenSQL, fmt.Errorf("expected %d, but got %d arguments", len(eargs), len(args))
}
for k, v := range args {
// custom argument matcher
if matcher, ok := eargs[k].(Argument); ok {
if !matcher.Match(v) {
return rewrittenSQL, fmt.Errorf("matcher %T could not match %d argument %T - %+v", matcher, k, args[k], args[k])
}
continue
}
if darg := eargs[k]; !reflect.DeepEqual(darg, v) {
return rewrittenSQL, fmt.Errorf("argument %d expected [%T - %+v] does not match actual [%T - %+v]", k, darg, darg, v, v)
}
}
return
}
// ExpectedClose is used to manage pgx.Close expectation
// returned by pgxmock.ExpectClose
type ExpectedClose struct {
commonExpectation
}
// String returns string representation
func (e *ExpectedClose) String() string {
return "ExpectedClose => expecting call to Close()\n" + e.commonExpectation.String()
}
// ExpectedBegin is used to manage *pgx.Begin expectation
// returned by pgxmock.ExpectBegin.
type ExpectedBegin struct {
commonExpectation
opts pgx.TxOptions
}
// String returns string representation
func (e *ExpectedBegin) String() string {
msg := "ExpectedBegin => expecting call to Begin() or to BeginTx()\n"
if e.opts != (pgx.TxOptions{}) {
msg += fmt.Sprintf("\t- transaction options awaited: %+v\n", e.opts)
}
return msg + e.commonExpectation.String()
}
// ExpectedCommit is used to manage pgx.Tx.Commit expectation
// returned by pgxmock.ExpectCommit.
type ExpectedCommit struct {
commonExpectation
}
// String returns string representation
func (e *ExpectedCommit) String() string {
return "ExpectedCommit => expecting call to Tx.Commit()\n" + e.commonExpectation.String()
}
// ExpectedExec is used to manage pgx.Exec, pgx.Tx.Exec or pgx.Stmt.Exec expectations.
// Returned by pgxmock.ExpectExec.
type ExpectedExec struct {
commonExpectation
queryBasedExpectation
result pgconn.CommandTag
}
// WithArgs will match given expected args to actual database exec operation arguments.
// if at least one argument does not match, it will return an error. For specific
// arguments an pgxmock.Argument interface can be used to match an argument.
func (e *ExpectedExec) WithArgs(args ...interface{}) *ExpectedExec {
e.args = args
return e
}
// WithRewrittenSQL will match given expected expression to a rewritten SQL statement by
// an pgx.QueryRewriter argument
func (e *ExpectedExec) WithRewrittenSQL(sql string) *ExpectedExec {
e.expectRewrittenSQL = sql
return e
}
// String returns string representation
func (e *ExpectedExec) String() string {
msg := "ExpectedExec => expecting call to Exec():\n"
msg += fmt.Sprintf("\t- matches sql: '%s'\n", e.expectSQL)
if len(e.args) == 0 {
msg += "\t- is without arguments\n"
} else {
msg += "\t- is with arguments:\n"
for i, arg := range e.args {
msg += fmt.Sprintf("\t\t%d - %+v\n", i, arg)
}
}
if e.result.String() != "" {
msg += fmt.Sprintf("\t- returns result: %s\n", e.result)
}
return msg + e.commonExpectation.String()
}
// WillReturnResult arranges for an expected Exec() to return a particular
// result, there is pgxmock.NewResult(op string, rowsAffected int64) method
// to build a corresponding result.
func (e *ExpectedExec) WillReturnResult(result pgconn.CommandTag) *ExpectedExec {
e.result = result
return e
}
// ExpectedBatch is used to manage pgx.Batch expectations.
// Returned by pgxmock.ExpectBatch.
type ExpectedBatch struct {
commonExpectation
mock *pgxmock
expectedQueries []*queryBasedExpectation
closed bool
mustBeClosed bool
}
// ExpectExec allows to expect Queue().Exec() on this batch.
func (e *ExpectedBatch) ExpectExec(query string) *ExpectedExec {
ee := &ExpectedExec{}
ee.expectSQL = query
e.expectedQueries = append(e.expectedQueries, &ee.queryBasedExpectation)
e.mock.expectations = append(e.mock.expectations, ee)
return ee
}
// ExpectQuery allows to expect Queue().Query() or Queue().QueryRow() on this batch.
func (e *ExpectedBatch) ExpectQuery(query string) *ExpectedQuery {
eq := &ExpectedQuery{}
eq.expectSQL = query
e.expectedQueries = append(e.expectedQueries, &eq.queryBasedExpectation)
e.mock.expectations = append(e.mock.expectations, eq)
return eq
}
// String returns string representation
func (e *ExpectedBatch) String() string {
msg := "ExpectedBatch => expecting call to SendBatch()\n"
if e.mustBeClosed {
msg += "\t- batch must be closed\n"
}
return msg + e.commonExpectation.String()
}
// ExpectedPrepare is used to manage pgx.Prepare or pgx.Tx.Prepare expectations.
// Returned by pgxmock.ExpectPrepare.
type ExpectedPrepare struct {
commonExpectation
expectStmtName string
expectSQL string
}
// String returns string representation
func (e *ExpectedPrepare) String() string {
msg := "ExpectedPrepare => expecting call to Prepare():\n"
msg += fmt.Sprintf("\t- matches statement name: '%s'\n", e.expectStmtName)
msg += fmt.Sprintf("\t- matches sql: '%s'\n", e.expectSQL)
return msg + e.commonExpectation.String()
}
// ExpectedDeallocate is used to manage pgx.Deallocate and pgx.DeallocateAll expectations.
// Returned by pgxmock.ExpectDeallocate(string) and pgxmock.ExpectDeallocateAll().
type ExpectedDeallocate struct {
commonExpectation
expectStmtName string
expectAll bool
}
// String returns string representation
func (e *ExpectedDeallocate) String() string {
msg := "ExpectedDeallocate => expecting call to Deallocate():\n"
if e.expectAll {
msg += "\t- matches all statements\n"
} else {
msg += fmt.Sprintf("\t- matches statement name: '%s'\n", e.expectStmtName)
}
return msg + e.commonExpectation.String()
}
// ExpectedPing is used to manage Ping() expectations
type ExpectedPing struct {
commonExpectation
}
// String returns string representation
func (e *ExpectedPing) String() string {
msg := "ExpectedPing => expecting call to Ping()\n"
return msg + e.commonExpectation.String()
}
// ExpectedQuery is used to manage *pgx.Conn.Query, *pgx.Conn.QueryRow, *pgx.Tx.Query,
// *pgx.Tx.QueryRow, *pgx.Stmt.Query or *pgx.Stmt.QueryRow expectations
type ExpectedQuery struct {
commonExpectation
queryBasedExpectation
rows pgx.Rows
rowsMustBeClosed bool
rowsWereClosed bool
}
// WithArgs will match given expected args to actual database query arguments.
// if at least one argument does not match, it will return an error. For specific
// arguments an pgxmock.Argument interface can be used to match an argument.
func (e *ExpectedQuery) WithArgs(args ...interface{}) *ExpectedQuery {
e.args = args
return e
}
// WithRewrittenSQL will match given expected expression to a rewritten SQL statement by
// an pgx.QueryRewriter argument
func (e *ExpectedQuery) WithRewrittenSQL(sql string) *ExpectedQuery {
e.expectRewrittenSQL = sql
return e
}
// RowsWillBeClosed expects this query rows to be closed.
func (e *ExpectedQuery) RowsWillBeClosed() *ExpectedQuery {
e.rowsMustBeClosed = true
return e
}
// String returns string representation
func (e *ExpectedQuery) String() string {
msg := "ExpectedQuery => expecting call to Query() or to QueryRow():\n"
msg += fmt.Sprintf("\t- matches sql: '%s'\n", e.expectSQL)
if len(e.args) == 0 {
msg += "\t- is without arguments\n"
} else {
msg += "\t- is with arguments:\n"
for i, arg := range e.args {
msg += fmt.Sprintf("\t\t%d - %+v\n", i, arg)
}
}
if e.rows != nil {
msg += fmt.Sprintf("%s\n", e.rows)
}
return msg + e.commonExpectation.String()
}
// WillReturnRows specifies the set of resulting rows that will be returned
// by the triggered query
func (e *ExpectedQuery) WillReturnRows(rows ...*Rows) *ExpectedQuery {
e.rows = &rowSets{sets: rows, ex: e}
return e
}
// ExpectedCopyFrom is used to manage *pgx.Conn.CopyFrom expectations.
// Returned by *Pgxmock.ExpectCopyFrom.
type ExpectedCopyFrom struct {
commonExpectation
expectedTableName pgx.Identifier
expectedColumns []string
rowsAffected int64
}
// String returns string representation
func (e *ExpectedCopyFrom) String() string {
msg := "ExpectedCopyFrom => expecting CopyFrom which:"
msg += "\n - matches table name: '" + e.expectedTableName.Sanitize() + "'"
msg += fmt.Sprintf("\n - matches column names: '%+v'", e.expectedColumns)
if e.err != nil {
msg += fmt.Sprintf("\n - should returns error: %s", e.err)
}
return msg
}
// WillReturnResult arranges for an expected CopyFrom() to return a number of rows affected
func (e *ExpectedCopyFrom) WillReturnResult(result int64) *ExpectedCopyFrom {
e.rowsAffected = result
return e
}
// ExpectedReset is used to manage pgx.Reset expectation
type ExpectedReset struct {
commonExpectation
}
func (e *ExpectedReset) String() string {
return "ExpectedReset => expecting database Reset"
}
// ExpectedRollback is used to manage pgx.Tx.Rollback expectation
// returned by pgxmock.ExpectRollback.
type ExpectedRollback struct {
commonExpectation
}
// String returns string representation
func (e *ExpectedRollback) String() string {
msg := "ExpectedRollback => expecting transaction Rollback"
if e.err != nil {
msg += fmt.Sprintf(", which should return error: %s", e.err)
}
return msg
}