mirror of
https://github.com/postgres/postgres.git
synced 2025-10-25 13:17:41 +03:00
Add a slot synchronization function.
This commit introduces a new SQL function pg_sync_replication_slots() which is used to synchronize the logical replication slots from the primary server to the physical standby so that logical replication can be resumed after a failover or planned switchover. A new 'synced' flag is introduced in pg_replication_slots view, indicating whether the slot has been synchronized from the primary server. On a standby, synced slots cannot be dropped or consumed, and any attempt to perform logical decoding on them will result in an error. The logical replication slots on the primary can be synchronized to the hot standby by using the 'failover' parameter of pg-create-logical-replication-slot(), or by using the 'failover' option of CREATE SUBSCRIPTION during slot creation, and then calling pg_sync_replication_slots() on standby. For the synchronization to work, it is mandatory to have a physical replication slot between the primary and the standby aka 'primary_slot_name' should be configured on the standby, and 'hot_standby_feedback' must be enabled on the standby. It is also necessary to specify a valid 'dbname' in the 'primary_conninfo'. If a logical slot is invalidated on the primary, then that slot on the standby is also invalidated. If a logical slot on the primary is valid but is invalidated on the standby, then that slot is dropped but will be recreated on the standby in the next pg_sync_replication_slots() call provided the slot still exists on the primary server. It is okay to recreate such slots as long as these are not consumable on standby (which is the case currently). This situation may occur due to the following reasons: - The 'max_slot_wal_keep_size' on the standby is insufficient to retain WAL records from the restart_lsn of the slot. - 'primary_slot_name' is temporarily reset to null and the physical slot is removed. The slot synchronization status on the standby can be monitored using the 'synced' column of pg_replication_slots view. A functionality to automatically synchronize slots by a background worker and allow logical walsenders to wait for the physical will be done in subsequent commits. Author: Hou Zhijie, Shveta Malik, Ajin Cherian based on an earlier version by Peter Eisentraut Reviewed-by: Masahiko Sawada, Bertrand Drouvot, Peter Smith, Dilip Kumar, Nisha Moond, Kuroda Hayato, Amit Kapila Discussion: https://postgr.es/m/514f6f2f-6833-4539-39f1-96cd1e011f23@enterprisedb.com
This commit is contained in:
@@ -358,6 +358,62 @@ postgres=# select * from pg_logical_slot_get_changes('regression_slot', NULL, NU
|
||||
So if a slot is no longer required it should be dropped.
|
||||
</para>
|
||||
</caution>
|
||||
|
||||
</sect2>
|
||||
|
||||
<sect2 id="logicaldecoding-replication-slots-synchronization">
|
||||
<title>Replication Slot Synchronization</title>
|
||||
<para>
|
||||
The logical replication slots on the primary can be synchronized to
|
||||
the hot standby by using the <literal>failover</literal> parameter of
|
||||
<link linkend="pg-create-logical-replication-slot">
|
||||
<function>pg_create_logical_replication_slot</function></link>, or by
|
||||
using the <link linkend="sql-createsubscription-params-with-failover">
|
||||
<literal>failover</literal></link> option of
|
||||
<command>CREATE SUBSCRIPTION</command> during slot creation, and then calling
|
||||
<link linkend="pg-sync-replication-slots">
|
||||
<function>pg_sync_replication_slots</function></link>
|
||||
on the standby. For the synchronization to work, it is mandatory to
|
||||
have a physical replication slot between the primary and the standby aka
|
||||
<link linkend="guc-primary-slot-name"><varname>primary_slot_name</varname></link>
|
||||
should be configured on the standby, and
|
||||
<link linkend="guc-hot-standby-feedback"><varname>hot_standby_feedback</varname></link>
|
||||
must be enabled on the standby. It is also necessary to specify a valid
|
||||
<literal>dbname</literal> in the
|
||||
<link linkend="guc-primary-conninfo"><varname>primary_conninfo</varname></link>.
|
||||
</para>
|
||||
|
||||
<para>
|
||||
The ability to resume logical replication after failover depends upon the
|
||||
<link linkend="view-pg-replication-slots">pg_replication_slots</link>.<structfield>synced</structfield>
|
||||
value for the synchronized slots on the standby at the time of failover.
|
||||
Only persistent slots that have attained synced state as true on the standby
|
||||
before failover can be used for logical replication after failover.
|
||||
Temporary synced slots cannot be used for logical decoding, therefore
|
||||
logical replication for those slots cannot be resumed. For example, if the
|
||||
synchronized slot could not become persistent on the standby due to a
|
||||
disabled subscription, then the subscription cannot be resumed after
|
||||
failover even when it is enabled.
|
||||
</para>
|
||||
|
||||
<para>
|
||||
To resume logical replication after failover from the synced logical
|
||||
slots, the subscription's 'conninfo' must be altered to point to the
|
||||
new primary server. This is done using
|
||||
<link linkend="sql-altersubscription-params-connection"><command>ALTER SUBSCRIPTION ... CONNECTION</command></link>.
|
||||
It is recommended that subscriptions are first disabled before promoting
|
||||
the standby and are re-enabled after altering the connection string.
|
||||
</para>
|
||||
<caution>
|
||||
<para>
|
||||
There is a chance that the old primary is up again during the promotion
|
||||
and if subscriptions are not disabled, the logical subscribers may
|
||||
continue to receive data from the old primary server even after promotion
|
||||
until the connection string is altered. This might result in data
|
||||
inconsistency issues, preventing the logical subscribers from being
|
||||
able to continue replication from the new primary server.
|
||||
</para>
|
||||
</caution>
|
||||
</sect2>
|
||||
|
||||
<sect2 id="logicaldecoding-explanation-output-plugins">
|
||||
|
||||
Reference in New Issue
Block a user