1
0
mirror of https://github.com/pgbackrest/pgbackrest.git synced 2025-01-06 03:53:59 +02:00
pgbackrest/lib/pgBackRest/File.pm

1713 lines
54 KiB
Perl
Raw Normal View History

2014-02-03 03:03:05 +03:00
####################################################################################################################################
# FILE MODULE
####################################################################################################################################
New simpler configuration and consistent project/exe/path naming. * The repo-path option now always refers to the repository where backups and archive are stored, whether local or remote, so the repo-remote-path option has been removed. The new spool-path option can be used to define a location for queueing WAL segments when archiving asynchronously. Otherwise, a local repository is no longer required. * Implemented a new config format which should be far simpler to use. See the User Guide and Configuration Reference for details but for a simple configuration all options can now be placed in the stanza section. Options that are shared between stanzas can be placed in the [global] section. More complex configurations can still make use of command sections though this should be a rare use case. * The default configuration filename is now pgbackrest.conf instead of pg_backrest.conf. This was done for consistency with other naming changes but also to prevent old config files from being loaded accidentally. * The default repository name was changed from /var/lib/backup to /var/lib/pgbackrest. * Lock files are now stored in /tmp/pgbackrest by default. These days /run/pgbackrest would be the preferred location but that would require init scripts which are not part of this release. The lock-path option can be used to configure the lock directory. * Log files are now stored in /var/log/pgbackrest by default and no longer have the date appended so they can be managed with logrotate. The log-path option can be used to configure the lock directory. * Executable filename changed from pg_backrest to pgbackrest.
2016-04-14 15:30:54 +02:00
package pgBackRest::File;
2014-02-03 03:03:05 +03:00
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(:mode :flock O_RDONLY O_WRONLY O_CREAT);
use File::Basename qw(dirname basename);
use File::Copy qw(cp);
use File::Path qw(make_path remove_tree);
use File::stat;
use IO::Handle;
2014-02-03 03:03:05 +03:00
New simpler configuration and consistent project/exe/path naming. * The repo-path option now always refers to the repository where backups and archive are stored, whether local or remote, so the repo-remote-path option has been removed. The new spool-path option can be used to define a location for queueing WAL segments when archiving asynchronously. Otherwise, a local repository is no longer required. * Implemented a new config format which should be far simpler to use. See the User Guide and Configuration Reference for details but for a simple configuration all options can now be placed in the stanza section. Options that are shared between stanzas can be placed in the [global] section. More complex configurations can still make use of command sections though this should be a rare use case. * The default configuration filename is now pgbackrest.conf instead of pg_backrest.conf. This was done for consistency with other naming changes but also to prevent old config files from being loaded accidentally. * The default repository name was changed from /var/lib/backup to /var/lib/pgbackrest. * Lock files are now stored in /tmp/pgbackrest by default. These days /run/pgbackrest would be the preferred location but that would require init scripts which are not part of this release. The lock-path option can be used to configure the lock directory. * Log files are now stored in /var/log/pgbackrest by default and no longer have the date appended so they can be managed with logrotate. The log-path option can be used to configure the lock directory. * Executable filename changed from pg_backrest to pgbackrest.
2016-04-14 15:30:54 +02:00
use pgBackRest::Common::Exception;
use pgBackRest::Common::Log;
use pgBackRest::Common::String;
use pgBackRest::Common::Wait;
use pgBackRest::FileCommon;
use pgBackRest::Protocol::Common;
use pgBackRest::Version;
2014-02-03 03:03:05 +03:00
2014-06-06 06:51:27 +03:00
####################################################################################################################################
# PATH_GET constants
2014-06-06 06:51:27 +03:00
####################################################################################################################################
use constant PATH_ABSOLUTE => 'absolute';
push @EXPORT, qw(PATH_ABSOLUTE);
use constant PATH_DB => 'db';
push @EXPORT, qw(PATH_DB);
use constant PATH_DB_ABSOLUTE => 'db:absolute';
push @EXPORT, qw(PATH_DB_ABSOLUTE);
use constant PATH_BACKUP => 'backup';
push @EXPORT, qw(PATH_BACKUP);
use constant PATH_BACKUP_ABSOLUTE => 'backup:absolute';
push @EXPORT, qw(PATH_BACKUP_ABSOLUTE);
use constant PATH_BACKUP_CLUSTER => 'backup:cluster';
push @EXPORT, qw(PATH_BACKUP_CLUSTER);
use constant PATH_BACKUP_TMP => 'backup:tmp';
push @EXPORT, qw(PATH_BACKUP_TMP);
use constant PATH_BACKUP_ARCHIVE => 'backup:archive';
push @EXPORT, qw(PATH_BACKUP_ARCHIVE);
use constant PATH_BACKUP_ARCHIVE_OUT => 'backup:archive:out';
push @EXPORT, qw(PATH_BACKUP_ARCHIVE_OUT);
2014-06-07 23:06:46 +03:00
####################################################################################################################################
# STD pipe constants
2014-06-07 23:06:46 +03:00
####################################################################################################################################
use constant PIPE_STDIN => '<STDIN>';
push @EXPORT, qw(PIPE_STDIN);
use constant PIPE_STDOUT => '<STDOUT>';
push @EXPORT, qw(PIPE_STDOUT);
use constant PIPE_STDERR => '<STDERR>';
push @EXPORT, qw(PIPE_STDERR);
2014-02-07 00:37:37 +03:00
####################################################################################################################################
# new
2014-02-07 00:37:37 +03:00
####################################################################################################################################
2014-10-10 22:13:28 +03:00
sub new
2014-02-03 03:03:05 +03:00
{
2014-10-10 22:13:28 +03:00
my $class = shift;
# Create the class hash
my $self = {};
bless $self, $class;
# Assign function parameters, defaults, and log debug info
(
my $strOperation,
$self->{strStanza},
$self->{strBackupPath},
$self->{oProtocol},
$self->{strDefaultPathMode},
$self->{strDefaultFileMode},
) =
logDebugParam
(
__PACKAGE__ . '->new', \@_,
{name => 'strStanza', required => false},
{name => 'strBackupPath'},
{name => 'oProtocol'},
{name => 'strDefaultPathMode', default => '0750'},
{name => 'strDefaultFileMode', default => '0640'},
);
2014-10-10 22:13:28 +03:00
# Default compression extension to gz
$self->{strCompressExtension} = 'gz';
# Remote object must be set
if (!defined($self->{oProtocol}))
{
confess &log(ASSERT, 'oProtocol must be defined');
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'self', value => $self}
);
2014-02-03 03:03:05 +03:00
}
####################################################################################################################################
# DESTROY
####################################################################################################################################
sub DESTROY
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my ($strOperation) = logDebugParam(__PACKAGE__ . '->DESTROY');
if (defined($self->{oProtocol}))
{
$self->{oProtocol} = undef;
}
# Return from function and log return values if any
return logDebugReturn($strOperation);
}
####################################################################################################################################
# stanza
####################################################################################################################################
sub stanza
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my ($strOperation) = logDebugParam(__PACKAGE__ . '->stanza');
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'strStanza', $self->{strStanza}, trace => true}
);
}
2014-04-28 16:13:25 +03:00
####################################################################################################################################
# pathTypeGet
2014-04-28 16:13:25 +03:00
####################################################################################################################################
sub pathTypeGet
2014-02-03 03:03:05 +03:00
{
2014-02-06 05:39:08 +03:00
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strType
) =
logDebugParam
(
__PACKAGE__ . '->pathTypeGet', \@_,
{name => 'strType', trace => true}
);
my $strPath;
2014-02-07 00:37:37 +03:00
2014-06-04 05:02:56 +03:00
# If absolute type
2014-06-04 02:03:03 +03:00
if ($strType eq PATH_ABSOLUTE)
{
$strPath = PATH_ABSOLUTE;
2014-06-04 02:03:03 +03:00
}
2014-06-04 05:02:56 +03:00
# If db type
2014-06-04 02:03:03 +03:00
elsif ($strType =~ /^db(\:.*){0,1}/)
2014-02-03 03:03:05 +03:00
{
$strPath = PATH_DB;
2014-02-03 03:03:05 +03:00
}
# Else if backup type
elsif ($strType =~ /^backup(\:.*){0,1}/)
{
$strPath = PATH_BACKUP;
}
# Else error when path type not recognized
else
{
confess &log(ASSERT, "no known path types in '${strType}'");
2014-02-03 03:03:05 +03:00
}
2014-02-07 00:37:37 +03:00
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'strPath', value => $strPath, trace => true}
);
2014-02-03 03:03:05 +03:00
}
2014-06-04 02:03:03 +03:00
####################################################################################################################################
# pathGet
# ??? Need to tackle the return paths in this function (i.e. there are to many ways to return)
2014-06-04 02:03:03 +03:00
####################################################################################################################################
sub pathGet
2014-02-03 03:03:05 +03:00
{
2014-02-06 05:39:08 +03:00
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strType, # Base type of the path to get (PATH_DB_ABSOLUTE, PATH_BACKUP_TMP, etc)
$strFile, # File to append to the base path (can include a path as well)
$bTemp # Return the temp file for this path type - only some types have temp files
) =
logDebugParam
(
__PACKAGE__ . '->pathGet', \@_,
{name => 'strType', trace => true},
{name => 'strFile', required => false, trace => true},
{name => 'bTemp', default => false, trace => true}
);
2014-02-03 03:03:05 +03:00
# Is this an absolute path type?
2014-06-04 02:03:03 +03:00
my $bAbsolute = $strType =~ /.*absolute.*/;
# Make sure a temp file is valid for this type and file
if ($bTemp)
2014-02-03 03:03:05 +03:00
{
# Only allow temp files for PATH_BACKUP_ARCHIVE, PATH_BACKUP_ARCHIVE_OUT, PATH_BACKUP_TMP and any absolute path
if (!($strType eq PATH_BACKUP_ARCHIVE || $strType eq PATH_BACKUP_ARCHIVE_OUT || $strType eq PATH_BACKUP_TMP || $bAbsolute))
{
confess &log(ASSERT, 'temp file not supported for path type ' . $strType);
}
2014-02-03 03:03:05 +03:00
# The file must be defined
if (!defined($strFile))
{
confess &log(ASSERT, 'strFile must be defined when temp file requested');
}
2014-06-02 00:23:33 +03:00
}
2014-06-04 02:03:03 +03:00
# Get absolute path
if ($bAbsolute)
2014-02-03 03:03:05 +03:00
{
# Make sure that any absolute path starts with /, otherwise it will actually be relative
if ($strFile !~ /^\/.*/)
{
confess &log(ASSERT, "absolute path ${strType}:${strFile} must start with /");
}
if ($bTemp)
{
return
($strFile =~ "\.$self->{strCompressExtension}\$" ?
substr($strFile, 0, length($strFile) - (length($self->{strCompressExtension}) + 1)) : $strFile) .
'.' . BACKREST_EXE . '.tmp';
}
2014-02-03 03:03:05 +03:00
return $strFile;
}
2014-02-05 23:56:05 +03:00
2014-06-04 02:03:03 +03:00
# Make sure the base backup path is defined (since all other path types are backup)
2014-02-06 05:39:08 +03:00
if (!defined($self->{strBackupPath}))
2014-02-05 21:10:36 +03:00
{
confess &log(ASSERT, 'strBackupPath not defined');
2014-02-05 21:10:36 +03:00
}
2014-02-05 23:56:05 +03:00
# Get base backup path
if ($strType eq PATH_BACKUP)
{
return $self->{strBackupPath} . (defined($strFile) ? "/${strFile}" : '');
2014-02-05 23:56:05 +03:00
}
2014-02-05 21:10:36 +03:00
2014-02-05 23:56:05 +03:00
# Make sure the cluster is defined
if (!defined($self->{strStanza}))
2014-02-05 23:56:05 +03:00
{
confess &log(ASSERT, 'strStanza not defined');
2014-02-05 23:56:05 +03:00
}
2014-02-05 21:10:36 +03:00
2014-05-13 18:23:15 +03:00
# Get the backup tmp path
2014-02-05 23:56:05 +03:00
if ($strType eq PATH_BACKUP_TMP)
{
# return
# "$self->{strBackupPath}/temp/$self->{strStanza}.tmp" . (defined($strFile) ? "/${strFile}" : '') .
# ($bTemp ? '.' . BACKREST_EXE . '.tmp' : '');
return
"$self->{strBackupPath}/temp/$self->{strStanza}.tmp" .
# (defined($strFile) ?
# '/' . ($strFile =~ "\.$self->{strCompressExtension}\$" ?
# substr($strFile, 0, length($strFile) - (length($self->{strCompressExtension}) + 1)) : $strFile) : '') .
($bTemp ?
'/' . ($strFile =~ "\.$self->{strCompressExtension}\$" ?
substr($strFile, 0, length($strFile) - (length($self->{strCompressExtension}) + 1)) : $strFile) .
'.' . BACKREST_EXE . '.tmp' :
(defined($strFile) ? "/${strFile}" : ''));
2014-02-04 03:03:17 +03:00
}
2014-02-05 23:56:05 +03:00
# Get the backup archive path
if ($strType eq PATH_BACKUP_ARCHIVE_OUT || $strType eq PATH_BACKUP_ARCHIVE)
2014-02-03 03:03:05 +03:00
{
my $strArchivePath = "$self->{strBackupPath}/archive/$self->{strStanza}";
if (!defined($strFile))
{
return $strArchivePath;
}
if ($strType eq PATH_BACKUP_ARCHIVE)
2014-02-03 03:03:05 +03:00
{
my $strArchiveId = (split('/', $strFile))[0];
my $strArchiveFile = (split('/', $strFile))[1];
2014-02-05 23:56:05 +03:00
if (!defined($strArchiveFile))
2014-02-03 03:03:05 +03:00
{
return "${strArchivePath}/${strFile}";
}
my $strArchive = substr(basename($strArchiveFile), 0, 24);
if ($strArchive !~ /^([0-F]){24}$/)
{
return "${strArchivePath}/${strFile}";
2014-02-03 03:03:05 +03:00
}
2014-02-04 03:03:17 +03:00
$strArchivePath =
"${strArchivePath}/${strArchiveId}/" . substr($strArchive, 0, 16) .
($bTemp ? "/${strArchive}." . BACKREST_EXE . '.tmp' : "/${strArchiveFile}");
}
else
{
$strArchivePath =
"${strArchivePath}/out" . (defined($strFile) ? '/' . $strFile : '') .
($bTemp ? '.' . BACKREST_EXE . '.tmp' : '');
}
return $strArchivePath;
2014-02-05 23:56:05 +03:00
}
2014-02-04 03:03:17 +03:00
2014-02-05 23:56:05 +03:00
if ($strType eq PATH_BACKUP_CLUSTER)
{
return $self->{strBackupPath} . "/backup/$self->{strStanza}" . (defined($strFile) ? "/${strFile}" : '');
2014-02-03 03:03:05 +03:00
}
# Error when path type not recognized
confess &log(ASSERT, "no known path types in '${strType}'");
}
####################################################################################################################################
# isRemote
#
2014-06-07 04:16:24 +03:00
# Determine whether the path type is remote
####################################################################################################################################
sub isRemote
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType
) =
logDebugParam
(
__PACKAGE__ . '->isRemote', \@_,
{name => 'strPathType', trace => true}
);
my $bRemote = $self->{oProtocol}->isRemote() && $self->{oProtocol}->remoteTypeTest($self->pathTypeGet($strPathType));
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'bRemote', value => $bRemote, trace => true}
);
}
2014-02-03 03:03:05 +03:00
####################################################################################################################################
# linkCreate
2014-02-03 03:03:05 +03:00
####################################################################################################################################
sub linkCreate
2014-02-03 03:03:05 +03:00
{
2014-02-06 05:39:08 +03:00
my $self = shift;
2014-02-07 00:37:37 +03:00
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strSourcePathType,
$strSourceFile,
$strDestinationPathType,
$strDestinationFile,
$bHard,
$bRelative,
$bPathCreate
) =
logDebugParam
(
__PACKAGE__ . '->linkCreate', \@_,
{name => 'strSourcePathType'},
{name => 'strSourceFile'},
{name => 'strDestinationPathType'},
{name => 'strDestinationFile'},
{name => 'bHard', default => false},
{name => 'bRelative', default => false},
{name => 'bPathCreate', default => true}
);
2014-06-04 18:58:30 +03:00
# Source and destination path types must be the same (e.g. both PATH_DB or both PATH_BACKUP, etc.)
if ($self->pathTypeGet($strSourcePathType) ne $self->pathTypeGet($strDestinationPathType))
2014-02-04 03:03:17 +03:00
{
confess &log(ASSERT, 'path types must be equal in link create');
2014-02-04 03:03:17 +03:00
}
# Generate source and destination files
my $strSource = $self->pathGet($strSourcePathType, $strSourceFile);
my $strDestination = $self->pathGet($strDestinationPathType, $strDestinationFile);
2014-02-03 03:03:05 +03:00
# Run remotely
if ($self->isRemote($strSourcePathType))
2014-02-03 03:03:05 +03:00
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
2014-02-03 03:03:05 +03:00
}
# Run locally
else
2014-02-03 03:03:05 +03:00
{
# If the destination path is backup and does not exist, create it
# ??? This should only happen when the link create errors
if ($bPathCreate && $self->pathTypeGet($strDestinationPathType) eq PATH_BACKUP)
2014-02-03 03:03:05 +03:00
{
filePathCreate(dirname($strDestination));
2014-02-03 03:03:05 +03:00
}
unless (-e $strSource)
2014-02-03 03:03:05 +03:00
{
if (-e $strSource . ".$self->{strCompressExtension}")
{
$strSource .= ".$self->{strCompressExtension}";
$strDestination .= ".$self->{strCompressExtension}";
}
else
{
# Error when a hardlink will be created on a missing file
if ($bHard)
{
confess &log(ASSERT, "unable to find ${strSource}(.$self->{strCompressExtension}) for link");
}
}
2014-02-03 03:03:05 +03:00
}
2014-02-07 00:37:37 +03:00
# Generate relative path if requested
if ($bRelative)
{
# Determine how much of the paths are common
my @strySource = split('/', $strSource);
my @stryDestination = split('/', $strDestination);
while (defined($strySource[0]) && defined($stryDestination[0]) && $strySource[0] eq $stryDestination[0])
{
shift(@strySource);
shift(@stryDestination);
}
# Add relative path sections
$strSource = '';
for (my $iIndex = 0; $iIndex < @stryDestination - 1; $iIndex++)
{
$strSource .= '../';
}
# Add path to source
$strSource .= join('/', @strySource);
logDebugMisc
(
$strOperation, 'apply relative path',
{name => 'strSource', value => $strSource, trace => true}
);
}
if ($bHard)
{
link($strSource, $strDestination)
or confess &log(ERROR, "unable to create hardlink from ${strSource} to ${strDestination}");
}
else
{
symlink($strSource, $strDestination)
or confess &log(ERROR, "unable to create symlink from ${strSource} to ${strDestination}");
}
2014-02-04 03:03:17 +03:00
}
# Return from function and log return values if any
return logDebugReturn($strOperation);
2014-02-03 03:03:05 +03:00
}
2014-02-05 02:48:39 +03:00
####################################################################################################################################
# move
2014-02-05 02:48:39 +03:00
#
# Moves a file locally or remotely.
####################################################################################################################################
2014-06-04 02:03:03 +03:00
sub move
2014-02-05 02:48:39 +03:00
{
2014-02-06 05:39:08 +03:00
my $self = shift;
2014-02-05 21:10:36 +03:00
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strSourcePathType,
$strSourceFile,
$strDestinationPathType,
$strDestinationFile,
$bDestinationPathCreate
) =
logDebugParam
(
__PACKAGE__ . '->move', \@_,
{name => 'strSourcePathType'},
{name => 'strSourceFile', required => false},
{name => 'strDestinationPathType'},
{name => 'strDestinationFile'},
{name => 'bDestinationPathCreate', default => false}
);
2014-06-21 21:19:03 +03:00
# Source and destination path types must be the same
if ($self->pathTypeGet($strSourcePathType) ne $self->pathTypeGet($strSourcePathType))
2014-02-05 21:10:36 +03:00
{
confess &log(ASSERT, 'source and destination path types must be equal');
2014-02-05 21:10:36 +03:00
}
# Set operation variables
my $strPathOpSource = $self->pathGet($strSourcePathType, $strSourceFile);
my $strPathOpDestination = $self->pathGet($strDestinationPathType, $strDestinationFile);
2014-02-05 02:48:39 +03:00
# Run remotely
if ($self->isRemote($strSourcePathType))
2014-02-05 02:48:39 +03:00
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
2014-02-05 02:48:39 +03:00
}
# Run locally
else
{
fileMove($strPathOpSource, $strPathOpDestination, $bDestinationPathCreate);
2014-02-05 02:48:39 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation
);
2014-02-05 02:48:39 +03:00
}
2014-06-21 22:17:09 +03:00
####################################################################################################################################
# compress
2014-06-21 22:17:09 +03:00
####################################################################################################################################
sub compress
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strFile,
$bRemoveSource
) =
logDebugParam
(
__PACKAGE__ . '->compress', \@_,
{name => 'strPathType'},
{name => 'strFile'},
{name => 'bRemoveSource', default => true}
);
2014-06-21 22:17:09 +03:00
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strFile);
2014-06-21 22:17:09 +03:00
# Run remotely
if ($self->isRemote($strPathType))
2014-06-21 22:17:09 +03:00
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
2014-06-21 22:17:09 +03:00
}
# Run locally
else
{
# Use copy to compress the file
$self->copy($strPathType, $strFile, $strPathType, "${strFile}.gz", false, true);
2014-06-22 03:08:49 +03:00
# Remove the old file
if ($bRemoveSource)
{
fileRemove($strPathOp);
}
2014-06-21 22:17:09 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation
);
2014-06-21 22:17:09 +03:00
}
####################################################################################################################################
# pathCreate
2014-06-21 22:17:09 +03:00
#
# Creates a path locally or remotely.
####################################################################################################################################
sub pathCreate
2014-06-21 22:17:09 +03:00
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPath,
$strMode,
$bIgnoreExists,
$bCreateParents
) =
logDebugParam
(
__PACKAGE__ . '->pathCreate', \@_,
{name => 'strPathType'},
{name => 'strPath', required => false},
{name => 'strMode', default => '0750'},
{name => 'bIgnoreExists', default => false},
{name => 'bCreateParents', default => false}
);
2014-06-21 22:17:09 +03:00
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strPath);
2014-06-21 22:17:09 +03:00
if ($self->isRemote($strPathType))
2014-06-21 22:17:09 +03:00
{
# Build param hash
my %oParamHash;
$oParamHash{path} = ${strPathOp};
if (defined($strMode))
2014-06-21 22:46:26 +03:00
{
$oParamHash{mode} = ${strMode};
2014-06-21 22:46:26 +03:00
}
2014-06-21 22:17:09 +03:00
# Execute the command
$self->{oProtocol}->cmdExecute(OP_FILE_PATH_CREATE, \%oParamHash);
2014-06-21 22:17:09 +03:00
}
else
{
filePathCreate($strPathOp, $strMode, $bIgnoreExists, $bCreateParents);
2014-06-21 22:17:09 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation
);
2014-06-21 22:17:09 +03:00
}
####################################################################################################################################
# exists
#
# Checks for the existence of a file, but does not imply that the file is readable/writeable.
#
# Return: true if file exists, false otherwise
####################################################################################################################################
sub exists
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPath
) =
logDebugParam
(
__PACKAGE__ . '->exists', \@_,
{name => 'strPathType'},
{name => 'strPath', required => false}
);
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strPath);
my $bExists = true;
# Run remotely
if ($self->isRemote($strPathType))
{
# Build param hash
my %oParamHash;
2014-06-22 01:19:37 +03:00
$oParamHash{path} = $strPathOp;
# Execute the command
$bExists = $self->{oProtocol}->cmdExecute(OP_FILE_EXISTS, \%oParamHash, true) eq 'Y' ? true : false;
}
# Run locally
else
{
$bExists = fileExists($strPathOp);
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'bExists', value => $bExists}
);
}
2014-06-22 02:16:55 +03:00
####################################################################################################################################
# remove
2014-06-22 02:16:55 +03:00
####################################################################################################################################
2015-01-07 19:58:21 +02:00
sub remove
2014-06-22 02:16:55 +03:00
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPath,
$bTemp,
$bIgnoreMissing
) =
logDebugParam
(
__PACKAGE__ . '->remove', \@_,
{name => 'strPathType'},
{name => 'strPath'},
{name => 'bTemp', required => false},
{name => 'bIgnoreMissing', default => true}
);
2014-09-30 02:08:08 +03:00
2014-06-22 02:16:55 +03:00
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strPath, $bTemp);
2015-01-07 19:58:21 +02:00
my $bRemoved = true;
2014-06-22 02:16:55 +03:00
2015-01-07 19:58:21 +02:00
# Run remotely
if ($self->isRemote($strPathType))
2014-06-22 02:16:55 +03:00
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
2014-06-22 02:16:55 +03:00
}
2015-01-07 19:58:21 +02:00
# Run locally
2014-06-22 02:16:55 +03:00
else
{
$bRemoved = fileRemove($strPathOp, $bIgnoreMissing);
2015-01-07 19:58:21 +02:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'bRemoved', value => $bRemoved}
);
2015-01-07 19:58:21 +02:00
}
####################################################################################################################################
# hash
####################################################################################################################################
sub hash
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strFile,
$bCompressed,
$strHashType
) =
logDebugParam
(
__PACKAGE__ . '->hash', \@_,
{name => 'strPathType'},
{name => 'strFile'},
{name => 'bCompressed', required => false},
{name => 'strHashType', required => false}
);
my ($strHash) = $self->hashSize($strPathType, $strFile, $bCompressed, $strHashType);
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'strHash', value => $strHash, trace => true}
);
}
####################################################################################################################################
# hashSize
####################################################################################################################################
sub hashSize
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strFile,
$bCompressed,
$strHashType
) =
logDebugParam
(
__PACKAGE__ . '->hashSize', \@_,
{name => 'strPathType'},
{name => 'strFile'},
{name => 'bCompressed', default => false},
{name => 'strHashType', default => 'sha1'}
);
# Set operation variables
my $strFileOp = $self->pathGet($strPathType, $strFile);
my $strHash;
my $iSize = 0;
if ($self->isRemote($strPathType))
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
}
else
{
($strHash, $iSize) = fileHashSize($strFileOp, $bCompressed, $strHashType, $self->{oProtocol});
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'strHash', value => $strHash},
{name => 'iSize', value => $iSize}
);
}
####################################################################################################################################
# owner
2015-01-07 19:58:21 +02:00
####################################################################################################################################
sub owner
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strFile,
$strUser,
$strGroup
) =
logDebugParam
(
__PACKAGE__ . '->owner', \@_,
{name => 'strPathType'},
{name => 'strFile'},
{name => 'strUser'},
{name => 'strGroup'}
);
2014-06-22 02:16:55 +03:00
# Set operation variables
my $strFileOp = $self->pathGet($strPathType, $strFile);
2014-06-22 02:16:55 +03:00
if ($self->isRemote($strPathType))
2015-01-07 19:58:21 +02:00
{
confess &log(ASSERT, "${strOperation}: remote operation not supported");
2015-01-07 19:58:21 +02:00
}
else
{
my $iUserId;
my $iGroupId;
my $oStat;
if (!defined($strUser) || !defined($strGroup))
2014-09-30 02:08:08 +03:00
{
2015-01-07 19:58:21 +02:00
$oStat = stat($strFileOp);
if (!defined($oStat))
{
confess &log(ERROR, 'unable to stat ${strFileOp}');
}
}
if (defined($strUser))
{
$iUserId = getpwnam($strUser);
2014-09-30 02:08:08 +03:00
}
else
{
2015-01-07 19:58:21 +02:00
$iUserId = $oStat->uid;
2014-09-30 02:08:08 +03:00
}
2014-06-22 02:16:55 +03:00
2015-01-07 19:58:21 +02:00
if (defined($strGroup))
{
$iGroupId = getgrnam($strGroup);
}
else
{
$iGroupId = $oStat->gid;
}
2014-06-22 02:16:55 +03:00
2015-01-07 19:58:21 +02:00
chown($iUserId, $iGroupId, $strFileOp)
or confess &log(ERROR, "unable to set ownership for ${strFileOp}");
2014-06-22 02:16:55 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation
);
2014-06-22 02:16:55 +03:00
}
####################################################################################################################################
# list
####################################################################################################################################
2014-06-22 01:19:37 +03:00
sub list
{
my $self = shift;
2014-02-03 03:03:05 +03:00
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPath,
$strExpression,
$strSortOrder,
$bIgnoreMissing
) =
logDebugParam
(
__PACKAGE__ . '->list', \@_,
{name => 'strPathType'},
{name => 'strPath', required => false},
{name => 'strExpression', required => false},
{name => 'strSortOrder', default => 'forward'},
{name => 'bIgnoreMissing', default => false}
);
2014-02-05 16:21:27 +03:00
2014-06-22 01:19:37 +03:00
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strPath);
2014-06-22 01:19:37 +03:00
my @stryFileList;
2014-02-03 03:03:05 +03:00
2014-06-22 01:19:37 +03:00
# Run remotely
if ($self->isRemote($strPathType))
{
2014-06-22 01:19:37 +03:00
# Build param hash
my %oParamHash;
2014-06-22 01:19:37 +03:00
$oParamHash{path} = $strPathOp;
$oParamHash{sort_order} = $strSortOrder;
$oParamHash{ignore_missing} = ${bIgnoreMissing};
2014-06-22 01:19:37 +03:00
if (defined($strExpression))
{
$oParamHash{expression} = $strExpression;
}
2014-06-22 01:19:37 +03:00
# Execute the command
my $strOutput = $self->{oProtocol}->cmdExecute(OP_FILE_LIST, \%oParamHash);
2014-06-22 03:08:49 +03:00
2014-06-22 01:19:37 +03:00
if (defined($strOutput))
{
@stryFileList = split(/\n/, $strOutput);
2014-06-16 13:12:38 +03:00
}
2014-06-06 06:51:27 +03:00
}
2014-06-22 01:19:37 +03:00
# Run locally
else
2014-02-03 03:03:05 +03:00
{
@stryFileList = fileList($strPathOp, $strExpression, $strSortOrder, $bIgnoreMissing);
2014-06-22 01:19:37 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'stryFileList', value => \@stryFileList}
);
2014-06-22 01:19:37 +03:00
}
####################################################################################################################################
# wait
#
# Wait until the next second. This is done in the file object because it must be performed on whichever side the db is on, local or
# remote. This function is used to make sure that no files are copied in the same second as the manifest is created. The reason is
# that the db might modify they file again in the same second as the copy and that change will not be visible to a subsequent
# incremental backup using timestamp/size to determine deltas.
####################################################################################################################################
sub wait
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$bWait
) =
logDebugParam
(
__PACKAGE__ . '->wait', \@_,
{name => 'strPathType'},
{name => 'bWait', default => true}
);
# Second when the function was called
my $lTimeBegin;
# Run remotely
if ($self->isRemote($strPathType))
{
# Build param hash
my %oParamHash;
$oParamHash{wait} = $bWait;
# Execute the command
$lTimeBegin = $self->{oProtocol}->cmdExecute(OP_FILE_WAIT, \%oParamHash, true);
}
# Run locally
else
{
# Wait the remainder of the current second
$lTimeBegin = waitRemainder($bWait);
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'lTimeBegin', value => $lTimeBegin, trace => true}
);
}
2014-06-22 01:19:37 +03:00
####################################################################################################################################
# manifest
2014-06-22 01:19:37 +03:00
#
# Builds a path/file manifest starting with the base path and including all subpaths. The manifest contains all the information
# needed to perform a backup or a delta with a previous backup.
####################################################################################################################################
sub manifest
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPath,
) =
logDebugParam
(
__PACKAGE__ . '->manifest', \@_,
{name => 'strPathType'},
{name => 'strPath', required => false},
);
# Set operation variables
my $strPathOp = $self->pathGet($strPathType, $strPath);
my $hManifest = {};
2014-06-22 01:19:37 +03:00
# Run remotely
if ($self->isRemote($strPathType))
2014-06-22 01:19:37 +03:00
{
# Build param hash
my %oParamHash;
$oParamHash{path} = $strPathOp;
# Execute the command
dataHashBuild($hManifest, $self->{oProtocol}->cmdExecute(OP_FILE_MANIFEST, \%oParamHash, true), "\t");
2014-02-03 03:03:05 +03:00
}
2014-06-22 01:19:37 +03:00
# Run locally
else
{
$self->manifestRecurse($strPathType, $strPathOp, undef, 0, $hManifest);
2014-06-22 01:19:37 +03:00
}
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'hManifest', value => $hManifest, trace => true}
);
2014-06-22 01:19:37 +03:00
}
sub manifestRecurse
2014-06-22 01:19:37 +03:00
{
2014-06-22 03:08:49 +03:00
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strPathType,
$strPathOp,
$strPathFileOp,
$iDepth,
$oManifestHashRef
) =
logDebugParam
(
__PACKAGE__ . '->manifestRecurse', \@_,
{name => 'strPathType'},
{name => 'strPathOp'},
{name => 'strPathFileOp', required => false},
{name => 'iDepth'},
{name => 'oManifestHashRef', required => false}
);
2014-06-22 01:19:37 +03:00
# Set operation and debug strings
my $strPathRead = $strPathOp . (defined($strPathFileOp) ? "/${strPathFileOp}" : '');
2014-06-22 01:19:37 +03:00
my $hPath;
my $strFilter;
# If this is the top level stat the path to discover if it is actually a file
if ($iDepth == 0 && !S_ISDIR((fileStat($strPathRead))->mode))
{
$strFilter = basename($strPathRead);
$strPathRead = dirname($strPathRead);
}
2014-06-22 01:19:37 +03:00
# Open the path
2014-06-22 01:19:37 +03:00
if (!opendir($hPath, $strPathRead))
{
my $strError = "${strPathRead} could not be read: " . $!;
my $iErrorCode = ERROR_PATH_OPEN;
2014-06-22 01:19:37 +03:00
# If the path does not exist and is not the root path requested then return, else error
# It's OK for paths to go away during execution (databases are a dynamic thing!)
if (!$self->exists(PATH_ABSOLUTE, $strPathRead))
2014-06-22 01:19:37 +03:00
{
if ($iDepth != 0)
{
return;
}
2014-06-22 01:19:37 +03:00
$strError = "${strPathRead} does not exist";
$iErrorCode = ERROR_PATH_MISSING;
2014-06-22 01:19:37 +03:00
}
confess &log(ERROR, $strError, $iErrorCode);
}
2014-06-22 01:19:37 +03:00
# Get a list of all files in the path (except ..)
2014-06-22 01:19:37 +03:00
my @stryFileList = grep(!/^\..$/i, readdir($hPath));
close($hPath);
# Loop through all subpaths/files in the path
foreach my $strFile (sort(@stryFileList))
2014-06-22 01:19:37 +03:00
{
# Skip this file if it does not match the filter
if (defined($strFilter) && $strFile ne $strFilter)
{
next;
}
2014-06-22 01:19:37 +03:00
my $strPathFile = "${strPathRead}/$strFile";
my $bCurrentDir = $strFile eq '.';
2014-06-22 01:19:37 +03:00
# Create the file and path names
2014-06-22 01:19:37 +03:00
if ($iDepth != 0)
{
if ($bCurrentDir)
{
$strFile = $strPathFileOp;
$strPathFile = $strPathRead;
}
else
{
$strFile = "${strPathFileOp}/${strFile}";
}
}
# Stat the path/file
2014-06-22 01:19:37 +03:00
my $oStat = lstat($strPathFile);
# Check for errors in stat
2014-06-22 01:19:37 +03:00
if (!defined($oStat))
{
2014-06-22 03:08:49 +03:00
my $strError = "${strPathFile} could not be read: " . $!;
my $iErrorCode = ERROR_FILE_READ;
2014-06-22 01:19:37 +03:00
# If the file does not exist then go to the next file, else error
# It's OK for files to go away during execution (databases are a dynamic thing!)
if (!$self->exists(PATH_ABSOLUTE, $strPathFile))
2014-06-22 03:08:49 +03:00
{
next;
2014-06-22 03:08:49 +03:00
}
2014-06-22 01:19:37 +03:00
confess &log(ERROR, $strError, $iErrorCode);
}
2014-06-22 01:19:37 +03:00
# Check for regular file
if (S_ISREG($oStat->mode))
{
${$oManifestHashRef}{$strFile}{type} = 'f';
2014-06-22 01:19:37 +03:00
# Get inode
${$oManifestHashRef}{$strFile}{inode} = $oStat->ino;
2014-06-22 01:19:37 +03:00
# Get size
${$oManifestHashRef}{$strFile}{size} = $oStat->size;
2014-06-22 01:19:37 +03:00
# Get modification time
${$oManifestHashRef}{$strFile}{modification_time} = $oStat->mtime;
2014-06-22 01:19:37 +03:00
}
# Check for directory
elsif (S_ISDIR($oStat->mode))
{
${$oManifestHashRef}{$strFile}{type} = 'd';
2014-06-22 01:19:37 +03:00
}
# Check for link
elsif (S_ISLNK($oStat->mode))
{
${$oManifestHashRef}{$strFile}{type} = 'l';
2014-06-22 01:19:37 +03:00
# Get link destination
${$oManifestHashRef}{$strFile}{link_destination} = readlink($strPathFile);
2014-06-22 01:19:37 +03:00
if (!defined(${$oManifestHashRef}{$strFile}{link_destination}))
2014-06-22 01:19:37 +03:00
{
if (-e $strPathFile)
{
my $strError = "${strPathFile} error reading link: " . $!;
if ($strPathType eq PATH_ABSOLUTE)
{
print $strError;
exit ERROR_LINK_OPEN;
2014-06-22 01:19:37 +03:00
}
confess &log(ERROR, $strError);
2014-06-22 01:19:37 +03:00
}
}
}
# Not a recognized type
2014-06-22 01:19:37 +03:00
else
{
my $strError = "${strPathFile} is not of type directory, file, or link";
if ($strPathType eq PATH_ABSOLUTE)
{
print $strError;
exit ERROR_FILE_INVALID;
2014-06-22 01:19:37 +03:00
}
confess &log(ERROR, $strError);
2014-06-22 01:19:37 +03:00
}
# Get user name
${$oManifestHashRef}{$strFile}{user} = getpwuid($oStat->uid);
2014-06-22 01:19:37 +03:00
# Get group name
${$oManifestHashRef}{$strFile}{group} = getgrgid($oStat->gid);
2014-06-22 01:19:37 +03:00
# Get mode
if (${$oManifestHashRef}{$strFile}{type} ne 'l')
2014-06-22 01:19:37 +03:00
{
${$oManifestHashRef}{$strFile}{mode} = sprintf('%04o', S_IMODE($oStat->mode));
2014-06-22 01:19:37 +03:00
}
# Recurse into directories
if (${$oManifestHashRef}{$strFile}{type} eq 'd' && !$bCurrentDir)
2014-06-22 01:19:37 +03:00
{
$self->manifestRecurse($strPathType, $strPathOp, $strFile, $iDepth + 1, $oManifestHashRef);
2014-06-22 01:19:37 +03:00
}
}
# Return from function and log return values if any
return logDebugReturn($strOperation);
2014-06-22 01:19:37 +03:00
}
####################################################################################################################################
# copy
2014-06-22 01:19:37 +03:00
#
# Copies a file from one location to another:
#
# * source and destination can be local or remote
# * wire and output compression/decompression are supported
# * intermediate temp files are used to prevent partial copies
# * modification time, mode, and ownership can be set on destination file
2014-06-22 01:19:37 +03:00
# * destination path can optionally be created
####################################################################################################################################
sub copy
{
my $self = shift;
# Assign function parameters, defaults, and log debug info
my
(
$strOperation,
$strSourcePathType,
$strSourceFile,
$strDestinationPathType,
$strDestinationFile,
$bSourceCompressed,
$bDestinationCompress,
$bIgnoreMissingSource,
$lModificationTime,
$strMode,
$bDestinationPathCreate,
$strUser,
$strGroup,
$bAppendChecksum
) =
logDebugParam
(
__PACKAGE__ . '->copy', \@_,
{name => 'strSourcePathType'},
{name => 'strSourceFile', required => false},
{name => 'strDestinationPathType'},
{name => 'strDestinationFile', required => false},
{name => 'bSourceCompressed', default => false},
{name => 'bDestinationCompress', default => false},
{name => 'bIgnoreMissingSource', default => false},
{name => 'lModificationTime', required => false},
{name => 'strMode', default => '0640'},
{name => 'bDestinationPathCreate', default => false},
{name => 'strUser', required => false},
{name => 'strGroup', required => false},
{name => 'bAppendChecksum', default => false}
);
2014-06-22 01:19:37 +03:00
# Set working variables
my $bSourceRemote = $self->isRemote($strSourcePathType) || $strSourcePathType eq PIPE_STDIN;
my $bDestinationRemote = $self->isRemote($strDestinationPathType) || $strDestinationPathType eq PIPE_STDOUT;
2014-06-22 01:19:37 +03:00
my $strSourceOp = $strSourcePathType eq PIPE_STDIN ?
$strSourcePathType : $self->pathGet($strSourcePathType, $strSourceFile);
2014-06-22 01:19:37 +03:00
my $strDestinationOp = $strDestinationPathType eq PIPE_STDOUT ?
$strDestinationPathType : $self->pathGet($strDestinationPathType, $strDestinationFile);
2014-06-22 01:19:37 +03:00
my $strDestinationTmpOp = $strDestinationPathType eq PIPE_STDOUT ?
undef : $self->pathGet($strDestinationPathType, $strDestinationFile, true);
2014-06-22 01:19:37 +03:00
# Checksum and size variables
my $strChecksum = undef;
my $iFileSize = undef;
my $bResult = true;
2014-06-22 01:19:37 +03:00
# Open the source and destination files (if needed)
my $hSourceFile;
my $hDestinationFile;
if (!$bSourceRemote)
{
if (!sysopen($hSourceFile, $strSourceOp, O_RDONLY))
2014-06-22 01:19:37 +03:00
{
my $strError = $!;
my $iErrorCode = ERROR_FILE_READ;
2014-06-22 01:19:37 +03:00
if ($!{ENOENT})
{
# $strError = 'file is missing';
$iErrorCode = ERROR_FILE_MISSING;
2014-06-22 01:19:37 +03:00
if ($bIgnoreMissingSource && $strDestinationPathType ne PIPE_STDOUT)
{
return false, undef, undef;
2014-06-22 01:19:37 +03:00
}
}
$strError = "cannot open source file ${strSourceOp}: " . $strError;
if ($strSourcePathType eq PATH_ABSOLUTE)
{
if ($strDestinationPathType eq PIPE_STDOUT)
{
$self->{oProtocol}->binaryXferAbort();
2014-06-22 01:19:37 +03:00
}
}
confess &log(ERROR, $strError, $iErrorCode);
2014-06-22 01:19:37 +03:00
}
}
if (!$bDestinationRemote)
{
my $iCreateFlag = O_WRONLY | O_CREAT;
2014-06-24 01:54:00 +03:00
# Open the destination temp file
if (!sysopen($hDestinationFile, $strDestinationTmpOp, $iCreateFlag))
2014-06-24 01:54:00 +03:00
{
if ($bDestinationPathCreate)
2014-06-24 01:54:00 +03:00
{
filePathCreate(dirname($strDestinationTmpOp), undef, true, true);
2014-06-24 01:54:00 +03:00
}
if (!$bDestinationPathCreate || !sysopen($hDestinationFile, $strDestinationTmpOp, $iCreateFlag))
2014-06-24 01:54:00 +03:00
{
my $strError = "unable to open ${strDestinationTmpOp}: " . $!;
my $iErrorCode = ERROR_FILE_READ;
if (!fileExists(dirname($strDestinationTmpOp)))
2014-06-24 01:54:00 +03:00
{
$strError = dirname($strDestinationTmpOp) . ' destination path does not exist';
$iErrorCode = ERROR_FILE_MISSING;
2014-06-24 01:54:00 +03:00
}
if (!($bDestinationPathCreate && $iErrorCode == ERROR_FILE_MISSING))
{
confess &log(ERROR, $strError, $iErrorCode);
}
2014-06-24 01:54:00 +03:00
}
}
# Now lock the file to be sure nobody else is operating on it
if (!flock($hDestinationFile, LOCK_EX | LOCK_NB))
{
confess &log(ERROR, "unable to acquire exclusive lock on ${strDestinationTmpOp}", ERROR_LOCK_ACQUIRE);
}
2014-06-22 01:19:37 +03:00
}
# If source or destination are remote
if ($bSourceRemote || $bDestinationRemote)
{
# Build the command and open the local file
my $hFile;
my %oParamHash;
my $hIn,
my $hOut;
my $strRemote;
my $strRemoteOp;
2014-06-22 01:19:37 +03:00
# If source is remote and destination is local
if ($bSourceRemote && !$bDestinationRemote)
{
$hOut = $hDestinationFile;
$strRemoteOp = OP_FILE_COPY_OUT;
2014-06-22 01:19:37 +03:00
$strRemote = 'in';
if ($strSourcePathType ne PIPE_STDIN)
2014-06-22 01:19:37 +03:00
{
$oParamHash{source_file} = $strSourceOp;
$oParamHash{source_compressed} = $bSourceCompressed;
$oParamHash{destination_compress} = $bDestinationCompress;
2014-06-22 01:19:37 +03:00
}
}
# Else if source is local and destination is remote
elsif (!$bSourceRemote && $bDestinationRemote)
{
$hIn = $hSourceFile;
$strRemoteOp = OP_FILE_COPY_IN;
2014-06-22 01:19:37 +03:00
$strRemote = 'out';
if ($strDestinationPathType ne PIPE_STDOUT)
2014-06-22 01:19:37 +03:00
{
$oParamHash{destination_file} = $strDestinationOp;
$oParamHash{source_compressed} = $bSourceCompressed;
2014-06-22 01:19:37 +03:00
$oParamHash{destination_compress} = $bDestinationCompress;
$oParamHash{destination_path_create} = $bDestinationPathCreate;
2014-06-22 01:19:37 +03:00
if (defined($strMode))
{
$oParamHash{mode} = $strMode;
}
if (defined($strUser))
{
$oParamHash{user} = $strUser;
}
if (defined($strGroup))
2014-06-28 21:32:34 +03:00
{
$oParamHash{group} = $strGroup;
2014-06-28 21:32:34 +03:00
}
if ($bAppendChecksum)
{
$oParamHash{append_checksum} = true;
}
2014-06-22 01:19:37 +03:00
}
}
# Else source and destination are remote
else
{
$strRemoteOp = OP_FILE_COPY;
2014-06-22 01:19:37 +03:00
$oParamHash{source_file} = $strSourceOp;
$oParamHash{source_compressed} = $bSourceCompressed;
$oParamHash{destination_file} = $strDestinationOp;
$oParamHash{destination_compress} = $bDestinationCompress;
$oParamHash{destination_path_create} = $bDestinationPathCreate;
2014-06-22 01:19:37 +03:00
if (defined($strMode))
{
$oParamHash{mode} = $strMode;
}
if (defined($strUser))
2014-06-28 21:32:34 +03:00
{
$oParamHash{user} = $strUser;
}
if (defined($strGroup))
{
$oParamHash{group} = $strGroup;
2014-06-28 21:32:34 +03:00
}
2014-06-22 01:19:37 +03:00
if ($bIgnoreMissingSource)
{
$oParamHash{ignore_missing_source} = $bIgnoreMissingSource;
}
if ($bAppendChecksum)
{
$oParamHash{append_checksum} = true;
}
2014-06-22 01:19:37 +03:00
}
# If an operation is defined then write it
if (%oParamHash)
{
$self->{oProtocol}->cmdWrite($strRemoteOp, \%oParamHash);
2014-06-22 01:19:37 +03:00
}
# Transfer the file (skip this for copies where both sides are remote)
if ($strRemoteOp ne OP_FILE_COPY)
2014-06-22 01:19:37 +03:00
{
($strChecksum, $iFileSize) =
$self->{oProtocol}->binaryXfer($hIn, $hOut, $strRemote, $bSourceCompressed, $bDestinationCompress);
2014-06-22 01:19:37 +03:00
}
# If this is the controlling process then wait for OK from remote
if (%oParamHash)
{
# Test for an error when reading output
my $strOutput;
2014-06-22 03:08:49 +03:00
2014-06-22 01:19:37 +03:00
eval
{
$strOutput = $self->{oProtocol}->outputRead(true, $bIgnoreMissingSource);
# Check the result of the remote call
if (substr($strOutput, 0, 1) eq 'Y')
{
# If the operation was purely remote, get checksum/size
if ($strRemoteOp eq OP_FILE_COPY ||
$strRemoteOp eq OP_FILE_COPY_IN && $bSourceCompressed && !$bDestinationCompress)
{
# Checksum shouldn't already be set
if (defined($strChecksum) || defined($iFileSize))
{
confess &log(ASSERT, "checksum and size are already defined, but shouldn't be");
}
# Parse output and check to make sure tokens are defined
my @stryToken = split(/ /, $strOutput);
if (!defined($stryToken[1]) || !defined($stryToken[2]) ||
$stryToken[1] eq '?' && $stryToken[2] eq '?')
{
confess &log(ERROR, "invalid return from copy" . (defined($strOutput) ? ": ${strOutput}" : ''));
}
# Read the checksum and size
if ($stryToken[1] ne '?')
{
$strChecksum = $stryToken[1];
}
if ($stryToken[2] ne '?')
{
$iFileSize = $stryToken[2];
}
}
}
# Remote called returned false
else
{
$bResult = false;
}
2014-06-22 03:08:49 +03:00
return true;
}
2014-06-22 01:19:37 +03:00
# If there is an error then evaluate
or do
2014-06-22 01:19:37 +03:00
{
my $oException = $EVAL_ERROR;
2014-06-22 03:08:49 +03:00
# Ignore error if source file was missing and missing file exception was returned and bIgnoreMissingSource is set
if ($bIgnoreMissingSource && $strRemote eq 'in' && isException($oException) &&
$oException->code() == ERROR_FILE_MISSING)
2014-06-22 01:19:37 +03:00
{
close($hDestinationFile)
or confess &log(ERROR, "cannot close file ${strDestinationTmpOp}");
fileRemove($strDestinationTmpOp);
2014-06-22 01:19:37 +03:00
return false, undef, undef;
2014-06-22 01:19:37 +03:00
}
confess $oException;
};
2014-06-22 01:19:37 +03:00
}
}
# Else this is a local operation
else
2014-02-05 19:35:09 +03:00
{
# If the source is not compressed and the destination is then compress
if (!$bSourceCompressed && $bDestinationCompress)
{
($strChecksum, $iFileSize) =
$self->{oProtocol}->binaryXfer($hSourceFile, $hDestinationFile, 'out', false, true, false);
}
# If the source is compressed and the destination is not then decompress
elsif ($bSourceCompressed && !$bDestinationCompress)
{
($strChecksum, $iFileSize) =
$self->{oProtocol}->binaryXfer($hSourceFile, $hDestinationFile, 'in', true, false, false);
}
# Else both side are compressed, so copy capturing checksum
elsif ($bSourceCompressed)
{
($strChecksum, $iFileSize) =
$self->{oProtocol}->binaryXfer($hSourceFile, $hDestinationFile, 'out', true, true, false);
}
else
{
($strChecksum, $iFileSize) =
$self->{oProtocol}->binaryXfer($hSourceFile, $hDestinationFile, 'in', false, true, false);
}
2014-02-05 19:35:09 +03:00
}
2014-02-07 00:37:37 +03:00
# Close the source file (if local)
if (defined($hSourceFile))
{
close($hSourceFile) or confess &log(ERROR, "cannot close file ${strSourceOp}");
}
2014-06-06 06:51:27 +03:00
# Sync and close the destination file (if local)
if (defined($hDestinationFile))
{
$hDestinationFile->sync()
or confess &log(ERROR, "unable to sync ${strDestinationTmpOp}", ERROR_FILE_SYNC);
close($hDestinationFile)
or confess &log(ERROR, "cannot close file ${strDestinationTmpOp}");
}
2014-06-06 06:51:27 +03:00
# Checksum and file size should be set if the destination is not remote
if ($bResult &&
!(!$bSourceRemote && $bDestinationRemote && $bSourceCompressed) &&
(!defined($strChecksum) || !defined($iFileSize)))
{
confess &log(ASSERT, 'checksum or file size not set');
}
2014-06-06 05:42:47 +03:00
# Where the destination is local, set mode, modification time, and perform move to final location
if ($bResult && !$bDestinationRemote)
{
# Set the file Mode if required
if (defined($strMode))
{
chmod(oct($strMode), $strDestinationTmpOp)
or confess &log(ERROR, "unable to set mode for local ${strDestinationTmpOp}");
}
2014-06-06 05:42:47 +03:00
# Set the file modification time if required
if (defined($lModificationTime))
{
utime($lModificationTime, $lModificationTime, $strDestinationTmpOp)
or confess &log(ERROR, "unable to set time for local ${strDestinationTmpOp}");
}
# set user and/or group if required
if (defined($strUser) || defined($strGroup))
{
$self->owner(PATH_ABSOLUTE, $strDestinationTmpOp, $strUser, $strGroup);
}
# Replace checksum in destination filename (if exists)
if ($bAppendChecksum)
{
# Replace destination filename
if ($bDestinationCompress)
{
$strDestinationOp =
substr($strDestinationOp, 0, length($strDestinationOp) - length($self->{strCompressExtension}) - 1) .
'-' . $strChecksum . '.' . $self->{strCompressExtension};
}
else
{
$strDestinationOp .= '-' . $strChecksum;
}
}
# Move the file from tmp to final destination
fileMove($strDestinationTmpOp, $strDestinationOp, $bDestinationPathCreate);
}
2014-02-07 00:37:37 +03:00
# Return from function and log return values if any
return logDebugReturn
(
$strOperation,
{name => 'bResult', value => $bResult, trace => true},
{name => 'strChecksum', value => $strChecksum, trace => true},
{name => 'iFileSize', value => $iFileSize, trace => true}
);
2014-02-03 03:03:05 +03:00
}
2014-06-02 00:23:33 +03:00
2014-10-10 22:13:28 +03:00
1;