mirror of
https://github.com/EnterpriseDB/repmgr.git
synced 2026-03-23 15:16:29 +00:00
Compare commits
1060 Commits
v1.1.0
...
REL3_1_STA
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3802b917e0 | ||
|
|
4f7a2a0614 | ||
|
|
06c7fe04b0 | ||
|
|
1fe01e9168 | ||
|
|
ed1136f443 | ||
|
|
a7ed60a533 | ||
|
|
fc5a18410d | ||
|
|
fd52c8ec3c | ||
|
|
47f1c6fa84 | ||
|
|
fba89ef37c | ||
|
|
4cc6cbe32f | ||
|
|
c715077c29 | ||
|
|
c178d8ed27 | ||
|
|
d4d06f43f7 | ||
|
|
0d346a9f54 | ||
|
|
abb16e4366 | ||
|
|
59b1924d5b | ||
|
|
c88ea62643 | ||
|
|
5b91a5e2e5 | ||
|
|
c2a1a35282 | ||
|
|
2b8b74ae75 | ||
|
|
08ef4d4be6 | ||
|
|
1a0049f086 | ||
|
|
af6f0fc2cf | ||
|
|
893d67473d | ||
|
|
a922cd5558 | ||
|
|
7bbc664230 | ||
|
|
a6998fe0f9 | ||
|
|
dadfdcc51f | ||
|
|
b8823d5c1f | ||
|
|
e59b57376d | ||
|
|
3db87e6a31 | ||
|
|
94d05619c3 | ||
|
|
807c7c926c | ||
|
|
df68f1f3f6 | ||
|
|
d4c75bb6c7 | ||
|
|
94d4e1128d | ||
|
|
dbd82ba687 | ||
|
|
0888fbc538 | ||
|
|
92a84bd950 | ||
|
|
a3318d65d2 | ||
|
|
374e9811c9 | ||
|
|
72f9b0145a | ||
|
|
5e03ef40cb | ||
|
|
091541619d | ||
|
|
5e9db47d12 | ||
|
|
e8a0cd33b5 | ||
|
|
8cd79fd7dd | ||
|
|
013b4b4b8a | ||
|
|
c5a721a3cf | ||
|
|
a6294b7da0 | ||
|
|
a0f02e454c | ||
|
|
69d9d137e0 | ||
|
|
60bceae905 | ||
|
|
746c9793ed | ||
|
|
c30447ac90 | ||
|
|
097024a32f | ||
|
|
66b7dbbed7 | ||
|
|
74f6f97f26 | ||
|
|
968c2f1954 | ||
|
|
bd76d0eb92 | ||
|
|
f1ee6e19b6 | ||
|
|
fbb65b4a43 | ||
|
|
3fac975de6 | ||
|
|
a2b5ba595a | ||
|
|
c16ab3c889 | ||
|
|
dd5b6f9f12 | ||
|
|
303bb22ee1 | ||
|
|
5d8b1a3a31 | ||
|
|
3d6c349d88 | ||
|
|
1ade1acb22 | ||
|
|
66fd003ab4 | ||
|
|
0d42b771f5 | ||
|
|
005640be51 | ||
|
|
b6ebd34e2f | ||
|
|
951879f80d | ||
|
|
46ff9fb587 | ||
|
|
cc610f995d | ||
|
|
384618cb33 | ||
|
|
0dd617cfca | ||
|
|
f18d629bd2 | ||
|
|
afc904f876 | ||
|
|
3bcea46c3b | ||
|
|
d7e85f7565 | ||
|
|
b14d8ddb74 | ||
|
|
9b2a907b09 | ||
|
|
f63d42fe77 | ||
|
|
560066fa9d | ||
|
|
3937670d14 | ||
|
|
0daa7381b3 | ||
|
|
e53545af4f | ||
|
|
45178c19d8 | ||
|
|
cf46834041 | ||
|
|
c30609426a | ||
|
|
1c49c4159c | ||
|
|
b6b6439819 | ||
|
|
16896510dc | ||
|
|
9a05999abb | ||
|
|
1c155a1088 | ||
|
|
4c463a66b7 | ||
|
|
31d57f4122 | ||
|
|
209de699ce | ||
|
|
7b313b9d71 | ||
|
|
cf126642bd | ||
|
|
52281fcde8 | ||
|
|
de573edaaa | ||
|
|
4cb7f301ad | ||
|
|
87d8de4441 | ||
|
|
6db742f81e | ||
|
|
c79933685c | ||
|
|
04ba672b9f | ||
|
|
4f4111063a | ||
|
|
3a3a536e6d | ||
|
|
6f7206a5a1 | ||
|
|
f9fd1dd227 | ||
|
|
8140ba9c27 | ||
|
|
32dba444e1 | ||
|
|
e814c1120e | ||
|
|
247823db4d | ||
|
|
beda22d5f9 | ||
|
|
2eb00a3e6f | ||
|
|
0a798bf6e4 | ||
|
|
21b2ff1a1f | ||
|
|
8212ff8d8a | ||
|
|
57f9432692 | ||
|
|
54d3c7a4ca | ||
|
|
7fd44a3d74 | ||
|
|
b0f6b7bad7 | ||
|
|
4dbbf40196 | ||
|
|
d5e24689a4 | ||
|
|
10e47441a2 | ||
|
|
1ccd0edad2 | ||
|
|
59b31dd1ca | ||
|
|
300b9f0cc2 | ||
|
|
274a30efa5 | ||
|
|
db63b5bb1c | ||
|
|
0efee4cf65 | ||
|
|
0cb2584886 | ||
|
|
b88d27248c | ||
|
|
683c54325e | ||
|
|
70d398cd47 | ||
|
|
7b7d80e5f2 | ||
|
|
e100728b93 | ||
|
|
d104f2a914 | ||
|
|
2946c097f0 | ||
|
|
a538ceb0ea | ||
|
|
5a2a8d1c82 | ||
|
|
b5a7efa58e | ||
|
|
96b0e26084 | ||
|
|
91c498f6f1 | ||
|
|
d48093e732 | ||
|
|
3f0d1754a4 | ||
|
|
f27979bbe1 | ||
|
|
9f6f58e4ed | ||
|
|
c22f4eaf6f | ||
|
|
925d82f7a4 | ||
|
|
1db577e294 | ||
|
|
a886fddccc | ||
|
|
83e5f98171 | ||
|
|
eb31a56186 | ||
|
|
8cd2c6fd05 | ||
|
|
e3e1c5de4e | ||
|
|
f9a150504a | ||
|
|
e9445a5d5e | ||
|
|
9a2717b5e3 | ||
|
|
5bc809466c | ||
|
|
dd6ea1cd77 | ||
|
|
5d32026b79 | ||
|
|
de5908c122 | ||
|
|
2a8d6f72c6 | ||
|
|
190cc7dcb4 | ||
|
|
4b5c84921c | ||
|
|
819937d4bd | ||
|
|
aaa8d70cef | ||
|
|
57299cb978 | ||
|
|
ca31b846e7 | ||
|
|
59f503835b | ||
|
|
33e626cd75 | ||
|
|
a27cecb559 | ||
|
|
491ec37adf | ||
|
|
cf0cdfa6a1 | ||
|
|
31489d92c0 | ||
|
|
c93790fc96 | ||
|
|
b7fd13aed2 | ||
|
|
ecabe2c294 | ||
|
|
2ba57e5938 | ||
|
|
3c4bf27aa7 | ||
|
|
0ebd9c15d9 | ||
|
|
f9dba283d4 | ||
|
|
205f1cebbb | ||
|
|
4d97c1ebf7 | ||
|
|
12c395e91f | ||
|
|
bd1e4f71d6 | ||
|
|
cb49071ea4 | ||
|
|
2eec17e25f | ||
|
|
c48c248c15 | ||
|
|
958e45f2b8 | ||
|
|
daafd70383 | ||
|
|
c828598bfb | ||
|
|
b55519c4a2 | ||
|
|
4cafd443e1 | ||
|
|
d400d7f9ac | ||
|
|
62bb3db1f8 | ||
|
|
5ad674edff | ||
|
|
ac09bad89c | ||
|
|
009d92fec8 | ||
|
|
b3d8a68a1d | ||
|
|
05b47cb2a8 | ||
|
|
dc542a1b7d | ||
|
|
6ce8058749 | ||
|
|
2edcac77f0 | ||
|
|
f740374392 | ||
|
|
d9961bbb17 | ||
|
|
e1b8982c14 | ||
|
|
2fe3b3c2a3 | ||
|
|
c6e1bc205a | ||
|
|
7241391ddc | ||
|
|
c8f449f178 | ||
|
|
49420c437f | ||
|
|
827ffef5f9 | ||
|
|
16296bb1c3 | ||
|
|
c9c18d6216 | ||
|
|
d21f506614 | ||
|
|
fbad18085e | ||
|
|
ca08b1c3bb | ||
|
|
3d95fab0ac | ||
|
|
12d6ce4629 | ||
|
|
dfb34ae7b6 | ||
|
|
98c4eb002a | ||
|
|
faed8a65f7 | ||
|
|
a81cf04614 | ||
|
|
ca6cbcf965 | ||
|
|
16c1e13019 | ||
|
|
1375adcac8 | ||
|
|
e859a58405 | ||
|
|
1a6d830314 | ||
|
|
a96f478a43 | ||
|
|
8f20ab16dd | ||
|
|
3ec436f30d | ||
|
|
61e00bf1c7 | ||
|
|
5d71869fc1 | ||
|
|
7598e08b6f | ||
|
|
ba71e1eedf | ||
|
|
a4c07b23fb | ||
|
|
0c36f921f7 | ||
|
|
8ac5a5444e | ||
|
|
f60e7346e2 | ||
|
|
855ca8fe1a | ||
|
|
daa79d1a0f | ||
|
|
211768d911 | ||
|
|
f982708b35 | ||
|
|
995083d66c | ||
|
|
be58d6af96 | ||
|
|
a52e97e622 | ||
|
|
cc1ea00333 | ||
|
|
ec3596521f | ||
|
|
66245ccc03 | ||
|
|
c7542063be | ||
|
|
2633d994ef | ||
|
|
5359d45463 | ||
|
|
efa60d142c | ||
|
|
f3d0ab9ab9 | ||
|
|
7e6bac1be6 | ||
|
|
b72058dba8 | ||
|
|
79d1332f9c | ||
|
|
cde721e3fc | ||
|
|
7b2439b824 | ||
|
|
787cd94142 | ||
|
|
056e64f635 | ||
|
|
6b5a609d30 | ||
|
|
7a4d84379c | ||
|
|
490e12b1af | ||
|
|
7b9df3ac8f | ||
|
|
d6bf870316 | ||
|
|
b15e8debe1 | ||
|
|
310faf1bd9 | ||
|
|
35caeaa66a | ||
|
|
ba300c58f7 | ||
|
|
f2370de2fa | ||
|
|
3920deb803 | ||
|
|
e452bf6601 | ||
|
|
167b4efbb3 | ||
|
|
56b9ca7992 | ||
|
|
9c002c7e38 | ||
|
|
cfec04d19f | ||
|
|
4f1c67a1bf | ||
|
|
2f4fd2b7fa | ||
|
|
aca2b9547f | ||
|
|
c9db7f57d2 | ||
|
|
96ac39ba0f | ||
|
|
88a3378203 | ||
|
|
4db0efab47 | ||
|
|
864d57953a | ||
|
|
84d2a292b2 | ||
|
|
62d53b7622 | ||
|
|
77d52adb53 | ||
|
|
7a3e2f2a3a | ||
|
|
120688013e | ||
|
|
f6d1db5edb | ||
|
|
02729d299b | ||
|
|
88a6a1376e | ||
|
|
67df082ee9 | ||
|
|
9ed71d6317 | ||
|
|
933647d6de | ||
|
|
f99018b202 | ||
|
|
ced87373cd | ||
|
|
1db22546a9 | ||
|
|
7ae0df9c85 | ||
|
|
7a80f7a096 | ||
|
|
8710e067d0 | ||
|
|
793950eabd | ||
|
|
d1b4280182 | ||
|
|
64d038c823 | ||
|
|
46dd734b3d | ||
|
|
0a2e4466aa | ||
|
|
17ab86f7ac | ||
|
|
d433982af7 | ||
|
|
869b6a7a06 | ||
|
|
9018dc65de | ||
|
|
9cbd8df089 | ||
|
|
67a81d1d47 | ||
|
|
ab70007b75 | ||
|
|
0145aa0fc3 | ||
|
|
493c307b23 | ||
|
|
fc6225a511 | ||
|
|
e3111d37ba | ||
|
|
2a1a9f2e61 | ||
|
|
71a667ecb8 | ||
|
|
3ab91730c3 | ||
|
|
dd7f9b79ae | ||
|
|
8ab1901a93 | ||
|
|
e0cbdd5b31 | ||
|
|
d62aaeedd0 | ||
|
|
05cc7091b5 | ||
|
|
d192d5665c | ||
|
|
3848b9011b | ||
|
|
487aadc4b9 | ||
|
|
3f5920a395 | ||
|
|
617ea8cb78 | ||
|
|
142517fcca | ||
|
|
d722e2c74b | ||
|
|
abb02cab76 | ||
|
|
8e66e4811c | ||
|
|
ce5a541960 | ||
|
|
e12be52fa8 | ||
|
|
c0911d3286 | ||
|
|
6e94432282 | ||
|
|
29d9232e2f | ||
|
|
8973812144 | ||
|
|
e775a962ad | ||
|
|
12204f7e56 | ||
|
|
684f7590b7 | ||
|
|
9d589a780d | ||
|
|
83e6d15410 | ||
|
|
6a10fe0cd9 | ||
|
|
c664682c05 | ||
|
|
44acc8d719 | ||
|
|
b911483d5e | ||
|
|
ee9270fe8d | ||
|
|
d0a4eebeec | ||
|
|
0f5e71f029 | ||
|
|
dbd90d45f5 | ||
|
|
c8d0fb401f | ||
|
|
afda3419cc | ||
|
|
a86fa4ad4a | ||
|
|
7e3007f6e8 | ||
|
|
8c797a8fea | ||
|
|
56cec22f22 | ||
|
|
b61649a3e3 | ||
|
|
ded716e403 | ||
|
|
d639dc3342 | ||
|
|
17ed81ebb7 | ||
|
|
b00c507ee4 | ||
|
|
55d8b2ad9c | ||
|
|
c918aaad4a | ||
|
|
6e7eee4c01 | ||
|
|
5c59e8fc5b | ||
|
|
eba0b6bb1e | ||
|
|
3bc0b80a71 | ||
|
|
06b9e0a8ec | ||
|
|
120be2db1c | ||
|
|
12bd7da836 | ||
|
|
2fd905cf9e | ||
|
|
7a8a50e229 | ||
|
|
e188044593 | ||
|
|
636f4b03c6 | ||
|
|
bf96b383a3 | ||
|
|
3a2e40f381 | ||
|
|
c608bb28ee | ||
|
|
ca9c2e1143 | ||
|
|
3a6d6b8899 | ||
|
|
4091cb7f18 | ||
|
|
870b0a53b6 | ||
|
|
6184cc57be | ||
|
|
e1254b6773 | ||
|
|
1c9121c2d8 | ||
|
|
6da03a6157 | ||
|
|
9bb6befa25 | ||
|
|
a8e5c68d03 | ||
|
|
b83e18c503 | ||
|
|
d4b845d213 | ||
|
|
75aad9a85e | ||
|
|
e115825cd6 | ||
|
|
6cf5ab2e53 | ||
|
|
f8119d20ea | ||
|
|
0caddf2d2c | ||
|
|
a4abbc6f0c | ||
|
|
d7e489ea0a | ||
|
|
2bcacff3b3 | ||
|
|
45eb0ea5d3 | ||
|
|
c3bd02b83d | ||
|
|
8e7d110a22 | ||
|
|
43874d5576 | ||
|
|
87ff9d09ba | ||
|
|
c429b0b186 | ||
|
|
dd7ebdc1c7 | ||
|
|
03b88178c1 | ||
|
|
5f33f4286f | ||
|
|
932f84910b | ||
|
|
1ef7f1368d | ||
|
|
640abed18f | ||
|
|
ef6b24551a | ||
|
|
42847e44d2 | ||
|
|
dd7cfce3d3 | ||
|
|
30fd111cba | ||
|
|
65e63b062e | ||
|
|
1636805fa1 | ||
|
|
053f672caa | ||
|
|
f6d02b85d8 | ||
|
|
6ebf3a7319 | ||
|
|
7345ddcf00 | ||
|
|
eb0af7ca23 | ||
|
|
ae47e5f413 | ||
|
|
46100a9549 | ||
|
|
9bd95cabdf | ||
|
|
f1584469bf | ||
|
|
a7f46d24de | ||
|
|
462d446477 | ||
|
|
23a72f489c | ||
|
|
f3f56b0cd6 | ||
|
|
00146b7fbd | ||
|
|
faf72a2514 | ||
|
|
7010b636e0 | ||
|
|
00deff9069 | ||
|
|
5240a5723a | ||
|
|
899d789699 | ||
|
|
cd7a3215df | ||
|
|
45e29c5b28 | ||
|
|
f8fd344d9f | ||
|
|
5def293ed6 | ||
|
|
ff7b4d3f02 | ||
|
|
a54478a045 | ||
|
|
7ad9a2c28a | ||
|
|
3deb6784e7 | ||
|
|
ba275bb0c2 | ||
|
|
9735bb63a1 | ||
|
|
1e5792f8df | ||
|
|
a01fefa7d0 | ||
|
|
34eaf94b2b | ||
|
|
68e3a9d7ab | ||
|
|
2ad4f68700 | ||
|
|
00aa0c8c87 | ||
|
|
e8025c7c9f | ||
|
|
6a17360b4c | ||
|
|
9e5e843a4f | ||
|
|
734ae1825e | ||
|
|
41fe58764e | ||
|
|
58a5249b7e | ||
|
|
90c0bd4638 | ||
|
|
359e81a6d6 | ||
|
|
0037e66034 | ||
|
|
07d220cb00 | ||
|
|
4dfeffe087 | ||
|
|
18544c82ca | ||
|
|
0f86bdcd05 | ||
|
|
7d33c1e411 | ||
|
|
fec65bde3d | ||
|
|
4863ea98bc | ||
|
|
c4505248b0 | ||
|
|
5774d86ac7 | ||
|
|
29e7733152 | ||
|
|
b64385bee8 | ||
|
|
419ce28d3d | ||
|
|
90cc6a5cb4 | ||
|
|
1b175ddfcf | ||
|
|
eabe618738 | ||
|
|
3e621f43d1 | ||
|
|
15a531fed8 | ||
|
|
96255b988a | ||
|
|
8de0deddf9 | ||
|
|
bd19a2c868 | ||
|
|
2cadb3424d | ||
|
|
bfe4585b91 | ||
|
|
230773d626 | ||
|
|
81b7b3bae7 | ||
|
|
8cfc26d3ea | ||
|
|
5b1a4d0ef1 | ||
|
|
22423aa51a | ||
|
|
98df2a5891 | ||
|
|
6f61c8285b | ||
|
|
0875b2aafa | ||
|
|
3e2c9ed410 | ||
|
|
5b4f832f3b | ||
|
|
66844d057a | ||
|
|
f096cca84f | ||
|
|
0fbb83262f | ||
|
|
a0a3ef58b0 | ||
|
|
b007fc8b39 | ||
|
|
ae19c9bd5d | ||
|
|
1b4a8917ca | ||
|
|
ce66a7c2d2 | ||
|
|
9a3196b671 | ||
|
|
46a3082055 | ||
|
|
ebabc68f8a | ||
|
|
c757985640 | ||
|
|
172a3d90cf | ||
|
|
86d24759a0 | ||
|
|
7bd54b5a70 | ||
|
|
3e04c8e720 | ||
|
|
7f98bb7aec | ||
|
|
9e2736be4c | ||
|
|
4b3966d6a7 | ||
|
|
5a1036cea2 | ||
|
|
e21448831d | ||
|
|
155f5075cb | ||
|
|
9cfd6680b3 | ||
|
|
874616f149 | ||
|
|
61ce18ebbe | ||
|
|
922dfd88e5 | ||
|
|
b41235b896 | ||
|
|
0307c51d4b | ||
|
|
6d608aea7f | ||
|
|
5d26e27b48 | ||
|
|
2fa2dfff95 | ||
|
|
9e5b3e0a2d | ||
|
|
e0b82393b0 | ||
|
|
5c64f09889 | ||
|
|
af7dee05a4 | ||
|
|
bdb8ee1a6f | ||
|
|
3799d089a1 | ||
|
|
d06bd0ddea | ||
|
|
7fed433df1 | ||
|
|
9517624297 | ||
|
|
25ea635689 | ||
|
|
0d971d9009 | ||
|
|
5bba37cebd | ||
|
|
886a9fd036 | ||
|
|
94d0d119f6 | ||
|
|
96c8cd4148 | ||
|
|
619f95d85c | ||
|
|
97ae6dbf57 | ||
|
|
2929ed9be0 | ||
|
|
33037dd7fb | ||
|
|
36db199882 | ||
|
|
728b71c700 | ||
|
|
97c9525479 | ||
|
|
bf957ac173 | ||
|
|
e358c20b84 | ||
|
|
aaf219a694 | ||
|
|
c69e4e93f2 | ||
|
|
8a4f4bbd83 | ||
|
|
36a9e17bd3 | ||
|
|
95ac7e889b | ||
|
|
0a5457efb2 | ||
|
|
aa67a4b7e9 | ||
|
|
f8a0e45f5b | ||
|
|
6b8f96b590 | ||
|
|
e01807ea20 | ||
|
|
080bb81aeb | ||
|
|
d8fe1ebf47 | ||
|
|
9b7cb5b0c0 | ||
|
|
2c69119eff | ||
|
|
cc1e285d90 | ||
|
|
4ee84f4f05 | ||
|
|
ad83f8f12e | ||
|
|
ce254ccde3 | ||
|
|
c833dd65f9 | ||
|
|
7e615c5e0d | ||
|
|
a0a2f87d32 | ||
|
|
dce16d794c | ||
|
|
60b14ab107 | ||
|
|
65afc42afa | ||
|
|
0bba5ad792 | ||
|
|
f3d5a4a7b0 | ||
|
|
6597a03559 | ||
|
|
78dad4fc52 | ||
|
|
b152cccd69 | ||
|
|
6e5d4e0235 | ||
|
|
5ade2a1f2d | ||
|
|
cd2f74a840 | ||
|
|
fb28ee6f1c | ||
|
|
c02d226d2f | ||
|
|
fe1bd5fb91 | ||
|
|
bc63099dcf | ||
|
|
ef5ef9c13f | ||
|
|
02646165be | ||
|
|
d08bd352c1 | ||
|
|
53b990c65d | ||
|
|
1979fe63ca | ||
|
|
b0ce1fc801 | ||
|
|
28c2652580 | ||
|
|
e3b8a8fc3a | ||
|
|
74d925475e | ||
|
|
bd96e0ca72 | ||
|
|
918ee3811f | ||
|
|
606d0afabc | ||
|
|
e16c3b2c9a | ||
|
|
4e6c250830 | ||
|
|
29110a6e11 | ||
|
|
abf92883a8 | ||
|
|
bdf1696b58 | ||
|
|
7ad001e0c5 | ||
|
|
fb8296644d | ||
|
|
e603498f43 | ||
|
|
d7365535ab | ||
|
|
a55587d75a | ||
|
|
5a9b6eea1f | ||
|
|
a572efe929 | ||
|
|
c6b0f77923 | ||
|
|
66b399b886 | ||
|
|
2339adba6c | ||
|
|
ff63954c68 | ||
|
|
1a9fbd623f | ||
|
|
491309f4ba | ||
|
|
072c2d70ff | ||
|
|
1f9b19f3ff | ||
|
|
04fe820aff | ||
|
|
0f8759d316 | ||
|
|
2b6415b339 | ||
|
|
defb1e819b | ||
|
|
7ae4434f6e | ||
|
|
e3b734f177 | ||
|
|
187a6b6d23 | ||
|
|
238fc589a1 | ||
|
|
63b9254be4 | ||
|
|
44196de2f4 | ||
|
|
46888de77f | ||
|
|
6b1f55ff1b | ||
|
|
3d3f082617 | ||
|
|
db5db06244 | ||
|
|
dd7193715c | ||
|
|
f8f81f4bf1 | ||
|
|
27a4780cc3 | ||
|
|
b79bce5821 | ||
|
|
a69176fc1b | ||
|
|
65dcce55e8 | ||
|
|
ff30df3d96 | ||
|
|
74e4b6aa3c | ||
|
|
22dd164cbb | ||
|
|
63c416bb76 | ||
|
|
e2c2f97307 | ||
|
|
41e9a370af | ||
|
|
2eb242553a | ||
|
|
e3ecd3cdc5 | ||
|
|
32611f5f04 | ||
|
|
ba254d2f06 | ||
|
|
69f842bbc2 | ||
|
|
99e1c33668 | ||
|
|
6791bd9ad3 | ||
|
|
3e6659dda0 | ||
|
|
ce498e4d1e | ||
|
|
0762b28679 | ||
|
|
5dff3017c8 | ||
|
|
b453c6a533 | ||
|
|
09c7ba95ac | ||
|
|
e213f3b99d | ||
|
|
eabb3cb1ab | ||
|
|
e886e72f14 | ||
|
|
1803a16c7e | ||
|
|
4f36b2c085 | ||
|
|
19aba38327 | ||
|
|
2d82ade170 | ||
|
|
21730899da | ||
|
|
94bc5bdf80 | ||
|
|
1369fe5c79 | ||
|
|
e71dd4945e | ||
|
|
49d1abf130 | ||
|
|
01a33198fd | ||
|
|
1d28a2d08e | ||
|
|
d4578e023e | ||
|
|
9ceef937eb | ||
|
|
f047ec9526 | ||
|
|
a3f0e89a05 | ||
|
|
3d9d0d98af | ||
|
|
49debcdf92 | ||
|
|
7a760c32ff | ||
|
|
b4b5e6cd30 | ||
|
|
2ece014952 | ||
|
|
01360c3d39 | ||
|
|
031a726f04 | ||
|
|
5c67d47881 | ||
|
|
f40b3ac48a | ||
|
|
109269f7fb | ||
|
|
36d94c88ac | ||
|
|
23ef305afb | ||
|
|
99dae5cdcb | ||
|
|
b552710767 | ||
|
|
7e4c26b8a0 | ||
|
|
f8639a7878 | ||
|
|
f2309bd0a9 | ||
|
|
061e72d7cd | ||
|
|
0a19bf1e23 | ||
|
|
84a4766f13 | ||
|
|
3be8bf8e4c | ||
|
|
1e6f1a88b0 | ||
|
|
4a8912c2b4 | ||
|
|
3279e9e47e | ||
|
|
5c4e77f8e2 | ||
|
|
b09f987341 | ||
|
|
fe758eda9f | ||
|
|
c413cff461 | ||
|
|
609453a848 | ||
|
|
a82d37e48a | ||
|
|
4b6c097a3e | ||
|
|
3dfa33d01d | ||
|
|
a20afe28ec | ||
|
|
692204e381 | ||
|
|
2ae27521a3 | ||
|
|
d141d2a8aa | ||
|
|
fc6f5ddaa7 | ||
|
|
4b7fb70de3 | ||
|
|
5db6489ae5 | ||
|
|
96f6695a52 | ||
|
|
f006a79af9 | ||
|
|
3a8aa86e0c | ||
|
|
9cbac64ad4 | ||
|
|
242fc8416c | ||
|
|
355d8b8f01 | ||
|
|
8df7966540 | ||
|
|
1a790549ea | ||
|
|
b80d25ad33 | ||
|
|
a1a61e5ec3 | ||
|
|
1221b73293 | ||
|
|
93d5d7905f | ||
|
|
41a5274b44 | ||
|
|
d9f83cf620 | ||
|
|
905a2d0a5e | ||
|
|
526803c5e0 | ||
|
|
2ff24fecf2 | ||
|
|
a5ead16d6f | ||
|
|
5fb84b9627 | ||
|
|
437485bf6a | ||
|
|
822867ec24 | ||
|
|
5b88a980b8 | ||
|
|
e958b8f2d7 | ||
|
|
74a963a10e | ||
|
|
dd064cb47c | ||
|
|
17de82493a | ||
|
|
f2fa60f5cf | ||
|
|
acf2744ed7 | ||
|
|
7bafd490c1 | ||
|
|
7c15176646 | ||
|
|
97cbd7d557 | ||
|
|
5798241205 | ||
|
|
5b5b4cd4cd | ||
|
|
9dd78f34db | ||
|
|
a5b5e3d384 | ||
|
|
241622694d | ||
|
|
687872e979 | ||
|
|
3033f2dfaf | ||
|
|
718024454e | ||
|
|
e3379c0fbf | ||
|
|
ba4413ce91 | ||
|
|
e85e0732ef | ||
|
|
97fb3dbb14 | ||
|
|
8e09e7b57c | ||
|
|
4f37515113 | ||
|
|
413cc6eb54 | ||
|
|
f23c43b986 | ||
|
|
38de150436 | ||
|
|
a80da61203 | ||
|
|
4fbe8ca70e | ||
|
|
1aa12a122f | ||
|
|
c54d5f9a9c | ||
|
|
db49206fc5 | ||
|
|
017f66e15e | ||
|
|
e06949db87 | ||
|
|
2f6ce44cab | ||
|
|
1f21040fb3 | ||
|
|
4e9c58c7db | ||
|
|
3e983b258c | ||
|
|
9093a9d1ce | ||
|
|
4305f76fa9 | ||
|
|
7700086703 | ||
|
|
8b69b1e16f | ||
|
|
f94626bf7b | ||
|
|
4c64d52afb | ||
|
|
6f80cd5441 | ||
|
|
ee69730f8b | ||
|
|
9306726e06 | ||
|
|
44fa98f828 | ||
|
|
28fd9d5aeb | ||
|
|
24b2ee09b3 | ||
|
|
8faf41dd94 | ||
|
|
763881b1ed | ||
|
|
94f520e1de | ||
|
|
f72bb2f093 | ||
|
|
927667905e | ||
|
|
684d66f48e | ||
|
|
406c325ccb | ||
|
|
c23ee3830b | ||
|
|
dc0dfe9b56 | ||
|
|
313d787ebf | ||
|
|
253992c5ec | ||
|
|
47b7c4ce06 | ||
|
|
99ed9a065e | ||
|
|
a87d859e56 | ||
|
|
7350a8bf57 | ||
|
|
75baed233b | ||
|
|
5b9ac4585b | ||
|
|
2cbee90f35 | ||
|
|
fa7d8df534 | ||
|
|
0cbd5d3933 | ||
|
|
a0e4c99ab4 | ||
|
|
98c5215871 | ||
|
|
e40b9db0a6 | ||
|
|
54e62c3d65 | ||
|
|
bfd482bebc | ||
|
|
6a0fc43086 | ||
|
|
8f47111072 | ||
|
|
0b5b3aaa4b | ||
|
|
d8bba0de03 | ||
|
|
a11bb60fd5 | ||
|
|
4f810aee2c | ||
|
|
73d352b2a2 | ||
|
|
143aa57bb8 | ||
|
|
5b15fcff5c | ||
|
|
4469de533e | ||
|
|
30045f3bec | ||
|
|
d93d42fadb | ||
|
|
7c89a4d762 | ||
|
|
b3c68dead8 | ||
|
|
b9ab9010c0 | ||
|
|
2a6c835a5a | ||
|
|
2d48d5aee4 | ||
|
|
653e11c2a7 | ||
|
|
91c29fe2a2 | ||
|
|
573f1d3b2e | ||
|
|
0a6ff7faec | ||
|
|
98b1f8d28a | ||
|
|
9eba986833 | ||
|
|
164cf9d08f | ||
|
|
d8b8bf0e2a | ||
|
|
fed5c77653 | ||
|
|
8429b43edf | ||
|
|
7e55ce737d | ||
|
|
98c7635fb5 | ||
|
|
90ecb2b107 | ||
|
|
50b9022a41 | ||
|
|
150ccc0662 | ||
|
|
0ff14a2aa1 | ||
|
|
5215265694 | ||
|
|
e45ac25348 | ||
|
|
a1ce01f033 | ||
|
|
516cde621a | ||
|
|
f0807923a3 | ||
|
|
10ca8037f8 | ||
|
|
0dc46f0dc8 | ||
|
|
c3b58658ad | ||
|
|
18f1fed77f | ||
|
|
d58fd080ca | ||
|
|
c4ac2d3343 | ||
|
|
5ff1beeea7 | ||
|
|
ca470647cb | ||
|
|
62ee287e3f | ||
|
|
729a1b848a | ||
|
|
701cf043fd | ||
|
|
bbb67c55f6 | ||
|
|
c2c48a9fe6 | ||
|
|
9d6ac2ebf9 | ||
|
|
680f23fb1d | ||
|
|
1159113c58 | ||
|
|
f25a709454 | ||
|
|
897daddcc7 | ||
|
|
0fdcce0477 | ||
|
|
de58eff7c1 | ||
|
|
f2a0b31a20 | ||
|
|
e007a55967 | ||
|
|
d235c696af | ||
|
|
4ef6fbb5fe | ||
|
|
2e61d7b156 | ||
|
|
4496a0761e | ||
|
|
3978ead184 | ||
|
|
b36dbf61fe | ||
|
|
84466ecca5 | ||
|
|
649086e5e4 | ||
|
|
7cf2eb440d | ||
|
|
388bbfb773 | ||
|
|
a89aa02c68 | ||
|
|
c81793b63f | ||
|
|
b4e83cf188 | ||
|
|
1db61ce277 | ||
|
|
41abf9a7ef | ||
|
|
abebc53ddc | ||
|
|
5fc4a0382f | ||
|
|
a7d3c9b93a | ||
|
|
ee9dc9e247 | ||
|
|
94cb5b94e7 | ||
|
|
a08aa50f92 | ||
|
|
9563877fbb | ||
|
|
4f3bd6612c | ||
|
|
192ee3cdb0 | ||
|
|
6f149ead8f | ||
|
|
77aa6aa326 | ||
|
|
18206b3a64 | ||
|
|
91446bcf93 | ||
|
|
dcdf8788ae | ||
|
|
4fabfbbbd0 | ||
|
|
c41030b40e | ||
|
|
a0fdadd5d2 | ||
|
|
4c3d7f80ed | ||
|
|
6e3fe059d8 | ||
|
|
9f26254ac3 | ||
|
|
0e8ff1730e | ||
|
|
634fdff303 | ||
|
|
cbce29f009 | ||
|
|
920f925e4b | ||
|
|
9fe2d6886e | ||
|
|
0068dd573a | ||
|
|
d0f3cb59c7 | ||
|
|
7428e92e10 | ||
|
|
a97065113d | ||
|
|
9e2f276fcf | ||
|
|
b0cd2b5e43 | ||
|
|
9209248420 | ||
|
|
6693b99288 | ||
|
|
8e7b487838 | ||
|
|
7f796e2d15 | ||
|
|
5e04ab6eae | ||
|
|
a1f4285e2b | ||
|
|
493133986d | ||
|
|
8b370dc581 | ||
|
|
43af00aa12 | ||
|
|
3c8df59eb9 | ||
|
|
b410772627 | ||
|
|
d99024ba11 | ||
|
|
1afaa3a26f | ||
|
|
079a7c9f16 | ||
|
|
3b66a31ac9 | ||
|
|
bdf957ca52 | ||
|
|
ad3630e7a9 | ||
|
|
67b451aa45 | ||
|
|
0a70d907ae | ||
|
|
2e7acf03c4 | ||
|
|
2bc8044fda | ||
|
|
ab1d380843 | ||
|
|
b0b44a157f | ||
|
|
49a2531930 | ||
|
|
672b237c4e | ||
|
|
7d94151494 | ||
|
|
4191b77e70 | ||
|
|
2a5d431481 | ||
|
|
81b8a944de | ||
|
|
93a999adc7 | ||
|
|
1b69282df9 | ||
|
|
06dd252f69 | ||
|
|
088ca29fe3 | ||
|
|
30e9d06172 | ||
|
|
d6bd5aa381 | ||
|
|
bbdcffa813 | ||
|
|
cd1a84252e | ||
|
|
5f33d9d715 | ||
|
|
2e19b3688b | ||
|
|
877f4cf82e | ||
|
|
de883a4c84 | ||
|
|
949f5ee498 | ||
|
|
eb2f7efb4a | ||
|
|
85ff3ec286 | ||
|
|
499a501afd | ||
|
|
0a9107d76d | ||
|
|
2803bb92a8 | ||
|
|
16fe41eecf | ||
|
|
95ec0450da | ||
|
|
57aa95f674 | ||
|
|
d365a309fc | ||
|
|
56d2ae4e81 | ||
|
|
d5a41bb587 | ||
|
|
474d3217b4 | ||
|
|
3edd87a041 | ||
|
|
740208da1c | ||
|
|
7a00d5a9a4 | ||
|
|
664e1a8321 | ||
|
|
5683b905dd | ||
|
|
d43c6334da | ||
|
|
f984b3fd33 | ||
|
|
aaf35947ed | ||
|
|
08ed0aa987 | ||
|
|
2d24518d9d | ||
|
|
a6c94b29de | ||
|
|
bf241ba1d6 | ||
|
|
41dbc39527 | ||
|
|
50b7147f15 | ||
|
|
f5e57aa433 | ||
|
|
ac5a9d1fd6 | ||
|
|
cb740b68be | ||
|
|
d58ea77798 | ||
|
|
e3c3c22b6e | ||
|
|
861a3c8f22 | ||
|
|
e51870b504 | ||
|
|
5651720560 | ||
|
|
d32a6cdb24 | ||
|
|
9e10987b90 | ||
|
|
64fce88e99 | ||
|
|
7a76f1998c | ||
|
|
4db046a8ea | ||
|
|
331eca447a | ||
|
|
b5b2f93f7e | ||
|
|
9d03d4a254 | ||
|
|
3b2ccc5b78 | ||
|
|
89a1e2bcbd | ||
|
|
7077a7c68f | ||
|
|
9b8fb7e960 | ||
|
|
55c7ea4b5e | ||
|
|
0240cb2bfb | ||
|
|
2e64028f26 | ||
|
|
27396f462a | ||
|
|
cb764b180d | ||
|
|
65cf045a63 | ||
|
|
3d53661d97 | ||
|
|
7b3a9cbbe4 | ||
|
|
276c947202 | ||
|
|
020e17b059 | ||
|
|
10817ecffc | ||
|
|
f26e37c087 | ||
|
|
61c31eeb0c | ||
|
|
0b5d2418df | ||
|
|
f44037088b | ||
|
|
e05dd729f8 | ||
|
|
ab8d2673dc | ||
|
|
28c5ac3d91 | ||
|
|
c983fdf83c | ||
|
|
158214acae | ||
|
|
04290c1f60 | ||
|
|
cedc5e20fb | ||
|
|
b61ac86b5f | ||
|
|
64da72b48b | ||
|
|
c4778a83c5 | ||
|
|
4d26e4d21e | ||
|
|
3f4ff5a862 | ||
|
|
1d05467df1 | ||
|
|
15ba3342aa | ||
|
|
3a950c9f8b | ||
|
|
43268f2345 | ||
|
|
8200b68cf7 | ||
|
|
06aecbf58a | ||
|
|
367d0b1afa | ||
|
|
e52f2e88a5 | ||
|
|
620f8090a3 | ||
|
|
534649eb03 | ||
|
|
1e8b9e8960 | ||
|
|
c4ae5741a3 | ||
|
|
f824189692 | ||
|
|
68b42d16c9 | ||
|
|
3ca0f2db06 | ||
|
|
235903e517 | ||
|
|
bf29601218 | ||
|
|
02a3fbb303 | ||
|
|
b30398bf2a | ||
|
|
e74e319e7a | ||
|
|
d77186c904 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,6 +1,9 @@
|
|||||||
*~
|
*~
|
||||||
*.o
|
*.o
|
||||||
|
*.so
|
||||||
repmgr
|
repmgr
|
||||||
repmgrd
|
repmgrd
|
||||||
README.htm*
|
README.htm*
|
||||||
README.pdf
|
README.pdf
|
||||||
|
sql/repmgr_funcs.so
|
||||||
|
sql/repmgr_funcs.sql
|
||||||
|
|||||||
29
CONTRIBUTING.md
Normal file
29
CONTRIBUTING.md
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
License and Contributions
|
||||||
|
=========================
|
||||||
|
|
||||||
|
`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
|
||||||
|
details.
|
||||||
|
|
||||||
|
The development of repmgr has primarily been sponsored by 2ndQuadrant customers.
|
||||||
|
|
||||||
|
Additional work has been sponsored by the 4CaaST project for cloud computing,
|
||||||
|
which has received funding from the European Union's Seventh Framework Programme
|
||||||
|
(FP7/2007-2013) under grant agreement 258862.
|
||||||
|
|
||||||
|
Contributions to `repmgr` are welcome, and will be listed in the file `CREDITS`.
|
||||||
|
2ndQuadrant Limited requires that any contributions provide a copyright
|
||||||
|
assignment and a disclaimer of any work-for-hire ownership claims from the
|
||||||
|
employer of the developer. This lets us make sure that all of the repmgr
|
||||||
|
distribution remains free code. Please contact info@2ndQuadrant.com for a
|
||||||
|
copy of the relevant Copyright Assignment Form.
|
||||||
|
|
||||||
|
Code style
|
||||||
|
----------
|
||||||
|
|
||||||
|
Code in repmgr is formatted to a consistent style using the following command:
|
||||||
|
|
||||||
|
astyle --style=ansi --indent=tab --suffix=none *.c *.h
|
||||||
|
|
||||||
|
Contributors should reformat their code similarly before submitting code to
|
||||||
|
the project, in order to minimize merge conflicts with other work.
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
Copyright (c) 2010-2011, 2ndQuadrant Limited
|
Copyright (c) 2010-2016, 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
|
||||||
|
|||||||
4
CREDITS
4
CREDITS
@@ -10,3 +10,7 @@ Hannu Krosing <hannu@2ndQuadrant.com>
|
|||||||
Cédric Villemain <cedric@2ndquadrant.com>
|
Cédric Villemain <cedric@2ndquadrant.com>
|
||||||
Charles Duffy <charles@dyfis.net>
|
Charles Duffy <charles@dyfis.net>
|
||||||
Daniel Farina <daniel@heroku.com>
|
Daniel Farina <daniel@heroku.com>
|
||||||
|
Shawn Ellis <shawn.ellis17@gmail.com>
|
||||||
|
Jay Taylor <jay@jaytaylor.com>
|
||||||
|
Christian Kruse <christian@2ndQuadrant.com>
|
||||||
|
Krzysztof Gajdemski <songo@debian.org.pl>
|
||||||
|
|||||||
1
FAILOVER.rst
Normal file
1
FAILOVER.rst
Normal file
@@ -0,0 +1 @@
|
|||||||
|
The contents of this file have been incorporated into the main README.md document.
|
||||||
171
FAQ.md
Normal file
171
FAQ.md
Normal file
@@ -0,0 +1,171 @@
|
|||||||
|
FAQ - Frequently Asked Questions about repmgr
|
||||||
|
=============================================
|
||||||
|
|
||||||
|
This FAQ applies to `repmgr` 3.0 and later.
|
||||||
|
|
||||||
|
General
|
||||||
|
-------
|
||||||
|
|
||||||
|
- What's the difference between the repmgr versions?
|
||||||
|
|
||||||
|
repmgr 3.x builds on the improved replication facilities added
|
||||||
|
in PostgreSQL 9.3, as well as improved automated failover support
|
||||||
|
via `repmgrd`, and is not compatible with PostgreSQL 9.2 and earlier.
|
||||||
|
|
||||||
|
repmgr 2.x supports PostgreSQL 9.0 onwards. While it is compatible
|
||||||
|
with PostgreSQL 9.3 and later, we recommend repmgr v3.
|
||||||
|
|
||||||
|
- What's the advantage of using replication slots?
|
||||||
|
|
||||||
|
Replication slots, introduced in PostgreSQL 9.4, ensure that the
|
||||||
|
master server will retain WAL files until they have been consumed
|
||||||
|
by all standby servers. This makes WAL file management much easier,
|
||||||
|
and if used `repmgr` will no longer insist on a fixed number (default: 5000)
|
||||||
|
of WAL files being preserved.
|
||||||
|
|
||||||
|
(However this does mean that if a standby is no longer connected to the
|
||||||
|
master, the master will retain WAL files indefinitely).
|
||||||
|
|
||||||
|
- How many replication slots should I define in `max_replication_slots`?
|
||||||
|
|
||||||
|
Normally at least same number as the number of standbys which will connect
|
||||||
|
to the node. Note that changes to `max_replication_slots` require a server
|
||||||
|
restart to take effect, and as there is no particular penalty for unused
|
||||||
|
replication slots, setting a higher figure will make adding new nodes
|
||||||
|
easier.
|
||||||
|
|
||||||
|
- Does `repmgr` support hash indexes?
|
||||||
|
|
||||||
|
No. Hash indexes and replication do not mix well and their use is
|
||||||
|
explicitly discouraged; see:
|
||||||
|
https://www.postgresql.org/docs/current/interactive/sql-createindex.html#AEN74175
|
||||||
|
|
||||||
|
`repmgr`
|
||||||
|
--------
|
||||||
|
|
||||||
|
- When should I use the --rsync-only option?
|
||||||
|
|
||||||
|
By default, `repmgr` uses `pg_basebackup` to clone a standby from
|
||||||
|
a master. However, `pg_basebackup` copies the entire data directory, which
|
||||||
|
can take some time depending on installation size. If you have an
|
||||||
|
existing but "stale" standby, `repmgr` can use `rsync` instead,
|
||||||
|
which means only changed or added files need to be copied.
|
||||||
|
|
||||||
|
- Can I register an existing master/standby?
|
||||||
|
|
||||||
|
Yes, this is no problem.
|
||||||
|
|
||||||
|
- How can a failed master be re-added as a standby?
|
||||||
|
|
||||||
|
This is a two-stage process. First, the failed master's data directory
|
||||||
|
must be re-synced with the current master; secondly the failed master
|
||||||
|
needs to be re-registered as a standby. The section "Converting a failed
|
||||||
|
master to a standby" in the `README.md` file contains more detailed
|
||||||
|
information on this process.
|
||||||
|
|
||||||
|
- Is there an easy way to check my master server is correctly configured
|
||||||
|
for use with `repmgr`?
|
||||||
|
|
||||||
|
Yes - execute `repmgr` with the `--check-upstream-config` option, and it
|
||||||
|
will let you know which items in `postgresql.conf` need to be modified.
|
||||||
|
|
||||||
|
- Even though I specified custom `rsync` options, `repmgr` appends
|
||||||
|
the `--checksum` - why?
|
||||||
|
|
||||||
|
When syncing a stale data directory from an active server, it's
|
||||||
|
essential that `rsync` compares the content of files rather than
|
||||||
|
just timestamp and size, to ensure that all changed files are
|
||||||
|
copied and prevent corruption.
|
||||||
|
|
||||||
|
- When cloning a standby, how can I prevent `repmgr` from copying
|
||||||
|
`postgresql.conf` and `pg_hba.conf` from the PostgreSQL configuration
|
||||||
|
directory in `/etc`?
|
||||||
|
|
||||||
|
Use the command line option `--ignore-external-config-files`
|
||||||
|
|
||||||
|
- How can I prevent `repmgr` from copying local configuration files
|
||||||
|
in the data directory?
|
||||||
|
|
||||||
|
If you're updating an existing but stale data directory which
|
||||||
|
contains e.g. configuration files you don't want to be overwritten
|
||||||
|
with the same file from the master, specify the files in the
|
||||||
|
`rsync_options` configuration option, e.g.
|
||||||
|
|
||||||
|
rsync_options=--exclude=postgresql.local.conf
|
||||||
|
|
||||||
|
This option is only available when using the `--rsync-only` option.
|
||||||
|
|
||||||
|
- How can I make the witness server use a particular port?
|
||||||
|
|
||||||
|
By default the witness server is configured to use port 5499; this
|
||||||
|
is intended to support running the witness server as a separate
|
||||||
|
instance on a normal node server, rather than on its own dedicated server.
|
||||||
|
|
||||||
|
To specify different port for the witness server, supply the port number
|
||||||
|
in the `conninfo` string in `repmgr.conf`
|
||||||
|
(repmgr 3.0.1 and earlier: use the `-l/--local-port` option)
|
||||||
|
|
||||||
|
- Do I need to include `shared_preload_libraries = 'repmgr_funcs'`
|
||||||
|
in `postgresql.conf` if I'm not using `repmgrd`?
|
||||||
|
|
||||||
|
No, the `repmgr_funcs` library is only needed when running `repmgrd`.
|
||||||
|
If you later decide to run `repmgrd`, you just need to add
|
||||||
|
`shared_preload_libraries = 'repmgr_funcs'` and restart PostgreSQL.
|
||||||
|
|
||||||
|
- I've provided replication permission for the `repmgr` user in `pg_hba.conf`
|
||||||
|
but `repmgr`/`repmgrd` complains it can't connect to the server... Why?
|
||||||
|
|
||||||
|
`repmgr`/`repmgrd` need to be able to connect to the repmgr database
|
||||||
|
with a normal connection to query metadata. The `replication` connection
|
||||||
|
permission is for PostgreSQL's streaming replication and doesn't
|
||||||
|
necessarily need to be the `repmgr` user.
|
||||||
|
|
||||||
|
- When cloning a standby, why do I need to provide the connection parameters
|
||||||
|
for the primary server on the command line, not in the configuration file?
|
||||||
|
|
||||||
|
Cloning a standby is a one-time action; the role of the server being cloned
|
||||||
|
from could change, so fixing it in the configuration file would create
|
||||||
|
confusion. If `repmgr` needs to establish a connection to the primary
|
||||||
|
server, it can retrieve this from the `repl_nodes` table or if necessary
|
||||||
|
scan the replication cluster until it locates the active primary.
|
||||||
|
|
||||||
|
- Why is there no foreign key on the `node_id` column in the `repl_events`
|
||||||
|
table?
|
||||||
|
|
||||||
|
Under some circumstances event notifications can be generated for servers
|
||||||
|
which have not yet been registered; it's also useful to retain a record
|
||||||
|
of events which includes servers removed from the replication cluster
|
||||||
|
which no longer have an entry in the `repl_nodes` table.
|
||||||
|
|
||||||
|
`repmgrd`
|
||||||
|
---------
|
||||||
|
|
||||||
|
- Do I need a witness server?
|
||||||
|
|
||||||
|
Not necessarily. However if you have an uneven number of nodes spread
|
||||||
|
over more than one network segment, a witness server will enable
|
||||||
|
better handling of a 'split brain' situation by providing a "casting
|
||||||
|
vote" on the preferred network segment.
|
||||||
|
|
||||||
|
- How can I prevent a node from ever being promoted to master?
|
||||||
|
|
||||||
|
In `repmgr.conf`, set its priority to a value of 0 or less.
|
||||||
|
|
||||||
|
- Does `repmgrd` support delayed standbys?
|
||||||
|
|
||||||
|
`repmgrd` can monitor delayed standbys - those set up with
|
||||||
|
`recovery_min_apply_delay` set to a non-zero value in `recovery.conf` -
|
||||||
|
but as it's not currently possible to directly examine the value
|
||||||
|
applied to the standby, `repmgrd` may not be able to properly evaluate
|
||||||
|
the node as a promotion candidate.
|
||||||
|
|
||||||
|
We recommend that delayed standbys are explicitly excluded from promotion
|
||||||
|
by setting `priority` to 0 in `repmgr.conf`.
|
||||||
|
|
||||||
|
Note that after registering a delayed standby, `repmgrd` will only start
|
||||||
|
once the metadata added in the master node has been replicated.
|
||||||
|
|
||||||
|
- How can I get `repmgrd` to rotate its logfile?
|
||||||
|
|
||||||
|
Configure your system's `logrotate` service to do this; see example
|
||||||
|
in README.md
|
||||||
180
HISTORY
180
HISTORY
@@ -1,5 +1,176 @@
|
|||||||
1.0.0 2010-12-05
|
3.1.5 2016-08-15
|
||||||
First public release
|
repmgrd: in a failover situation, prevent endless looping when
|
||||||
|
attempting to establish the status of a node with
|
||||||
|
`failover=manual` (Ian)
|
||||||
|
repmgrd: improve handling of failover events on standbys with
|
||||||
|
`failover=manual`, and create a new event notification
|
||||||
|
for this, `standby_disconnect_manual` (Ian)
|
||||||
|
repmgr: add further event notifications (Gianni)
|
||||||
|
repmgr: when executing `standby switchover`, don't collect remote
|
||||||
|
command output unless required (Gianni, Ian)
|
||||||
|
repmgrd: improve standby monitoring query (Ian, based on suggestion
|
||||||
|
from Álvaro)
|
||||||
|
repmgr: various command line handling improvements (Ian)
|
||||||
|
|
||||||
|
3.1.4 2016-07-12
|
||||||
|
repmgr: new configuration option for setting "restore_command"
|
||||||
|
in the recovery.conf file generated by repmgr (Martín)
|
||||||
|
repmgr: add --csv option to "repmgr cluster show" (Gianni)
|
||||||
|
repmgr: enable provision of a conninfo string as the -d/--dbname
|
||||||
|
parameter, similar to other PostgreSQL utilities (Ian)
|
||||||
|
repmgr: during switchover operations improve detection of
|
||||||
|
demotion candidate shutdown (Ian)
|
||||||
|
various bugfixes and documentation updates (Ian, Martín)
|
||||||
|
|
||||||
|
3.1.3 2016-05-17
|
||||||
|
repmgrd: enable monitoring when a standby is catching up by
|
||||||
|
replaying archived WAL (Ian)
|
||||||
|
repmgrd: when upstream_node_id is NULL, assume upstream node
|
||||||
|
to be current master (Ian)
|
||||||
|
repmgrd: check for reappearance of the master node if standby
|
||||||
|
promotion fails (Ian)
|
||||||
|
improve handling of rsync failure conditions (Martín)
|
||||||
|
|
||||||
|
3.1.2 2016-04-12
|
||||||
|
Fix pg_ctl path generation in do_standby_switchover() (Ian)
|
||||||
|
Regularly sync witness server repl_nodes table (Ian)
|
||||||
|
Documentation improvements (Gianni, dhyannataraj)
|
||||||
|
(Experimental) ensure repmgr handles failover slots when copying
|
||||||
|
in rsync mode (Craig, Ian)
|
||||||
|
rsync mode handling fixes (Martín)
|
||||||
|
Enable repmgr to compile against 9.6devel (Ian)
|
||||||
|
|
||||||
|
3.1.1 2016-02-24
|
||||||
|
Add '-P/--pwprompt' option for "repmgr create witness" (Ian)
|
||||||
|
Prevent repmgr/repmgrd running as root (Ian)
|
||||||
|
|
||||||
|
3.1.0 2016-02-01
|
||||||
|
Add "repmgr standby switchover" command (Ian)
|
||||||
|
Revised README file (Ian)
|
||||||
|
Remove requirement for 'archive_mode' to be enabled (Ian)
|
||||||
|
Improve -?/--help output, showing default values if relevant (Ian)
|
||||||
|
Various bugfixes to command line/configuration parameter handling (Ian)
|
||||||
|
|
||||||
|
3.0.3 2016-01-04
|
||||||
|
Create replication slot if required before base backup is run (Abhijit)
|
||||||
|
standy clone: when using rsync, clean up "pg_replslot" directory (Ian)
|
||||||
|
Improve --help output (Ian)
|
||||||
|
Improve config file parsing (Ian)
|
||||||
|
Various logging output improvements, including explicit HINTS (Ian)
|
||||||
|
Add --log-level to explicitly set log level on command line (Ian)
|
||||||
|
Repurpose --verbose to display extra log output (Ian)
|
||||||
|
Add --terse to hide hints and other non-critical output (Ian)
|
||||||
|
Reference internal functions with explicit catalog path (Ian)
|
||||||
|
When following a new primary, have repmgr (not repmgrd) create the new slot (Ian)
|
||||||
|
Add /etc/repmgr.conf as a default configuration file location (Ian)
|
||||||
|
Prevent repmgrd's -v/--verbose option expecting a parameter (Ian)
|
||||||
|
Prevent invalid replication_lag values being written to the monitoring table (Ian)
|
||||||
|
Improve repmgrd behaviour when monitored standby node is temporarily
|
||||||
|
unavailable (Martín)
|
||||||
|
|
||||||
|
3.0.2 2015-10-02
|
||||||
|
Improve handling of --help/--version options; and improve help output (Ian)
|
||||||
|
Improve handling of situation where logfile can't be opened (Ian)
|
||||||
|
Always pass -D/--pgdata option to pg_basebackup (Ian)
|
||||||
|
Bugfix: standby clone --force does not empty pg_xlog (Gianni)
|
||||||
|
Bugfix: autofailover with reconnect_attempts > 1 (Gianni)
|
||||||
|
Bugfix: ignore comments after values (soxwellfb)
|
||||||
|
Bugfix: handle string values in 'node' parameter correctly (Gregory Duchatelet)
|
||||||
|
Allow repmgr to be compiled with a newer libpq (Marco)
|
||||||
|
Bugfix: call update_node_record_set_upstream() for STANDBY FOLLOW (Tomas)
|
||||||
|
Update `repmgr --help` output (per Github report from renard)
|
||||||
|
Update tablespace remapping in --rsync-only mode for 9.5 and later (Ian)
|
||||||
|
Deprecate `-l/--local-port` option - the port can be extracted
|
||||||
|
from the conninfo string in repmgr.conf (Ian)
|
||||||
|
Add STANDBY UNREGISTER (Vik Fearing)
|
||||||
|
Don't fail with error when registering master if schema already defined (Ian)
|
||||||
|
Fixes to whitespace handling when parsing config file (Ian)
|
||||||
|
|
||||||
|
3.0.1 2015-04-16
|
||||||
|
Prevent repmgrd from looping infinitely if node was not registered (Ian)
|
||||||
|
When promoting a standby, have repmgr (not repmgrd) handle metadata updates (Ian)
|
||||||
|
Re-use replication slot if it already exists (Ian)
|
||||||
|
Prevent a test SSH connection being made when not needed (Ian)
|
||||||
|
Correct monitoring table column names (Ian)
|
||||||
|
|
||||||
|
3.0 2015-03-27
|
||||||
|
Require PostgreSQL 9.3 or later (Ian)
|
||||||
|
Use `pg_basebackup` by default (instead of `rsync`) to clone standby servers (Ian)
|
||||||
|
Use `pg_ctl promote` to promote a standby to primary
|
||||||
|
Enable tablespace remapping using `pg_basebackup` (in PostgreSQL 9.3 with `rsync`) (Ian)
|
||||||
|
Support cascaded standbys (Ian)
|
||||||
|
"pg_bindir" no longer required as a configuration parameter (Ian)
|
||||||
|
Enable replication slots to be used (PostgreSQL 9.4 and later (Ian)
|
||||||
|
Command line option "--check-upstream-config" (Ian)
|
||||||
|
Add event logging table and option to execute an external program when an event occurs (Ian)
|
||||||
|
General usability and logging message improvements (Ian)
|
||||||
|
Code consolidation and cleanup (Ian)
|
||||||
|
|
||||||
|
2.0.3 2015-04-16
|
||||||
|
Add -S/--superuser option for witness database creation Ian)
|
||||||
|
Add -c/--fast-checkpoint option for cloning (Christoph)
|
||||||
|
Add option "--initdb-no-pwprompt" (Ian)
|
||||||
|
|
||||||
|
2.0.2 2015-02-17
|
||||||
|
Add "--checksum" in rsync when using "--force" (Jaime)
|
||||||
|
Use createdb/createuser instead of psql (Jaime)
|
||||||
|
Fixes to witness creation and monitoring (wamonite)
|
||||||
|
Use default master port if none supplied (Martín)
|
||||||
|
Documentation fixes and improvements (Ian)
|
||||||
|
|
||||||
|
2.0.1 2014-07-16
|
||||||
|
Documentation fixes and new QUICKSTART file (Ian)
|
||||||
|
Explicitly specify directories to ignore when cloning (Ian)
|
||||||
|
Fix log level for some log messages (Ian)
|
||||||
|
RHEL/CentOS specfile, init script and Makefile fixes (Nathan Van Overloop)
|
||||||
|
Debian init script and config file documentation fixes (József Kószó)
|
||||||
|
Typo fixes (Riegie Godwin Jeyaranchen, PriceChild)
|
||||||
|
|
||||||
|
2.0stable 2014-01-30
|
||||||
|
Documentation fixes (Christian)
|
||||||
|
General refactoring, code quality improvements and stabilization work (Christian)
|
||||||
|
Added proper daemonizing (-d/--daemonize) (Christian)
|
||||||
|
Added PID file handling (-p/--pid-file) (Christian)
|
||||||
|
New config option: monitor_interval_secs (Christian)
|
||||||
|
New config option: retry_promote_interval (Christian)
|
||||||
|
New config option: logfile (Christian)
|
||||||
|
New config option: pg_bindir (Christian)
|
||||||
|
New config option: pgctl_options (Christian)
|
||||||
|
|
||||||
|
2.0beta2 2013-12-19
|
||||||
|
Improve autofailover logic and algorithms (Jaime, Andres)
|
||||||
|
Ignore pg_log when cloning (Jaime)
|
||||||
|
Add timestamps to log line in stderr (Christian)
|
||||||
|
Correctly check wal_keep_segments (Jay Taylor)
|
||||||
|
Add a ssh_options parameter (Jay Taylor)
|
||||||
|
|
||||||
|
2.0beta1 2012-07-27
|
||||||
|
Make CLONE command try to make an exact copy including $PGDATA location (Cedric)
|
||||||
|
Add detection of master failure (Jaime)
|
||||||
|
Add the notion of a witness server (Jaime)
|
||||||
|
Add autofailover capabilities (Jaime)
|
||||||
|
Add a configuration parameter to indicate the script to execute on failover or follow (Jaime)
|
||||||
|
Make the monitoring optional and turned off by default, it can be turned on with --monitoring-history switch (Jaime)
|
||||||
|
Add tunables to specify number of retries to reconnect to master and the time between them (Jaime)
|
||||||
|
|
||||||
|
1.2.0 2012-07-27
|
||||||
|
Test ssh connection before trying to rsync (Cédric)
|
||||||
|
Add CLUSTER SHOW command (Carlo)
|
||||||
|
Add CLUSTER CLEANUP command (Jaime)
|
||||||
|
Add function write_primary_conninfo (Marco)
|
||||||
|
Teach repmgr how to get tablespace's location in different pg version (Jaime)
|
||||||
|
Improve version message (Carlo)
|
||||||
|
|
||||||
|
1.1.1 2012-04-18
|
||||||
|
Add --ignore-rsync-warning (Cédric)
|
||||||
|
Add strnlen for compatibility with OS X (Greg)
|
||||||
|
Improve performance of the repl_status view (Jaime)
|
||||||
|
Remove last argument from log_err (Jaime, Reported by Jeroen Dekkers)
|
||||||
|
Complete documentation about possible error conditions (Jaime)
|
||||||
|
Document how to clean history (Jaime)
|
||||||
|
|
||||||
|
1.1.0 2011-03-09
|
||||||
|
Make options -U, -R and -p not mandatory (Jaime)
|
||||||
|
|
||||||
1.1.0b1 2011-02-24
|
1.1.0b1 2011-02-24
|
||||||
Fix missing "--force" option in help (Greg Smith)
|
Fix missing "--force" option in help (Greg Smith)
|
||||||
@@ -28,6 +199,5 @@
|
|||||||
Map old verbose flag into a useful setting for the new logger (Greg)
|
Map old verbose flag into a useful setting for the new logger (Greg)
|
||||||
Document repmgrd startup restrictions and log info about them (Greg)
|
Document repmgrd startup restrictions and log info about them (Greg)
|
||||||
|
|
||||||
1.1.0 2011-03-09
|
1.0.0 2010-12-05
|
||||||
Make options -U, -R and -p not mandatory (Jaime)
|
First public release
|
||||||
|
|
||||||
|
|||||||
74
Makefile
74
Makefile
@@ -1,25 +1,37 @@
|
|||||||
#
|
#
|
||||||
# Makefile
|
# Makefile
|
||||||
# Copyright (c) 2ndQuadrant, 2010-2011
|
# Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
|
|
||||||
|
HEADERS = $(wildcard *.h)
|
||||||
|
|
||||||
repmgrd_OBJS = dbutils.o config.o repmgrd.o log.o strutil.o
|
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
|
repmgr_OBJS = dbutils.o check_dir.o config.o repmgr.o log.o strutil.o dirmod.o
|
||||||
|
|
||||||
DATA = repmgr.sql uninstall_repmgr.sql
|
DATA = repmgr.sql uninstall_repmgr.sql
|
||||||
|
|
||||||
PG_CPPFLAGS = -I$(libpq_srcdir)
|
PG_CPPFLAGS = -I$(libpq_srcdir)
|
||||||
PG_LIBS = $(libpq_pgport)
|
PG_LIBS = $(libpq_pgport)
|
||||||
|
|
||||||
all: repmgrd repmgr
|
|
||||||
|
all: repmgrd repmgr
|
||||||
|
$(MAKE) -C sql
|
||||||
|
|
||||||
repmgrd: $(repmgrd_OBJS)
|
repmgrd: $(repmgrd_OBJS)
|
||||||
$(CC) $(CFLAGS) $(repmgrd_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS) -o repmgrd
|
$(CC) -o repmgrd $(CFLAGS) $(repmgrd_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS)
|
||||||
|
$(MAKE) -C sql
|
||||||
|
|
||||||
repmgr: $(repmgr_OBJS)
|
repmgr: $(repmgr_OBJS)
|
||||||
$(CC) $(CFLAGS) $(repmgr_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS) -o repmgr
|
$(CC) -o repmgr $(CFLAGS) $(repmgr_OBJS) $(PG_LIBS) $(LDFLAGS) $(LDFLAGS_EX) $(LIBS)
|
||||||
|
|
||||||
|
# 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
|
||||||
|
# is very fast anyway.
|
||||||
|
$(repmgr_OBJS): $(HEADERS)
|
||||||
|
$(repmgrd_OBJS): $(HEADERS)
|
||||||
|
|
||||||
ifdef USE_PGXS
|
ifdef USE_PGXS
|
||||||
PGXS := $(shell pg_config --pgxs)
|
PG_CONFIG = pg_config
|
||||||
|
PGXS := $(shell $(PG_CONFIG) --pgxs)
|
||||||
include $(PGXS)
|
include $(PGXS)
|
||||||
else
|
else
|
||||||
subdir = contrib/repmgr
|
subdir = contrib/repmgr
|
||||||
@@ -28,11 +40,34 @@ include $(top_builddir)/src/Makefile.global
|
|||||||
include $(top_srcdir)/contrib/contrib-global.mk
|
include $(top_srcdir)/contrib/contrib-global.mk
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# XXX: Try to use PROGRAM construct (see pgxs.mk) someday. Right now
|
# XXX: This overrides the pgxs install target - we're building two binaries,
|
||||||
# is overriding pgxs install.
|
# which is not supported by pgxs.mk's PROGRAM construct.
|
||||||
install:
|
install: install_prog install_ext
|
||||||
$(INSTALL_PROGRAM) repmgrd$(X) '$(DESTDIR)$(bindir)'
|
|
||||||
$(INSTALL_PROGRAM) repmgr$(X) '$(DESTDIR)$(bindir)'
|
install_prog:
|
||||||
|
mkdir -p '$(DESTDIR)$(bindir)'
|
||||||
|
$(INSTALL_PROGRAM) repmgrd$(X) '$(DESTDIR)$(bindir)/'
|
||||||
|
$(INSTALL_PROGRAM) repmgr$(X) '$(DESTDIR)$(bindir)/'
|
||||||
|
|
||||||
|
install_ext:
|
||||||
|
$(MAKE) -C sql install
|
||||||
|
|
||||||
|
# Distribution-specific package building targets
|
||||||
|
# ----------------------------------------------
|
||||||
|
#
|
||||||
|
# XXX we recommend using the PGDG-supplied packages where possible;
|
||||||
|
# see README.md for details.
|
||||||
|
|
||||||
|
install_rhel:
|
||||||
|
mkdir -p '$(DESTDIR)/etc/init.d/'
|
||||||
|
$(INSTALL_PROGRAM) RHEL/repmgrd.init '$(DESTDIR)/etc/init.d/repmgrd'
|
||||||
|
mkdir -p '$(DESTDIR)/etc/sysconfig/'
|
||||||
|
$(INSTALL_PROGRAM) RHEL/repmgrd.sysconfig '$(DESTDIR)/etc/sysconfig/repmgrd'
|
||||||
|
mkdir -p '$(DESTDIR)/etc/repmgr/'
|
||||||
|
$(INSTALL_PROGRAM) repmgr.conf.sample '$(DESTDIR)/etc/repmgr/'
|
||||||
|
mkdir -p '$(DESTDIR)/usr/bin/'
|
||||||
|
$(INSTALL_PROGRAM) repmgrd$(X) '$(DESTDIR)/usr/bin/'
|
||||||
|
$(INSTALL_PROGRAM) repmgr$(X) '$(DESTDIR)/usr/bin/'
|
||||||
|
|
||||||
ifneq (,$(DATA)$(DATA_built))
|
ifneq (,$(DATA)$(DATA_built))
|
||||||
@for file in $(addprefix $(srcdir)/, $(DATA)) $(DATA_built); do \
|
@for file in $(addprefix $(srcdir)/, $(DATA)) $(DATA_built); do \
|
||||||
@@ -45,10 +80,23 @@ clean:
|
|||||||
rm -f *.o
|
rm -f *.o
|
||||||
rm -f repmgrd
|
rm -f repmgrd
|
||||||
rm -f repmgr
|
rm -f repmgr
|
||||||
|
$(MAKE) -C sql clean
|
||||||
|
|
||||||
|
# Get correct version numbers and install paths, depending on your postgres version
|
||||||
|
PG_VERSION = $(shell pg_config --version | cut -d ' ' -f 2 | cut -d '.' -f 1,2)
|
||||||
|
REPMGR_VERSION = $(shell grep REPMGR_VERSION version.h | cut -d ' ' -f 3 | cut -d '"' -f 2)
|
||||||
|
PKGLIBDIR = $(shell pg_config --pkglibdir)
|
||||||
|
SHAREDIR = $(shell pg_config --sharedir)
|
||||||
|
|
||||||
deb: repmgrd repmgr
|
deb: repmgrd repmgr
|
||||||
mkdir -p ./debian/usr/bin
|
mkdir -p ./debian/usr/bin
|
||||||
cp repmgrd repmgr ./debian/usr/bin/
|
cp repmgrd repmgr ./debian/usr/bin/
|
||||||
|
mkdir -p ./debian$(SHAREDIR)/contrib/
|
||||||
|
cp sql/repmgr_funcs.sql ./debian$(SHAREDIR)/contrib/
|
||||||
|
cp sql/uninstall_repmgr_funcs.sql ./debian$(SHAREDIR)/contrib/
|
||||||
|
mkdir -p ./debian$(PKGLIBDIR)/
|
||||||
|
cp sql/repmgr_funcs.so ./debian$(PKGLIBDIR)/
|
||||||
dpkg-deb --build debian
|
dpkg-deb --build debian
|
||||||
mv debian.deb ../postgresql-repmgr-9.0_1.0.0.deb
|
mv debian.deb ../postgresql-repmgr-$(PG_VERSION)_$(REPMGR_VERSION).deb
|
||||||
|
rm -rf ./debian/usr
|
||||||
|
|
||||||
|
|||||||
127
PACKAGES.md
Normal file
127
PACKAGES.md
Normal file
@@ -0,0 +1,127 @@
|
|||||||
|
Packaging
|
||||||
|
=========
|
||||||
|
|
||||||
|
Notes on RedHat Linux, Fedora, and CentOS Builds
|
||||||
|
------------------------------------------------
|
||||||
|
|
||||||
|
The RPM packages of PostgreSQL put `pg_config` into the `postgresql-devel`
|
||||||
|
package, not the main server one. And if you have a RPM install of PostgreSQL
|
||||||
|
9.0, the entire PostgreSQL binary directory will not be in your PATH by default
|
||||||
|
either. Individual utilities are made available via the `alternatives`
|
||||||
|
mechanism, but not all commands will be wrapped that way. The files installed
|
||||||
|
by repmgr will certainly not be in the default PATH for the postgres user
|
||||||
|
on such a system. They will instead be in /usr/pgsql-9.0/bin/ on this
|
||||||
|
type of system.
|
||||||
|
|
||||||
|
When building repmgr against a RPM packaged build, you may discover that some
|
||||||
|
development packages are needed as well. The following build errors can
|
||||||
|
occur:
|
||||||
|
|
||||||
|
/usr/bin/ld: cannot find -lxslt
|
||||||
|
/usr/bin/ld: cannot find -lpam
|
||||||
|
|
||||||
|
Install the following packages to correct those:
|
||||||
|
|
||||||
|
|
||||||
|
yum install libxslt-devel
|
||||||
|
yum install pam-devel
|
||||||
|
|
||||||
|
If building repmgr as a regular user, then doing the install into the system
|
||||||
|
directories using sudo, the syntax is hard. `pg_config` won't be in root's
|
||||||
|
path either. The following recipe should work:
|
||||||
|
|
||||||
|
sudo PATH="/usr/pgsql-9.0/bin:$PATH" make USE_PGXS=1 install
|
||||||
|
|
||||||
|
|
||||||
|
Issues with 32 and 64 bit RPMs
|
||||||
|
------------------------------
|
||||||
|
|
||||||
|
If when building, you receive a series of errors of this form:
|
||||||
|
|
||||||
|
/usr/bin/ld: skipping incompatible /usr/pgsql-9.0/lib/libpq.so when searching for -lpq
|
||||||
|
|
||||||
|
This is likely because you have both the 32 and 64 bit versions of the
|
||||||
|
`postgresql90-devel` package installed. You can check that like this:
|
||||||
|
|
||||||
|
rpm -qa --queryformat '%{NAME}\t%{ARCH}\n' | grep postgresql90-devel
|
||||||
|
|
||||||
|
And if two packages appear, one for i386 and one for x86_64, that's not supposed
|
||||||
|
to be allowed.
|
||||||
|
|
||||||
|
This can happen when using the PGDG repo to install that package;
|
||||||
|
here is an example sessions demonstrating the problem case appearing:
|
||||||
|
|
||||||
|
|
||||||
|
# yum install postgresql-devel
|
||||||
|
..
|
||||||
|
Setting up Install Process
|
||||||
|
Resolving Dependencies
|
||||||
|
--> Running transaction check
|
||||||
|
---> Package postgresql90-devel.i386 0:9.0.2-2PGDG.rhel5 set to be updated
|
||||||
|
---> Package postgresql90-devel.x86_64 0:9.0.2-2PGDG.rhel5 set to be updated
|
||||||
|
--> Finished Dependency Resolution
|
||||||
|
|
||||||
|
Dependencies Resolved
|
||||||
|
|
||||||
|
=========================================================================
|
||||||
|
Package Arch Version Repository Size
|
||||||
|
=========================================================================
|
||||||
|
Installing:
|
||||||
|
postgresql90-devel i386 9.0.2-2PGDG.rhel5 pgdg90 1.5 M
|
||||||
|
postgresql90-devel x86_64 9.0.2-2PGDG.rhel5 pgdg90 1.6 M
|
||||||
|
|
||||||
|
|
||||||
|
Note how both the i386 and x86_64 platform architectures are selected for
|
||||||
|
installation. Your main PostgreSQL package will only be compatible with one of
|
||||||
|
those, and if the repmgr build finds the wrong postgresql90-devel these
|
||||||
|
"skipping incompatible" messages appear.
|
||||||
|
|
||||||
|
In this case, you can temporarily remove both packages, then just install the
|
||||||
|
correct one for your architecture. Example:
|
||||||
|
|
||||||
|
rpm -e postgresql90-devel --allmatches
|
||||||
|
yum install postgresql90-devel-9.0.2-2PGDG.rhel5.x86_64
|
||||||
|
|
||||||
|
Instead just deleting the package from the wrong platform might not leave behind
|
||||||
|
the correct files, due to the way in which these accidentally happen to interact.
|
||||||
|
If you already tried to build repmgr before doing this, you'll need to do:
|
||||||
|
|
||||||
|
make USE_PGXS=1 clean
|
||||||
|
|
||||||
|
to get rid of leftover files from the wrong architecture.
|
||||||
|
|
||||||
|
Notes on Ubuntu, Debian or other Debian-based Builds
|
||||||
|
----------------------------------------------------
|
||||||
|
|
||||||
|
The Debian packages of PostgreSQL put `pg_config` into the development package
|
||||||
|
called `postgresql-server-dev-$version`.
|
||||||
|
|
||||||
|
When building repmgr against a Debian packages build, you may discover that some
|
||||||
|
development packages are needed as well. You will need the following development
|
||||||
|
packages installed:
|
||||||
|
|
||||||
|
sudo apt-get install libxslt-dev libxml2-dev libpam-dev libedit-dev
|
||||||
|
|
||||||
|
If you're using Debian packages for PostgreSQL and are building repmgr with the
|
||||||
|
USE_PGXS option you also need to install the corresponding development package:
|
||||||
|
|
||||||
|
sudo apt-get install postgresql-server-dev-9.0
|
||||||
|
|
||||||
|
If you build and install repmgr manually it will not be on the system path. The
|
||||||
|
binaries will be installed in /usr/lib/postgresql/$version/bin/ which is not on
|
||||||
|
the default path. The reason behind this is that Ubuntu/Debian systems manage
|
||||||
|
multiple installed versions of PostgreSQL on the same system through a wrapper
|
||||||
|
called pg_wrapper and repmgr is not (yet) known to this wrapper.
|
||||||
|
|
||||||
|
You can solve this in many different ways, the most Debian like is to make an
|
||||||
|
alternate for repmgr and repmgrd:
|
||||||
|
|
||||||
|
sudo update-alternatives --install /usr/bin/repmgr repmgr /usr/lib/postgresql/9.0/bin/repmgr 10
|
||||||
|
sudo update-alternatives --install /usr/bin/repmgrd repmgrd /usr/lib/postgresql/9.0/bin/repmgrd 10
|
||||||
|
|
||||||
|
You can also make a deb package of repmgr using:
|
||||||
|
|
||||||
|
make USE_PGXS=1 deb
|
||||||
|
|
||||||
|
This will build a Debian package one level up from where you build, normally the
|
||||||
|
same directory that you have your repmgr/ directory in.
|
||||||
1
QUICKSTART.md
Normal file
1
QUICKSTART.md
Normal file
@@ -0,0 +1 @@
|
|||||||
|
The contents of this file have been incorporated into the main README.md document.
|
||||||
1049
README.rst
1049
README.rst
File diff suppressed because it is too large
Load Diff
36
SSH-RSYNC.md
Normal file
36
SSH-RSYNC.md
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
Set up trusted copy between postgres accounts
|
||||||
|
---------------------------------------------
|
||||||
|
|
||||||
|
If you need to use `rsync` to clone standby servers, the `postgres` account
|
||||||
|
on your primary and standby servers must be each able to access the other
|
||||||
|
using SSH without a password.
|
||||||
|
|
||||||
|
First generate an ssh key, using an empty passphrase, and copy the resulting
|
||||||
|
keys and a matching authorization file to a privileged user account on the other
|
||||||
|
system:
|
||||||
|
|
||||||
|
[postgres@node1]$ ssh-keygen -t rsa
|
||||||
|
Generating public/private rsa key pair.
|
||||||
|
Enter file in which to save the key (/var/lib/pgsql/.ssh/id_rsa):
|
||||||
|
Enter passphrase (empty for no passphrase):
|
||||||
|
Enter same passphrase again:
|
||||||
|
Your identification has been saved in /var/lib/pgsql/.ssh/id_rsa.
|
||||||
|
Your public key has been saved in /var/lib/pgsql/.ssh/id_rsa.pub.
|
||||||
|
The key fingerprint is:
|
||||||
|
aa:bb:cc:dd:ee:ff:aa:11:22:33:44:55:66:77:88:99 postgres@db1.domain.com
|
||||||
|
[postgres@node1]$ cat ~/.ssh/id_rsa.pub >> ~/.ssh/authorized_keys
|
||||||
|
[postgres@node1]$ chmod go-rwx ~/.ssh/*
|
||||||
|
[postgres@node1]$ cd ~/.ssh
|
||||||
|
[postgres@node1]$ scp id_rsa.pub id_rsa authorized_keys user@node2:
|
||||||
|
|
||||||
|
Login as a user on the other system, and install the files into the `postgres`
|
||||||
|
user's account:
|
||||||
|
|
||||||
|
[user@node2 ~]$ sudo chown postgres.postgres authorized_keys id_rsa.pub id_rsa
|
||||||
|
[user@node2 ~]$ sudo mkdir -p ~postgres/.ssh
|
||||||
|
[user@node2 ~]$ sudo chown postgres.postgres ~postgres/.ssh
|
||||||
|
[user@node2 ~]$ sudo mv authorized_keys id_rsa.pub id_rsa ~postgres/.ssh
|
||||||
|
[user@node2 ~]$ sudo chmod -R go-rwx ~postgres/.ssh
|
||||||
|
|
||||||
|
Now test that ssh in both directions works. You may have to accept some new
|
||||||
|
known hosts in the process.
|
||||||
73
TODO
73
TODO
@@ -1,21 +1,74 @@
|
|||||||
Known issues in repmgr
|
Known issues in repmgr
|
||||||
======================
|
======================
|
||||||
|
|
||||||
* The check for whether ``wal_keep_segments`` is considered large enough
|
|
||||||
does a string comparison rather than an integer one. It can give both
|
|
||||||
false positive (setting is large enough but flagged as too small) and
|
|
||||||
false negative (setting is too small but not noted as such) errors.
|
|
||||||
|
|
||||||
* When running repmgr against a remote machine, operations that start
|
* When running repmgr against a remote machine, operations that start
|
||||||
the database server using the ``pg_ctl`` command may accidentally
|
the database server using the ``pg_ctl`` command may accidentally
|
||||||
terminate after their associated ssh session ends.
|
terminate after their associated ssh session ends.
|
||||||
|
|
||||||
* After running repmgrd as a regular foreground application, hitting
|
* PGPASSFILE may not be passed to pg_basebackup
|
||||||
control-C causes the program to crash.
|
|
||||||
|
|
||||||
Planned feature improvements
|
Planned feature improvements
|
||||||
============================
|
============================
|
||||||
|
|
||||||
* Before running ``pg_start_backup()``, a sanity check that there is a
|
* Use 'primary' instead of 'master' in documentation and log output
|
||||||
a working ssh connection to the destination would help find
|
for consistency with PostgreSQL documentation. See also commit
|
||||||
configuration errors before disturbing the database.
|
870b0a53b627eeb9aca1fc14cbafe25b5beafe12.
|
||||||
|
|
||||||
|
* A better check which standby did receive most of the data
|
||||||
|
|
||||||
|
* Make the fact that a standby may be delayed a factor in the voting
|
||||||
|
algorithm
|
||||||
|
|
||||||
|
* include support for delayed standbys
|
||||||
|
|
||||||
|
* Create the repmgr user/database on "master register".
|
||||||
|
|
||||||
|
* Use pg_basebackup for the data directory, and ALSO rsync for the
|
||||||
|
configuration files.
|
||||||
|
|
||||||
|
* If no configuration file supplied, search in sensible default locations
|
||||||
|
(currently: current directory and `pg_config --sysconfdir`); if
|
||||||
|
possible this should include the location provided by the package,
|
||||||
|
if installed.
|
||||||
|
|
||||||
|
* repmgrd: if connection to the upstream node fails on startup, optionally
|
||||||
|
retry for a certain period before giving up; this will cover cases when
|
||||||
|
e.g. primary and standby are both starting up, and the standby comes up
|
||||||
|
before the primary. See github issue #80.
|
||||||
|
|
||||||
|
* make old master node ID available for event notification commands
|
||||||
|
(See github issue #80).
|
||||||
|
|
||||||
|
* repmgr standby clone: possibility to use barman instead of performing a new base backup
|
||||||
|
|
||||||
|
* possibility to transform a failed master into a new standby with pg_rewind
|
||||||
|
|
||||||
|
* "repmgr standby switchover" to promote a standby in a controlled manner
|
||||||
|
and convert the existing primary into a standby
|
||||||
|
|
||||||
|
* make repmgrd more robust
|
||||||
|
|
||||||
|
* repmgr: when cloning a standby using pg_basebackup and replication slots are
|
||||||
|
requested, activate the replication slot using pg_receivexlog to negate the
|
||||||
|
need to set `wal_keep_segments` just for the initial clone (9.4 and 9.5).
|
||||||
|
|
||||||
|
* repmgr: enable "standby follow" to point a standby at another standby, not
|
||||||
|
just the replication cluster master (see GitHub #130)
|
||||||
|
|
||||||
|
|
||||||
|
Usability improvements
|
||||||
|
======================
|
||||||
|
|
||||||
|
* repmgr: add interrupt handler, so that if the program is interrupted
|
||||||
|
while running a backup, an attempt can be made to execute pg_stop_backup()
|
||||||
|
on the primary, to prevent an orphaned backup state existing.
|
||||||
|
|
||||||
|
* repmgr: when unregistering a node, delete any entries in the repl_monitoring
|
||||||
|
table.
|
||||||
|
|
||||||
|
* repmgr: for "standby unregister", accept connection parameters for the
|
||||||
|
primary and perform metadata updates (and slot removal) directly on
|
||||||
|
the primary, to allow a shutdown standby to be unregistered
|
||||||
|
(currently the standby must still be running, which means the replication
|
||||||
|
slot can't be dropped).
|
||||||
|
|||||||
163
check_dir.c
163
check_dir.c
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* check_dir.c - Directories management functions
|
* check_dir.c - Directories management functions
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -23,15 +23,18 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
#include <ftw.h>
|
||||||
|
|
||||||
/* NB: postgres_fe must be included BEFORE check_dir */
|
/* NB: postgres_fe must be included BEFORE check_dir */
|
||||||
#include "postgres_fe.h"
|
#include <libpq-fe.h>
|
||||||
#include "check_dir.h"
|
#include <postgres_fe.h>
|
||||||
|
|
||||||
|
#include "check_dir.h"
|
||||||
#include "strutil.h"
|
#include "strutil.h"
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
|
|
||||||
static int mkdir_p(char *path, mode_t omode);
|
static bool _create_pg_dir(char *dir, bool force, bool for_witness);
|
||||||
|
static int unlink_dir_callback(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* make sure the directory either doesn't exist or is empty
|
* make sure the directory either doesn't exist or is empty
|
||||||
@@ -46,9 +49,9 @@ static int mkdir_p(char *path, mode_t omode);
|
|||||||
int
|
int
|
||||||
check_dir(char *dir)
|
check_dir(char *dir)
|
||||||
{
|
{
|
||||||
DIR *chkdir;
|
DIR *chkdir;
|
||||||
struct dirent *file;
|
struct dirent *file;
|
||||||
int result = 1;
|
int result = 1;
|
||||||
|
|
||||||
errno = 0;
|
errno = 0;
|
||||||
|
|
||||||
@@ -60,7 +63,7 @@ check_dir(char *dir)
|
|||||||
while ((file = readdir(chkdir)) != NULL)
|
while ((file = readdir(chkdir)) != NULL)
|
||||||
{
|
{
|
||||||
if (strcmp(".", file->d_name) == 0 ||
|
if (strcmp(".", file->d_name) == 0 ||
|
||||||
strcmp("..", file->d_name) == 0)
|
strcmp("..", file->d_name) == 0)
|
||||||
{
|
{
|
||||||
/* skip this and parent directory */
|
/* skip this and parent directory */
|
||||||
continue;
|
continue;
|
||||||
@@ -73,6 +76,7 @@ check_dir(char *dir)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifdef WIN32
|
#ifdef WIN32
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This fix is in mingw cvs (runtime/mingwex/dirent.c rev 1.4), but not in
|
* This fix is in mingw cvs (runtime/mingwex/dirent.c rev 1.4), but not in
|
||||||
* released version
|
* released version
|
||||||
@@ -84,29 +88,29 @@ check_dir(char *dir)
|
|||||||
closedir(chkdir);
|
closedir(chkdir);
|
||||||
|
|
||||||
if (errno != 0)
|
if (errno != 0)
|
||||||
return -1; /* some kind of I/O error? */
|
return -1; /* some kind of I/O error? */
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Create directory
|
* Create directory with error log message when failing
|
||||||
*/
|
*/
|
||||||
bool
|
bool
|
||||||
create_directory(char *dir)
|
create_dir(char *dir)
|
||||||
{
|
{
|
||||||
if (mkdir_p(dir, 0700) == 0)
|
if (mkdir_p(dir, 0700) == 0)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
log_err(_("Could not create directory \"%s\": %s\n"),
|
log_err(_("unable to create directory \"%s\": %s\n"),
|
||||||
dir, strerror(errno));
|
dir, strerror(errno));
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool
|
bool
|
||||||
set_directory_permissions(char *dir)
|
set_dir_permissions(char *dir)
|
||||||
{
|
{
|
||||||
return (chmod(dir, 0700) != 0) ? false : true;
|
return (chmod(dir, 0700) != 0) ? false : true;
|
||||||
}
|
}
|
||||||
@@ -124,15 +128,15 @@ set_directory_permissions(char *dir)
|
|||||||
* note that on failure, the path arg has been modified to show the particular
|
* note that on failure, the path arg has been modified to show the particular
|
||||||
* directory level we had problems with.
|
* directory level we had problems with.
|
||||||
*/
|
*/
|
||||||
static int
|
int
|
||||||
mkdir_p(char *path, mode_t omode)
|
mkdir_p(char *path, mode_t omode)
|
||||||
{
|
{
|
||||||
struct stat sb;
|
struct stat sb;
|
||||||
mode_t numask,
|
mode_t numask,
|
||||||
oumask;
|
oumask;
|
||||||
int first,
|
int first,
|
||||||
last,
|
last,
|
||||||
retval;
|
retval;
|
||||||
char *p;
|
char *p;
|
||||||
|
|
||||||
p = path;
|
p = path;
|
||||||
@@ -151,8 +155,8 @@ mkdir_p(char *path, mode_t omode)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
else if (p[1] == ':' &&
|
else if (p[1] == ':' &&
|
||||||
((p[0] >= 'a' && p[0] <= 'z') ||
|
((p[0] >= 'a' && p[0] <= 'z') ||
|
||||||
(p[0] >= 'A' && p[0] <= 'Z')))
|
(p[0] >= 'A' && p[0] <= 'Z')))
|
||||||
{
|
{
|
||||||
/* local drive */
|
/* local drive */
|
||||||
p += 2;
|
p += 2;
|
||||||
@@ -223,10 +227,123 @@ bool
|
|||||||
is_pg_dir(char *dir)
|
is_pg_dir(char *dir)
|
||||||
{
|
{
|
||||||
const size_t buf_sz = 8192;
|
const size_t buf_sz = 8192;
|
||||||
char path[buf_sz];
|
char path[buf_sz];
|
||||||
struct stat sb;
|
struct stat sb;
|
||||||
|
int r;
|
||||||
|
|
||||||
|
/* test pgdata */
|
||||||
xsnprintf(path, buf_sz, "%s/PG_VERSION", dir);
|
xsnprintf(path, buf_sz, "%s/PG_VERSION", dir);
|
||||||
|
if (stat(path, &sb) == 0)
|
||||||
|
return true;
|
||||||
|
|
||||||
return (stat(path, &sb) == 0) ? true : false;
|
/* test tablespace dir */
|
||||||
|
sprintf(path, "ls %s/PG_*/ -I*", dir);
|
||||||
|
r = system(path);
|
||||||
|
if (r == 0)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
create_pg_dir(char *dir, bool force)
|
||||||
|
{
|
||||||
|
return _create_pg_dir(dir, force, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool
|
||||||
|
create_witness_pg_dir(char *dir, bool force)
|
||||||
|
{
|
||||||
|
return _create_pg_dir(dir, force, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static bool
|
||||||
|
_create_pg_dir(char *dir, bool force, bool for_witness)
|
||||||
|
{
|
||||||
|
bool pg_dir = false;
|
||||||
|
|
||||||
|
/* Check this directory could be used as a PGDATA dir */
|
||||||
|
switch (check_dir(dir))
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
/* dir not there, must create it */
|
||||||
|
log_info(_("creating directory \"%s\"...\n"), dir);
|
||||||
|
|
||||||
|
if (!create_dir(dir))
|
||||||
|
{
|
||||||
|
log_err(_("unable to create directory \"%s\"...\n"),
|
||||||
|
dir);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
/* Present but empty, fix permissions and use it */
|
||||||
|
log_info(_("checking and correcting permissions on existing directory %s ...\n"),
|
||||||
|
dir);
|
||||||
|
|
||||||
|
if (!set_dir_permissions(dir))
|
||||||
|
{
|
||||||
|
log_err(_("unable to change permissions of directory \"%s\": %s\n"),
|
||||||
|
dir, strerror(errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
/* Present and not empty */
|
||||||
|
log_warning(_("directory \"%s\" exists but is not empty\n"),
|
||||||
|
dir);
|
||||||
|
|
||||||
|
pg_dir = is_pg_dir(dir);
|
||||||
|
|
||||||
|
|
||||||
|
if (pg_dir && force)
|
||||||
|
{
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The witness server does not store any data other than a copy of the
|
||||||
|
* repmgr metadata, so in --force mode we can simply overwrite the
|
||||||
|
* directory.
|
||||||
|
*
|
||||||
|
* For non-witness servers, we'll leave the data in place, both to reduce
|
||||||
|
* the risk of unintentional data loss and to make it possible for the
|
||||||
|
* data directory to be brought up-to-date with rsync.
|
||||||
|
*/
|
||||||
|
if (for_witness)
|
||||||
|
{
|
||||||
|
log_notice(_("deleting existing data directory \"%s\"\n"), dir);
|
||||||
|
nftw(dir, unlink_dir_callback, 64, FTW_DEPTH | FTW_PHYS);
|
||||||
|
}
|
||||||
|
/* Let it continue */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
else if (pg_dir && !force)
|
||||||
|
{
|
||||||
|
log_hint(_("This looks like a PostgreSQL directory.\n"
|
||||||
|
"If you are sure you want to clone here, "
|
||||||
|
"please check there is no PostgreSQL server "
|
||||||
|
"running and use the -F/--force option\n"));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
default:
|
||||||
|
/* Trouble accessing directory */
|
||||||
|
log_err(_("could not access directory \"%s\": %s\n"),
|
||||||
|
dir, strerror(errno));
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
unlink_dir_callback(const char *fpath, const struct stat *sb, int typeflag, struct FTW *ftwbuf)
|
||||||
|
{
|
||||||
|
int rv = remove(fpath);
|
||||||
|
|
||||||
|
if (rv)
|
||||||
|
perror(fpath);
|
||||||
|
|
||||||
|
return rv;
|
||||||
}
|
}
|
||||||
|
|||||||
13
check_dir.h
13
check_dir.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* check_dir.h
|
* check_dir.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2011
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -20,9 +20,12 @@
|
|||||||
#ifndef _REPMGR_CHECK_DIR_H_
|
#ifndef _REPMGR_CHECK_DIR_H_
|
||||||
#define _REPMGR_CHECK_DIR_H_
|
#define _REPMGR_CHECK_DIR_H_
|
||||||
|
|
||||||
int check_dir(char *dir);
|
int mkdir_p(char *path, mode_t omode);
|
||||||
bool create_directory(char *dir);
|
int check_dir(char *dir);
|
||||||
bool set_directory_permissions(char *dir);
|
bool create_dir(char *dir);
|
||||||
bool is_pg_dir(char *dir);
|
bool set_dir_permissions(char *dir);
|
||||||
|
bool is_pg_dir(char *dir);
|
||||||
|
bool create_pg_dir(char *dir, bool force);
|
||||||
|
bool create_witness_pg_dir(char *dir, bool force);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
105
config.h
105
config.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* config.h
|
* config.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2011
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -20,21 +20,102 @@
|
|||||||
#ifndef _REPMGR_CONFIG_H_
|
#ifndef _REPMGR_CONFIG_H_
|
||||||
#define _REPMGR_CONFIG_H_
|
#define _REPMGR_CONFIG_H_
|
||||||
|
|
||||||
#include "repmgr.h"
|
#include "postgres_fe.h"
|
||||||
|
|
||||||
#include "strutil.h"
|
#include "strutil.h"
|
||||||
|
|
||||||
|
#define CONFIG_FILE_NAME "repmgr.conf"
|
||||||
|
|
||||||
|
typedef struct EventNotificationListCell
|
||||||
|
{
|
||||||
|
struct EventNotificationListCell *next;
|
||||||
|
char event_type[MAXLEN];
|
||||||
|
} EventNotificationListCell;
|
||||||
|
|
||||||
|
typedef struct EventNotificationList
|
||||||
|
{
|
||||||
|
EventNotificationListCell *head;
|
||||||
|
EventNotificationListCell *tail;
|
||||||
|
} EventNotificationList;
|
||||||
|
|
||||||
|
typedef struct TablespaceListCell
|
||||||
|
{
|
||||||
|
struct TablespaceListCell *next;
|
||||||
|
char old_dir[MAXPGPATH];
|
||||||
|
char new_dir[MAXPGPATH];
|
||||||
|
} TablespaceListCell;
|
||||||
|
|
||||||
|
typedef struct TablespaceList
|
||||||
|
{
|
||||||
|
TablespaceListCell *head;
|
||||||
|
TablespaceListCell *tail;
|
||||||
|
} TablespaceList;
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
char cluster_name[MAXLEN];
|
char cluster_name[MAXLEN];
|
||||||
int node;
|
int node;
|
||||||
char conninfo[MAXLEN];
|
int upstream_node;
|
||||||
char loglevel[MAXLEN];
|
char conninfo[MAXLEN];
|
||||||
char logfacility[MAXLEN];
|
int failover;
|
||||||
char rsync_options[QUERY_STR_LEN];
|
int priority;
|
||||||
} t_configuration_options;
|
char node_name[MAXLEN];
|
||||||
|
char promote_command[MAXLEN];
|
||||||
|
char follow_command[MAXLEN];
|
||||||
|
char stop_command[MAXLEN];
|
||||||
|
char start_command[MAXLEN];
|
||||||
|
char restart_command[MAXLEN];
|
||||||
|
char loglevel[MAXLEN];
|
||||||
|
char logfacility[MAXLEN];
|
||||||
|
char rsync_options[QUERY_STR_LEN];
|
||||||
|
char ssh_options[QUERY_STR_LEN];
|
||||||
|
int master_response_timeout;
|
||||||
|
int reconnect_attempts;
|
||||||
|
int reconnect_interval;
|
||||||
|
char pg_bindir[MAXLEN];
|
||||||
|
char pg_ctl_options[MAXLEN];
|
||||||
|
char pg_basebackup_options[MAXLEN];
|
||||||
|
char restore_command[MAXLEN];
|
||||||
|
char logfile[MAXLEN];
|
||||||
|
int monitor_interval_secs;
|
||||||
|
int retry_promote_interval_secs;
|
||||||
|
int witness_repl_nodes_sync_interval_secs;
|
||||||
|
int use_replication_slots;
|
||||||
|
char event_notification_command[MAXLEN];
|
||||||
|
EventNotificationList event_notifications;
|
||||||
|
TablespaceList tablespace_mapping;
|
||||||
|
} t_configuration_options;
|
||||||
|
|
||||||
void parse_config(const char* config_file, t_configuration_options* options);
|
/*
|
||||||
void parse_line(char *buff, char *name, char *value);
|
* The following will initialize the structure with a minimal set of options;
|
||||||
char *trim(char *s);
|
* actual defaults are set in parse_config() before parsing the configuration file
|
||||||
|
*/
|
||||||
|
#define T_CONFIGURATION_OPTIONS_INITIALIZER { "", -1, NO_UPSTREAM_NODE, "", MANUAL_FAILOVER, -1, "", "", "", "", "", "", "", "", "", "", -1, -1, -1, "", "", "", "", "", 0, 0, 0, 0, "", { NULL, NULL }, {NULL, NULL} }
|
||||||
|
|
||||||
|
typedef struct ErrorListCell
|
||||||
|
{
|
||||||
|
struct ErrorListCell *next;
|
||||||
|
char *error_message;
|
||||||
|
} ErrorListCell;
|
||||||
|
|
||||||
|
typedef struct ErrorList
|
||||||
|
{
|
||||||
|
ErrorListCell *head;
|
||||||
|
ErrorListCell *tail;
|
||||||
|
} ErrorList;
|
||||||
|
|
||||||
|
void set_progname(const char *argv0);
|
||||||
|
const char * progname(void);
|
||||||
|
|
||||||
|
bool load_config(const char *config_file, bool verbose, t_configuration_options *options, char *argv0);
|
||||||
|
bool reload_config(t_configuration_options *orig_options);
|
||||||
|
bool parse_config(t_configuration_options *options);
|
||||||
|
void parse_line(char *buff, char *name, char *value);
|
||||||
|
char *trim(char *s);
|
||||||
|
void error_list_append(ErrorList *error_list, char *error_message);
|
||||||
|
int repmgr_atoi(const char *s,
|
||||||
|
const char *config_item,
|
||||||
|
ErrorList *error_list,
|
||||||
|
bool allow_negative);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
128
dbutils.h
128
dbutils.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* dbutils.h
|
* dbutils.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2011
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -20,16 +20,120 @@
|
|||||||
#ifndef _REPMGR_DBUTILS_H_
|
#ifndef _REPMGR_DBUTILS_H_
|
||||||
#define _REPMGR_DBUTILS_H_
|
#define _REPMGR_DBUTILS_H_
|
||||||
|
|
||||||
PGconn *establishDBConnection(const char *conninfo, const bool exit_on_error);
|
#include "access/xlogdefs.h"
|
||||||
PGconn *establishDBConnectionByParams(const char *keywords[],
|
|
||||||
const char *values[],
|
|
||||||
const bool exit_on_error);
|
|
||||||
bool is_standby(PGconn *conn);
|
|
||||||
char *pg_version(PGconn *conn, char* major_version);
|
|
||||||
bool guc_setted(PGconn *conn, const char *parameter, const char *op,
|
|
||||||
const char *value);
|
|
||||||
const char *get_cluster_size(PGconn *conn);
|
|
||||||
PGconn *getMasterConnection(PGconn *standby_conn, int id, char *cluster,
|
|
||||||
int *master_id, char *master_conninfo_out);
|
|
||||||
|
|
||||||
|
#include "config.h"
|
||||||
|
#include "strutil.h"
|
||||||
|
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
UNKNOWN = 0,
|
||||||
|
MASTER,
|
||||||
|
STANDBY,
|
||||||
|
WITNESS
|
||||||
|
} t_server_type;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Struct to store node information
|
||||||
|
*/
|
||||||
|
typedef struct s_node_info
|
||||||
|
{
|
||||||
|
int node_id;
|
||||||
|
int upstream_node_id;
|
||||||
|
t_server_type type;
|
||||||
|
char name[MAXLEN];
|
||||||
|
char conninfo_str[MAXLEN];
|
||||||
|
char slot_name[MAXLEN];
|
||||||
|
int priority;
|
||||||
|
bool active;
|
||||||
|
bool is_ready;
|
||||||
|
bool is_visible;
|
||||||
|
XLogRecPtr xlog_location;
|
||||||
|
} t_node_info;
|
||||||
|
|
||||||
|
|
||||||
|
#define T_NODE_INFO_INITIALIZER { \
|
||||||
|
NODE_NOT_FOUND, \
|
||||||
|
NO_UPSTREAM_NODE, \
|
||||||
|
UNKNOWN, \
|
||||||
|
"", \
|
||||||
|
"", \
|
||||||
|
"", \
|
||||||
|
DEFAULT_PRIORITY, \
|
||||||
|
true, \
|
||||||
|
false, \
|
||||||
|
false, \
|
||||||
|
InvalidXLogRecPtr \
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Struct to store replication slot information
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef struct s_replication_slot
|
||||||
|
{
|
||||||
|
char slot_name[MAXLEN];
|
||||||
|
char slot_type[MAXLEN];
|
||||||
|
bool active;
|
||||||
|
} t_replication_slot;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
PGconn *_establish_db_connection(const char *conninfo,
|
||||||
|
const bool exit_on_error,
|
||||||
|
const bool log_notice,
|
||||||
|
const bool verbose_only);
|
||||||
|
PGconn *establish_db_connection(const char *conninfo,
|
||||||
|
const bool exit_on_error);
|
||||||
|
PGconn *establish_db_connection_quiet(const char *conninfo);
|
||||||
|
PGconn *test_db_connection(const char *conninfo);
|
||||||
|
PGconn *establish_db_connection_by_params(const char *keywords[],
|
||||||
|
const char *values[],
|
||||||
|
const bool exit_on_error);
|
||||||
|
bool begin_transaction(PGconn *conn);
|
||||||
|
bool commit_transaction(PGconn *conn);
|
||||||
|
bool rollback_transaction(PGconn *conn);
|
||||||
|
bool check_cluster_schema(PGconn *conn);
|
||||||
|
int is_standby(PGconn *conn);
|
||||||
|
bool is_pgup(PGconn *conn, int timeout);
|
||||||
|
int get_master_node_id(PGconn *conn, char *cluster);
|
||||||
|
int get_server_version(PGconn *conn, char *server_version);
|
||||||
|
bool get_cluster_size(PGconn *conn, char *size);
|
||||||
|
bool get_pg_setting(PGconn *conn, const char *setting, char *output);
|
||||||
|
|
||||||
|
int guc_set(PGconn *conn, const char *parameter, const char *op,
|
||||||
|
const char *value);
|
||||||
|
int guc_set_typed(PGconn *conn, const char *parameter, const char *op,
|
||||||
|
const char *value, const char *datatype);
|
||||||
|
bool get_conninfo_value(const char *conninfo, const char *keyword, char *output);
|
||||||
|
PGconn *get_upstream_connection(PGconn *standby_conn, char *cluster,
|
||||||
|
int node_id,
|
||||||
|
int *upstream_node_id_ptr,
|
||||||
|
char *upstream_conninfo_out);
|
||||||
|
PGconn *get_master_connection(PGconn *standby_conn, char *cluster,
|
||||||
|
int *master_id, char *master_conninfo_out);
|
||||||
|
|
||||||
|
int wait_connection_availability(PGconn *conn, long long timeout);
|
||||||
|
bool cancel_query(PGconn *conn, int timeout);
|
||||||
|
char *get_repmgr_schema(void);
|
||||||
|
char *get_repmgr_schema_quoted(PGconn *conn);
|
||||||
|
bool create_replication_slot(PGconn *conn, char *slot_name, int server_version_num);
|
||||||
|
int get_slot_record(PGconn *conn, char *slot_name, t_replication_slot *record);
|
||||||
|
bool drop_replication_slot(PGconn *conn, char *slot_name);
|
||||||
|
bool start_backup(PGconn *conn, char *first_wal_segment, bool fast_checkpoint);
|
||||||
|
bool stop_backup(PGconn *conn, char *last_wal_segment);
|
||||||
|
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 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 delete_node_record(PGconn *conn, int node, char *action);
|
||||||
|
int get_node_record(PGconn *conn, char *cluster, int node_id, t_node_info *node_info);
|
||||||
|
int get_node_record_by_name(PGconn *conn, char *cluster, const char *node_name, t_node_info *node_info);
|
||||||
|
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 create_event_record(PGconn *conn, t_configuration_options *options, int node_id, char *event, bool successful, char *details);
|
||||||
|
|
||||||
|
int get_node_replication_state(PGconn *conn, char *node_name, char *output);
|
||||||
|
t_server_type parse_node_type(const char *type);
|
||||||
|
int get_data_checksum_version(const char *data_directory);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
8
debian/DEBIAN/control
vendored
8
debian/DEBIAN/control
vendored
@@ -1,9 +1,9 @@
|
|||||||
Package: repmgr
|
Package: repmgr-auto
|
||||||
Version: 1.0-1
|
Version: 3.1.3
|
||||||
Section: database
|
Section: database
|
||||||
Priority: optional
|
Priority: optional
|
||||||
Architecture: all
|
Architecture: all
|
||||||
Depends: rsync, postgresql-9.0
|
Depends: rsync, postgresql-9.3 | postgresql-9.4 | postgresql-9.5
|
||||||
Maintainer: Greg Smith <greg@2ndQuadrant.com>
|
Maintainer: Self built package <user@localhost>
|
||||||
Description: PostgreSQL replication setup, magament and monitoring
|
Description: PostgreSQL replication setup, magament and monitoring
|
||||||
has two main executables
|
has two main executables
|
||||||
|
|||||||
18
debian/repmgr.repmgrd.default
vendored
Normal file
18
debian/repmgr.repmgrd.default
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# default settings for repmgrd. This file is source by /bin/sh from
|
||||||
|
# /etc/init.d/repmgrd
|
||||||
|
|
||||||
|
# disable repmgrd by default so it won't get started upon installation
|
||||||
|
# valid values: yes/no
|
||||||
|
REPMGRD_ENABLED=no
|
||||||
|
|
||||||
|
# Options for repmgrd (required)
|
||||||
|
#REPMGRD_OPTS="--config-file /path/to/repmgr.conf"
|
||||||
|
|
||||||
|
# User to run repmgrd as
|
||||||
|
#REPMGRD_USER=postgres
|
||||||
|
|
||||||
|
# repmgrd binary
|
||||||
|
#REPMGRD_BIN=/usr/bin/repmgrd
|
||||||
|
|
||||||
|
# pid file
|
||||||
|
#REPMGRD_PIDFILE=/var/run/repmgrd.pid
|
||||||
101
debian/repmgr.repmgrd.init
vendored
Normal file
101
debian/repmgr.repmgrd.init
vendored
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
### BEGIN INIT INFO
|
||||||
|
# Provides: repmgrd
|
||||||
|
# Required-Start: $local_fs $remote_fs $network $syslog postgresql
|
||||||
|
# Required-Stop: $local_fs $remote_fs $network $syslog postgresql
|
||||||
|
# Should-Start: $syslog postgresql
|
||||||
|
# Default-Start: 2 3 4 5
|
||||||
|
# Default-Stop: 0 1 6
|
||||||
|
# Short-Description: Start/stop repmgrd
|
||||||
|
# Description: Enable repmgrd replication management and monitoring daemon for PostgreSQL
|
||||||
|
### END INIT INFO
|
||||||
|
|
||||||
|
set -e
|
||||||
|
|
||||||
|
DESC="PostgreSQL replication management and monitoring daemon"
|
||||||
|
NAME=repmgrd
|
||||||
|
|
||||||
|
REPMGRD_ENABLED=no
|
||||||
|
REPMGRD_OPTS=
|
||||||
|
REPMGRD_USER=postgres
|
||||||
|
REPMGRD_BIN=/usr/bin/repmgrd
|
||||||
|
REPMGRD_PIDFILE=/var/run/repmgrd.pid
|
||||||
|
|
||||||
|
# Read configuration variable file if it is present
|
||||||
|
[ -r /etc/default/$NAME ] && . /etc/default/$NAME
|
||||||
|
|
||||||
|
test -x $REPMGRD_BIN || exit 0
|
||||||
|
|
||||||
|
case "$REPMGRD_ENABLED" in
|
||||||
|
[Yy]*)
|
||||||
|
break
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
exit 0
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
# Define LSB log_* functions.
|
||||||
|
. /lib/lsb/init-functions
|
||||||
|
|
||||||
|
if [ -z "$REPMGRD_OPTS" ]
|
||||||
|
then
|
||||||
|
log_warning_msg "Not starting $NAME, REPMGRD_OPTS not set in /etc/default/$NAME"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
do_start()
|
||||||
|
{
|
||||||
|
# Return
|
||||||
|
# 0 if daemon has been started
|
||||||
|
# 1 if daemon was already running
|
||||||
|
# other if daemon could not be started or a failure occured
|
||||||
|
start-stop-daemon --start --quiet --background --chuid $REPMGRD_USER --make-pidfile --pidfile $REPMGRD_PIDFILE --exec $REPMGRD_BIN -- $REPMGRD_OPTS
|
||||||
|
}
|
||||||
|
|
||||||
|
do_stop()
|
||||||
|
{
|
||||||
|
# Return
|
||||||
|
# 0 if daemon has been stopped
|
||||||
|
# 1 if daemon was already stopped
|
||||||
|
# other if daemon could not be stopped or a failure occurred
|
||||||
|
start-stop-daemon --stop --quiet --retry=TERM/30/KILL/5 --pidfile $REPMGRD_PIDFILE --name "$(basename $REPMGRD_BIN)"
|
||||||
|
}
|
||||||
|
|
||||||
|
case "$1" in
|
||||||
|
start)
|
||||||
|
log_daemon_msg "Starting $DESC" "$NAME"
|
||||||
|
do_start
|
||||||
|
case "$?" in
|
||||||
|
0) log_end_msg 0 ;;
|
||||||
|
1) log_progress_msg "already started"
|
||||||
|
log_end_msg 0 ;;
|
||||||
|
*) log_end_msg 1 ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
;;
|
||||||
|
stop)
|
||||||
|
log_daemon_msg "Stopping $DESC" "$NAME"
|
||||||
|
do_stop
|
||||||
|
case "$?" in
|
||||||
|
0) log_end_msg 0 ;;
|
||||||
|
1) log_progress_msg "already stopped"
|
||||||
|
log_end_msg 0 ;;
|
||||||
|
*) log_end_msg 1 ;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
;;
|
||||||
|
restart|force-reload)
|
||||||
|
$0 stop
|
||||||
|
$0 start
|
||||||
|
;;
|
||||||
|
status)
|
||||||
|
status_of_proc -p $REPMGRD_PIDFILE $REPMGRD_BIN $NAME && exit 0 || exit $?
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "Usage: $SCRIPTNAME {start|stop|restart|force-reload|status}" >&2
|
||||||
|
exit 3
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
exit 0
|
||||||
194
dirmod.c
Normal file
194
dirmod.c
Normal file
@@ -0,0 +1,194 @@
|
|||||||
|
/*
|
||||||
|
*
|
||||||
|
* dirmod.c
|
||||||
|
* directory handling functions
|
||||||
|
*
|
||||||
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
|
*
|
||||||
|
* Portions Copyright (c) 1996-2013, PostgreSQL Global Development Group
|
||||||
|
* Portions Copyright (c) 1994, Regents of the University of California
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "postgres_fe.h"
|
||||||
|
|
||||||
|
/* Don't modify declarations in system headers */
|
||||||
|
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <dirent.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
|
||||||
|
/*
|
||||||
|
* pgfnames
|
||||||
|
*
|
||||||
|
* return a list of the names of objects in the argument directory. Caller
|
||||||
|
* must call pgfnames_cleanup later to free the memory allocated by this
|
||||||
|
* function.
|
||||||
|
*/
|
||||||
|
char **
|
||||||
|
pgfnames(const char *path)
|
||||||
|
{
|
||||||
|
DIR *dir;
|
||||||
|
struct dirent *file;
|
||||||
|
char **filenames;
|
||||||
|
int numnames = 0;
|
||||||
|
int fnsize = 200; /* enough for many small dbs */
|
||||||
|
|
||||||
|
dir = opendir(path);
|
||||||
|
if (dir == NULL)
|
||||||
|
{
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
filenames = (char **) palloc(fnsize * sizeof(char *));
|
||||||
|
|
||||||
|
while (errno = 0, (file = readdir(dir)) != NULL)
|
||||||
|
{
|
||||||
|
if (strcmp(file->d_name, ".") != 0 && strcmp(file->d_name, "..") != 0)
|
||||||
|
{
|
||||||
|
if (numnames + 1 >= fnsize)
|
||||||
|
{
|
||||||
|
fnsize *= 2;
|
||||||
|
filenames = (char **) repalloc(filenames,
|
||||||
|
fnsize * sizeof(char *));
|
||||||
|
}
|
||||||
|
filenames[numnames++] = pstrdup(file->d_name);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (errno)
|
||||||
|
{
|
||||||
|
fprintf(stderr, _("could not read directory \"%s\": %s\n"),
|
||||||
|
path, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
filenames[numnames] = NULL;
|
||||||
|
|
||||||
|
if (closedir(dir))
|
||||||
|
{
|
||||||
|
fprintf(stderr, _("could not close directory \"%s\": %s\n"),
|
||||||
|
path, strerror(errno));
|
||||||
|
}
|
||||||
|
|
||||||
|
return filenames;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* pgfnames_cleanup
|
||||||
|
*
|
||||||
|
* deallocate memory used for filenames
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
pgfnames_cleanup(char **filenames)
|
||||||
|
{
|
||||||
|
char **fn;
|
||||||
|
|
||||||
|
for (fn = filenames; *fn; fn++)
|
||||||
|
pfree(*fn);
|
||||||
|
|
||||||
|
pfree(filenames);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* rmtree
|
||||||
|
*
|
||||||
|
* Delete a directory tree recursively.
|
||||||
|
* Assumes path points to a valid directory.
|
||||||
|
* Deletes everything under path.
|
||||||
|
* If rmtopdir is true deletes the directory too.
|
||||||
|
* Returns true if successful, false if there was any problem.
|
||||||
|
* (The details of the problem are reported already, so caller
|
||||||
|
* doesn't really have to say anything more, but most do.)
|
||||||
|
*/
|
||||||
|
bool
|
||||||
|
rmtree(const char *path, bool rmtopdir)
|
||||||
|
{
|
||||||
|
bool result = true;
|
||||||
|
char pathbuf[MAXPGPATH];
|
||||||
|
char **filenames;
|
||||||
|
char **filename;
|
||||||
|
struct stat statbuf;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* we copy all the names out of the directory before we start modifying
|
||||||
|
* it.
|
||||||
|
*/
|
||||||
|
filenames = pgfnames(path);
|
||||||
|
|
||||||
|
if (filenames == NULL)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
/* now we have the names we can start removing things */
|
||||||
|
for (filename = filenames; *filename; filename++)
|
||||||
|
{
|
||||||
|
snprintf(pathbuf, MAXPGPATH, "%s/%s", path, *filename);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* It's ok if the file is not there anymore; we were just about to
|
||||||
|
* delete it anyway.
|
||||||
|
*
|
||||||
|
* This is not an academic possibility. One scenario where this
|
||||||
|
* happens is when bgwriter has a pending unlink request for a file in
|
||||||
|
* a database that's being dropped. In dropdb(), we call
|
||||||
|
* ForgetDatabaseFsyncRequests() to flush out any such pending unlink
|
||||||
|
* requests, but because that's asynchronous, it's not guaranteed that
|
||||||
|
* the bgwriter receives the message in time.
|
||||||
|
*/
|
||||||
|
if (lstat(pathbuf, &statbuf) != 0)
|
||||||
|
{
|
||||||
|
if (errno != ENOENT)
|
||||||
|
{
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (S_ISDIR(statbuf.st_mode))
|
||||||
|
{
|
||||||
|
/* call ourselves recursively for a directory */
|
||||||
|
if (!rmtree(pathbuf, true))
|
||||||
|
{
|
||||||
|
/* we already reported the error */
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (unlink(pathbuf) != 0)
|
||||||
|
{
|
||||||
|
if (errno != ENOENT)
|
||||||
|
{
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rmtopdir)
|
||||||
|
{
|
||||||
|
if (rmdir(path) != 0)
|
||||||
|
{
|
||||||
|
result = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pgfnames_cleanup(filenames);
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
23
dirmod.h
Normal file
23
dirmod.h
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
/*
|
||||||
|
* dirmod.h
|
||||||
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
|
*
|
||||||
|
* 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
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _DIRMOD_H_
|
||||||
|
#define _DIRMOD_H_
|
||||||
|
|
||||||
|
#endif
|
||||||
15
errcode.h
15
errcode.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* errcode.h
|
* errcode.h
|
||||||
* Copyright (C) 2ndQuadrant, 2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -25,13 +25,18 @@
|
|||||||
#define SUCCESS 0
|
#define SUCCESS 0
|
||||||
#define ERR_BAD_CONFIG 1
|
#define ERR_BAD_CONFIG 1
|
||||||
#define ERR_BAD_RSYNC 2
|
#define ERR_BAD_RSYNC 2
|
||||||
#define ERR_STOP_BACKUP 3
|
|
||||||
#define ERR_NO_RESTART 4
|
#define ERR_NO_RESTART 4
|
||||||
#define ERR_NEEDS_XLOG 5
|
|
||||||
#define ERR_DB_CON 6
|
#define ERR_DB_CON 6
|
||||||
#define ERR_DB_QUERY 7
|
#define ERR_DB_QUERY 7
|
||||||
#define ERR_PROMOTED 8
|
#define ERR_PROMOTED 8
|
||||||
#define ERR_BAD_PASSWORD 9
|
|
||||||
#define ERR_STR_OVERFLOW 10
|
#define ERR_STR_OVERFLOW 10
|
||||||
|
#define ERR_FAILOVER_FAIL 11
|
||||||
|
#define ERR_BAD_SSH 12
|
||||||
|
#define ERR_SYS_FAILURE 13
|
||||||
|
#define ERR_BAD_BASEBACKUP 14
|
||||||
|
#define ERR_INTERNAL 15
|
||||||
|
#define ERR_MONITORING_FAIL 16
|
||||||
|
#define ERR_BAD_BACKUP_LABEL 17
|
||||||
|
#define ERR_SWITCHOVER_FAIL 18
|
||||||
|
|
||||||
#endif /* _ERRCODE_H_ */
|
#endif /* _ERRCODE_H_ */
|
||||||
|
|||||||
259
log.c
259
log.c
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* log.c - Logging methods
|
* log.c - Logging methods
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* This module is a set of methods for logging (currently only syslog)
|
* This module is a set of methods for logging (currently only syslog)
|
||||||
*
|
*
|
||||||
@@ -25,9 +25,11 @@
|
|||||||
|
|
||||||
#ifdef HAVE_SYSLOG
|
#ifdef HAVE_SYSLOG
|
||||||
#include <syslog.h>
|
#include <syslog.h>
|
||||||
#include <stdarg.h>
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
|
|
||||||
#define DEFAULT_IDENT "repmgr"
|
#define DEFAULT_IDENT "repmgr"
|
||||||
@@ -37,20 +39,119 @@
|
|||||||
|
|
||||||
/* #define REPMGR_DEBUG */
|
/* #define REPMGR_DEBUG */
|
||||||
|
|
||||||
static int detect_log_level(const char* level);
|
static int detect_log_facility(const char *facility);
|
||||||
static int detect_log_facility(const char* facility);
|
static void _stderr_log_with_level(const char *level_name, int level, const char *fmt, va_list ap)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 0)));
|
||||||
|
|
||||||
int log_type = REPMGR_STDERR;
|
int log_type = REPMGR_STDERR;
|
||||||
int log_level = LOG_NOTICE;
|
int log_level = LOG_NOTICE;
|
||||||
|
int last_log_level = LOG_NOTICE;
|
||||||
|
int verbose_logging = false;
|
||||||
|
int terse_logging = false;
|
||||||
|
|
||||||
bool logger_init(const char* ident, const char* level, const char* facility)
|
extern void
|
||||||
|
stderr_log_with_level(const char *level_name, int level, const char *fmt, ...)
|
||||||
{
|
{
|
||||||
|
va_list arglist;
|
||||||
|
|
||||||
int l;
|
va_start(arglist, fmt);
|
||||||
int f;
|
_stderr_log_with_level(level_name, level, fmt, arglist);
|
||||||
|
va_end(arglist);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
_stderr_log_with_level(const char *level_name, int level, const char *fmt, va_list ap)
|
||||||
|
{
|
||||||
|
time_t t;
|
||||||
|
struct tm *tm;
|
||||||
|
char buff[100];
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Store the requested level so that if there's a subsequent
|
||||||
|
* log_hint(), we can suppress that if appropriate.
|
||||||
|
*/
|
||||||
|
last_log_level = level;
|
||||||
|
|
||||||
|
if (log_level >= level)
|
||||||
|
{
|
||||||
|
time(&t);
|
||||||
|
tm = localtime(&t);
|
||||||
|
strftime(buff, 100, "[%Y-%m-%d %H:%M:%S]", tm);
|
||||||
|
fprintf(stderr, "%s [%s] ", buff, level_name);
|
||||||
|
|
||||||
|
vfprintf(stderr, fmt, ap);
|
||||||
|
|
||||||
|
fflush(stderr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
log_hint(const char *fmt, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
if (terse_logging == false)
|
||||||
|
{
|
||||||
|
va_start(ap, fmt);
|
||||||
|
_stderr_log_with_level("HINT", last_log_level, fmt, ap);
|
||||||
|
va_end(ap);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void
|
||||||
|
log_verbose(int level, const char *fmt, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
|
||||||
|
va_start(ap, fmt);
|
||||||
|
|
||||||
|
if (verbose_logging == true)
|
||||||
|
{
|
||||||
|
switch(level)
|
||||||
|
{
|
||||||
|
case LOG_EMERG:
|
||||||
|
_stderr_log_with_level("EMERG", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_ALERT:
|
||||||
|
_stderr_log_with_level("ALERT", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_CRIT:
|
||||||
|
_stderr_log_with_level("CRIT", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_ERR:
|
||||||
|
_stderr_log_with_level("ERR", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_WARNING:
|
||||||
|
_stderr_log_with_level("WARNING", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_NOTICE:
|
||||||
|
_stderr_log_with_level("NOTICE", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_INFO:
|
||||||
|
_stderr_log_with_level("INFO", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
case LOG_DEBUG:
|
||||||
|
_stderr_log_with_level("DEBUG", level, fmt, ap);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
va_end(ap);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
logger_init(t_configuration_options *opts, const char *ident)
|
||||||
|
{
|
||||||
|
char *level = opts->loglevel;
|
||||||
|
char *facility = opts->logfacility;
|
||||||
|
|
||||||
|
int l;
|
||||||
|
int f;
|
||||||
|
|
||||||
#ifdef HAVE_SYSLOG
|
#ifdef HAVE_SYSLOG
|
||||||
int syslog_facility = DEFAULT_SYSLOG_FACILITY;
|
int syslog_facility = DEFAULT_SYSLOG_FACILITY;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef REPMGR_DEBUG
|
#ifdef REPMGR_DEBUG
|
||||||
@@ -69,10 +170,10 @@ bool logger_init(const char* ident, const char* level, const char* facility)
|
|||||||
printf("Assigned level for logger: %d\n", l);
|
printf("Assigned level for logger: %d\n", l);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (l > 0)
|
if (l >= 0)
|
||||||
log_level = l;
|
log_level = l;
|
||||||
else
|
else
|
||||||
stderr_log_warning(_("Cannot detect log level %s (use any of DEBUG, INFO, NOTICE, WARNING, ERR, ALERT, CRIT or EMERG)\n"), level);
|
stderr_log_warning(_("Invalid log level \"%s\" (available values: DEBUG, INFO, NOTICE, WARNING, ERR, ALERT, CRIT or EMERG)\n"), level);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (facility && *facility)
|
if (facility && *facility)
|
||||||
@@ -107,21 +208,53 @@ bool logger_init(const char* ident, const char* level, const char* facility)
|
|||||||
|
|
||||||
if (log_type == REPMGR_SYSLOG)
|
if (log_type == REPMGR_SYSLOG)
|
||||||
{
|
{
|
||||||
setlogmask (LOG_UPTO (log_level));
|
setlogmask(LOG_UPTO(log_level));
|
||||||
openlog (ident, LOG_CONS | LOG_PID | LOG_NDELAY, syslog_facility);
|
openlog(ident, LOG_CONS | LOG_PID | LOG_NDELAY, syslog_facility);
|
||||||
|
|
||||||
stderr_log_notice(_("Setup syslog (level: %s, facility: %s)\n"), level, facility);
|
stderr_log_notice(_("Setup syslog (level: %s, facility: %s)\n"), level, facility);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return true;
|
if (*opts->logfile)
|
||||||
|
{
|
||||||
|
FILE *fd;
|
||||||
|
|
||||||
|
/* Check if we can write to the specified file before redirecting
|
||||||
|
* stderr - if freopen() fails, stderr output will vanish into
|
||||||
|
* the ether and the user won't know what's going on.
|
||||||
|
*/
|
||||||
|
|
||||||
|
fd = fopen(opts->logfile, "a");
|
||||||
|
if (fd == NULL)
|
||||||
|
{
|
||||||
|
stderr_log_err(_("Unable to open specified logfile '%s' for writing: %s\n"), opts->logfile, strerror(errno));
|
||||||
|
stderr_log_err(_("Terminating\n"));
|
||||||
|
exit(ERR_BAD_CONFIG);
|
||||||
|
}
|
||||||
|
fclose(fd);
|
||||||
|
|
||||||
|
stderr_log_notice(_("Redirecting logging output to '%s'\n"), opts->logfile);
|
||||||
|
fd = freopen(opts->logfile, "a", stderr);
|
||||||
|
|
||||||
|
/* It's possible freopen() may still fail due to e.g. a race condition;
|
||||||
|
as it's not feasible to restore stderr after a failed freopen(),
|
||||||
|
we'll write to stdout as a last resort.
|
||||||
|
*/
|
||||||
|
if (fd == NULL)
|
||||||
|
{
|
||||||
|
printf(_("Unable to open specified logfile %s for writing: %s\n"), opts->logfile, strerror(errno));
|
||||||
|
printf(_("Terminating\n"));
|
||||||
|
exit(ERR_BAD_CONFIG);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool logger_shutdown(void)
|
|
||||||
{
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
logger_shutdown(void)
|
||||||
|
{
|
||||||
#ifdef HAVE_SYSLOG
|
#ifdef HAVE_SYSLOG
|
||||||
if (log_type == REPMGR_SYSLOG)
|
if (log_type == REPMGR_SYSLOG)
|
||||||
closelog();
|
closelog();
|
||||||
@@ -131,17 +264,34 @@ bool logger_shutdown(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Set a minimum logging level. Intended for command line verbosity
|
* Indicate whether extra-verbose logging is required. This will
|
||||||
* options, which might increase requested logging over what's specified
|
* generate a lot of output, particularly debug logging, and should
|
||||||
* in the regular configuration file.
|
* not be permanently enabled in production.
|
||||||
|
*
|
||||||
|
* NOTE: in previous repmgr versions, this option forced the log
|
||||||
|
* level to INFO.
|
||||||
*/
|
*/
|
||||||
void logger_min_verbose(int minimum)
|
void
|
||||||
|
logger_set_verbose(void)
|
||||||
{
|
{
|
||||||
if (log_level < minimum)
|
verbose_logging = true;
|
||||||
log_level = minimum;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int detect_log_level(const char* level)
|
|
||||||
|
/*
|
||||||
|
* Indicate whether some non-critical log messages can be omitted.
|
||||||
|
* Currently this includes warnings about irrelevant command line
|
||||||
|
* options and hints.
|
||||||
|
*/
|
||||||
|
|
||||||
|
void logger_set_terse(void)
|
||||||
|
{
|
||||||
|
terse_logging = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
detect_log_level(const char *level)
|
||||||
{
|
{
|
||||||
if (!strcmp(level, "DEBUG"))
|
if (!strcmp(level, "DEBUG"))
|
||||||
return LOG_DEBUG;
|
return LOG_DEBUG;
|
||||||
@@ -160,43 +310,44 @@ int detect_log_level(const char* level)
|
|||||||
if (!strcmp(level, "EMERG"))
|
if (!strcmp(level, "EMERG"))
|
||||||
return LOG_EMERG;
|
return LOG_EMERG;
|
||||||
|
|
||||||
return 0;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int detect_log_facility(const char* facility)
|
static int
|
||||||
|
detect_log_facility(const char *facility)
|
||||||
{
|
{
|
||||||
int local = 0;
|
int local = 0;
|
||||||
|
|
||||||
if (!strncmp(facility, "LOCAL", 5) && strlen(facility) == 6)
|
if (!strncmp(facility, "LOCAL", 5) && strlen(facility) == 6)
|
||||||
{
|
{
|
||||||
|
local = atoi(&facility[5]);
|
||||||
local = atoi (&facility[5]);
|
|
||||||
|
|
||||||
switch (local)
|
switch (local)
|
||||||
{
|
{
|
||||||
case 0:
|
case 0:
|
||||||
return LOG_LOCAL0;
|
return LOG_LOCAL0;
|
||||||
break;
|
break;
|
||||||
case 1:
|
case 1:
|
||||||
return LOG_LOCAL1;
|
return LOG_LOCAL1;
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
return LOG_LOCAL2;
|
return LOG_LOCAL2;
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
return LOG_LOCAL3;
|
return LOG_LOCAL3;
|
||||||
break;
|
break;
|
||||||
case 4:
|
case 4:
|
||||||
return LOG_LOCAL4;
|
return LOG_LOCAL4;
|
||||||
break;
|
break;
|
||||||
case 5:
|
case 5:
|
||||||
return LOG_LOCAL5;
|
return LOG_LOCAL5;
|
||||||
break;
|
break;
|
||||||
case 6:
|
case 6:
|
||||||
return LOG_LOCAL6;
|
return LOG_LOCAL6;
|
||||||
break;
|
break;
|
||||||
case 7:
|
case 7:
|
||||||
return LOG_LOCAL7;
|
return LOG_LOCAL7;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
65
log.h
65
log.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* log.h
|
* log.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2011
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -25,15 +25,19 @@
|
|||||||
#define REPMGR_SYSLOG 1
|
#define REPMGR_SYSLOG 1
|
||||||
#define REPMGR_STDERR 2
|
#define REPMGR_STDERR 2
|
||||||
|
|
||||||
|
extern void
|
||||||
|
stderr_log_with_level(const char *level_name, int level, const char *fmt,...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 4)));
|
||||||
|
|
||||||
/* Standard error logging */
|
/* Standard error logging */
|
||||||
#define stderr_log_debug(...) if (log_level >= LOG_DEBUG) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_debug(...) stderr_log_with_level("DEBUG", LOG_DEBUG, __VA_ARGS__)
|
||||||
#define stderr_log_info(...) if (log_level >= LOG_INFO) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_info(...) stderr_log_with_level("INFO", LOG_INFO, __VA_ARGS__)
|
||||||
#define stderr_log_notice(...) if (log_level >= LOG_NOTICE) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_notice(...) stderr_log_with_level("NOTICE", LOG_NOTICE, __VA_ARGS__)
|
||||||
#define stderr_log_warning(...) if (log_level >= LOG_WARNING) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_warning(...) stderr_log_with_level("WARNING", LOG_WARNING, __VA_ARGS__)
|
||||||
#define stderr_log_err(...) if (log_level >= LOG_ERR) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_err(...) stderr_log_with_level("ERROR", LOG_ERR, __VA_ARGS__)
|
||||||
#define stderr_log_crit(...) if (log_level >= LOG_CRIT) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_crit(...) stderr_log_with_level("CRITICAL", LOG_CRIT, __VA_ARGS__)
|
||||||
#define stderr_log_alert(...) if (log_level >= LOG_ALERT) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_alert(...) stderr_log_with_level("ALERT", LOG_ALERT, __VA_ARGS__)
|
||||||
#define stderr_log_emerg(...) if (log_level >= LOG_EMERG) fprintf(stderr, __VA_ARGS__)
|
#define stderr_log_emerg(...) stderr_log_with_level("EMERGENCY", LOG_EMERG, __VA_ARGS__)
|
||||||
|
|
||||||
#ifdef HAVE_SYSLOG
|
#ifdef HAVE_SYSLOG
|
||||||
|
|
||||||
@@ -86,17 +90,16 @@
|
|||||||
if (log_type == REPMGR_SYSLOG) syslog(LOG_ALERT, __VA_ARGS__); \
|
if (log_type == REPMGR_SYSLOG) syslog(LOG_ALERT, __VA_ARGS__); \
|
||||||
else stderr_log_alert(__VA_ARGS__); \
|
else stderr_log_alert(__VA_ARGS__); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
#define LOG_EMERG 0 /* system is unusable */
|
#define LOG_EMERG 0 /* system is unusable */
|
||||||
#define LOG_ALERT 1 /* action must be taken immediately */
|
#define LOG_ALERT 1 /* action must be taken immediately */
|
||||||
#define LOG_CRIT 2 /* critical conditions */
|
#define LOG_CRIT 2 /* critical conditions */
|
||||||
#define LOG_ERR 3 /* error conditions */
|
#define LOG_ERR 3 /* error conditions */
|
||||||
#define LOG_WARNING 4 /* warning conditions */
|
#define LOG_WARNING 4 /* warning conditions */
|
||||||
#define LOG_NOTICE 5 /* normal but significant condition */
|
#define LOG_NOTICE 5 /* normal but significant condition */
|
||||||
#define LOG_INFO 6 /* informational */
|
#define LOG_INFO 6 /* informational */
|
||||||
#define LOG_DEBUG 7 /* debug-level messages */
|
#define LOG_DEBUG 7 /* debug-level messages */
|
||||||
|
|
||||||
#define log_debug(...) stderr_log_debug(__VA_ARGS__)
|
#define log_debug(...) stderr_log_debug(__VA_ARGS__)
|
||||||
#define log_info(...) stderr_log_info(__VA_ARGS__)
|
#define log_info(...) stderr_log_info(__VA_ARGS__)
|
||||||
@@ -106,16 +109,28 @@
|
|||||||
#define log_crit(...) stderr_log_crit(__VA_ARGS__)
|
#define log_crit(...) stderr_log_crit(__VA_ARGS__)
|
||||||
#define log_alert(...) stderr_log_alert(__VA_ARGS__)
|
#define log_alert(...) stderr_log_alert(__VA_ARGS__)
|
||||||
#define log_emerg(...) stderr_log_emerg(__VA_ARGS__)
|
#define log_emerg(...) stderr_log_emerg(__VA_ARGS__)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
int detect_log_level(const char *level);
|
||||||
|
|
||||||
/* Logger initialisation and shutdown */
|
/* Logger initialisation and shutdown */
|
||||||
bool logger_shutdown(void);
|
|
||||||
bool logger_init(const char* ident, const char* level, const char* facility);
|
|
||||||
void logger_min_verbose(int minimum);
|
|
||||||
|
|
||||||
extern int log_type;
|
bool logger_init(t_configuration_options * opts, const char *ident);
|
||||||
extern int log_level;
|
|
||||||
|
|
||||||
#endif
|
bool logger_shutdown(void);
|
||||||
|
|
||||||
|
void logger_set_verbose(void);
|
||||||
|
void logger_set_terse(void);
|
||||||
|
|
||||||
|
void log_hint(const char *fmt, ...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 1, 2)));
|
||||||
|
void log_verbose(int level, const char *fmt, ...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3)));
|
||||||
|
|
||||||
|
extern int log_type;
|
||||||
|
extern int log_level;
|
||||||
|
extern int verbose_logging;
|
||||||
|
extern int terse_logging;
|
||||||
|
|
||||||
|
#endif /* _REPMGR_LOG_H_ */
|
||||||
|
|||||||
21
repmgr.conf
21
repmgr.conf
@@ -1,21 +0,0 @@
|
|||||||
###################################################
|
|
||||||
# Replication Manager configuration file
|
|
||||||
###################################################
|
|
||||||
|
|
||||||
# Cluster name
|
|
||||||
cluster=test
|
|
||||||
|
|
||||||
# Node ID
|
|
||||||
node=2
|
|
||||||
|
|
||||||
# Connection information
|
|
||||||
conninfo='host=192.168.204.104'
|
|
||||||
rsync_options=--archive --checksum --compress --progress --rsh=ssh
|
|
||||||
|
|
||||||
# Log level: possible values are DEBUG, INFO, NOTICE, WARNING, ERR, ALERT, CRIT or EMERG
|
|
||||||
# Default: NOTICE
|
|
||||||
loglevel=NOTICE
|
|
||||||
|
|
||||||
# Logging facility: possible values are STDERR or - for Syslog integration - one of LOCAL0, LOCAL1, ..., LOCAL7, USER
|
|
||||||
# Default: STDERR
|
|
||||||
logfacility=STDERR
|
|
||||||
195
repmgr.conf.sample
Normal file
195
repmgr.conf.sample
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
###################################################
|
||||||
|
# Replication Manager sample configuration file
|
||||||
|
###################################################
|
||||||
|
|
||||||
|
# Some configuration items will be set with a default value; this
|
||||||
|
# is noted for each item. Where no default value is shown, the
|
||||||
|
# parameter will be treated as empty or false.
|
||||||
|
|
||||||
|
# Required configuration items
|
||||||
|
# ============================
|
||||||
|
#
|
||||||
|
# repmgr and repmgrd require these items to be configured:
|
||||||
|
|
||||||
|
# Cluster name - this will be used by repmgr to generate its internal
|
||||||
|
# schema (pattern: "repmgr_{cluster}"); while this name will be quoted
|
||||||
|
# to preserve case, we recommend using lower case and avoiding whitespace
|
||||||
|
# to facilitate easier querying of the repmgr views and tables.
|
||||||
|
#cluster=example_cluster
|
||||||
|
|
||||||
|
# Node ID and name
|
||||||
|
# (Note: we recommend to avoid naming nodes after their initial
|
||||||
|
# replication function, as this will cause confusion when e.g.
|
||||||
|
# "standby2" is promoted to primary)
|
||||||
|
#node=2 # a unique integer
|
||||||
|
#node_name=node2 # an arbitrary (but unique) string; we recommend using
|
||||||
|
# the server's hostname or another identifier unambiguously
|
||||||
|
# associated with the server to avoid confusion
|
||||||
|
|
||||||
|
# Database connection information as a conninfo string
|
||||||
|
# This must be accessible to all servers in the cluster; for details see:
|
||||||
|
#
|
||||||
|
# https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNSTRING
|
||||||
|
#
|
||||||
|
#conninfo='host=192.168.204.104 dbname=repmgr_db user=repmgr_usr'
|
||||||
|
#
|
||||||
|
# If repmgrd is in use, consider explicitly setting `connect_timeout` in the
|
||||||
|
# conninfo string to determine the length of time which elapses before
|
||||||
|
# a network connection attempt is abandoned; for details see:
|
||||||
|
#
|
||||||
|
# https://www.postgresql.org/docs/current/static/libpq-connect.html#LIBPQ-CONNECT-CONNECT-TIMEOUT
|
||||||
|
|
||||||
|
# Optional configuration items
|
||||||
|
# ============================
|
||||||
|
|
||||||
|
# Replication settings
|
||||||
|
# ---------------------
|
||||||
|
|
||||||
|
# When using cascading replication, a standby can connect to another
|
||||||
|
# upstream standby node which is specified by setting 'upstream_node'.
|
||||||
|
# In that case, the upstream node must exist before the new standby
|
||||||
|
# can be registered. If 'upstream_node' is not set, then the standby
|
||||||
|
# will connect directly to the primary node.
|
||||||
|
#upstream_node=1
|
||||||
|
|
||||||
|
# use physical replication slots - PostgreSQL 9.4 and later only
|
||||||
|
# (default: 0)
|
||||||
|
#use_replication_slots=0
|
||||||
|
|
||||||
|
# NOTE: 'max_replication_slots' should be configured for at least the
|
||||||
|
# number of standbys which will connect to the primary.
|
||||||
|
|
||||||
|
# Logging and monitoring settings
|
||||||
|
# -------------------------------
|
||||||
|
|
||||||
|
# Log level: possible values are DEBUG, INFO, NOTICE, WARNING, ERR, ALERT, CRIT or EMERG
|
||||||
|
# (default: NOTICE)
|
||||||
|
#loglevel=NOTICE
|
||||||
|
|
||||||
|
# Logging facility: possible values are STDERR or - for Syslog integration - one of LOCAL0, LOCAL1, ..., LOCAL7, USER
|
||||||
|
# (default: STDERR)
|
||||||
|
#logfacility=STDERR
|
||||||
|
|
||||||
|
# stderr can be redirected to an arbitrary file:
|
||||||
|
#
|
||||||
|
#logfile='/var/log/repmgr/repmgr.log'
|
||||||
|
|
||||||
|
# event notifications can be passed to an arbitrary external program
|
||||||
|
# together with the following parameters:
|
||||||
|
#
|
||||||
|
# %n - node ID
|
||||||
|
# %e - event type
|
||||||
|
# %s - success (1 or 0)
|
||||||
|
# %t - timestamp
|
||||||
|
# %d - details
|
||||||
|
#
|
||||||
|
# the values provided for "%t" and "%d" will probably contain spaces,
|
||||||
|
# so should be quoted in the provided command configuration, e.g.:
|
||||||
|
#
|
||||||
|
#event_notification_command='/path/to/some/script %n %e %s "%t" "%d"'
|
||||||
|
|
||||||
|
# By default, all notifications will be passed; the notification types
|
||||||
|
# can be filtered to explicitly named ones:
|
||||||
|
#
|
||||||
|
#event_notifications=master_register,standby_register,witness_create
|
||||||
|
|
||||||
|
|
||||||
|
# Environment/command settings
|
||||||
|
# ----------------------------
|
||||||
|
|
||||||
|
# path to PostgreSQL binary directory (location of pg_ctl, pg_basebackup etc.)
|
||||||
|
# (if not provided, defaults to system $PATH)
|
||||||
|
#pg_bindir=/usr/bin/
|
||||||
|
|
||||||
|
# service control commands
|
||||||
|
#
|
||||||
|
# repmgr provides options to to override the default pg_ctl commands
|
||||||
|
# used to stop, start and restart the PostgreSQL cluster
|
||||||
|
#
|
||||||
|
# NOTE: These commands must be runnable on remote nodes as well for switchover
|
||||||
|
# to function correctly.
|
||||||
|
#
|
||||||
|
# If you use sudo, the user repmgr runs as (usually 'postgres') must have
|
||||||
|
# passwordless sudo access to execute the command
|
||||||
|
#
|
||||||
|
# For example, to use systemd, you may use the following configuration:
|
||||||
|
#
|
||||||
|
# # this is required when running sudo over ssh without -t:
|
||||||
|
# Defaults:postgres !requiretty
|
||||||
|
# postgres ALL = NOPASSWD: /usr/bin/systemctl stop postgresql-9.5, \
|
||||||
|
# /usr/bin/systemctl start postgresql-9.5, \
|
||||||
|
# /usr/bin/systemctl restart postgresql-9.5
|
||||||
|
#
|
||||||
|
# start_command = systemctl start postgresql-9.5
|
||||||
|
# stop_command = systemctl stop postgresql-9.5
|
||||||
|
# restart_command = systemctl restart postgresql-9.5
|
||||||
|
|
||||||
|
# external command options
|
||||||
|
|
||||||
|
#rsync_options=--archive --checksum --compress --progress --rsh="ssh -o \"StrictHostKeyChecking no\""
|
||||||
|
#ssh_options=-o "StrictHostKeyChecking no"
|
||||||
|
|
||||||
|
# external command arguments. Values shown are examples.
|
||||||
|
|
||||||
|
#pg_ctl_options='-s'
|
||||||
|
#pg_basebackup_options='--xlog-method=s'
|
||||||
|
|
||||||
|
|
||||||
|
# Standby clone settings
|
||||||
|
# ----------------------
|
||||||
|
#
|
||||||
|
# These settings apply when cloning a standby (`repmgr standby clone`).
|
||||||
|
|
||||||
|
# Tablespaces can be remapped from one file system location to another:
|
||||||
|
#
|
||||||
|
# tablespace_mapping=/path/to/original/tablespace=/path/to/new/tablespace
|
||||||
|
|
||||||
|
# You can specify a restore_command to be used in the recovery.conf that
|
||||||
|
# will be placed in the cloned standby
|
||||||
|
#
|
||||||
|
# restore_command = cp /path/to/archived/wals/%f %p
|
||||||
|
|
||||||
|
# Failover settings (repmgrd)
|
||||||
|
# ---------------------------
|
||||||
|
#
|
||||||
|
# These settings are only applied when repmgrd is running. Values shown
|
||||||
|
# are defaults.
|
||||||
|
|
||||||
|
# Number of seconds to wait for a response from the primary server before
|
||||||
|
# deciding it has failed.
|
||||||
|
|
||||||
|
#master_response_timeout=60
|
||||||
|
|
||||||
|
# Number of attempts at what interval (in seconds) to try and
|
||||||
|
# connect to a server to establish its status (e.g. master
|
||||||
|
# during failover)
|
||||||
|
#reconnect_attempts=6
|
||||||
|
#reconnect_interval=10
|
||||||
|
|
||||||
|
# Autofailover options
|
||||||
|
#failover=manual # one of 'automatic', 'manual' (default: manual)
|
||||||
|
# defines the action to take in the event of upstream failure
|
||||||
|
#
|
||||||
|
# 'automatic': repmgrd will automatically attempt to promote the
|
||||||
|
# node or follow the new upstream node
|
||||||
|
# 'manual': repmgrd will take no action and the mode will require
|
||||||
|
# manual attention to reattach it to replication
|
||||||
|
|
||||||
|
#priority=100 # indicate a preferred priorty for promoting nodes
|
||||||
|
# a value of zero or less prevents the node being promoted to primary
|
||||||
|
# (default: 100)
|
||||||
|
|
||||||
|
#promote_command='repmgr standby promote -f /path/to/repmgr.conf'
|
||||||
|
#follow_command='repmgr standby follow -f /path/to/repmgr.conf -W'
|
||||||
|
|
||||||
|
# monitoring interval in seconds; default is 2
|
||||||
|
#monitor_interval_secs=2
|
||||||
|
|
||||||
|
# change wait time for primary; before we bail out and exit when the primary
|
||||||
|
# disappears, we wait 'reconnect_attempts' * 'retry_promote_interval_secs'
|
||||||
|
# seconds; by default this would be half an hour, as 'retry_promote_interval_secs'
|
||||||
|
# default value is 300)
|
||||||
|
#retry_promote_interval_secs=300
|
||||||
|
|
||||||
|
# Number of seconds after which the witness server resyncs the repl_nodes table
|
||||||
|
#witness_repl_nodes_sync_interval_secs=15
|
||||||
109
repmgr.h
109
repmgr.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* repmgr.h
|
* repmgr.h
|
||||||
* Copyright (c) 2ndQuadrant, 2010-2011
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -20,44 +20,105 @@
|
|||||||
#ifndef _REPMGR_H_
|
#ifndef _REPMGR_H_
|
||||||
#define _REPMGR_H_
|
#define _REPMGR_H_
|
||||||
|
|
||||||
#include "postgres_fe.h"
|
#include <libpq-fe.h>
|
||||||
#include "getopt_long.h"
|
#include <postgres_fe.h>
|
||||||
#include "libpq-fe.h"
|
#include <getopt_long.h>
|
||||||
|
|
||||||
#include "strutil.h"
|
#include "strutil.h"
|
||||||
#include "dbutils.h"
|
#include "dbutils.h"
|
||||||
#include "errcode.h"
|
#include "errcode.h"
|
||||||
|
|
||||||
#define PRIMARY_MODE 0
|
|
||||||
#define STANDBY_MODE 1
|
|
||||||
|
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#define MAXFILENAME 1024
|
#include "dirmod.h"
|
||||||
#define ERRBUFF_SIZE 512
|
|
||||||
|
#define MIN_SUPPORTED_VERSION "9.3"
|
||||||
|
#define MIN_SUPPORTED_VERSION_NUM 90300
|
||||||
|
|
||||||
|
#define ERRBUFF_SIZE 512
|
||||||
|
|
||||||
#define DEFAULT_CONFIG_FILE "./repmgr.conf"
|
|
||||||
#define DEFAULT_WAL_KEEP_SEGMENTS "5000"
|
#define DEFAULT_WAL_KEEP_SEGMENTS "5000"
|
||||||
#define DEFAULT_DEST_DIR "."
|
#define DEFAULT_DEST_DIR "."
|
||||||
#define DEFAULT_MASTER_PORT "5432"
|
|
||||||
#define DEFAULT_DBNAME "postgres"
|
|
||||||
#define DEFAULT_REPMGR_SCHEMA_PREFIX "repmgr_"
|
#define DEFAULT_REPMGR_SCHEMA_PREFIX "repmgr_"
|
||||||
|
#define DEFAULT_PRIORITY 100
|
||||||
|
#define FAILOVER_NODES_MAX_CHECK 50
|
||||||
|
|
||||||
|
#define MANUAL_FAILOVER 0
|
||||||
|
#define AUTOMATIC_FAILOVER 1
|
||||||
|
#define NODE_NOT_FOUND -1
|
||||||
|
#define NO_UPSTREAM_NODE -1
|
||||||
|
#define UNKNOWN_NODE_ID -1
|
||||||
|
|
||||||
|
#define OPT_HELP 1
|
||||||
|
#define OPT_CHECK_UPSTREAM_CONFIG 2
|
||||||
|
#define OPT_RECOVERY_MIN_APPLY_DELAY 3
|
||||||
|
#define OPT_IGNORE_EXTERNAL_CONFIG_FILES 4
|
||||||
|
#define OPT_CONFIG_ARCHIVE_DIR 5
|
||||||
|
#define OPT_PG_REWIND 6
|
||||||
|
#define OPT_PWPROMPT 7
|
||||||
|
#define OPT_CSV 8
|
||||||
|
#define OPT_INITDB_NO_PWPROMPT 9
|
||||||
|
|
||||||
|
|
||||||
/* Run time options type */
|
/* Run time options type */
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
|
|
||||||
char dbname[MAXLEN];
|
char dbname[MAXLEN];
|
||||||
char host[MAXLEN];
|
char host[MAXLEN];
|
||||||
char username[MAXLEN];
|
char username[MAXLEN];
|
||||||
char dest_dir[MAXFILENAME];
|
char dest_dir[MAXPGPATH];
|
||||||
char config_file[MAXFILENAME];
|
char config_file[MAXPGPATH];
|
||||||
char remote_user[MAXLEN];
|
char remote_user[MAXLEN];
|
||||||
char wal_keep_segments[MAXLEN];
|
char superuser[MAXLEN];
|
||||||
bool verbose;
|
char wal_keep_segments[MAXLEN];
|
||||||
bool force;
|
bool verbose;
|
||||||
|
bool terse;
|
||||||
|
bool force;
|
||||||
|
bool wait_for_master;
|
||||||
|
bool ignore_rsync_warn;
|
||||||
|
bool witness_pwprompt;
|
||||||
|
bool rsync_only;
|
||||||
|
bool fast_checkpoint;
|
||||||
|
bool ignore_external_config_files;
|
||||||
|
bool csv_mode;
|
||||||
|
char masterport[MAXLEN];
|
||||||
|
/*
|
||||||
|
* configuration file parameters which can be overridden on the
|
||||||
|
* command line
|
||||||
|
*/
|
||||||
|
char loglevel[MAXLEN];
|
||||||
|
|
||||||
char masterport[MAXLEN];
|
/* parameter used by STANDBY SWITCHOVER */
|
||||||
|
char remote_config_file[MAXLEN];
|
||||||
|
char pg_rewind[MAXPGPATH];
|
||||||
|
char pg_ctl_mode[MAXLEN];
|
||||||
|
/* parameter used by STANDBY {ARCHIVE_CONFIG | RESTORE_CONFIG} */
|
||||||
|
char config_archive_dir[MAXLEN];
|
||||||
|
/* parameter used by CLUSTER CLEANUP */
|
||||||
|
int keep_history;
|
||||||
|
|
||||||
} t_runtime_options;
|
char pg_bindir[MAXLEN];
|
||||||
|
|
||||||
|
char recovery_min_apply_delay[MAXLEN];
|
||||||
|
|
||||||
|
/* deprecated command line options */
|
||||||
|
char localport[MAXLEN];
|
||||||
|
} t_runtime_options;
|
||||||
|
|
||||||
|
#define T_RUNTIME_OPTIONS_INITIALIZER { "", "", "", "", "", "", "", DEFAULT_WAL_KEEP_SEGMENTS, false, false, false, false, false, false, false, false, false, false, "", "", "", "", "fast", "", 0, "", "", ""}
|
||||||
|
|
||||||
|
struct BackupLabel
|
||||||
|
{
|
||||||
|
XLogRecPtr start_wal_location;
|
||||||
|
char start_wal_file[MAXLEN];
|
||||||
|
XLogRecPtr checkpoint_location;
|
||||||
|
char backup_from[MAXLEN];
|
||||||
|
char backup_method[MAXLEN];
|
||||||
|
char start_time[MAXLEN];
|
||||||
|
char label[MAXLEN];
|
||||||
|
XLogRecPtr min_failover_slot_lsn;
|
||||||
|
};
|
||||||
|
|
||||||
|
extern char repmgr_schema[MAXLEN];
|
||||||
|
extern bool config_file_found;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
33
repmgr.sql
33
repmgr.sql
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* repmgr.sql
|
* repmgr.sql
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -14,8 +14,11 @@ CREATE SCHEMA repmgr;
|
|||||||
*/
|
*/
|
||||||
CREATE TABLE repl_nodes (
|
CREATE TABLE repl_nodes (
|
||||||
id integer primary key,
|
id integer primary key,
|
||||||
cluster text not null, -- Name to identify the cluster
|
cluster text not null, -- Name to identify the cluster
|
||||||
conninfo text not null
|
name text not null,
|
||||||
|
conninfo text not null,
|
||||||
|
priority integer not null,
|
||||||
|
witness boolean not null default false
|
||||||
);
|
);
|
||||||
ALTER TABLE repl_nodes OWNER TO repmgr;
|
ALTER TABLE repl_nodes OWNER TO repmgr;
|
||||||
|
|
||||||
@@ -28,13 +31,12 @@ CREATE TABLE repl_monitor (
|
|||||||
standby_node INTEGER NOT NULL,
|
standby_node INTEGER NOT NULL,
|
||||||
last_monitor_time TIMESTAMP WITH TIME ZONE NOT NULL,
|
last_monitor_time TIMESTAMP WITH TIME ZONE NOT NULL,
|
||||||
last_wal_primary_location TEXT NOT NULL,
|
last_wal_primary_location TEXT NOT NULL,
|
||||||
last_wal_standby_location TEXT NOT NULL,
|
last_wal_standby_location TEXT, -- In case of a witness server this will be NULL
|
||||||
replication_lag BIGINT NOT NULL,
|
replication_lag BIGINT NOT NULL,
|
||||||
apply_lag BIGINT NOT NULL
|
apply_lag BIGINT NOT NULL
|
||||||
);
|
);
|
||||||
ALTER TABLE repl_monitor OWNER TO repmgr;
|
ALTER TABLE repl_monitor OWNER TO repmgr;
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This view shows the latest monitor info about every node.
|
* This view shows the latest monitor info about every node.
|
||||||
* Interesting thing to see:
|
* Interesting thing to see:
|
||||||
@@ -46,14 +48,23 @@ ALTER TABLE repl_monitor OWNER TO repmgr;
|
|||||||
* time_lag: how many seconds are we from being up-to-date with master
|
* time_lag: how many seconds are we from being up-to-date with master
|
||||||
*/
|
*/
|
||||||
CREATE VIEW repl_status AS
|
CREATE VIEW repl_status AS
|
||||||
WITH monitor_info AS (SELECT *, ROW_NUMBER() OVER (PARTITION BY primary_node, standby_node
|
SELECT primary_node, standby_node, name AS standby_name, last_monitor_time, last_wal_primary_location,
|
||||||
ORDER BY last_monitor_time desc)
|
|
||||||
FROM repl_monitor)
|
|
||||||
SELECT primary_node, standby_node, last_monitor_time, last_wal_primary_location,
|
|
||||||
last_wal_standby_location, pg_size_pretty(replication_lag) replication_lag,
|
last_wal_standby_location, pg_size_pretty(replication_lag) replication_lag,
|
||||||
pg_size_pretty(apply_lag) apply_lag,
|
pg_size_pretty(apply_lag) apply_lag,
|
||||||
age(now(), last_monitor_time) AS time_lag
|
age(now(), last_monitor_time) AS time_lag
|
||||||
FROM monitor_info a
|
FROM repl_monitor JOIN repl_nodes ON standby_node = id
|
||||||
WHERE row_number = 1;
|
WHERE (standby_node, last_monitor_time) IN (SELECT standby_node, MAX(last_monitor_time)
|
||||||
|
FROM repl_monitor GROUP BY 1);
|
||||||
|
|
||||||
ALTER VIEW repl_status OWNER TO repmgr;
|
ALTER VIEW repl_status OWNER TO repmgr;
|
||||||
|
|
||||||
|
CREATE INDEX idx_repl_status_sort ON repl_monitor(last_monitor_time, standby_node);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This view shows the list of nodes with the information of which one is the upstream
|
||||||
|
* in each case (when appliable)
|
||||||
|
*/
|
||||||
|
CREATE VIEW repl_show_nodes AS
|
||||||
|
SELECT rn.id, rn.conninfo, rn.type, rn.name, rn.cluster,
|
||||||
|
rn.priority, rn.active, sq.name AS upstream_node_name
|
||||||
|
FROM repl_nodes as rn LEFT JOIN repl_nodes AS sq ON sq.id=rn.upstream_node_id;
|
||||||
|
|||||||
21
sql/Makefile
Normal file
21
sql/Makefile
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
#
|
||||||
|
# Makefile
|
||||||
|
#
|
||||||
|
# Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
|
#
|
||||||
|
|
||||||
|
MODULE_big = repmgr_funcs
|
||||||
|
DATA_built=repmgr_funcs.sql
|
||||||
|
DATA=uninstall_repmgr_funcs.sql
|
||||||
|
OBJS=repmgr_funcs.o
|
||||||
|
|
||||||
|
ifdef USE_PGXS
|
||||||
|
PG_CONFIG = pg_config
|
||||||
|
PGXS := $(shell $(PG_CONFIG) --pgxs)
|
||||||
|
include $(PGXS)
|
||||||
|
else
|
||||||
|
subdir = contrib/repmgr/sql
|
||||||
|
top_builddir = ../../..
|
||||||
|
include $(top_builddir)/src/Makefile.global
|
||||||
|
include $(top_srcdir)/contrib/contrib-global.mk
|
||||||
|
endif
|
||||||
85
sql/repmgr2_repmgr3.sql
Normal file
85
sql/repmgr2_repmgr3.sql
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
/*
|
||||||
|
* Update a repmgr 2.x installation to repmgr 3.0
|
||||||
|
* ----------------------------------------------
|
||||||
|
*
|
||||||
|
* 1. Stop any running repmgrd instances
|
||||||
|
* 2. On the master node, execute the SQL statements listed below,
|
||||||
|
* taking care to identify the master node and any inactive
|
||||||
|
* nodes
|
||||||
|
* 3. Restart repmgrd (being sure to use repmgr 3.0)
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Set the search path to the name of the schema used by
|
||||||
|
* your repmgr installation
|
||||||
|
* (this should be "repmgr_" + the cluster name defined in
|
||||||
|
* 'repmgr.conf')
|
||||||
|
*/
|
||||||
|
|
||||||
|
-- SET search_path TO 'name_of_repmgr_schema';
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE repl_nodes RENAME TO repl_nodes2_0;
|
||||||
|
|
||||||
|
CREATE TABLE repl_nodes (
|
||||||
|
id INTEGER PRIMARY KEY,
|
||||||
|
type TEXT NOT NULL CHECK (type IN('master','standby','witness')),
|
||||||
|
upstream_node_id INTEGER NULL REFERENCES repl_nodes (id),
|
||||||
|
cluster TEXT NOT NULL,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
conninfo TEXT NOT NULL,
|
||||||
|
slot_name TEXT NULL,
|
||||||
|
priority INTEGER NOT NULL,
|
||||||
|
active BOOLEAN NOT NULL DEFAULT TRUE
|
||||||
|
);
|
||||||
|
|
||||||
|
INSERT INTO repl_nodes
|
||||||
|
(id, type, cluster, name, conninfo, priority)
|
||||||
|
SELECT id,
|
||||||
|
CASE
|
||||||
|
WHEN witness IS TRUE THEN 'witness'
|
||||||
|
ELSE 'standby'
|
||||||
|
END AS type,
|
||||||
|
cluster,
|
||||||
|
name,
|
||||||
|
conninfo,
|
||||||
|
priority + 100
|
||||||
|
FROM repl_nodes2_0;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* You'll need to set the master explicitly; the following query
|
||||||
|
* should identify the master node ID but will only work if all
|
||||||
|
* standby servers are connected:
|
||||||
|
*
|
||||||
|
* SELECT id FROM repmgr_test.repl_nodes WHERE name NOT IN (SELECT application_name FROM pg_stat_replication)
|
||||||
|
*
|
||||||
|
* If in doubt, execute 'repmgr cluster show' will definitively identify
|
||||||
|
* the master.
|
||||||
|
*/
|
||||||
|
UPDATE repl_nodes SET type = 'master' WHERE id = $master_id;
|
||||||
|
|
||||||
|
/* If any nodes are known to be inactive, update them here */
|
||||||
|
|
||||||
|
-- UPDATE repl_nodes SET active = FALSE WHERE id IN (...);
|
||||||
|
|
||||||
|
/* There's also an event table which we need to create */
|
||||||
|
CREATE TABLE repl_events (
|
||||||
|
node_id INTEGER NOT NULL,
|
||||||
|
event TEXT NOT NULL,
|
||||||
|
successful BOOLEAN NOT NULL DEFAULT TRUE,
|
||||||
|
event_timestamp TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
details TEXT NULL
|
||||||
|
);
|
||||||
|
|
||||||
|
/* When you're sure of your changes, commit them */
|
||||||
|
|
||||||
|
-- COMMIT;
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* execute the following command when you are sure you no longer
|
||||||
|
* require the old table:
|
||||||
|
*/
|
||||||
|
|
||||||
|
-- DROP TABLE repl_nodes2_0;
|
||||||
35
sql/repmgr3.0_repmgr3.1.sql
Normal file
35
sql/repmgr3.0_repmgr3.1.sql
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
/*
|
||||||
|
* Update a repmgr 3.0 installation to repmgr 3.1
|
||||||
|
* ----------------------------------------------
|
||||||
|
*
|
||||||
|
* The new repmgr package should be installed first. Then
|
||||||
|
* carry out these steps:
|
||||||
|
*
|
||||||
|
* 1. (If repmgrd is used) stop any running repmgrd instances
|
||||||
|
* 2. On the master node, execute the SQL statements listed below
|
||||||
|
* 3. (If repmgrd is used) restart repmgrd
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If your repmgr installation is not included in your repmgr
|
||||||
|
* user's search path, please set the search path to the name
|
||||||
|
* of the repmgr schema to ensure objects are installed in
|
||||||
|
* the correct location.
|
||||||
|
*
|
||||||
|
* The repmgr schema is "repmgr_" + the cluster name defined in
|
||||||
|
* 'repmgr.conf'.
|
||||||
|
*/
|
||||||
|
|
||||||
|
-- SET search_path TO 'name_of_repmgr_schema';
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
-- New view "repl_show_nodes" which also displays the server's
|
||||||
|
-- upstream node
|
||||||
|
|
||||||
|
CREATE VIEW repl_show_nodes AS
|
||||||
|
SELECT rn.id, rn.conninfo, rn.type, rn.name, rn.cluster,
|
||||||
|
rn.priority, rn.active, sq.name AS upstream_node_name
|
||||||
|
FROM repl_nodes as rn LEFT JOIN repl_nodes AS sq ON sq.id=rn.upstream_node_id;
|
||||||
|
|
||||||
|
COMMIT;
|
||||||
32
sql/repmgr3.1.1_repmgr3.1.2.sql
Normal file
32
sql/repmgr3.1.1_repmgr3.1.2.sql
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
/*
|
||||||
|
* Update a repmgr 3.1.1 installation to repmgr 3.1.2
|
||||||
|
* --------------------------------------------------
|
||||||
|
*
|
||||||
|
* This update is only required if repmgrd is being used in conjunction
|
||||||
|
* with a witness server.
|
||||||
|
*
|
||||||
|
* The new repmgr package should be installed first. Then
|
||||||
|
* carry out these steps:
|
||||||
|
*
|
||||||
|
* 1. (If repmgrd is used) stop any running repmgrd instances
|
||||||
|
* 2. On the master node, execute the SQL statement listed below
|
||||||
|
* 3. (If repmgrd is used) restart repmgrd
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If your repmgr installation is not included in your repmgr
|
||||||
|
* user's search path, please set the search path to the name
|
||||||
|
* of the repmgr schema to ensure objects are installed in
|
||||||
|
* the correct location.
|
||||||
|
*
|
||||||
|
* The repmgr schema is "repmgr_" + the cluster name defined in
|
||||||
|
* 'repmgr.conf'.
|
||||||
|
*/
|
||||||
|
|
||||||
|
-- SET search_path TO 'name_of_repmgr_schema';
|
||||||
|
|
||||||
|
BEGIN;
|
||||||
|
|
||||||
|
ALTER TABLE repl_nodes DROP CONSTRAINT repl_nodes_upstream_node_id_fkey,
|
||||||
|
ADD CONSTRAINT repl_nodes_upstream_node_id_fkey FOREIGN KEY (upstream_node_id) REFERENCES repl_nodes(id) DEFERRABLE;
|
||||||
|
COMMIT;
|
||||||
245
sql/repmgr_funcs.c
Normal file
245
sql/repmgr_funcs.c
Normal file
@@ -0,0 +1,245 @@
|
|||||||
|
/*
|
||||||
|
* repmgr_funcs.c
|
||||||
|
* Copyright (c) 2ndQuadrant, 2010
|
||||||
|
*
|
||||||
|
* Shared memory state management and some backend functions in SQL
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "postgres.h"
|
||||||
|
#include "fmgr.h"
|
||||||
|
#include "access/xlog.h"
|
||||||
|
#include "miscadmin.h"
|
||||||
|
#include "replication/walreceiver.h"
|
||||||
|
#include "storage/ipc.h"
|
||||||
|
#include "storage/lwlock.h"
|
||||||
|
#include "storage/procarray.h"
|
||||||
|
#include "storage/shmem.h"
|
||||||
|
#include "storage/spin.h"
|
||||||
|
#include "utils/builtins.h"
|
||||||
|
#include "utils/timestamp.h"
|
||||||
|
|
||||||
|
/* same definition as the one in xlog_internal.h */
|
||||||
|
#define MAXFNAMELEN 64
|
||||||
|
|
||||||
|
PG_MODULE_MAGIC;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Global shared state
|
||||||
|
*/
|
||||||
|
typedef struct repmgrSharedState
|
||||||
|
{
|
||||||
|
LWLockId lock; /* protects search/modification */
|
||||||
|
char location[MAXFNAMELEN]; /* last known xlog location */
|
||||||
|
TimestampTz last_updated;
|
||||||
|
} repmgrSharedState;
|
||||||
|
|
||||||
|
/* Links to shared memory state */
|
||||||
|
static repmgrSharedState *shared_state = NULL;
|
||||||
|
|
||||||
|
static shmem_startup_hook_type prev_shmem_startup_hook = NULL;
|
||||||
|
|
||||||
|
void _PG_init(void);
|
||||||
|
void _PG_fini(void);
|
||||||
|
|
||||||
|
static void repmgr_shmem_startup(void);
|
||||||
|
static Size repmgr_memsize(void);
|
||||||
|
|
||||||
|
static bool repmgr_set_standby_location(char *locationstr);
|
||||||
|
|
||||||
|
Datum repmgr_update_standby_location(PG_FUNCTION_ARGS);
|
||||||
|
Datum repmgr_get_last_standby_location(PG_FUNCTION_ARGS);
|
||||||
|
|
||||||
|
PG_FUNCTION_INFO_V1(repmgr_update_standby_location);
|
||||||
|
PG_FUNCTION_INFO_V1(repmgr_get_last_standby_location);
|
||||||
|
|
||||||
|
Datum repmgr_update_last_updated(PG_FUNCTION_ARGS);
|
||||||
|
Datum repmgr_get_last_updated(PG_FUNCTION_ARGS);
|
||||||
|
|
||||||
|
PG_FUNCTION_INFO_V1(repmgr_update_last_updated);
|
||||||
|
PG_FUNCTION_INFO_V1(repmgr_get_last_updated);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Module load callback
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
_PG_init(void)
|
||||||
|
{
|
||||||
|
/*
|
||||||
|
* In order to create our shared memory area, we have to be loaded via
|
||||||
|
* shared_preload_libraries. If not, fall out without hooking into any of
|
||||||
|
* the main system. (We don't throw error here because it seems useful to
|
||||||
|
* allow the repmgr functions to be created even when the module isn't
|
||||||
|
* active. The functions must protect themselves against being called
|
||||||
|
* then, however.)
|
||||||
|
*/
|
||||||
|
if (!process_shared_preload_libraries_in_progress)
|
||||||
|
return;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Request additional shared resources. (These are no-ops if we're not in
|
||||||
|
* the postmaster process.) We'll allocate or attach to the shared
|
||||||
|
* resources in repmgr_shmem_startup().
|
||||||
|
*/
|
||||||
|
RequestAddinShmemSpace(repmgr_memsize());
|
||||||
|
|
||||||
|
#if (PG_VERSION_NUM >= 90600)
|
||||||
|
RequestNamedLWLockTranche("repmgr", 1);
|
||||||
|
#else
|
||||||
|
RequestAddinLWLocks(1);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Install hooks.
|
||||||
|
*/
|
||||||
|
prev_shmem_startup_hook = shmem_startup_hook;
|
||||||
|
shmem_startup_hook = repmgr_shmem_startup;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Module unload callback
|
||||||
|
*/
|
||||||
|
void
|
||||||
|
_PG_fini(void)
|
||||||
|
{
|
||||||
|
/* Uninstall hooks. */
|
||||||
|
shmem_startup_hook = prev_shmem_startup_hook;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* shmem_startup hook: allocate or attach to shared memory,
|
||||||
|
*/
|
||||||
|
static void
|
||||||
|
repmgr_shmem_startup(void)
|
||||||
|
{
|
||||||
|
bool found;
|
||||||
|
|
||||||
|
if (prev_shmem_startup_hook)
|
||||||
|
prev_shmem_startup_hook();
|
||||||
|
|
||||||
|
/* reset in case this is a restart within the postmaster */
|
||||||
|
shared_state = NULL;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Create or attach to the shared memory state, including hash table
|
||||||
|
*/
|
||||||
|
LWLockAcquire(AddinShmemInitLock, LW_EXCLUSIVE);
|
||||||
|
|
||||||
|
shared_state = ShmemInitStruct("repmgr shared state",
|
||||||
|
sizeof(repmgrSharedState),
|
||||||
|
&found);
|
||||||
|
|
||||||
|
if (!found)
|
||||||
|
{
|
||||||
|
/* First time through ... */
|
||||||
|
#if (PG_VERSION_NUM >= 90600)
|
||||||
|
shared_state->lock = &(GetNamedLWLockTranche("repmgr"))->lock;
|
||||||
|
#else
|
||||||
|
shared_state->lock = LWLockAssign();
|
||||||
|
#endif
|
||||||
|
snprintf(shared_state->location,
|
||||||
|
sizeof(shared_state->location), "%X/%X", 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
LWLockRelease(AddinShmemInitLock);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Estimate shared memory space needed.
|
||||||
|
*/
|
||||||
|
static Size
|
||||||
|
repmgr_memsize(void)
|
||||||
|
{
|
||||||
|
return MAXALIGN(sizeof(repmgrSharedState));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static bool
|
||||||
|
repmgr_set_standby_location(char *locationstr)
|
||||||
|
{
|
||||||
|
/* Safety check... */
|
||||||
|
if (!shared_state)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
LWLockAcquire(shared_state->lock, LW_EXCLUSIVE);
|
||||||
|
strncpy(shared_state->location, locationstr, MAXFNAMELEN);
|
||||||
|
LWLockRelease(shared_state->lock);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* SQL Functions */
|
||||||
|
|
||||||
|
/* Read last xlog location reported by this standby from shared memory */
|
||||||
|
Datum
|
||||||
|
repmgr_get_last_standby_location(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
char location[MAXFNAMELEN];
|
||||||
|
|
||||||
|
/* Safety check... */
|
||||||
|
if (!shared_state)
|
||||||
|
PG_RETURN_NULL();
|
||||||
|
|
||||||
|
LWLockAcquire(shared_state->lock, LW_SHARED);
|
||||||
|
strncpy(location, shared_state->location, MAXFNAMELEN);
|
||||||
|
LWLockRelease(shared_state->lock);
|
||||||
|
|
||||||
|
PG_RETURN_TEXT_P(cstring_to_text(location));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* Set update last xlog location reported by this standby to shared memory */
|
||||||
|
Datum
|
||||||
|
repmgr_update_standby_location(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
text *location = PG_GETARG_TEXT_P(0);
|
||||||
|
char *locationstr;
|
||||||
|
|
||||||
|
/* Safety check... */
|
||||||
|
if (!shared_state)
|
||||||
|
PG_RETURN_BOOL(false);
|
||||||
|
|
||||||
|
locationstr = text_to_cstring(location);
|
||||||
|
|
||||||
|
PG_RETURN_BOOL(repmgr_set_standby_location(locationstr));
|
||||||
|
}
|
||||||
|
|
||||||
|
/* update and return last updated with current timestamp */
|
||||||
|
Datum
|
||||||
|
repmgr_update_last_updated(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
TimestampTz last_updated = GetCurrentTimestamp();
|
||||||
|
|
||||||
|
/* Safety check... */
|
||||||
|
if (!shared_state)
|
||||||
|
PG_RETURN_NULL();
|
||||||
|
|
||||||
|
LWLockAcquire(shared_state->lock, LW_SHARED);
|
||||||
|
shared_state->last_updated = last_updated;
|
||||||
|
LWLockRelease(shared_state->lock);
|
||||||
|
|
||||||
|
PG_RETURN_TIMESTAMPTZ(last_updated);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/* get last updated timestamp */
|
||||||
|
Datum
|
||||||
|
repmgr_get_last_updated(PG_FUNCTION_ARGS)
|
||||||
|
{
|
||||||
|
TimestampTz last_updated;
|
||||||
|
|
||||||
|
/* Safety check... */
|
||||||
|
if (!shared_state)
|
||||||
|
PG_RETURN_NULL();
|
||||||
|
|
||||||
|
LWLockAcquire(shared_state->lock, LW_EXCLUSIVE);
|
||||||
|
last_updated = shared_state->last_updated;
|
||||||
|
LWLockRelease(shared_state->lock);
|
||||||
|
|
||||||
|
PG_RETURN_TIMESTAMPTZ(last_updated);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
23
sql/repmgr_funcs.sql.in
Normal file
23
sql/repmgr_funcs.sql.in
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
/*
|
||||||
|
* repmgr_function.sql
|
||||||
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
-- SET SEARCH_PATH TO 'repmgr';
|
||||||
|
|
||||||
|
CREATE FUNCTION repmgr_update_standby_location(text) RETURNS boolean
|
||||||
|
AS 'MODULE_PATHNAME', 'repmgr_update_standby_location'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
|
||||||
|
CREATE FUNCTION repmgr_get_last_standby_location() RETURNS text
|
||||||
|
AS 'MODULE_PATHNAME', 'repmgr_get_last_standby_location'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
|
||||||
|
CREATE FUNCTION repmgr_update_last_updated() RETURNS TIMESTAMP WITH TIME ZONE
|
||||||
|
AS 'MODULE_PATHNAME', 'repmgr_update_last_updated'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
|
|
||||||
|
CREATE FUNCTION repmgr_get_last_updated() RETURNS TIMESTAMP WITH TIME ZONE
|
||||||
|
AS 'MODULE_PATHNAME', 'repmgr_get_last_updated'
|
||||||
|
LANGUAGE C STRICT;
|
||||||
11
sql/uninstall_repmgr_funcs.sql
Normal file
11
sql/uninstall_repmgr_funcs.sql
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
/*
|
||||||
|
* uninstall_repmgr_funcs.sql
|
||||||
|
* Copyright (c) 2ndQuadrant, 2010-2016
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
DROP FUNCTION repmgr_update_standby_location(text);
|
||||||
|
DROP FUNCTION repmgr_get_last_standby_location();
|
||||||
|
|
||||||
|
DROP FUNCTION repmgr_update_last_updated();
|
||||||
|
DROP FUNCTION repmgr_get_last_updated();
|
||||||
24
strutil.c
24
strutil.c
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* strutil.c
|
* strutil.c
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
* 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
|
||||||
@@ -25,20 +25,21 @@
|
|||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "strutil.h"
|
#include "strutil.h"
|
||||||
|
|
||||||
static int xvsnprintf(char *str, size_t size, const char *format, va_list ap);
|
static int
|
||||||
|
xvsnprintf(char *str, size_t size, const char *format, va_list ap)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 0)));
|
||||||
|
|
||||||
static int
|
static int
|
||||||
xvsnprintf(char *str, size_t size, const char *format, va_list ap)
|
xvsnprintf(char *str, size_t size, const char *format, va_list ap)
|
||||||
{
|
{
|
||||||
int retval;
|
int retval;
|
||||||
|
|
||||||
retval = vsnprintf(str, size, format, ap);
|
retval = vsnprintf(str, size, format, ap);
|
||||||
|
|
||||||
if (retval >= size)
|
if (retval >= (int) size)
|
||||||
{
|
{
|
||||||
log_err(_("Buffer of size not large enough to format entire string '%s'\n"),
|
log_err(_("Buffer of size not large enough to format entire string '%s'\n"),
|
||||||
str);
|
str);
|
||||||
exit(ERR_STR_OVERFLOW);
|
exit(ERR_STR_OVERFLOW);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -47,10 +48,10 @@ xvsnprintf(char *str, size_t size, const char *format, va_list ap)
|
|||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
xsnprintf(char *str, size_t size, const char *format, ...)
|
xsnprintf(char *str, size_t size, const char *format,...)
|
||||||
{
|
{
|
||||||
va_list arglist;
|
va_list arglist;
|
||||||
int retval;
|
int retval;
|
||||||
|
|
||||||
va_start(arglist, format);
|
va_start(arglist, format);
|
||||||
retval = xvsnprintf(str, size, format, arglist);
|
retval = xvsnprintf(str, size, format, arglist);
|
||||||
@@ -61,7 +62,7 @@ xsnprintf(char *str, size_t size, const char *format, ...)
|
|||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
sqlquery_snprintf(char *str, const char *format, ...)
|
sqlquery_snprintf(char *str, const char *format,...)
|
||||||
{
|
{
|
||||||
va_list arglist;
|
va_list arglist;
|
||||||
int retval;
|
int retval;
|
||||||
@@ -74,7 +75,8 @@ sqlquery_snprintf(char *str, const char *format, ...)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int maxlen_snprintf(char *str, const char *format, ...)
|
int
|
||||||
|
maxlen_snprintf(char *str, const char *format,...)
|
||||||
{
|
{
|
||||||
va_list arglist;
|
va_list arglist;
|
||||||
int retval;
|
int retval;
|
||||||
|
|||||||
25
strutil.h
25
strutil.h
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* strutil.h
|
* strutil.h
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* This program is free software: you can redistribute it and/or modify
|
||||||
@@ -22,7 +22,8 @@
|
|||||||
#define _STRUTIL_H_
|
#define _STRUTIL_H_
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <errcode.h>
|
#include "errcode.h"
|
||||||
|
|
||||||
|
|
||||||
#define QUERY_STR_LEN 8192
|
#define QUERY_STR_LEN 8192
|
||||||
#define MAXLEN 1024
|
#define MAXLEN 1024
|
||||||
@@ -30,9 +31,21 @@
|
|||||||
#define MAXVERSIONSTR 16
|
#define MAXVERSIONSTR 16
|
||||||
#define MAXCONNINFO 1024
|
#define MAXCONNINFO 1024
|
||||||
|
|
||||||
|
/* Why? http://stackoverflow.com/a/5459929/398670 */
|
||||||
|
#define STR(x) CppAsString(x)
|
||||||
|
|
||||||
extern int xsnprintf(char *str, size_t size, const char *format, ...);
|
#define MAXLEN_STR STR(MAXLEN)
|
||||||
extern int sqlquery_snprintf(char *str, const char *format, ...);
|
|
||||||
extern int maxlen_snprintf(char *str, const char *format, ...);
|
|
||||||
|
|
||||||
#endif /* _STRUTIL_H_ */
|
extern int
|
||||||
|
xsnprintf(char *str, size_t size, const char *format,...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 3, 4)));
|
||||||
|
|
||||||
|
extern int
|
||||||
|
sqlquery_snprintf(char *str, const char *format,...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3)));
|
||||||
|
|
||||||
|
extern int
|
||||||
|
maxlen_snprintf(char *str, const char *format,...)
|
||||||
|
__attribute__((format(PG_PRINTF_ATTRIBUTE, 2, 3)));
|
||||||
|
|
||||||
|
#endif /* _STRUTIL_H_ */
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
/*
|
/*
|
||||||
* uninstall_repmgr.sql
|
* uninstall_repmgr.sql
|
||||||
*
|
*
|
||||||
* Copyright (C) 2ndQuadrant, 2010-2011
|
* Copyright (C) 2ndQuadrant, 2010-2016
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user