Commit 4ad2a548 authored by Fujii Masao's avatar Fujii Masao

Add GUC to enable logging of replication commands.

Previously replication commands like IDENTIFY_COMMAND were not logged
even when log_statements is set to all. Some users who want to audit
all types of statements were not satisfied with this situation. To
address the problem, this commit adds new GUC log_replication_commands.
If it's enabled, all replication commands are logged in the server log.

There are many ways to allow us to enable that logging. For example,
we can extend log_statement so that replication commands are logged
when it's set to all. But per discussion in the community, we reached
the consensus to add separate GUC for that.

Reviewed by Ian Barwick, Robert Haas and Heikki Linnakangas.
parent a2dabf0e
...@@ -4672,6 +4672,22 @@ FROM pg_stat_activity; ...@@ -4672,6 +4672,22 @@ FROM pg_stat_activity;
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry id="guc-log-replication-commands" xreflabel="log_replication_commands">
<term><varname>log_replication_commands</varname> (<type>boolean</type>)
<indexterm>
<primary><varname>log_replication_commands</> configuration parameter</primary>
</indexterm>
</term>
<listitem>
<para>
Causes each replication command to be logged in the server log.
See <xref linkend="protocol-replication"> for more information about
replication command. The default value is <literal>off</>.
Only superusers can change this setting.
</para>
</listitem>
</varlistentry>
<varlistentry id="guc-log-temp-files" xreflabel="log_temp_files"> <varlistentry id="guc-log-temp-files" xreflabel="log_temp_files">
<term><varname>log_temp_files</varname> (<type>integer</type>) <term><varname>log_temp_files</varname> (<type>integer</type>)
<indexterm> <indexterm>
......
...@@ -1306,6 +1306,8 @@ To initiate streaming replication, the frontend sends the ...@@ -1306,6 +1306,8 @@ To initiate streaming replication, the frontend sends the
of <literal>true</> tells the backend to go into walsender mode, wherein a of <literal>true</> tells the backend to go into walsender mode, wherein a
small set of replication commands can be issued instead of SQL statements. Only small set of replication commands can be issued instead of SQL statements. Only
the simple query protocol can be used in walsender mode. the simple query protocol can be used in walsender mode.
Replication commands are logged in the server log when
<xref linkend="guc-log-replication-commands"> is enabled.
Passing <literal>database</> as the value instructs walsender to connect to Passing <literal>database</> as the value instructs walsender to connect to
the database specified in the <literal>dbname</> parameter, which will allow the database specified in the <literal>dbname</> parameter, which will allow
the connection to be used for logical replication from that database. the connection to be used for logical replication from that database.
......
...@@ -108,6 +108,7 @@ bool am_db_walsender = false; /* Connected to a database? */ ...@@ -108,6 +108,7 @@ bool am_db_walsender = false; /* Connected to a database? */
int max_wal_senders = 0; /* the maximum number of concurrent walsenders */ int max_wal_senders = 0; /* the maximum number of concurrent walsenders */
int wal_sender_timeout = 60 * 1000; /* maximum time to send one int wal_sender_timeout = 60 * 1000; /* maximum time to send one
* WAL data message */ * WAL data message */
bool log_replication_commands = false;
/* /*
* State for WalSndWakeupRequest * State for WalSndWakeupRequest
...@@ -1267,14 +1268,20 @@ exec_replication_command(const char *cmd_string) ...@@ -1267,14 +1268,20 @@ exec_replication_command(const char *cmd_string)
MemoryContext cmd_context; MemoryContext cmd_context;
MemoryContext old_context; MemoryContext old_context;
/*
* Log replication command if log_replication_commands is enabled.
* Even when it's disabled, log the command with DEBUG1 level for
* backward compatibility.
*/
ereport(log_replication_commands ? LOG : DEBUG1,
(errmsg("received replication command: %s", cmd_string)));
/* /*
* CREATE_REPLICATION_SLOT ... LOGICAL exports a snapshot until the next * CREATE_REPLICATION_SLOT ... LOGICAL exports a snapshot until the next
* command arrives. Clean up the old stuff if there's anything. * command arrives. Clean up the old stuff if there's anything.
*/ */
SnapBuildClearExportedSnapshot(); SnapBuildClearExportedSnapshot();
elog(DEBUG1, "received replication command: %s", cmd_string);
CHECK_FOR_INTERRUPTS(); CHECK_FOR_INTERRUPTS();
cmd_context = AllocSetContextCreate(CurrentMemoryContext, cmd_context = AllocSetContextCreate(CurrentMemoryContext,
......
...@@ -924,6 +924,15 @@ static struct config_bool ConfigureNamesBool[] = ...@@ -924,6 +924,15 @@ static struct config_bool ConfigureNamesBool[] =
false, false,
NULL, NULL, NULL NULL, NULL, NULL
}, },
{
{"log_replication_commands", PGC_SUSET, LOGGING_WHAT,
gettext_noop("Logs each replication command."),
NULL
},
&log_replication_commands,
false,
NULL, NULL, NULL
},
{ {
{"debug_assertions", PGC_INTERNAL, PRESET_OPTIONS, {"debug_assertions", PGC_INTERNAL, PRESET_OPTIONS,
gettext_noop("Shows whether the running server has assertion checks enabled."), gettext_noop("Shows whether the running server has assertion checks enabled."),
......
...@@ -431,6 +431,7 @@ ...@@ -431,6 +431,7 @@
# e.g. '<%u%%%d> ' # e.g. '<%u%%%d> '
#log_lock_waits = off # log lock waits >= deadlock_timeout #log_lock_waits = off # log lock waits >= deadlock_timeout
#log_statement = 'none' # none, ddl, mod, all #log_statement = 'none' # none, ddl, mod, all
#log_replication_commands = off
#log_temp_files = -1 # log temporary files equal or larger #log_temp_files = -1 # log temporary files equal or larger
# than the specified size in kilobytes; # than the specified size in kilobytes;
# -1 disables, 0 logs all temp files # -1 disables, 0 logs all temp files
......
...@@ -25,6 +25,7 @@ extern bool wake_wal_senders; ...@@ -25,6 +25,7 @@ extern bool wake_wal_senders;
/* user-settable parameters */ /* user-settable parameters */
extern int max_wal_senders; extern int max_wal_senders;
extern int wal_sender_timeout; extern int wal_sender_timeout;
extern bool log_replication_commands;
extern void InitWalSender(void); extern void InitWalSender(void);
extern void exec_replication_command(const char *query_string); extern void exec_replication_command(const char *query_string);
......
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