2016-08-24 18:39:27 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# ARCHIVE COMMON MODULE
|
|
|
|
####################################################################################################################################
|
|
|
|
package pgBackRest::ArchiveCommon;
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings FATAL => qw(all);
|
|
|
|
use Carp qw(confess);
|
|
|
|
|
|
|
|
use Exporter qw(import);
|
|
|
|
our @EXPORT = qw();
|
|
|
|
|
|
|
|
use pgBackRest::DbVersion;
|
2017-01-05 01:39:50 +02:00
|
|
|
use pgBackRest::Common::Exception;
|
2016-08-24 18:39:27 +02:00
|
|
|
use pgBackRest::Common::Log;
|
2017-01-05 01:39:50 +02:00
|
|
|
use pgBackRest::Config::Config;
|
2016-12-03 17:28:08 +02:00
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# RegEx constants
|
|
|
|
####################################################################################################################################
|
|
|
|
use constant REGEX_ARCHIVE_DIR_DB_VERSION => '^[0-9]+\.[0-9]+-[0-9]+$';
|
|
|
|
push @EXPORT, qw(REGEX_ARCHIVE_DIR_DB_VERSION);
|
|
|
|
use constant REGEX_ARCHIVE_DIR_WAL => '^[0-F]{16}$';
|
|
|
|
push @EXPORT, qw(REGEX_ARCHIVE_DIR_WAL);
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# PostgreSQL WAL system id offset
|
|
|
|
####################################################################################################################################
|
|
|
|
use constant PG_WAL_SYSTEM_ID_OFFSET_GTE_93 => 20;
|
|
|
|
push @EXPORT, qw(PG_WAL_SYSTEM_ID_OFFSET_GTE_93);
|
|
|
|
use constant PG_WAL_SYSTEM_ID_OFFSET_LT_93 => 12;
|
|
|
|
push @EXPORT, qw(PG_WAL_SYSTEM_ID_OFFSET_LT_93);
|
|
|
|
|
2016-08-24 18:39:27 +02:00
|
|
|
####################################################################################################################################
|
2016-08-25 17:25:46 +02:00
|
|
|
# lsnNormalize
|
|
|
|
#
|
|
|
|
# Generates a normalized form from an LSN that can be used for comparison.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub lsnNormalize
|
|
|
|
{
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strLsn,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . 'lsnFile', \@_,
|
|
|
|
{name => 'strLsn', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
# Split the LSN into major and minor parts
|
|
|
|
my @stryLsnSplit = split('/', $strLsn);
|
|
|
|
|
|
|
|
if (@stryLsnSplit != 2)
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "invalid lsn ${strLsn}");
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strLsnNormal = uc(sprintf("%08s%08s", $stryLsnSplit[0], $stryLsnSplit[1]));
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strLsnNormal', value => $strLsnNormal, trace => true}
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
push @EXPORT, qw(lsnNormalize);
|
|
|
|
|
|
|
|
####################################################################################################################################
|
2016-08-24 18:39:27 +02:00
|
|
|
# lsnFileRange
|
|
|
|
#
|
|
|
|
# Generates a range of WAL filenames given the start and stop LSN. For pre-9.3 databases, use bSkipFF to exclude the FF that
|
|
|
|
# prior versions did not generate.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub lsnFileRange
|
|
|
|
{
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strLsnStart,
|
|
|
|
$strLsnStop,
|
|
|
|
$strDbVersion,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . 'lsnFileRange', \@_,
|
|
|
|
{name => 'strLsnStart'},
|
|
|
|
{name => 'strLsnStop'},
|
|
|
|
{name => '$strDbVersion'},
|
|
|
|
);
|
|
|
|
|
|
|
|
# Working variables
|
|
|
|
my @stryArchive;
|
|
|
|
my $iArchiveIdx = 0;
|
|
|
|
my $bSkipFF = $strDbVersion < PG_VERSION_93;
|
|
|
|
|
|
|
|
# Iterate through all archive logs between start and stop
|
|
|
|
my @stryArchiveSplit = split('/', $strLsnStart);
|
|
|
|
my $iStartMajor = hex($stryArchiveSplit[0]);
|
|
|
|
my $iStartMinor = hex(substr(sprintf("%08s", $stryArchiveSplit[1]), 0, 2));
|
|
|
|
|
|
|
|
@stryArchiveSplit = split('/', $strLsnStop);
|
|
|
|
my $iStopMajor = hex($stryArchiveSplit[0]);
|
|
|
|
my $iStopMinor = hex(substr(sprintf("%08s", $stryArchiveSplit[1]), 0, 2));
|
|
|
|
|
|
|
|
$stryArchive[$iArchiveIdx] = uc(sprintf("%08x%08x", $iStartMajor, $iStartMinor));
|
|
|
|
$iArchiveIdx += 1;
|
|
|
|
|
|
|
|
while (!($iStartMajor == $iStopMajor && $iStartMinor == $iStopMinor))
|
|
|
|
{
|
|
|
|
$iStartMinor += 1;
|
|
|
|
|
|
|
|
if ($bSkipFF && $iStartMinor == 255 || !$bSkipFF && $iStartMinor == 256)
|
|
|
|
{
|
|
|
|
$iStartMajor += 1;
|
|
|
|
$iStartMinor = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$stryArchive[$iArchiveIdx] = uc(sprintf("%08x%08x", $iStartMajor, $iStartMinor));
|
|
|
|
$iArchiveIdx += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'stryWalFileName', value => \@stryArchive}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
push @EXPORT, qw(lsnFileRange);
|
|
|
|
|
2017-01-05 01:39:50 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# walPath
|
|
|
|
#
|
|
|
|
# Generates the location of the pg_xlog directory using a relative xlog path and the supplied db path.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub walPath
|
|
|
|
{
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strWalFile,
|
|
|
|
$strDbPath,
|
|
|
|
$strCommand,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . 'walPath', \@_,
|
|
|
|
{name => 'strWalFile', trace => true},
|
|
|
|
{name => 'strDbPath', trace => true, required => false},
|
|
|
|
{name => 'strCommand', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
if (index($strWalFile, '/') != 0)
|
|
|
|
{
|
|
|
|
if (!defined($strDbPath))
|
|
|
|
{
|
|
|
|
confess &log(ERROR,
|
|
|
|
"option 'db-path' must be specified when relative xlog paths are used\n" .
|
|
|
|
"HINT: Is \%f passed to ${strCommand} instead of \%p?\n" .
|
|
|
|
"HINT: PostgreSQL may pass relative paths even with \%p depending on the environment.",
|
|
|
|
ERROR_OPTION_REQUIRED);
|
|
|
|
}
|
|
|
|
|
|
|
|
$strWalFile = "${strDbPath}/${strWalFile}";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strWalFile', value => $strWalFile, trace => true}
|
|
|
|
);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
push @EXPORT, qw(walPath);
|
|
|
|
|
2016-08-24 18:39:27 +02:00
|
|
|
1;
|