mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
* [enhancement]: Provide native backup/restore mechanism - db.go: add Backup/Restore functions that utilize Sqlite's built-in online backup mechanism - support automatic backup with schedule, limit number of files - provide commands to manually backup/restore Navidrome Notes: `Step(-1)` results in a read-only lock being held for the entire duration of the backup. This will block out any other write operation (and may hold additional locks. An alternate implementation that doesn't block but instead retries is available at https://www.sqlite.org/backup.html#:~:text=of%20a%20Running-,database,-%2F*%0A**%20Perform%20an%20online (easily adaptable to go), but has the potential problem of continually getting restarted by background writes. Additionally, the restore should still only be called when Navidrome is offline, as the restore process does not run migrations that are missing. * remove empty line * add more granular backup schedule * do not remove files when bypass is set * move prune * refactor: small nitpicks * change commands and flags * tests, return path from backup * refactor: small nitpicks --------- Co-authored-by: Deluan <deluan@navidrome.org>
216 lines
4.6 KiB
Go
216 lines
4.6 KiB
Go
package db
|
|
|
|
import (
|
|
"context"
|
|
"database/sql"
|
|
"embed"
|
|
"fmt"
|
|
"runtime"
|
|
"time"
|
|
|
|
"github.com/mattn/go-sqlite3"
|
|
"github.com/navidrome/navidrome/conf"
|
|
_ "github.com/navidrome/navidrome/db/migrations"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/utils/hasher"
|
|
"github.com/navidrome/navidrome/utils/singleton"
|
|
"github.com/pressly/goose/v3"
|
|
)
|
|
|
|
var (
|
|
Driver = "sqlite3"
|
|
Path string
|
|
)
|
|
|
|
//go:embed migrations/*.sql
|
|
var embedMigrations embed.FS
|
|
|
|
const migrationsFolder = "migrations"
|
|
|
|
type DB interface {
|
|
ReadDB() *sql.DB
|
|
WriteDB() *sql.DB
|
|
Close()
|
|
|
|
Backup(ctx context.Context) (string, error)
|
|
Prune(ctx context.Context) (int, error)
|
|
Restore(ctx context.Context, path string) error
|
|
}
|
|
|
|
type db struct {
|
|
readDB *sql.DB
|
|
writeDB *sql.DB
|
|
}
|
|
|
|
func (d *db) ReadDB() *sql.DB {
|
|
return d.readDB
|
|
}
|
|
|
|
func (d *db) WriteDB() *sql.DB {
|
|
return d.writeDB
|
|
}
|
|
|
|
func (d *db) Close() {
|
|
if err := d.readDB.Close(); err != nil {
|
|
log.Error("Error closing read DB", err)
|
|
}
|
|
if err := d.writeDB.Close(); err != nil {
|
|
log.Error("Error closing write DB", err)
|
|
}
|
|
}
|
|
|
|
func (d *db) Backup(ctx context.Context) (string, error) {
|
|
destPath := backupPath(time.Now())
|
|
err := d.backupOrRestore(ctx, true, destPath)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return destPath, nil
|
|
}
|
|
|
|
func (d *db) Prune(ctx context.Context) (int, error) {
|
|
return prune(ctx)
|
|
}
|
|
|
|
func (d *db) Restore(ctx context.Context, path string) error {
|
|
return d.backupOrRestore(ctx, false, path)
|
|
}
|
|
|
|
func Db() DB {
|
|
return singleton.GetInstance(func() *db {
|
|
sql.Register(Driver+"_custom", &sqlite3.SQLiteDriver{
|
|
ConnectHook: func(conn *sqlite3.SQLiteConn) error {
|
|
return conn.RegisterFunc("SEEDEDRAND", hasher.HashFunc(), false)
|
|
},
|
|
})
|
|
|
|
Path = conf.Server.DbPath
|
|
if Path == ":memory:" {
|
|
Path = "file::memory:?cache=shared&_foreign_keys=on"
|
|
conf.Server.DbPath = Path
|
|
}
|
|
log.Debug("Opening DataBase", "dbPath", Path, "driver", Driver)
|
|
|
|
// Create a read database connection
|
|
rdb, err := sql.Open(Driver+"_custom", Path)
|
|
if err != nil {
|
|
log.Fatal("Error opening read database", err)
|
|
}
|
|
rdb.SetMaxOpenConns(max(4, runtime.NumCPU()))
|
|
|
|
// Create a write database connection
|
|
wdb, err := sql.Open(Driver+"_custom", Path)
|
|
if err != nil {
|
|
log.Fatal("Error opening write database", err)
|
|
}
|
|
wdb.SetMaxOpenConns(1)
|
|
|
|
return &db{
|
|
readDB: rdb,
|
|
writeDB: wdb,
|
|
}
|
|
})
|
|
}
|
|
|
|
func Close() {
|
|
log.Info("Closing Database")
|
|
Db().Close()
|
|
}
|
|
|
|
func Init() func() {
|
|
db := Db().WriteDB()
|
|
|
|
// Disable foreign_keys to allow re-creating tables in migrations
|
|
_, err := db.Exec("PRAGMA foreign_keys=off")
|
|
defer func() {
|
|
_, err := db.Exec("PRAGMA foreign_keys=on")
|
|
if err != nil {
|
|
log.Error("Error re-enabling foreign_keys", err)
|
|
}
|
|
}()
|
|
if err != nil {
|
|
log.Error("Error disabling foreign_keys", err)
|
|
}
|
|
|
|
gooseLogger := &logAdapter{silent: isSchemaEmpty(db)}
|
|
goose.SetBaseFS(embedMigrations)
|
|
|
|
err = goose.SetDialect(Driver)
|
|
if err != nil {
|
|
log.Fatal("Invalid DB driver", "driver", Driver, err)
|
|
}
|
|
if !isSchemaEmpty(db) && hasPendingMigrations(db, migrationsFolder) {
|
|
log.Info("Upgrading DB Schema to latest version")
|
|
}
|
|
goose.SetLogger(gooseLogger)
|
|
err = goose.Up(db, migrationsFolder)
|
|
if err != nil {
|
|
log.Fatal("Failed to apply new migrations", err)
|
|
}
|
|
|
|
return Close
|
|
}
|
|
|
|
type statusLogger struct{ numPending int }
|
|
|
|
func (*statusLogger) Fatalf(format string, v ...interface{}) { log.Fatal(fmt.Sprintf(format, v...)) }
|
|
func (l *statusLogger) Printf(format string, v ...interface{}) {
|
|
if len(v) < 1 {
|
|
return
|
|
}
|
|
if v0, ok := v[0].(string); !ok {
|
|
return
|
|
} else if v0 == "Pending" {
|
|
l.numPending++
|
|
}
|
|
}
|
|
|
|
func hasPendingMigrations(db *sql.DB, folder string) bool {
|
|
l := &statusLogger{}
|
|
goose.SetLogger(l)
|
|
err := goose.Status(db, folder)
|
|
if err != nil {
|
|
log.Fatal("Failed to check for pending migrations", err)
|
|
}
|
|
return l.numPending > 0
|
|
}
|
|
|
|
func isSchemaEmpty(db *sql.DB) bool {
|
|
rows, err := db.Query("SELECT name FROM sqlite_master WHERE type='table' AND name='goose_db_version';") // nolint:rowserrcheck
|
|
if err != nil {
|
|
log.Fatal("Database could not be opened!", err)
|
|
}
|
|
defer rows.Close()
|
|
return !rows.Next()
|
|
}
|
|
|
|
type logAdapter struct {
|
|
silent bool
|
|
}
|
|
|
|
func (l *logAdapter) Fatal(v ...interface{}) {
|
|
log.Fatal(fmt.Sprint(v...))
|
|
}
|
|
|
|
func (l *logAdapter) Fatalf(format string, v ...interface{}) {
|
|
log.Fatal(fmt.Sprintf(format, v...))
|
|
}
|
|
|
|
func (l *logAdapter) Print(v ...interface{}) {
|
|
if !l.silent {
|
|
log.Info(fmt.Sprint(v...))
|
|
}
|
|
}
|
|
|
|
func (l *logAdapter) Println(v ...interface{}) {
|
|
if !l.silent {
|
|
log.Info(fmt.Sprintln(v...))
|
|
}
|
|
}
|
|
|
|
func (l *logAdapter) Printf(format string, v ...interface{}) {
|
|
if !l.silent {
|
|
log.Info(fmt.Sprintf(format, v...))
|
|
}
|
|
}
|