monorepo/cloud/maplefile-backend/pkg/storage/database/cassandradb/migration.go

146 lines
4.1 KiB
Go

// File Path: monorepo/cloud/maplefile-backend/pkg/storage/database/cassandradb/migration.go
package cassandradb
import (
"fmt"
"go.uber.org/zap"
"github.com/golang-migrate/migrate/v4"
_ "github.com/golang-migrate/migrate/v4/database/cassandra"
_ "github.com/golang-migrate/migrate/v4/source/file"
"codeberg.org/mapleopentech/monorepo/cloud/maplefile-backend/config"
)
// Migrator handles database schema migrations
// This encapsulates all migration logic and makes it testable
type Migrator struct {
config config.DatabaseConfig
logger *zap.Logger
}
// NewMigrator creates a new migration manager that works with fx dependency injection
func NewMigrator(cfg *config.Configuration, logger *zap.Logger) *Migrator {
return &Migrator{
config: cfg.Database,
logger: logger.Named("Migrator"),
}
}
// Up runs all pending migrations with dirty state recovery
func (m *Migrator) Up() error {
m.logger.Info("Creating migrator")
migrateInstance, err := m.createMigrate()
if err != nil {
return fmt.Errorf("failed to create migrator: %w", err)
}
defer migrateInstance.Close()
m.logger.Info("Checking migration version")
version, dirty, err := migrateInstance.Version()
if err != nil && err != migrate.ErrNilVersion {
return fmt.Errorf("failed to get migration version: %w", err)
}
if dirty {
m.logger.Warn("Database is in dirty state, attempting to force clean state",
zap.Uint("version", version))
if err := migrateInstance.Force(int(version)); err != nil {
return fmt.Errorf("failed to force clean migration state: %w", err)
}
}
// Run migrations
if err := migrateInstance.Up(); err != nil && err != migrate.ErrNoChange {
return fmt.Errorf("failed to run migrations: %w", err)
}
// Get final version
finalVersion, _, err := migrateInstance.Version()
if err != nil && err != migrate.ErrNilVersion {
m.logger.Warn("Could not get final migration version",
zap.Error(err))
} else if err != migrate.ErrNilVersion {
m.logger.Info("Database migrations completed successfully",
zap.Uint("version", finalVersion))
} else {
m.logger.Info("Database migrations completed successfully (no migrations applied)")
}
return nil
}
// Down rolls back the last migration
// Useful for development and rollback scenarios
func (m *Migrator) Down() error {
migrate, err := m.createMigrate()
if err != nil {
return fmt.Errorf("failed to create migrator: %w", err)
}
defer migrate.Close()
if err := migrate.Steps(-1); err != nil {
return fmt.Errorf("failed to rollback migration: %w", err)
}
return nil
}
// Version returns the current migration version
func (m *Migrator) Version() (uint, bool, error) {
migrate, err := m.createMigrate()
if err != nil {
return 0, false, fmt.Errorf("failed to create migrator: %w", err)
}
defer migrate.Close()
return migrate.Version()
}
// ForceVersion forces the migration version (useful for fixing dirty states)
func (m *Migrator) ForceVersion(version int) error {
migrateInstance, err := m.createMigrate()
if err != nil {
return fmt.Errorf("failed to create migrator: %w", err)
}
defer migrateInstance.Close()
if err := migrateInstance.Force(version); err != nil {
return fmt.Errorf("failed to force version %d: %w", version, err)
}
m.logger.Info("Successfully forced migration version",
zap.Int("version", version))
return nil
}
// createMigrate creates a migrate instance with proper configuration
func (m *Migrator) createMigrate() (*migrate.Migrate, error) {
// Build Cassandra connection string
// Format: cassandra://host:port/keyspace?consistency=level
databaseURL := fmt.Sprintf("cassandra://%s/%s?consistency=%s",
m.config.Hosts[0], // Use first host for migrations
m.config.Keyspace,
m.config.Consistency,
)
// Add authentication if configured
if m.config.Username != "" && m.config.Password != "" {
databaseURL = fmt.Sprintf("cassandra://%s:%s@%s/%s?consistency=%s",
m.config.Username,
m.config.Password,
m.config.Hosts[0],
m.config.Keyspace,
m.config.Consistency,
)
}
// Create migrate instance
migrate, err := migrate.New(m.config.MigrationsPath, databaseURL)
if err != nil {
return nil, fmt.Errorf("failed to initialize migrate: %w", err)
}
return migrate, nil
}