PostgreSql基于Standby的异步流主从复制

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
云原生数据库 PolarDB MySQL 版,通用型 2核4GB 50GB
云原生数据库 PolarDB PostgreSQL 版,标准版 2核4GB 50GB
简介:

一、概述

PostgreSQl9.0版本之后推出一个类似于Oracleactive dataguardMySql中继日志一样的日志传送。我们借助这个功能就可实现PostgreSql的主从复制。

基本原理就是,通常一台主数据库提供读写,然后把数据同步到另一台从库。从库不断apply从主库接收到的数据,从库不提供写服务,只对外提供读服务。在postgresql中提供读写全功能的服务器称为primary databasemaster database,在接收主库同步数据的同时又能提供读服务的从库服务器称为hot standby server

PostgreSQL在数据目录下的pg_xlog子目录中维护了一个WAL日志文件,该文件用于记录数据库文件的每次改变,这种日志文件机制提供了一种数据库热备份的方案,即:在把数据库使用文件系统的方式备份出来的同时也把相应的WAL日志进行备份,即使备份出来的数据块不一致,也可以重放WAL日志把备份的内容推到一致状态。这也就是基于时间点的备份(Point-in-Time Recovery),简称PITR。而把WAL日志传送到另一台服务器有两种方式,分别是:

1. WAL日志归档(base-file

2. 流复制(streaming replication

第一种是写完一个WAL日志后,才把WAL日志文件拷贝到standby数据库中,简言之就是通过cp命令实现远程备份,这样通常备库会落后主库一个WAL日志文件。而第二种流复制是postgresql9.x之后才提供的新的传递WAL日志的方法,它的好处是只要master库一产生日志,就会马上传递到standby库,同第一种相比有更低的同步延迟,所以我们肯定也会选择流复制的方式。

在实际操作之前还有一点需要说明就是standby的搭建中最关键的一步——在standby中生成master的基础备份。postgresql9.1之后提供了一个很方便的工具—— pg_basebackup,关于它的详细介绍和参数说明可以在官网中查看(pg_basebackup tool),下面在搭建过程中再做相关具体说明

二、基础环境介绍

系统平台:CentOS release 6.6 (Final)

Postgresqlpostgresql-9.6.6

SELINUX=disabled

Iptables关闭

主库(master)IP:192.168.221.161

从库(standby)IP:192.168.221.160

 

基础环境搭建可以参考前一篇文章(Centos6.6Postgresql9.6.6安装与配置),也就是PostgreSql的基本安装与配置。

三、主库配置

1. 在主库增加同步的用户名与密码

1
2
3
4
5
6
7
[postgres@MidApp ~]$ psql 
psql (9.6.6)
Type  "help"  for  help.
  
postgres= # CREATE ROLE repluser REPLICATION LOGIN PASSWORD '123456';
CREATE ROLE
postgres= #


2. 修改/home/postgres/pgsql/data/pg_hba.conf,最后一行添加

1
2
3
4
5
6
7
[root@MidApp tmp] # tail -6 /home/postgres/pgsql/data/pg_hba.conf 
# Allow replication connections from localhost, by a user with the
# replication privilege.
#local replication postgres trust
#host replication postgres 127.0.0.1/32 trust
#host replication postgres ::1/128 trust
Host replication repluser  192.168.221.160 /32  md5


这行配置意思是允许用户repluser192.168.221.160这台主机上以md5 加密的形式发起到本数据库的流复制连接

3. 在主配置文件下配置下面几个参数

1
2
3
4
5
listen_address = ‘*’(默认localhost)
wal_level = hot_standby(默认是minimal)
max_wal_senders=5(默认是0)
wal_keep_segments=64(默认是0)
synchronous_standby_names =  'standby01'


第一个参数表示监听所有IP;第二个参数表示启动hot standby;第三个参数表示主库可以有多少个并发的standby数据库,这里设置为5;第四个参数表示一个WAL日志文件大小,默认为16M

 blob.png

第五个参数指定同步复制的Standby名称(从库的recovery.conf中有要定义的地方,不过这一个参数可以不设置

4. 重启主库,让配置生效。

如果启动有报错,可以去日志排查。

四、从库配置

首先要保证主库、从库之间的同步之前的环境的是一致的,这样才方便做同步。我因为之前就在从库机器上配置过PG数据库,所以一开始走了不少弯路。最后把PG的家目录清空,重新再来一次才算成功。

1. 在从库上通过pg_basebackup命令行工具生成基础备份,命令如下,看到100%说明备份成功

1
2
3
4
[root@DB tmp] # pg_basebackup -h 192.168.221.161 -F p -P -D /home/postgres/pgsql/data -p5432 -U repluser --password
Password: 
22802 /22802  kB (100%), 1 /1  tablespace
NOTICE:  WAL archiving is not enabled; you must ensure that all required WAL segments are copied through other means to comple


参数说明:-h 指定连接的数据库IP

-F 指定输出的格式,支持p(plain原样输出)或者t(tar格式输出)

-P 在备份过程中实时打印备份进度

-D 指定备份的目录

-U 指定连接的用户名

-p 指定要连接的端口

--password 指定要连接的用户密码

其他参数介绍:

-R 会在备份后自动生成recovery.conf文件,我也是事后才知道这个参数

-l 指定一个备份的标识

具体的参数介绍可以使用pg_basebackup --help查看,也可以查看官网介绍https://www.postgresql.org/docs/current/static/app-pgbasebackup.html

特别备注:我这里没有加-R参数,所以要手动拷贝一下recovery.conf

1
cp  /home/postgres/pgsql/share/recovery .conf.sample  /home/postgres/pgsql/data/recovery .conf


添加以下信息:

1
2
standby_mode = on
primary_conninfo =  'application_name=standby01 user=repluser password=123456 host=192.168.221.161 port=5432 sslmode=disable sslcompression=1'


2. 修改从库的主配置文件/home/postgres/pgsql/data/postgresql.conf

1
hot_standby = on


hot_standby改为启用状态

3. 接下来可以启动从库了

1
2
3
[root@DB tmp] # /etc/init.d/postgresql start
Starting PostgreSQL: - bash /home/postgres/pgsql/data/serverlog : Permission denied
ok


第一次启动报错,这是因为上面生成备份的命令我使用root用户执行的,导致PG家目录的属性变成了root,所以要重新设置权限

1
chown  -R postgres:postgres  /home/postgres/pgsql/ *


再次启动正常,查看进程也OK

1
2
3
4
5
6
7
8
9
10
11
[root@DB tmp] # /etc/init.d/postgresql start
Starting PostgreSQL: ok
[root@DB tmp] # ps -ef | grep postg
root      52577  82731  0 15:12 pts /0     00:00:00  su  - postgres
postgres  52578  52577  0 15:12 pts /0     00:00:00 - bash
postgres  74295      1  0 20:01 ?        00:00:00  /home/postgres/pgsql/bin/postmaster  -D  /home/postgres/pgsql/data
postgres  74296  74295  0 20:01 ?        00:00:00 postgres: startup process   recovering 000000010000000000000003 
postgres  74297  74295  5 20:01 ?        00:00:00 postgres: wal receiver process   streaming 0 /3000140            
postgres  74298  74295  0 20:01 ?        00:00:00 postgres: checkpointer process                                  
postgres  74299  74295  0 20:01 ?        00:00:00 postgres: writer process                                        
postgres  74300  74295  0 20:01 ?        00:00:00 postgres: stats collector process

                         

五、结果验证

1. 在主库通过select usename,application_name,client_addr,state from pg_stat_replication查询一下:

1
2
3
4
5
6
7
8
9
10
11
[postgres@MidApp ~]$ psql 
psql (9.6.6)
Type  "help"  for  help.
  
postgres= # select usename,application_name,client_addr,state from pg_stat_replication;
  usename  | application_name |   client_addr   |   state   
----------+------------------+-----------------+-----------
  repluser | standby01        | 192.168.221.160 | streaming
(1 row)
  
postgres= #


可以看到192.168.221.160上的repluser在通过流复制的方式同步主库的数据

2. 创建表验证一下

主库上建表,并插入数据验证

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
postgres= # create table test01(id int primary key,note text);
CREATE TABLE
postgres= # \d
          List of relations
  Schema |  Name  | Type  |  Owner   
--------+--------+-------+----------
  public | test01 | table | postgres
(1 row)
  
postgres= # insert into test01 values(1,'1111111');
INSERT 0 1
postgres= # select * from test01;
  id  |  note   
----+---------
   1 | 1111111
(1 row)


在从库上查看:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
[postgres@DB data]$ psql 
psql (9.6.6)
Type  "help"  for  help.
  
postgres= # \d
          List of relations
  Schema |  Name  | Type  |  Owner   
--------+--------+-------+----------
  public | test01 | table | postgres
(1 row)
  
postgres= # select * from test01;
  id  |  note   
----+---------
   1 | 1111111
(1 row)


尝试插入数据看一下:

1
2
postgres= # insert into test01 values(2,'2222222');
ERROR:  cannot execute INSERT  in  read -only transaction


可以看到,从库可以查看从主库同步过来的数据,但并不能写数据。

六、总结

以上是搭建Postgresql主从同步的全过程,一路踩了好多坑,记录下了,希望能帮助到别人




本文转自 青苗飞扬 51CTO博客,原文链接:http://blog.51cto.com/qingmiao/2047284
相关实践学习
使用PolarDB和ECS搭建门户网站
本场景主要介绍基于PolarDB和ECS实现搭建门户网站。
阿里云数据库产品家族及特性
阿里云智能数据库产品团队一直致力于不断健全产品体系,提升产品性能,打磨产品功能,从而帮助客户实现更加极致的弹性能力、具备更强的扩展能力、并利用云设施进一步降低企业成本。以云原生+分布式为核心技术抓手,打造以自研的在线事务型(OLTP)数据库Polar DB和在线分析型(OLAP)数据库Analytic DB为代表的新一代企业级云原生数据库产品体系, 结合NoSQL数据库、数据库生态工具、云原生智能化数据库管控平台,为阿里巴巴经济体以及各个行业的企业客户和开发者提供从公共云到混合云再到私有云的完整解决方案,提供基于云基础设施进行数据从处理、到存储、再到计算与分析的一体化解决方案。本节课带你了解阿里云数据库产品家族及特性。
相关文章
|
负载均衡 监控 关系型数据库
百度搜索:蓝易云【PostgreSQL 主从复制方案】
请注意,上述仅为一种主从复制方案的概述,实际实施时可能需要根据特定环境和需求进行调整。建议参考PostgreSQL官方文档和其他可靠资源获取更详细的指南和说明。
121 1
|
弹性计算 网络协议 容灾
PostgreSQL 时间点恢复(PITR)在异步流复制主从模式下,如何避免主备切换后PITR恢复(备库、容灾节点、只读节点)走错时间线(timeline , history , partial , restore_command , recovery.conf)
标签 PostgreSQL , 恢复 , 时间点恢复 , PITR , restore_command , recovery.conf , partial , history , 任意时间点恢复 , timeline , 时间线 背景 政治正确非常重要,对于数据库来说亦如此,一个基于流复制的HA架构的集群,如果还有一堆只读节点,当HA集群发生了主备切换后,这些只读节点能否与新的主节点保持
1837 0
|
8月前
|
缓存 负载均衡 关系型数据库
postgresql|数据库|centos7下基于postgresql-12的主从复制的pgpool-4.4的部署和使用
postgresql|数据库|centos7下基于postgresql-12的主从复制的pgpool-4.4的部署和使用
198 0
|
8月前
|
SQL 安全 关系型数据库
postgresql|数据库|【postgresql-12的基于pg_basebackup的主从复制部署】
postgresql|数据库|【postgresql-12的基于pg_basebackup的主从复制部署】
162 0
|
SQL 存储 关系型数据库
PostgreSQL 流复制搭建主从环境,同步和异步的解释,压力测试,主从角色切换|学习笔记
快速学习PostgreSQL 流复制搭建主从环境,同步和异步的解释,压力测试,主从角色切换
PostgreSQL 流复制搭建主从环境,同步和异步的解释,压力测试,主从角色切换|学习笔记
|
关系型数据库 测试技术 数据库
PostgreSQL pg_rewind,时间线修复,脑裂修复,flashback - 从库开启读写后,回退为只读从库。异步主从发生角色切换后,主库rewind为新主库的从库
PostgreSQL pg_rewind,时间线修复,脑裂修复,flashback - 从库开启读写后,回退为只读从库。异步主从发生角色切换后,主库rewind为新主库的从库
2250 1
|
SQL Oracle 关系型数据库
PostgreSQL pg_rewind,时间线修复,脑裂修复,flashback - 从库开启读写后,回退为只读从库。异步主从发生角色切换后,主库rewind为新主库的从库
标签 PostgreSQL , pg_rewind , 主从切换 , 时间线修复 , 脑裂修复 , 从库开启读写后,回退为只读从库 , 异步主从发生角色切换后,主库rewind为新主库的从库 背景 1、PG物理流复制的从库,当激活后,可以开启读写,使用pg_rewind可以将从库回退为只读从库的角色。而不需要重建整个从库。 2、当异步主从发生角色切换后,主库的wal目录中可能还有没完全
1157 0
|
SQL 弹性计算 Oracle
PostgreSQL 使用逻辑decode实现异步主从切换后,时间线分歧变化量补齐、修复
标签 PostgreSQL , pg_rewind , 时间线 , 变化量 , 业务补齐 背景 pg_rewind类似Oracle flashback,可以将一个数据库回退到一个以前的状态,例如用于: 1、PG物理流复制的从库,当激活后,可以开启读写,使用pg_rewind可以将从库回退为只读从库的角色。而不需要重建整个从库。 2、当异步主从发生角色切换后,主库的wal目录中可能还有
1085 0
|
弹性计算 关系型数据库 数据库
PostgreSQL 如何让心跳永远不死,支持半同步自动同步、异步升降级 - udf 心跳
标签 PostgreSQL , 同步 , 半同步 , 流复制 , 心跳 , 自动降级 , 自动升级 , dblink , 异步调用 背景 在心跳时,通过自定义UDF,实现心跳永远不被堵塞,并且支持更加当前的配置自动的进行同步、异步模式的升降级。实现半同步的功能。 UDF输入 1、优先模式(同步、异步) 2、同步等待超时时间 当优先为同步模式时,假设当前为同步配置,如果备库异常导致
1946 0