Commit bf8a662c authored by Tom Lane's avatar Tom Lane

Introduce a new GUC_REPORT setting "in_hot_standby".

Aside from being queriable via SHOW, this value is sent to the client
immediately at session startup, and again later on if the server gets
promoted to primary during the session.  The immediate report will be
used in an upcoming patch to avoid an extra round trip when trying to
connect to a primary server.

Haribabu Kommi, Greg Nancarrow, Tom Lane; reviewed at various times
by Laurenz Albe, Takayuki Tsunakawa, Peter Smith.

Discussion: https://postgr.es/m/CAF3+xM+8-ztOkaV9gHiJ3wfgENTq97QcjXQt+rbFQ6F7oNzt9A@mail.gmail.com
parent 47b2ed1d
...@@ -9600,13 +9600,14 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir' ...@@ -9600,13 +9600,14 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir'
<title>Preset Options</title> <title>Preset Options</title>
<para> <para>
The following <quote>parameters</quote> are read-only, and are determined The following <quote>parameters</quote> are read-only.
when <productname>PostgreSQL</productname> is compiled or when it is As such, they have been excluded from the sample
installed. As such, they have been excluded from the sample
<filename>postgresql.conf</filename> file. These options report <filename>postgresql.conf</filename> file. These options report
various aspects of <productname>PostgreSQL</productname> behavior various aspects of <productname>PostgreSQL</productname> behavior
that might be of interest to certain applications, particularly that might be of interest to certain applications, particularly
administrative front-ends. administrative front-ends.
Most of them are determined when <productname>PostgreSQL</productname>
is compiled or when it is installed.
</para> </para>
<variablelist> <variablelist>
...@@ -9651,10 +9652,11 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir' ...@@ -9651,10 +9652,11 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir'
</term> </term>
<listitem> <listitem>
<para> <para>
On Unix systems this parameter reports the permissions of the data On Unix systems this parameter reports the permissions the data
directory defined by (<xref linkend="guc-data-directory"/>) at startup. directory (defined by <xref linkend="guc-data-directory"/>)
had at server startup.
(On Microsoft Windows this parameter will always display (On Microsoft Windows this parameter will always display
<literal>0700</literal>). See <literal>0700</literal>.) See
<xref linkend="app-initdb-allow-group-access"/> for more information. <xref linkend="app-initdb-allow-group-access"/> for more information.
</para> </para>
</listitem> </listitem>
...@@ -9695,6 +9697,23 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir' ...@@ -9695,6 +9697,23 @@ dynamic_library_path = 'C:\tools\postgresql;H:\my_project\lib;$libdir'
</listitem> </listitem>
</varlistentry> </varlistentry>
<varlistentry id="guc-in-hot-standby" xreflabel="in_hot_standby">
<term><varname>in_hot_standby</varname> (<type>boolean</type>)
<indexterm>
<primary><varname>in_hot_standby</varname> configuration parameter</primary>
</indexterm>
</term>
<listitem>
<para>
Reports whether the server is currently in hot standby mode. When
this is <literal>on</literal>, all transactions are forced to be
read-only. Within a session, this can change only if the server is
promoted to be primary. See <xref linkend="hot-standby"/> for more
information.
</para>
</listitem>
</varlistentry>
<varlistentry id="guc-lc-collate" xreflabel="lc_collate"> <varlistentry id="guc-lc-collate" xreflabel="lc_collate">
<term><varname>lc_collate</varname> (<type>string</type>) <term><varname>lc_collate</varname> (<type>string</type>)
<indexterm> <indexterm>
......
...@@ -1859,8 +1859,11 @@ if (!triggered) ...@@ -1859,8 +1859,11 @@ if (!triggered)
</para> </para>
<para> <para>
Users will be able to tell whether their session is read-only by Users can determine whether hot standby is currently active for their
issuing <command>SHOW transaction_read_only</command>. In addition, a set of session by issuing <command>SHOW in_hot_standby</command>.
(In server versions before 14, the <varname>in_hot_standby</varname>
parameter did not exist; a workable substitute method for older servers
is <command>SHOW transaction_read_only</command>.) In addition, a set of
functions (<xref linkend="functions-recovery-info-table"/>) allow users to functions (<xref linkend="functions-recovery-info-table"/>) allow users to
access information about the standby server. These allow you to write access information about the standby server. These allow you to write
programs that are aware of the current state of the database. These programs that are aware of the current state of the database. These
......
...@@ -2150,6 +2150,7 @@ const char *PQparameterStatus(const PGconn *conn, const char *paramName); ...@@ -2150,6 +2150,7 @@ const char *PQparameterStatus(const PGconn *conn, const char *paramName);
<varname>server_encoding</varname>, <varname>server_encoding</varname>,
<varname>client_encoding</varname>, <varname>client_encoding</varname>,
<varname>application_name</varname>, <varname>application_name</varname>,
<varname>in_hot_standby</varname>,
<varname>is_superuser</varname>, <varname>is_superuser</varname>,
<varname>session_authorization</varname>, <varname>session_authorization</varname>,
<varname>DateStyle</varname>, <varname>DateStyle</varname>,
...@@ -2162,7 +2163,10 @@ const char *PQparameterStatus(const PGconn *conn, const char *paramName); ...@@ -2162,7 +2163,10 @@ const char *PQparameterStatus(const PGconn *conn, const char *paramName);
<varname>standard_conforming_strings</varname> was not reported by releases <varname>standard_conforming_strings</varname> was not reported by releases
before 8.1; before 8.1;
<varname>IntervalStyle</varname> was not reported by releases before 8.4; <varname>IntervalStyle</varname> was not reported by releases before 8.4;
<varname>application_name</varname> was not reported by releases before 9.0.) <varname>application_name</varname> was not reported by releases before
9.0;
<varname>in_hot_standby</varname> was not reported by releases before
14.)
Note that Note that
<varname>server_version</varname>, <varname>server_version</varname>,
<varname>server_encoding</varname> and <varname>server_encoding</varname> and
......
...@@ -1278,6 +1278,7 @@ SELCT 1/0;<!-- this typo is intentional --> ...@@ -1278,6 +1278,7 @@ SELCT 1/0;<!-- this typo is intentional -->
<varname>server_encoding</varname>, <varname>server_encoding</varname>,
<varname>client_encoding</varname>, <varname>client_encoding</varname>,
<varname>application_name</varname>, <varname>application_name</varname>,
<varname>in_hot_standby</varname>,
<varname>is_superuser</varname>, <varname>is_superuser</varname>,
<varname>session_authorization</varname>, <varname>session_authorization</varname>,
<varname>DateStyle</varname>, <varname>DateStyle</varname>,
...@@ -1290,7 +1291,10 @@ SELCT 1/0;<!-- this typo is intentional --> ...@@ -1290,7 +1291,10 @@ SELCT 1/0;<!-- this typo is intentional -->
<varname>standard_conforming_strings</varname> was not reported by releases <varname>standard_conforming_strings</varname> was not reported by releases
before 8.1; before 8.1;
<varname>IntervalStyle</varname> was not reported by releases before 8.4; <varname>IntervalStyle</varname> was not reported by releases before 8.4;
<varname>application_name</varname> was not reported by releases before 9.0.) <varname>application_name</varname> was not reported by releases before
9.0;
<varname>in_hot_standby</varname> was not reported by releases before
14.)
Note that Note that
<varname>server_version</varname>, <varname>server_version</varname>,
<varname>server_encoding</varname> and <varname>server_encoding</varname> and
......
...@@ -16,7 +16,7 @@ ...@@ -16,7 +16,7 @@
## if an option is valid but shows up in only one file (guc.c but not ## if an option is valid but shows up in only one file (guc.c but not
## postgresql.conf.sample), it should be listed here so that it ## postgresql.conf.sample), it should be listed here so that it
## can be ignored ## can be ignored
INTENTIONALLY_NOT_INCLUDED="debug_deadlocks \ INTENTIONALLY_NOT_INCLUDED="debug_deadlocks in_hot_standby \
is_superuser lc_collate lc_ctype lc_messages lc_monetary lc_numeric lc_time \ is_superuser lc_collate lc_ctype lc_messages lc_monetary lc_numeric lc_time \
pre_auth_delay role seed server_encoding server_version server_version_num \ pre_auth_delay role seed server_encoding server_version server_version_num \
session_authorization trace_lock_oidmin trace_lock_table trace_locks trace_lwlocks \ session_authorization trace_lock_oidmin trace_lock_table trace_locks trace_lwlocks \
......
...@@ -209,6 +209,7 @@ static bool check_cluster_name(char **newval, void **extra, GucSource source); ...@@ -209,6 +209,7 @@ static bool check_cluster_name(char **newval, void **extra, GucSource source);
static const char *show_unix_socket_permissions(void); static const char *show_unix_socket_permissions(void);
static const char *show_log_file_mode(void); static const char *show_log_file_mode(void);
static const char *show_data_directory_mode(void); static const char *show_data_directory_mode(void);
static const char *show_in_hot_standby(void);
static bool check_backtrace_functions(char **newval, void **extra, GucSource source); static bool check_backtrace_functions(char **newval, void **extra, GucSource source);
static void assign_backtrace_functions(const char *newval, void *extra); static void assign_backtrace_functions(const char *newval, void *extra);
static bool check_recovery_target_timeline(char **newval, void **extra, GucSource source); static bool check_recovery_target_timeline(char **newval, void **extra, GucSource source);
...@@ -610,6 +611,7 @@ static int wal_block_size; ...@@ -610,6 +611,7 @@ static int wal_block_size;
static bool data_checksums; static bool data_checksums;
static bool integer_datetimes; static bool integer_datetimes;
static bool assert_enabled; static bool assert_enabled;
static bool in_hot_standby;
static char *recovery_target_timeline_string; static char *recovery_target_timeline_string;
static char *recovery_target_string; static char *recovery_target_string;
static char *recovery_target_xid_string; static char *recovery_target_xid_string;
...@@ -1844,6 +1846,17 @@ static struct config_bool ConfigureNamesBool[] = ...@@ -1844,6 +1846,17 @@ static struct config_bool ConfigureNamesBool[] =
NULL, NULL, NULL NULL, NULL, NULL
}, },
{
{"in_hot_standby", PGC_INTERNAL, PRESET_OPTIONS,
gettext_noop("Shows whether hot standby is currently active."),
NULL,
GUC_REPORT | GUC_NOT_IN_SAMPLE | GUC_DISALLOW_IN_FILE
},
&in_hot_standby,
false,
NULL, NULL, show_in_hot_standby
},
{ {
{"allow_system_table_mods", PGC_SUSET, DEVELOPER_OPTIONS, {"allow_system_table_mods", PGC_SUSET, DEVELOPER_OPTIONS,
gettext_noop("Allows modifications of the structure of system tables."), gettext_noop("Allows modifications of the structure of system tables."),
...@@ -6248,6 +6261,14 @@ BeginReportingGUCOptions(void) ...@@ -6248,6 +6261,14 @@ BeginReportingGUCOptions(void)
reporting_enabled = true; reporting_enabled = true;
/*
* Hack for in_hot_standby: initialize with the value we're about to send.
* (This could be out of date by the time we actually send it, in which
* case the next ReportChangedGUCOptions call will send a duplicate
* report.)
*/
in_hot_standby = RecoveryInProgress();
/* Transmit initial values of interesting variables */ /* Transmit initial values of interesting variables */
for (i = 0; i < num_guc_variables; i++) for (i = 0; i < num_guc_variables; i++)
{ {
...@@ -6280,6 +6301,23 @@ ReportChangedGUCOptions(void) ...@@ -6280,6 +6301,23 @@ ReportChangedGUCOptions(void)
if (!reporting_enabled) if (!reporting_enabled)
return; return;
/*
* Since in_hot_standby isn't actually changed by normal GUC actions, we
* need a hack to check whether a new value needs to be reported to the
* client. For speed, we rely on the assumption that it can never
* transition from false to true.
*/
if (in_hot_standby && !RecoveryInProgress())
{
struct config_generic *record;
record = find_option("in_hot_standby", false, ERROR);
Assert(record != NULL);
record->status |= GUC_NEEDS_REPORT;
report_needed = true;
in_hot_standby = false;
}
/* Quick exit if no values have been changed */ /* Quick exit if no values have been changed */
if (!report_needed) if (!report_needed)
return; return;
...@@ -11773,6 +11811,18 @@ show_data_directory_mode(void) ...@@ -11773,6 +11811,18 @@ show_data_directory_mode(void)
return buf; return buf;
} }
static const char *
show_in_hot_standby(void)
{
/*
* We display the actual state based on shared memory, so that this GUC
* reports up-to-date state if examined intra-query. The underlying
* variable in_hot_standby changes only when we transmit a new value to
* the client.
*/
return RecoveryInProgress() ? "on" : "off";
}
/* /*
* We split the input string, where commas separate function names * We split the input string, where commas separate function names
* and certain whitespace chars are ignored, into a \0-separated (and * and certain whitespace chars are ignored, into a \0-separated (and
......
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