storj/satellite/satellitedb/migrate_postgres_test.go

204 lines
5.3 KiB
Go
Raw Normal View History

2019-02-14 21:55:21 +00:00
// Copyright (C) 2019 Storj Labs, Inc.
// See LICENSE for copying information.
package satellitedb_test
import (
"fmt"
"io/ioutil"
"path/filepath"
"strconv"
"strings"
"sync"
"testing"
"github.com/lib/pq"
2019-02-14 21:55:21 +00:00
"github.com/stretchr/testify/require"
"github.com/zeebo/errs"
2019-02-14 21:55:21 +00:00
"go.uber.org/zap/zaptest"
"storj.io/storj/private/dbutil/dbschema"
"storj.io/storj/private/dbutil/pgutil"
"storj.io/storj/private/dbutil/pgutil/pgtest"
"storj.io/storj/private/dbutil/tempdb"
2019-02-14 21:55:21 +00:00
"storj.io/storj/satellite/satellitedb"
)
// loadSnapshots loads all the dbschemas from testdata/postgres.* caching the result
func loadSnapshots(connstr string) (*dbschema.Snapshots, error) {
snapshots := &dbschema.Snapshots{}
// find all postgres sql files
matches, err := filepath.Glob("testdata/postgres.*")
if err != nil {
return nil, err
}
for _, match := range matches {
versionStr := match[19 : len(match)-4] // hack to avoid trim issues with path differences in windows/linux
version, err := strconv.Atoi(versionStr)
if err != nil {
return nil, errs.New("invalid testdata file %q: %v", match, err)
2019-02-14 21:55:21 +00:00
}
scriptData, err := ioutil.ReadFile(match)
if err != nil {
return nil, errs.New("could not read testdata file for version %d: %v", version, err)
2019-02-14 21:55:21 +00:00
}
snapshot, err := loadSnapshotFromSQL(connstr, string(scriptData))
2019-02-14 21:55:21 +00:00
if err != nil {
if pqErr, ok := err.(*pq.Error); ok && pqErr.Detail != "" {
return nil, fmt.Errorf("Version %d error: %v\nDetail: %s\nHint: %s", version, pqErr, pqErr.Detail, pqErr.Hint)
}
return nil, fmt.Errorf("Version %d error: %+v", version, err)
2019-02-14 21:55:21 +00:00
}
snapshot.Version = version
snapshots.Add(snapshot)
}
snapshots.Sort()
return snapshots, nil
}
// loadSnapshotFromSQL inserts script into connstr and loads schema.
func loadSnapshotFromSQL(connstr, script string) (_ *dbschema.Snapshot, err error) {
db, err := tempdb.OpenUnique(connstr, "load-schema")
if err != nil {
return nil, err
}
defer func() { err = errs.Combine(err, db.Close()) }()
_, err = db.Exec(script)
if err != nil {
return nil, err
}
snapshot, err := pgutil.QuerySnapshot(db)
if err != nil {
return nil, err
}
snapshot.Script = script
return snapshot, nil
}
2019-02-14 21:55:21 +00:00
const newDataSeparator = `-- NEW DATA --`
func newData(snap *dbschema.Snapshot) string {
tokens := strings.SplitN(snap.Script, newDataSeparator, 2)
if len(tokens) != 2 {
return ""
}
return tokens[1]
}
var (
dbxschema struct {
sync.Once
*dbschema.Schema
err error
}
)
// loadDBXSChema loads dbxscript schema only once and caches it,
// it shouldn't change during the test
func loadDBXSchema(connstr, dbxscript string) (*dbschema.Schema, error) {
dbxschema.Do(func() {
dbxschema.Schema, dbxschema.err = loadSchemaFromSQL(connstr, dbxscript)
2019-02-14 21:55:21 +00:00
})
return dbxschema.Schema, dbxschema.err
}
// loadSchemaFromSQL inserts script into connstr and loads schema.
func loadSchemaFromSQL(connstr, script string) (_ *dbschema.Schema, err error) {
db, err := tempdb.OpenUnique(connstr, "load-schema")
if err != nil {
return nil, err
}
defer func() { err = errs.Combine(err, db.Close()) }()
_, err = db.Exec(script)
if err != nil {
return nil, err
}
return pgutil.QuerySchema(db)
}
2019-02-14 21:55:21 +00:00
func TestMigratePostgres(t *testing.T) {
2019-04-26 14:39:11 +01:00
if *pgtest.ConnStr == "" {
t.Skip("Postgres flag missing, example: -postgres-test-db=" + pgtest.DefaultConnStr)
2019-02-14 21:55:21 +00:00
}
pgMigrateTest(t, *pgtest.ConnStr)
}
func pgMigrateTest(t *testing.T, connStr string) {
log := zaptest.NewLogger(t)
snapshots, err := loadSnapshots(connStr)
2019-02-14 21:55:21 +00:00
require.NoError(t, err)
// create tempDB
tempDB, err := tempdb.OpenUnique(connStr, "migrate")
require.NoError(t, err)
defer func() { require.NoError(t, tempDB.Close()) }()
2019-02-14 21:55:21 +00:00
// create a new satellitedb connection
db, err := satellitedb.New(log, tempDB.ConnStr)
require.NoError(t, err)
defer func() { require.NoError(t, db.Close()) }()
2019-02-14 21:55:21 +00:00
// we need raw database access unfortunately
rawdb := db.(*satellitedb.DB).TestDBAccess()
2019-02-14 21:55:21 +00:00
var finalSchema *dbschema.Schema
2019-02-14 21:55:21 +00:00
// get migration for this database
migrations := db.(*satellitedb.DB).PostgresMigration()
for i, step := range migrations.Steps {
tag := fmt.Sprintf("#%d - v%d", i, step.Version)
2019-02-14 21:55:21 +00:00
// run migration up to a specific version
err := migrations.TargetVersion(step.Version).Run(log.Named("migrate"))
require.NoError(t, err, tag)
2019-02-14 21:55:21 +00:00
// find the matching expected version
expected, ok := snapshots.FindVersion(step.Version)
require.True(t, ok, "Missing snapshot v%d. Did you forget to add a snapshot for the new migration?", step.Version)
2019-02-14 21:55:21 +00:00
// insert data for new tables
if newdata := newData(expected); newdata != "" {
_, err = rawdb.Exec(newdata)
require.NoError(t, err, tag)
}
2019-02-14 21:55:21 +00:00
// load schema from database
currentSchema, err := pgutil.QuerySchema(rawdb)
require.NoError(t, err, tag)
2019-02-14 21:55:21 +00:00
// we don't care changes in versions table
currentSchema.DropTable("versions")
2019-02-14 21:55:21 +00:00
// load data from database
currentData, err := pgutil.QueryData(rawdb, currentSchema)
require.NoError(t, err, tag)
2019-02-14 21:55:21 +00:00
// verify schema and data
require.Equal(t, expected.Schema, currentSchema, tag)
require.Equal(t, expected.Data, currentData, tag)
2019-02-14 21:55:21 +00:00
// keep the last version around
finalSchema = currentSchema
}
2019-02-14 21:55:21 +00:00
// verify that we also match the dbx version
dbxschema, err := loadDBXSchema(connStr, rawdb.Schema())
require.NoError(t, err)
2019-02-14 21:55:21 +00:00
require.Equal(t, dbxschema, finalSchema, "dbx")
2019-02-14 21:55:21 +00:00
}