2017-01-11 02:59:32 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# ARCHIVE GET MODULE
|
|
|
|
####################################################################################################################################
|
|
|
|
package pgBackRest::Archive::ArchiveGet;
|
|
|
|
use parent 'pgBackRest::Archive::Archive';
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings FATAL => qw(all);
|
|
|
|
use Carp qw(confess);
|
|
|
|
use English '-no_match_vars';
|
|
|
|
|
|
|
|
use Exporter qw(import);
|
|
|
|
our @EXPORT = qw();
|
|
|
|
use Fcntl qw(SEEK_CUR O_RDONLY O_WRONLY O_CREAT);
|
|
|
|
use File::Basename qw(dirname basename);
|
|
|
|
use Scalar::Util qw(blessed);
|
|
|
|
|
|
|
|
use pgBackRest::Common::Exception;
|
|
|
|
use pgBackRest::Common::Lock;
|
|
|
|
use pgBackRest::Common::Log;
|
|
|
|
use pgBackRest::Archive::ArchiveCommon;
|
|
|
|
use pgBackRest::Archive::ArchiveInfo;
|
|
|
|
use pgBackRest::Common::String;
|
|
|
|
use pgBackRest::Common::Wait;
|
|
|
|
use pgBackRest::Config::Config;
|
|
|
|
use pgBackRest::Db;
|
|
|
|
use pgBackRest::DbVersion;
|
2017-05-15 17:12:14 +02:00
|
|
|
use pgBackRest::Protocol::Helper;
|
2017-06-09 23:51:41 +02:00
|
|
|
use pgBackRest::Protocol::Storage::Helper;
|
|
|
|
use pgBackRest::Storage::Base;
|
|
|
|
use pgBackRest::Storage::Filter::Gzip;
|
|
|
|
use pgBackRest::Storage::Helper;
|
2017-01-11 02:59:32 +02:00
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# process
|
|
|
|
####################################################################################################################################
|
|
|
|
sub process
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '->process');
|
|
|
|
|
|
|
|
# Make sure the archive file is defined
|
|
|
|
if (!defined($ARGV[1]))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'WAL segment not provided', ERROR_PARAM_REQUIRED);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Make sure the destination file is defined
|
|
|
|
if (!defined($ARGV[2]))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'WAL segment destination not provided', ERROR_PARAM_REQUIRED);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Info for the Postgres log
|
|
|
|
&log(INFO, 'get WAL segment ' . $ARGV[1]);
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'iResult', value => $self->get($ARGV[1], $ARGV[2]), trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# get
|
|
|
|
####################################################################################################################################
|
|
|
|
sub get
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strSourceArchive,
|
|
|
|
$strDestinationFile
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->get', \@_,
|
|
|
|
{name => 'strSourceArchive'},
|
|
|
|
{name => 'strDestinationFile'}
|
|
|
|
);
|
|
|
|
|
|
|
|
lockStopTest();
|
|
|
|
|
2017-06-09 23:51:41 +02:00
|
|
|
# Get the repo storage
|
|
|
|
my $oStorageRepo = storageRepo();
|
2017-01-11 02:59:32 +02:00
|
|
|
|
|
|
|
# Construct absolute path to the WAL file when it is relative
|
|
|
|
$strDestinationFile = walPath($strDestinationFile, optionGet(OPTION_DB_PATH, false), commandGet());
|
|
|
|
|
|
|
|
# Get the wal segment filename
|
2017-01-27 18:02:27 +02:00
|
|
|
my ($strArchiveId, $strArchiveFile) = $self->getCheck(
|
2017-06-09 23:51:41 +02:00
|
|
|
undef, undef, walIsSegment($strSourceArchive) ? $strSourceArchive : undef);
|
2017-01-27 18:02:27 +02:00
|
|
|
|
|
|
|
if (!defined($strArchiveFile) && !walIsSegment($strSourceArchive) &&
|
2017-06-09 23:51:41 +02:00
|
|
|
$oStorageRepo->exists(STORAGE_REPO_ARCHIVE . "/${strArchiveId}/${strSourceArchive}"))
|
2017-01-27 18:02:27 +02:00
|
|
|
{
|
|
|
|
$strArchiveFile = $strSourceArchive;
|
|
|
|
}
|
2017-01-11 02:59:32 +02:00
|
|
|
|
|
|
|
# If there are no matching archive files then there are two possibilities:
|
|
|
|
# 1) The end of the archive stream has been reached, this is normal and a 1 will be returned
|
|
|
|
# 2) There is a hole in the archive stream and a hard error should be returned. However, holes are possible due to async
|
|
|
|
# archiving - so when to report a hole? Since a hard error will cause PG to terminate, for now treat as case #1.
|
|
|
|
my $iResult = 0;
|
|
|
|
|
|
|
|
if (!defined($strArchiveFile))
|
|
|
|
{
|
|
|
|
&log(INFO, "unable to find ${strSourceArchive} in the archive");
|
|
|
|
|
|
|
|
$iResult = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Determine if the source file is already compressed
|
2017-06-09 23:51:41 +02:00
|
|
|
my $bSourceCompressed = $strArchiveFile =~ ('^.*\.' . COMPRESS_EXT . '$') ? true : false;
|
2017-01-11 02:59:32 +02:00
|
|
|
|
|
|
|
# Copy the archive file to the requested location
|
2017-06-09 23:51:41 +02:00
|
|
|
$oStorageRepo->copy(
|
|
|
|
$oStorageRepo->openRead(
|
|
|
|
STORAGE_REPO_ARCHIVE . "/${strArchiveId}/${strArchiveFile}", {bProtocolCompress => !$bSourceCompressed}),
|
|
|
|
storageDb()->openWrite(
|
|
|
|
$strDestinationFile,
|
|
|
|
{rhyFilter => $bSourceCompressed ?
|
|
|
|
[{strClass => STORAGE_FILTER_GZIP, rxyParam => [{strCompressType => STORAGE_DECOMPRESS}]}] : undef}));
|
2017-01-11 02:59:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'iResult', value => $iResult}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# getArchiveId
|
|
|
|
#
|
|
|
|
# CAUTION: Only to be used by commands where the DB Version and DB System ID are not important such that the db-path is not valid
|
|
|
|
# for the command (i.e. expire command). Since this function will not check validity of the database version call getCheck()
|
|
|
|
# instead.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub getArchiveId
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my ($strOperation) = logDebugParam(__PACKAGE__ . '->getArchiveId');
|
|
|
|
|
|
|
|
my $strArchiveId;
|
|
|
|
|
2017-06-09 23:51:41 +02:00
|
|
|
if (!isRepoLocal())
|
2017-01-11 02:59:32 +02:00
|
|
|
{
|
2017-06-09 23:51:41 +02:00
|
|
|
$strArchiveId = protocolGet(BACKUP)->cmdExecute(OP_ARCHIVE_GET_ARCHIVE_ID, undef, true);
|
2017-01-11 02:59:32 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-06-09 23:51:41 +02:00
|
|
|
$strArchiveId = (new pgBackRest::Archive::ArchiveInfo(storageRepo()->pathGet(STORAGE_REPO_ARCHIVE), true))->archiveId();
|
2017-01-11 02:59:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strArchiveId', value => $strArchiveId, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|