2014-02-03 03:03:05 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# FILE MODULE
|
|
|
|
####################################################################################################################################
|
|
|
|
package pg_backrest_file;
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
use Carp;
|
|
|
|
use Net::OpenSSH;
|
|
|
|
use IPC::Open3;
|
|
|
|
use File::Basename;
|
|
|
|
use IPC::System::Simple qw(capture);
|
|
|
|
|
|
|
|
use lib dirname($0);
|
|
|
|
use pg_backrest_utility;
|
|
|
|
|
|
|
|
use Exporter qw(import);
|
|
|
|
|
2014-02-03 22:50:23 +03:00
|
|
|
our @EXPORT = qw(file_init_archive file_init_backup
|
2014-02-05 21:10:36 +03:00
|
|
|
path_get path_type_get is_remote
|
|
|
|
link_create path_create file_move file_copy file_list_get manifest_get file_hash_get
|
2014-02-05 02:48:39 +03:00
|
|
|
psql_execute
|
2014-02-04 03:03:17 +03:00
|
|
|
PATH_DB PATH_DB_ABSOLUTE PATH_BACKUP PATH_BACKUP_ABSOLUTE PATH_BACKUP_CLUSTER PATH_BACKUP_TMP PATH_BACKUP_ARCHIVE);
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
# Extension and permissions
|
|
|
|
our $strCompressExtension = "gz";
|
2014-02-04 04:48:02 +03:00
|
|
|
my $strDefaultPathPermission = "0750";
|
2014-02-05 19:35:09 +03:00
|
|
|
our $strDefaultFilePermission = "0640";
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
# Command strings
|
2014-02-04 04:48:02 +03:00
|
|
|
my $strCommandChecksum;
|
|
|
|
my $strCommandCompress;
|
|
|
|
my $strCommandDecompress;
|
|
|
|
my $strCommandCat = "cat %file%";
|
|
|
|
my $strCommandManifest;
|
|
|
|
my $strCommandPsql;
|
|
|
|
|
|
|
|
# Module variables
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strDbUser; # Database user
|
2014-02-04 04:48:02 +03:00
|
|
|
my $strDbHost; # Database host
|
|
|
|
my $oDbSSH; # Database SSH object
|
|
|
|
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strBackupUser; # Backup user
|
2014-02-04 04:48:02 +03:00
|
|
|
my $strBackupHost; # Backup host
|
|
|
|
my $oBackupSSH; # Backup SSH object
|
|
|
|
my $strBackupPath; # Backup base path
|
2014-02-06 00:53:25 +03:00
|
|
|
my $strBackupClusterPath; # Backup cluster path
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
# Process flags
|
2014-02-04 04:48:02 +03:00
|
|
|
my $bNoCompression;
|
2014-02-05 21:10:36 +03:00
|
|
|
my $strCluster;
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
####################################################################################################################################
|
2014-02-03 22:50:23 +03:00
|
|
|
# FILE_INIT_ARCHIVE
|
2014-02-03 03:03:05 +03:00
|
|
|
####################################################################################################################################
|
2014-02-03 22:50:23 +03:00
|
|
|
sub file_init_archive
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-04 04:23:04 +03:00
|
|
|
my $bNoCompressionParam = shift;
|
2014-02-03 22:50:23 +03:00
|
|
|
my $strCommandChecksumParam = shift;
|
|
|
|
my $strCommandCompressParam = shift;
|
|
|
|
my $strCommandDecompressParam = shift;
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strBackupUserParam = shift;
|
2014-02-03 22:50:23 +03:00
|
|
|
my $strBackupHostParam = shift;
|
|
|
|
my $strBackupPathParam = shift;
|
2014-02-05 21:10:36 +03:00
|
|
|
my $strClusterParam = shift;
|
2014-02-03 03:03:05 +03:00
|
|
|
|
2014-02-03 22:50:23 +03:00
|
|
|
# Assign parameters to module variables
|
2014-02-04 04:23:04 +03:00
|
|
|
$bNoCompression = $bNoCompressionParam;
|
2014-02-03 22:50:23 +03:00
|
|
|
$strCommandChecksum = $strCommandChecksumParam;
|
|
|
|
$strCommandCompress = $strCommandCompressParam;
|
|
|
|
$strCommandDecompress = $strCommandDecompressParam;
|
|
|
|
$strBackupPath = $strBackupPathParam;
|
2014-02-05 21:10:36 +03:00
|
|
|
$strCluster = $strClusterParam;
|
2014-02-05 02:48:39 +03:00
|
|
|
$strBackupUser = $strBackupUserParam;
|
2014-02-03 22:50:23 +03:00
|
|
|
$strBackupHost = $strBackupHostParam;
|
|
|
|
|
|
|
|
# Make sure the backup path is defined
|
2014-02-04 04:48:02 +03:00
|
|
|
if (!defined($strBackupPath))
|
2014-02-03 22:50:23 +03:00
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
confess &log(ERROR, "common:backup_path undefined");
|
2014-02-03 22:50:23 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
# Create the backup cluster path
|
|
|
|
$strBackupClusterPath = "${strBackupPath}/${strCluster}";
|
|
|
|
|
2014-02-04 04:48:02 +03:00
|
|
|
# Connect SSH object if backup host is defined
|
|
|
|
if (defined($strBackupHost))
|
2014-02-03 22:50:23 +03:00
|
|
|
{
|
2014-02-04 04:48:02 +03:00
|
|
|
&log(INFO, "connecting to backup ssh host ${strBackupHost}");
|
2014-02-03 22:50:23 +03:00
|
|
|
|
|
|
|
# !!! This could be improved by redirecting stderr to a file to get a better error message
|
2014-02-05 02:48:39 +03:00
|
|
|
$oBackupSSH = Net::OpenSSH->new($strBackupHost, master_stderr_discard => true, user => $strBackupUser);
|
2014-02-04 04:48:02 +03:00
|
|
|
$oBackupSSH->error and confess &log(ERROR, "unable to connect to $strBackupHost}: " . $oBackupSSH->error);
|
2014-02-03 22:50:23 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sub file_init_backup
|
|
|
|
{
|
|
|
|
my $strCommandManifestParam = shift;
|
|
|
|
my $strCommandPsqlParam = shift;
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strDbUserParam = shift;
|
2014-02-03 22:50:23 +03:00
|
|
|
my $strDbHostParam = shift;
|
|
|
|
|
|
|
|
$strCommandManifest = $strCommandManifestParam;
|
|
|
|
$strCommandPsql = $strCommandPsqlParam;
|
2014-02-05 02:48:39 +03:00
|
|
|
$strDbUser = $strDbUserParam;
|
2014-02-03 22:50:23 +03:00
|
|
|
$strDbHost = $strDbHostParam;
|
|
|
|
|
2014-02-04 04:48:02 +03:00
|
|
|
# Connect SSH object if db host is defined
|
2014-02-03 22:50:23 +03:00
|
|
|
if (defined($strDbHost))
|
|
|
|
{
|
|
|
|
&log(INFO, "connecting to database ssh host ${strDbHost}");
|
|
|
|
|
|
|
|
# !!! This could be improved by redirecting stderr to a file to get a better error message
|
2014-02-05 02:48:39 +03:00
|
|
|
$oDbSSH = Net::OpenSSH->new($strDbHost, master_stderr_discard => true, user => $strDbUser);
|
2014-02-04 04:48:02 +03:00
|
|
|
$oDbSSH->error and confess &log(ERROR, "unable to connect to ${strDbHost}: " . $oDbSSH->error);
|
2014-02-03 22:50:23 +03:00
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# PATH_GET
|
|
|
|
####################################################################################################################################
|
|
|
|
use constant
|
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
PATH_DB => 'db',
|
|
|
|
PATH_DB_ABSOLUTE => 'db:absolute',
|
|
|
|
PATH_BACKUP => 'backup',
|
|
|
|
PATH_BACKUP_ABSOLUTE => 'backup:absolute',
|
|
|
|
PATH_BACKUP_CLUSTER => 'backup:cluster',
|
|
|
|
PATH_BACKUP_TMP => 'backup:tmp',
|
|
|
|
PATH_BACKUP_ARCHIVE => 'backup:archive'
|
2014-02-03 03:03:05 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
sub path_type_get
|
|
|
|
{
|
|
|
|
my $strType = shift;
|
|
|
|
|
|
|
|
# If db type
|
|
|
|
if ($strType =~ /^db(\:.*){0,1}/)
|
|
|
|
{
|
|
|
|
return PATH_DB;
|
|
|
|
}
|
|
|
|
# Else if backup type
|
|
|
|
elsif ($strType =~ /^backup(\:.*){0,1}/)
|
|
|
|
{
|
|
|
|
return PATH_BACKUP;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Error when path type not recognized
|
|
|
|
confess &log(ASSERT, "no known path types in '${strType}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
sub path_get
|
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
my $strType = shift; # Base type of the path to get (PATH_DB_ABSOLUTE, PATH_BACKUP_TMP, etc)
|
|
|
|
my $strFile = shift; # File to append to the base path (can include a path as well)
|
|
|
|
my $bTemp = shift; # Return the temp file for this path type - only some types have temp files
|
2014-02-03 03:03:05 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
# Only allow temp files for PATH_BACKUP_ARCHIVE and PATH_BACKUP_TMP
|
2014-02-03 03:03:05 +03:00
|
|
|
if (defined($bTemp) && $bTemp && !($strType eq PATH_BACKUP_ARCHIVE || $strType eq PATH_BACKUP_TMP))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "temp file not supported on path " . $strType);
|
|
|
|
}
|
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
# Get absolute db path
|
2014-02-03 03:03:05 +03:00
|
|
|
if ($strType eq PATH_DB_ABSOLUTE)
|
|
|
|
{
|
|
|
|
return $strFile;
|
|
|
|
}
|
2014-02-05 23:56:05 +03:00
|
|
|
|
|
|
|
# Make sure the base backup path is defined
|
|
|
|
if (!defined($strBackupPath))
|
2014-02-05 21:10:36 +03:00
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
confess &log(ASSERT, "\$strBackupPath not yet defined");
|
2014-02-05 21:10:36 +03:00
|
|
|
}
|
2014-02-05 23:56:05 +03:00
|
|
|
|
|
|
|
# Get absolute backup path
|
|
|
|
if ($strType eq PATH_BACKUP_ABSOLUTE)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
# Need a check in here to make sure this is relative to the backup path
|
2014-02-04 03:03:17 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
return $strFile;
|
|
|
|
}
|
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 $strBackupPath . (defined($strFile) ? "/${strFile}" : "");
|
|
|
|
}
|
2014-02-05 21:10:36 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
# Make sure the cluster is defined
|
|
|
|
if (!defined($strCluster))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "\$strCluster not yet defined");
|
|
|
|
}
|
2014-02-05 21:10:36 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
# Get the backup tmp path
|
|
|
|
if ($strType eq PATH_BACKUP_TMP)
|
|
|
|
{
|
|
|
|
my $strTempPath = "${strBackupPath}/tmp/${strCluster}.tmp";
|
2014-02-05 21:10:36 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
if (defined($bTemp) && $bTemp)
|
|
|
|
{
|
|
|
|
return "${strTempPath}/file.tmp";
|
2014-02-05 21:10:36 +03:00
|
|
|
}
|
2014-02-05 23:56:05 +03:00
|
|
|
|
|
|
|
return "${strTempPath}" . (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)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
my $strArchivePath = "$strBackupPath/archive/${strCluster}";
|
|
|
|
my $strArchive;
|
2014-02-04 03:03:17 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
if (defined($strFile))
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
$strArchive = substr(basename($strFile), 0, 24);
|
|
|
|
|
|
|
|
if ($strArchive !~ /^([0-F]){24}$/)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-05 23:56:05 +03:00
|
|
|
return "${strArchivePath}/${strFile}";
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
2014-02-05 23:56:05 +03:00
|
|
|
}
|
2014-02-04 03:03:17 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
return $strArchivePath . (defined($strArchive) ? "/" . substr($strArchive, 0, 16) : "") .
|
|
|
|
(defined($strFile) ? "/" . $strFile : "");
|
|
|
|
}
|
2014-02-04 03:03:17 +03:00
|
|
|
|
2014-02-05 23:56:05 +03:00
|
|
|
if ($strType eq PATH_BACKUP_CLUSTER)
|
|
|
|
{
|
|
|
|
return $strBackupPath . "/backup/${strCluster}" . (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}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# LINK_CREATE
|
|
|
|
####################################################################################################################################
|
|
|
|
sub link_create
|
|
|
|
{
|
|
|
|
my $strSourcePathType = shift;
|
|
|
|
my $strSourceFile = shift;
|
|
|
|
my $strDestinationPathType = shift;
|
|
|
|
my $strDestinationFile = shift;
|
|
|
|
my $bHard = shift;
|
|
|
|
my $bRelative = shift;
|
|
|
|
|
2014-02-05 18:40:49 +03:00
|
|
|
# if bHard is not defined default to false
|
|
|
|
$bHard = defined($bHard) ? $bHard : false;
|
|
|
|
|
|
|
|
# if bRelative is not defined or bHard is true, default to false
|
|
|
|
$bRelative = !defined($bRelative) || $bHard ? false : $bRelative;
|
|
|
|
|
|
|
|
# Source and destination path types must be the same (both PATH_DB or both PATH_BACKUP)
|
2014-02-04 03:03:17 +03:00
|
|
|
if (path_type_get($strSourcePathType) ne path_type_get($strDestinationPathType))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "path types must be equal in link create");
|
|
|
|
}
|
|
|
|
|
2014-02-05 18:40:49 +03:00
|
|
|
# Generate source and destination files
|
2014-02-03 03:03:05 +03:00
|
|
|
my $strSource = path_get($strSourcePathType, $strSourceFile);
|
|
|
|
my $strDestination = path_get($strDestinationPathType, $strDestinationFile);
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# If the destination path is backup and does not exist, create it
|
|
|
|
if (path_type_get($strDestinationPathType) eq PATH_BACKUP)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
path_create(PATH_BACKUP_ABSOLUTE, dirname($strDestination));
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
2014-02-05 18:40:49 +03:00
|
|
|
|
2014-02-03 03:03:05 +03:00
|
|
|
unless (-e $strSource)
|
|
|
|
{
|
|
|
|
if (-e $strSource . ".${strCompressExtension}")
|
|
|
|
{
|
|
|
|
$strSource .= ".${strCompressExtension}";
|
|
|
|
$strDestination .= ".${strCompressExtension}";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-02-05 18:40:49 +03:00
|
|
|
# Error when a hardlink will be created on a missing file
|
|
|
|
if ($bHard)
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "unable to find ${strSource}(.${strCompressExtension}) for link");
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-05 18:40:49 +03:00
|
|
|
# Generate relative path if requested
|
|
|
|
if ($bRelative)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-05 18:40:49 +03:00
|
|
|
my $iCommonLen = common_prefix($strSource, $strDestination);
|
|
|
|
|
|
|
|
if ($iCommonLen != 0)
|
|
|
|
{
|
|
|
|
$strSource = ("../" x substr($strDestination, $iCommonLen) =~ tr/\///) . substr($strSource, $iCommonLen);
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
2014-02-05 18:40:49 +03:00
|
|
|
|
|
|
|
# Create the command
|
|
|
|
my $strCommand = "ln" . (!$bHard ? " -s" : "") . " ${strSource} ${strDestination}";
|
2014-02-04 03:03:17 +03:00
|
|
|
|
|
|
|
# Run remotely
|
|
|
|
if (is_remote($strSourcePathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " link_create: remote ${strSourcePathType} '${strCommand}'");
|
|
|
|
|
|
|
|
my $oSSH = remote_get($strSourcePathType);
|
|
|
|
$oSSH->system($strCommand) or confess &log("unable to create link from ${strSource} to ${strDestination}");
|
|
|
|
}
|
|
|
|
# Run locally
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " link_create: local '${strCommand}'");
|
|
|
|
system($strCommand) == 0 or confess &log("unable to create link from ${strSource} to ${strDestination}");
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# PATH_CREATE
|
2014-02-04 03:03:17 +03:00
|
|
|
#
|
|
|
|
# Creates a path locally or remotely. Currently does not error if the path already exists. Also does not set permissions if the
|
|
|
|
# path aleady exists.
|
2014-02-03 03:03:05 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
sub path_create
|
|
|
|
{
|
|
|
|
my $strPathType = shift;
|
|
|
|
my $strPath = shift;
|
|
|
|
my $strPermission = shift;
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# If no permissions are given then use the default
|
2014-02-03 03:03:05 +03:00
|
|
|
if (!defined($strPermission))
|
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
$strPermission = $strDefaultPathPermission;
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
# Get the path to create
|
|
|
|
my $strPathCreate = $strPath;
|
|
|
|
|
|
|
|
if (defined($strPathType))
|
|
|
|
{
|
|
|
|
$strPathCreate = path_get($strPathType, $strPath);
|
|
|
|
}
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
my $strCommand = "mkdir -p -m ${strPermission} ${strPathCreate}";
|
|
|
|
|
|
|
|
# Run remotely
|
|
|
|
if (is_remote($strPathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " path_create: remote ${strPathType} '${strCommand}'");
|
|
|
|
|
|
|
|
my $oSSH = remote_get($strPathType);
|
|
|
|
$oSSH->system($strCommand) or confess &log("unable to create remote path ${strPathType}:${strPath}");
|
|
|
|
}
|
|
|
|
# Run locally
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " path_create: local '${strCommand}'");
|
|
|
|
system($strCommand) == 0 or confess &log(ERROR, "unable to create path ${strPath}");
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# IS_REMOTE
|
|
|
|
#
|
|
|
|
# Determine whether any operations are being performed remotely. If $strPathType is defined, the function will return true if that
|
|
|
|
# path is remote. If $strPathType is not defined, then function will return true if any path is remote.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub is_remote
|
|
|
|
{
|
|
|
|
my $strPathType = shift;
|
|
|
|
|
|
|
|
# If the SSH object is defined then some paths are remote
|
|
|
|
if (defined($oDbSSH) || defined($oBackupSSH))
|
|
|
|
{
|
|
|
|
# If path type is not defined but the SSH object is, then some paths are remote
|
|
|
|
if (!defined($strPathType))
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
# If a host is defined for the path then it is remote
|
|
|
|
if (defined($strBackupHost) && path_type_get($strPathType) eq PATH_BACKUP ||
|
|
|
|
defined($strDbHost) && path_type_get($strPathType) eq PATH_DB)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# REMOTE_GET
|
|
|
|
#
|
|
|
|
# Get remote SSH object depending on the path type.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub remote_get
|
|
|
|
{
|
|
|
|
my $strPathType = shift;
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# Get the db SSH object
|
2014-02-03 03:03:05 +03:00
|
|
|
if (path_type_get($strPathType) eq PATH_DB && defined($oDbSSH))
|
|
|
|
{
|
|
|
|
return $oDbSSH;
|
|
|
|
}
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# Get the backup SSH object
|
2014-02-03 03:03:05 +03:00
|
|
|
if (path_type_get($strPathType) eq PATH_BACKUP && defined($oBackupSSH))
|
|
|
|
{
|
|
|
|
return $oBackupSSH
|
|
|
|
}
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# Error when no ssh object is found
|
2014-02-03 03:03:05 +03:00
|
|
|
confess &log(ASSERT, "path type ${strPathType} does not have a defined ssh object");
|
|
|
|
}
|
|
|
|
|
2014-02-05 02:48:39 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# FILE_MOVE
|
|
|
|
#
|
|
|
|
# Moves a file locally or remotely.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub file_move
|
|
|
|
{
|
2014-02-05 21:10:36 +03:00
|
|
|
my $strSourcePathType = shift;
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strSourceFile = shift;
|
2014-02-05 21:10:36 +03:00
|
|
|
my $strDestinationPathType = shift;
|
2014-02-05 02:48:39 +03:00
|
|
|
my $strDestinationFile = shift;
|
2014-02-05 21:10:36 +03:00
|
|
|
|
|
|
|
if (path_type_get($strSourcePathType) ne path_type_get($strSourcePathType))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "source and destination path types must be equal");
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strSource = path_get($strSourcePathType, $strSourceFile);
|
|
|
|
my $strDestination = path_get($strDestinationPathType, $strDestinationFile);
|
2014-02-05 02:48:39 +03:00
|
|
|
|
|
|
|
# If the destination path is backup and does not exist, create it
|
2014-02-05 21:10:36 +03:00
|
|
|
if (path_type_get($strDestinationPathType) eq PATH_BACKUP)
|
2014-02-05 02:48:39 +03:00
|
|
|
{
|
|
|
|
path_create(PATH_BACKUP_ABSOLUTE, dirname($strDestination));
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strCommand = "mv ${strSource} ${strDestination}";
|
|
|
|
|
|
|
|
# Run remotely
|
2014-02-05 21:10:36 +03:00
|
|
|
if (is_remote($strDestinationPathType))
|
2014-02-05 02:48:39 +03:00
|
|
|
{
|
2014-02-05 21:10:36 +03:00
|
|
|
&log(DEBUG, " file_move: remote ${strDestinationPathType} '${strCommand}'");
|
2014-02-05 02:48:39 +03:00
|
|
|
|
2014-02-05 21:10:36 +03:00
|
|
|
my $oSSH = remote_get($strDestinationPathType);
|
|
|
|
$oSSH->system($strCommand)
|
|
|
|
or confess &log("unable to move remote ${strDestinationPathType}:${strSourceFile} to ${strDestinationFile}");
|
2014-02-05 02:48:39 +03:00
|
|
|
}
|
|
|
|
# Run locally
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_move: '${strCommand}'");
|
|
|
|
|
|
|
|
system($strCommand) == 0 or confess &log("unable to move local ${strSourceFile} to ${strDestinationFile}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-03 03:03:05 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# FILE_COPY
|
|
|
|
####################################################################################################################################
|
|
|
|
sub file_copy
|
|
|
|
{
|
|
|
|
my $strSourcePathType = shift;
|
|
|
|
my $strSourceFile = shift;
|
|
|
|
my $strDestinationPathType = shift;
|
|
|
|
my $strDestinationFile = shift;
|
|
|
|
my $bNoCompressionOverride = shift;
|
2014-02-05 19:35:09 +03:00
|
|
|
my $lModificationTime = shift;
|
|
|
|
my $strPermission = shift;
|
|
|
|
|
|
|
|
# Modification time and permissions cannot be set remotely
|
|
|
|
if ((defined($lModificationTime) || defined($strPermission)) && is_remote($strDestinationPathType))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "modification time and permissions cannot be set on remote destination file");
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
# Generate source, destination and tmp filenames
|
|
|
|
my $strSource = path_get($strSourcePathType, $strSourceFile);
|
|
|
|
my $strDestination = path_get($strDestinationPathType, $strDestinationFile);
|
|
|
|
my $strDestinationTmp = path_get($strDestinationPathType, $strDestinationFile, true);
|
|
|
|
|
|
|
|
# Is this already a compressed file?
|
|
|
|
my $bAlreadyCompressed = $strSource =~ "^.*\.${strCompressExtension}\$";
|
|
|
|
|
|
|
|
if ($bAlreadyCompressed && $strDestination !~ "^.*\.${strCompressExtension}\$")
|
|
|
|
{
|
|
|
|
$strDestination .= ".${strCompressExtension}";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Does the file need compression?
|
2014-02-05 16:21:27 +03:00
|
|
|
my $bCompress = !((defined($bNoCompressionOverride) && $bNoCompressionOverride) ||
|
2014-02-03 03:03:05 +03:00
|
|
|
(!defined($bNoCompressionOverride) && $bNoCompression));
|
|
|
|
|
|
|
|
# If the destination path is backup and does not exist, create it
|
2014-02-04 03:03:17 +03:00
|
|
|
if (path_type_get($strDestinationPathType) eq PATH_BACKUP)
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
path_create(PATH_BACKUP_ABSOLUTE, dirname($strDestination));
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
2014-02-05 16:21:27 +03:00
|
|
|
# Generate the command string depending on compression/decompression/cat
|
|
|
|
my $strCommand = $strCommandCat;
|
|
|
|
|
|
|
|
if ($bAlreadyCompressed && $bCompress)
|
|
|
|
{
|
|
|
|
$strDestination .= $strDestination =~ "^.*\.${strCompressExtension}\$" ? ".gz" : "";
|
|
|
|
}
|
|
|
|
elsif (!$bAlreadyCompressed && $bCompress)
|
|
|
|
{
|
|
|
|
$strCommand = $strCommandCompress;
|
|
|
|
$strDestination .= ".gz";
|
|
|
|
}
|
|
|
|
elsif ($bAlreadyCompressed && !$bCompress)
|
|
|
|
{
|
|
|
|
$strCommand = $strCommandDecompress;
|
|
|
|
$strDestination = substr($strDestination, 0, length($strDestination) - length($strCompressExtension) - 1);
|
|
|
|
}
|
|
|
|
|
2014-02-03 03:03:05 +03:00
|
|
|
$strCommand =~ s/\%file\%/${strSource}/g;
|
|
|
|
|
|
|
|
# If this command is remote on only one side
|
|
|
|
if (is_remote($strSourcePathType) && !is_remote($strDestinationPathType) ||
|
|
|
|
!is_remote($strSourcePathType) && is_remote($strDestinationPathType))
|
|
|
|
{
|
|
|
|
# Else if the source is remote
|
|
|
|
if (is_remote($strSourcePathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: remote ${strSource} to local ${strDestination}");
|
|
|
|
|
|
|
|
# Open the destination file for writing (will be streamed from the ssh session)
|
|
|
|
my $hFile;
|
|
|
|
open($hFile, ">", $strDestinationTmp) or confess &log(ERROR, "cannot open ${strDestination}");
|
|
|
|
|
|
|
|
# Execute the command through ssh
|
|
|
|
my $oSSH = remote_get($strSourcePathType);
|
|
|
|
$oSSH->system({stdout_fh => $hFile}, $strCommand) or confess &log(ERROR, "unable to execute ssh '$strCommand'");
|
|
|
|
|
|
|
|
# Close the destination file handle
|
|
|
|
close($hFile) or confess &log(ERROR, "cannot close file");
|
|
|
|
}
|
|
|
|
# Else if the destination is remote
|
|
|
|
elsif (is_remote($strDestinationPathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: local ${strSource} ($strCommand) to remote ${strDestination}");
|
|
|
|
|
|
|
|
# Open the input command as a stream
|
|
|
|
my $hOut;
|
|
|
|
my $pId = open3(undef, $hOut, undef, $strCommand) or confess(ERROR, "unable to execute '${strCommand}'");
|
|
|
|
|
|
|
|
# Execute the command though ssh
|
|
|
|
my $oSSH = remote_get($strDestinationPathType);
|
|
|
|
$oSSH->system({stdin_fh => $hOut}, "cat > ${strDestinationTmp}") or confess &log(ERROR, "unable to execute ssh 'cat'");
|
|
|
|
|
|
|
|
# Wait for the stream process to finish
|
|
|
|
waitpid($pId, 0);
|
|
|
|
my $iExitStatus = ${^CHILD_ERROR_NATIVE} >> 8;
|
|
|
|
|
|
|
|
if ($iExitStatus != 0)
|
|
|
|
{
|
|
|
|
confess &log(ERROR, "command '${strCommand}' returned", $iExitStatus);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
# If the source and destination are both remote but not the same remote
|
|
|
|
elsif (is_remote($strSourcePathType) && is_remote($strDestinationPathType) &&
|
|
|
|
path_type_get($strSourcePathType) ne path_type_get($strDestinationPathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: remote ${strSource} to remote ${strDestination}");
|
|
|
|
confess &log(ASSERT, "remote source and destination not supported");
|
|
|
|
}
|
|
|
|
# Else this is a local command or remote where both sides are the same remote
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Complete the command by redirecting to the destination tmp file
|
|
|
|
$strCommand .= " > ${strDestinationTmp}";
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
if (is_remote($strSourcePathType))
|
2014-02-03 03:03:05 +03:00
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: remote ${strSourcePathType} '${strCommand}'");
|
|
|
|
|
|
|
|
my $oSSH = remote_get($strSourcePathType);
|
|
|
|
$oSSH->system($strCommand) or confess &log(ERROR, "unable to execute remote command ${strCommand}:" . oSSH->error);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: local '${strCommand}'");
|
|
|
|
system($strCommand) == 0 or confess &log(ERROR, "unable to copy local $strSource to local $strDestinationTmp");
|
|
|
|
}
|
|
|
|
}
|
2014-02-05 19:35:09 +03:00
|
|
|
|
|
|
|
# Set the file permission if required (this only works locally for now)
|
|
|
|
if (defined($strPermission))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: chmod ${strPermission}");
|
|
|
|
|
|
|
|
system("chmod ${strPermission} ${strDestinationTmp}") == 0
|
|
|
|
or confess &log(ERROR, "unable to set permissions for local ${strDestinationTmp}");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Set the file modification time if required (this only works locally for now)
|
|
|
|
if (defined($lModificationTime))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " file_copy: time ${lModificationTime}");
|
|
|
|
|
|
|
|
utime($lModificationTime, $lModificationTime, $strDestinationTmp)
|
|
|
|
or confess &log(ERROR, "unable to set time for local ${strDestinationTmp}");
|
|
|
|
}
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
# Move the file from tmp to final destination
|
2014-02-05 21:10:36 +03:00
|
|
|
file_move(path_type_get($strSourcePathType) . ":absolute", $strDestinationTmp,
|
|
|
|
path_type_get($strDestinationPathType) . ":absolute", $strDestination);
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# FILE_HASH_GET
|
|
|
|
####################################################################################################################################
|
|
|
|
sub file_hash_get
|
|
|
|
{
|
|
|
|
my $strPathType = shift;
|
|
|
|
my $strFile = shift;
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
# For now this operation is not supported remotely. Not currently needed.
|
|
|
|
if (is_remote($strPathType))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "remote operation not supported");
|
|
|
|
}
|
|
|
|
|
2014-02-03 03:03:05 +03:00
|
|
|
if (!defined($strCommandChecksum))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "\$strCommandChecksum not defined");
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strPath = path_get($strPathType, $strFile);
|
|
|
|
my $strCommand;
|
|
|
|
|
|
|
|
if (-e $strPath)
|
|
|
|
{
|
|
|
|
$strCommand = $strCommandChecksum;
|
2014-02-05 16:21:27 +03:00
|
|
|
$strCommand =~ s/\%file\%/${strPath}/g;
|
2014-02-03 03:03:05 +03:00
|
|
|
}
|
|
|
|
elsif (-e $strPath . ".${strCompressExtension}")
|
|
|
|
{
|
|
|
|
$strCommand = $strCommandDecompress;
|
|
|
|
$strCommand =~ s/\%file\%/${strPath}/g;
|
|
|
|
$strCommand .= " | " . $strCommandChecksum;
|
|
|
|
$strCommand =~ s/\%file\%//g;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "unable to find $strPath(.${strCompressExtension}) for checksum");
|
|
|
|
}
|
|
|
|
|
|
|
|
return trim(capture($strCommand)) or confess &log(ERROR, "unable to checksum ${strPath}");
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# FILE_LIST_GET
|
|
|
|
####################################################################################################################################
|
|
|
|
sub file_list_get
|
|
|
|
{
|
2014-02-04 03:03:17 +03:00
|
|
|
my $strPathType = shift;
|
2014-02-03 03:03:05 +03:00
|
|
|
my $strPath = shift;
|
|
|
|
my $strExpression = shift;
|
|
|
|
my $strSortOrder = shift;
|
2014-02-04 03:03:17 +03:00
|
|
|
|
|
|
|
# For now this operation is not supported remotely. Not currently needed.
|
|
|
|
if (is_remote($strPathType))
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "remote operation not supported");
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strPathList = path_get($strPathType, $strPath);
|
2014-02-03 03:03:05 +03:00
|
|
|
|
|
|
|
my $hDir;
|
|
|
|
|
2014-02-04 03:03:17 +03:00
|
|
|
opendir $hDir, $strPathList or confess &log(ERROR, "unable to open path ${strPathList}");
|
|
|
|
my @stryFileAll = readdir $hDir or confess &log(ERROR, "unable to get files for path ${strPathList}, expression ${strExpression}");
|
2014-02-03 03:03:05 +03:00
|
|
|
close $hDir;
|
|
|
|
|
|
|
|
my @stryFile;
|
|
|
|
|
|
|
|
if (@stryFileAll)
|
|
|
|
{
|
|
|
|
@stryFile = grep(/$strExpression/i, @stryFileAll)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (@stryFile)
|
|
|
|
{
|
|
|
|
if (defined($strSortOrder) && $strSortOrder eq "reverse")
|
|
|
|
{
|
|
|
|
return sort {$b cmp $a} @stryFile;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return sort @stryFile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return @stryFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# MANIFEST_GET
|
|
|
|
#
|
|
|
|
# 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_get
|
|
|
|
{
|
|
|
|
my $strPathType = shift;
|
|
|
|
my $strPath = shift;
|
|
|
|
|
|
|
|
&log(DEBUG, "manifest: " . $strCommandManifest);
|
|
|
|
|
|
|
|
# Get the root path for the manifest
|
|
|
|
my $strPathManifest = path_get($strPathType, $strPath);
|
|
|
|
|
|
|
|
# Builds the manifest command
|
|
|
|
my $strCommand = $strCommandManifest;
|
|
|
|
$strCommand =~ s/\%path\%/${strPathManifest}/g;
|
|
|
|
|
|
|
|
# Run the manifest command
|
|
|
|
my $strManifest;
|
|
|
|
|
|
|
|
# Run remotely
|
|
|
|
if (is_remote($strPathType))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " manifest_get: remote ${strPathType}:${strPathManifest}");
|
|
|
|
|
|
|
|
my $oSSH = remote_get($strPathType);
|
|
|
|
$strManifest = $oSSH->capture($strCommand) or confess &log(ERROR, "unable to execute remote command '${strCommand}'");
|
|
|
|
}
|
|
|
|
# Run locally
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " manifest_get: local ${strPathType}:${strPathManifest}");
|
|
|
|
$strManifest = capture($strCommand) or confess &log(ERROR, "unable to execute local command '${strCommand}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load the manifest into a hash
|
|
|
|
return data_hash_build("name\ttype\tuser\tgroup\tpermission\tmodification_time\tinode\tsize\tlink_destination\n" .
|
|
|
|
$strManifest, "\t", ".");
|
|
|
|
}
|
|
|
|
|
2014-02-05 02:48:39 +03:00
|
|
|
####################################################################################################################################
|
|
|
|
# PSQL_EXECUTE
|
|
|
|
####################################################################################################################################
|
|
|
|
sub psql_execute
|
|
|
|
{
|
|
|
|
my $strScript = shift; # psql script to execute
|
|
|
|
|
|
|
|
# Get the user-defined command for psql
|
|
|
|
my $strCommand = $strCommandPsql . " -c \"${strScript}\" postgres";
|
|
|
|
my $strResult;
|
|
|
|
|
|
|
|
# Run remotely
|
|
|
|
if (is_remote(PATH_DB))
|
|
|
|
{
|
|
|
|
&log(DEBUG, " psql execute: remote ${strScript}");
|
|
|
|
|
|
|
|
my $oSSH = remote_get(PATH_DB);
|
|
|
|
$strResult = $oSSH->capture($strCommand) or confess &log(ERROR, "unable to execute remote psql command '${strCommand}'");
|
|
|
|
}
|
|
|
|
# Run locally
|
|
|
|
else
|
|
|
|
{
|
|
|
|
&log(DEBUG, " psql execute: ${strScript}");
|
|
|
|
$strResult = capture($strCommand) or confess &log(ERROR, "unable to execute local psql command '${strCommand}'");
|
|
|
|
}
|
|
|
|
|
|
|
|
return $strResult;
|
|
|
|
}
|
|
|
|
|
2014-02-03 03:03:05 +03:00
|
|
|
1;
|