1
0
mirror of https://github.com/postgrespro/pg_probackup.git synced 2025-01-24 11:46:31 +02:00
pg_probackup/tests/incr_restore.py

2457 lines
80 KiB
Python

import os
import unittest
from .helpers.ptrack_helpers import ProbackupTest, ProbackupException
import subprocess
from datetime import datetime
import sys
from time import sleep
from datetime import datetime, timedelta
import hashlib
import shutil
import json
from testgres import QueryException
module_name = 'incr_restore'
class IncrRestoreTest(ProbackupTest, unittest.TestCase):
# @unittest.skip("skip")
def test_basic_incr_restore(self):
"""incremental restore in CHECKSUM mode"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.pgbench_init(scale=50)
self.backup_node(backup_dir, 'node', node)
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
backup_id = self.backup_node(backup_dir, 'node', node, backup_type='page')
pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_basic_incr_restore_into_missing_directory(self):
""""""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.pgbench_init(scale=10)
self.backup_node(backup_dir, 'node', node)
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgdata = self.pgdata_content(node.data_dir)
node.cleanup()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_checksum_corruption_detection(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.pgbench_init(scale=10)
self.backup_node(backup_dir, 'node', node)
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
backup_id = self.backup_node(backup_dir, 'node', node, backup_type='page')
pgdata = self.pgdata_content(node.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=lsn"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node, options=['--stream'])
tblspace = self.get_tblspace_path(node, 'tblspace')
some_directory = self.get_tblspace_path(node, 'some_directory')
# stuff new destination with garbage
self.restore_node(backup_dir, 'node', node, data_dir=some_directory)
self.create_tblspace_in_node(node, 'tblspace')
node.pgbench_init(scale=10, tablespace='tblspace')
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4", "--incremental-mode=checksum", "--force",
"-T{0}={1}".format(tblspace, some_directory)])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_1(self):
"""recovery to target timeline"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
set_replication=True,
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node, options=['--stream'])
tblspace = self.get_tblspace_path(node, 'tblspace')
some_directory = self.get_tblspace_path(node, 'some_directory')
self.restore_node(backup_dir, 'node', node, data_dir=some_directory)
self.create_tblspace_in_node(node, 'tblspace')
node.pgbench_init(scale=10, tablespace='tblspace')
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(
backup_dir, 'node', node, backup_type='delta', options=['--stream'])
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
pgbench.stdout.close()
self.backup_node(
backup_dir, 'node', node, backup_type='delta', options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_2(self):
"""
If "--tablespace-mapping" option is used with incremental restore,
then new directory must be empty.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
set_replication=True,
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node, options=['--stream'])
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
# fill node1 with data
out = self.restore_node(
backup_dir, 'node', node,
data_dir=node_1.data_dir,
options=['--incremental-mode=checksum', '--force'])
self.assertIn("WARNING: Backup catalog was initialized for system id", out)
tblspace = self.get_tblspace_path(node, 'tblspace')
self.create_tblspace_in_node(node, 'tblspace')
node.pgbench_init(scale=5, tablespace='tblspace')
node.safe_psql(
'postgres',
'vacuum')
self.backup_node(backup_dir, 'node', node, backup_type='delta', options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
try:
self.restore_node(
backup_dir, 'node', node,
data_dir=node_1.data_dir,
options=['--incremental-mode=checksum', '-T{0}={1}'.format(tblspace, tblspace)])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because remapped directory is not empty.\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'ERROR: Remapped tablespace destination is not empty',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
out = self.restore_node(
backup_dir, 'node', node,
data_dir=node_1.data_dir,
options=[
'--force', '--incremental-mode=checksum',
'-T{0}={1}'.format(tblspace, tblspace)])
pgdata_restored = self.pgdata_content(node_1.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_3(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.create_tblspace_in_node(node, 'tblspace1')
node.pgbench_init(scale=10, tablespace='tblspace1')
# take backup with tblspace1
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
self.drop_tblspace(node, 'tblspace1')
self.create_tblspace_in_node(node, 'tblspace2')
node.pgbench_init(scale=10, tablespace='tblspace2')
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4",
"--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_4(self):
"""
Check that system ID mismatch is detected,
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.create_tblspace_in_node(node, 'tblspace1')
node.pgbench_init(scale=10, tablespace='tblspace1')
# take backup of node1 with tblspace1
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
self.drop_tblspace(node, 'tblspace1')
node.cleanup()
# recreate node
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
node.slow_start()
self.create_tblspace_in_node(node, 'tblspace1')
node.pgbench_init(scale=10, tablespace='tblspace1')
node.stop()
try:
self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4",
"--incremental-mode=checksum"])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because destination directory has wrong system id.\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'WARNING: Backup catalog was initialized for system id',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
self.assertIn(
'ERROR: Incremental restore is not allowed',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
out = self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4", "--force",
"--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.expectedFailure
@unittest.skip("skip")
def test_incr_restore_with_tablespace_5(self):
"""
More complicated case, we restore backup
with tablespace, which we remap into directory
with some old content, that belongs to an instance
with different system id.
"""
fname = self.id().split('.')[3]
node1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node1'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node1)
node1.slow_start()
self.create_tblspace_in_node(node1, 'tblspace')
node1.pgbench_init(scale=10, tablespace='tblspace')
# take backup of node1 with tblspace
self.backup_node(backup_dir, 'node', node1, options=['--stream'])
pgdata = self.pgdata_content(node1.data_dir)
node1.stop()
# recreate node
node2 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node2'),
set_replication=True,
initdb_params=['--data-checksums'])
node2.slow_start()
self.create_tblspace_in_node(node2, 'tblspace')
node2.pgbench_init(scale=10, tablespace='tblspace')
node2.stop()
tblspc1_path = self.get_tblspace_path(node1, 'tblspace')
tblspc2_path = self.get_tblspace_path(node2, 'tblspace')
out = self.restore_node(
backup_dir, 'node', node1,
options=[
"-j", "4", "--force",
"--incremental-mode=checksum",
"-T{0}={1}".format(tblspc1_path, tblspc2_path)])
# check that tblspc1_path is empty
self.assertFalse(
os.listdir(tblspc1_path),
"Dir is not empty: '{0}'".format(tblspc1_path))
pgdata_restored = self.pgdata_content(node1.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_6(self):
"""
Empty pgdata, not empty tablespace
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.create_tblspace_in_node(node, 'tblspace')
node.pgbench_init(scale=10, tablespace='tblspace')
# take backup of node with tblspace
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
node.cleanup()
try:
self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4",
"--incremental-mode=checksum"])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because there is running postmaster "
"process in destination directory.\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'ERROR: PGDATA is empty, but tablespace destination is not',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
out = self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4", "--force",
"--incremental-mode=checksum"])
self.assertIn(
"INFO: Destination directory and tablespace directories are empty, "
"disable incremental restore", out)
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_restore_with_tablespace_7(self):
"""
Restore backup without tablespace into
PGDATA with tablespace.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'])
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
# take backup of node with tblspace
self.backup_node(backup_dir, 'node', node, options=['--stream'])
pgdata = self.pgdata_content(node.data_dir)
self.create_tblspace_in_node(node, 'tblspace')
node.pgbench_init(scale=5, tablespace='tblspace')
node.stop()
# try:
# self.restore_node(
# backup_dir, 'node', node,
# options=[
# "-j", "4",
# "--incremental-mode=checksum"])
# # we should die here because exception is what we expect to happen
# self.assertEqual(
# 1, 0,
# "Expecting Error because there is running postmaster "
# "process in destination directory.\n "
# "Output: {0} \n CMD: {1}".format(
# repr(self.output), self.cmd))
# except ProbackupException as e:
# self.assertIn(
# 'ERROR: PGDATA is empty, but tablespace destination is not',
# e.message,
# '\n Unexpected Error Message: {0}\n CMD: {1}'.format(
# repr(e.message), self.cmd))
out = self.restore_node(
backup_dir, 'node', node,
options=[
"-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_basic_incr_restore_sanity(self):
"""recovery to target timeline"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
set_replication=True,
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node, options=['--stream'])
try:
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=checksum"])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because there is running postmaster "
"process in destination directory.\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'WARNING: Postmaster with pid',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
self.assertIn(
'ERROR: Incremental restore is not allowed',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
try:
self.restore_node(
backup_dir, 'node', node_1, data_dir=node_1.data_dir,
options=["-j", "4", "--incremental-mode=checksum"])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because destination directory has wrong system id.\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'WARNING: Backup catalog was initialized for system id',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
self.assertIn(
'ERROR: Incremental restore is not allowed',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
# Clean after yourself
self.del_test_dir(module_name, fname, [node])
# @unittest.skip("skip")
def test_incr_checksum_restore(self):
"""
/----C-----D
------A----B---*--------X
X - is instance, we want to return it to C state.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.pgbench_init(scale=50)
self.backup_node(backup_dir, 'node', node)
pgbench = node.pgbench(options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
xid = node.safe_psql(
'postgres',
'select txid_current()').decode('utf-8').rstrip()
# --A-----B--------X
pgbench = node.pgbench(options=['-T', '30', '-c', '1', '--no-vacuum'])
pgbench.wait()
node.stop(['-m', 'immediate', '-D', node.data_dir])
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
node_1.cleanup()
self.restore_node(
backup_dir, 'node', node_1, data_dir=node_1.data_dir,
options=[
'--recovery-target-action=promote',
'--recovery-target-xid={0}'.format(xid)])
self.set_auto_conf(node_1, {'port': node_1.port})
node_1.slow_start()
# /--
# --A-----B----*----X
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# /--C
# --A-----B----*----X
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
# /--C------
# --A-----B----*----X
pgbench = node_1.pgbench(options=['-T', '50', '-c', '1'])
pgbench.wait()
# /--C------D
# --A-----B----*----X
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
pgdata = self.pgdata_content(node_1.data_dir)
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.set_auto_conf(node, {'port': node.port})
node.slow_start()
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname, [node, node_1])
# @unittest.skip("skip")
def test_incr_lsn_restore(self):
"""
/----C-----D
------A----B---*--------X
X - is instance, we want to return it to C state.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.pgbench_init(scale=50)
self.backup_node(backup_dir, 'node', node)
pgbench = node.pgbench(options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node, backup_type='page')
pgbench = node.pgbench(options=['-T', '10', '-c', '1', '--no-vacuum'])
pgbench.wait()
xid = node.safe_psql(
'postgres',
'select txid_current()').decode('utf-8').rstrip()
# --A-----B--------X
pgbench = node.pgbench(options=['-T', '30', '-c', '1', '--no-vacuum'])
pgbench.wait()
node.stop(['-m', 'immediate', '-D', node.data_dir])
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
node_1.cleanup()
self.restore_node(
backup_dir, 'node', node_1, data_dir=node_1.data_dir,
options=[
'--recovery-target-action=promote',
'--recovery-target-xid={0}'.format(xid)])
self.set_auto_conf(node_1, {'port': node_1.port})
node_1.slow_start()
# /--
# --A-----B----*----X
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# /--C
# --A-----B----*----X
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
# /--C------
# --A-----B----*----X
pgbench = node_1.pgbench(options=['-T', '50', '-c', '1'])
pgbench.wait()
# /--C------D
# --A-----B----*----X
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
pgdata = self.pgdata_content(node_1.data_dir)
self.restore_node(
backup_dir, 'node', node, options=["-j", "4", "--incremental-mode=lsn"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.set_auto_conf(node, {'port': node.port})
node.slow_start()
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname, [node, node_1])
# @unittest.skip("skip")
def test_incr_lsn_sanity(self):
"""
/----A-----B
F------*--------X
X - is instance, we want to return it to state B.
fail is expected behaviour in case of lsn restore.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node)
node.pgbench_init(scale=10)
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
node_1.cleanup()
self.restore_node(
backup_dir, 'node', node_1, data_dir=node_1.data_dir)
self.set_auto_conf(node_1, {'port': node_1.port})
node_1.slow_start()
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='full')
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
node.stop()
try:
self.restore_node(
backup_dir, 'node', node, data_dir=node.data_dir,
options=["-j", "4", "--incremental-mode=lsn"])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because incremental restore in lsn mode is impossible\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
"ERROR: Cannot perform incremental restore of "
"backup chain {0} in 'lsn' mode".format(page_id),
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
# Clean after yourself
self.del_test_dir(module_name, fname, [node_1])
# @unittest.skip("skip")
def test_incr_checksum_sanity(self):
"""
/----A-----B
F------*--------X
X - is instance, we want to return it to state B.
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node)
node.pgbench_init(scale=20)
node_1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node_1'))
node_1.cleanup()
self.restore_node(
backup_dir, 'node', node_1, data_dir=node_1.data_dir)
self.set_auto_conf(node_1, {'port': node_1.port})
node_1.slow_start()
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='full')
pgbench = node_1.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(backup_dir, 'node', node_1,
data_dir=node_1.data_dir, backup_type='page')
pgdata = self.pgdata_content(node_1.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node, data_dir=node.data_dir,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname, [node_1])
# @unittest.skip("skip")
def test_incr_checksum_corruption_detection(self):
"""
check that corrupted page got detected and replaced
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
# initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node)
node.pgbench_init(scale=20)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node,
data_dir=node.data_dir, backup_type='full')
heap_path = node.safe_psql(
"postgres",
"select pg_relation_filepath('pgbench_accounts')").decode('utf-8').rstrip()
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(backup_dir, 'node', node,
data_dir=node.data_dir, backup_type='page')
pgdata = self.pgdata_content(node.data_dir)
node.stop()
path = os.path.join(node.data_dir, heap_path)
with open(path, "rb+", 0) as f:
f.seek(22000)
f.write(b"bla")
f.flush()
f.close
self.restore_node(
backup_dir, 'node', node, data_dir=node.data_dir,
options=["-j", "4", "--incremental-mode=checksum"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_incr_lsn_corruption_detection(self):
"""
check that corrupted page got detected and replaced
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
self.backup_node(backup_dir, 'node', node)
node.pgbench_init(scale=20)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
self.backup_node(backup_dir, 'node', node,
data_dir=node.data_dir, backup_type='full')
heap_path = node.safe_psql(
"postgres",
"select pg_relation_filepath('pgbench_accounts')").decode('utf-8').rstrip()
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(backup_dir, 'node', node,
data_dir=node.data_dir, backup_type='page')
pgdata = self.pgdata_content(node.data_dir)
node.stop()
path = os.path.join(node.data_dir, heap_path)
with open(path, "rb+", 0) as f:
f.seek(22000)
f.write(b"bla")
f.flush()
f.close
self.restore_node(
backup_dir, 'node', node, data_dir=node.data_dir,
options=["-j", "4", "--incremental-mode=lsn"])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_restore_multiple_external(self):
"""check that cmdline has priority over config"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
external_dir1 = self.get_tblspace_path(node, 'external_dir1')
external_dir2 = self.get_tblspace_path(node, 'external_dir2')
# FULL backup
node.pgbench_init(scale=20)
self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4"])
# fill external directories with data
self.restore_node(
backup_dir, 'node', node,
data_dir=external_dir1, options=["-j", "4"])
self.restore_node(
backup_dir, 'node', node,
data_dir=external_dir2, options=["-j", "4"])
self.set_config(
backup_dir, 'node',
options=['-E{0}{1}{2}'.format(
external_dir1, self.EXTERNAL_DIRECTORY_DELIMITER, external_dir2)])
# cmdline option MUST override options in config
self.backup_node(
backup_dir, 'node', node,
backup_type='full', options=["-j", "4"])
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# cmdline option MUST override options in config
self.backup_node(
backup_dir, 'node', node,
backup_type='page', options=["-j", "4"])
pgdata = self.pgdata_content(
node.base_dir, exclude_dirs=['logs'])
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", '--incremental-mode=checksum'])
pgdata_restored = self.pgdata_content(
node.base_dir, exclude_dirs=['logs'])
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_lsn_restore_multiple_external(self):
"""check that cmdline has priority over config"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
external_dir1 = self.get_tblspace_path(node, 'external_dir1')
external_dir2 = self.get_tblspace_path(node, 'external_dir2')
# FULL backup
node.pgbench_init(scale=20)
self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4"])
# fill external directories with data
self.restore_node(
backup_dir, 'node', node,
data_dir=external_dir1, options=["-j", "4"])
self.restore_node(
backup_dir, 'node', node,
data_dir=external_dir2, options=["-j", "4"])
self.set_config(
backup_dir, 'node',
options=['-E{0}{1}{2}'.format(
external_dir1, self.EXTERNAL_DIRECTORY_DELIMITER, external_dir2)])
# cmdline option MUST override options in config
self.backup_node(
backup_dir, 'node', node,
backup_type='full', options=["-j", "4"])
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# cmdline option MUST override options in config
self.backup_node(
backup_dir, 'node', node,
backup_type='page', options=["-j", "4"])
pgdata = self.pgdata_content(
node.base_dir, exclude_dirs=['logs'])
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
node.stop()
self.restore_node(
backup_dir, 'node', node,
options=["-j", "4", '--incremental-mode=lsn'])
pgdata_restored = self.pgdata_content(
node.base_dir, exclude_dirs=['logs'])
self.compare_pgdata(pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_lsn_restore_backward(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off', 'wal_log_hints': 'on', 'hot_standby': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
# FULL backup
node.pgbench_init(scale=2)
full_id = self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4"])
full_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(
backup_dir, 'node', node,
backup_type='page', options=["-j", "4"])
page_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
delta_id = self.backup_node(
backup_dir, 'node', node,
backup_type='delta', options=["-j", "4"])
delta_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=full_id,
options=[
"-j", "4",
'--incremental-mode=lsn',
'--recovery-target=immediate',
'--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(full_pgdata, pgdata_restored)
node.slow_start(replica=True)
node.stop()
try:
self.restore_node(
backup_dir, 'node', node, backup_id=page_id,
options=[
"-j", "4", '--incremental-mode=lsn',
'--recovery-target=immediate', '--recovery-target-action=pause'])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because incremental restore in lsn mode is impossible\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
"Cannot perform incremental restore of backup chain",
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
self.restore_node(
backup_dir, 'node', node, backup_id=page_id,
options=[
"-j", "4", '--incremental-mode=checksum',
'--recovery-target=immediate', '--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(page_pgdata, pgdata_restored)
node.slow_start(replica=True)
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=delta_id,
options=[
"-j", "4",
'--incremental-mode=lsn',
'--recovery-target=immediate',
'--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(delta_pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_checksum_restore_backward(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={
'autovacuum': 'off',
'hot_standby': 'on'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
# FULL backup
node.pgbench_init(scale=20)
full_id = self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4"])
full_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
page_id = self.backup_node(
backup_dir, 'node', node,
backup_type='page', options=["-j", "4"])
page_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
delta_id = self.backup_node(
backup_dir, 'node', node,
backup_type='delta', options=["-j", "4"])
delta_pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=full_id,
options=[
"-j", "4",
'--incremental-mode=checksum',
'--recovery-target=immediate',
'--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(full_pgdata, pgdata_restored)
node.slow_start(replica=True)
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=page_id,
options=[
"-j", "4",
'--incremental-mode=checksum',
'--recovery-target=immediate',
'--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(page_pgdata, pgdata_restored)
node.slow_start(replica=True)
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=delta_id,
options=[
"-j", "4",
'--incremental-mode=checksum',
'--recovery-target=immediate',
'--recovery-target-action=pause'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(delta_pgdata, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
def test_make_replica_via_incr_checksum_restore(self):
"""
"""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
master = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'master'),
set_replication=True,
initdb_params=['--data-checksums'])
if self.get_version(master) < self.version_to_num('9.6.0'):
self.del_test_dir(module_name, fname)
return unittest.skip(
'Skipped because backup from replica is not supported in PG 9.5')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', master)
self.set_archiving(backup_dir, 'node', master, replica=True)
master.slow_start()
replica = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'replica'))
replica.cleanup()
master.pgbench_init(scale=20)
self.backup_node(backup_dir, 'node', master)
self.restore_node(
backup_dir, 'node', replica, options=['-R'])
# Settings for Replica
self.set_replica(master, replica, synchronous=False)
replica.slow_start(replica=True)
pgbench = master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# PROMOTIONS
replica.promote()
new_master = replica
# old master is going a bit further
old_master = master
pgbench = old_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
old_master.stop()
pgbench = new_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# take backup from new master
self.backup_node(
backup_dir, 'node', new_master,
data_dir=new_master.data_dir, backup_type='page')
# restore old master as replica
self.restore_node(
backup_dir, 'node', old_master, data_dir=old_master.data_dir,
options=['-R', '--incremental-mode=checksum'])
self.set_replica(new_master, old_master, synchronous=True)
old_master.slow_start(replica=True)
pgbench = new_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# Clean after yourself
self.del_test_dir(module_name, fname, [new_master, old_master])
# @unittest.skip("skip")
def test_make_replica_via_incr_lsn_restore(self):
"""
"""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
master = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'master'),
set_replication=True,
initdb_params=['--data-checksums'])
if self.get_version(master) < self.version_to_num('9.6.0'):
self.del_test_dir(module_name, fname)
return unittest.skip(
'Skipped because backup from replica is not supported in PG 9.5')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', master)
self.set_archiving(backup_dir, 'node', master, replica=True)
master.slow_start()
replica = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'replica'))
replica.cleanup()
master.pgbench_init(scale=20)
self.backup_node(backup_dir, 'node', master)
self.restore_node(
backup_dir, 'node', replica, options=['-R'])
# Settings for Replica
self.set_replica(master, replica, synchronous=False)
replica.slow_start(replica=True)
pgbench = master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# PROMOTIONS
replica.promote()
new_master = replica
# old master is going a bit further
old_master = master
pgbench = old_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
old_master.stop()
pgbench = new_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# take backup from new master
self.backup_node(
backup_dir, 'node', new_master,
data_dir=new_master.data_dir, backup_type='page')
# restore old master as replica
self.restore_node(
backup_dir, 'node', old_master, data_dir=old_master.data_dir,
options=['-R', '--incremental-mode=lsn'])
self.set_replica(new_master, old_master, synchronous=True)
old_master.slow_start(replica=True)
pgbench = new_master.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# Clean after yourself
self.del_test_dir(module_name, fname, [new_master, old_master])
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_checksum_long_xact(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
# initdb_params=['--data-checksums'],
pg_options={
'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.safe_psql(
'postgres',
'create extension pageinspect')
# FULL backup
con = node.connect("postgres")
con.execute("CREATE TABLE t1 (a int)")
con.commit()
con.execute("INSERT INTO t1 values (1)")
con.commit()
# leave uncommited
con2 = node.connect("postgres")
con.execute("INSERT INTO t1 values (2)")
con2.execute("INSERT INTO t1 values (3)")
full_id = self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4", "--stream"])
self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
con.commit()
node.safe_psql(
'postgres',
'select * from t1')
con2.commit()
node.safe_psql(
'postgres',
'select * from t1')
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=full_id,
options=["-j", "4", '--incremental-mode=checksum'])
node.slow_start()
self.assertEqual(
node.safe_psql(
'postgres',
'select count(*) from t1').decode('utf-8').rstrip(),
'1')
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
# This test will pass with Enterprise
# because it has checksums enabled by default
@unittest.skipIf(ProbackupTest.enterprise, 'skip')
def test_incr_lsn_long_xact_1(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
# initdb_params=['--data-checksums'],
pg_options={
'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.safe_psql(
'postgres',
'create extension pageinspect')
# FULL backup
con = node.connect("postgres")
con.execute("CREATE TABLE t1 (a int)")
con.commit()
con.execute("INSERT INTO t1 values (1)")
con.commit()
# leave uncommited
con2 = node.connect("postgres")
con.execute("INSERT INTO t1 values (2)")
con2.execute("INSERT INTO t1 values (3)")
full_id = self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4", "--stream"])
self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
con.commit()
# when does LSN gets stamped when checksum gets updated ?
node.safe_psql(
'postgres',
'select * from t1')
con2.commit()
node.safe_psql(
'postgres',
'select * from t1')
node.stop()
try:
self.restore_node(
backup_dir, 'node', node, backup_id=full_id,
options=["-j", "4", '--incremental-mode=lsn'])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because incremental restore in lsn mode is impossible\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
"ERROR: Incremental restore in 'lsn' mode require data_checksums to be "
"enabled in destination data directory",
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_lsn_long_xact_2(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={
'autovacuum': 'off',
'full_page_writes': 'off',
'wal_log_hints': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node.safe_psql(
'postgres',
'create extension pageinspect')
# FULL backup
con = node.connect("postgres")
con.execute("CREATE TABLE t1 (a int)")
con.commit()
con.execute("INSERT INTO t1 values (1)")
con.commit()
# leave uncommited
con2 = node.connect("postgres")
con.execute("INSERT INTO t1 values (2)")
con2.execute("INSERT INTO t1 values (3)")
full_id = self.backup_node(
backup_dir, 'node', node,
backup_type="full", options=["-j", "4", "--stream"])
self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
# print(node.safe_psql(
# 'postgres',
# "select * from page_header(get_raw_page('t1', 0))"))
con.commit()
# when does LSN gets stamped when checksum gets updated ?
node.safe_psql(
'postgres',
'select * from t1')
# print(node.safe_psql(
# 'postgres',
# "select * from page_header(get_raw_page('t1', 0))"))
con2.commit()
node.safe_psql(
'postgres',
'select * from t1')
# print(node.safe_psql(
# 'postgres',
# "select * from page_header(get_raw_page('t1', 0))"))
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=full_id,
options=["-j", "4", '--incremental-mode=lsn'])
node.slow_start()
self.assertEqual(
node.safe_psql(
'postgres',
'select count(*) from t1').decode('utf-8').rstrip(),
'1')
# Clean after yourself
self.del_test_dir(module_name, fname, [node])
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_restore_zero_size_file_checksum(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
fullpath = os.path.join(node.data_dir, 'simple_file')
with open(fullpath, "w+b", 0) as f:
f.flush()
f.close
# FULL backup
id1 = self.backup_node(
backup_dir, 'node', node,
options=["-j", "4", "--stream"])
pgdata1 = self.pgdata_content(node.data_dir)
with open(fullpath, "rb+", 0) as f:
f.seek(9000)
f.write(b"bla")
f.flush()
f.close
id2 = self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
pgdata2 = self.pgdata_content(node.data_dir)
with open(fullpath, "w") as f:
f.close()
id3 = self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
pgdata3 = self.pgdata_content(node.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=id1,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata1, pgdata_restored)
self.restore_node(
backup_dir, 'node', node, backup_id=id2,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata2, pgdata_restored)
self.restore_node(
backup_dir, 'node', node, backup_id=id3,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata3, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
# @unittest.skip("skip")
# @unittest.expectedFailure
def test_incr_restore_zero_size_file_lsn(self):
"""
"""
fname = self.id().split('.')[3]
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
set_replication=True,
initdb_params=['--data-checksums'],
pg_options={'autovacuum': 'off'})
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
node.slow_start()
fullpath = os.path.join(node.data_dir, 'simple_file')
with open(fullpath, "w+b", 0) as f:
f.flush()
f.close
# FULL backup
id1 = self.backup_node(
backup_dir, 'node', node,
options=["-j", "4", "--stream"])
pgdata1 = self.pgdata_content(node.data_dir)
with open(fullpath, "rb+", 0) as f:
f.seek(9000)
f.write(b"bla")
f.flush()
f.close
id2 = self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
pgdata2 = self.pgdata_content(node.data_dir)
with open(fullpath, "w") as f:
f.close()
id3 = self.backup_node(
backup_dir, 'node', node,
backup_type="delta", options=["-j", "4", "--stream"])
pgdata3 = self.pgdata_content(node.data_dir)
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=id1,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata1, pgdata_restored)
node.slow_start()
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=id2,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata2, pgdata_restored)
node.slow_start()
node.stop()
self.restore_node(
backup_dir, 'node', node, backup_id=id3,
options=["-j", "4", '-I', 'checksum'])
pgdata_restored = self.pgdata_content(node.data_dir)
self.compare_pgdata(pgdata3, pgdata_restored)
# Clean after yourself
self.del_test_dir(module_name, fname)
def test_incremental_partial_restore_exclude_checksum(self):
""""""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'])
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
for i in range(1, 10, 1):
node.safe_psql(
'postgres',
'CREATE database db{0}'.format(i))
db_list_raw = node.safe_psql(
'postgres',
'SELECT to_json(a) '
'FROM (SELECT oid, datname FROM pg_database) a').rstrip()
db_list_splitted = db_list_raw.splitlines()
db_list = {}
for line in db_list_splitted:
line = json.loads(line)
db_list[line['datname']] = line['oid']
node.pgbench_init(scale=20)
# FULL backup
self.backup_node(backup_dir, 'node', node)
pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# PAGE backup
backup_id = self.backup_node(backup_dir, 'node', node, backup_type='page')
# restore FULL backup into second node2
node1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node1'))
node1.cleanup()
node2 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node2'))
node2.cleanup()
# restore some data into node2
self.restore_node(backup_dir, 'node', node2)
# partial restore backup into node1
self.restore_node(
backup_dir, 'node',
node1, options=[
"--db-exclude=db1",
"--db-exclude=db5"])
pgdata1 = self.pgdata_content(node1.data_dir)
# partial incremental restore backup into node2
self.restore_node(
backup_dir, 'node',
node2, options=[
"--db-exclude=db1",
"--db-exclude=db5",
"-I", "checksum"])
pgdata2 = self.pgdata_content(node2.data_dir)
self.compare_pgdata(pgdata1, pgdata2)
self.set_auto_conf(node2, {'port': node2.port})
node2.slow_start()
node2.safe_psql(
'postgres',
'select 1')
try:
node2.safe_psql(
'db1',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
try:
node2.safe_psql(
'db5',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
with open(node2.pg_log_file, 'r') as f:
output = f.read()
self.assertNotIn('PANIC', output)
# Clean after yourself
self.del_test_dir(module_name, fname, [node, node2])
def test_incremental_partial_restore_exclude_lsn(self):
""""""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'])
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
for i in range(1, 10, 1):
node.safe_psql(
'postgres',
'CREATE database db{0}'.format(i))
db_list_raw = node.safe_psql(
'postgres',
'SELECT to_json(a) '
'FROM (SELECT oid, datname FROM pg_database) a').rstrip()
db_list_splitted = db_list_raw.splitlines()
db_list = {}
for line in db_list_splitted:
line = json.loads(line)
db_list[line['datname']] = line['oid']
node.pgbench_init(scale=20)
# FULL backup
self.backup_node(backup_dir, 'node', node)
pgdata = self.pgdata_content(node.data_dir)
pgbench = node.pgbench(options=['-T', '10', '-c', '1'])
pgbench.wait()
# PAGE backup
backup_id = self.backup_node(backup_dir, 'node', node, backup_type='page')
node.stop()
# restore FULL backup into second node2
node1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node1'))
node1.cleanup()
node2 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node2'))
node2.cleanup()
# restore some data into node2
self.restore_node(backup_dir, 'node', node2)
# partial restore backup into node1
self.restore_node(
backup_dir, 'node',
node1, options=[
"--db-exclude=db1",
"--db-exclude=db5"])
pgdata1 = self.pgdata_content(node1.data_dir)
# partial incremental restore backup into node2
node2.port = node.port
node2.slow_start()
node2.stop()
self.restore_node(
backup_dir, 'node',
node2, options=[
"--db-exclude=db1",
"--db-exclude=db5",
"-I", "lsn"])
pgdata2 = self.pgdata_content(node2.data_dir)
self.compare_pgdata(pgdata1, pgdata2)
self.set_auto_conf(node2, {'port': node2.port})
node2.slow_start()
node2.safe_psql(
'postgres',
'select 1')
try:
node2.safe_psql(
'db1',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
try:
node2.safe_psql(
'db5',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
with open(node2.pg_log_file, 'r') as f:
output = f.read()
self.assertNotIn('PANIC', output)
# Clean after yourself
self.del_test_dir(module_name, fname, [node2])
def test_incremental_partial_restore_exclude_tablespace_checksum(self):
""""""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'])
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
# cat_version = node.get_control_data()["Catalog version number"]
# version_specific_dir = 'PG_' + node.major_version_str + '_' + cat_version
# PG_10_201707211
# pg_tblspc/33172/PG_9.5_201510051/16386/
self.create_tblspace_in_node(node, 'somedata')
node_tablespace = self.get_tblspace_path(node, 'somedata')
tbl_oid = node.safe_psql(
'postgres',
"SELECT oid "
"FROM pg_tablespace "
"WHERE spcname = 'somedata'").rstrip()
for i in range(1, 10, 1):
node.safe_psql(
'postgres',
'CREATE database db{0} tablespace somedata'.format(i))
db_list_raw = node.safe_psql(
'postgres',
'SELECT to_json(a) '
'FROM (SELECT oid, datname FROM pg_database) a').rstrip()
db_list_splitted = db_list_raw.splitlines()
db_list = {}
for line in db_list_splitted:
line = json.loads(line)
db_list[line['datname']] = line['oid']
# FULL backup
backup_id = self.backup_node(backup_dir, 'node', node)
# node1
node1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node1'))
node1.cleanup()
node1_tablespace = self.get_tblspace_path(node1, 'somedata')
# node2
node2 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node2'))
node2.cleanup()
node2_tablespace = self.get_tblspace_path(node2, 'somedata')
# in node2 restore full backup
self.restore_node(
backup_dir, 'node',
node2, options=[
"-T", "{0}={1}".format(
node_tablespace, node2_tablespace)])
# partial restore into node1
self.restore_node(
backup_dir, 'node',
node1, options=[
"--db-exclude=db1",
"--db-exclude=db5",
"-T", "{0}={1}".format(
node_tablespace, node1_tablespace)])
pgdata1 = self.pgdata_content(node1.data_dir)
# partial incremental restore into node2
try:
self.restore_node(
backup_dir, 'node',
node2, options=[
"-I", "checksum",
"--db-exclude=db1",
"--db-exclude=db5",
"-T", "{0}={1}".format(
node_tablespace, node2_tablespace)])
# we should die here because exception is what we expect to happen
self.assertEqual(
1, 0,
"Expecting Error because remapped tablespace contain old data .\n "
"Output: {0} \n CMD: {1}".format(
repr(self.output), self.cmd))
except ProbackupException as e:
self.assertIn(
'ERROR: Remapped tablespace destination is not empty:',
e.message,
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
repr(e.message), self.cmd))
self.restore_node(
backup_dir, 'node',
node2, options=[
"-I", "checksum", "--force",
"--db-exclude=db1",
"--db-exclude=db5",
"-T", "{0}={1}".format(
node_tablespace, node2_tablespace)])
pgdata2 = self.pgdata_content(node2.data_dir)
self.compare_pgdata(pgdata1, pgdata2)
self.set_auto_conf(node2, {'port': node2.port})
node2.slow_start()
node2.safe_psql(
'postgres',
'select 1')
try:
node2.safe_psql(
'db1',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
try:
node2.safe_psql(
'db5',
'select 1')
except QueryException as e:
self.assertIn('FATAL', e.message)
with open(node2.pg_log_file, 'r') as f:
output = f.read()
self.assertNotIn('PANIC', output)
# Clean after yourself
self.del_test_dir(module_name, fname, [node2])
def test_incremental_pg_filenode_map(self):
"""
https://github.com/postgrespro/pg_probackup/issues/320
"""
fname = self.id().split('.')[3]
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
node = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node'),
initdb_params=['--data-checksums'])
self.init_pb(backup_dir)
self.add_instance(backup_dir, 'node', node)
self.set_archiving(backup_dir, 'node', node)
node.slow_start()
node1 = self.make_simple_node(
base_dir=os.path.join(module_name, fname, 'node1'),
initdb_params=['--data-checksums'])
node1.cleanup()
node.pgbench_init(scale=5)
# FULL backup
backup_id = self.backup_node(backup_dir, 'node', node)
# in node1 restore full backup
self.restore_node(backup_dir, 'node', node1)
self.set_auto_conf(node1, {'port': node1.port})
node1.slow_start()
pgbench = node.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1'])
pgbench = node1.pgbench(
stdout=subprocess.PIPE, stderr=subprocess.STDOUT,
options=['-T', '10', '-c', '1'])
node.safe_psql(
'postgres',
'reindex index pg_type_oid_index')
# FULL backup
backup_id = self.backup_node(backup_dir, 'node', node)
node1.stop()
# incremental restore into node1
self.restore_node(backup_dir, 'node', node1, options=["-I", "checksum"])
self.set_auto_conf(node1, {'port': node1.port})
node1.slow_start()
node1.safe_psql(
'postgres',
'select 1')
# Clean after yourself
self.del_test_dir(module_name, fname)
# check that MinRecPoint and BackupStartLsn are correctly used in case of --incrementa-lsn