Ptrack is a block-level incremental backup engine for PostgreSQL. You can effectively use ptrack
engine for taking incremental backups with pg_probackup backup and recovery manager for PostgreSQL.
It is designed to allow false positives (i.e. block/page is marked in the ptrack
map, but actually has not been changed), but to never allow false negatives (i.e. loosing any PGDATA
changes, excepting hint-bits).
Currently, ptrack
codebase is split between small PostgreSQL core patch and extension. All public SQL API methods and main engine are placed in the ptrack
extension, while the core patch contains only certain hooks and modifies binary utilities to ignore ptrack.map.*
files.
This extension is compatible with PostgreSQL 11, 12, 13, 14.
- Get latest
ptrack
sources:
git clone https://github.com/postgrespro/ptrack.git
- Get latest PostgreSQL sources:
git clone https://github.com/postgres/postgres.git -b REL_14_STABLE && cd postgres
- Apply PostgreSQL core patch:
git apply -3 ../ptrack/patches/REL_14_STABLE-ptrack-core.diff
-
Compile and install PostgreSQL
-
Set
ptrack.map_size
(in MB)
echo "shared_preload_libraries = 'ptrack'" >> postgres_data/postgresql.conf
echo "ptrack.map_size = 64" >> postgres_data/postgresql.conf
- Compile and install
ptrack
extension
USE_PGXS=1 make -C /path/to/ptrack/ install
- Run PostgreSQL and create
ptrack
extension
postgres=# CREATE EXTENSION ptrack;
The only one configurable option is ptrack.map_size
(in MB). Default is -1
, which means ptrack
is turned off. In order to reduce number of false positives it is recommended to set ptrack.map_size
to 1 / 1000
of expected PGDATA
size (i.e. 1000
for a 1 TB database).
To disable ptrack
and clean up all remaining service files set ptrack.map_size
to 0
.
- ptrack_version() — returns ptrack version string.
- ptrack_init_lsn() — returns LSN of the last ptrack map initialization.
- ptrack_get_pagemapset(start_lsn pg_lsn) — returns a set of changed data files with a number of changed blocks and their bitmaps since specified
start_lsn
. - ptrack_get_change_stat(start_lsn pg_lsn) — returns statistic of changes (number of files, pages and size in MB) since specified
start_lsn
.
Usage example:
postgres=# SELECT ptrack_version();
ptrack_version
----------------
2.2
(1 row)
postgres=# SELECT ptrack_init_lsn();
ptrack_init_lsn
-----------------
0/1814408
(1 row)
postgres=# SELECT * FROM ptrack_get_pagemapset('0/185C8C0');
path | pagecount | pagemap
---------------------+-----------+----------------------------------------
base/16384/1255 | 3 | \x001000000005000000000000
base/16384/2674 | 3 | \x0000000900010000000000000000
base/16384/2691 | 1 | \x00004000000000000000000000
base/16384/2608 | 1 | \x000000000000000400000000000000000000
base/16384/2690 | 1 | \x000400000000000000000000
(5 rows)
postgres=# SELECT * FROM ptrack_get_change_stat('0/285C8C8');
files | pages | size, MB
-------+-------+------------------------
20 | 25 | 0.19531250000000000000
(1 row)
Usually, you have to only install new version of ptrack
and do ALTER EXTENSION 'ptrack' UPDATE;
. However, some specific actions may be required as well:
- Put
shared_preload_libraries = 'ptrack'
intopostgresql.conf
. - Rename
ptrack_map_size
toptrack.map_size
. - Do
ALTER EXTENSION 'ptrack' UPDATE;
. - Restart your server.
Since version 2.2 we use a different algorithm for tracking changed pages. Thus, data recorded in the ptrack.map
using pre 2.2 versions of ptrack
is incompatible with newer versions. After extension upgrade and server restart old ptrack.map
will be discarded with WARNING
and initialized from the scratch.
-
You can only use
ptrack
safely withwal_level >= 'replica'
. Otherwise, you can lose tracking of some changes if crash-recovery occurs, since certain commands are designed not to write WAL at all if wal_level is minimal, but we only durably flushptrack
map at checkpoint time. -
The only one production-ready backup utility, that fully supports
ptrack
is pg_probackup. -
Currently, you cannot resize
ptrack
map in runtime, only on postmaster start. Also, you will loose all tracked changes, so it is recommended to do so in the maintainance window and accompany this operation with full backup. See TODO for details. -
You will need up to
ptrack.map_size * 3
of additional disk space, sinceptrack
uses two additional temporary files for durability purpose. See Architecture section for details.
Briefly, an overhead of using ptrack
on TPS usually does not exceed a couple of percent (~1-3%) for a database of dozens to hundreds of gigabytes in size, while the backup time scales down linearly with backup size with a coefficient ~1. It means that an incremental ptrack
backup of a database with only 20% of changed pages will be 5 times faster than a full backup. More details here.
We use a single shared hash table in ptrack
, which is mapped in memory from the file on disk using mmap
. Due to the fixed size of the map there may be false positives (when some block is marked as changed without being actually modified), but not false negative results. However, these false postives may be completely eliminated by setting a high enough ptrack.map_size
.
All reads/writes are made using atomic operations on uint64
entries, so the map is completely lockless during the normal PostgreSQL operation. Because we do not use locks for read/write access and cannot control mmap
eviction back to disk, ptrack
keeps a map (ptrack.map
) since the last checkpoint intact and uses up to 2 additional temporary files:
- working copy
ptrack.map.mmap
for doingmmap
on it (there is a TODO item); - temporary file
ptrack.map.tmp
to durably replaceptrack.map
during checkpoint.
Map is written on disk at the end of checkpoint atomically block by block involving the CRC32 checksum calculation that is checked on the next whole map re-read after crash-recovery or restart.
To gather the whole changeset of modified blocks in ptrack_get_pagemapset()
we walk the entire PGDATA
(base/**/*
, global/*
, pg_tblspc/**/*
) and verify using map whether each block of each relation was modified since the specified LSN or not.
Feel free to send pull requests, fill up issues, or just reach one of us directly (e.g. <Alexey Kondratov, @ololobus>) if you are interested in ptrack
.
Everything is tested automatically with travis-ci.com and codecov.io, but you can also run tests locally via Docker
:
export PG_BRANCH=REL_14_STABLE
export TEST_CASE=all
export MODE=paranoia
./make_dockerfile.sh
docker-compose build
docker-compose run tests
Available test modes (MODE
) are basic
(default) and paranoia
(per-block checksum comparison of PGDATA
content before and after backup-restore process). Available test cases (TEST_CASE
) are tap
(minimalistic PostgreSQL tap test), all
or any specific pg_probackup test, e.g. test_ptrack_simple
.
- Use POSIX
shm_open()
instead ofopen()
to do not create an additional working copy ofptrack
map file. - Should we introduce
ptrack.map_path
to allowptrack
service files storage outside ofPGDATA
? Doing that we will avoid patching PostgreSQL binary utilities to ignoreptrack.map.*
files. - Can we resize
ptrack
map on restart but keep the previously tracked changes? - Can we resize
ptrack
map dynamicaly? - Can we write a formal proof, that we never loose any modified page with
ptrack
? With TLA+?