mirror of
https://github.com/postgres/postgres.git
synced 2025-07-26 01:22:12 +03:00
Allow to enable failover property for replication slots via SQL API.
This commit adds the failover property to the replication slot. The failover property indicates whether the slot will be synced to the standby servers, enabling the resumption of corresponding logical replication after failover. But note that this commit does not yet include the capability to sync the replication slot; the subsequent commits will add that capability. A new optional parameter 'failover' is added to the pg_create_logical_replication_slot() function. We will also enable to set 'failover' option for slots via the subscription commands in the subsequent commits. The value of the 'failover' flag is displayed as part of pg_replication_slots view. Author: Hou Zhijie, Shveta Malik, Ajin Cherian Reviewed-by: Peter Smith, Bertrand Drouvot, Dilip Kumar, Masahiko Sawada, Nisha Moond, Kuroda, Hayato, Amit Kapila Discussion: https://postgr.es/m/514f6f2f-6833-4539-39f1-96cd1e011f23@enterprisedb.com
This commit is contained in:
@ -90,7 +90,7 @@ typedef struct ReplicationSlotOnDisk
|
||||
sizeof(ReplicationSlotOnDisk) - ReplicationSlotOnDiskConstantSize
|
||||
|
||||
#define SLOT_MAGIC 0x1051CA1 /* format identifier */
|
||||
#define SLOT_VERSION 3 /* version for new files */
|
||||
#define SLOT_VERSION 4 /* version for new files */
|
||||
|
||||
/* Control array for replication slot management */
|
||||
ReplicationSlotCtlData *ReplicationSlotCtl = NULL;
|
||||
@ -248,10 +248,13 @@ ReplicationSlotValidateName(const char *name, int elevel)
|
||||
* during getting changes, if the two_phase option is enabled it can skip
|
||||
* prepare because by that time start decoding point has been moved. So the
|
||||
* user will only get commit prepared.
|
||||
* failover: If enabled, allows the slot to be synced to standbys so
|
||||
* that logical replication can be resumed after failover.
|
||||
*/
|
||||
void
|
||||
ReplicationSlotCreate(const char *name, bool db_specific,
|
||||
ReplicationSlotPersistency persistency, bool two_phase)
|
||||
ReplicationSlotPersistency persistency,
|
||||
bool two_phase, bool failover)
|
||||
{
|
||||
ReplicationSlot *slot = NULL;
|
||||
int i;
|
||||
@ -311,6 +314,7 @@ ReplicationSlotCreate(const char *name, bool db_specific,
|
||||
slot->data.persistency = persistency;
|
||||
slot->data.two_phase = two_phase;
|
||||
slot->data.two_phase_at = InvalidXLogRecPtr;
|
||||
slot->data.failover = failover;
|
||||
|
||||
/* and then data only present in shared memory */
|
||||
slot->just_dirtied = false;
|
||||
|
@ -42,7 +42,8 @@ create_physical_replication_slot(char *name, bool immediately_reserve,
|
||||
|
||||
/* acquire replication slot, this will check for conflicting names */
|
||||
ReplicationSlotCreate(name, false,
|
||||
temporary ? RS_TEMPORARY : RS_PERSISTENT, false);
|
||||
temporary ? RS_TEMPORARY : RS_PERSISTENT, false,
|
||||
false);
|
||||
|
||||
if (immediately_reserve)
|
||||
{
|
||||
@ -117,6 +118,7 @@ pg_create_physical_replication_slot(PG_FUNCTION_ARGS)
|
||||
static void
|
||||
create_logical_replication_slot(char *name, char *plugin,
|
||||
bool temporary, bool two_phase,
|
||||
bool failover,
|
||||
XLogRecPtr restart_lsn,
|
||||
bool find_startpoint)
|
||||
{
|
||||
@ -133,7 +135,8 @@ create_logical_replication_slot(char *name, char *plugin,
|
||||
* error as well.
|
||||
*/
|
||||
ReplicationSlotCreate(name, true,
|
||||
temporary ? RS_TEMPORARY : RS_EPHEMERAL, two_phase);
|
||||
temporary ? RS_TEMPORARY : RS_EPHEMERAL, two_phase,
|
||||
failover);
|
||||
|
||||
/*
|
||||
* Create logical decoding context to find start point or, if we don't
|
||||
@ -171,6 +174,7 @@ pg_create_logical_replication_slot(PG_FUNCTION_ARGS)
|
||||
Name plugin = PG_GETARG_NAME(1);
|
||||
bool temporary = PG_GETARG_BOOL(2);
|
||||
bool two_phase = PG_GETARG_BOOL(3);
|
||||
bool failover = PG_GETARG_BOOL(4);
|
||||
Datum result;
|
||||
TupleDesc tupdesc;
|
||||
HeapTuple tuple;
|
||||
@ -188,6 +192,7 @@ pg_create_logical_replication_slot(PG_FUNCTION_ARGS)
|
||||
NameStr(*plugin),
|
||||
temporary,
|
||||
two_phase,
|
||||
failover,
|
||||
InvalidXLogRecPtr,
|
||||
true);
|
||||
|
||||
@ -232,7 +237,7 @@ pg_drop_replication_slot(PG_FUNCTION_ARGS)
|
||||
Datum
|
||||
pg_get_replication_slots(PG_FUNCTION_ARGS)
|
||||
{
|
||||
#define PG_GET_REPLICATION_SLOTS_COLS 15
|
||||
#define PG_GET_REPLICATION_SLOTS_COLS 16
|
||||
ReturnSetInfo *rsinfo = (ReturnSetInfo *) fcinfo->resultinfo;
|
||||
XLogRecPtr currlsn;
|
||||
int slotno;
|
||||
@ -426,6 +431,8 @@ pg_get_replication_slots(PG_FUNCTION_ARGS)
|
||||
}
|
||||
}
|
||||
|
||||
values[i++] = BoolGetDatum(slot_contents.data.failover);
|
||||
|
||||
Assert(i == PG_GET_REPLICATION_SLOTS_COLS);
|
||||
|
||||
tuplestore_putvalues(rsinfo->setResult, rsinfo->setDesc,
|
||||
@ -693,6 +700,7 @@ copy_replication_slot(FunctionCallInfo fcinfo, bool logical_slot)
|
||||
XLogRecPtr src_restart_lsn;
|
||||
bool src_islogical;
|
||||
bool temporary;
|
||||
bool failover;
|
||||
char *plugin;
|
||||
Datum values[2];
|
||||
bool nulls[2];
|
||||
@ -748,6 +756,7 @@ copy_replication_slot(FunctionCallInfo fcinfo, bool logical_slot)
|
||||
src_islogical = SlotIsLogical(&first_slot_contents);
|
||||
src_restart_lsn = first_slot_contents.data.restart_lsn;
|
||||
temporary = (first_slot_contents.data.persistency == RS_TEMPORARY);
|
||||
failover = first_slot_contents.data.failover;
|
||||
plugin = logical_slot ? NameStr(first_slot_contents.data.plugin) : NULL;
|
||||
|
||||
/* Check type of replication slot */
|
||||
@ -787,6 +796,7 @@ copy_replication_slot(FunctionCallInfo fcinfo, bool logical_slot)
|
||||
plugin,
|
||||
temporary,
|
||||
false,
|
||||
failover,
|
||||
src_restart_lsn,
|
||||
false);
|
||||
}
|
||||
|
@ -1212,7 +1212,7 @@ CreateReplicationSlot(CreateReplicationSlotCmd *cmd)
|
||||
{
|
||||
ReplicationSlotCreate(cmd->slotname, false,
|
||||
cmd->temporary ? RS_TEMPORARY : RS_PERSISTENT,
|
||||
false);
|
||||
false, false);
|
||||
|
||||
if (reserve_wal)
|
||||
{
|
||||
@ -1243,7 +1243,7 @@ CreateReplicationSlot(CreateReplicationSlotCmd *cmd)
|
||||
*/
|
||||
ReplicationSlotCreate(cmd->slotname, true,
|
||||
cmd->temporary ? RS_TEMPORARY : RS_EPHEMERAL,
|
||||
two_phase);
|
||||
two_phase, false);
|
||||
|
||||
/*
|
||||
* Do options check early so that we can bail before calling the
|
||||
|
Reference in New Issue
Block a user