背景和原理

有没有被突发的IO惊到过,有没有见到过大量的autovacuum for prevent wrap。
PostgreSQL 的版本冻结是一个比较蛋疼的事情,为什么要做版本冻结呢?
因为PG的版本号是uint32的,是重复使用的,所以每隔大约20亿个事务后,必须要冻结,否则记录会变成未来的,对当前事务”不可见”。
冻结的事务号是2

  1. src/include/access/transam.h
  2. #define InvalidTransactionId ((TransactionId) 0)
  3. #define BootstrapTransactionId ((TransactionId) 1)
  4. #define FrozenTransactionId ((TransactionId) 2)
  5. #define FirstNormalTransactionId ((TransactionId) 3)
  6. #define MaxTransactionId ((TransactionId) 0xFFFFFFFF)

现在,还可以通过行的t_infomask来区分行是否为冻结行

  1. src/include/access/htup_details.h
  2. /*
  3. * information stored in t_infomask:
  4. */
  5. #define HEAP_XMIN_COMMITTED 0x0100 /* t_xmin committed */
  6. #define HEAP_XMIN_INVALID 0x0200 /* t_xmin invalid/aborted */
  7. #define HEAP_XMIN_FROZEN (HEAP_XMIN_COMMITTED|HEAP_XMIN_INVALID)

表的最老事务号则是记录在pg_class.relfrozenxid里面的。

执行vacuum freeze table,除了修改t_infomask,还需要修改该表对应的pg_class.relfrozenxid的值。

那么系统什么时候会触发对表进行冻结呢?

当表的年龄大于autovacuum_freeze_max_age时(默认是2亿),autovacuum进程会自动对表进行freeze。
freeze后,还可以清除掉比整个集群的最老事务号早的clog文件。
那么可能会出现这样的情形:
可能有很多大表的年龄会先后到达2亿,数据库的autovacuum会开始对这些表依次进行vacuum freeze,从而集中式的爆发大量的读IO(DATAFILE)和写IO(DATAFILE以及XLOG)。
如果又碰上业务高峰,会出现很不好的影响。

为什么集中爆发Freeze很常见?
因为默认情况下,所有表的autovacuum_freeze_max_age是一样的,并且大多数的业务,一个事务或者相邻的事务都会涉及多个表的操作,所以这些大表的最老的事务号可能都是相差不大的。
这样,就有非常大的概率导致很多表的年龄是相仿的,从而导致集中的爆发多表的autovacuum freeze。

PostgreSQL有什么机制能尽量的减少多个表的年龄相仿吗?
目前来看,有一个机制,也许能降低年龄相仿性,但是要求表有发生UPDATE,对于只有INSERT的表无效。
vacuum_freeze_min_age 这个参数,当发生vacuum或者autovacuum时,扫过的记录,只要年龄大于它,就会置为freeze。因此有一定的概率可以促使频繁更新的表年龄不一致。

那么还有什么手段能放在或者尽量避免大表的年龄相仿呢?
为每个表设置不同的autovacuum_freeze_max_age值,从认为的错开来进行vacuum freeze的时机。
例如有10个大表,把全局的autovacuum_freeze_max_age设置为5亿,然后针对这些表,从2亿开始每个表间隔1000万事务设置autovacuum_freeze_max_age。 如2亿,2.1亿,2.2亿,2.3亿,2.4亿….2.9亿。
除非这些表同时达到 2亿,2.1亿,2.2亿,2.3亿,2.4亿….2.9亿。 否则不会出现同时需要vacuum freeze的情况。

但是,如果有很多大表,这样做可能就不太合适了。
建议还是人为的在业务空闲时间,对大表进行vacuum freeze。

优化建议

  1. 分区,把大表分成小表。每个表的数据量取决于系统的IO能力,前面说了VACUUM FREEZE是扫全表的, 现代的硬件每个表建议不超过32GB。
  2. 对大表设置不同的vacuum年龄. alter table t set (autovacuum_freeze_max_age=xxxx);
  3. 用户自己调度 freeze,如在业务低谷的时间窗口,对年龄较大,数据量较大的表进行vacuum freeze。
  4. 年龄只能降到系统存在的最早的长事务即 min pg_stat_activity.(backend_xid, backend_xmin)。 因此也需要密切关注长事务。

讲完了Freeze的背景,接下来给大家讲讲如何预测Freeze IO风暴。

预测 IO 风暴

如何预测此类(prevent wrapped vacuum freeze) IO 风暴的来临呢?
首先需要测量几个维度的值。

  1. 表的大小以及距离它需要被强制vacuum freeze prevent wrap的年龄
  2. 每隔一段时间的XID值的采样(例如每分钟一次),采样越多越好,因为需要用于预测下一个时间窗口的XID。(其实就是每分钟消耗多少个事务号的数据)
  3. 通过第二步得到的结果,预测下一个时间窗口的每分钟的pXID(可以使用线性回归来进行预测) 预测方法这里不在细说,也可以参考我以前写的一些预测类的文章。

预测的结论包括”未来一段时间的总Freeze IO量,以及分时的Freeze IO量”。
预测结果范例
Freeze IO 时段总量
screenshot Freeze IO 分时走势
_

预测过程

  • 每隔一段时间的XID值的采样(例如每分钟一次),采样越多越好,因为需要用于预测下一个时间窗口的XID。(其实就是每分钟消耗多少个事务号的数据)
  1. vi xids.sh
  2. #!/bin/bash
  3. export PATH=/home/digoal/pgsql9.5/bin:$PATH
  4. export PGHOST=127.0.0.1
  5. export PGPORT=1921
  6. export PGDATABASE=postgres
  7. export PGUSER=postgres
  8. export PGPASSWORD=postgres
  9. psql -c "create table xids(crt_time timestamp, xids int8)"
  10. for ((i=1;i>0;))
  11. do
  12. # 保留1个月的数据
  13. psql -c "with a as (select ctid from xids order by crt_time desc limit 100 offset 43200) delete from xids where ctid in (select ctid from a);"
  14. psql -c "insert into xids values (now(), txid_current());"
  15. sleep 60
  16. done
  17. chmod 500 xids.sh
  18. nohup ./xids.sh >/dev/null 2>&1 &

采集1天的数据可能是这样的

  1. postgres=# select * from xids ;
  2. crt_time | xids
  3. ----------------------------+------
  4. 2016-06-12 12:36:13.201315 | 2020
  5. 2016-06-12 12:37:13.216002 | 9021
  6. 2016-06-12 12:38:13.240739 | 21022
  7. 2016-06-12 12:39:13.259203 | 32023
  8. 2016-06-12 12:40:13.300604 | 42024
  9. 2016-06-12 12:41:13.325874 | 52025
  10. 2016-06-12 12:42:13.361152 | 62026
  11. 2016-06-12 12:43:15.481609 | 72027
  12. ...
  • 表的大小以及距离它需要被强制vacuum freeze prevent wrap的年龄(因为freeze是全集群的,所以需要把所有库得到的数据汇总到一起)
  1. vi pred_io.sh
  2. #!/bin/bash
  3. export PATH=/home/digoal/pgsql9.5/bin:$PATH
  4. export PGHOST=127.0.0.1
  5. export PGPORT=1921
  6. export PGDATABASE=postgres
  7. export PGUSER=postgres
  8. export PGPASSWORD=postgres
  9. psql -c "drop table pred_io; create table pred_io(crt_time timestamp, bytes int8, left_live int8);"
  10. for db in `psql -A -t -q -c "select datname from pg_database where datname <> 'template0'"`
  11. do
  12. psql -d $db -c " copy (
  13. select now(), bytes, case when max_age>age then max_age-age else 0 end as xids from
  14. (select block_size*relpages bytes,
  15. case when d_max_age is not null and d_max_age<max_age then d_max_age else max_age end as max_age,
  16. age from
  17. (select
  18. (select setting from pg_settings where name='block_size')::int8 as block_size,
  19. (select setting from pg_settings where name='autovacuum_freeze_max_age')::int8 as max_age,
  20. relpages,
  21. substring(reloptions::text,'autovacuum_freeze_max_age=(\d+)')::int8 as d_max_age,
  22. age(relfrozenxid) age
  23. from pg_class where relkind in ('r', 't')) t) t
  24. ) to stdout;" | psql -d $PGDATABASE -c "copy pred_io from stdin"
  25. done
  26. . ./pred_io.sh

得到的数据可能是这样的

  1. postgres=# select * from pred_io limit 10;
  2. crt_time | bytes | left_live
  3. ----------------------------+--------+-----------
  4. 2016-06-12 13:24:08.666995 | 131072 | 199999672
  5. 2016-06-12 13:24:08.666995 | 65536 | 199999672
  6. 2016-06-12 13:24:08.666995 | 0 | 199999672
  7. 2016-06-12 13:24:08.666995 | 0 | 199999672
  8. 2016-06-12 13:24:08.666995 | 0 | 199999672
  9. 2016-06-12 13:24:08.666995 | 0 | 199999672
  10. ...
  • 预测XIDs走势(略),本文直接取昨天的同一时间点开始后的数据。
  1. create view v_pred_xids as
  2. with b as (select min(crt_time) tbase from pred_io),
  3. a as (select crt_time + interval '1 day' as crt_time, xids from xids,b where crt_time >= b.tbase - interval '1 day')
  4. select crt_time, xids - (select min(xids) from a) as xids from a ;

数据可能是这样的,预测未来分时的相对XIDs消耗量

  1. crt_time | xids
  2. ----------------------------+------
  3. 2016-06-13 12:36:13.201315 | 0
  4. 2016-06-13 12:37:13.216002 | 100
  5. 2016-06-13 12:38:13.240739 | 200
  6. 2016-06-13 12:39:13.259203 | 300
  7. 2016-06-13 12:40:13.300604 | 400
  • 结合pred_io与v_pred_xids 进行 io风暴预测
    基准视图,后面的数据通过这个基准视图得到
  1. create view pred_tbased_io as
  2. with a as (select crt_time, xids as s_xids, lead(xids) over(order by crt_time) as e_xids from v_pred_xids)
  3. select a.crt_time, sum(b.bytes) bytes from a, pred_io b where b.left_live >=a.s_xids and b.left_live < a.e_xids group by a.crt_time order by a.crt_time;

未来一天的总freeze io bytes预测

  1. postgres=# select min(crt_time),max(crt_time),sum(bytes) from pred_tbased_io ;
  2. min | max | sum
  3. ----------------------------+----------------------------+----------
  4. 2016-06-13 12:36:13.201315 | 2016-06-14 12:35:26.104025 | 19685376
  5. (1 row)

未来一天的freeze io bytes分时走势
得到的结果可能是这样的

  1. postgres=# select * from pred_tbased_io ;
  2. crt_time | bytes
  3. ----------------------------+----------
  4. 2016-06-13 12:36:13.201315 | 65536
  5. 2016-06-13 12:37:13.216002 | 581632
  6. 2016-06-13 12:38:13.240739 | 0
  7. 2016-06-13 12:39:13.259203 | 0
  8. 2016-06-13 12:40:13.300604 | 0
  9. 2016-06-13 12:41:13.325874 | 0
  10. 2016-06-13 12:43:15.481609 | 106496
  11. 2016-06-13 12:43:24.133055 | 8192
  12. 2016-06-13 12:45:24.193318 | 0
  13. 2016-06-13 12:46:24.225559 | 16384
  14. 2016-06-13 12:48:24.296223 | 13434880
  15. 2016-06-13 12:49:24.325652 | 24576
  16. 2016-06-13 12:50:24.367232 | 401408
  17. 2016-06-13 12:51:24.426199 | 0
  18. 2016-06-13 12:52:24.457375 | 393216
  19. ......

小结

预测主要用到哪些PostgreSQL的手段?

  1. 线性回归
  2. with语法
  3. 窗口函数
  4. xid分时消耗统计
  5. 强制prevent wrap freeze vacuum的剩余XIDs统计