165 lines
6.0 KiB
Go
165 lines
6.0 KiB
Go
package repository
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"testing"
|
|
"testing/fstest"
|
|
|
|
sqlmock "github.com/DATA-DOG/go-sqlmock"
|
|
"github.com/stretchr/testify/require"
|
|
)
|
|
|
|
func TestValidateMigrationExecutionMode(t *testing.T) {
|
|
t.Run("事务迁移包含CONCURRENTLY会被拒绝", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_add_idx.sql", "CREATE INDEX CONCURRENTLY idx_a ON t(a);")
|
|
require.False(t, nonTx)
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("notx迁移要求CREATE使用IF NOT EXISTS", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_add_idx_notx.sql", "CREATE INDEX CONCURRENTLY idx_a ON t(a);")
|
|
require.False(t, nonTx)
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("notx迁移要求DROP使用IF EXISTS", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_drop_idx_notx.sql", "DROP INDEX CONCURRENTLY idx_a;")
|
|
require.False(t, nonTx)
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("notx迁移禁止事务控制语句", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_add_idx_notx.sql", "BEGIN; CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_a ON t(a); COMMIT;")
|
|
require.False(t, nonTx)
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("notx迁移禁止混用非CONCURRENTLY语句", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_add_idx_notx.sql", "CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_a ON t(a); UPDATE t SET a = 1;")
|
|
require.False(t, nonTx)
|
|
require.Error(t, err)
|
|
})
|
|
|
|
t.Run("notx迁移允许幂等并发索引语句", func(t *testing.T) {
|
|
nonTx, err := validateMigrationExecutionMode("001_add_idx_notx.sql", `
|
|
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_a ON t(a);
|
|
DROP INDEX CONCURRENTLY IF EXISTS idx_b;
|
|
`)
|
|
require.True(t, nonTx)
|
|
require.NoError(t, err)
|
|
})
|
|
}
|
|
|
|
func TestApplyMigrationsFS_NonTransactionalMigration(t *testing.T) {
|
|
db, mock, err := sqlmock.New()
|
|
require.NoError(t, err)
|
|
defer func() { _ = db.Close() }()
|
|
|
|
prepareMigrationsBootstrapExpectations(mock)
|
|
mock.ExpectQuery("SELECT checksum FROM schema_migrations WHERE filename = \\$1").
|
|
WithArgs("001_add_idx_notx.sql").
|
|
WillReturnError(sql.ErrNoRows)
|
|
mock.ExpectExec("CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_a ON t\\(a\\)").
|
|
WillReturnResult(sqlmock.NewResult(0, 0))
|
|
mock.ExpectExec("INSERT INTO schema_migrations \\(filename, checksum\\) VALUES \\(\\$1, \\$2\\)").
|
|
WithArgs("001_add_idx_notx.sql", sqlmock.AnyArg()).
|
|
WillReturnResult(sqlmock.NewResult(1, 1))
|
|
mock.ExpectExec("SELECT pg_advisory_unlock\\(\\$1\\)").
|
|
WithArgs(migrationsAdvisoryLockID).
|
|
WillReturnResult(sqlmock.NewResult(0, 1))
|
|
|
|
fsys := fstest.MapFS{
|
|
"001_add_idx_notx.sql": &fstest.MapFile{
|
|
Data: []byte("CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_a ON t(a);"),
|
|
},
|
|
}
|
|
|
|
err = applyMigrationsFS(context.Background(), db, fsys)
|
|
require.NoError(t, err)
|
|
require.NoError(t, mock.ExpectationsWereMet())
|
|
}
|
|
|
|
func TestApplyMigrationsFS_NonTransactionalMigration_MultiStatements(t *testing.T) {
|
|
db, mock, err := sqlmock.New()
|
|
require.NoError(t, err)
|
|
defer func() { _ = db.Close() }()
|
|
|
|
prepareMigrationsBootstrapExpectations(mock)
|
|
mock.ExpectQuery("SELECT checksum FROM schema_migrations WHERE filename = \\$1").
|
|
WithArgs("001_add_multi_idx_notx.sql").
|
|
WillReturnError(sql.ErrNoRows)
|
|
mock.ExpectExec("CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_a ON t\\(a\\)").
|
|
WillReturnResult(sqlmock.NewResult(0, 0))
|
|
mock.ExpectExec("CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_b ON t\\(b\\)").
|
|
WillReturnResult(sqlmock.NewResult(0, 0))
|
|
mock.ExpectExec("INSERT INTO schema_migrations \\(filename, checksum\\) VALUES \\(\\$1, \\$2\\)").
|
|
WithArgs("001_add_multi_idx_notx.sql", sqlmock.AnyArg()).
|
|
WillReturnResult(sqlmock.NewResult(1, 1))
|
|
mock.ExpectExec("SELECT pg_advisory_unlock\\(\\$1\\)").
|
|
WithArgs(migrationsAdvisoryLockID).
|
|
WillReturnResult(sqlmock.NewResult(0, 1))
|
|
|
|
fsys := fstest.MapFS{
|
|
"001_add_multi_idx_notx.sql": &fstest.MapFile{
|
|
Data: []byte(`
|
|
-- first
|
|
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_a ON t(a);
|
|
-- second
|
|
CREATE INDEX CONCURRENTLY IF NOT EXISTS idx_t_b ON t(b);
|
|
`),
|
|
},
|
|
}
|
|
|
|
err = applyMigrationsFS(context.Background(), db, fsys)
|
|
require.NoError(t, err)
|
|
require.NoError(t, mock.ExpectationsWereMet())
|
|
}
|
|
|
|
func TestApplyMigrationsFS_TransactionalMigration(t *testing.T) {
|
|
db, mock, err := sqlmock.New()
|
|
require.NoError(t, err)
|
|
defer func() { _ = db.Close() }()
|
|
|
|
prepareMigrationsBootstrapExpectations(mock)
|
|
mock.ExpectQuery("SELECT checksum FROM schema_migrations WHERE filename = \\$1").
|
|
WithArgs("001_add_col.sql").
|
|
WillReturnError(sql.ErrNoRows)
|
|
mock.ExpectBegin()
|
|
mock.ExpectExec("ALTER TABLE t ADD COLUMN name TEXT").
|
|
WillReturnResult(sqlmock.NewResult(0, 0))
|
|
mock.ExpectExec("INSERT INTO schema_migrations \\(filename, checksum\\) VALUES \\(\\$1, \\$2\\)").
|
|
WithArgs("001_add_col.sql", sqlmock.AnyArg()).
|
|
WillReturnResult(sqlmock.NewResult(1, 1))
|
|
mock.ExpectCommit()
|
|
mock.ExpectExec("SELECT pg_advisory_unlock\\(\\$1\\)").
|
|
WithArgs(migrationsAdvisoryLockID).
|
|
WillReturnResult(sqlmock.NewResult(0, 1))
|
|
|
|
fsys := fstest.MapFS{
|
|
"001_add_col.sql": &fstest.MapFile{
|
|
Data: []byte("ALTER TABLE t ADD COLUMN name TEXT;"),
|
|
},
|
|
}
|
|
|
|
err = applyMigrationsFS(context.Background(), db, fsys)
|
|
require.NoError(t, err)
|
|
require.NoError(t, mock.ExpectationsWereMet())
|
|
}
|
|
|
|
func prepareMigrationsBootstrapExpectations(mock sqlmock.Sqlmock) {
|
|
mock.ExpectQuery("SELECT pg_try_advisory_lock\\(\\$1\\)").
|
|
WithArgs(migrationsAdvisoryLockID).
|
|
WillReturnRows(sqlmock.NewRows([]string{"pg_try_advisory_lock"}).AddRow(true))
|
|
mock.ExpectExec("CREATE TABLE IF NOT EXISTS schema_migrations").
|
|
WillReturnResult(sqlmock.NewResult(0, 0))
|
|
mock.ExpectQuery("SELECT EXISTS \\(").
|
|
WithArgs("schema_migrations").
|
|
WillReturnRows(sqlmock.NewRows([]string{"exists"}).AddRow(true))
|
|
mock.ExpectQuery("SELECT EXISTS \\(").
|
|
WithArgs("atlas_schema_revisions").
|
|
WillReturnRows(sqlmock.NewRows([]string{"exists"}).AddRow(true))
|
|
mock.ExpectQuery("SELECT COUNT\\(\\*\\) FROM atlas_schema_revisions").
|
|
WillReturnRows(sqlmock.NewRows([]string{"count"}).AddRow(1))
|
|
}
|