2019-05-16 19:14:32 +02:00
|
|
|
package upgrades
|
|
|
|
|
|
|
|
import (
|
|
|
|
"database/sql"
|
2022-01-07 15:45:20 +01:00
|
|
|
"errors"
|
2019-05-16 19:14:32 +02:00
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
log "maunium.net/go/maulogger/v2"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Dialect int
|
|
|
|
|
|
|
|
const (
|
|
|
|
Postgres Dialect = iota
|
|
|
|
SQLite
|
|
|
|
)
|
|
|
|
|
2020-07-10 14:56:45 +02:00
|
|
|
func (dialect Dialect) String() string {
|
|
|
|
switch dialect {
|
|
|
|
case Postgres:
|
|
|
|
return "postgres"
|
|
|
|
case SQLite:
|
|
|
|
return "sqlite3"
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-25 16:25:19 +02:00
|
|
|
type upgradeFunc func(*sql.Tx, context) error
|
|
|
|
|
|
|
|
type context struct {
|
|
|
|
dialect Dialect
|
|
|
|
db *sql.DB
|
|
|
|
log log.Logger
|
|
|
|
}
|
2019-05-16 19:14:32 +02:00
|
|
|
|
|
|
|
type upgrade struct {
|
|
|
|
message string
|
2019-08-25 16:25:19 +02:00
|
|
|
fn upgradeFunc
|
2019-05-16 19:14:32 +02:00
|
|
|
}
|
|
|
|
|
2022-03-05 20:22:31 +01:00
|
|
|
const NumberOfUpgrades = 39
|
2019-05-24 01:33:26 +02:00
|
|
|
|
|
|
|
var upgrades [NumberOfUpgrades]upgrade
|
2019-05-16 19:14:32 +02:00
|
|
|
|
2022-02-06 20:03:04 +01:00
|
|
|
var UnsupportedDatabaseVersion = fmt.Errorf("unsupported database schema version")
|
2019-05-28 20:29:43 +02:00
|
|
|
|
2019-08-25 16:25:19 +02:00
|
|
|
func GetVersion(db *sql.DB) (int, error) {
|
2019-05-16 19:14:32 +02:00
|
|
|
_, err := db.Exec("CREATE TABLE IF NOT EXISTS version (version INTEGER)")
|
|
|
|
if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
|
|
|
|
version := 0
|
2022-01-07 15:45:20 +01:00
|
|
|
err = db.QueryRow("SELECT version FROM version LIMIT 1").Scan(&version)
|
|
|
|
if err != nil && !errors.Is(err, sql.ErrNoRows) {
|
|
|
|
return -1, err
|
2019-05-16 19:14:32 +02:00
|
|
|
}
|
|
|
|
return version, nil
|
|
|
|
}
|
|
|
|
|
2019-08-25 16:25:19 +02:00
|
|
|
func SetVersion(tx *sql.Tx, version int) error {
|
2019-05-16 19:14:32 +02:00
|
|
|
_, err := tx.Exec("DELETE FROM version")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
_, err = tx.Exec("INSERT INTO version (version) VALUES ($1)", version)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-08-19 18:17:19 +02:00
|
|
|
func execMany(tx *sql.Tx, queries ...string) error {
|
|
|
|
for _, query := range queries {
|
|
|
|
_, err := tx.Exec(query)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-05-16 19:14:32 +02:00
|
|
|
func Run(log log.Logger, dialectName string, db *sql.DB) error {
|
|
|
|
var dialect Dialect
|
|
|
|
switch strings.ToLower(dialectName) {
|
|
|
|
case "postgres":
|
|
|
|
dialect = Postgres
|
|
|
|
case "sqlite3":
|
|
|
|
dialect = SQLite
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("unknown dialect %s", dialectName)
|
|
|
|
}
|
|
|
|
|
2019-08-25 16:25:19 +02:00
|
|
|
version, err := GetVersion(db)
|
2019-05-16 19:14:32 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-05-28 20:29:43 +02:00
|
|
|
if version > NumberOfUpgrades {
|
2022-02-06 19:43:36 +01:00
|
|
|
return fmt.Errorf("%w: currently on v%d, latest known: v%d", UnsupportedDatabaseVersion, version, NumberOfUpgrades)
|
2019-05-28 20:29:43 +02:00
|
|
|
}
|
|
|
|
|
2019-05-24 01:33:26 +02:00
|
|
|
log.Infofln("Database currently on v%d, latest: v%d", version, NumberOfUpgrades)
|
2021-08-19 18:17:19 +02:00
|
|
|
for i, upgradeItem := range upgrades[version:] {
|
2021-10-30 22:40:38 +02:00
|
|
|
if upgradeItem.fn == nil {
|
|
|
|
continue
|
|
|
|
}
|
2021-08-19 18:17:19 +02:00
|
|
|
log.Infofln("Upgrading database to v%d: %s", version+i+1, upgradeItem.message)
|
|
|
|
var tx *sql.Tx
|
|
|
|
tx, err = db.Begin()
|
2019-05-16 19:14:32 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-08-19 18:17:19 +02:00
|
|
|
err = upgradeItem.fn(tx, context{dialect, db, log})
|
2019-05-16 19:14:32 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-08-25 16:25:19 +02:00
|
|
|
err = SetVersion(tx, version+i+1)
|
2019-05-16 19:14:32 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = tx.Commit()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|