From f99b63de51ba54438e650f5dd7d9293eda553e06 Mon Sep 17 00:00:00 2001 From: Marcel Gregoriadis Date: Mon, 5 Sep 2022 12:06:59 +0300 Subject: [PATCH] Add --number flag on dbmate down --- main.go | 12 ++++++++- pkg/dbmate/db.go | 67 +++++++++++++++++++++++------------------------- 2 files changed, 43 insertions(+), 36 deletions(-) diff --git a/main.go b/main.go index 04a777f5..7d6e5062 100644 --- a/main.go +++ b/main.go @@ -150,10 +150,20 @@ func NewApp() *cli.App { EnvVars: []string{"DBMATE_VERBOSE"}, Usage: "print the result of each statement execution", }, + &cli.IntFlag{ + Name: "number", + Aliases: []string{"n"}, + Usage: "number of migrations to rollback (-1 means all)", + DefaultText: "1", + }, }, Action: action(func(db *dbmate.DB, c *cli.Context) error { db.Verbose = c.Bool("verbose") - return db.Rollback() + number := c.Int("number") + if number == 0 { + number = 1 + } + return db.Rollback(number) }), }, { diff --git a/pkg/dbmate/db.go b/pkg/dbmate/db.go index 5c1c3a28..d06c8ad7 100644 --- a/pkg/dbmate/db.go +++ b/pkg/dbmate/db.go @@ -464,7 +464,7 @@ func migrationVersion(filename string) string { } // Rollback rolls back the most recent migration -func (db *DB) Rollback() error { +func (db *DB) Rollback(number int) error { drv, err := db.GetDriver() if err != nil { return err @@ -483,55 +483,52 @@ func (db *DB) Rollback() error { } defer dbutil.MustClose(sqlDB) - applied, err := drv.SelectMigrations(sqlDB, 1) + applied, err := drv.SelectMigrations(sqlDB, number) if err != nil { return err } - // grab most recent applied migration (applied has len=1) - latest := "" - for ver := range applied { - latest = ver - } - if latest == "" { + if len(applied) == 0 { return ErrNoRollback } - filename, err := findMigrationFile(db.MigrationsDir, latest) - if err != nil { - return err - } - - fmt.Fprintf(db.Log, "Rolling back: %s\n", filename) + for ver, _ := range applied { + filename, err := findMigrationFile(db.MigrationsDir, ver) + if err != nil { + return err + } - _, down, err := parseMigration(filepath.Join(db.MigrationsDir, filename)) - if err != nil { - return err - } + fmt.Fprintf(db.Log, "Rolling back: %s\n", filename) - execMigration := func(tx dbutil.Transaction) error { - // rollback migration - result, err := tx.Exec(down.Contents) + _, down, err := parseMigration(filepath.Join(db.MigrationsDir, filename)) if err != nil { return err - } else if db.Verbose { - db.printVerbose(result) } - // remove migration record - return drv.DeleteMigration(tx, latest) - } + execMigration := func(tx dbutil.Transaction) error { + // rollback migration + result, err := tx.Exec(down.Contents) + if err != nil { + return err + } else if db.Verbose { + db.printVerbose(result) + } - if down.Options.Transaction() { - // begin transaction - err = doTransaction(sqlDB, execMigration) - } else { - // run outside of transaction - err = execMigration(sqlDB) - } + // remove migration record + return drv.DeleteMigration(tx, ver) + } - if err != nil { - return err + if down.Options.Transaction() { + // begin transaction + err = doTransaction(sqlDB, execMigration) + } else { + // run outside of transaction + err = execMigration(sqlDB) + } + + if err != nil { + return err + } } // automatically update schema file, silence errors