mirror of
https://github.com/postgrespro/pg_probackup.git
synced 2025-01-08 13:44:32 +02:00
618 lines
20 KiB
Python
618 lines
20 KiB
Python
# coding: utf-8
|
|
|
|
import unittest
|
|
import os
|
|
from .helpers.ptrack_helpers import ProbackupTest
|
|
|
|
module_name = "merge"
|
|
|
|
|
|
class MergeTest(ProbackupTest, unittest.TestCase):
|
|
|
|
def test_merge_full_page(self):
|
|
"""
|
|
Test MERGE command, it merges FULL backup with target PAGE backups
|
|
"""
|
|
fname = self.id().split(".")[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, "backup")
|
|
|
|
# Initialize instance and backup directory
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
initdb_params=["--data-checksums"]
|
|
)
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, "node", node)
|
|
self.set_archiving(backup_dir, "node", node)
|
|
node.start()
|
|
|
|
# Do full backup
|
|
self.backup_node(backup_dir, "node", node)
|
|
show_backup = self.show_pb(backup_dir, "node")[0]
|
|
|
|
self.assertEqual(show_backup["status"], "OK")
|
|
self.assertEqual(show_backup["backup-mode"], "FULL")
|
|
|
|
# Fill with data
|
|
with node.connect() as conn:
|
|
conn.execute("create table test (id int)")
|
|
conn.execute(
|
|
"insert into test select i from generate_series(1,10) s(i)")
|
|
conn.commit()
|
|
|
|
# Do first page backup
|
|
self.backup_node(backup_dir, "node", node, backup_type="page")
|
|
show_backup = self.show_pb(backup_dir, "node")[1]
|
|
|
|
# sanity check
|
|
self.assertEqual(show_backup["status"], "OK")
|
|
self.assertEqual(show_backup["backup-mode"], "PAGE")
|
|
|
|
# Fill with data
|
|
with node.connect() as conn:
|
|
conn.execute(
|
|
"insert into test select i from generate_series(1,10) s(i)")
|
|
count1 = conn.execute("select count(*) from test")
|
|
conn.commit()
|
|
|
|
# Do second page backup
|
|
self.backup_node(backup_dir, "node", node, backup_type="page")
|
|
show_backup = self.show_pb(backup_dir, "node")[2]
|
|
page_id = show_backup["id"]
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
# sanity check
|
|
self.assertEqual(show_backup["status"], "OK")
|
|
self.assertEqual(show_backup["backup-mode"], "PAGE")
|
|
|
|
# Merge all backups
|
|
self.merge_backup(backup_dir, "node", page_id,
|
|
options=["-j", "4"])
|
|
show_backups = self.show_pb(backup_dir, "node")
|
|
|
|
# sanity check
|
|
self.assertEqual(len(show_backups), 1)
|
|
self.assertEqual(show_backups[0]["status"], "OK")
|
|
self.assertEqual(show_backups[0]["backup-mode"], "FULL")
|
|
|
|
# Drop node and restore it
|
|
node.cleanup()
|
|
self.restore_node(backup_dir, 'node', node)
|
|
|
|
# Check physical correctness
|
|
if self.paranoia:
|
|
pgdata_restored = self.pgdata_content(
|
|
node.data_dir, ignore_ptrack=False)
|
|
self.compare_pgdata(pgdata, pgdata_restored)
|
|
|
|
node.slow_start()
|
|
|
|
# Check restored node
|
|
count2 = node.execute("postgres", "select count(*) from test")
|
|
self.assertEqual(count1, count2)
|
|
|
|
# Clean after yourself
|
|
node.cleanup()
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
def test_merge_compressed_backups(self):
|
|
"""
|
|
Test MERGE command with compressed backups
|
|
"""
|
|
fname = self.id().split(".")[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, "backup")
|
|
|
|
# Initialize instance and backup directory
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
initdb_params=["--data-checksums"]
|
|
)
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, "node", node)
|
|
self.set_archiving(backup_dir, "node", node)
|
|
node.start()
|
|
|
|
# Do full compressed backup
|
|
self.backup_node(backup_dir, "node", node, options=[
|
|
'--compress-algorithm=zlib'])
|
|
show_backup = self.show_pb(backup_dir, "node")[0]
|
|
|
|
self.assertEqual(show_backup["status"], "OK")
|
|
self.assertEqual(show_backup["backup-mode"], "FULL")
|
|
|
|
# Fill with data
|
|
with node.connect() as conn:
|
|
conn.execute("create table test (id int)")
|
|
conn.execute(
|
|
"insert into test select i from generate_series(1,10) s(i)")
|
|
count1 = conn.execute("select count(*) from test")
|
|
conn.commit()
|
|
|
|
# Do compressed page backup
|
|
self.backup_node(
|
|
backup_dir, "node", node, backup_type="page",
|
|
options=['--compress-algorithm=zlib'])
|
|
show_backup = self.show_pb(backup_dir, "node")[1]
|
|
page_id = show_backup["id"]
|
|
|
|
self.assertEqual(show_backup["status"], "OK")
|
|
self.assertEqual(show_backup["backup-mode"], "PAGE")
|
|
|
|
# Merge all backups
|
|
self.merge_backup(backup_dir, "node", page_id)
|
|
show_backups = self.show_pb(backup_dir, "node")
|
|
|
|
self.assertEqual(len(show_backups), 1)
|
|
self.assertEqual(show_backups[0]["status"], "OK")
|
|
self.assertEqual(show_backups[0]["backup-mode"], "FULL")
|
|
|
|
# Drop node and restore it
|
|
node.cleanup()
|
|
self.restore_node(backup_dir, 'node', node)
|
|
node.slow_start()
|
|
|
|
# Check restored node
|
|
count2 = node.execute("postgres", "select count(*) from test")
|
|
self.assertEqual(count1, count2)
|
|
|
|
# Clean after yourself
|
|
node.cleanup()
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
# @unittest.skip("skip")
|
|
def test_merge_tablespaces(self):
|
|
"""
|
|
Some test here
|
|
"""
|
|
|
|
def test_merge_page_truncate(self):
|
|
"""
|
|
make node, create table, take full backup,
|
|
delete last 3 pages, vacuum relation,
|
|
take page backup, merge full and page,
|
|
restore last page backup and check data correctness
|
|
"""
|
|
fname = self.id().split('.')[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
set_replication=True,
|
|
initdb_params=['--data-checksums'],
|
|
pg_options={
|
|
'wal_level': 'replica',
|
|
'max_wal_senders': '2',
|
|
'checkpoint_timeout': '300s',
|
|
'autovacuum': 'off'
|
|
}
|
|
)
|
|
node_restored = self.make_simple_node(
|
|
base_dir="{0}/{1}/node_restored".format(module_name, fname))
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, 'node', node)
|
|
self.set_archiving(backup_dir, 'node', node)
|
|
node_restored.cleanup()
|
|
node.start()
|
|
self.create_tblspace_in_node(node, 'somedata')
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"create sequence t_seq; "
|
|
"create table t_heap tablespace somedata as select i as id, "
|
|
"md5(i::text) as text, "
|
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
|
"from generate_series(0,1024) i;")
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(backup_dir, 'node', node)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"delete from t_heap where ctid >= '(11,0)'")
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(
|
|
backup_dir, 'node', node, backup_type='page')
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
page_id = self.show_pb(backup_dir, "node")[1]["id"]
|
|
self.merge_backup(backup_dir, "node", page_id)
|
|
|
|
self.validate_pb(backup_dir)
|
|
|
|
old_tablespace = self.get_tblspace_path(node, 'somedata')
|
|
new_tablespace = self.get_tblspace_path(node_restored, 'somedata_new')
|
|
|
|
self.restore_node(
|
|
backup_dir, 'node', node_restored,
|
|
options=[
|
|
"-j", "4",
|
|
"-T", "{0}={1}".format(old_tablespace, new_tablespace),
|
|
"--recovery-target-action=promote"])
|
|
|
|
# Physical comparison
|
|
if self.paranoia:
|
|
pgdata_restored = self.pgdata_content(node_restored.data_dir)
|
|
self.compare_pgdata(pgdata, pgdata_restored)
|
|
|
|
node_restored.append_conf(
|
|
"postgresql.auto.conf", "port = {0}".format(node_restored.port))
|
|
node_restored.slow_start()
|
|
|
|
# Logical comparison
|
|
result1 = node.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
result2 = node_restored.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
self.assertEqual(result1, result2)
|
|
|
|
# Clean after yourself
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
def test_merge_delta_truncate(self):
|
|
"""
|
|
make node, create table, take full backup,
|
|
delete last 3 pages, vacuum relation,
|
|
take page backup, merge full and page,
|
|
restore last page backup and check data correctness
|
|
"""
|
|
fname = self.id().split('.')[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
set_replication=True,
|
|
initdb_params=['--data-checksums'],
|
|
pg_options={
|
|
'wal_level': 'replica',
|
|
'max_wal_senders': '2',
|
|
'checkpoint_timeout': '300s',
|
|
'autovacuum': 'off'
|
|
}
|
|
)
|
|
node_restored = self.make_simple_node(
|
|
base_dir="{0}/{1}/node_restored".format(module_name, fname))
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, 'node', node)
|
|
self.set_archiving(backup_dir, 'node', node)
|
|
node_restored.cleanup()
|
|
node.start()
|
|
self.create_tblspace_in_node(node, 'somedata')
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"create sequence t_seq; "
|
|
"create table t_heap tablespace somedata as select i as id, "
|
|
"md5(i::text) as text, "
|
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
|
"from generate_series(0,1024) i;")
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(backup_dir, 'node', node)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"delete from t_heap where ctid >= '(11,0)'")
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(
|
|
backup_dir, 'node', node, backup_type='delta')
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
page_id = self.show_pb(backup_dir, "node")[1]["id"]
|
|
self.merge_backup(backup_dir, "node", page_id)
|
|
|
|
self.validate_pb(backup_dir)
|
|
|
|
old_tablespace = self.get_tblspace_path(node, 'somedata')
|
|
new_tablespace = self.get_tblspace_path(node_restored, 'somedata_new')
|
|
|
|
self.restore_node(
|
|
backup_dir, 'node', node_restored,
|
|
options=[
|
|
"-j", "4",
|
|
"-T", "{0}={1}".format(old_tablespace, new_tablespace),
|
|
"--recovery-target-action=promote"])
|
|
|
|
# Physical comparison
|
|
if self.paranoia:
|
|
pgdata_restored = self.pgdata_content(node_restored.data_dir)
|
|
self.compare_pgdata(pgdata, pgdata_restored)
|
|
|
|
node_restored.append_conf(
|
|
"postgresql.auto.conf", "port = {0}".format(node_restored.port))
|
|
node_restored.slow_start()
|
|
|
|
# Logical comparison
|
|
result1 = node.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
result2 = node_restored.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
self.assertEqual(result1, result2)
|
|
|
|
# Clean after yourself
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
def test_merge_ptrack_truncate(self):
|
|
"""
|
|
make node, create table, take full backup,
|
|
delete last 3 pages, vacuum relation,
|
|
take page backup, merge full and page,
|
|
restore last page backup and check data correctness
|
|
"""
|
|
fname = self.id().split('.')[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
set_replication=True,
|
|
initdb_params=['--data-checksums'],
|
|
pg_options={
|
|
'wal_level': 'replica',
|
|
'max_wal_senders': '2',
|
|
'checkpoint_timeout': '300s',
|
|
'autovacuum': 'off',
|
|
'ptrack_enable': 'on'
|
|
}
|
|
)
|
|
node_restored = self.make_simple_node(
|
|
base_dir="{0}/{1}/node_restored".format(module_name, fname))
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, 'node', node)
|
|
self.set_archiving(backup_dir, 'node', node)
|
|
node_restored.cleanup()
|
|
node.start()
|
|
self.create_tblspace_in_node(node, 'somedata')
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"create sequence t_seq; "
|
|
"create table t_heap tablespace somedata as select i as id, "
|
|
"md5(i::text) as text, "
|
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
|
"from generate_series(0,1024) i;")
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(backup_dir, 'node', node)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"delete from t_heap where ctid >= '(11,0)'")
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap")
|
|
|
|
self.backup_node(
|
|
backup_dir, 'node', node, backup_type='ptrack')
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
page_id = self.show_pb(backup_dir, "node")[1]["id"]
|
|
self.merge_backup(backup_dir, "node", page_id)
|
|
|
|
self.validate_pb(backup_dir)
|
|
|
|
old_tablespace = self.get_tblspace_path(node, 'somedata')
|
|
new_tablespace = self.get_tblspace_path(node_restored, 'somedata_new')
|
|
|
|
self.restore_node(
|
|
backup_dir, 'node', node_restored,
|
|
options=[
|
|
"-j", "4",
|
|
"-T", "{0}={1}".format(old_tablespace, new_tablespace),
|
|
"--recovery-target-action=promote"])
|
|
|
|
# Physical comparison
|
|
if self.paranoia:
|
|
pgdata_restored = self.pgdata_content(node_restored.data_dir)
|
|
self.compare_pgdata(pgdata, pgdata_restored)
|
|
|
|
node_restored.append_conf(
|
|
"postgresql.auto.conf", "port = {0}".format(node_restored.port))
|
|
node_restored.slow_start()
|
|
|
|
# Logical comparison
|
|
result1 = node.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
result2 = node_restored.safe_psql(
|
|
"postgres",
|
|
"select * from t_heap")
|
|
|
|
self.assertEqual(result1, result2)
|
|
|
|
# Clean after yourself
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
# @unittest.skip("skip")
|
|
def test_merge_delta_delete(self):
|
|
"""
|
|
Make node, create tablespace with table, take full backup,
|
|
alter tablespace location, take delta backup, merge full and delta,
|
|
restore database.
|
|
"""
|
|
fname = self.id().split('.')[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
set_replication=True, initdb_params=['--data-checksums'],
|
|
pg_options={
|
|
'wal_level': 'replica',
|
|
'max_wal_senders': '2',
|
|
'checkpoint_timeout': '30s',
|
|
'autovacuum': 'off'
|
|
}
|
|
)
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, 'node', node)
|
|
self.set_archiving(backup_dir, 'node', node)
|
|
node.start()
|
|
|
|
self.create_tblspace_in_node(node, 'somedata')
|
|
|
|
# FULL backup
|
|
self.backup_node(backup_dir, 'node', node, options=["--stream"])
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"create table t_heap tablespace somedata as select i as id,"
|
|
" md5(i::text) as text, md5(i::text)::tsvector as tsvector"
|
|
" from generate_series(0,100) i"
|
|
)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"delete from t_heap"
|
|
)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap"
|
|
)
|
|
|
|
# DELTA BACKUP
|
|
self.backup_node(
|
|
backup_dir, 'node', node,
|
|
backup_type='delta',
|
|
options=["--stream"]
|
|
)
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
backup_id = self.show_pb(backup_dir, "node")[1]["id"]
|
|
self.merge_backup(backup_dir, "node", backup_id, options=["-j", "4"])
|
|
|
|
# RESTORE
|
|
node_restored = self.make_simple_node(
|
|
base_dir="{0}/{1}/node_restored".format(module_name, fname)
|
|
)
|
|
node_restored.cleanup()
|
|
|
|
self.restore_node(
|
|
backup_dir, 'node', node_restored,
|
|
options=[
|
|
"-j", "4",
|
|
"-T", "{0}={1}".format(
|
|
self.get_tblspace_path(node, 'somedata'),
|
|
self.get_tblspace_path(node_restored, 'somedata')
|
|
)
|
|
]
|
|
)
|
|
|
|
# GET RESTORED PGDATA AND COMPARE
|
|
if self.paranoia:
|
|
pgdata_restored = self.pgdata_content(node_restored.data_dir)
|
|
self.compare_pgdata(pgdata, pgdata_restored)
|
|
|
|
# START RESTORED NODE
|
|
node_restored.append_conf(
|
|
'postgresql.auto.conf', 'port = {0}'.format(node_restored.port))
|
|
node_restored.start()
|
|
|
|
# Clean after yourself
|
|
self.del_test_dir(module_name, fname)
|
|
|
|
# @unittest.skip("skip")
|
|
def test_continue_failed_merge(self):
|
|
"""
|
|
Check that failed MERGE can be continued
|
|
"""
|
|
fname = self.id().split('.')[3]
|
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
|
node = self.make_simple_node(
|
|
base_dir="{0}/{1}/node".format(module_name, fname),
|
|
set_replication=True, initdb_params=['--data-checksums'],
|
|
pg_options={
|
|
'wal_level': 'replica'
|
|
}
|
|
)
|
|
|
|
self.init_pb(backup_dir)
|
|
self.add_instance(backup_dir, 'node', node)
|
|
self.set_archiving(backup_dir, 'node', node)
|
|
node.start()
|
|
|
|
# FULL backup
|
|
self.backup_node(backup_dir, 'node', node)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"create table t_heap as select i as id,"
|
|
" md5(i::text) as text, md5(i::text)::tsvector as tsvector"
|
|
" from generate_series(0,1000) i"
|
|
)
|
|
|
|
# DELTA BACKUP
|
|
self.backup_node(
|
|
backup_dir, 'node', node, backup_type='delta'
|
|
)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"delete from t_heap"
|
|
)
|
|
|
|
node.safe_psql(
|
|
"postgres",
|
|
"vacuum t_heap"
|
|
)
|
|
|
|
# DELTA BACKUP
|
|
self.backup_node(
|
|
backup_dir, 'node', node, backup_type='delta'
|
|
)
|
|
|
|
if self.paranoia:
|
|
pgdata = self.pgdata_content(node.data_dir)
|
|
|
|
backup_id = self.show_pb(backup_dir, "node")[2]["id"]
|
|
|
|
gdb = self.merge_backup(backup_dir, "node", backup_id, gdb=True)
|
|
|
|
gdb.set_breakpoint('move_file')
|
|
gdb.run_until_break()
|
|
|
|
if gdb.continue_execution_until_break(20) != 'breakpoint-hit':
|
|
print('Failed to hit breakpoint')
|
|
exit(1)
|
|
|
|
gdb._execute('signal SIGKILL')
|
|
|
|
print(self.show_pb(backup_dir, as_text=True, as_json=False))
|
|
|
|
# Try to continue failed MERGE
|
|
self.merge_backup(backup_dir, "node", backup_id)
|