Commit c3fe108c authored by Fujii Masao's avatar Fujii Masao

Rename wal_keep_segments to wal_keep_size.

max_slot_wal_keep_size that was added in v13 and wal_keep_segments are
the GUC parameters to specify how much WAL files to retain for
the standby servers. While max_slot_wal_keep_size accepts the number of
bytes of WAL files, wal_keep_segments accepts the number of WAL files.
This difference of setting units between those similar parameters could
be confusing to users.

To alleviate this situation, this commit renames wal_keep_segments to
wal_keep_size, and make users specify the WAL size in it instead of
the number of WAL files.

There was also the idea to rename max_slot_wal_keep_size to
max_slot_wal_keep_segments, in the discussion. But we have been moving
away from measuring in segments, for example, checkpoint_segments was
replaced by max_wal_size. So we concluded to rename wal_keep_segments
to wal_keep_size.

Back-patch to v13 where max_slot_wal_keep_size was added.

Author: Fujii Masao
Reviewed-by: Álvaro Herrera, Kyotaro Horiguchi, David Steele
Discussion: https://postgr.es/m/574b4ea3-e0f9-b175-ead2-ebea7faea855@oss.nttdata.com
parent 0bead9af
...@@ -11278,7 +11278,7 @@ SELECT * FROM pg_locks pl LEFT JOIN pg_prepared_xacts ppx ...@@ -11278,7 +11278,7 @@ SELECT * FROM pg_locks pl LEFT JOIN pg_prepared_xacts ppx
<para><literal>extended</literal> means <para><literal>extended</literal> means
that <varname>max_wal_size</varname> is exceeded but the files are that <varname>max_wal_size</varname> is exceeded but the files are
still retained, either by the replication slot or still retained, either by the replication slot or
by <varname>wal_keep_segments</varname>. by <varname>wal_keep_size</varname>.
</para> </para>
</listitem> </listitem>
<listitem> <listitem>
......
...@@ -3151,7 +3151,7 @@ include_dir 'conf.d' ...@@ -3151,7 +3151,7 @@ include_dir 'conf.d'
checkpoints. This is a soft limit; WAL size can exceed checkpoints. This is a soft limit; WAL size can exceed
<varname>max_wal_size</varname> under special circumstances, such as <varname>max_wal_size</varname> under special circumstances, such as
heavy load, a failing <varname>archive_command</varname>, or a high heavy load, a failing <varname>archive_command</varname>, or a high
<varname>wal_keep_segments</varname> setting. <varname>wal_keep_size</varname> setting.
If this value is specified without units, it is taken as megabytes. If this value is specified without units, it is taken as megabytes.
The default is 1 GB. The default is 1 GB.
Increasing this parameter can increase the amount of time needed for Increasing this parameter can increase the amount of time needed for
...@@ -3778,21 +3778,21 @@ restore_command = 'copy "C:\\server\\archivedir\\%f" "%p"' # Windows ...@@ -3778,21 +3778,21 @@ restore_command = 'copy "C:\\server\\archivedir\\%f" "%p"' # Windows
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry id="guc-wal-keep-segments" xreflabel="wal_keep_segments"> <varlistentry id="guc-wal-keep-size" xreflabel="wal_keep_size">
<term><varname>wal_keep_segments</varname> (<type>integer</type>) <term><varname>wal_keep_size</varname> (<type>integer</type>)
<indexterm> <indexterm>
<primary><varname>wal_keep_segments</varname> configuration parameter</primary> <primary><varname>wal_keep_size</varname> configuration parameter</primary>
</indexterm> </indexterm>
</term> </term>
<listitem> <listitem>
<para> <para>
Specifies the minimum number of past log file segments kept in the Specifies the minimum size of past log file segments kept in the
<filename>pg_wal</filename> <filename>pg_wal</filename>
directory, in case a standby server needs to fetch them for streaming directory, in case a standby server needs to fetch them for streaming
replication. Each segment is normally 16 megabytes. If a standby replication. If a standby
server connected to the sending server falls behind by more than server connected to the sending server falls behind by more than
<varname>wal_keep_segments</varname> segments, the sending server might remove <varname>wal_keep_size</varname> megabytes, the sending server might
a WAL segment still needed by the standby, in which case the remove a WAL segment still needed by the standby, in which case the
replication connection will be terminated. Downstream connections replication connection will be terminated. Downstream connections
will also eventually fail as a result. (However, the standby will also eventually fail as a result. (However, the standby
server can recover by fetching the segment from archive, if WAL server can recover by fetching the segment from archive, if WAL
...@@ -3800,14 +3800,15 @@ restore_command = 'copy "C:\\server\\archivedir\\%f" "%p"' # Windows ...@@ -3800,14 +3800,15 @@ restore_command = 'copy "C:\\server\\archivedir\\%f" "%p"' # Windows
</para> </para>
<para> <para>
This sets only the minimum number of segments retained in This sets only the minimum size of segments retained in
<filename>pg_wal</filename>; the system might need to retain more segments <filename>pg_wal</filename>; the system might need to retain more segments
for WAL archival or to recover from a checkpoint. If for WAL archival or to recover from a checkpoint. If
<varname>wal_keep_segments</varname> is zero (the default), the system <varname>wal_keep_size</varname> is zero (the default), the system
doesn't keep any extra segments for standby purposes, so the number doesn't keep any extra segments for standby purposes, so the number
of old WAL segments available to standby servers is a function of of old WAL segments available to standby servers is a function of
the location of the previous checkpoint and status of WAL the location of the previous checkpoint and status of WAL
archiving. archiving.
If this value is specified without units, it is taken as megabytes.
This parameter can only be set in the This parameter can only be set in the
<filename>postgresql.conf</filename> file or on the server command line. <filename>postgresql.conf</filename> file or on the server command line.
</para> </para>
......
...@@ -785,7 +785,7 @@ archive_cleanup_command = 'pg_archivecleanup /path/to/archive %r' ...@@ -785,7 +785,7 @@ archive_cleanup_command = 'pg_archivecleanup /path/to/archive %r'
archiving, the server might recycle old WAL segments before the standby archiving, the server might recycle old WAL segments before the standby
has received them. If this occurs, the standby will need to be has received them. If this occurs, the standby will need to be
reinitialized from a new base backup. You can avoid this by setting reinitialized from a new base backup. You can avoid this by setting
<varname>wal_keep_segments</varname> to a value large enough to ensure that <varname>wal_keep_size</varname> to a value large enough to ensure that
WAL segments are not recycled too early, or by configuring a replication WAL segments are not recycled too early, or by configuring a replication
slot for the standby. If you set up a WAL archive that's accessible from slot for the standby. If you set up a WAL archive that's accessible from
the standby, these solutions are not required, since the standby can the standby, these solutions are not required, since the standby can
...@@ -929,7 +929,7 @@ primary_conninfo = 'host=192.168.1.50 port=5432 user=foo password=foopass' ...@@ -929,7 +929,7 @@ primary_conninfo = 'host=192.168.1.50 port=5432 user=foo password=foopass'
</para> </para>
<para> <para>
In lieu of using replication slots, it is possible to prevent the removal In lieu of using replication slots, it is possible to prevent the removal
of old WAL segments using <xref linkend="guc-wal-keep-segments"/>, or by of old WAL segments using <xref linkend="guc-wal-keep-size"/>, or by
storing the segments in an archive using storing the segments in an archive using
<xref linkend="guc-archive-command"/>. <xref linkend="guc-archive-command"/>.
However, these methods often result in retaining more WAL segments than However, these methods often result in retaining more WAL segments than
......
...@@ -305,7 +305,7 @@ PostgreSQL documentation ...@@ -305,7 +305,7 @@ PostgreSQL documentation
<para> <para>
The write-ahead log files are collected at the end of the backup. The write-ahead log files are collected at the end of the backup.
Therefore, it is necessary for the Therefore, it is necessary for the
<xref linkend="guc-wal-keep-segments"/> parameter to be set high <xref linkend="guc-wal-keep-size"/> parameter to be set high
enough that the log is not removed before the end of the backup. enough that the log is not removed before the end of the backup.
If the log has been rotated when it's time to transfer it, the If the log has been rotated when it's time to transfer it, the
backup will fail and be unusable. backup will fail and be unusable.
......
...@@ -578,7 +578,8 @@ ...@@ -578,7 +578,8 @@
<para> <para>
Independently of <varname>max_wal_size</varname>, Independently of <varname>max_wal_size</varname>,
<xref linkend="guc-wal-keep-segments"/> + 1 most recent WAL files are the most recent <xref linkend="guc-wal-keep-size"/> megabytes of
WAL files plus one additional WAL file are
kept at all times. Also, if WAL archiving is used, old segments can not be kept at all times. Also, if WAL archiving is used, old segments can not be
removed or recycled until they are archived. If WAL archiving cannot keep up removed or recycled until they are archived. If WAL archiving cannot keep up
with the pace that WAL is generated, or if <varname>archive_command</varname> with the pace that WAL is generated, or if <varname>archive_command</varname>
......
...@@ -88,7 +88,7 @@ extern uint32 bootstrap_data_checksum_version; ...@@ -88,7 +88,7 @@ extern uint32 bootstrap_data_checksum_version;
/* User-settable parameters */ /* User-settable parameters */
int max_wal_size_mb = 1024; /* 1 GB */ int max_wal_size_mb = 1024; /* 1 GB */
int min_wal_size_mb = 80; /* 80 MB */ int min_wal_size_mb = 80; /* 80 MB */
int wal_keep_segments = 0; int wal_keep_size_mb = 0;
int XLOGbuffers = -1; int XLOGbuffers = -1;
int XLogArchiveTimeout = 0; int XLogArchiveTimeout = 0;
int XLogArchiveMode = ARCHIVE_MODE_OFF; int XLogArchiveMode = ARCHIVE_MODE_OFF;
...@@ -9525,7 +9525,7 @@ GetWALAvailability(XLogRecPtr targetLSN) ...@@ -9525,7 +9525,7 @@ GetWALAvailability(XLogRecPtr targetLSN)
/* /*
* Calculate the oldest segment currently reserved by all slots, * Calculate the oldest segment currently reserved by all slots,
* considering wal_keep_segments and max_slot_wal_keep_size. Initialize * considering wal_keep_size and max_slot_wal_keep_size. Initialize
* oldestSlotSeg to the current segment. * oldestSlotSeg to the current segment.
*/ */
currpos = GetXLogWriteRecPtr(); currpos = GetXLogWriteRecPtr();
...@@ -9576,9 +9576,9 @@ GetWALAvailability(XLogRecPtr targetLSN) ...@@ -9576,9 +9576,9 @@ GetWALAvailability(XLogRecPtr targetLSN)
/* /*
* Retreat *logSegNo to the last segment that we need to retain because of * Retreat *logSegNo to the last segment that we need to retain because of
* either wal_keep_segments or replication slots. * either wal_keep_size or replication slots.
* *
* This is calculated by subtracting wal_keep_segments from the given xlog * This is calculated by subtracting wal_keep_size from the given xlog
* location, recptr and by making sure that that result is below the * location, recptr and by making sure that that result is below the
* requirement of replication slots. For the latter criterion we do consider * requirement of replication slots. For the latter criterion we do consider
* the effects of max_slot_wal_keep_size: reserve at most that much space back * the effects of max_slot_wal_keep_size: reserve at most that much space back
...@@ -9616,14 +9616,20 @@ KeepLogSeg(XLogRecPtr recptr, XLogSegNo *logSegNo) ...@@ -9616,14 +9616,20 @@ KeepLogSeg(XLogRecPtr recptr, XLogSegNo *logSegNo)
} }
} }
/* but, keep at least wal_keep_segments if that's set */ /* but, keep at least wal_keep_size if that's set */
if (wal_keep_segments > 0 && currSegNo - segno < wal_keep_segments) if (wal_keep_size_mb > 0)
{ {
/* avoid underflow, don't go below 1 */ uint64 keep_segs;
if (currSegNo <= wal_keep_segments)
segno = 1; keep_segs = ConvertToXSegs(wal_keep_size_mb, wal_segment_size);
else if (currSegNo - segno < keep_segs)
segno = currSegNo - wal_keep_segments; {
/* avoid underflow, don't go below 1 */
if (currSegNo <= keep_segs)
segno = 1;
else
segno = currSegNo - keep_segs;
}
} }
/* don't delete WAL segments newer than the calculated segment */ /* don't delete WAL segments newer than the calculated segment */
...@@ -11328,7 +11334,7 @@ do_pg_stop_backup(char *labelfile, bool waitforarchive, TimeLineID *stoptli_p) ...@@ -11328,7 +11334,7 @@ do_pg_stop_backup(char *labelfile, bool waitforarchive, TimeLineID *stoptli_p)
* If archiving is enabled, wait for all the required WAL files to be * If archiving is enabled, wait for all the required WAL files to be
* archived before returning. If archiving isn't enabled, the required WAL * archived before returning. If archiving isn't enabled, the required WAL
* needs to be transported via streaming replication (hopefully with * needs to be transported via streaming replication (hopefully with
* wal_keep_segments set high enough), or some more exotic mechanism like * wal_keep_size set high enough), or some more exotic mechanism like
* polling and copying files from pg_wal with script. We have no knowledge * polling and copying files from pg_wal with script. We have no knowledge
* of those mechanisms, so it's up to the user to ensure that he gets all * of those mechanisms, so it's up to the user to ensure that he gets all
* the required WAL. * the required WAL.
......
...@@ -413,19 +413,20 @@ pg_get_replication_slots(PG_FUNCTION_ARGS) ...@@ -413,19 +413,20 @@ pg_get_replication_slots(PG_FUNCTION_ARGS)
else else
{ {
XLogSegNo targetSeg; XLogSegNo targetSeg;
XLogSegNo keepSegs; uint64 slotKeepSegs;
uint64 keepSegs;
XLogSegNo failSeg; XLogSegNo failSeg;
XLogRecPtr failLSN; XLogRecPtr failLSN;
XLByteToSeg(slot_contents.data.restart_lsn, targetSeg, wal_segment_size); XLByteToSeg(slot_contents.data.restart_lsn, targetSeg, wal_segment_size);
/* determine how many segments slots can be kept by slots ... */ /* determine how many segments slots can be kept by slots */
keepSegs = XLogMBVarToSegs(max_slot_wal_keep_size_mb, wal_segment_size); slotKeepSegs = XLogMBVarToSegs(max_slot_wal_keep_size_mb, wal_segment_size);
/* ... and override by wal_keep_segments as needed */ /* ditto for wal_keep_size */
keepSegs = Max(keepSegs, wal_keep_segments); keepSegs = XLogMBVarToSegs(wal_keep_size_mb, wal_segment_size);
/* if currpos reaches failLSN, we lose our segment */ /* if currpos reaches failLSN, we lose our segment */
failSeg = targetSeg + keepSegs + 1; failSeg = targetSeg + Max(slotKeepSegs, keepSegs) + 1;
XLogSegNoOffsetToRecPtr(failSeg, 0, wal_segment_size, failLSN); XLogSegNoOffsetToRecPtr(failSeg, 0, wal_segment_size, failLSN);
values[i++] = Int64GetDatum(failLSN - currlsn); values[i++] = Int64GetDatum(failLSN - currlsn);
......
...@@ -2636,12 +2636,13 @@ static struct config_int ConfigureNamesInt[] = ...@@ -2636,12 +2636,13 @@ static struct config_int ConfigureNamesInt[] =
}, },
{ {
{"wal_keep_segments", PGC_SIGHUP, REPLICATION_SENDING, {"wal_keep_size", PGC_SIGHUP, REPLICATION_SENDING,
gettext_noop("Sets the number of WAL files held for standby servers."), gettext_noop("Sets the size of WAL files held for standby servers."),
NULL NULL,
GUC_UNIT_MB
}, },
&wal_keep_segments, &wal_keep_size_mb,
0, 0, INT_MAX, 0, 0, MAX_KILOBYTES,
NULL, NULL, NULL NULL, NULL, NULL
}, },
......
...@@ -290,7 +290,7 @@ ...@@ -290,7 +290,7 @@
#max_wal_senders = 10 # max number of walsender processes #max_wal_senders = 10 # max number of walsender processes
# (change requires restart) # (change requires restart)
#wal_keep_segments = 0 # in logfile segments; 0 disables #wal_keep_size = 0 # in megabytes; 0 disables
#max_slot_wal_keep_size = -1 # in megabytes; -1 disables #max_slot_wal_keep_size = -1 # in megabytes; -1 disables
#wal_sender_timeout = 60s # in milliseconds; 0 disables #wal_sender_timeout = 60s # in milliseconds; 0 disables
......
...@@ -135,11 +135,11 @@ sub setup_cluster ...@@ -135,11 +135,11 @@ sub setup_cluster
extra => $extra, extra => $extra,
auth_extra => [ '--create-role', 'rewind_user' ]); auth_extra => [ '--create-role', 'rewind_user' ]);
# Set wal_keep_segments to prevent WAL segment recycling after enforced # Set wal_keep_size to prevent WAL segment recycling after enforced
# checkpoints in the tests. # checkpoints in the tests.
$node_primary->append_conf( $node_primary->append_conf(
'postgresql.conf', qq( 'postgresql.conf', qq(
wal_keep_segments = 20 wal_keep_size = 320MB
)); ));
return; return;
} }
......
...@@ -107,7 +107,7 @@ extern bool reachedConsistency; ...@@ -107,7 +107,7 @@ extern bool reachedConsistency;
extern int wal_segment_size; extern int wal_segment_size;
extern int min_wal_size_mb; extern int min_wal_size_mb;
extern int max_wal_size_mb; extern int max_wal_size_mb;
extern int wal_keep_segments; extern int wal_keep_size_mb;
extern int max_slot_wal_keep_size_mb; extern int max_slot_wal_keep_size_mb;
extern int XLOGbuffers; extern int XLOGbuffers;
extern int XLogArchiveTimeout; extern int XLogArchiveTimeout;
...@@ -273,7 +273,7 @@ typedef enum WALAvailability ...@@ -273,7 +273,7 @@ typedef enum WALAvailability
WALAVAIL_INVALID_LSN, /* parameter error */ WALAVAIL_INVALID_LSN, /* parameter error */
WALAVAIL_RESERVED, /* WAL segment is within max_wal_size */ WALAVAIL_RESERVED, /* WAL segment is within max_wal_size */
WALAVAIL_EXTENDED, /* WAL segment is reserved by a slot or WALAVAIL_EXTENDED, /* WAL segment is reserved by a slot or
* wal_keep_segments */ * wal_keep_size */
WALAVAIL_UNRESERVED, /* no longer reserved, but not removed yet */ WALAVAIL_UNRESERVED, /* no longer reserved, but not removed yet */
WALAVAIL_REMOVED /* WAL segment has been removed */ WALAVAIL_REMOVED /* WAL segment has been removed */
} WALAvailability; } WALAvailability;
......
...@@ -116,19 +116,19 @@ $start_lsn = $node_primary->lsn('write'); ...@@ -116,19 +116,19 @@ $start_lsn = $node_primary->lsn('write');
$node_primary->wait_for_catchup($node_standby, 'replay', $start_lsn); $node_primary->wait_for_catchup($node_standby, 'replay', $start_lsn);
$node_standby->stop; $node_standby->stop;
# wal_keep_segments overrides max_slot_wal_keep_size # wal_keep_size overrides max_slot_wal_keep_size
$result = $node_primary->safe_psql('postgres', $result = $node_primary->safe_psql('postgres',
"ALTER SYSTEM SET wal_keep_segments to 8; SELECT pg_reload_conf();"); "ALTER SYSTEM SET wal_keep_size to '8MB'; SELECT pg_reload_conf();");
# Advance WAL again then checkpoint, reducing remain by 6 MB. # Advance WAL again then checkpoint, reducing remain by 6 MB.
advance_wal($node_primary, 6); advance_wal($node_primary, 6);
$result = $node_primary->safe_psql('postgres', $result = $node_primary->safe_psql('postgres',
"SELECT wal_status as remain FROM pg_replication_slots WHERE slot_name = 'rep1'" "SELECT wal_status as remain FROM pg_replication_slots WHERE slot_name = 'rep1'"
); );
is($result, "extended", is($result, "extended",
'check that wal_keep_segments overrides max_slot_wal_keep_size'); 'check that wal_keep_size overrides max_slot_wal_keep_size');
# restore wal_keep_segments # restore wal_keep_size
$result = $node_primary->safe_psql('postgres', $result = $node_primary->safe_psql('postgres',
"ALTER SYSTEM SET wal_keep_segments to 0; SELECT pg_reload_conf();"); "ALTER SYSTEM SET wal_keep_size to 0; SELECT pg_reload_conf();");
# The standby can reconnect to primary # The standby can reconnect to primary
$node_standby->start; $node_standby->start;
......
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