25d00e3518
Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
123 lines
3.7 KiB
Go
123 lines
3.7 KiB
Go
package infra
|
|
|
|
import (
|
|
"database/sql"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
_ "github.com/mattn/go-sqlite3"
|
|
)
|
|
|
|
func openMigrationTestDB(t *testing.T) *sql.DB {
|
|
t.Helper()
|
|
db, err := sql.Open("sqlite3", ":memory:")
|
|
if err != nil {
|
|
t.Fatalf("cannot open test DB: %v", err)
|
|
}
|
|
t.Cleanup(func() { db.Close() })
|
|
return db
|
|
}
|
|
|
|
func writeMigrationFile(t *testing.T, dir, filename, content string) {
|
|
t.Helper()
|
|
if err := os.WriteFile(filepath.Join(dir, filename), []byte(content), 0o644); err != nil {
|
|
t.Fatalf("cannot write migration file %s: %v", filename, err)
|
|
}
|
|
}
|
|
|
|
func TestMigrationUp(t *testing.T) {
|
|
t.Run("base de datos vacia aplica todas las migraciones", func(t *testing.T) {
|
|
db := openMigrationTestDB(t)
|
|
dir := t.TempDir()
|
|
|
|
writeMigrationFile(t, dir, "001_create_users.sql",
|
|
"-- +up\nCREATE TABLE users (id TEXT PRIMARY KEY, name TEXT NOT NULL);\n-- +down\nDROP TABLE IF EXISTS users;\n")
|
|
writeMigrationFile(t, dir, "002_create_roles.sql",
|
|
"-- +up\nCREATE TABLE roles (id TEXT PRIMARY KEY);\n-- +down\nDROP TABLE IF EXISTS roles;\n")
|
|
|
|
applied, err := MigrationUp(db, dir)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if len(applied) != 2 {
|
|
t.Errorf("applied count: got %d, want 2", len(applied))
|
|
}
|
|
if applied[0].Version != 1 || applied[1].Version != 2 {
|
|
t.Errorf("applied versions: got %v", []int{applied[0].Version, applied[1].Version})
|
|
}
|
|
|
|
// Verify tables were created
|
|
var count int
|
|
if err := db.QueryRow("SELECT COUNT(*) FROM users").Scan(&count); err != nil {
|
|
t.Errorf("users table not created: %v", err)
|
|
}
|
|
if err := db.QueryRow("SELECT COUNT(*) FROM roles").Scan(&count); err != nil {
|
|
t.Errorf("roles table not created: %v", err)
|
|
}
|
|
})
|
|
|
|
t.Run("migraciones ya aplicadas se omiten", func(t *testing.T) {
|
|
db := openMigrationTestDB(t)
|
|
dir := t.TempDir()
|
|
|
|
writeMigrationFile(t, dir, "001_create_users.sql",
|
|
"-- +up\nCREATE TABLE users (id TEXT PRIMARY KEY);\n-- +down\nDROP TABLE IF EXISTS users;\n")
|
|
writeMigrationFile(t, dir, "002_create_roles.sql",
|
|
"-- +up\nCREATE TABLE roles (id TEXT PRIMARY KEY);\n-- +down\nDROP TABLE IF EXISTS roles;\n")
|
|
|
|
// Apply all
|
|
_, err := MigrationUp(db, dir)
|
|
if err != nil {
|
|
t.Fatalf("first up failed: %v", err)
|
|
}
|
|
|
|
// Apply again — should apply nothing
|
|
applied, err := MigrationUp(db, dir)
|
|
if err != nil {
|
|
t.Fatalf("second up failed: %v", err)
|
|
}
|
|
if len(applied) != 0 {
|
|
t.Errorf("expected 0 applied on second run, got %d", len(applied))
|
|
}
|
|
})
|
|
|
|
t.Run("migracion con SQL invalido retorna error y deja las anteriores aplicadas", func(t *testing.T) {
|
|
db := openMigrationTestDB(t)
|
|
dir := t.TempDir()
|
|
|
|
writeMigrationFile(t, dir, "001_create_users.sql",
|
|
"-- +up\nCREATE TABLE users (id TEXT PRIMARY KEY);\n-- +down\nDROP TABLE IF EXISTS users;\n")
|
|
writeMigrationFile(t, dir, "002_bad_sql.sql",
|
|
"-- +up\nTHIS IS NOT VALID SQL!!!;\n-- +down\n\n")
|
|
|
|
applied, err := MigrationUp(db, dir)
|
|
if err == nil {
|
|
t.Fatal("expected error for invalid SQL, got nil")
|
|
}
|
|
// Version 1 should be applied
|
|
if len(applied) != 1 || applied[0].Version != 1 {
|
|
t.Errorf("expected [1] applied before failure, got versions: %v", applied)
|
|
}
|
|
|
|
// users table should still exist (migration 1 committed)
|
|
var count int
|
|
if err2 := db.QueryRow("SELECT COUNT(*) FROM users").Scan(&count); err2 != nil {
|
|
t.Errorf("users table not present after partial apply: %v", err2)
|
|
}
|
|
})
|
|
|
|
t.Run("directorio sin archivos sql no aplica nada", func(t *testing.T) {
|
|
db := openMigrationTestDB(t)
|
|
dir := t.TempDir()
|
|
|
|
applied, err := MigrationUp(db, dir)
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if len(applied) != 0 {
|
|
t.Errorf("expected 0 applied for empty dir, got %d", len(applied))
|
|
}
|
|
})
|
|
}
|