cc-backend/internal/repository/migration.go

115 lines
2.5 KiB
Go
Raw Normal View History

2023-02-21 10:57:22 +01:00
// Copyright (C) 2022 NHR@FAU, University Erlangen-Nuremberg.
// All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package repository
import (
"database/sql"
"embed"
"fmt"
"github.com/ClusterCockpit/cc-backend/pkg/log"
"github.com/golang-migrate/migrate/v4"
2023-02-21 14:45:59 +01:00
"github.com/golang-migrate/migrate/v4/database/mysql"
2023-02-21 10:57:22 +01:00
"github.com/golang-migrate/migrate/v4/database/sqlite3"
"github.com/golang-migrate/migrate/v4/source/iofs"
)
const Version uint = 3
2023-02-21 10:57:22 +01:00
2023-02-21 14:45:59 +01:00
//go:embed migrations/*
2023-02-21 10:57:22 +01:00
var migrationFiles embed.FS
func checkDBVersion(backend string, db *sql.DB) error {
2023-02-21 14:45:59 +01:00
var m *migrate.Migrate
2023-02-21 10:57:22 +01:00
switch backend {
case "sqlite3":
2023-02-21 14:45:59 +01:00
driver, err := sqlite3.WithInstance(db, &sqlite3.Config{})
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
d, err := iofs.New(migrationFiles, "migrations/sqlite3")
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
m, err = migrate.NewWithInstance("iofs", d, "sqlite3", driver)
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
case "mysql":
2023-02-21 14:45:59 +01:00
driver, err := mysql.WithInstance(db, &mysql.Config{})
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
d, err := iofs.New(migrationFiles, "migrations/mysql")
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
m, err = migrate.NewWithInstance("iofs", d, "mysql", driver)
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
2023-02-21 10:57:22 +01:00
}
v, _, err := m.Version()
if err != nil {
if err == migrate.ErrNilVersion {
log.Warn("Legacy database without version or missing database file!")
} else {
return err
2023-02-21 10:57:22 +01:00
}
}
if v < Version {
return fmt.Errorf("unsupported database version %d, need %d.\nPlease backup your database file and run cc-backend --migrate-db", v, Version)
2023-02-21 10:57:22 +01:00
}
2023-02-21 13:50:15 +01:00
if v > Version {
return fmt.Errorf("unsupported database version %d, need %d.\nPlease refer to documentation how to downgrade db with external migrate tool", v, Version)
2023-02-21 13:50:15 +01:00
}
return nil
2023-02-21 10:57:22 +01:00
}
func MigrateDB(backend string, db string) error {
2023-02-21 14:45:59 +01:00
var m *migrate.Migrate
2023-02-21 10:57:22 +01:00
switch backend {
case "sqlite3":
2023-02-21 14:45:59 +01:00
d, err := iofs.New(migrationFiles, "migrations/sqlite3")
if err != nil {
log.Fatal(err)
}
m, err = migrate.NewWithSourceInstance("iofs", d, fmt.Sprintf("sqlite3://%s?_foreign_keys=on", db))
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
case "mysql":
2023-02-21 14:45:59 +01:00
d, err := iofs.New(migrationFiles, "migrations/mysql")
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
m, err = migrate.NewWithSourceInstance("iofs", d, fmt.Sprintf("mysql://%s?multiStatements=true", db))
if err != nil {
return err
2023-02-21 14:45:59 +01:00
}
2023-02-21 10:57:22 +01:00
}
2023-02-21 14:45:59 +01:00
2023-02-21 10:57:22 +01:00
if err := m.Up(); err != nil {
2023-02-25 08:26:56 +01:00
if err == migrate.ErrNoChange {
log.Info("DB already up to date!")
} else {
return err
2023-02-25 08:26:56 +01:00
}
2023-02-21 10:57:22 +01:00
}
m.Close()
return nil
2023-02-21 10:57:22 +01:00
}