mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-24 22:01:33 +01:00
176 lines
4.0 KiB
Go
176 lines
4.0 KiB
Go
package sstore
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"strconv"
|
|
"time"
|
|
|
|
_ "github.com/golang-migrate/migrate/v4/database/sqlite3"
|
|
_ "github.com/golang-migrate/migrate/v4/source/file"
|
|
"github.com/golang-migrate/migrate/v4/source/iofs"
|
|
_ "github.com/mattn/go-sqlite3"
|
|
sh2db "github.com/scripthaus-dev/sh2-server/db"
|
|
|
|
"github.com/golang-migrate/migrate/v4"
|
|
)
|
|
|
|
const MaxMigration = 17
|
|
const MigratePrimaryScreenVersion = 9
|
|
|
|
func MakeMigrate() (*migrate.Migrate, error) {
|
|
fsVar, err := iofs.New(sh2db.MigrationFS, "migrations")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("opening iofs: %w", err)
|
|
}
|
|
// migrationPathUrl := fmt.Sprintf("file://%s", path.Join(wd, "db", "migrations"))
|
|
dbUrl := fmt.Sprintf("sqlite3://%s", GetDBName())
|
|
m, err := migrate.NewWithSourceInstance("iofs", fsVar, dbUrl)
|
|
// m, err := migrate.New(migrationPathUrl, dbUrl)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("making migration db[%s]: %w", GetDBName(), err)
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func copyFile(srcFile string, dstFile string) error {
|
|
if srcFile == dstFile {
|
|
return fmt.Errorf("cannot copy %s to itself", srcFile)
|
|
}
|
|
srcFd, err := os.Open(srcFile)
|
|
if err != nil {
|
|
return fmt.Errorf("cannot open %s: %v", err)
|
|
}
|
|
defer srcFd.Close()
|
|
dstFd, err := os.OpenFile(dstFile, os.O_RDWR|os.O_CREATE|os.O_TRUNC, 0600)
|
|
if err != nil {
|
|
return fmt.Errorf("cannot open destination file %s: %v", err)
|
|
}
|
|
_, err = io.Copy(dstFd, srcFd)
|
|
if err != nil {
|
|
dstFd.Close()
|
|
return fmt.Errorf("error copying file: %v", err)
|
|
}
|
|
return dstFd.Close()
|
|
}
|
|
|
|
func MigrateUp() error {
|
|
m, err := MakeMigrate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
curVersion, dirty, err := m.Version()
|
|
if err == migrate.ErrNilVersion {
|
|
curVersion = 0
|
|
err = nil
|
|
}
|
|
if dirty {
|
|
return fmt.Errorf("cannot migrate up, database is dirty")
|
|
}
|
|
if err != nil {
|
|
return fmt.Errorf("cannot get current migration version: %v", err)
|
|
}
|
|
if curVersion >= MaxMigration {
|
|
return nil
|
|
}
|
|
log.Printf("[db] migrating from %d to %d\n", curVersion, MaxMigration)
|
|
log.Printf("[db] backing up database %s to %s\n", DBFileName, DBFileNameBackup)
|
|
err = copyFile(GetDBName(), GetDBBackupName())
|
|
if err != nil {
|
|
return fmt.Errorf("error creating database backup: %v", err)
|
|
}
|
|
startTime := time.Now()
|
|
err = m.Migrate(MaxMigration)
|
|
log.Printf("[db] migration took %v\n", time.Since(startTime))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func MigrateVersion() (uint, bool, error) {
|
|
m, err := MakeMigrate()
|
|
if err != nil {
|
|
return 0, false, err
|
|
}
|
|
return m.Version()
|
|
}
|
|
|
|
func MigrateDown() error {
|
|
m, err := MakeMigrate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = m.Down()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func MigrateGoto(n uint) error {
|
|
m, err := MakeMigrate()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = m.Migrate(n)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func TryMigrateUp() error {
|
|
err := MigrateUp()
|
|
if err != nil && err.Error() == migrate.ErrNoChange.Error() {
|
|
err = nil
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return MigratePrintVersion()
|
|
}
|
|
|
|
func MigratePrintVersion() error {
|
|
version, dirty, err := MigrateVersion()
|
|
if err != nil {
|
|
return fmt.Errorf("error getting db version: %v", err)
|
|
}
|
|
if dirty {
|
|
return fmt.Errorf("error db is dirty, version=%d", version)
|
|
}
|
|
log.Printf("[db] version=%d\n", version)
|
|
return nil
|
|
}
|
|
|
|
func MigrateCommandOpts(opts []string) error {
|
|
var err error
|
|
if opts[0] == "--migrate-up" {
|
|
fmt.Printf("migrate-up %v\n", GetDBName())
|
|
time.Sleep(3 * time.Second)
|
|
err = MigrateUp()
|
|
} else if opts[0] == "--migrate-down" {
|
|
fmt.Printf("migrate-down %v\n", GetDBName())
|
|
time.Sleep(3 * time.Second)
|
|
err = MigrateDown()
|
|
} else if opts[0] == "--migrate-goto" {
|
|
n, err := strconv.Atoi(opts[1])
|
|
if err == nil {
|
|
fmt.Printf("migrate-goto %v => %d\n", GetDBName(), n)
|
|
time.Sleep(3 * time.Second)
|
|
err = MigrateGoto(uint(n))
|
|
}
|
|
} else {
|
|
err = fmt.Errorf("invalid migration command")
|
|
}
|
|
if err != nil && err.Error() == migrate.ErrNoChange.Error() {
|
|
err = nil
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return MigratePrintVersion()
|
|
}
|