Pause recovery for insufficient parameter settings
When certain parameters are changed on a physical replication primary, this is communicated to standbys using the XLOG_PARAMETER_CHANGE WAL record. The standby then checks whether its own settings are at least as big as the ones on the primary. If not, the standby shuts down with a fatal error. This patch changes this behavior for hot standbys to pause recovery at that point instead. That allows read traffic on the standby to continue while database administrators figure out next steps. When recovery is unpaused, the server shuts down (as before). The idea is to fix the parameters while recovery is paused and then restart when there is a maintenance window. Reviewed-by: Sergei Kornilov <sk@zsrv.org> Discussion: https://www.postgresql.org/message-id/flat/4ad69a4c-cc9b-0dfe-0352-8b1b0cd36c7b@2ndquadrant.com
This commit is contained in:
parent
708d165ddb
commit
15251c0a60
@ -2129,18 +2129,14 @@ LOG: database system is ready to accept read only connections
|
|||||||
</para>
|
</para>
|
||||||
|
|
||||||
<para>
|
<para>
|
||||||
The setting of some parameters on the standby will need reconfiguration
|
The settings of some parameters determine the size of shared memory for
|
||||||
if they have been changed on the primary. For these parameters,
|
tracking transaction IDs, locks, and prepared transactions. These shared
|
||||||
the value on the standby must
|
memory structures must be no smaller on a standby than on the primary in
|
||||||
be equal to or greater than the value on the primary.
|
order to ensure that the standby does not run out of shared memory during
|
||||||
Therefore, if you want to increase these values, you should do so on all
|
recovery. For example, if the primary had used a prepared transaction but
|
||||||
standby servers first, before applying the changes to the primary server.
|
the standby had not allocated any shared memory for tracking prepared
|
||||||
Conversely, if you want to decrease these values, you should do so on the
|
transactions, then recovery could not continue until the standby's
|
||||||
primary server first, before applying the changes to all standby servers.
|
configuration is changed. The parameters affected are:
|
||||||
If these parameters
|
|
||||||
are not set high enough then the standby will refuse to start.
|
|
||||||
Higher values can then be supplied and the server
|
|
||||||
restarted to begin recovery again. These parameters are:
|
|
||||||
|
|
||||||
<itemizedlist>
|
<itemizedlist>
|
||||||
<listitem>
|
<listitem>
|
||||||
@ -2169,6 +2165,37 @@ LOG: database system is ready to accept read only connections
|
|||||||
</para>
|
</para>
|
||||||
</listitem>
|
</listitem>
|
||||||
</itemizedlist>
|
</itemizedlist>
|
||||||
|
|
||||||
|
The easiest way to ensure this does not become a problem is to have these
|
||||||
|
parameters set on the standbys to values equal to or greater than on the
|
||||||
|
primary. Therefore, if you want to increase these values, you should do
|
||||||
|
so on all standby servers first, before applying the changes to the
|
||||||
|
primary server. Conversely, if you want to decrease these values, you
|
||||||
|
should do so on the primary server first, before applying the changes to
|
||||||
|
all standby servers. Keep in mind that when a standby is promoted, it
|
||||||
|
becomes the new reference for the required parameter settings for the
|
||||||
|
standbys that follow it. Therefore, to avoid this becoming a problem
|
||||||
|
during a switchover or failover, it is recommended to keep these settings
|
||||||
|
the same on all standby servers.
|
||||||
|
</para>
|
||||||
|
|
||||||
|
<para>
|
||||||
|
The WAL tracks changes to these parameters on the
|
||||||
|
primary. If a hot standby processes WAL that indicates that the current
|
||||||
|
value on the primary is higher than its own value, it will log a warning
|
||||||
|
and pause recovery, for example:
|
||||||
|
<screen>
|
||||||
|
WARNING: hot standby is not possible because of insufficient parameter settings
|
||||||
|
DETAIL: max_connections = 80 is a lower setting than on the primary server, where its value was 100.
|
||||||
|
LOG: recovery has paused
|
||||||
|
DETAIL: If recovery is unpaused, the server will shut down.
|
||||||
|
HINT: You can then restart the server after making the necessary configuration changes.
|
||||||
|
</screen>
|
||||||
|
At that point, the settings on the standby need to be updated and the
|
||||||
|
instance restarted before recovery can continue. If the standby is not a
|
||||||
|
hot standby, then when it encounters the incompatible parameter change, it
|
||||||
|
will shut down immediately without pausing, since there is then no value
|
||||||
|
in keeping it up.
|
||||||
</para>
|
</para>
|
||||||
|
|
||||||
<para>
|
<para>
|
||||||
|
@ -6261,12 +6261,61 @@ static void
|
|||||||
RecoveryRequiresIntParameter(const char *param_name, int currValue, int minValue)
|
RecoveryRequiresIntParameter(const char *param_name, int currValue, int minValue)
|
||||||
{
|
{
|
||||||
if (currValue < minValue)
|
if (currValue < minValue)
|
||||||
ereport(ERROR,
|
{
|
||||||
|
if (LocalHotStandbyActive)
|
||||||
|
{
|
||||||
|
bool warned_for_promote = false;
|
||||||
|
|
||||||
|
ereport(WARNING,
|
||||||
|
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
|
||||||
|
errmsg("hot standby is not possible because of insufficient parameter settings"),
|
||||||
|
errdetail("%s = %d is a lower setting than on the primary server, where its value was %d.",
|
||||||
|
param_name,
|
||||||
|
currValue,
|
||||||
|
minValue)));
|
||||||
|
|
||||||
|
SetRecoveryPause(true);
|
||||||
|
|
||||||
|
ereport(LOG,
|
||||||
|
(errmsg("recovery has paused"),
|
||||||
|
errdetail("If recovery is unpaused, the server will shut down."),
|
||||||
|
errhint("You can then restart the server after making the necessary configuration changes.")));
|
||||||
|
|
||||||
|
while (RecoveryIsPaused())
|
||||||
|
{
|
||||||
|
HandleStartupProcInterrupts();
|
||||||
|
|
||||||
|
if (CheckForStandbyTrigger())
|
||||||
|
{
|
||||||
|
if (!warned_for_promote)
|
||||||
|
ereport(WARNING,
|
||||||
|
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
|
||||||
|
errmsg("promotion is not possible because of insufficient parameter settings"),
|
||||||
|
/* Repeat the detail from above so it's easy to find in the log. */
|
||||||
|
errdetail("%s = %d is a lower setting than on the primary server, where its value was %d.",
|
||||||
|
param_name,
|
||||||
|
currValue,
|
||||||
|
minValue),
|
||||||
|
errhint("Restart the server after making the necessary configuration changes.")));
|
||||||
|
warned_for_promote = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
pgstat_report_wait_start(WAIT_EVENT_RECOVERY_PAUSE);
|
||||||
|
pg_usleep(1000000L); /* 1000 ms */
|
||||||
|
pgstat_report_wait_end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ereport(FATAL,
|
||||||
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
|
(errcode(ERRCODE_INVALID_PARAMETER_VALUE),
|
||||||
errmsg("hot standby is not possible because %s = %d is a lower setting than on the primary server (its value was %d)",
|
errmsg("recovery aborted because of insufficient parameter settings"),
|
||||||
param_name,
|
/* Repeat the detail from above so it's easy to find in the log. */
|
||||||
currValue,
|
errdetail("%s = %d is a lower setting than on the primary server, where its value was %d.",
|
||||||
minValue)));
|
param_name,
|
||||||
|
currValue,
|
||||||
|
minValue),
|
||||||
|
errhint("You can restart the server after making the necessary configuration changes.")));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
Loading…
x
Reference in New Issue
Block a user