mirror of
https://github.com/EnterpriseDB/repmgr.git
synced 2026-03-24 07:36:30 +00:00
Compare commits
79 Commits
v3.3
...
REL3_3_STA
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9ae8f5780b | ||
|
|
dd9df04334 | ||
|
|
5411225b6f | ||
|
|
cf90bc3224 | ||
|
|
6ba9077ba5 | ||
|
|
ead4866719 | ||
|
|
a0937e959f | ||
|
|
00391ba95d | ||
|
|
01edae1b20 | ||
|
|
b92d0cc696 | ||
|
|
2264848601 | ||
|
|
657125a3fb | ||
|
|
8fefb799ee | ||
|
|
37b458dfcd | ||
|
|
72b14a7274 | ||
|
|
19684f965b | ||
|
|
9690aeb030 | ||
|
|
774a3abf24 | ||
|
|
95d6f08ff4 | ||
|
|
33af998a1e | ||
|
|
18a56b266b | ||
|
|
b7d1e7a091 | ||
|
|
c7f9fbf524 | ||
|
|
e0ea9c3be4 | ||
|
|
318f1dac40 | ||
|
|
bda4b0995c | ||
|
|
c14449f0a7 | ||
|
|
557e34b70c | ||
|
|
333083869b | ||
|
|
57fae00844 | ||
|
|
3de336f1c0 | ||
|
|
5493b57443 | ||
|
|
e53f1bf844 | ||
|
|
90638811c8 | ||
|
|
892e3b93d1 | ||
|
|
6f15a7e52e | ||
|
|
98998f73bf | ||
|
|
34ac2d8141 | ||
|
|
c820b61f28 | ||
|
|
9e620656c5 | ||
|
|
2fa277cc53 | ||
|
|
6a4f5944a1 | ||
|
|
c02a12a113 | ||
|
|
01b3933922 | ||
|
|
39b3b32814 | ||
|
|
846e0f73b2 | ||
|
|
7467525c8d | ||
|
|
b27a94ccbe | ||
|
|
2e69d155da | ||
|
|
870a367d3b | ||
|
|
9c28d3626b | ||
|
|
0916d8f2ad | ||
|
|
1964f890be | ||
|
|
976a61005e | ||
|
|
0c82278fd4 | ||
|
|
0abfde3773 | ||
|
|
1746831486 | ||
|
|
8c8e368a69 | ||
|
|
0ef532dcff | ||
|
|
478407fd86 | ||
|
|
05bfdfab2c | ||
|
|
29740dc41b | ||
|
|
ad6ecef2ab | ||
|
|
5318d37462 | ||
|
|
7244dda20f | ||
|
|
e651284927 | ||
|
|
72a2ac284a | ||
|
|
cec01c6620 | ||
|
|
989f683bc6 | ||
|
|
fa30382f2c | ||
|
|
defc2653e0 | ||
|
|
67e8ca73b5 | ||
|
|
a1a1d64e1f | ||
|
|
76509038cc | ||
|
|
7f8e50c882 | ||
|
|
5deb6c8ce4 | ||
|
|
175ee8acfc | ||
|
|
d1491f51a3 | ||
|
|
bc9febdc48 |
@@ -2,7 +2,7 @@ License and Contributions
|
|||||||
=========================
|
=========================
|
||||||
|
|
||||||
`repmgr` is licensed under the GPL v3. All of its code and documentation is
|
`repmgr` is licensed under the GPL v3. All of its code and documentation is
|
||||||
Copyright 2010-2016, 2ndQuadrant Limited. See the files COPYRIGHT and LICENSE for
|
Copyright 2010-2017, 2ndQuadrant Limited. See the files COPYRIGHT and LICENSE for
|
||||||
details.
|
details.
|
||||||
|
|
||||||
The development of repmgr has primarily been sponsored by 2ndQuadrant customers.
|
The development of repmgr has primarily been sponsored by 2ndQuadrant customers.
|
||||||
@@ -21,9 +21,11 @@ copy of the relevant Copyright Assignment Form.
|
|||||||
Code style
|
Code style
|
||||||
----------
|
----------
|
||||||
|
|
||||||
Code in repmgr is formatted to a consistent style using the following command:
|
Code in repmgr should be formatted to the same standards as the main PostgreSQL
|
||||||
|
project. For more details see:
|
||||||
|
|
||||||
astyle --style=ansi --indent=tab --suffix=none *.c *.h
|
https://www.postgresql.org/docs/current/static/source-format.html
|
||||||
|
|
||||||
Contributors should reformat their code similarly before submitting code to
|
Contributors should reformat their code similarly before submitting code to
|
||||||
the project, in order to minimize merge conflicts with other work.
|
the project, in order to minimize merge conflicts with other work.
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
Copyright (c) 2010-2016, 2ndQuadrant Limited
|
Copyright (c) 2010-2017, 2ndQuadrant Limited
|
||||||
All rights reserved.
|
All rights reserved.
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
|
|||||||
27
HISTORY
27
HISTORY
@@ -1,3 +1,30 @@
|
|||||||
|
3.3.3 2017-06
|
||||||
|
repmgr: fix `standby register --force` when updating existing node record (Ian)
|
||||||
|
|
||||||
|
3.3.2 2017-06-01
|
||||||
|
Add support for PostgreSQL 10 (Ian)
|
||||||
|
repmgr: ensure --replication-user option is honoured when passing database
|
||||||
|
connection parameters as a conninfo string (Ian)
|
||||||
|
repmgr: improve detection of pg_rewind on remote server (Ian)
|
||||||
|
repmgr: add DETAIL log output for additional clarification of error messages (Ian)
|
||||||
|
repmgr: suppress various spurious error messages in `standby follow` and
|
||||||
|
`standby switchover` (Ian)
|
||||||
|
repmgr: add missing `-P` option (Ian)
|
||||||
|
repmgrd: monitoring statistic reporting fixes (Ian)
|
||||||
|
|
||||||
|
3.3.1 2017-03-13
|
||||||
|
repmgrd: prevent invalid apply lag value being written to the
|
||||||
|
monitoring table (Ian)
|
||||||
|
repmgrd: fix error in XLogRecPtr conversion when calculating
|
||||||
|
monitoring statistics (Ian)
|
||||||
|
repmgr: if replication slots in use, where possible delete slot on old
|
||||||
|
upstream node after following new upstream (Ian)
|
||||||
|
repmgr: improve logging of rsync actions (Ian)
|
||||||
|
repmgr: improve `standby clone` when synchronous replication in use (Ian)
|
||||||
|
repmgr: stricter checking of allowed node id values
|
||||||
|
repmgr: enable `master register --force` when there is a foreign key
|
||||||
|
dependency from a standby node (Ian)
|
||||||
|
|
||||||
3.3 2016-12-27
|
3.3 2016-12-27
|
||||||
repmgr: always log to STDERR even if log facility defined (Ian)
|
repmgr: always log to STDERR even if log facility defined (Ian)
|
||||||
repmgr: add --log-to-file to log repmgr output to the defined
|
repmgr: add --log-to-file to log repmgr output to the defined
|
||||||
|
|||||||
9
Makefile
9
Makefile
@@ -1,6 +1,6 @@
|
|||||||
#
|
#
|
||||||
# Makefile
|
# Makefile
|
||||||
# Copyright (c) 2ndQuadrant, 2010-2016
|
# Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
|
|
||||||
HEADERS = $(wildcard *.h)
|
HEADERS = $(wildcard *.h)
|
||||||
|
|
||||||
@@ -8,8 +8,9 @@ repmgrd_OBJS = dbutils.o config.o repmgrd.o log.o strutil.o
|
|||||||
repmgr_OBJS = dbutils.o check_dir.o config.o repmgr.o log.o strutil.o dirmod.o compat.o
|
repmgr_OBJS = dbutils.o check_dir.o config.o repmgr.o log.o strutil.o dirmod.o compat.o
|
||||||
|
|
||||||
DATA = repmgr.sql uninstall_repmgr.sql
|
DATA = repmgr.sql uninstall_repmgr.sql
|
||||||
|
REGRESS = repmgr_funcs repmgr_test
|
||||||
|
|
||||||
PG_CPPFLAGS = -I$(libpq_srcdir)
|
PG_CPPFLAGS = -I$(includedir_internal) -I$(libpq_srcdir)
|
||||||
PG_LIBS = $(libpq_pgport)
|
PG_LIBS = $(libpq_pgport)
|
||||||
|
|
||||||
|
|
||||||
@@ -17,11 +18,11 @@ all: repmgrd repmgr
|
|||||||
$(MAKE) -C sql
|
$(MAKE) -C sql
|
||||||
|
|
||||||
repmgrd: $(repmgrd_OBJS)
|
repmgrd: $(repmgrd_OBJS)
|
||||||
$(CC) -o repmgrd $(CFLAGS) $(repmgrd_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS)
|
$(CC) -o repmgrd $(CFLAGS) $(repmgrd_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX)
|
||||||
$(MAKE) -C sql
|
$(MAKE) -C sql
|
||||||
|
|
||||||
repmgr: $(repmgr_OBJS)
|
repmgr: $(repmgr_OBJS)
|
||||||
$(CC) -o repmgr $(CFLAGS) $(repmgr_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS)
|
$(CC) -o repmgr $(CFLAGS) $(repmgr_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX)
|
||||||
|
|
||||||
# Make all objects depend on all include files. This is a bit of a
|
# Make all objects depend on all include files. This is a bit of a
|
||||||
# shotgun approach, but the codebase is small enough that a complete rebuild
|
# shotgun approach, but the codebase is small enough that a complete rebuild
|
||||||
|
|||||||
68
README.md
68
README.md
@@ -7,7 +7,7 @@ replication capabilities with utilities to set up standby servers, monitor
|
|||||||
replication, and perform administrative tasks such as failover or switchover
|
replication, and perform administrative tasks such as failover or switchover
|
||||||
operations.
|
operations.
|
||||||
|
|
||||||
The current `repmgr` version, 3.2, supports all PostgreSQL versions from
|
The current `repmgr` version (3.3) supports all PostgreSQL versions from
|
||||||
9.3 to 9.6.
|
9.3 to 9.6.
|
||||||
|
|
||||||
Overview
|
Overview
|
||||||
@@ -189,6 +189,14 @@ system.
|
|||||||
Instructions can be found in the APT section of the PostgreSQL Wiki
|
Instructions can be found in the APT section of the PostgreSQL Wiki
|
||||||
( https://wiki.postgresql.org/wiki/Apt ).
|
( https://wiki.postgresql.org/wiki/Apt ).
|
||||||
|
|
||||||
|
*NOTE*: repmgr 3.3 packages are now only available via a 2ndQuadrant-hosted
|
||||||
|
repository which can be installed like this:
|
||||||
|
|
||||||
|
apt-key adv --fetch-keys http://packages.2ndquadrant.com/repmgr3/apt/0xD3FA41F6.asc
|
||||||
|
|
||||||
|
echo deb http://packages.2ndquadrant.com/repmgr3/apt/ $(lsb_release -cs)-2ndquadrant main > /etc/apt/sources.list.d/repmgr3.list
|
||||||
|
|
||||||
|
|
||||||
See `PACKAGES.md` for details on building .deb and .rpm packages from the
|
See `PACKAGES.md` for details on building .deb and .rpm packages from the
|
||||||
`repmgr` source code.
|
`repmgr` source code.
|
||||||
|
|
||||||
@@ -202,7 +210,7 @@ See `PACKAGES.md` for details on building .deb and .rpm packages from the
|
|||||||
Release tarballs are also available:
|
Release tarballs are also available:
|
||||||
|
|
||||||
https://github.com/2ndQuadrant/repmgr/releases
|
https://github.com/2ndQuadrant/repmgr/releases
|
||||||
http://repmgr.org/downloads.php
|
http://repmgr.org/
|
||||||
|
|
||||||
`repmgr` is compiled in the same way as a PostgreSQL extension using the PGXS
|
`repmgr` is compiled in the same way as a PostgreSQL extension using the PGXS
|
||||||
infrastructure, e.g.:
|
infrastructure, e.g.:
|
||||||
@@ -314,7 +322,13 @@ The following replication settings may need to be adjusted:
|
|||||||
max_wal_senders = 10
|
max_wal_senders = 10
|
||||||
|
|
||||||
# Ensure WAL files contain enough information to enable read-only queries
|
# Ensure WAL files contain enough information to enable read-only queries
|
||||||
# on the standby
|
# on the standby.
|
||||||
|
#
|
||||||
|
# PostgreSQL 9.5 and earlier: one of 'hot_standby' or 'logical'
|
||||||
|
# PostgreSQL 9.6 and later: one of 'replica' or 'logical'
|
||||||
|
# ('hot_standby' will still be accepted as an alias for 'replica')
|
||||||
|
#
|
||||||
|
# See: https://www.postgresql.org/docs/current/static/runtime-config-wal.html#GUC-WAL-LEVEL
|
||||||
|
|
||||||
wal_level = 'hot_standby'
|
wal_level = 'hot_standby'
|
||||||
|
|
||||||
@@ -333,10 +347,11 @@ The following replication settings may need to be adjusted:
|
|||||||
archive_command = '/bin/true'
|
archive_command = '/bin/true'
|
||||||
|
|
||||||
# If cloning using rsync, or you have configured `pg_basebackup_options`
|
# If cloning using rsync, or you have configured `pg_basebackup_options`
|
||||||
# in `repmgr.conf` to include the setting `--xlog-method=fetch`, *and*
|
# in `repmgr.conf` to include the setting `--xlog-method=fetch` (from
|
||||||
# you have not set `restore_command` in `repmgr.conf`to fetch WAL files
|
# PostgreSQL 10 `--wal-method=fetch`), *and* you have not set
|
||||||
# from another source such as Barman, you'll need to set `wal_keep_segments`
|
# `restore_command` in `repmgr.conf`to fetch WAL files from another
|
||||||
# to a high enough value to ensure that all WAL files generated while
|
# source such as Barman, you'll need to set `wal_keep_segments` to a
|
||||||
|
# high enough value to ensure that all WAL files generated while
|
||||||
# the standby is being cloned are retained until the standby starts up.
|
# the standby is being cloned are retained until the standby starts up.
|
||||||
|
|
||||||
# wal_keep_segments = 5000
|
# wal_keep_segments = 5000
|
||||||
@@ -390,7 +405,8 @@ least the following parameters:
|
|||||||
|
|
||||||
- `cluster`: an arbitrary name for the replication cluster; this must be identical
|
- `cluster`: an arbitrary name for the replication cluster; this must be identical
|
||||||
on all nodes
|
on all nodes
|
||||||
- `node`: a unique integer identifying the node
|
- `node`: a unique integer identifying the node; note this must be a positive
|
||||||
|
32 bit signed integer between 1 and 2147483647
|
||||||
- `node_name`: a unique string identifying the node; we recommend a name
|
- `node_name`: a unique string identifying the node; we recommend a name
|
||||||
specific to the server (e.g. 'server_1'); avoid names indicating the
|
specific to the server (e.g. 'server_1'); avoid names indicating the
|
||||||
current replication role like 'master' or 'standby' as the server's
|
current replication role like 'master' or 'standby' as the server's
|
||||||
@@ -398,7 +414,8 @@ least the following parameters:
|
|||||||
- `conninfo`: a valid connection string for the `repmgr` database on the
|
- `conninfo`: a valid connection string for the `repmgr` database on the
|
||||||
*current* server. (On the standby, the database will not yet exist, but
|
*current* server. (On the standby, the database will not yet exist, but
|
||||||
`repmgr` needs to know the connection details to complete the setup
|
`repmgr` needs to know the connection details to complete the setup
|
||||||
process).
|
process). *NOTE* this must be a keyword/value string, not a connection
|
||||||
|
URI; this limitation will be removed in a future `repmgr` version.
|
||||||
|
|
||||||
`repmgr.conf` should not be stored inside the PostgreSQL data directory,
|
`repmgr.conf` should not be stored inside the PostgreSQL data directory,
|
||||||
as it could be overwritten when setting up or reinitialising the PostgreSQL
|
as it could be overwritten when setting up or reinitialising the PostgreSQL
|
||||||
@@ -423,7 +440,7 @@ to include this schema name, e.g.
|
|||||||
### Initialise the master server
|
### Initialise the master server
|
||||||
|
|
||||||
To enable `repmgr` to support a replication cluster, the master node must
|
To enable `repmgr` to support a replication cluster, the master node must
|
||||||
be registered with `repmgr`, which creates the `repmgr` database and adds
|
be registered with `repmgr`, which creates the `repmgr` metadatabase and adds
|
||||||
a metadata record for the server:
|
a metadata record for the server:
|
||||||
|
|
||||||
$ repmgr -f repmgr.conf master register
|
$ repmgr -f repmgr.conf master register
|
||||||
@@ -501,7 +518,8 @@ place. To ensure this happens when using the default `pg_basebackup` method,
|
|||||||
`repmgr` will set `pg_basebackup`'s `--xlog-method` parameter to `stream`,
|
`repmgr` will set `pg_basebackup`'s `--xlog-method` parameter to `stream`,
|
||||||
which will ensure all WAL files generated during the cloning process are
|
which will ensure all WAL files generated during the cloning process are
|
||||||
streamed in parallel with the main backup. Note that this requires two
|
streamed in parallel with the main backup. Note that this requires two
|
||||||
replication connections to be available.
|
replication connections to be available (`repmgr` will verify sufficient
|
||||||
|
connections are available before attempting to clone).
|
||||||
|
|
||||||
To override this behaviour, in `repmgr.conf` set `pg_basebackup`'s
|
To override this behaviour, in `repmgr.conf` set `pg_basebackup`'s
|
||||||
`--xlog-method` parameter to `fetch`:
|
`--xlog-method` parameter to `fetch`:
|
||||||
@@ -513,6 +531,9 @@ See the `pg_basebackup` documentation for details:
|
|||||||
|
|
||||||
https://www.postgresql.org/docs/current/static/app-pgbasebackup.html
|
https://www.postgresql.org/docs/current/static/app-pgbasebackup.html
|
||||||
|
|
||||||
|
> *NOTE*: From PostgreSQL 10, `pg_basebackup`'s `--xlog-method` parameter
|
||||||
|
> has been renamed to `--wal-method`.
|
||||||
|
|
||||||
Make any adjustments to the standby's PostgreSQL configuration files now,
|
Make any adjustments to the standby's PostgreSQL configuration files now,
|
||||||
then start the server.
|
then start the server.
|
||||||
|
|
||||||
@@ -625,7 +646,7 @@ In order to enable Barman support for `repmgr standby clone`, you must
|
|||||||
ensure that:
|
ensure that:
|
||||||
|
|
||||||
- the name of the server configured in Barman is equal to the
|
- the name of the server configured in Barman is equal to the
|
||||||
`cluster_name` setting in `repmgr.conf`;
|
`cluster` setting in `repmgr.conf`;
|
||||||
- the `barman_server` setting in `repmgr.conf` is set to the SSH
|
- the `barman_server` setting in `repmgr.conf` is set to the SSH
|
||||||
hostname of the Barman server;
|
hostname of the Barman server;
|
||||||
- the `restore_command` setting in `repmgr.conf` is configured to
|
- the `restore_command` setting in `repmgr.conf` is configured to
|
||||||
@@ -990,6 +1011,13 @@ both passwordless SSH access and the path of `repmgr.conf` on that server.
|
|||||||
> careful preparation and with adequate attention. In particular you should
|
> careful preparation and with adequate attention. In particular you should
|
||||||
> be confident that your network environment is stable and reliable.
|
> be confident that your network environment is stable and reliable.
|
||||||
>
|
>
|
||||||
|
> Additionally you should be sure that the current master can be shut down
|
||||||
|
> quickly and cleanly. In particular, access from applications should be
|
||||||
|
> minimalized or preferably blocked completely. Also check that there is
|
||||||
|
> no backlog of files waiting to be archived, as PostgreSQL will not shut
|
||||||
|
> down until archiving completes, and that any standbys attached to the
|
||||||
|
> current primary don't have a significant amount of replication lag.
|
||||||
|
>
|
||||||
> We recommend running `repmgr standby switchover` at the most verbose
|
> We recommend running `repmgr standby switchover` at the most verbose
|
||||||
> logging level (`--log-level DEBUG --verbose`) and capturing all output
|
> logging level (`--log-level DEBUG --verbose`) and capturing all output
|
||||||
> to assist troubleshooting any problems.
|
> to assist troubleshooting any problems.
|
||||||
@@ -1056,7 +1084,7 @@ should have been updated to reflect this:
|
|||||||
|
|
||||||
### Caveats
|
### Caveats
|
||||||
|
|
||||||
- The functionality provided `repmgr standby switchover` is primarily aimed
|
- The functionality provided by `repmgr standby switchover` is primarily aimed
|
||||||
at a two-server master/standby replication cluster and currently does
|
at a two-server master/standby replication cluster and currently does
|
||||||
not support additional standbys.
|
not support additional standbys.
|
||||||
- `repmgr standby switchover` is designed to use the `pg_rewind` utility,
|
- `repmgr standby switchover` is designed to use the `pg_rewind` utility,
|
||||||
@@ -1070,11 +1098,6 @@ should have been updated to reflect this:
|
|||||||
the `repmgrd` may try and promote a standby by itself.
|
the `repmgrd` may try and promote a standby by itself.
|
||||||
- Any other standbys attached to the old master will need to be manually
|
- Any other standbys attached to the old master will need to be manually
|
||||||
instructed to point to the new master (e.g. with `repmgr standby follow`).
|
instructed to point to the new master (e.g. with `repmgr standby follow`).
|
||||||
- You must ensure that following a server start using `pg_ctl`, log output
|
|
||||||
is not send to STDERR (the default behaviour). If logging is not configured,
|
|
||||||
we recommend setting `logging_collector=on` in `postgresql.conf` and
|
|
||||||
providing an explicit `-l/--log` setting in `repmgr.conf`'s `pg_ctl_options`
|
|
||||||
parameter.
|
|
||||||
|
|
||||||
We hope to remove some of these restrictions in future versions of `repmgr`.
|
We hope to remove some of these restrictions in future versions of `repmgr`.
|
||||||
|
|
||||||
@@ -1153,11 +1176,10 @@ Additionally the following `repmgrd` options must be set in `repmgr.conf`:
|
|||||||
promote_command='repmgr standby promote -f /etc/repmgr.conf --log-to-file'
|
promote_command='repmgr standby promote -f /etc/repmgr.conf --log-to-file'
|
||||||
follow_command='repmgr standby follow -f /etc/repmgr.conf --log-to-file'
|
follow_command='repmgr standby follow -f /etc/repmgr.conf --log-to-file'
|
||||||
|
|
||||||
Note that the `--log-to-file` option will cause `repmgr` output to be logged to
|
Note that the `--log-to-file` option will cause `repmgr`'s output to be logged to
|
||||||
the destination configured to receive log output `repmgrd`.
|
the destination configured to receive log output for `repmgrd`.
|
||||||
See `repmgr.conf.sample` for further `repmgrd`-specific settings
|
See `repmgr.conf.sample` for further `repmgrd`-specific settings
|
||||||
|
|
||||||
|
|
||||||
When `failover` is set to `automatic`, upon detecting failure of the current
|
When `failover` is set to `automatic`, upon detecting failure of the current
|
||||||
master, `repmgrd` will execute one of `promote_command` or `follow_command`,
|
master, `repmgrd` will execute one of `promote_command` or `follow_command`,
|
||||||
depending on whether the current server is becoming the new master or
|
depending on whether the current server is becoming the new master or
|
||||||
@@ -1461,7 +1483,7 @@ In general `repmgr` can be upgraded as-is without any further action required,
|
|||||||
however feature releases may require the `repmgr` database to be upgraded.
|
however feature releases may require the `repmgr` database to be upgraded.
|
||||||
An SQL script will be provided - please check the release notes for details:
|
An SQL script will be provided - please check the release notes for details:
|
||||||
|
|
||||||
* http://repmgr.org/release-notes-3.2.1.html#UPGRADING
|
* http://repmgr.org/release-notes-3.3.html#UPGRADING
|
||||||
|
|
||||||
|
|
||||||
Distribution-specific configuration
|
Distribution-specific configuration
|
||||||
@@ -1605,7 +1627,7 @@ which contains connection details for the local database.
|
|||||||
|
|
||||||
Creates a witness server as a separate PostgreSQL instance. This instance
|
Creates a witness server as a separate PostgreSQL instance. This instance
|
||||||
can be on a separate server or a server running an existing node. The
|
can be on a separate server or a server running an existing node. The
|
||||||
witness server contain a copy of the repmgr metadata tables but will not
|
witness server contains a copy of the repmgr metadata tables but will not
|
||||||
be set up as a standby; instead it will update its metadata copy each
|
be set up as a standby; instead it will update its metadata copy each
|
||||||
time a failover occurs.
|
time a failover occurs.
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* check_dir.c - Directories management functions
|
* check_dir.c - Directories management functions
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* check_dir.h
|
* check_dir.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
18
compat.c
18
compat.c
@@ -1,10 +1,12 @@
|
|||||||
/*
|
/*
|
||||||
*
|
*
|
||||||
* compat.c
|
* compat.c
|
||||||
* Provide backports of various functions not publicly
|
* Provides a couple of useful string utility functions adapted
|
||||||
* exposed before PostgreSQL 9.6
|
* from the backend code, which are not publicly exposed. They're
|
||||||
|
* unlikely to change but it would be worth keeping an eye on them
|
||||||
|
* for any fixes/improvements
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
||||||
* Portions Copyright (c) 1994, Regents of the University of California
|
* Portions Copyright (c) 1994, Regents of the University of California
|
||||||
@@ -24,8 +26,6 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#if (PG_VERSION_NUM < 90600)
|
|
||||||
|
|
||||||
#include "repmgr.h"
|
#include "repmgr.h"
|
||||||
#include "compat.h"
|
#include "compat.h"
|
||||||
|
|
||||||
@@ -34,8 +34,8 @@
|
|||||||
* the string as a value, in a keyword/pair value in a libpq connection
|
* the string as a value, in a keyword/pair value in a libpq connection
|
||||||
* string
|
* string
|
||||||
*
|
*
|
||||||
* This function is copied from src/bin/pg_dump/dumputils.c
|
* This function is adapted from src/fe_utils/string_utils.c (before 9.6
|
||||||
* as it is only publicly exposed from 9.6
|
* located in: src/bin/pg_dump/dumputils.c)
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
appendConnStrVal(PQExpBuffer buf, const char *str)
|
appendConnStrVal(PQExpBuffer buf, const char *str)
|
||||||
@@ -79,8 +79,6 @@ appendConnStrVal(PQExpBuffer buf, const char *str)
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Adapted from: src/fe_utils/string_utils.c
|
* Adapted from: src/fe_utils/string_utils.c
|
||||||
*
|
|
||||||
* Function not publicly available before PostgreSQL 9.6.
|
|
||||||
*/
|
*/
|
||||||
void
|
void
|
||||||
appendShellString(PQExpBuffer buf, const char *str)
|
appendShellString(PQExpBuffer buf, const char *str)
|
||||||
@@ -107,5 +105,3 @@ appendShellString(PQExpBuffer buf, const char *str)
|
|||||||
appendPQExpBufferChar(buf, '\'');
|
appendPQExpBufferChar(buf, '\'');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|||||||
2
compat.h
2
compat.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* compat.h
|
* compat.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
8
config.c
8
config.c
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* config.c - Functions to parse the config file
|
* config.c - Functions to parse the config file
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -30,7 +30,7 @@ static void tablespace_list_append(t_configuration_options *options, const char
|
|||||||
static void exit_with_errors(ItemList *config_errors);
|
static void exit_with_errors(ItemList *config_errors);
|
||||||
|
|
||||||
const static char *_progname = NULL;
|
const static char *_progname = NULL;
|
||||||
static char config_file_path[MAXPGPATH];
|
static char config_file_path[MAXPGPATH] = "";
|
||||||
static bool config_file_provided = false;
|
static bool config_file_provided = false;
|
||||||
bool config_file_found = false;
|
bool config_file_found = false;
|
||||||
|
|
||||||
@@ -448,6 +448,10 @@ _parse_config(t_configuration_options *options, ItemList *error_list)
|
|||||||
{
|
{
|
||||||
item_list_append(error_list, _("\"node\": must be greater than zero"));
|
item_list_append(error_list, _("\"node\": must be greater than zero"));
|
||||||
}
|
}
|
||||||
|
else if (options->node < 0)
|
||||||
|
{
|
||||||
|
item_list_append(error_list, _("\"node\": must be a positive signed 32 bit integer, i.e. 2147483647 or less"));
|
||||||
|
}
|
||||||
|
|
||||||
if (strlen(options->conninfo))
|
if (strlen(options->conninfo))
|
||||||
{
|
{
|
||||||
|
|||||||
2
config.h
2
config.h
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* config.h
|
* config.h
|
||||||
*
|
*
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
155
dbutils.c
155
dbutils.c
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* dbutils.c - Database connection/management functions
|
* dbutils.c - Database connection/management functions
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -33,6 +33,15 @@ char repmgr_schema[MAXLEN] = "";
|
|||||||
char repmgr_schema_quoted[MAXLEN] = "";
|
char repmgr_schema_quoted[MAXLEN] = "";
|
||||||
|
|
||||||
static int _get_node_record(PGconn *conn, char *cluster, char *sqlquery, t_node_info *node_info);
|
static int _get_node_record(PGconn *conn, char *cluster, char *sqlquery, t_node_info *node_info);
|
||||||
|
static bool _set_config(PGconn *conn, const char *config_param, const char *sqlquery);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* _establish_db_connection()
|
||||||
|
*
|
||||||
|
* Connect to a database using a conninfo string.
|
||||||
|
*
|
||||||
|
* NOTE: *do not* use this for replication connections; use establish_db_connection_by_params() instead.
|
||||||
|
*/
|
||||||
|
|
||||||
PGconn *
|
PGconn *
|
||||||
_establish_db_connection(const char *conninfo, const bool exit_on_error, const bool log_notice, const bool verbose_only)
|
_establish_db_connection(const char *conninfo, const bool exit_on_error, const bool log_notice, const bool verbose_only)
|
||||||
@@ -77,6 +86,19 @@ _establish_db_connection(const char *conninfo, const bool exit_on_error, const b
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* set "synchronous_commit" to "local" in case synchronous replication is in use
|
||||||
|
*/
|
||||||
|
|
||||||
|
else if (set_config(conn, "synchronous_commit", "local") == false)
|
||||||
|
{
|
||||||
|
if (exit_on_error)
|
||||||
|
{
|
||||||
|
PQfinish(conn);
|
||||||
|
exit(ERR_DB_CON);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return conn;
|
return conn;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,8 +138,12 @@ PGconn *
|
|||||||
establish_db_connection_by_params(const char *keywords[], const char *values[],
|
establish_db_connection_by_params(const char *keywords[], const char *values[],
|
||||||
const bool exit_on_error)
|
const bool exit_on_error)
|
||||||
{
|
{
|
||||||
/* Make a connection to the database */
|
PGconn *conn;
|
||||||
PGconn *conn = PQconnectdbParams(keywords, values, true);
|
bool replication_connection = false;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* Connect to the database using the provided parameters */
|
||||||
|
conn = PQconnectdbParams(keywords, values, true);
|
||||||
|
|
||||||
/* Check to see that the backend connection was successfully made */
|
/* Check to see that the backend connection was successfully made */
|
||||||
if ((PQstatus(conn) != CONNECTION_OK))
|
if ((PQstatus(conn) != CONNECTION_OK))
|
||||||
@@ -130,6 +156,28 @@ establish_db_connection_by_params(const char *keywords[], const char *values[],
|
|||||||
exit(ERR_DB_CON);
|
exit(ERR_DB_CON);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* set "synchronous_commit" to "local" in case synchronous replication is in
|
||||||
|
* use (provided this is not a replication connection)
|
||||||
|
*/
|
||||||
|
|
||||||
|
for (i = 0; keywords[i]; i++)
|
||||||
|
{
|
||||||
|
if (strcmp(keywords[i], "replication") == 0)
|
||||||
|
replication_connection = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (replication_connection == false && set_config(conn, "synchronous_commit", "local") == false)
|
||||||
|
{
|
||||||
|
if (exit_on_error)
|
||||||
|
{
|
||||||
|
PQfinish(conn);
|
||||||
|
exit(ERR_DB_CON);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return conn;
|
return conn;
|
||||||
}
|
}
|
||||||
@@ -380,6 +428,8 @@ int
|
|||||||
get_server_version(PGconn *conn, char *server_version)
|
get_server_version(PGconn *conn, char *server_version)
|
||||||
{
|
{
|
||||||
PGresult *res;
|
PGresult *res;
|
||||||
|
int server_version_num;
|
||||||
|
|
||||||
res = PQexec(conn,
|
res = PQexec(conn,
|
||||||
"SELECT current_setting('server_version_num'), "
|
"SELECT current_setting('server_version_num'), "
|
||||||
" current_setting('server_version')");
|
" current_setting('server_version')");
|
||||||
@@ -393,9 +443,12 @@ get_server_version(PGconn *conn, char *server_version)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (server_version != NULL)
|
if (server_version != NULL)
|
||||||
strcpy(server_version, PQgetvalue(res, 0, 0));
|
strcpy(server_version, PQgetvalue(res, 0, 1));
|
||||||
|
|
||||||
return atoi(PQgetvalue(res, 0, 0));
|
server_version_num = atoi(PQgetvalue(res, 0, 0));
|
||||||
|
|
||||||
|
PQclear(res);
|
||||||
|
return server_version_num;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -1084,15 +1137,25 @@ drop_replication_slot(PGconn *conn, char *slot_name)
|
|||||||
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint)
|
start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint, int server_version_num)
|
||||||
{
|
{
|
||||||
char sqlquery[QUERY_STR_LEN];
|
char sqlquery[QUERY_STR_LEN];
|
||||||
PGresult *res;
|
PGresult *res;
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
{
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
"SELECT pg_catalog.pg_walfile_name(pg_catalog.pg_start_backup('repmgr_standby_clone_%ld', %s))",
|
||||||
|
time(NULL),
|
||||||
|
fast_checkpoint ? "TRUE" : "FALSE");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
sqlquery_snprintf(sqlquery,
|
sqlquery_snprintf(sqlquery,
|
||||||
"SELECT pg_catalog.pg_xlogfile_name(pg_catalog.pg_start_backup('repmgr_standby_clone_%ld', %s))",
|
"SELECT pg_catalog.pg_xlogfile_name(pg_catalog.pg_start_backup('repmgr_standby_clone_%ld', %s))",
|
||||||
time(NULL),
|
time(NULL),
|
||||||
fast_checkpoint ? "TRUE" : "FALSE");
|
fast_checkpoint ? "TRUE" : "FALSE");
|
||||||
|
}
|
||||||
|
|
||||||
log_verbose(LOG_DEBUG, "start_backup():\n%s\n", sqlquery);
|
log_verbose(LOG_DEBUG, "start_backup():\n%s\n", sqlquery);
|
||||||
|
|
||||||
@@ -1120,12 +1183,19 @@ start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint)
|
|||||||
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
stop_backup(PGconn *conn, char *last_wal_segment)
|
stop_backup(PGconn *conn, char *last_wal_segment, int server_version_num)
|
||||||
{
|
{
|
||||||
char sqlquery[QUERY_STR_LEN];
|
char sqlquery[QUERY_STR_LEN];
|
||||||
PGresult *res;
|
PGresult *res;
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
{
|
||||||
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_walfile_name(pg_catalog.pg_stop_backup())");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_xlogfile_name(pg_catalog.pg_stop_backup())");
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_xlogfile_name(pg_catalog.pg_stop_backup())");
|
||||||
|
}
|
||||||
|
|
||||||
res = PQexec(conn, sqlquery);
|
res = PQexec(conn, sqlquery);
|
||||||
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
||||||
@@ -1150,19 +1220,12 @@ stop_backup(PGconn *conn, char *last_wal_segment)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
set_config_bool(PGconn *conn, const char *config_param, bool state)
|
_set_config(PGconn *conn, const char *config_param, const char *sqlquery)
|
||||||
{
|
{
|
||||||
char sqlquery[QUERY_STR_LEN];
|
|
||||||
PGresult *res;
|
PGresult *res;
|
||||||
|
|
||||||
sqlquery_snprintf(sqlquery,
|
|
||||||
"SET %s TO %s",
|
|
||||||
config_param,
|
|
||||||
state ? "TRUE" : "FALSE");
|
|
||||||
|
|
||||||
log_verbose(LOG_DEBUG, "set_config_bool():\n%s\n", sqlquery);
|
|
||||||
|
|
||||||
res = PQexec(conn, sqlquery);
|
res = PQexec(conn, sqlquery);
|
||||||
|
|
||||||
if (PQresultStatus(res) != PGRES_COMMAND_OK)
|
if (PQresultStatus(res) != PGRES_COMMAND_OK)
|
||||||
@@ -1177,6 +1240,36 @@ set_config_bool(PGconn *conn, const char *config_param, bool state)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
set_config(PGconn *conn, const char *config_param, const char *config_value)
|
||||||
|
{
|
||||||
|
char sqlquery[QUERY_STR_LEN];
|
||||||
|
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
"SET %s TO '%s'",
|
||||||
|
config_param,
|
||||||
|
config_value);
|
||||||
|
|
||||||
|
log_verbose(LOG_DEBUG, "set_config():\n%s\n", sqlquery);
|
||||||
|
|
||||||
|
return _set_config(conn, config_param, sqlquery);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
set_config_bool(PGconn *conn, const char *config_param, bool state)
|
||||||
|
{
|
||||||
|
char sqlquery[QUERY_STR_LEN];
|
||||||
|
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
"SET %s TO %s",
|
||||||
|
config_param,
|
||||||
|
state ? "TRUE" : "FALSE");
|
||||||
|
|
||||||
|
log_verbose(LOG_DEBUG, "set_config_bool():\n%s\n", sqlquery);
|
||||||
|
|
||||||
|
return _set_config(conn, config_param, sqlquery);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* witness_copy_node_records()
|
* witness_copy_node_records()
|
||||||
@@ -1628,6 +1721,27 @@ create_event_record(PGconn *conn, t_configuration_options *options, int node_id,
|
|||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
create_checkpoint(PGconn *conn)
|
||||||
|
{
|
||||||
|
char sqlquery[MAXLEN];
|
||||||
|
PGresult *res;
|
||||||
|
|
||||||
|
sqlquery_snprintf(sqlquery, "CHECKPOINT");
|
||||||
|
log_verbose(LOG_DEBUG, "checkpoint:\n%s\n", sqlquery);
|
||||||
|
|
||||||
|
res = PQexec(conn, sqlquery);
|
||||||
|
if (!res || PQresultStatus(res) != PGRES_COMMAND_OK)
|
||||||
|
{
|
||||||
|
log_err(_("Unable to create CHECKPOINT:\n%s\n"),
|
||||||
|
PQerrorMessage(conn));
|
||||||
|
PQfinish(conn);
|
||||||
|
exit(ERR_DB_QUERY);
|
||||||
|
}
|
||||||
|
|
||||||
|
log_notice(_("CHECKPOINT created\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
bool
|
bool
|
||||||
update_node_record(PGconn *conn, char *action, int node, char *type, int upstream_node, char *cluster_name, char *node_name, char *conninfo, int priority, char *slot_name, bool active)
|
update_node_record(PGconn *conn, char *action, int node, char *type, int upstream_node, char *cluster_name, char *node_name, char *conninfo, int priority, char *slot_name, bool active)
|
||||||
@@ -1872,7 +1986,16 @@ _get_node_record(PGconn *conn, char *cluster, char *sqlquery, t_node_info *node_
|
|||||||
|
|
||||||
node_info->node_id = atoi(PQgetvalue(res, 0, 0));
|
node_info->node_id = atoi(PQgetvalue(res, 0, 0));
|
||||||
node_info->type = parse_node_type(PQgetvalue(res, 0, 1));
|
node_info->type = parse_node_type(PQgetvalue(res, 0, 1));
|
||||||
|
|
||||||
|
if (PQgetisnull(res, 0, 2))
|
||||||
|
{
|
||||||
|
node_info->upstream_node_id = NO_UPSTREAM_NODE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
node_info->upstream_node_id = atoi(PQgetvalue(res, 0, 2));
|
node_info->upstream_node_id = atoi(PQgetvalue(res, 0, 2));
|
||||||
|
}
|
||||||
|
|
||||||
strncpy(node_info->name, PQgetvalue(res, 0, 3), MAXLEN);
|
strncpy(node_info->name, PQgetvalue(res, 0, 3), MAXLEN);
|
||||||
strncpy(node_info->conninfo_str, PQgetvalue(res, 0, 4), MAXLEN);
|
strncpy(node_info->conninfo_str, PQgetvalue(res, 0, 4), MAXLEN);
|
||||||
strncpy(node_info->slot_name, PQgetvalue(res, 0, 5), MAXLEN);
|
strncpy(node_info->slot_name, PQgetvalue(res, 0, 5), MAXLEN);
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* dbutils.h
|
* dbutils.h
|
||||||
*
|
*
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -122,8 +122,9 @@ char *get_repmgr_schema_quoted(PGconn *conn);
|
|||||||
bool create_replication_slot(PGconn *conn, char *slot_name, int server_version_num, PQExpBufferData *error_msg);
|
bool create_replication_slot(PGconn *conn, char *slot_name, int server_version_num, PQExpBufferData *error_msg);
|
||||||
int get_slot_record(PGconn *conn, char *slot_name, t_replication_slot *record);
|
int get_slot_record(PGconn *conn, char *slot_name, t_replication_slot *record);
|
||||||
bool drop_replication_slot(PGconn *conn, char *slot_name);
|
bool drop_replication_slot(PGconn *conn, char *slot_name);
|
||||||
bool start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint);
|
bool start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint, int server_version_num);
|
||||||
bool stop_backup(PGconn *conn, char *last_wal_segment);
|
bool stop_backup(PGconn *conn, char *last_wal_segment, int server_version_num);
|
||||||
|
bool set_config(PGconn *conn, const char *config_param, const char *config_value);
|
||||||
bool set_config_bool(PGconn *conn, const char *config_param, bool state);
|
bool set_config_bool(PGconn *conn, const char *config_param, bool state);
|
||||||
bool witness_copy_node_records(PGconn *masterconn, PGconn *witnessconn, char *cluster_name);
|
bool witness_copy_node_records(PGconn *masterconn, PGconn *witnessconn, char *cluster_name);
|
||||||
bool create_node_record(PGconn *conn, char *action, int node, char *type, int upstream_node, char *cluster_name, char *node_name, char *conninfo, int priority, char *slot_name, bool active);
|
bool create_node_record(PGconn *conn, char *action, int node, char *type, int upstream_node, char *cluster_name, char *node_name, char *conninfo, int priority, char *slot_name, bool active);
|
||||||
@@ -134,9 +135,9 @@ bool update_node_record(PGconn *conn, char *action, int node, char *type,
|
|||||||
bool update_node_record_status(PGconn *conn, char *cluster_name, int this_node_id, char *type, int upstream_node_id, bool active);
|
bool update_node_record_status(PGconn *conn, char *cluster_name, int this_node_id, char *type, int upstream_node_id, bool active);
|
||||||
bool update_node_record_set_upstream(PGconn *conn, char *cluster_name, int this_node_id, int new_upstream_node_id);
|
bool update_node_record_set_upstream(PGconn *conn, char *cluster_name, int this_node_id, int new_upstream_node_id);
|
||||||
bool create_event_record(PGconn *conn, t_configuration_options *options, int node_id, char *event, bool successful, char *details);
|
bool create_event_record(PGconn *conn, t_configuration_options *options, int node_id, char *event, bool successful, char *details);
|
||||||
|
void create_checkpoint(PGconn *conn);
|
||||||
|
|
||||||
int get_node_replication_state(PGconn *conn, char *node_name, char *output);
|
int get_node_replication_state(PGconn *conn, char *node_name, char *output);
|
||||||
t_server_type parse_node_type(const char *type);
|
t_server_type parse_node_type(const char *type);
|
||||||
int get_data_checksum_version(const char *data_directory);
|
int get_data_checksum_version(const char *data_directory);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
2
dirmod.c
2
dirmod.c
@@ -3,7 +3,7 @@
|
|||||||
* dirmod.c
|
* dirmod.c
|
||||||
* directory handling functions
|
* directory handling functions
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
||||||
* Portions Copyright (c) 1994, Regents of the University of California
|
* Portions Copyright (c) 1994, Regents of the University of California
|
||||||
|
|||||||
2
dirmod.h
2
dirmod.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* dirmod.h
|
* dirmod.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -53,7 +53,7 @@ will be carried out:
|
|||||||
e.g. if a replication cluster is spread over multiple data centres, a split-brain
|
e.g. if a replication cluster is spread over multiple data centres, a split-brain
|
||||||
situation does not occur if there is a network failure between datacentres. Note
|
situation does not occur if there is a network failure between datacentres. Note
|
||||||
that if nodes are split evenly between data centres, a witness server can be
|
that if nodes are split evenly between data centres, a witness server can be
|
||||||
used to establish the "majority" daat centre.
|
used to establish the "majority" data centre.
|
||||||
|
|
||||||
* `repmgrd` polls all visible servers and waits for each node to return a valid LSN;
|
* `repmgrd` polls all visible servers and waits for each node to return a valid LSN;
|
||||||
it updates the LSN previously stored for this node if it has increased since
|
it updates the LSN previously stored for this node if it has increased since
|
||||||
|
|||||||
@@ -49,6 +49,14 @@ the `%include` directive (available from PgBouncer 1.6) to include a separate
|
|||||||
configuration file, `/etc/pgbouncer.database.ini`, which will be modified by
|
configuration file, `/etc/pgbouncer.database.ini`, which will be modified by
|
||||||
`repmgr`.
|
`repmgr`.
|
||||||
|
|
||||||
|
* * *
|
||||||
|
|
||||||
|
> *NOTE*: in this self-contained demonstration, `pgbouncer` is running on the
|
||||||
|
> database servers, however in a production environment it will make more
|
||||||
|
> sense to run `pgbouncer` on either separate nodes or the application server.
|
||||||
|
|
||||||
|
* * *
|
||||||
|
|
||||||
`/etc/pgbouncer.ini` should look something like this:
|
`/etc/pgbouncer.ini` should look something like this:
|
||||||
|
|
||||||
[pgbouncer]
|
[pgbouncer]
|
||||||
@@ -125,7 +133,7 @@ The actual script is as follows; adjust the configurable items as appropriate:
|
|||||||
|
|
||||||
psql -d $REPMGR_DB -U $REPMGR_USER -t -A \
|
psql -d $REPMGR_DB -U $REPMGR_USER -t -A \
|
||||||
-c "SELECT '${PGBOUNCER_DATABASE}-ro= ' || conninfo || ' application_name=pgbouncer_${HOST}' \
|
-c "SELECT '${PGBOUNCER_DATABASE}-ro= ' || conninfo || ' application_name=pgbouncer_${HOST}' \
|
||||||
FROM $REPMGR_SCHEMA.repl_nodes \
|
FROM ${REPMGR_SCHEMA}.repl_nodes \
|
||||||
WHERE node_name='${HOST}'" >> $PGBOUNCER_DATABASE_INI_NEW
|
WHERE node_name='${HOST}'" >> $PGBOUNCER_DATABASE_INI_NEW
|
||||||
|
|
||||||
rsync $PGBOUNCER_DATABASE_INI_NEW $HOST:$PGBOUNCER_DATABASE_INI
|
rsync $PGBOUNCER_DATABASE_INI_NEW $HOST:$PGBOUNCER_DATABASE_INI
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* errcode.h
|
* errcode.h
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
18
expected/repmgr_funcs.out
Normal file
18
expected/repmgr_funcs.out
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
/*
|
||||||
|
* repmgr_function.sql
|
||||||
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
-- SET SEARCH_PATH TO 'repmgr';
|
||||||
|
CREATE FUNCTION repmgr_update_standby_location(text) RETURNS boolean
|
||||||
|
AS '$libdir/repmgr_funcs', 'repmgr_update_standby_location'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
CREATE FUNCTION repmgr_get_last_standby_location() RETURNS text
|
||||||
|
AS '$libdir/repmgr_funcs', 'repmgr_get_last_standby_location'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
CREATE FUNCTION repmgr_update_last_updated() RETURNS TIMESTAMP WITH TIME ZONE
|
||||||
|
AS '$libdir/repmgr_funcs', 'repmgr_update_last_updated'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
CREATE FUNCTION repmgr_get_last_updated() RETURNS TIMESTAMP WITH TIME ZONE
|
||||||
|
AS '$libdir/repmgr_funcs', 'repmgr_get_last_updated'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
24
expected/repmgr_test.out
Normal file
24
expected/repmgr_test.out
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
select * from repmgr_update_standby_location('');
|
||||||
|
repmgr_update_standby_location
|
||||||
|
--------------------------------
|
||||||
|
f
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select * from repmgr_get_last_standby_location();
|
||||||
|
repmgr_get_last_standby_location
|
||||||
|
----------------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select * from repmgr_update_last_updated();
|
||||||
|
repmgr_update_last_updated
|
||||||
|
----------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
|
select * from repmgr_get_last_updated();
|
||||||
|
repmgr_get_last_updated
|
||||||
|
-------------------------
|
||||||
|
|
||||||
|
(1 row)
|
||||||
|
|
||||||
18
log.c
18
log.c
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* log.c - Logging methods
|
* log.c - Logging methods
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This module is a set of methods for logging (currently only syslog)
|
* This module is a set of methods for logging (currently only syslog)
|
||||||
*
|
*
|
||||||
@@ -71,7 +71,7 @@ _stderr_log_with_level(const char *level_name, int level, const char *fmt, va_li
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Store the requested level so that if there's a subsequent
|
* Store the requested level so that if there's a subsequent
|
||||||
* log_hint(), we can suppress that if appropriate.
|
* log_hint() or log_detail(), we can suppress that if appropriate.
|
||||||
*/
|
*/
|
||||||
last_log_level = level;
|
last_log_level = level;
|
||||||
|
|
||||||
@@ -113,6 +113,20 @@ log_hint(const char *fmt, ...)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
log_detail(const char *fmt, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
if (terse_logging == false)
|
||||||
|
{
|
||||||
|
va_start(ap, fmt);
|
||||||
|
_stderr_log_with_level("DETAIL", last_log_level, fmt, ap);
|
||||||
|
va_end(ap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
log_verbose(int level, const char *fmt, ...)
|
log_verbose(int level, const char *fmt, ...)
|
||||||
{
|
{
|
||||||
|
|||||||
4
log.h
4
log.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* log.h
|
* log.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -126,6 +126,8 @@ bool logger_shutdown(void);
|
|||||||
void logger_set_verbose(void);
|
void logger_set_verbose(void);
|
||||||
void logger_set_terse(void);
|
void logger_set_terse(void);
|
||||||
|
|
||||||
|
void log_detail(const char *fmt, ...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 1, 2)));
|
||||||
void log_hint(const char *fmt, ...)
|
void log_hint(const char *fmt, ...)
|
||||||
__attribute__((format(PG_PRINTF_ATTRIBUTE, 1, 2)));
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 1, 2)));
|
||||||
void log_verbose(int level, const char *fmt, ...)
|
void log_verbose(int level, const char *fmt, ...)
|
||||||
|
|||||||
@@ -26,11 +26,14 @@
|
|||||||
# the server's hostname or another identifier unambiguously
|
# the server's hostname or another identifier unambiguously
|
||||||
# associated with the server to avoid confusion
|
# associated with the server to avoid confusion
|
||||||
|
|
||||||
# Database connection information as a conninfo string
|
# Database connection information as a conninfo string (this must be a
|
||||||
# This must be accessible to all servers in the cluster; for details see:
|
# keyword/value string, not a connection URI).
|
||||||
#
|
#
|
||||||
# https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNSTRING
|
# https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNSTRING
|
||||||
#
|
#
|
||||||
|
# All servers in the cluster must be able to access the database
|
||||||
|
# using this connection string.
|
||||||
|
#
|
||||||
#conninfo='host=192.168.204.104 dbname=repmgr user=repmgr'
|
#conninfo='host=192.168.204.104 dbname=repmgr user=repmgr'
|
||||||
#
|
#
|
||||||
# If repmgrd is in use, consider explicitly setting `connect_timeout` in the
|
# If repmgrd is in use, consider explicitly setting `connect_timeout` in the
|
||||||
@@ -80,12 +83,6 @@
|
|||||||
#
|
#
|
||||||
#logfile='/var/log/repmgr/repmgr.log'
|
#logfile='/var/log/repmgr/repmgr.log'
|
||||||
|
|
||||||
# By default only repmgrd log output will be written to a file,
|
|
||||||
# if defined in "logfile"
|
|
||||||
# enable this to restore old behaviour where output from the repmgr
|
|
||||||
# client will be written to the logfile too
|
|
||||||
#log_repmgr_to_file = 0
|
|
||||||
|
|
||||||
# event notifications can be passed to an arbitrary external program
|
# event notifications can be passed to an arbitrary external program
|
||||||
# together with the following parameters:
|
# together with the following parameters:
|
||||||
#
|
#
|
||||||
@@ -149,8 +146,15 @@
|
|||||||
# external command arguments. Values shown are examples.
|
# external command arguments. Values shown are examples.
|
||||||
|
|
||||||
#pg_ctl_options='-s'
|
#pg_ctl_options='-s'
|
||||||
#pg_basebackup_options='--xlog-method=s'
|
#pg_basebackup_options='--label=repmgr_backup'
|
||||||
|
|
||||||
|
# This is the host name of the barman server, which is used for connecting over
|
||||||
|
# to the barman server (passwordless ssh keys should be in place)
|
||||||
|
#barman_server='backup_server'
|
||||||
|
# If you are placing the barman.conf file in a non-standard path, or using
|
||||||
|
# a name other than barman.conf, use this parameter to specify the path and
|
||||||
|
# name of the barman configuration file.
|
||||||
|
#barman_config='/path/to/barman.conf'
|
||||||
|
|
||||||
# Standby clone settings
|
# Standby clone settings
|
||||||
# ----------------------
|
# ----------------------
|
||||||
|
|||||||
8
repmgr.h
8
repmgr.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* repmgr.h
|
* repmgr.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -55,7 +55,6 @@
|
|||||||
#define OPT_COPY_EXTERNAL_CONFIG_FILES 4
|
#define OPT_COPY_EXTERNAL_CONFIG_FILES 4
|
||||||
#define OPT_CONFIG_ARCHIVE_DIR 5
|
#define OPT_CONFIG_ARCHIVE_DIR 5
|
||||||
#define OPT_PG_REWIND 6
|
#define OPT_PG_REWIND 6
|
||||||
#define OPT_PWPROMPT 7
|
|
||||||
#define OPT_CSV 8
|
#define OPT_CSV 8
|
||||||
#define OPT_NODE 9
|
#define OPT_NODE 9
|
||||||
#define OPT_WITHOUT_BARMAN 10
|
#define OPT_WITHOUT_BARMAN 10
|
||||||
@@ -118,7 +117,7 @@ typedef struct
|
|||||||
|
|
||||||
char recovery_min_apply_delay[MAXLEN];
|
char recovery_min_apply_delay[MAXLEN];
|
||||||
|
|
||||||
/* standby register paarameters */
|
/* standby register parameters */
|
||||||
bool wait_register_sync;
|
bool wait_register_sync;
|
||||||
int wait_register_sync_seconds;
|
int wait_register_sync_seconds;
|
||||||
|
|
||||||
@@ -193,9 +192,10 @@ typedef struct
|
|||||||
{
|
{
|
||||||
char slot[MAXLEN];
|
char slot[MAXLEN];
|
||||||
char xlog_method[MAXLEN];
|
char xlog_method[MAXLEN];
|
||||||
|
bool no_slot; /* from PostgreSQL 10 */
|
||||||
} t_basebackup_options;
|
} t_basebackup_options;
|
||||||
|
|
||||||
#define T_BASEBACKUP_OPTIONS_INITIALIZER { "", "" }
|
#define T_BASEBACKUP_OPTIONS_INITIALIZER { "", "", false }
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* repmgr.sql
|
* repmgr.sql
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
108
repmgrd.c
108
repmgrd.c
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* repmgrd.c - Replication manager daemon
|
* repmgrd.c - Replication manager daemon
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This module connects to the nodes of a replication cluster and monitors
|
* This module connects to the nodes of a replication cluster and monitors
|
||||||
* how far are they from master
|
* how far are they from master
|
||||||
@@ -30,18 +30,10 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#include "repmgr.h"
|
#include "repmgr.h"
|
||||||
#include "config.h"
|
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "strutil.h"
|
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
|
|
||||||
/* Required PostgreSQL headers */
|
|
||||||
#include "access/xlogdefs.h"
|
|
||||||
#include "pqexpbuffer.h"
|
|
||||||
|
|
||||||
/* Message strings passed in repmgrSharedState->location */
|
/* Message strings passed in repmgrSharedState->location */
|
||||||
|
|
||||||
#define PASSIVE_NODE "PASSIVE_NODE"
|
#define PASSIVE_NODE "PASSIVE_NODE"
|
||||||
@@ -71,6 +63,7 @@ bool failover_done = false;
|
|||||||
bool manual_mode_upstream_disconnected = false;
|
bool manual_mode_upstream_disconnected = false;
|
||||||
|
|
||||||
char *pid_file = NULL;
|
char *pid_file = NULL;
|
||||||
|
int server_version_num = 0;
|
||||||
|
|
||||||
static void help(void);
|
static void help(void);
|
||||||
static void usage(void);
|
static void usage(void);
|
||||||
@@ -145,8 +138,6 @@ main(int argc, char **argv)
|
|||||||
|
|
||||||
FILE *fd;
|
FILE *fd;
|
||||||
|
|
||||||
int server_version_num = 0;
|
|
||||||
|
|
||||||
set_progname(argv[0]);
|
set_progname(argv[0]);
|
||||||
|
|
||||||
/* Disallow running as root to prevent directory ownership problems */
|
/* Disallow running as root to prevent directory ownership problems */
|
||||||
@@ -718,12 +709,31 @@ witness_monitor(void)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
strcpy(monitor_witness_timestamp, PQgetvalue(res, 0, 0));
|
strncpy(monitor_witness_timestamp, PQgetvalue(res, 0, 0), MAXLEN);
|
||||||
PQclear(res);
|
PQclear(res);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Build the SQL to execute on master
|
* Build the SQL to execute on master
|
||||||
*/
|
*/
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
{
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
"INSERT INTO %s.repl_monitor "
|
||||||
|
" (primary_node, standby_node, "
|
||||||
|
" last_monitor_time, last_apply_time, "
|
||||||
|
" last_wal_primary_location, last_wal_standby_location, "
|
||||||
|
" replication_lag, apply_lag )"
|
||||||
|
" VALUES(%d, %d, "
|
||||||
|
" '%s'::TIMESTAMP WITH TIME ZONE, NULL, "
|
||||||
|
" pg_catalog.pg_current_wal_lsn(), NULL, "
|
||||||
|
" 0, 0) ",
|
||||||
|
get_repmgr_schema_quoted(my_local_conn),
|
||||||
|
master_options.node,
|
||||||
|
local_options.node,
|
||||||
|
monitor_witness_timestamp);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
sqlquery_snprintf(sqlquery,
|
sqlquery_snprintf(sqlquery,
|
||||||
"INSERT INTO %s.repl_monitor "
|
"INSERT INTO %s.repl_monitor "
|
||||||
" (primary_node, standby_node, "
|
" (primary_node, standby_node, "
|
||||||
@@ -738,6 +748,7 @@ witness_monitor(void)
|
|||||||
master_options.node,
|
master_options.node,
|
||||||
local_options.node,
|
local_options.node,
|
||||||
monitor_witness_timestamp);
|
monitor_witness_timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Execute the query asynchronously, but don't check for a result. We will
|
* Execute the query asynchronously, but don't check for a result. We will
|
||||||
@@ -1125,6 +1136,27 @@ standby_monitor(void)
|
|||||||
* If receive_location is less than replay location, we were streaming WAL but are
|
* If receive_location is less than replay location, we were streaming WAL but are
|
||||||
* somehow disconnected and evidently in archive recovery
|
* somehow disconnected and evidently in archive recovery
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
{
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
" SELECT ts, "
|
||||||
|
" CASE WHEN (receive_location IS NULL OR receive_location < replay_location) "
|
||||||
|
" THEN replay_location "
|
||||||
|
" ELSE receive_location"
|
||||||
|
" END AS receive_location,"
|
||||||
|
" replay_location, "
|
||||||
|
" replay_timestamp, "
|
||||||
|
" COALESCE(receive_location, '0/0') >= replay_location AS receiving_streamed_wal "
|
||||||
|
" FROM (SELECT CURRENT_TIMESTAMP AS ts, "
|
||||||
|
" pg_catalog.pg_last_wal_receive_lsn() AS receive_location, "
|
||||||
|
" pg_catalog.pg_last_wal_replay_lsn() AS replay_location, "
|
||||||
|
" pg_catalog.pg_last_xact_replay_timestamp() AS replay_timestamp "
|
||||||
|
" ) q ");
|
||||||
|
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
sqlquery_snprintf(sqlquery,
|
sqlquery_snprintf(sqlquery,
|
||||||
" SELECT ts, "
|
" SELECT ts, "
|
||||||
" CASE WHEN (receive_location IS NULL OR receive_location < replay_location) "
|
" CASE WHEN (receive_location IS NULL OR receive_location < replay_location) "
|
||||||
@@ -1139,7 +1171,7 @@ standby_monitor(void)
|
|||||||
" pg_catalog.pg_last_xlog_replay_location() AS replay_location, "
|
" pg_catalog.pg_last_xlog_replay_location() AS replay_location, "
|
||||||
" pg_catalog.pg_last_xact_replay_timestamp() AS replay_timestamp "
|
" pg_catalog.pg_last_xact_replay_timestamp() AS replay_timestamp "
|
||||||
" ) q ");
|
" ) q ");
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
res = PQexec(my_local_conn, sqlquery);
|
res = PQexec(my_local_conn, sqlquery);
|
||||||
@@ -1173,6 +1205,10 @@ standby_monitor(void)
|
|||||||
* TODO: investigate whether pg_current_xlog_insert_location() would be a better
|
* TODO: investigate whether pg_current_xlog_insert_location() would be a better
|
||||||
* choice; see: https://github.com/2ndQuadrant/repmgr/issues/189
|
* choice; see: https://github.com/2ndQuadrant/repmgr/issues/189
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_current_wal_lsn()");
|
||||||
|
else
|
||||||
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_current_xlog_location()");
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_current_xlog_location()");
|
||||||
|
|
||||||
res = PQexec(master_conn, sqlquery);
|
res = PQexec(master_conn, sqlquery);
|
||||||
@@ -1187,10 +1223,22 @@ standby_monitor(void)
|
|||||||
PQclear(res);
|
PQclear(res);
|
||||||
|
|
||||||
lsn_master_current_xlog_location = lsn_to_xlogrecptr(last_wal_primary_location, NULL);
|
lsn_master_current_xlog_location = lsn_to_xlogrecptr(last_wal_primary_location, NULL);
|
||||||
lsn_last_xlog_replay_location = lsn_to_xlogrecptr(last_xlog_replay_location, NULL);
|
|
||||||
lsn_last_xlog_receive_location = lsn_to_xlogrecptr(last_xlog_receive_location, NULL);
|
lsn_last_xlog_receive_location = lsn_to_xlogrecptr(last_xlog_receive_location, NULL);
|
||||||
|
lsn_last_xlog_replay_location = lsn_to_xlogrecptr(last_xlog_replay_location, NULL);
|
||||||
|
|
||||||
|
if (lsn_last_xlog_receive_location >= lsn_last_xlog_replay_location)
|
||||||
|
{
|
||||||
apply_lag = (long long unsigned int)lsn_last_xlog_receive_location - lsn_last_xlog_replay_location;
|
apply_lag = (long long unsigned int)lsn_last_xlog_receive_location - lsn_last_xlog_replay_location;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* This should never happen, but in case it does set apply lag to zero */
|
||||||
|
log_warning("Standby receive (%s) location appears less than standby replay location (%s)\n",
|
||||||
|
last_xlog_receive_location,
|
||||||
|
last_xlog_replay_location);
|
||||||
|
apply_lag = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Calculate replication lag */
|
/* Calculate replication lag */
|
||||||
if (lsn_master_current_xlog_location >= lsn_last_xlog_receive_location)
|
if (lsn_master_current_xlog_location >= lsn_last_xlog_receive_location)
|
||||||
@@ -1199,7 +1247,7 @@ standby_monitor(void)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* This should never happen, but in case it does set lag to zero */
|
/* This should never happen, but in case it does set replication lag to zero */
|
||||||
log_warning("Master xlog (%s) location appears less than standby receive location (%s)\n",
|
log_warning("Master xlog (%s) location appears less than standby receive location (%s)\n",
|
||||||
last_wal_primary_location,
|
last_wal_primary_location,
|
||||||
last_xlog_receive_location);
|
last_xlog_receive_location);
|
||||||
@@ -1244,8 +1292,23 @@ standby_monitor(void)
|
|||||||
log_verbose(LOG_DEBUG, "standby_monitor:() %s\n", sqlquery);
|
log_verbose(LOG_DEBUG, "standby_monitor:() %s\n", sqlquery);
|
||||||
|
|
||||||
if (PQsendQuery(master_conn, sqlquery) == 0)
|
if (PQsendQuery(master_conn, sqlquery) == 0)
|
||||||
log_warning(_("query could not be sent to master. %s\n"),
|
{
|
||||||
|
log_warning(_("query could not be sent to master: %s\n"),
|
||||||
PQerrorMessage(master_conn));
|
PQerrorMessage(master_conn));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
sqlquery_snprintf(sqlquery,
|
||||||
|
"SELECT %s.repmgr_update_last_updated();",
|
||||||
|
get_repmgr_schema_quoted(my_local_conn));
|
||||||
|
res = PQexec(my_local_conn, sqlquery);
|
||||||
|
|
||||||
|
/* not critical if the above query fails*/
|
||||||
|
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
||||||
|
log_warning(_("unable to set last_updated: %s\n"), PQerrorMessage(my_local_conn));
|
||||||
|
|
||||||
|
PQclear(res);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -1437,7 +1500,11 @@ do_master_failover(void)
|
|||||||
terminate(ERR_FAILOVER_FAIL);
|
terminate(ERR_FAILOVER_FAIL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_last_wal_receive_lsn()");
|
||||||
|
else
|
||||||
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_last_xlog_receive_location()");
|
sqlquery_snprintf(sqlquery, "SELECT pg_catalog.pg_last_xlog_receive_location()");
|
||||||
|
|
||||||
res = PQexec(node_conn, sqlquery);
|
res = PQexec(node_conn, sqlquery);
|
||||||
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
||||||
{
|
{
|
||||||
@@ -1469,7 +1536,12 @@ do_master_failover(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* last we get info about this node, and update shared memory */
|
/* last we get info about this node, and update shared memory */
|
||||||
|
|
||||||
|
if (server_version_num >= 100000)
|
||||||
|
sprintf(sqlquery, "SELECT pg_catalog.pg_last_wal_receive_lsn()");
|
||||||
|
else
|
||||||
sprintf(sqlquery, "SELECT pg_catalog.pg_last_xlog_receive_location()");
|
sprintf(sqlquery, "SELECT pg_catalog.pg_last_xlog_receive_location()");
|
||||||
|
|
||||||
res = PQexec(my_local_conn, sqlquery);
|
res = PQexec(my_local_conn, sqlquery);
|
||||||
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
if (PQresultStatus(res) != PGRES_TUPLES_OK)
|
||||||
{
|
{
|
||||||
@@ -2327,13 +2399,13 @@ lsn_to_xlogrecptr(char *lsn, bool *format_ok)
|
|||||||
if (format_ok != NULL)
|
if (format_ok != NULL)
|
||||||
*format_ok = true;
|
*format_ok = true;
|
||||||
|
|
||||||
return (((XLogRecPtr) xlogid * 16 * 1024 * 1024 * 255) + xrecoff);
|
return (XLogRecPtr) ((uint64) xlogid) << 32 | (uint64) xrecoff;
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
usage(void)
|
usage(void)
|
||||||
{
|
{
|
||||||
log_err(_("%s: Replicator manager daemon \n"), progname());
|
log_err(_("%s: replication management daemon for PostgreSQL\n"), progname());
|
||||||
log_err(_("Try \"%s --help\" for more information.\n"), progname());
|
log_err(_("Try \"%s --help\" for more information.\n"), progname());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#
|
#
|
||||||
# Makefile
|
# Makefile
|
||||||
#
|
#
|
||||||
# Copyright (c) 2ndQuadrant, 2010-2016
|
# Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
#
|
#
|
||||||
|
|
||||||
MODULE_big = repmgr_funcs
|
MODULE_big = repmgr_funcs
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* repmgr_function.sql
|
* repmgr_function.sql
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
4
sql/repmgr_test.sql
Normal file
4
sql/repmgr_test.sql
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
select * from repmgr_update_standby_location('');
|
||||||
|
select * from repmgr_get_last_standby_location();
|
||||||
|
select * from repmgr_update_last_updated();
|
||||||
|
select * from repmgr_get_last_updated();
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uninstall_repmgr_funcs.sql
|
* uninstall_repmgr_funcs.sql
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* strutil.c
|
* strutil.c
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* strutil.h
|
* strutil.h
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* uninstall_repmgr.sql
|
* uninstall_repmgr.sql
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2016
|
* Copyright (c) 2ndQuadrant, 2010-2017
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user