Commit c303e9e7 authored by Andres Freund's avatar Andres Freund

Fix (re-)starting from a basebackup taken off a standby after a failure.

When starting up from a basebackup taken off a standby extra logic has
to be applied to compute the point where the data directory is
consistent. Normal base backups use a WAL record for that purpose, but
that isn't possible on a standby.

That logic had a error check ensuring that the cluster's control file
indicates being in recovery. Unfortunately that check was too strict,
disregarding the fact that the control file could also indicate that
the cluster was shut down while in recovery.

That's possible when the a cluster starting from a basebackup is shut
down before the backup label has been removed. When everything goes
well that's a short window, but when either restore_command or
primary_conninfo isn't configured correctly the window can get much
wider. That's because inbetween reading and unlinking the label we
restore the last checkpoint from WAL which can need additional WAL.

To fix simply also allow starting when the control file indicates
"shutdown in recovery". There's nicer fixes imaginable, but they'd be
more invasive.

Backpatch to 9.2 where support for taking basebackups from standbys
was added.
parent 40c598fa
...@@ -6070,11 +6070,17 @@ StartupXLOG(void) ...@@ -6070,11 +6070,17 @@ StartupXLOG(void)
/* /*
* Set backupStartPoint if we're starting recovery from a base backup. * Set backupStartPoint if we're starting recovery from a base backup.
* *
* Set backupEndPoint and use minRecoveryPoint as the backup end * Also set backupEndPoint and use minRecoveryPoint as the backup end
* location if we're starting recovery from a base backup which was * location if we're starting recovery from a base backup which was
* taken from the standby. In this case, the database system status in * taken from a standby. In this case, the database system status in
* pg_control must indicate DB_IN_ARCHIVE_RECOVERY. If not, which * pg_control must indicate that the database was already in
* means that backup is corrupted, so we cancel recovery. * recovery. Usually that will be DB_IN_ARCHIVE_RECOVERY but also can
* be DB_SHUTDOWNED_IN_RECOVERY if recovery previously was interrupted
* before reaching this point; e.g. because restore_command or
* primary_conninfo were faulty.
*
* Any other state indicates that the backup somehow became corrupted
* and we can't sensibly continue with recovery.
*/ */
if (haveBackupLabel) if (haveBackupLabel)
{ {
...@@ -6083,7 +6089,8 @@ StartupXLOG(void) ...@@ -6083,7 +6089,8 @@ StartupXLOG(void)
if (backupFromStandby) if (backupFromStandby)
{ {
if (dbstate_at_startup != DB_IN_ARCHIVE_RECOVERY) if (dbstate_at_startup != DB_IN_ARCHIVE_RECOVERY &&
dbstate_at_startup != DB_SHUTDOWNED_IN_RECOVERY)
ereport(FATAL, ereport(FATAL,
(errmsg("backup_label contains data inconsistent with control file"), (errmsg("backup_label contains data inconsistent with control file"),
errhint("This means that the backup is corrupted and you will " errhint("This means that the backup is corrupted and you will "
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment