package datastore import ( "fmt" "runtime/debug" portainer "github.com/portainer/portainer/api" "github.com/portainer/portainer/api/cli" "github.com/portainer/portainer/api/dataservices/errors" "github.com/portainer/portainer/api/datastore/migrator" "github.com/portainer/portainer/api/internal/authorization" werrors "github.com/pkg/errors" "github.com/rs/zerolog/log" ) const beforePortainerVersionUpgradeBackup = "portainer.db.bak" func (store *Store) MigrateData() error { version, err := store.version() if err != nil { return err } // Backup Database backupPath, err := store.Backup() if err != nil { return werrors.Wrap(err, "while backing up db before migration") } migratorParams := &migrator.MigratorParameters{ DatabaseVersion: version, EndpointGroupService: store.EndpointGroupService, EndpointService: store.EndpointService, EndpointRelationService: store.EndpointRelationService, ExtensionService: store.ExtensionService, FDOProfilesService: store.FDOProfilesService, 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), } // restore on error err = store.connectionMigrateData(migratorParams) if err != nil { log.Error().Err(err).Msg("while DB migration, restoring DB") // Restore options options := BackupOptions{ BackupPath: backupPath, } err := store.restoreWithOptions(&options) if err != nil { log.Fatal(). Str("database_file", store.databasePath()). Str("backup", options.BackupPath).Err(err). Msg("failed restoring the backup, Portainer database file needs to restored manually by replacing the database file with a recent backup") } } return err } // FailSafeMigrate backup and restore DB if migration fail func (store *Store) FailSafeMigrate(migrator *migrator.Migrator) (err error) { defer func() { if e := recover(); e != nil { store.Rollback(true) // return error with cause and stacktrace (recover() doesn't include a stacktrace) err = fmt.Errorf("%v %s", e, string(debug.Stack())) } }() // !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 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) connectionMigrateData(migratorParams *migrator.MigratorParameters) error { migrator := migrator.NewMigrator(migratorParams) // backup db file before upgrading DB to support rollback isUpdating, err := migratorParams.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 { log.Info(). Int("migrator_version", migrator.Version()). Int("db_version", portainer.DBVersion). Msg("migrating database") err = store.FailSafeMigrate(migrator) if err != nil { log.Error().Err(err).Msg("an error occurred during database migration") return err } } return nil } // backupVersion will backup the database or panic if any errors occur func (store *Store) backupVersion(migrator *migrator.Migrator) error { log.Info().Msg("backing up database prior to version upgrade") options := getBackupRestoreOptions(store.commonBackupDir()) _, err := store.backupWithOptions(options) if err != nil { log.Error().Err(err).Msg("an error occurred during database backup") removalErr := store.removeWithOptions(options) if removalErr != nil { log.Error().Err(err).Msg("an error occurred during store removal prior to backup") } return err } return nil } // Rollback to a pre-upgrade backup copy/snapshot of portainer.db func (store *Store) connectionRollback(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.commonBackupDir()) err := store.restoreWithOptions(options) if err != nil { return err } return store.connection.Close() }