- Notifications
You must be signed in to change notification settings - Fork16
Block-level incremental backup engine for PostgreSQL
License
postgrespro/ptrack
Folders and files
Name | Name | Last commit message | Last commit date | |
---|---|---|---|---|
Repository files navigation
Ptrack is a block-level incremental backup engine for PostgreSQL. You caneffectively useptrack
engine for taking incremental backups withpg_probackup backup and recovery manager for PostgreSQL.
It is designed to allow false positives (i.e. block/page is marked in theptrack
map, but actually has not been changed), but to never allow false negatives (i.e. loosing anyPGDATA
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 theptrack
extension, while the core patch contains only certain hooks and modifies binary utilities to ignoreptrack.map.*
files.
This extension is compatible with PostgreSQL11,12,13,14,15.
- Specify the PostgreSQL branch to work with:
export PG_BRANCH=REL_15_STABLE
- Get the latest PostgreSQL sources:
git clone https://github.com/postgres/postgres.git -b$PG_BRANCH
- Get the latest
ptrack
sources:
git clone https://github.com/postgrespro/ptrack.git postgres/contrib/ptrack
- Change to the
ptrack
directory:
cd postgres/contrib/ptrack
- Apply the PostgreSQL core patch:
make patch
- Compile and install PostgreSQL:
make install-postgres prefix=$PWD/pgsql# or some other prefix of your choice
- Add the newly created binaries to the PATH:
export PATH=$PWD/pgsql/bin:$PATH
- Compile and install
ptrack
:
make install USE_PGXS=1
- Set
ptrack.map_size
(in MB):
echo"shared_preload_libraries = 'ptrack'">><DATA_DIR>/postgresql.confecho"ptrack.map_size = 64">><DATA_DIR>/postgresql.conf
- Run PostgreSQL and create the
ptrack
extension:
postgres=# CREATE EXTENSION ptrack;
The only one configurable option isptrack.map_size
(in MB). Default is0
, which meansptrack
is turned off. In order to reduce number of false positives it is recommended to setptrack.map_size
to1 / 1000
of expectedPGDATA
size (i.e.1000
for a 1 TB database).
To disableptrack
and clean up all remaining service files setptrack.map_size
to0
.
- 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.4(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 ofptrack
and doALTER 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 theptrack.map
using pre 2.2 versions ofptrack
is incompatible with newer versions. After extension upgrade and server restart oldptrack.map
will be discarded withWARNING
and initialized from the scratch.
- Stop your server
- Update ptrack binaries
- Remove global/ptrack.map.mmap if it exist in server data directory
- Start server
- Do
ALTER EXTENSION ptrack UPDATE;
.
- Stop your server
- Update ptrack binaries
- Start server
- Do
ALTER EXTENSION ptrack UPDATE;
.
You can only use
ptrack
safely withwal_level >= 'replica'
. Otherwise, you can lose tracking of some changes if crash-recovery occurs, sincecertain 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
ispg_probackup.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.You will need up to
ptrack.map_size * 2
of additional disk space, sinceptrack
uses additional temporary file for durability purpose. SeeArchitecture section for details.
Briefly, an overhead of usingptrack
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 incrementalptrack
backup of a database with only 20% of changed pages will be 5 times faster than a full backup. More detailshere.
We use a single shared hash table inptrack
. 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 enoughptrack.map_size
.
All reads/writes are made using atomic operations onuint64
entries, so the map is completely lockless during the normal PostgreSQL operation. Because we do not use locks for read/write access,ptrack
keeps a map (ptrack.map
) since the last checkpoint intact and uses up to 1 additional temporary file:
- 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 inptrack_get_pagemapset()
we walk the entirePGDATA
(base/**/*
,global/*
,pg_tblspc/**/*
) and verify using map whether each block of each relation was modified since the specified LSN or not.
Feel free tosend a pull request,create an issue orreach us by e-mail if you are interested inptrack
.
All changes of the source code in this repository are checked by CI - see commit statuses and the project status badge. You can also run tests locally by executing a few Makefile targets.
To run Python tests install the following packages:
OS packages:
- python3-pip
- python3-six
- python3-pytest
- python3-pytest-xdist
PIP packages:
- testgres
For example, for Ubuntu:
sudo apt updatesudo apt install python3-pip python3-six python3-pytest python3-pytest-xdistsudo pip3 install testgres
Install PostgreSQL and ptrack as described inInstallation, install the testing prerequisites, then do (assuming the current directory isptrack
):
git clone https://github.com/postgrespro/pg_probackup.git ../pg_probackup# clone the repository into postgres/contrib/pg_probackup# remember to export PATH=/path/to/pgsql/bin:$PATHmake install-pg-probackup USE_PGXS=1 top_srcdir=../..make test-tap USE_PGXS=1make test-python
Ifpg_probackup
is not located inpostgres/contrib
then additionally specify the path to thepg_probackup
directory when buildingpg_probackup
:
make install-pg-probackup USE_PGXS=1 top_srcdir=/path/to/postgres pg_probackup_dir=/path/to/pg_probackup
You can use a public Docker image which already has the necessary build environment (but not the testing prerequisites):
docker run -e USER_ID=`id -u` -it -v$PWD:/work --name=ptrack ghcr.io/postgres-dev/ubuntu-22.04:1.0dev@a033797d2f73:~$
Variable | Possible values | Required | Default value | Description |
---|---|---|---|---|
NPROC | An integer greater than 0 | No | Output ofnproc | The number of threads used for building and running tests |
PG_CONFIG | File path | No | pg_config (from the PATH) | The path to thepg_config binary |
TESTS | A Pytest filter expression | No | Not set (run all Python tests) | A filter to include only selected tests into the run. See the Pytest-k option for more information. This variable is only applicable totest-python for the tests located intests. |
TEST_MODE | normal, legacy, paranoia | No | normal | The "legacy" mode runs tests in an environment similar to a 32-bit Windows system. This mode is only applicable totest-tap . The "paranoia" mode compares the checksums of each block of the database catalog (PGDATA) contents before making a backup and after the restoration. This mode is only applicable totest-python . |
- 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 write a formal proof, that we never loose any modified page with
ptrack
? With TLA+?
About
Block-level incremental backup engine for PostgreSQL
Topics
Resources
License
Uh oh!
There was an error while loading.Please reload this page.
Stars
Watchers
Forks
Packages0
Uh oh!
There was an error while loading.Please reload this page.