2023-10-17 06:31:13 +02:00
|
|
|
// Copyright 2023, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2022-07-01 19:48:14 +02:00
|
|
|
package sstore
|
|
|
|
|
|
|
|
import (
|
2023-11-02 22:05:43 +01:00
|
|
|
"errors"
|
2022-07-01 19:48:14 +02:00
|
|
|
"fmt"
|
2023-03-12 22:42:18 +01:00
|
|
|
"io"
|
2023-11-02 22:05:43 +01:00
|
|
|
"io/fs"
|
2022-10-31 20:40:45 +01:00
|
|
|
"log"
|
2023-03-12 22:42:18 +01:00
|
|
|
"os"
|
2022-07-01 19:48:14 +02:00
|
|
|
"strconv"
|
2023-01-23 08:10:18 +01:00
|
|
|
"time"
|
2022-07-01 19:48:14 +02:00
|
|
|
|
|
|
|
_ "github.com/golang-migrate/migrate/v4/database/sqlite3"
|
|
|
|
_ "github.com/golang-migrate/migrate/v4/source/file"
|
2022-10-31 20:24:21 +01:00
|
|
|
"github.com/golang-migrate/migrate/v4/source/iofs"
|
2022-07-01 19:48:14 +02:00
|
|
|
_ "github.com/mattn/go-sqlite3"
|
2023-10-26 07:07:00 +02:00
|
|
|
sh2db "github.com/wavetermdev/waveterm/wavesrv/db"
|
2022-07-01 19:48:14 +02:00
|
|
|
|
|
|
|
"github.com/golang-migrate/migrate/v4"
|
|
|
|
)
|
|
|
|
|
2023-12-16 02:43:54 +01:00
|
|
|
const MaxMigration = 25
|
2023-03-12 22:42:18 +01:00
|
|
|
const MigratePrimaryScreenVersion = 9
|
2023-07-28 20:48:11 +02:00
|
|
|
const CmdScreenSpecialMigration = 13
|
2023-07-31 02:16:43 +02:00
|
|
|
const CmdLineSpecialMigration = 20
|
2023-03-12 22:42:18 +01:00
|
|
|
|
2022-07-01 19:48:14 +02:00
|
|
|
func MakeMigrate() (*migrate.Migrate, error) {
|
2022-10-31 20:24:21 +01:00
|
|
|
fsVar, err := iofs.New(sh2db.MigrationFS, "migrations")
|
2022-07-01 19:48:14 +02:00
|
|
|
if err != nil {
|
2022-10-31 20:24:21 +01:00
|
|
|
return nil, fmt.Errorf("opening iofs: %w", err)
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
2022-10-31 20:24:21 +01:00
|
|
|
// migrationPathUrl := fmt.Sprintf("file://%s", path.Join(wd, "db", "migrations"))
|
2023-03-12 22:42:18 +01:00
|
|
|
dbUrl := fmt.Sprintf("sqlite3://%s", GetDBName())
|
2022-10-31 20:24:21 +01:00
|
|
|
m, err := migrate.NewWithSourceInstance("iofs", fsVar, dbUrl)
|
|
|
|
// m, err := migrate.New(migrationPathUrl, dbUrl)
|
2022-07-01 19:48:14 +02:00
|
|
|
if err != nil {
|
2023-03-12 22:42:18 +01:00
|
|
|
return nil, fmt.Errorf("making migration db[%s]: %w", GetDBName(), err)
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
2023-11-02 22:05:43 +01:00
|
|
|
func copyFile(srcFile string, dstFile string, notFoundOk bool) error {
|
2023-03-12 22:42:18 +01:00
|
|
|
if srcFile == dstFile {
|
|
|
|
return fmt.Errorf("cannot copy %s to itself", srcFile)
|
|
|
|
}
|
|
|
|
srcFd, err := os.Open(srcFile)
|
2023-11-02 22:05:43 +01:00
|
|
|
if notFoundOk && err != nil && errors.Is(err, fs.ErrNotExist) {
|
|
|
|
return nil
|
|
|
|
}
|
2023-03-12 22:42:18 +01:00
|
|
|
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()
|
|
|
|
}
|
|
|
|
|
2023-07-28 20:48:11 +02:00
|
|
|
func MigrateUpStep(m *migrate.Migrate, newVersion uint) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
err := m.Migrate(newVersion)
|
2022-07-01 19:48:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
if newVersion == CmdScreenSpecialMigration {
|
2023-07-31 02:16:43 +02:00
|
|
|
mErr := RunMigration13()
|
2023-07-28 20:48:11 +02:00
|
|
|
if mErr != nil {
|
2023-07-31 02:16:43 +02:00
|
|
|
return fmt.Errorf("migrating to v%d: %w", newVersion, mErr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if newVersion == CmdLineSpecialMigration {
|
|
|
|
mErr := RunMigration20()
|
|
|
|
if mErr != nil {
|
|
|
|
return fmt.Errorf("migrating to v%d: %w", newVersion, mErr)
|
2023-07-28 20:48:11 +02:00
|
|
|
}
|
2023-03-12 22:42:18 +01:00
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
log.Printf("[db] migration v%d, elapsed %v\n", newVersion, time.Since(startTime))
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func MigrateUp(targetVersion uint) error {
|
|
|
|
m, err := MakeMigrate()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
curVersion, dirty, err := MigrateVersion(m)
|
2023-03-12 22:42:18 +01:00
|
|
|
if dirty {
|
|
|
|
return fmt.Errorf("cannot migrate up, database is dirty")
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("cannot get current migration version: %v", err)
|
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
if curVersion >= targetVersion {
|
2023-03-12 22:42:18 +01:00
|
|
|
return nil
|
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
log.Printf("[db] migrating from %d to %d\n", curVersion, targetVersion)
|
2023-03-12 22:42:18 +01:00
|
|
|
log.Printf("[db] backing up database %s to %s\n", DBFileName, DBFileNameBackup)
|
2023-11-02 22:05:43 +01:00
|
|
|
os.Remove(GetDBBackupName()) // don't report error
|
|
|
|
os.Remove(GetDBWALBackupName()) // don't report error
|
|
|
|
err = copyFile(GetDBName(), GetDBBackupName(), false)
|
2023-03-12 22:42:18 +01:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error creating database backup: %v", err)
|
|
|
|
}
|
2023-11-02 22:05:43 +01:00
|
|
|
err = copyFile(GetDBWALName(), GetDBWALBackupName(), true)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error creating database(wal) backup: %v", err)
|
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
for newVersion := curVersion + 1; newVersion <= targetVersion; newVersion++ {
|
|
|
|
err = MigrateUpStep(m, newVersion)
|
|
|
|
if err != nil {
|
2023-07-31 03:46:57 +02:00
|
|
|
return fmt.Errorf("during migration v%d: %w", newVersion, err)
|
2023-07-28 20:48:11 +02:00
|
|
|
}
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
log.Printf("[db] migration done, new version = %d\n", targetVersion)
|
2022-07-01 19:48:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-07-28 20:48:11 +02:00
|
|
|
// returns curVersion, dirty, error
|
|
|
|
func MigrateVersion(m *migrate.Migrate) (uint, bool, error) {
|
|
|
|
if m == nil {
|
|
|
|
var err error
|
|
|
|
m, err = MakeMigrate()
|
|
|
|
if err != nil {
|
|
|
|
return 0, false, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
curVersion, dirty, err := m.Version()
|
|
|
|
if err == migrate.ErrNilVersion {
|
|
|
|
return 0, false, nil
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
return curVersion, dirty, err
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2023-07-28 20:48:11 +02:00
|
|
|
curVersion, _, _ := MigrateVersion(nil)
|
|
|
|
if curVersion == n {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if curVersion < n {
|
|
|
|
return MigrateUp(n)
|
|
|
|
}
|
2022-07-01 19:48:14 +02:00
|
|
|
m, err := MakeMigrate()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = m.Migrate(n)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TryMigrateUp() error {
|
2023-07-28 20:48:11 +02:00
|
|
|
curVersion, _, _ := MigrateVersion(nil)
|
|
|
|
log.Printf("[db] db version = %d\n", curVersion)
|
|
|
|
if curVersion >= MaxMigration {
|
|
|
|
return nil
|
2022-07-01 19:48:14 +02:00
|
|
|
}
|
2023-07-28 20:48:11 +02:00
|
|
|
err := MigrateUp(MaxMigration)
|
2022-07-01 19:48:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return MigratePrintVersion()
|
|
|
|
}
|
|
|
|
|
|
|
|
func MigratePrintVersion() error {
|
2023-07-28 20:48:11 +02:00
|
|
|
version, dirty, err := MigrateVersion(nil)
|
2022-07-01 19:48:14 +02:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error getting db version: %v", err)
|
|
|
|
}
|
|
|
|
if dirty {
|
|
|
|
return fmt.Errorf("error db is dirty, version=%d", version)
|
|
|
|
}
|
2022-10-31 20:40:45 +01:00
|
|
|
log.Printf("[db] version=%d\n", version)
|
2022-07-01 19:48:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func MigrateCommandOpts(opts []string) error {
|
|
|
|
var err error
|
|
|
|
if opts[0] == "--migrate-up" {
|
2023-03-12 22:42:18 +01:00
|
|
|
fmt.Printf("migrate-up %v\n", GetDBName())
|
2023-01-23 08:10:18 +01:00
|
|
|
time.Sleep(3 * time.Second)
|
2023-07-28 20:48:11 +02:00
|
|
|
err = MigrateUp(MaxMigration)
|
2022-07-01 19:48:14 +02:00
|
|
|
} else if opts[0] == "--migrate-down" {
|
2023-03-12 22:42:18 +01:00
|
|
|
fmt.Printf("migrate-down %v\n", GetDBName())
|
2023-01-23 08:10:18 +01:00
|
|
|
time.Sleep(3 * time.Second)
|
2022-07-01 19:48:14 +02:00
|
|
|
err = MigrateDown()
|
|
|
|
} else if opts[0] == "--migrate-goto" {
|
|
|
|
n, err := strconv.Atoi(opts[1])
|
|
|
|
if err == nil {
|
2023-03-12 22:42:18 +01:00
|
|
|
fmt.Printf("migrate-goto %v => %d\n", GetDBName(), n)
|
2023-01-23 08:10:18 +01:00
|
|
|
time.Sleep(3 * time.Second)
|
2022-07-01 19:48:14 +02:00
|
|
|
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()
|
|
|
|
}
|