2021-09-27 00:52:50 +00:00
|
|
|
package bolt
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2021-11-24 02:58:43 +00:00
|
|
|
"runtime/debug"
|
2021-09-27 00:52:50 +00:00
|
|
|
|
|
|
|
"github.com/portainer/portainer/api/cli"
|
|
|
|
|
|
|
|
werrors "github.com/pkg/errors"
|
|
|
|
portainer "github.com/portainer/portainer/api"
|
|
|
|
"github.com/portainer/portainer/api/bolt/errors"
|
|
|
|
plog "github.com/portainer/portainer/api/bolt/log"
|
|
|
|
"github.com/portainer/portainer/api/bolt/migrator"
|
|
|
|
"github.com/portainer/portainer/api/internal/authorization"
|
|
|
|
)
|
|
|
|
|
|
|
|
const beforePortainerVersionUpgradeBackup = "portainer.db.bak"
|
|
|
|
|
|
|
|
var migrateLog = plog.NewScopedLog("bolt, migrate")
|
|
|
|
|
|
|
|
// FailSafeMigrate backup and restore DB if migration fail
|
2021-10-30 09:32:57 +00:00
|
|
|
func (store *Store) FailSafeMigrate(migrator *migrator.Migrator) (err error) {
|
2021-09-27 00:52:50 +00:00
|
|
|
defer func() {
|
2021-10-30 09:32:57 +00:00
|
|
|
if e := recover(); e != nil {
|
2021-09-27 00:52:50 +00:00
|
|
|
store.Rollback(true)
|
2021-11-24 02:58:43 +00:00
|
|
|
// return error with cause and stacktrace (recover() doesn't include a stacktrace)
|
|
|
|
err = fmt.Errorf("%v %s", e, string(debug.Stack()))
|
2021-09-27 00:52:50 +00:00
|
|
|
}
|
|
|
|
}()
|
2021-10-30 09:32:57 +00:00
|
|
|
|
|
|
|
// !Important: we must use a named return value in the function definition and not a local
|
|
|
|
// !variable referenced from the closure or else the return value will be incorrectly set
|
2021-09-27 00:52:50 +00:00
|
|
|
return migrator.Migrate()
|
|
|
|
}
|
|
|
|
|
|
|
|
// MigrateData automatically migrate the data based on the DBVersion.
|
|
|
|
// This process is only triggered on an existing database, not if the database was just created.
|
|
|
|
// if force is true, then migrate regardless.
|
|
|
|
func (store *Store) MigrateData(force bool) error {
|
|
|
|
if store.isNew && !force {
|
|
|
|
return store.VersionService.StoreDBVersion(portainer.DBVersion)
|
|
|
|
}
|
|
|
|
|
|
|
|
migrator, err := store.newMigrator()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// backup db file before upgrading DB to support rollback
|
|
|
|
isUpdating, err := store.VersionService.IsUpdating()
|
|
|
|
if err != nil && err != errors.ErrObjectNotFound {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !isUpdating && migrator.Version() != portainer.DBVersion {
|
|
|
|
err = store.backupVersion(migrator)
|
|
|
|
if err != nil {
|
|
|
|
return werrors.Wrapf(err, "failed to backup database")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if migrator.Version() < portainer.DBVersion {
|
|
|
|
migrateLog.Info(fmt.Sprintf("Migrating database from version %v to %v.\n", migrator.Version(), portainer.DBVersion))
|
|
|
|
err = store.FailSafeMigrate(migrator)
|
|
|
|
if err != nil {
|
|
|
|
migrateLog.Error("An error occurred during database migration", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (store *Store) newMigrator() (*migrator.Migrator, error) {
|
|
|
|
version, err := store.version()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
migratorParams := &migrator.Parameters{
|
|
|
|
DB: store.connection.DB,
|
|
|
|
DatabaseVersion: version,
|
|
|
|
EndpointGroupService: store.EndpointGroupService,
|
|
|
|
EndpointService: store.EndpointService,
|
|
|
|
EndpointRelationService: store.EndpointRelationService,
|
|
|
|
ExtensionService: store.ExtensionService,
|
|
|
|
RegistryService: store.RegistryService,
|
|
|
|
ResourceControlService: store.ResourceControlService,
|
|
|
|
RoleService: store.RoleService,
|
|
|
|
ScheduleService: store.ScheduleService,
|
|
|
|
SettingsService: store.SettingsService,
|
|
|
|
StackService: store.StackService,
|
|
|
|
TagService: store.TagService,
|
|
|
|
TeamMembershipService: store.TeamMembershipService,
|
|
|
|
UserService: store.UserService,
|
|
|
|
VersionService: store.VersionService,
|
|
|
|
FileService: store.fileService,
|
|
|
|
DockerhubService: store.DockerHubService,
|
|
|
|
AuthorizationService: authorization.NewService(store),
|
|
|
|
}
|
|
|
|
return migrator.NewMigrator(migratorParams), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// getBackupRestoreOptions returns options to store db at common backup dir location; used by:
|
|
|
|
// - db backup prior to version upgrade
|
|
|
|
// - db rollback
|
|
|
|
func getBackupRestoreOptions(store *Store) *BackupOptions {
|
|
|
|
return &BackupOptions{
|
|
|
|
BackupDir: store.commonBackupDir(),
|
|
|
|
BackupFileName: beforePortainerVersionUpgradeBackup,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// backupVersion will backup the database or panic if any errors occur
|
|
|
|
func (store *Store) backupVersion(migrator *migrator.Migrator) error {
|
|
|
|
migrateLog.Info("Backing up database prior to version upgrade...")
|
|
|
|
|
|
|
|
options := getBackupRestoreOptions(store)
|
|
|
|
|
|
|
|
_, err := store.BackupWithOptions(options)
|
|
|
|
if err != nil {
|
|
|
|
migrateLog.Error("An error occurred during database backup", err)
|
|
|
|
removalErr := store.RemoveWithOptions(options)
|
|
|
|
if removalErr != nil {
|
|
|
|
migrateLog.Error("An error occurred during store removal prior to backup", err)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rollback to a pre-upgrade backup copy/snapshot of portainer.db
|
|
|
|
func (store *Store) Rollback(force bool) error {
|
|
|
|
|
|
|
|
if !force {
|
|
|
|
confirmed, err := cli.Confirm("Are you sure you want to rollback your database to the previous backup?")
|
|
|
|
if err != nil || !confirmed {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
options := getBackupRestoreOptions(store)
|
|
|
|
|
|
|
|
err := store.RestoreWithOptions(options)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return store.Close()
|
|
|
|
}
|