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)) }