-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathtestdb.go
487 lines (444 loc) · 18.1 KB
/
testdb.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
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
package pgtestdb
import (
"context"
"crypto/rand"
"database/sql"
"encoding/hex"
"fmt"
"github.com/peterldowns/pgtestdb/internal/once"
"github.com/peterldowns/pgtestdb/internal/sessionlock"
"github.com/peterldowns/pgtestdb/migrators/common"
)
const (
// DefaultRoleUsername is the default name for the role that is created and
// used to create and connect to each test database.
DefaultRoleUsername = "pgtdbuser"
// DefaultRolePassword is the default password for the role that is created and
// used to create and connect to each test database.
DefaultRolePassword = "pgtdbpass"
// DefaultRoleCapabilities is the default set of capabilities for the role
// that is created and used to create and conect to each test database.
// This is locked down by default, and will not allow the creation of
// extensions.
DefaultRoleCapabilities = "NOSUPERUSER NOCREATEDB NOCREATEROLE"
// Deprecated: prefer [DefaultRoleUsername].
TestUser = DefaultRoleUsername
// Deprecated: prefer [DefaultRolePassword].
TestPassword = DefaultRolePassword
)
// DefaultRole returns the default Role used to create and connect to the
// template database and each test database. It is a function, not a struct, to
// prevent accidental overriding.
func DefaultRole() Role {
return Role{
Username: DefaultRoleUsername,
Password: DefaultRolePassword,
Capabilities: DefaultRoleCapabilities,
}
}
// Config contains the details needed to connect to a postgres server/database.
type Config struct {
DriverName string // the name of a driver to use when calling sql.Open() to connect to a database, "pgx" (pgx) or "postgres" (lib/pq)
Host string // the host of the database, "localhost"
Port string // the port of the database, "5433"
User string // the user to connect as, "postgres"
Password string // the password to connect with, "password"
Database string // the database to connect to, "postgres"
Options string // URL-formatted additional options to pass in the connection string, "sslmode=disable&something=value"
// TestRole is the role used to create and connect to the template database
// and each test database. If not provided, defaults to [DefaultRole]. The
// capabilities of this role should match the capabilities of the role that
// your application uses to connect to its database and run migrations.
TestRole *Role
// If true, ForceTerminateConnections will force-disconnect any remaining
// database connections prior to dropping the test database. This may be
// necessary if your code leaks database connections, intentionally or
// unintentionally. By default, if you leak a connection to a test database,
// pgtestdb will be unable to drop the database, and the test will be failed
// with a warning.
ForceTerminateConnections bool
}
// Role contains the details of a postgres role (user) that will be used
// when creating and connecting to the template and test databases.
type Role struct {
// The username for the role, defaults to [DefaultRoleUsername].
Username string
// The password for the role, defaults to [DefaultRolePassword].
Password string
// The capabilities that will be granted to the role, defaults to
// [DefaultRoleCapabilities].
Capabilities string
}
// URL returns a postgres connection string in the format
// "postgres://user:password@host:port/database?options=..."
func (c Config) URL() string {
return fmt.Sprintf(
"postgres://%s:%s@%s:%s/%s?%s",
c.User, c.Password, c.Host, c.Port, c.Database, c.Options,
)
}
// Connect calls `sql.Open()“ and connects to the database.
func (c Config) Connect() (*sql.DB, error) {
db, err := sql.Open(c.DriverName, c.URL())
if err != nil {
return nil, err
}
return db, nil
}
// A Migrator is necessary to provision the database that will be used as as template
// for each test.
type Migrator interface {
// Hash should return a unique identifier derived from the state of the database
// after it has been fully migrated. For instance, it may return a hash of all
// of the migration names and contents.
//
// pgtestdb will use the returned Hash to identify a template database. If a
// Migrator returns a Hash that has already been used to create a template
// database, it is assumed that the database need not be recreated since it
// would result in the same schema and data.
Hash() (string, error)
// Migrate is a function that actually performs the schema and data
// migrations to provision a template database. The connection given to this
// function is to an entirely new, empty, database. Migrate will be called
// only once, when the template database is being created.
Migrate(context.Context, *sql.DB, Config) error
}
// TB is a subset of the `testing.TB` testing interface implemented by
// `*testing.T`, `*testing.B`, and `*testing.F`, so you can use pgtestdb to get
// a database for tests, benchmarks, and fuzzes. It contains only the methods
// actually needed by pgtestdb, defined so that we can more easily mock it.
type TB interface {
Cleanup(func())
Failed() bool
Fatalf(format string, args ...any)
Helper()
Logf(format string, args ...any)
}
// New connects to a postgres server and creates and connects to a fresh
// database instance. This database is migrated by the given
// migrator, by get-or-creating a template database and then cloning it. This is
// a concurrency-safe primitive. If there is an error creating the database, the
// test will be immediately failed with `t.Fatalf()`.
//
// If this method succeeds, it will `t.Log()` the connection string to the
// created database, so that if your test fails, you can connect to the database
// manually and see what happened.
//
// If this method succeeds and your test succeeds, the database will be removed
// as part of the test cleanup process.
//
// `TB` is a subset of the `testing.TB` testing interface implemented by
// `*testing.T`, `*testing.B`, and `*testing.F`, so you can use pgtestdb to get
// a database for tests, benchmarks, and fuzzes.
func New(t TB, conf Config, migrator Migrator) *sql.DB {
t.Helper()
_, db := create(t, conf, migrator)
return db
}
// Custom is like [New] but after creating the new database instance, it closes
// any connections and returns the configuration details of that database so
// that you can connect to it explicitly, potentially via a different SQL
// interface.
func Custom(t TB, conf Config, migrator Migrator) *Config {
t.Helper()
config, db := create(t, conf, migrator)
// Close `*sql.DB` connection that was opened during the creation process so
// that it the caller can connect to the database in any method of their
// choosing without interference from this existing connection.
if err := db.Close(); err != nil {
t.Fatalf("could not close test database: '%s': %s", config.Database, err)
return nil // unreachable
}
return config
}
// create contains the implementation of [New] and [Custom], and is responsible
// for actually creating the instance database to be used by a testcase.
//
// create will use at most one connection to the underlying database at any
// given time.
func create(t TB, conf Config, migrator Migrator) (*Config, *sql.DB) {
t.Helper()
ctx := context.Background()
baseDB, err := conf.Connect()
if err != nil {
t.Fatalf("could not connect to database: %s", err)
return nil, nil // unreachable
}
// From this point onward, all functions assume that `conf.TestRole` is not nil.
// We default to the
if conf.TestRole == nil {
role := DefaultRole()
conf.TestRole = &role
}
if err := ensureUser(ctx, baseDB, conf); err != nil {
t.Fatalf("could not create pgtestdb user: %s", err)
return nil, nil // unreachable
}
template, err := getOrCreateTemplate(ctx, baseDB, conf, migrator)
if err != nil {
t.Fatalf("%s", err)
return nil, nil // unreachable
}
instance, err := createInstance(ctx, baseDB, *template)
if err != nil {
t.Fatalf("failed to create instance: %s", err)
return nil, nil // unreachable
}
t.Logf("testdbconf: %s", instance.URL())
db, err := instance.Connect()
if err != nil {
t.Fatalf("failed to connect to instance: %s", err)
return nil, nil // unreachable
}
if err := baseDB.Close(); err != nil {
t.Fatalf("could not close base database: '%s': %s", conf.Database, err)
return nil, nil // unreachable
}
t.Cleanup(func() {
// Close the testDB
if err := db.Close(); err != nil {
t.Fatalf("could not close test database: '%s': %s", instance.Database, err)
return // unreachable
}
// If the test failed, leave the instance around for further investigation
if t.Failed() {
return
}
// Otherwise, reconnect to the basedb and remove the instance from the server
baseDB, err := conf.Connect()
if err != nil {
t.Fatalf("could not connect to database: '%s': %s", conf.Database, err)
return
}
if conf.ForceTerminateConnections {
termConnections := fmt.Sprintf(`SELECT pg_terminate_backend(pg_stat_activity.pid)
FROM pg_stat_activity
WHERE pg_stat_activity.datname = '%s'
AND pid <> pg_backend_pid();`, instance.Database)
if _, err := baseDB.ExecContext(ctx, termConnections); err != nil {
t.Fatalf("could not terminate open connections on database '%s': %w",
instance.Database, err)
return // unreachable
}
}
query := fmt.Sprintf(`DROP DATABASE IF EXISTS "%s"`, instance.Database)
if _, err := baseDB.ExecContext(ctx, query); err != nil {
if !conf.ForceTerminateConnections {
t.Logf("pgtestdb failed to clean up the test database because there are still open connections to it.")
t.Logf("This usually means that your code is leaking database connections, which is usually bad.")
t.Logf("If you would like pgtestdb to force-terminate any open connections at the end of the testcase, set `ForceTerminateConnections = true` on your `pgtestdb.Config`")
}
t.Fatalf("could not drop test database '%s': %s", instance.Database, err)
return // unreachable
}
if err := baseDB.Close(); err != nil {
t.Fatalf("could not close base database: '%s': %s", conf.Database, err)
return // unreachable
}
})
return instance, db
}
// user is used to guarantee that each testdb user/role is only get-or-created
// at most once per program. Different calls to pgtestdb can specify different
// roles, but each will be get-or-created at most one time per program, and will
// be created only once no matter how many different programs or test suites run
// at once, thanks to the use of session locks.
var users once.Map[string, any] = once.NewMap[string, any]() //nolint:gochecknoglobals
func ensureUser(
ctx context.Context,
baseDB *sql.DB,
conf Config,
) error {
username := conf.TestRole.Username
_, err := users.Set(username, func() (*any, error) {
return nil, sessionlock.With(ctx, baseDB, username, func(conn *sql.Conn) error {
// Get-or-create a role/user dedicated to connecting to these test databases.
var roleExists bool
query := "SELECT EXISTS (SELECT from pg_catalog.pg_roles WHERE rolname = $1)"
if err := conn.QueryRowContext(ctx, query, username).Scan(&roleExists); err != nil {
return fmt.Errorf("failed to detect if role %s exists: %w", username, err)
}
if roleExists {
return nil
}
if !roleExists {
query = fmt.Sprintf(`CREATE ROLE "%s"`, username)
if _, err := conn.ExecContext(ctx, query); err != nil {
return fmt.Errorf("failed to create role %s: %w", username, err)
}
query = fmt.Sprintf(
`ALTER ROLE "%s" WITH LOGIN PASSWORD '%s' %s`,
username,
conf.TestRole.Password,
conf.TestRole.Capabilities,
)
if _, err := conn.ExecContext(ctx, query); err != nil {
return fmt.Errorf("failed to set password and capabilities for '%s': %w", username, err)
}
}
return nil
})
})
return err
}
// templateState keeps the state of a single template, so that each program only
// attempts to create/migrate the template at most once.
type templateState struct {
conf Config
hash string
}
var templates once.Map[string, templateState] = once.NewMap[string, templateState]() //nolint:gochecknoglobals
// getOrCreateTemplate will get-or-create a template, synchronizing at
// the golang level (with the states map, so that each template is
// get-or-created at most once) and at the postgres level (with advisory locks,
// so that there are no conflicts between multiple programs trying to create the
// template.)
//
// If there was a database error during template creation, the program that
// attempted the creation will set state.error, so subsequent attempts to access
// the template from within the same golang program will just return that error.
//
// This means that:
// - migrations are only run once per template per golang program / package under test.
// - you don't need to manually clear out "broken" templates between test suite runs.
func getOrCreateTemplate(
ctx context.Context,
baseDB *sql.DB,
dbconf Config,
migrator Migrator,
) (*templateState, error) {
mhash, err := migrator.Hash()
if err != nil {
return nil, fmt.Errorf("failed to calculate template hash: %w", err)
}
// The migrator Hash() implementation is included, along with the role
// details, so that if the user runs tests in parallel with different role
// information, they each get their own database.
hash := common.NewRecursiveHash(
common.Field("Username", dbconf.TestRole.Username),
common.Field("Password", dbconf.TestRole.Password),
common.Field("Capabilities", dbconf.TestRole.Capabilities),
common.Field("MigratorHash", mhash),
).String()
return templates.Set(hash, func() (*templateState, error) {
// This function runs once per program, but only synchronizes access
// within a single program. When running larger test suites, each
// package's tests may run in parallel, which means this does not
// perfectly synchronize interaction with the database.
state := templateState{}
state.hash = hash
state.conf = dbconf
state.conf.TestRole = dbconf.TestRole
state.conf.User = dbconf.TestRole.Username
state.conf.Password = dbconf.TestRole.Password
state.conf.Database = fmt.Sprintf("testdb_tpl_%s", hash)
// sessionlock synchronizes the creation of the template with a
// session-scoped advisory lock.
err := sessionlock.With(ctx, baseDB, state.conf.Database, func(conn *sql.Conn) error {
return ensureTemplate(ctx, conn, migrator, state)
})
if err != nil {
return nil, err
}
return &state, nil
})
}
// ensureTemplate uses the 'datistemplate' column to mark a template as having
// been successfully created, and does not set 'datistemplate = true' until the
// database has been successfully created and migrated. If it finds a template
// database where 'datistemplate = false', it drops and then attempts to
// recreate that database.
func ensureTemplate(
ctx context.Context,
conn *sql.Conn,
migrator Migrator,
state templateState,
) error {
// If the template database already exists, and is marked as a template,
// there is no more work to be done.
var templateExists bool
query := "SELECT EXISTS (SELECT FROM pg_database WHERE datname = $1 AND datistemplate = true)"
if err := conn.QueryRowContext(ctx, query, state.conf.Database).Scan(&templateExists); err != nil {
return fmt.Errorf("failed to check if template %s already exists: %w", state.conf.Database, err)
}
if templateExists {
return nil
}
// If the template database already exists, but it is not marked as a
// template, there was a failure at some point during the creation process
// so it needs to be deleted.
query = fmt.Sprintf(`DROP DATABASE IF EXISTS "%s"`, state.conf.Database)
if _, err := conn.ExecContext(ctx, query); err != nil {
return fmt.Errorf("failed to drop broken template %s: %w", state.conf.Database, err)
}
query = fmt.Sprintf(`CREATE DATABASE "%s" OWNER "%s"`, state.conf.Database, state.conf.User)
if _, err := conn.ExecContext(ctx, query); err != nil {
return fmt.Errorf("failed to create template %s: %w", state.conf.Database, err)
}
// Connect to the template.
template, err := state.conf.Connect()
if err != nil {
return fmt.Errorf("failed to connect to template %s: %w", state.conf.Database, err)
}
defer template.Close()
// Apply the Migrator logic one time, when creating the template. If this
// fails, the template will remain and the developer can connect to it and
// investigate the failure. Subsequent attempts to create the template will
// remove it, since it didn't get marked as complete (datistemplate=true).
if err := migrator.Migrate(ctx, template, state.conf); err != nil {
return fmt.Errorf("failed to migrator.Migrate template %s: %w", state.conf.Database, err)
}
// Finalize the creation of the template by marking it as a
// template.
query = "UPDATE pg_database SET datistemplate = true WHERE datname=$1"
if _, err := conn.ExecContext(ctx, query, state.conf.Database); err != nil {
return fmt.Errorf("failed to confirm template %s: %w", state.conf.Database, err)
}
return nil
}
// createInstance creates a new test database instance by cloning a template.
func createInstance(
ctx context.Context,
baseDB *sql.DB,
template templateState,
) (*Config, error) {
testConf := template.conf
testConf.Database = fmt.Sprintf(
"testdb_tpl_%s_inst_%s",
template.hash,
randomID(),
)
query := fmt.Sprintf(
`CREATE DATABASE "%s" WITH TEMPLATE "%s" OWNER "%s"`,
testConf.Database,
template.conf.Database,
testConf.User,
)
if _, err := baseDB.ExecContext(ctx, query); err != nil {
return nil, fmt.Errorf("failed to create instance from template %s: %w", template.conf.Database, err)
}
return &testConf, nil
}
// randomID is a helper for coming up with the names of the instance databases.
// It uses 32 random bits in the name, which means collisions are unlikely.
func randomID() string {
bytes := make([]byte, 4)
_, err := rand.Read(bytes)
if err != nil {
panic(err)
}
return hex.EncodeToString(bytes)
}
// NoopMigrator fulfills the Migrator interface but does absolutely nothing.
// You can use this to get empty databases in your tests, or if you are trying
// out testdb and aren't sure which migrator to use yet.
//
// For more documentation on migrators, see
// https://github.com/peterldowns/pgtestdb#testdbmigrator
type NoopMigrator struct{}
func (NoopMigrator) Hash() (string, error) {
return "noop", nil
}
func (NoopMigrator) Migrate(_ context.Context, _ *sql.DB, _ Config) error {
return nil
}