storj/private/migrate/versions_test.go
Yaroslav Vorobiov 139a7ee959 private/migrate: add ablity to create dbs during migration
Use tagsql.DB pointer as step database, to propagate changes
back and forth between actual database and migration.
Adds CreateDB operation to the migration step to be able to
create new dbs before executing migration action.
Adjusts storagenode database migration to use inner tagsql.DB
pointer of each database as step.DB.
Adjusts satellite dabase migration, adds proxy migrationDB field
to satellite db that wraps itself as tagsql.DB, pointer of which
is used as step.DB.

Change-Id: Ifed4de5b01a356cf7b37db64d2eaeb7b61982c5c
2020-10-15 15:28:04 +03:00

327 lines
7.7 KiB
Go

// Copyright (C) 2019 Storj Labs, Inc.
// See LICENSE for copying information.
package migrate_test
import (
"context"
"database/sql"
"fmt"
"io/ioutil"
"os"
"strings"
"testing"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/zeebo/errs"
"go.uber.org/zap"
"storj.io/common/testcontext"
"storj.io/storj/private/dbutil/pgtest"
"storj.io/storj/private/dbutil/tempdb"
"storj.io/storj/private/migrate"
"storj.io/storj/private/tagsql"
)
func TestBasicMigrationSqliteNoRebind(t *testing.T) {
db, err := tagsql.Open("sqlite3", ":memory:")
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
ctx := testcontext.New(t)
defer ctx.Cleanup()
basicMigration(ctx, t, db, db)
}
func TestBasicMigrationSqlite(t *testing.T) {
db, err := tagsql.Open("sqlite3", ":memory:")
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
ctx := testcontext.New(t)
defer ctx.Cleanup()
basicMigration(ctx, t, db, &sqliteDB{DB: db})
}
func TestBasicMigration(t *testing.T) {
pgtest.Run(t, func(ctx *testcontext.Context, t *testing.T, connstr string) {
db, err := tempdb.OpenUnique(ctx, connstr, "create-")
if err != nil {
t.Fatal(err)
}
defer func() { assert.NoError(t, db.Close()) }()
basicMigration(ctx, t, db.DB, &postgresDB{DB: db.DB})
})
}
func basicMigration(ctx *testcontext.Context, t *testing.T, db tagsql.DB, testDB tagsql.DB) {
dbName := strings.ToLower(`versions_` + strings.ReplaceAll(t.Name(), "/", "_"))
defer func() { assert.NoError(t, dropTables(ctx, db, dbName, "users")) }()
/* #nosec G306 */ // This is a test besides the file contains just test data.
err := ioutil.WriteFile(ctx.File("alpha.txt"), []byte("test"), 0644)
require.NoError(t, err)
m := migrate.Migration{
Table: dbName,
Steps: []*migrate.Step{
{
DB: &testDB,
Description: "Initialize Table",
Version: 1,
Action: migrate.SQL{
`CREATE TABLE users (id int)`,
`INSERT INTO users (id) VALUES (1)`,
},
},
{
DB: &testDB,
Description: "Move files",
Version: 2,
Action: migrate.Func(func(_ context.Context, log *zap.Logger, _ tagsql.DB, tx tagsql.Tx) error {
return os.Rename(ctx.File("alpha.txt"), ctx.File("beta.txt"))
}),
},
},
}
dbVersion, err := m.CurrentVersion(ctx, nil, testDB)
assert.NoError(t, err)
assert.Equal(t, dbVersion, -1)
err = m.Run(ctx, zap.NewNop())
assert.NoError(t, err)
dbVersion, err = m.CurrentVersion(ctx, nil, testDB)
assert.NoError(t, err)
assert.Equal(t, dbVersion, 2)
m2 := migrate.Migration{
Table: dbName,
Steps: []*migrate.Step{
{
DB: &testDB,
Version: 3,
},
},
}
dbVersion, err = m2.CurrentVersion(ctx, nil, testDB)
assert.NoError(t, err)
assert.Equal(t, dbVersion, 2)
var version int
/* #nosec G202 */ // This is a test besides the dbName value is generated in
// a controlled way
err = db.QueryRow(ctx, `SELECT MAX(version) FROM `+dbName).Scan(&version)
assert.NoError(t, err)
assert.Equal(t, 2, version)
var id int
err = db.QueryRow(ctx, `SELECT MAX(id) FROM users`).Scan(&id)
assert.NoError(t, err)
assert.Equal(t, 1, id)
// file not exists
_, err = os.Stat(ctx.File("alpha.txt"))
assert.Error(t, err)
// file exists
_, err = os.Stat(ctx.File("beta.txt"))
assert.NoError(t, err)
data, err := ioutil.ReadFile(ctx.File("beta.txt"))
assert.NoError(t, err)
assert.Equal(t, []byte("test"), data)
}
func TestMultipleMigrationSqlite(t *testing.T) {
db, err := tagsql.Open("sqlite3", ":memory:")
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
multipleMigration(t, db, &sqliteDB{DB: db})
}
func TestMultipleMigrationPostgres(t *testing.T) {
connstr := pgtest.PickPostgres(t)
db, err := tagsql.Open("pgx", connstr)
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
multipleMigration(t, db, &postgresDB{DB: db})
}
func multipleMigration(t *testing.T, db tagsql.DB, testDB tagsql.DB) {
ctx := testcontext.New(t)
defer ctx.Cleanup()
dbName := strings.ToLower(`versions_` + t.Name())
defer func() { assert.NoError(t, dropTables(ctx, db, dbName)) }()
steps := 0
m := migrate.Migration{
Table: dbName,
Steps: []*migrate.Step{
{
DB: &testDB,
Description: "Step 1",
Version: 1,
Action: migrate.Func(func(ctx context.Context, log *zap.Logger, _ tagsql.DB, tx tagsql.Tx) error {
steps++
return nil
}),
},
{
DB: &testDB,
Description: "Step 2",
Version: 2,
Action: migrate.Func(func(ctx context.Context, log *zap.Logger, _ tagsql.DB, tx tagsql.Tx) error {
steps++
return nil
}),
},
},
}
err := m.Run(ctx, zap.NewNop())
assert.NoError(t, err)
assert.Equal(t, 2, steps)
m.Steps = append(m.Steps, &migrate.Step{
DB: &testDB,
Description: "Step 3",
Version: 3,
Action: migrate.Func(func(ctx context.Context, log *zap.Logger, _ tagsql.DB, tx tagsql.Tx) error {
steps++
return nil
}),
})
err = m.Run(ctx, zap.NewNop())
assert.NoError(t, err)
var version int
/* #nosec G202 */ // This is a test besides the dbName value is generated in
// a controlled way
err = db.QueryRow(ctx, `SELECT MAX(version) FROM `+dbName).Scan(&version)
assert.NoError(t, err)
assert.Equal(t, 3, version)
assert.Equal(t, 3, steps)
}
func TestFailedMigrationSqlite(t *testing.T) {
db, err := tagsql.Open("sqlite3", ":memory:")
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
failedMigration(t, db, &sqliteDB{DB: db})
}
func TestFailedMigrationPostgres(t *testing.T) {
connstr := pgtest.PickPostgres(t)
db, err := tagsql.Open("pgx", connstr)
require.NoError(t, err)
defer func() { assert.NoError(t, db.Close()) }()
failedMigration(t, db, &postgresDB{DB: db})
}
func failedMigration(t *testing.T, db tagsql.DB, testDB tagsql.DB) {
ctx := testcontext.New(t)
defer ctx.Cleanup()
dbName := strings.ToLower(`versions_` + t.Name())
defer func() { assert.NoError(t, dropTables(ctx, db, dbName)) }()
m := migrate.Migration{
Table: dbName,
Steps: []*migrate.Step{
{
DB: &testDB,
Description: "Step 1",
Version: 1,
Action: migrate.Func(func(ctx context.Context, log *zap.Logger, _ tagsql.DB, tx tagsql.Tx) error {
return fmt.Errorf("migration failed")
}),
},
},
}
err := m.Run(ctx, zap.NewNop())
require.Error(t, err, "migration failed")
var version sql.NullInt64
/* #nosec G202 */ // This is a test besides the dbName value is generated in
// a controlled way
err = db.QueryRow(ctx, `SELECT MAX(version) FROM `+dbName).Scan(&version)
assert.NoError(t, err)
assert.Equal(t, false, version.Valid)
}
func TestTargetVersion(t *testing.T) {
m := migrate.Migration{
Table: "test",
Steps: []*migrate.Step{
{
Description: "Step 1",
Version: 1,
Action: migrate.SQL{},
},
{
Description: "Step 2",
Version: 2,
Action: migrate.SQL{},
},
{
Description: "Step 2.2",
Version: 2,
Action: migrate.SQL{},
},
{
Description: "Step 3",
Version: 3,
Action: migrate.SQL{},
},
},
}
testedMigration := m.TargetVersion(2)
assert.Equal(t, 3, len(testedMigration.Steps))
}
func TestInvalidStepsOrder(t *testing.T) {
m := migrate.Migration{
Table: "test",
Steps: []*migrate.Step{
{
Version: 0,
},
{
Version: 1,
},
{
Version: 4,
},
{
Version: 2,
},
},
}
err := m.ValidateSteps()
require.Error(t, err, "migrate: steps have incorrect order")
}
func dropTables(ctx context.Context, db tagsql.DB, names ...string) error {
var errlist errs.Group
for _, name := range names {
_, err := db.Exec(ctx, `DROP TABLE `+name)
errlist.Add(err)
}
return errlist.Err()
}