#################################################################################################################################### # 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; use pgBackRest::Common::Exception; use pgBackRest::Common::Log; use pgBackRest::Config::Config; #################################################################################################################################### # 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); #################################################################################################################################### # 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); #################################################################################################################################### # 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); #################################################################################################################################### # 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); 1;