1
0
mirror of https://github.com/pgbackrest/pgbackrest.git synced 2025-01-18 04:58:51 +02:00
pgbackrest/test/lib/BackRestTest/FileTest.pm

1286 lines
48 KiB
Perl
Raw Normal View History

#!/usr/bin/perl
####################################################################################################################################
# FileTest.pl - Unit Tests for BackRest::File
####################################################################################################################################
package BackRestTest::FileTest;
####################################################################################################################################
# Perl includes
####################################################################################################################################
use strict;
use warnings FATAL => qw(all);
use Carp qw(confess);
2014-06-07 20:15:55 +03:00
use File::Basename;
use Cwd 'abs_path';
use File::stat;
use Fcntl ':mode';
2014-06-07 20:15:55 +03:00
use Scalar::Util 'blessed';
use Time::HiRes qw(gettimeofday usleep);
use POSIX qw(ceil);
use lib dirname($0) . '/../lib';
2014-06-07 23:13:41 +03:00
use BackRest::Utility;
use BackRest::Config;
use BackRest::File;
use BackRest::Remote;
2014-06-22 17:54:31 +03:00
use BackRestTest::CommonTest;
use Exporter qw(import);
2014-06-22 17:54:31 +03:00
our @EXPORT = qw(BackRestTestFile_Test);
2014-06-22 17:54:31 +03:00
my $strTestPath;
my $strHost;
my $strUserBackRest;
2014-06-21 21:19:03 +03:00
####################################################################################################################################
2014-06-22 17:54:31 +03:00
# BackRestTestFile_Setup
2014-06-21 21:19:03 +03:00
####################################################################################################################################
2014-06-22 17:54:31 +03:00
sub BackRestTestFile_Setup
2014-06-21 21:19:03 +03:00
{
2014-06-22 01:19:37 +03:00
my $bPrivate = shift;
2014-06-21 21:19:03 +03:00
my $bDropOnly = shift;
2014-06-22 03:08:49 +03:00
2014-06-21 21:19:03 +03:00
# Remove the backrest private directory
2014-06-22 01:19:37 +03:00
if (-e "${strTestPath}/private")
{
system("ssh ${strUserBackRest}\@${strHost} 'rm -rf ${strTestPath}/private'");
}
2014-06-22 03:08:49 +03:00
2014-06-21 21:19:03 +03:00
# Remove the test directory
system("rm -rf ${strTestPath}") == 0 or die 'unable to drop test path';
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
if (!defined($bDropOnly) || !$bDropOnly)
2014-06-21 21:19:03 +03:00
{
# Create the test directory
mkdir($strTestPath, oct('0770')) or confess 'Unable to create test directory';
2014-06-22 03:08:49 +03:00
2014-06-21 21:19:03 +03:00
# Create the backrest private directory
2014-06-22 01:19:37 +03:00
if (defined($bPrivate) && $bPrivate)
{
system("ssh backrest\@${strHost} 'mkdir -m 700 ${strTestPath}/private'") == 0 or die 'unable to create test/private path';
}
2014-06-21 21:19:03 +03:00
}
}
####################################################################################################################################
2014-06-22 17:54:31 +03:00
# BackRestTestFile_Test
2014-06-21 21:19:03 +03:00
####################################################################################################################################
2014-06-22 17:54:31 +03:00
sub BackRestTestFile_Test
{
my $strTest = shift;
2014-06-06 05:42:47 +03:00
# If no test was specified, then run them all
if (!defined($strTest))
{
$strTest = 'all';
}
# Setup test variables
2014-06-02 00:23:33 +03:00
my $iRun;
2014-06-22 17:54:31 +03:00
$strTestPath = BackRestTestCommon_TestPathGet();
my $strStanza = BackRestTestCommon_StanzaGet();
my $strUser = BackRestTestCommon_UserGet();
my $strGroup = BackRestTestCommon_GroupGet();
$strHost = BackRestTestCommon_HostGet();
$strUserBackRest = BackRestTestCommon_UserBackRestGet();
2014-06-02 00:23:33 +03:00
# Print test banner
&log(INFO, 'FILE MODULE ********************************************************************');
2014-06-07 04:16:24 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
# Create remotes
2014-06-07 04:16:24 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
my $oRemote = BackRest::Remote->new
2014-06-07 04:16:24 +03:00
(
$strHost, # Host
$strUser, # User
BackRestTestCommon_CommandRemoteGet(), # Command
$strStanza, # Stanza
'', # Repo Path
OPTION_DEFAULT_BUFFER_SIZE, # Buffer size
OPTION_DEFAULT_COMPRESS_LEVEL, # Compress level
OPTION_DEFAULT_COMPRESS_LEVEL_NETWORK, # Compress network level
);
my $oLocal = new BackRest::Remote
(
undef, # Host
undef, # User
undef, # Command
undef, # Stanza
undef, # Repo Path
OPTION_DEFAULT_BUFFER_SIZE, # Buffer size
OPTION_DEFAULT_COMPRESS_LEVEL, # Compress level
OPTION_DEFAULT_COMPRESS_LEVEL_NETWORK, # Compress network level
2014-06-07 04:16:24 +03:00
);
#-------------------------------------------------------------------------------------------------------------------------------
# Test path_create()
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'path_create')
{
$iRun = 0;
&log(INFO, "Test File->path_create()\n");
# Loop through local/remote
for (my $bRemote = 0; $bRemote <= 1; $bRemote++)
{
2014-06-21 22:46:26 +03:00
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-10-10 22:13:28 +03:00
);
2014-06-24 01:54:00 +03:00
# Loop through error
2014-06-21 22:17:09 +03:00
for (my $bError = 0; $bError <= 1; $bError++)
{
# Loop through mode (mode will be set on true)
for (my $bMode = 0; $bMode <= 1; $bMode++)
2014-06-21 22:17:09 +03:00
{
2014-06-22 03:08:49 +03:00
my $strPathType = PATH_BACKUP_CLUSTER;
# Increment the run, log, and decide whether this unit test should be run
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, err ${bError}, mode ${bMode}")) {next}
2014-06-21 22:17:09 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
mkdir("${strTestPath}/backup") or confess 'Unable to create test/backup directory';
mkdir("${strTestPath}/backup/db") or confess 'Unable to create test/backup/db directory';
2014-06-22 03:08:49 +03:00
my $strPath = 'path';
my $strMode;
# If mode then set one (other than the default)
if ($bMode)
2014-06-21 22:17:09 +03:00
{
$strMode = '0700';
2014-06-21 22:17:09 +03:00
}
# If not exists then set the path to something bogus
if ($bError)
{
$strPath = "${strTestPath}/private/path";
2014-06-22 03:08:49 +03:00
$strPathType = PATH_BACKUP_ABSOLUTE;
2014-06-21 22:17:09 +03:00
}
# Execute in eval to catch errors
2014-06-24 01:54:00 +03:00
my $bErrorExpected = $bError;
2014-06-21 22:17:09 +03:00
eval
{
$oFile->path_create($strPathType, $strPath, $strMode);
2014-06-21 22:17:09 +03:00
};
2014-06-21 22:17:09 +03:00
# Check for errors
if ($@)
{
# Ignore errors if the path did not exist
if ($bErrorExpected)
{
2014-06-21 22:17:09 +03:00
next;
}
2014-06-04 18:58:30 +03:00
2014-06-21 22:17:09 +03:00
confess "error raised: " . $@ . "\n";
}
2014-06-04 18:58:30 +03:00
2014-06-21 22:17:09 +03:00
if ($bErrorExpected)
{
confess 'error was expected';
}
2014-06-21 22:17:09 +03:00
# Make sure the path was actually created
2014-06-22 03:08:49 +03:00
my $strPathCheck = $oFile->path_get($strPathType, $strPath);
unless (-e $strPathCheck)
2014-06-21 22:17:09 +03:00
{
confess 'path was not created';
2014-06-21 22:17:09 +03:00
}
2014-06-04 18:58:30 +03:00
# Check that the mode was set correctly
2014-06-22 03:08:49 +03:00
my $oStat = lstat($strPathCheck);
2014-06-04 18:58:30 +03:00
2014-06-21 22:17:09 +03:00
if (!defined($oStat))
{
2014-06-22 03:08:49 +03:00
confess "unable to stat ${strPathCheck}";
2014-06-21 22:17:09 +03:00
}
if ($bMode)
2014-06-21 22:17:09 +03:00
{
if ($strMode ne sprintf('%04o', S_IMODE($oStat->mode)))
2014-06-21 22:46:26 +03:00
{
confess "mode were not set to {$strMode}";
2014-06-21 22:46:26 +03:00
}
}
}
2014-06-21 22:17:09 +03:00
}
}
}
2014-06-02 00:23:33 +03:00
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-04 02:03:03 +03:00
# Test move()
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'move')
{
$iRun = 0;
2014-06-04 02:03:03 +03:00
&log(INFO, '--------------------------------------------------------------------------------');
&log(INFO, "Test File->move()\n");
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
for (my $bRemote = 0; $bRemote <= 0; $bRemote++)
2014-06-04 02:03:03 +03:00
{
2014-06-21 21:19:03 +03:00
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = (new BackRest::File
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-10-10 22:13:28 +03:00
))->clone(1);
# Loop through source exists
for (my $bSourceExists = 0; $bSourceExists <= 1; $bSourceExists++)
2014-06-04 02:03:03 +03:00
{
2014-06-21 21:19:03 +03:00
# Loop through source errors
for (my $bSourceError = 0; $bSourceError <= 1; $bSourceError++)
{
# Loop through destination exists
for (my $bDestinationExists = 0; $bDestinationExists <= 1; $bDestinationExists++)
{
# Loop through source errors
for (my $bDestinationError = 0; $bDestinationError <= 1; $bDestinationError++)
{
# Loop through create
for (my $bCreate = 0; $bCreate <= $bDestinationExists; $bCreate++)
{
# Increment the run, log, and decide whether this unit test should be run
if (!BackRestTestCommon_Run(++$iRun,
"src_exists ${bSourceExists}, src_error ${bSourceError}, " .
", dst_exists ${bDestinationExists}, dst_error ${bDestinationError}, " .
"dst_create ${bCreate}")) {next}
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bSourceError || $bDestinationError);
2014-06-04 18:58:30 +03:00
2014-06-21 21:19:03 +03:00
my $strSourceFile = "${strTestPath}/test.txt";
my $strDestinationFile = "${strTestPath}/test-dest.txt";
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
if ($bSourceError)
{
$strSourceFile = "${strTestPath}/private/test.txt";
}
elsif ($bSourceExists)
{
system("echo 'TESTDATA' > ${strSourceFile}");
}
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
if ($bDestinationError)
{
$strSourceFile = "${strTestPath}/private/test.txt";
}
elsif (!$bDestinationExists)
{
$strDestinationFile = "${strTestPath}/sub/test-dest.txt";
}
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
# Execute in eval in case of error
eval
{
$oFile->move(PATH_BACKUP_ABSOLUTE, $strSourceFile, PATH_BACKUP_ABSOLUTE, $strDestinationFile, $bCreate);
};
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
if ($@)
{
if (!$bSourceExists || (!$bDestinationExists && !$bCreate) || $bSourceError || $bDestinationError)
{
next;
}
2014-06-04 02:03:03 +03:00
confess 'error raised: ' . $@ . "\n";
2014-06-21 21:19:03 +03:00
}
2014-06-21 21:19:03 +03:00
if (!$bSourceExists || (!$bDestinationExists && !$bCreate) || $bSourceError || $bDestinationError)
{
confess 'error should have been raised';
2014-06-21 21:19:03 +03:00
}
2014-06-04 02:03:03 +03:00
2014-06-21 21:19:03 +03:00
unless (-e $strDestinationFile)
{
confess 'file was not moved';
2014-06-04 02:03:03 +03:00
}
}
2014-06-21 21:19:03 +03:00
}
}
}
}
2014-06-04 02:03:03 +03:00
}
}
#-------------------------------------------------------------------------------------------------------------------------------
# Test compress()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'compress')
{
$iRun = 0;
2014-06-03 00:48:07 +03:00
&log(INFO, '--------------------------------------------------------------------------------');
&log(INFO, "Test File->compress()\n");
2014-06-03 00:48:07 +03:00
2014-06-21 22:17:09 +03:00
for (my $bRemote = 0; $bRemote <= 0; $bRemote++)
2014-06-03 00:48:07 +03:00
{
2014-06-21 22:17:09 +03:00
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
);
# Loop through exists
for (my $bExists = 0; $bExists <= 1; $bExists++)
2014-06-21 22:17:09 +03:00
{
for (my $bError = 0; $bError <= 1; $bError++)
{
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, exists ${bExists}, err ${bError}")) {next}
2014-06-03 00:48:07 +03:00
2014-06-21 22:17:09 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-03 00:48:07 +03:00
my $strFile = "${strTestPath}/test.txt";
2014-06-21 22:17:09 +03:00
my $strSourceHash;
my $iSourceSize;
2014-06-03 00:48:07 +03:00
2014-06-21 22:17:09 +03:00
if ($bError)
{
$strFile = "${strTestPath}/private/test.txt";
}
elsif ($bExists)
{
system("echo 'TESTDATA' > ${strFile}");
($strSourceHash, $iSourceSize) = $oFile->hash_size(PATH_BACKUP_ABSOLUTE, $strFile);
}
2014-06-03 00:48:07 +03:00
# Execute in eval in case of error
eval
{
$oFile->compress(PATH_BACKUP_ABSOLUTE, $strFile);
};
2014-06-03 00:48:07 +03:00
2014-06-21 22:17:09 +03:00
if ($@)
{
2014-06-21 22:17:09 +03:00
if (!$bExists || $bError)
{
next;
}
2014-06-22 03:08:49 +03:00
confess 'error raised: ' . $@ . "\n";
}
2014-06-22 03:08:49 +03:00
2014-06-21 22:17:09 +03:00
if (!$bExists || $bError)
{
confess 'expected error';
2014-06-21 22:17:09 +03:00
}
2014-06-04 18:58:30 +03:00
my $strDestinationFile = $strFile . '.gz';
2014-06-21 22:17:09 +03:00
if (-e $strFile)
{
confess 'source file still exists';
}
2014-06-21 22:17:09 +03:00
unless (-e $strDestinationFile)
{
confess 'file was not compressed';
2014-06-21 22:17:09 +03:00
}
2014-06-22 03:08:49 +03:00
2014-06-21 22:17:09 +03:00
system("gzip -d ${strDestinationFile}") == 0 or die "could not decompress ${strDestinationFile}";
2014-06-22 03:08:49 +03:00
my ($strDestinationHash, $iDestinationSize) = $oFile->hash_size(PATH_BACKUP_ABSOLUTE, $strFile);
2014-06-22 03:08:49 +03:00
2014-06-21 22:17:09 +03:00
if ($strSourceHash ne $strDestinationHash)
{
2014-06-21 22:17:09 +03:00
confess "source ${strSourceHash} and destination ${strDestinationHash} file hashes do not match";
}
}
2014-06-21 22:17:09 +03:00
}
}
}
#-------------------------------------------------------------------------------------------------------------------------------
# Test wait()
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'wait')
{
$iRun = 0;
&log(INFO, '--------------------------------------------------------------------------------');
&log(INFO, "Test File->wait()\n");
for (my $bRemote = 0; $bRemote <= 1; $bRemote++)
{
# Create the file object
my $oFile = new BackRest::File
(
$strStanza,
$strTestPath,
$bRemote ? 'db' : undef,
$bRemote ? $oRemote : $oLocal
);
my $lTimeBegin = gettimeofday();
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, begin ${lTimeBegin}")) {next}
# If there is not enough time to complete the test then sleep
if (ceil($lTimeBegin) - $lTimeBegin < .250)
{
my $lSleepMs = ceil(((int($lTimeBegin) + 1) - $lTimeBegin) * 1000);
usleep($lSleepMs * 1000);
&log(DEBUG, "slept ${lSleepMs}ms: begin ${lTimeBegin}, end " . gettimeofday());
$lTimeBegin = gettimeofday();
}
# Run the test
my $lTimeBeginCheck = $oFile->wait(PATH_DB_ABSOLUTE);
&log(DEBUG, "begin ${lTimeBegin}, check ${lTimeBeginCheck}, end " . time());
# Current time should have advanced by 1 second
if (time() == int($lTimeBegin))
{
confess "time was not advanced by 1 second";
}
# lTimeBegin and lTimeBeginCheck should be equal
if (int($lTimeBegin) != $lTimeBeginCheck)
{
confess 'time begin ' || int($lTimeBegin) || "and check ${lTimeBeginCheck} should be equal";
}
}
}
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-21 22:17:09 +03:00
# Test manifest()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-22 01:19:37 +03:00
if ($strTest eq 'all' || $strTest eq 'manifest')
{
$iRun = 0;
&log(INFO, '--------------------------------------------------------------------------------');
2014-06-22 01:19:37 +03:00
&log(INFO, "Test File->manifest()\n");
my $strManifestCompare =
".,d,${strUser},${strGroup},0770,,,,\n" .
"sub1,d,${strUser},${strGroup},0750,,,,\n" .
"sub1/sub2,d,${strUser},${strGroup},0750,,,,\n" .
"sub1/sub2/test,l,${strUser},${strGroup},,,,,../..\n" .
"sub1/sub2/test-hardlink.txt,f,${strUser},${strGroup},1640,1111111111,0,9,\n" .
"sub1/sub2/test-sub2.txt,f,${strUser},${strGroup},0666,1111111113,0,11,\n" .
"sub1/test,l,${strUser},${strGroup},,,,,..\n" .
"sub1/test-hardlink.txt,f,${strUser},${strGroup},1640,1111111111,0,9,\n" .
"sub1/test-sub1.txt,f,${strUser},${strGroup},0646,1111111112,0,10,\n" .
"test.txt,f,${strUser},${strGroup},1640,1111111111,0,9,";
2014-06-22 01:19:37 +03:00
for (my $bRemote = 0; $bRemote <= 1; $bRemote++)
{
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
2014-06-22 01:19:37 +03:00
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-06-22 01:19:37 +03:00
);
for (my $bError = 0; $bError <= 1; $bError++)
{
for (my $bExists = 0; $bExists <= 1; $bExists++)
{
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, exists ${bExists}, err ${bError}")) {next}
2014-06-22 01:19:37 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-22 01:19:37 +03:00
# Setup test data
system("mkdir -m 750 ${strTestPath}/sub1") == 0 or confess 'Unable to create test directory';
system("mkdir -m 750 ${strTestPath}/sub1/sub2") == 0 or confess 'Unable to create test directory';
2014-06-22 01:19:37 +03:00
system("echo 'TESTDATA' > ${strTestPath}/test.txt");
utime(1111111111, 1111111111, "${strTestPath}/test.txt");
system("chmod 1640 ${strTestPath}/test.txt");
system("echo 'TESTDATA_' > ${strTestPath}/sub1/test-sub1.txt");
utime(1111111112, 1111111112, "${strTestPath}/sub1/test-sub1.txt");
system("chmod 0640 ${strTestPath}/sub1/test-sub1.txt");
system("echo 'TESTDATA__' > ${strTestPath}/sub1/sub2/test-sub2.txt");
utime(1111111113, 1111111113, "${strTestPath}/sub1/sub2/test-sub2.txt");
2014-06-22 01:19:37 +03:00
system("chmod 0646 ${strTestPath}/sub1/test-sub1.txt");
system("ln ${strTestPath}/test.txt ${strTestPath}/sub1/test-hardlink.txt");
system("ln ${strTestPath}/test.txt ${strTestPath}/sub1/sub2/test-hardlink.txt");
system("ln -s .. ${strTestPath}/sub1/test");
system("chmod 0700 ${strTestPath}/sub1/test");
system("ln -s ../.. ${strTestPath}/sub1/sub2/test");
system("chmod 0750 ${strTestPath}/sub1/sub2/test");
system("chmod 0770 ${strTestPath}");
# Create path
my $strPath = $strTestPath;
if ($bError)
{
$strPath = $strTestPath . '/private';
2014-06-22 01:19:37 +03:00
}
elsif (!$bExists)
{
$strPath = $strTestPath . '/error';
2014-06-22 01:19:37 +03:00
}
# Execute in eval in case of error
my %oManifestHash;
my $bErrorExpected = !$bExists || $bError;
2014-06-22 01:19:37 +03:00
eval
{
$oFile->manifest(PATH_BACKUP_ABSOLUTE, $strPath, \%oManifestHash);
2014-06-22 01:19:37 +03:00
};
# Check for an error
if ($@)
{
if ($bErrorExpected)
{
next;
}
2014-06-22 03:08:49 +03:00
confess 'error raised: ' . $@ . "\n";
2014-06-22 01:19:37 +03:00
}
2014-06-22 03:08:49 +03:00
2014-06-22 01:19:37 +03:00
# Check for an expected error
if ($bErrorExpected)
{
confess 'error was expected';
}
my $strManifest;
# Validate the manifest
foreach my $strName (sort(keys $oManifestHash{name}))
{
if (!defined($strManifest))
{
$strManifest = '';
2014-06-22 01:19:37 +03:00
}
else
{
$strManifest .= "\n";
}
if (defined($oManifestHash{name}{"${strName}"}{inode}))
{
$oManifestHash{name}{"${strName}"}{inode} = 0;
}
$strManifest .=
"${strName}," .
$oManifestHash{name}{"${strName}"}{type} . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{user}) ?
$oManifestHash{name}{"${strName}"}{user} : '') . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{group}) ?
$oManifestHash{name}{"${strName}"}{group} : '') . ',' .
(defined($oManifestHash{name}{"${strName}"}{mode}) ?
$oManifestHash{name}{"${strName}"}{mode} : '') . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{modification_time}) ?
$oManifestHash{name}{"${strName}"}{modification_time} : '') . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{inode}) ?
$oManifestHash{name}{"${strName}"}{inode} : '') . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{size}) ?
$oManifestHash{name}{"${strName}"}{size} : '') . ',' .
2014-06-22 01:19:37 +03:00
(defined($oManifestHash{name}{"${strName}"}{link_destination}) ?
$oManifestHash{name}{"${strName}"}{link_destination} : '');
2014-06-22 01:19:37 +03:00
}
if ($strManifest ne $strManifestCompare)
{
confess "manifest is not equal:\n\n${strManifest}\n\ncompare:\n\n${strManifestCompare}\n\n";
}
}
}
}
}
2014-06-21 22:17:09 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
# Test list()
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-22 01:19:37 +03:00
if ($strTest eq 'all' || $strTest eq 'list')
{
$iRun = 0;
&log(INFO, '--------------------------------------------------------------------------------');
2014-06-22 01:19:37 +03:00
&log(INFO, "Test File->list()\n");
for (my $bRemote = false; $bRemote <= true; $bRemote++)
2014-06-22 01:19:37 +03:00
{
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
2014-06-22 01:19:37 +03:00
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-06-22 01:19:37 +03:00
);
for (my $bSort = false; $bSort <= true; $bSort++)
2014-06-22 01:19:37 +03:00
{
my $strSort = $bSort ? undef : 'reverse';
2014-06-22 01:19:37 +03:00
# Loop through expression
for (my $iExpression = 0; $iExpression <= 2; $iExpression++)
{
my $strExpression;
# Expression tha returns results
if ($iExpression == 1)
{
$strExpression = "^test2\\..*\$";
}
# Expression that does not return results
elsif ($iExpression == 2)
{
$strExpression = "^du\$";
}
# Loop through exists
for (my $bExists = false; $bExists <= true; $bExists++)
{
# Loop through ignore missing
for (my $bIgnoreMissing = false; $bIgnoreMissing <= $bExists; $bIgnoreMissing++)
2014-06-22 01:19:37 +03:00
{
# Loop through error
for (my $bError = false; $bError <= true; $bError++)
2014-06-22 01:19:37 +03:00
{
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, err ${bError}, exists ${bExists}, ignmis ${bIgnoreMissing}, " .
'expression ' . (defined($strExpression) ? $strExpression : '[undef]') . ', ' .
'sort ' . (defined($strSort) ? $strSort : '[undef]'))) {next}
2014-06-22 01:19:37 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-22 01:19:37 +03:00
my $strPath = $strTestPath;
2014-06-22 01:19:37 +03:00
if ($bError)
{
$strPath = "${strTestPath}/private";
}
elsif (!$bExists)
{
$strPath = "${strTestPath}/error";
}
else
{
system("echo 'TESTDATA' > ${strPath}/test.txt");
system("echo 'TESTDATA2' > ${strPath}/test2.txt");
}
my @stryFileCompare = split(/\n/, "test.txt\ntest2.txt");
# Execute in eval in case of error
my @stryFileList;
my $bErrorExpected = (!$bExists && !$bIgnoreMissing) || $bError;
2014-06-22 01:19:37 +03:00
eval
{
@stryFileList = $oFile->list(PATH_BACKUP_ABSOLUTE, $strPath, $strExpression, $strSort, $bIgnoreMissing);
2014-06-22 01:19:37 +03:00
};
if ($@)
{
if ($bErrorExpected)
{
next;
}
confess 'error raised: ' . $@ . "\n";
2014-06-22 01:19:37 +03:00
}
2014-06-22 03:08:49 +03:00
2014-06-22 01:19:37 +03:00
if ($bErrorExpected)
{
confess 'error was expected';
}
# Validate the list
if (defined($strExpression))
{
@stryFileCompare = grep(/$strExpression/i, @stryFileCompare);
}
if (defined($strSort))
{
@stryFileCompare = sort {$b cmp $a} @stryFileCompare;
}
my $strFileList = sprintf("@stryFileList");
my $strFileCompare = sprintf("@stryFileCompare");
if ($strFileList ne $strFileCompare)
{
confess "list (${strFileList})[" . @stryFileList .
"] does not match compare (${strFileCompare})[" . @stryFileCompare . ']';
2014-06-22 01:19:37 +03:00
}
}
}
}
2014-06-22 01:19:37 +03:00
}
}
}
}
2014-06-02 00:23:33 +03:00
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-02 00:23:33 +03:00
# Test remove()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-22 02:16:55 +03:00
if ($strTest eq 'all' || $strTest eq 'remove')
{
$iRun = 0;
&log(INFO, '--------------------------------------------------------------------------------');
2014-06-22 02:16:55 +03:00
&log(INFO, "Test File->remove()\n");
for (my $bRemote = 0; $bRemote <= 1; $bRemote++)
{
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
2014-06-22 02:16:55 +03:00
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-06-22 02:16:55 +03:00
);
# Loop through exists
for (my $bError = 0; $bError <= 1; $bError++)
{
# Loop through exists
for (my $bExists = 0; $bExists <= 1; $bExists++)
{
# Loop through temp
for (my $bTemp = 0; $bTemp <= 1; $bTemp++)
{
# Loop through ignore missing
for (my $bIgnoreMissing = 0; $bIgnoreMissing <= 1; $bIgnoreMissing++)
{
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, err = ${bError}, exists ${bExists}, tmp ${bTemp}, " .
"ignore missing ${bIgnoreMissing}")) {next}
2014-06-22 02:16:55 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-22 02:16:55 +03:00
my $strFile = "${strTestPath}/test.txt";
if ($bError)
{
$strFile = "${strTestPath}/private/test.txt"
}
elsif (!$bExists)
{
$strFile = "${strTestPath}/private/error.txt"
}
else
{
system("echo 'TESTDATA' > ${strFile}" . ($bTemp ? '.backrest.tmp' : ''));
2014-06-22 02:16:55 +03:00
}
# Execute in eval in case of error
my $bRemoved;
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
eval
{
$bRemoved = $oFile->remove(PATH_BACKUP_ABSOLUTE, $strFile, $bTemp, $bIgnoreMissing);
};
if ($@)
{
if ($bError || $bRemote)
{
next;
}
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
if (!$bExists && !$bIgnoreMissing)
{
next;
}
confess 'unexpected error raised: ' . $@;
2014-06-22 02:16:55 +03:00
}
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
if ($bError || $bRemote)
{
confess 'error should have been returned';
}
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
if (!$bRemoved)
{
if (!$bExists && $bIgnoreMissing)
{
next;
}
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
confess 'remove returned false, but something should have been removed';
}
if (-e ($strFile . ($bTemp ? '.backrest.tmp' : '')))
2014-06-22 02:16:55 +03:00
{
confess 'file still exists';
2014-06-22 02:16:55 +03:00
}
}
}
}
}
}
}
2014-05-27 16:00:24 +03:00
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-02 00:23:33 +03:00
# Test hash()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-22 02:16:55 +03:00
if ($strTest eq 'all' || $strTest eq 'hash')
{
$iRun = 0;
&log(INFO, '--------------------------------------------------------------------------------');
2014-06-22 02:16:55 +03:00
&log(INFO, "test File->hash()\n");
for (my $bRemote = false; $bRemote <= true; $bRemote++)
2014-06-22 02:16:55 +03:00
{
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
2014-06-22 02:16:55 +03:00
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
2014-06-22 02:16:55 +03:00
);
# Loop through error
for (my $bError = false; $bError <= true; $bError++)
2014-06-22 02:16:55 +03:00
{
# Loop through exists
for (my $bExists = false; $bExists <= true; $bExists++)
2014-09-30 02:08:08 +03:00
{
# Loop through exists
for (my $bCompressed = false; $bCompressed <= true; $bCompressed++)
2014-06-22 02:16:55 +03:00
{
if (!BackRestTestCommon_Run(++$iRun,
2014-09-30 02:08:08 +03:00
"rmt ${bRemote}, err ${bError}, exists ${bExists}, cmp ${bCompressed}")) {next}
2014-06-22 02:16:55 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-22 02:16:55 +03:00
my $strFile = "${strTestPath}/test.txt";
if ($bError)
{
$strFile = "${strTestPath}/private/test.txt";
}
elsif (!$bExists)
{
$strFile = "${strTestPath}/error.txt";
}
else
{
system("echo 'TESTDATA' > ${strFile}");
if ($bCompressed && !$bRemote)
{
$oFile->compress(PATH_BACKUP_ABSOLUTE, $strFile);
$strFile = $strFile . '.gz';
}
2014-06-22 02:16:55 +03:00
}
# Execute in eval in case of error
my $strHash;
my $iSize;
2014-06-22 02:16:55 +03:00
my $bErrorExpected = !$bExists || $bError || $bRemote;
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
eval
{
($strHash, $iSize) = $oFile->hash_size(PATH_BACKUP_ABSOLUTE, $strFile, $bCompressed)
2014-06-22 02:16:55 +03:00
};
if ($@)
{
if ($bErrorExpected)
{
next;
}
confess 'unexpected error raised: ' . $@;
2014-06-22 02:16:55 +03:00
}
if ($bErrorExpected)
{
confess 'error was expected';
2014-06-22 02:16:55 +03:00
}
2014-06-22 03:08:49 +03:00
2014-06-22 02:16:55 +03:00
if ($strHash ne '06364afe79d801433188262478a76d19777ef351')
{
confess 'hashes do not match';
}
}
}
2014-09-30 02:08:08 +03:00
}
2014-06-22 02:16:55 +03:00
}
}
2014-06-02 00:23:33 +03:00
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
2014-06-02 00:23:33 +03:00
# Test exists()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'exists')
{
$iRun = 0;
2014-06-04 18:58:30 +03:00
&log(INFO, '--------------------------------------------------------------------------------');
&log(INFO, "test File->exists()\n");
2014-06-02 00:23:33 +03:00
for (my $bRemote = 0; $bRemote <= 1; $bRemote++)
2014-06-02 00:23:33 +03:00
{
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$bRemote ? 'backup' : undef,
$bRemote ? $oRemote : $oLocal
);
# Loop through exists
for (my $bExists = 0; $bExists <= 1; $bExists++)
{
2014-06-07 20:15:55 +03:00
# Loop through exists
for (my $bError = 0; $bError <= $bExists; $bError++)
{
if (!BackRestTestCommon_Run(++$iRun,
"rmt ${bRemote}, err ${bError}, exists ${bExists}")) {next}
2014-06-02 00:23:33 +03:00
2014-06-21 22:17:09 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup($bError);
2014-06-02 00:23:33 +03:00
2014-06-07 20:15:55 +03:00
my $strFile = "${strTestPath}/test.txt";
2014-06-02 00:23:33 +03:00
2014-06-07 20:15:55 +03:00
if ($bError)
{
$strFile = "${strTestPath}/private/test.txt";
}
elsif ($bExists)
{
system("echo 'TESTDATA' > ${strFile}");
}
2014-06-04 18:58:30 +03:00
2014-06-07 20:15:55 +03:00
# Execute in eval in case of error
eval
{
2014-06-07 20:15:55 +03:00
if ($oFile->exists(PATH_BACKUP_ABSOLUTE, $strFile) != $bExists)
{
confess "bExists is set to ${bExists}, but exists() returned " . !$bExists;
}
};
if ($@)
{
my $oMessage = $@;
my $iCode;
my $strMessage;
2014-06-07 22:30:13 +03:00
2014-06-07 20:15:55 +03:00
if (blessed($oMessage))
{
if ($oMessage->isa('BackRest::Exception'))
2014-06-07 20:15:55 +03:00
{
$iCode = $oMessage->code();
$strMessage = $oMessage->message();
}
else
{
confess 'unknown error object';
}
}
else
{
$strMessage = $oMessage;
}
2014-06-07 22:30:13 +03:00
2014-06-24 01:54:00 +03:00
if ($bError)
2014-06-07 20:15:55 +03:00
{
next;
}
2014-06-07 22:30:13 +03:00
confess 'error raised: ' . $strMessage . "\n";
2014-06-07 20:15:55 +03:00
}
}
2014-05-27 16:00:24 +03:00
}
}
}
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
# Test copy()
2014-06-03 00:48:07 +03:00
#-------------------------------------------------------------------------------------------------------------------------------
if ($strTest eq 'all' || $strTest eq 'copy')
{
$iRun = 0;
# Loop through small/large
for (my $bLarge = false; $bLarge <= 2; $bLarge++)
{
# Loop through backup local vs remote
for (my $bBackupRemote = 0; $bBackupRemote <= 1; $bBackupRemote++)
{
# Loop through db local vs remote
for (my $bDbRemote = 0; $bDbRemote <= 1; $bDbRemote++)
{
# Backup and db cannot both be remote
if ($bBackupRemote && $bDbRemote)
{
next;
}
# Determine side is remote
my $strRemote = $bBackupRemote ? 'backup' : $bDbRemote ? 'db' : undef;
# Create the file object
2014-10-10 22:13:28 +03:00
my $oFile = new BackRest::File
(
2014-10-10 22:13:28 +03:00
$strStanza,
$strTestPath,
$strRemote,
defined($strRemote) ? $oRemote : $oLocal
);
# Loop through source path types
for (my $bSourcePathType = 0; $bSourcePathType <= 1; $bSourcePathType++)
{
# Loop through destination path types
for (my $bDestinationPathType = 0; $bDestinationPathType <= 1; $bDestinationPathType++)
{
# Loop through source missing/present
for (my $bSourceMissing = 0; $bSourceMissing <= !$bLarge; $bSourceMissing++)
{
# Loop through source ignore/require
for (my $bSourceIgnoreMissing = 0; $bSourceIgnoreMissing <= !$bLarge; $bSourceIgnoreMissing++)
{
# Loop through checksum append
for (my $bChecksumAppend = 0; $bChecksumAppend <= !$bLarge; $bChecksumAppend++)
{
# Loop through source compression
for (my $bSourceCompressed = 0; $bSourceCompressed <= !$bSourceMissing; $bSourceCompressed++)
{
# Loop through destination compression
for (my $bDestinationCompress = 0; $bDestinationCompress <= !$bSourceMissing; $bDestinationCompress++)
{
my $strSourcePathType = $bSourcePathType ? PATH_DB_ABSOLUTE : PATH_BACKUP_ABSOLUTE;
my $strSourcePath = $bSourcePathType ? 'db' : 'backup';
my $strDestinationPathType = $bDestinationPathType ? PATH_DB_ABSOLUTE : PATH_BACKUP_ABSOLUTE;
my $strDestinationPath = $bDestinationPathType ? 'db' : 'backup';
if (!BackRestTestCommon_Run(++$iRun,
"lrg ${bLarge}, rmt " .
(defined($strRemote) && ($strRemote eq $strSourcePath ||
$strRemote eq $strDestinationPath) ? 1 : 0) .
', srcpth ' . (defined($strRemote) && $strRemote eq $strSourcePath ? 'rmt' : 'lcl') .
":${strSourcePath}, srcmiss ${bSourceMissing}, " .
"srcignmiss ${bSourceIgnoreMissing}, srccmp $bSourceCompressed, " .
'dstpth ' .
(defined($strRemote) && $strRemote eq $strDestinationPath ? 'rmt' : 'lcl') .
":${strDestinationPath}, chkapp ${bChecksumAppend}, " .
"dstcmp $bDestinationCompress")) {next}
2014-06-21 22:46:26 +03:00
# Setup test directory
2014-06-22 17:54:31 +03:00
BackRestTestFile_Setup(false);
system("mkdir ${strTestPath}/backup") == 0 or confess 'Unable to create test/backup directory';
system("mkdir ${strTestPath}/db") == 0 or confess 'Unable to create test/db directory';
my $strSourceFile = "${strTestPath}/${strSourcePath}/test-source";
my $strDestinationFile = "${strTestPath}/${strDestinationPath}/test-destination";
my $strCopyHash;
my $iCopySize;
# Create the compressed or uncompressed test file
my $strSourceHash;
my $iSourceSize;
2014-06-22 03:08:49 +03:00
if (!$bSourceMissing)
{
if ($bLarge)
{
$strSourceFile .= '.bin';
$strDestinationFile .= '.bin';
BackRestTestCommon_Execute('cp ' . BackRestTestCommon_DataPathGet() . "/test.archive${bLarge}.bin ${strSourceFile}");
}
else
{
$strSourceFile .= '.txt';
$strDestinationFile .= '.txt';
system("echo 'TESTDATA' > ${strSourceFile}");
}
if ($bLarge == 1)
{
$strSourceHash = 'c2e63b6a49d53a53d6df1aa6b70c7c16747ca099';
$iSourceSize = 16777216;
}
elsif ($bLarge == 2)
{
$strSourceHash = '1c7e00fd09b9dd11fc2966590b3e3274645dd031';
$iSourceSize = 16777216;
}
else
{
$strSourceHash = '06364afe79d801433188262478a76d19777ef351';
$iSourceSize = 9;
}
if ($bSourceCompressed)
{
system("gzip ${strSourceFile}");
$strSourceFile .= '.gz';
}
}
2014-06-06 05:42:47 +03:00
if ($bDestinationCompress)
{
$strDestinationFile .= '.gz';
}
# Run file copy in an eval block because some errors are expected
my $bReturn;
eval
{
($bReturn, $strCopyHash, $iCopySize) =
$oFile->copy($strSourcePathType, $strSourceFile,
$strDestinationPathType, $strDestinationFile,
$bSourceCompressed, $bDestinationCompress,
$bSourceIgnoreMissing, undef, '0700', false, undef, undef,
$bChecksumAppend);
};
# Check for errors after copy
if ($@)
{
my $oMessage = $@;
2014-06-22 03:08:49 +03:00
if (blessed($oMessage))
{
if ($oMessage->isa('BackRest::Exception'))
{
if ($bSourceMissing && !$bSourceIgnoreMissing)
{
next;
}
2014-06-22 03:08:49 +03:00
confess $oMessage->message();
}
else
{
confess 'unknown error object: ' . $oMessage;
}
}
2014-06-28 21:32:34 +03:00
confess $oMessage;
}
if ($bSourceMissing)
{
if ($bSourceIgnoreMissing)
{
if ($bReturn)
{
confess 'copy() returned ' . $bReturn . ' when ignore missing set';
}
2014-06-22 03:08:49 +03:00
next;
}
2014-06-22 03:08:49 +03:00
confess 'expected source file missing error';
}
2014-06-22 03:08:49 +03:00
if (!defined($strCopyHash))
{
confess 'copy hash must be defined';
}
if ($bChecksumAppend)
{
if ($bDestinationCompress)
{
$strDestinationFile =
substr($strDestinationFile, 0, length($strDestinationFile) -3) . "-${strSourceHash}.gz";
}
else
{
$strDestinationFile .= '-' . $strSourceHash;
}
}
unless (-e $strDestinationFile)
{
confess "could not find destination file ${strDestinationFile}";
}
my $strDestinationTest = $strDestinationFile;
if ($bDestinationCompress)
{
$strDestinationTest = substr($strDestinationFile, 0, length($strDestinationFile) - 3) . '.test';
2014-06-22 03:08:49 +03:00
system("gzip -dc ${strDestinationFile} > ${strDestinationTest}") == 0
or die "could not decompress ${strDestinationFile}";
}
my ($strDestinationHash, $iDestinationSize) = $oFile->hash_size(PATH_ABSOLUTE, $strDestinationTest);
if ($strSourceHash ne $strDestinationHash || $strSourceHash ne $strCopyHash)
{
confess "source ${strSourceHash}, copy ${strCopyHash} and destination ${strDestinationHash} file hashes do not match";
}
if ($iSourceSize != $iDestinationSize || $iSourceSize != $iCopySize)
{
confess "source ${iSourceSize}, copy ${iCopySize} and destination ${iDestinationSize} sizes do not match";
}
}
}
}
}
}
}
}
}
}
}
}
if (BackRestTestCommon_Cleanup())
{
BackRestTestFile_Setup(undef, true);
}
}
2014-06-21 21:19:03 +03:00
2014-06-22 03:08:49 +03:00
1;