feat: configuration backup management with manual and auto backup
Add backup/restore functionality for the SQLite database. Users can trigger manual backups, configure automatic backups on an interval with retention policies, list/download/delete backups, and restore from any backup. - Backup engine using VACUUM INTO (safe with WAL mode) - Backup metadata tracked in DB, files stored in DATA_DIR/backups/ - Settings: backup_enabled, backup_interval_hours, backup_retention_count - API: POST/GET/DELETE /api/backups, download, restore endpoints - Autobackup via cron scheduler with configurable interval - Retention: prune on startup, after each backup (manual and auto) - Orphan cleanup: removes backup files without metadata on startup - Restore: replaces DB and triggers graceful server shutdown - Settings UI: /settings/backup with toggle, interval, retention config - Backup list with download, delete, restore actions - i18n: English and Russian translations
This commit is contained in:
@@ -3,6 +3,7 @@ package main
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"io/fs"
|
||||
"log/slog"
|
||||
"net/http"
|
||||
@@ -19,6 +20,7 @@ import (
|
||||
"github.com/alexei/docker-watcher/internal/auth"
|
||||
"github.com/alexei/docker-watcher/internal/config"
|
||||
"github.com/alexei/docker-watcher/internal/crypto"
|
||||
"github.com/alexei/docker-watcher/internal/backup"
|
||||
"github.com/alexei/docker-watcher/internal/deployer"
|
||||
"github.com/alexei/docker-watcher/internal/dns"
|
||||
"github.com/alexei/docker-watcher/internal/docker"
|
||||
@@ -201,10 +203,56 @@ func main() {
|
||||
slog.Info("DNS provider initialized", "provider", settings.DNSProvider)
|
||||
}
|
||||
|
||||
// Initialize backup engine.
|
||||
backupEngine, err := backup.New(db, dbPath, dataDir)
|
||||
if err != nil {
|
||||
slog.Error("create backup engine", "error", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
|
||||
// Clean orphaned backup files and prune on startup.
|
||||
if cleaned, err := backupEngine.CleanOrphans(); err != nil {
|
||||
slog.Warn("backup: clean orphans on startup", "error", err)
|
||||
} else if cleaned > 0 {
|
||||
slog.Info("backup: cleaned orphaned files on startup", "count", cleaned)
|
||||
}
|
||||
if settings.BackupRetentionCount > 0 {
|
||||
if pruned, err := backupEngine.Prune(settings.BackupRetentionCount); err != nil {
|
||||
slog.Warn("backup: prune on startup", "error", err)
|
||||
} else if pruned > 0 {
|
||||
slog.Info("backup: pruned old backups on startup", "count", pruned)
|
||||
}
|
||||
}
|
||||
|
||||
// Schedule autobackup if enabled.
|
||||
if settings.BackupEnabled && settings.BackupIntervalHours > 0 {
|
||||
interval := fmt.Sprintf("@every %dh", settings.BackupIntervalHours)
|
||||
if _, err := cronScheduler.AddFunc(interval, func() {
|
||||
b, err := backupEngine.CreateBackup("auto")
|
||||
if err != nil {
|
||||
slog.Error("autobackup failed", "error", err)
|
||||
return
|
||||
}
|
||||
slog.Info("autobackup completed", "id", b.ID, "filename", b.Filename)
|
||||
|
||||
// Prune after auto backup.
|
||||
currentSettings, err := db.GetSettings()
|
||||
if err == nil && currentSettings.BackupRetentionCount > 0 {
|
||||
backupEngine.Prune(currentSettings.BackupRetentionCount)
|
||||
}
|
||||
}); err != nil {
|
||||
slog.Warn("failed to schedule autobackup", "error", err)
|
||||
} else {
|
||||
slog.Info("autobackup scheduled", "interval_hours", settings.BackupIntervalHours)
|
||||
}
|
||||
}
|
||||
|
||||
// Build API server.
|
||||
apiServer := api.NewServer(db, dockerClient, npmClient, dep, webhookHandler, eventBus, encKey)
|
||||
apiServer.SetStaleScanner(staleScanner)
|
||||
apiServer.SetProxyManager(proxyManager)
|
||||
apiServer.SetBackupEngine(backupEngine)
|
||||
apiServer.SetDBPath(dbPath)
|
||||
apiServer.SetDNSProvider(dnsProvider)
|
||||
apiServer.SetDNSProviderChangedCallback(func(provider dns.Provider) {
|
||||
dep.SetDNSProvider(provider)
|
||||
@@ -239,6 +287,11 @@ func main() {
|
||||
done := make(chan os.Signal, 1)
|
||||
signal.Notify(done, os.Interrupt, syscall.SIGTERM)
|
||||
|
||||
// Allow restore to trigger shutdown.
|
||||
apiServer.SetShutdownFunc(func() {
|
||||
done <- syscall.SIGTERM
|
||||
})
|
||||
|
||||
go func() {
|
||||
slog.Info("Docker Watcher started", "addr", addr)
|
||||
if err := httpServer.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
||||
|
||||
Reference in New Issue
Block a user