mirror of
https://github.com/postgrespro/pg_probackup.git
synced 2025-01-23 11:45:36 +02:00
Merge branch 'master' into issue_79
This commit is contained in:
commit
037b7cf53b
29
src/delete.c
29
src/delete.c
@ -259,6 +259,7 @@ do_retention_internal(parray *backup_list, parray *to_keep_list, parray *to_purg
|
|||||||
{
|
{
|
||||||
|
|
||||||
bool redundancy_keep = false;
|
bool redundancy_keep = false;
|
||||||
|
time_t backup_time = 0;
|
||||||
pgBackup *backup = (pgBackup *) parray_get(backup_list, (size_t) i);
|
pgBackup *backup = (pgBackup *) parray_get(backup_list, (size_t) i);
|
||||||
|
|
||||||
/* check if backup`s FULL ancestor is in redundancy list */
|
/* check if backup`s FULL ancestor is in redundancy list */
|
||||||
@ -280,10 +281,16 @@ do_retention_internal(parray *backup_list, parray *to_keep_list, parray *to_purg
|
|||||||
cur_full_backup_num++;
|
cur_full_backup_num++;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if backup in needed by retention policy
|
/* Invalid and running backups most likely to have recovery_time == 0,
|
||||||
* TODO: consider that ERROR backup most likely to have recovery_time == 0
|
* so in this case use start_time instead.
|
||||||
*/
|
*/
|
||||||
if ((days_threshold == 0 || (days_threshold > backup->recovery_time)) &&
|
if (backup->recovery_time)
|
||||||
|
backup_time = backup->recovery_time;
|
||||||
|
else
|
||||||
|
backup_time = backup->start_time;
|
||||||
|
|
||||||
|
/* Check if backup in needed by retention policy */
|
||||||
|
if ((days_threshold == 0 || (days_threshold > backup_time)) &&
|
||||||
(instance_config.retention_redundancy == 0 || !redundancy_keep))
|
(instance_config.retention_redundancy == 0 || !redundancy_keep))
|
||||||
{
|
{
|
||||||
/* This backup is not guarded by retention
|
/* This backup is not guarded by retention
|
||||||
@ -622,6 +629,7 @@ do_retention_wal(void)
|
|||||||
XLogRecPtr oldest_lsn = InvalidXLogRecPtr;
|
XLogRecPtr oldest_lsn = InvalidXLogRecPtr;
|
||||||
TimeLineID oldest_tli = 0;
|
TimeLineID oldest_tli = 0;
|
||||||
bool backup_list_is_empty = false;
|
bool backup_list_is_empty = false;
|
||||||
|
int i;
|
||||||
|
|
||||||
/* Get list of backups. */
|
/* Get list of backups. */
|
||||||
backup_list = catalog_get_backup_list(INVALID_BACKUP_ID);
|
backup_list = catalog_get_backup_list(INVALID_BACKUP_ID);
|
||||||
@ -630,14 +638,17 @@ do_retention_wal(void)
|
|||||||
backup_list_is_empty = true;
|
backup_list_is_empty = true;
|
||||||
|
|
||||||
/* Save LSN and Timeline to remove unnecessary WAL segments */
|
/* Save LSN and Timeline to remove unnecessary WAL segments */
|
||||||
if (!backup_list_is_empty)
|
for (i = (int) parray_num(backup_list) - 1; i >= 0; i--)
|
||||||
{
|
{
|
||||||
pgBackup *backup = NULL;
|
pgBackup *backup = (pgBackup *) parray_get(backup_list, i);
|
||||||
/* Get LSN and TLI of oldest alive backup */
|
|
||||||
backup = (pgBackup *) parray_get(backup_list, parray_num(backup_list) -1);
|
|
||||||
|
|
||||||
oldest_tli = backup->tli;
|
/* Get LSN and TLI of the oldest backup with valid start_lsn and tli */
|
||||||
oldest_lsn = backup->start_lsn;
|
if (backup->tli > 0 && !XLogRecPtrIsInvalid(backup->start_lsn))
|
||||||
|
{
|
||||||
|
oldest_tli = backup->tli;
|
||||||
|
oldest_lsn = backup->start_lsn;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Be paranoid */
|
/* Be paranoid */
|
||||||
|
@ -663,7 +663,7 @@ main(int argc, char *argv[])
|
|||||||
recovery_target_options,
|
recovery_target_options,
|
||||||
restore_params);
|
restore_params);
|
||||||
case VALIDATE_CMD:
|
case VALIDATE_CMD:
|
||||||
if (current.backup_id == 0 && target_time == 0 && target_xid == 0)
|
if (current.backup_id == 0 && target_time == 0 && target_xid == 0 && !target_lsn)
|
||||||
return do_validate_all();
|
return do_validate_all();
|
||||||
else
|
else
|
||||||
return do_restore_or_validate(current.backup_id,
|
return do_restore_or_validate(current.backup_id,
|
||||||
|
@ -143,10 +143,16 @@ do_restore_or_validate(time_t target_backup_id, pgRecoveryTarget *rt,
|
|||||||
/*
|
/*
|
||||||
* [PGPRO-1164] If BACKUP_ID is not provided for restore command,
|
* [PGPRO-1164] If BACKUP_ID is not provided for restore command,
|
||||||
* we must find the first valid(!) backup.
|
* we must find the first valid(!) backup.
|
||||||
|
|
||||||
|
* If target_backup_id is not provided, we can be sure that
|
||||||
|
* PITR for restore or validate is requested.
|
||||||
|
* So we can assume that user is more interested in recovery to specific point
|
||||||
|
* in time and NOT interested in revalidation of invalid backups.
|
||||||
|
* So based on that assumptions we should choose only OK and DONE backups
|
||||||
|
* as candidates for validate and restore.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
if (params->is_restore &&
|
if (target_backup_id == INVALID_BACKUP_ID &&
|
||||||
target_backup_id == INVALID_BACKUP_ID &&
|
|
||||||
(current_backup->status != BACKUP_STATUS_OK &&
|
(current_backup->status != BACKUP_STATUS_OK &&
|
||||||
current_backup->status != BACKUP_STATUS_DONE))
|
current_backup->status != BACKUP_STATUS_DONE))
|
||||||
{
|
{
|
||||||
@ -1079,7 +1085,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
if (parse_time(target_time, &dummy_time, false))
|
if (parse_time(target_time, &dummy_time, false))
|
||||||
rt->target_time = dummy_time;
|
rt->target_time = dummy_time;
|
||||||
else
|
else
|
||||||
elog(ERROR, "Invalid value for --recovery-target-time option %s",
|
elog(ERROR, "Invalid value for '--recovery-target-time' option %s",
|
||||||
target_time);
|
target_time);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1097,7 +1103,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
#endif
|
#endif
|
||||||
rt->target_xid = dummy_xid;
|
rt->target_xid = dummy_xid;
|
||||||
else
|
else
|
||||||
elog(ERROR, "Invalid value for --recovery-target-xid option %s",
|
elog(ERROR, "Invalid value for '--recovery-target-xid' option %s",
|
||||||
target_xid);
|
target_xid);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1110,7 +1116,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
if (parse_lsn(target_lsn, &dummy_lsn))
|
if (parse_lsn(target_lsn, &dummy_lsn))
|
||||||
rt->target_lsn = dummy_lsn;
|
rt->target_lsn = dummy_lsn;
|
||||||
else
|
else
|
||||||
elog(ERROR, "Invalid value of --recovery-target-lsn option %s",
|
elog(ERROR, "Invalid value of '--recovery-target-lsn' option %s",
|
||||||
target_lsn);
|
target_lsn);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1120,7 +1126,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
if (parse_bool(target_inclusive, &dummy_bool))
|
if (parse_bool(target_inclusive, &dummy_bool))
|
||||||
rt->target_inclusive = dummy_bool;
|
rt->target_inclusive = dummy_bool;
|
||||||
else
|
else
|
||||||
elog(ERROR, "Invalid value for --recovery-target-inclusive option %s",
|
elog(ERROR, "Invalid value for '--recovery-target-inclusive' option %s",
|
||||||
target_inclusive);
|
target_inclusive);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1129,7 +1135,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
{
|
{
|
||||||
if ((strcmp(target_stop, "immediate") != 0)
|
if ((strcmp(target_stop, "immediate") != 0)
|
||||||
&& (strcmp(target_stop, "latest") != 0))
|
&& (strcmp(target_stop, "latest") != 0))
|
||||||
elog(ERROR, "Invalid value for --recovery-target option %s",
|
elog(ERROR, "Invalid value for '--recovery-target' option %s",
|
||||||
target_stop);
|
target_stop);
|
||||||
|
|
||||||
recovery_target_specified++;
|
recovery_target_specified++;
|
||||||
@ -1147,7 +1153,7 @@ parseRecoveryTargetOptions(const char *target_time,
|
|||||||
if ((strcmp(target_action, "pause") != 0)
|
if ((strcmp(target_action, "pause") != 0)
|
||||||
&& (strcmp(target_action, "promote") != 0)
|
&& (strcmp(target_action, "promote") != 0)
|
||||||
&& (strcmp(target_action, "shutdown") != 0))
|
&& (strcmp(target_action, "shutdown") != 0))
|
||||||
elog(ERROR, "Invalid value for --recovery-target-action option %s",
|
elog(ERROR, "Invalid value for '--recovery-target-action' option %s",
|
||||||
target_action);
|
target_action);
|
||||||
|
|
||||||
rt->target_action = target_action;
|
rt->target_action = target_action;
|
||||||
|
@ -1437,6 +1437,9 @@ class BackupTest(ProbackupTest, unittest.TestCase):
|
|||||||
# @unittest.skip("skip")
|
# @unittest.skip("skip")
|
||||||
def test_basic_missing_file_permissions(self):
|
def test_basic_missing_file_permissions(self):
|
||||||
""""""
|
""""""
|
||||||
|
if os.name == 'nt':
|
||||||
|
return unittest.skip('Skipped because it is POSIX only test')
|
||||||
|
|
||||||
fname = self.id().split('.')[3]
|
fname = self.id().split('.')[3]
|
||||||
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
||||||
node = self.make_simple_node(
|
node = self.make_simple_node(
|
||||||
@ -1481,6 +1484,9 @@ class BackupTest(ProbackupTest, unittest.TestCase):
|
|||||||
# @unittest.skip("skip")
|
# @unittest.skip("skip")
|
||||||
def test_basic_missing_dir_permissions(self):
|
def test_basic_missing_dir_permissions(self):
|
||||||
""""""
|
""""""
|
||||||
|
if os.name == 'nt':
|
||||||
|
return unittest.skip('Skipped because it is POSIX only test')
|
||||||
|
|
||||||
fname = self.id().split('.')[3]
|
fname = self.id().split('.')[3]
|
||||||
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
backup_dir = os.path.join(self.tmp_path, module_name, fname, 'backup')
|
||||||
node = self.make_simple_node(
|
node = self.make_simple_node(
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import os
|
import os
|
||||||
import unittest
|
import unittest
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from .helpers.ptrack_helpers import ProbackupTest
|
from .helpers.ptrack_helpers import ProbackupTest, ProbackupException
|
||||||
from time import sleep
|
from time import sleep
|
||||||
|
|
||||||
|
|
||||||
@ -1459,3 +1459,45 @@ class RetentionTest(ProbackupTest, unittest.TestCase):
|
|||||||
|
|
||||||
# Clean after yourself
|
# Clean after yourself
|
||||||
self.del_test_dir(module_name, fname)
|
self.del_test_dir(module_name, fname)
|
||||||
|
|
||||||
|
def test_wal_purge_victim(self):
|
||||||
|
"""
|
||||||
|
https://github.com/postgrespro/pg_probackup/issues/103
|
||||||
|
"""
|
||||||
|
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()
|
||||||
|
|
||||||
|
# Make ERROR incremental backup
|
||||||
|
try:
|
||||||
|
self.backup_node(backup_dir, 'node', node, backup_type='page')
|
||||||
|
# we should die here because exception is what we expect to happen
|
||||||
|
self.assertEqual(
|
||||||
|
1, 0,
|
||||||
|
"Expecting Error because page backup should not be possible "
|
||||||
|
"without valid full backup.\n Output: {0} \n CMD: {1}".format(
|
||||||
|
repr(self.output), self.cmd))
|
||||||
|
except ProbackupException as e:
|
||||||
|
self.assertIn(
|
||||||
|
"ERROR: Valid backup on current timeline 1 is not found. "
|
||||||
|
"Create new FULL backup before an incremental one.",
|
||||||
|
e.message,
|
||||||
|
"\n Unexpected Error Message: {0}\n CMD: {1}".format(
|
||||||
|
repr(e.message), self.cmd))
|
||||||
|
|
||||||
|
page_id = self.show_pb(backup_dir, 'node')[0]['id']
|
||||||
|
|
||||||
|
sleep(1)
|
||||||
|
|
||||||
|
# Make FULL backup
|
||||||
|
self.backup_node(backup_dir, 'node', node, options=['--delete-wal'])
|
||||||
|
|
||||||
|
# Clean after yourself
|
||||||
|
self.del_test_dir(module_name, fname)
|
||||||
|
@ -3443,6 +3443,119 @@ class ValidateTest(ProbackupTest, unittest.TestCase):
|
|||||||
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
|
'\n Unexpected Error Message: {0}\n CMD: {1}'.format(
|
||||||
repr(e.message), self.cmd))
|
repr(e.message), self.cmd))
|
||||||
|
|
||||||
|
# @unittest.expectedFailure
|
||||||
|
# @unittest.skip("skip")
|
||||||
|
def test_validate_target_lsn(self):
|
||||||
|
"""
|
||||||
|
Check validation to specific LSN
|
||||||
|
"""
|
||||||
|
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'),
|
||||||
|
set_replication=True,
|
||||||
|
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()
|
||||||
|
|
||||||
|
# FULL backup
|
||||||
|
self.backup_node(backup_dir, 'node', node)
|
||||||
|
|
||||||
|
node.safe_psql(
|
||||||
|
"postgres",
|
||||||
|
"create table t_heap as select 1 as id, md5(i::text) as text, "
|
||||||
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
||||||
|
"from generate_series(0,10000) i")
|
||||||
|
|
||||||
|
node_restored = self.make_simple_node(
|
||||||
|
base_dir=os.path.join(module_name, fname, 'node_restored'))
|
||||||
|
node_restored.cleanup()
|
||||||
|
|
||||||
|
self.restore_node(backup_dir, 'node', node_restored)
|
||||||
|
|
||||||
|
node_restored.append_conf(
|
||||||
|
"postgresql.auto.conf", "port = {0}".format(node_restored.port))
|
||||||
|
|
||||||
|
node_restored.slow_start()
|
||||||
|
|
||||||
|
self.switch_wal_segment(node)
|
||||||
|
|
||||||
|
backup_id = self.backup_node(
|
||||||
|
backup_dir, 'node', node_restored,
|
||||||
|
data_dir=node_restored.data_dir)
|
||||||
|
|
||||||
|
target_lsn = self.show_pb(backup_dir, 'node')[1]['stop-lsn']
|
||||||
|
|
||||||
|
self.delete_pb(backup_dir, 'node', backup_id)
|
||||||
|
|
||||||
|
self.validate_pb(
|
||||||
|
backup_dir, 'node',
|
||||||
|
options=[
|
||||||
|
'--recovery-target-timeline=2',
|
||||||
|
'--recovery-target-lsn={0}'.format(target_lsn)])
|
||||||
|
|
||||||
|
# @unittest.expectedFailure
|
||||||
|
# @unittest.skip("skip")
|
||||||
|
def test_recovery_target_backup_victim(self):
|
||||||
|
"""
|
||||||
|
Check that for validation to recovery target
|
||||||
|
probackup chooses valid backup
|
||||||
|
https://github.com/postgrespro/pg_probackup/issues/104
|
||||||
|
"""
|
||||||
|
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'),
|
||||||
|
set_replication=True,
|
||||||
|
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()
|
||||||
|
|
||||||
|
# FULL backup
|
||||||
|
self.backup_node(backup_dir, 'node', node)
|
||||||
|
|
||||||
|
node.safe_psql(
|
||||||
|
"postgres",
|
||||||
|
"create table t_heap as select 1 as id, md5(i::text) as text, "
|
||||||
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
||||||
|
"from generate_series(0,10000) i")
|
||||||
|
|
||||||
|
target_time = node.safe_psql(
|
||||||
|
"postgres",
|
||||||
|
"select now()").rstrip()
|
||||||
|
|
||||||
|
node.safe_psql(
|
||||||
|
"postgres",
|
||||||
|
"create table t_heap1 as select 1 as id, md5(i::text) as text, "
|
||||||
|
"md5(repeat(i::text,10))::tsvector as tsvector "
|
||||||
|
"from generate_series(0,100) i")
|
||||||
|
|
||||||
|
gdb = self.backup_node(backup_dir, 'node', node, gdb=True)
|
||||||
|
|
||||||
|
gdb.set_breakpoint('pg_stop_backup')
|
||||||
|
gdb.run_until_break()
|
||||||
|
gdb.remove_all_breakpoints()
|
||||||
|
gdb._execute('signal SIGINT')
|
||||||
|
gdb.continue_execution_until_error()
|
||||||
|
|
||||||
|
backup_id = self.show_pb(backup_dir, 'node')[1]['id']
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
'ERROR',
|
||||||
|
self.show_pb(backup_dir, 'node', backup_id)['status'],
|
||||||
|
'Backup STATUS should be "ERROR"')
|
||||||
|
|
||||||
|
self.validate_pb(
|
||||||
|
backup_dir, 'node',
|
||||||
|
options=['--recovery-target-time={0}'.format(target_time)])
|
||||||
|
|
||||||
|
|
||||||
# validate empty backup list
|
# validate empty backup list
|
||||||
# page from future during validate
|
# page from future during validate
|
||||||
# page from future during backup
|
# page from future during backup
|
||||||
|
Loading…
x
Reference in New Issue
Block a user