#!/usr/bin/perl #################################################################################################################################### # test.pl - pgBackRest Unit Tests #################################################################################################################################### #################################################################################################################################### # Perl includes #################################################################################################################################### use strict; use warnings FATAL => qw(all); use Carp qw(confess longmess); # Convert die to confess to capture the stack trace $SIG{__DIE__} = sub { Carp::confess @_ }; use File::Basename qw(dirname); use Getopt::Long qw(GetOptions); use Cwd qw(abs_path); use Pod::Usage qw(pod2usage); use POSIX qw(ceil); use Time::HiRes qw(gettimeofday); use Scalar::Util qw(blessed); use lib dirname($0) . '/../lib'; use BackRest::Common::Ini; use BackRest::Common::Log; use BackRest::Common::String; use BackRest::Common::Wait; use BackRest::Db; use lib dirname($0) . '/lib'; use BackRestTest::BackupTest; use BackRestTest::Common::ExecuteTest; use BackRestTest::Common::VmTest; use BackRestTest::CommonTest; use BackRestTest::CompareTest; use BackRestTest::ConfigTest; use BackRestTest::Docker::ContainerTest; use BackRestTest::FileTest; use BackRestTest::HelpTest; #################################################################################################################################### # Usage #################################################################################################################################### =head1 NAME test.pl - pgBackRest Unit Tests =head1 SYNOPSIS test.pl [options] Test Options: --module test module to execute --test execute the specified test in a module --run execute only the specified test run --thread-max max threads to run for backup/restore (default 4) --dry-run show only the tests that would be executed but don't execute them --no-cleanup don't cleaup after the last test is complete - useful for debugging --infinite repeat selected tests forever --db-version version of postgres to test (or all) --log-force force overwrite of current test log files Configuration Options: --exe pgBackRest executable --psql-bin path to the psql executables (e.g. /usr/lib/postgresql/9.3/bin/) --test-path path where tests are executed (defaults to ./test) --log-level log level to use for tests (defaults to INFO) --quiet, -q equivalent to --log-level=off VM Options: --vm-build build Docker containers --vm execute in a docker container (u12, u14, co6, co7) --vm-out Show VM output (default false) --process-max max VMs to run in parallel (default 1) General Options: --version display version and exit --help display usage and exit =cut #################################################################################################################################### # Command line parameters #################################################################################################################################### my $strLogLevel = 'info'; my $strOS = 'all'; my $bVmOut = false; my $strModule = 'all'; my $strModuleTest = 'all'; my $iModuleTestRun = undef; my $iThreadMax = undef; my $iProcessMax = 1; my $bDryRun = false; my $bNoCleanup = false; my $strPgSqlBin; my $strExe; my $strTestPath; my $bVersion = false; my $bHelp = false; my $bQuiet = false; my $bInfinite = false; my $strDbVersion = 'all'; my $bLogForce = false; my $bVmBuild = false; my $strCommandLine = join(' ', @ARGV); GetOptions ('q|quiet' => \$bQuiet, 'version' => \$bVersion, 'help' => \$bHelp, 'pgsql-bin=s' => \$strPgSqlBin, 'exes=s' => \$strExe, 'test-path=s' => \$strTestPath, 'log-level=s' => \$strLogLevel, 'vm=s' => \$strOS, 'vm-out' => \$bVmOut, 'vm-build' => \$bVmBuild, 'module=s' => \$strModule, 'test=s' => \$strModuleTest, 'run=s' => \$iModuleTestRun, 'thread-max=s' => \$iThreadMax, 'process-max=s' => \$iProcessMax, 'dry-run' => \$bDryRun, 'no-cleanup' => \$bNoCleanup, 'infinite' => \$bInfinite, 'db-version=s' => \$strDbVersion, 'log-force' => \$bLogForce) or pod2usage(2); # Display version and exit if requested if ($bVersion || $bHelp) { syswrite(*STDOUT, 'pgBackRest ' . BACKREST_VERSION . " Unit Tests\n"); if ($bHelp) { syswrite(*STDOUT, "\n"); pod2usage(); } exit 0; } if (@ARGV > 0) { syswrite(*STDOUT, "invalid parameter\n\n"); pod2usage(); } #################################################################################################################################### # Setup #################################################################################################################################### # Set a neutral umask so tests work as expected umask(0); if (defined($strExe) && !-e $strExe) { confess '--exe must exist and be fully qualified' } # Set console log level if ($bQuiet) { $strLogLevel = 'off'; } logLevelSet(undef, uc($strLogLevel)); if ($strModuleTest ne 'all' && $strModule eq 'all') { confess "--module must be provided for --test=\"${strModuleTest}\""; } if (defined($iModuleTestRun) && $strModuleTest eq 'all') { confess "--test must be provided for --run=\"${iModuleTestRun}\""; } # Check thread total if (defined($iThreadMax) && ($iThreadMax < 1 || $iThreadMax > 32)) { confess 'thread-max must be between 1 and 32'; } #################################################################################################################################### # Build Docker containers #################################################################################################################################### if ($bVmBuild) { containerBuild(); exit 0; } #################################################################################################################################### # Make sure version number matches in the change log. #################################################################################################################################### my $hReadMe; my $strLine; my $bMatch = false; my $strChangeLogFile = abs_path(dirname($0) . '/../CHANGELOG.md'); if (!open($hReadMe, '<', $strChangeLogFile)) { confess "unable to open ${strChangeLogFile}"; } while ($strLine = readline($hReadMe)) { if ($strLine =~ /^\#\# v/) { $bMatch = substr($strLine, 4, length(BACKREST_VERSION)) eq BACKREST_VERSION; last; } } if (!$bMatch) { confess 'unable to find version ' . BACKREST_VERSION . " as last revision in ${strChangeLogFile}"; } eval { ################################################################################################################################ # Define tests ################################################################################################################################ my $oTestDefinition = { module => [ # Help tests { name => 'help', test => [ { name => 'help' } ] }, # Config tests { name => 'config', test => [ { name => 'option' }, { name => 'config' } ] }, # File tests { name => 'file', test => [ { name => 'path_create' }, { name => 'move' }, { name => 'compress' }, { name => 'wait' }, { name => 'manifest' }, { name => 'list' }, { name => 'remove' }, { name => 'hash' }, { name => 'exists' }, { name => 'copy' } ] }, # Backup tests { name => 'backup', test => [ { name => 'archive-push', total => 8 }, { name => 'archive-stop', total => 6 }, { name => 'archive-get', total => 8 }, { name => 'expire', total => 1 }, { name => 'synthetic', total => 8, thread => true }, { name => 'full', total => 8, thread => true, db => true } ] } ] }; my $oyTestRun = []; ################################################################################################################################ # Start VM and run ################################################################################################################################ if ($strOS ne 'none') { my $oyVm = vmGet(); if ($strOS ne 'all' && !defined($${oyVm}{$strOS})) { confess &log(ERROR, "${strOS} is not a valid VM"); } # Determine which tests to run my $iTestsToRun = 0; my $stryTestOS = []; if ($strOS eq 'all') { $stryTestOS = ['co6', 'u12', 'co7', 'u14']; } else { $stryTestOS = [$strOS]; } foreach my $strTestOS (@{$stryTestOS}) { foreach my $oModule (@{$$oTestDefinition{module}}) { if ($strModule eq $$oModule{name} || $strModule eq 'all') { foreach my $oTest (@{$$oModule{test}}) { if ($strModuleTest eq $$oTest{name} || $strModuleTest eq 'all') { my $iDbVersionMin = -1; my $iDbVersionMax = -1; if (defined($$oTest{db}) && $$oTest{db}) { $iDbVersionMin = 0; $iDbVersionMax = @{$$oyVm{$strTestOS}{db}} - 1; } my $bFirstDbVersion = true; for (my $iDbVersionIdx = $iDbVersionMax; $iDbVersionIdx >= $iDbVersionMin; $iDbVersionIdx--) { if ($iDbVersionIdx == -1 || $strDbVersion eq 'all' || ($strDbVersion ne 'all' && $strDbVersion eq ${$$oyVm{$strTestOS}{db}}[$iDbVersionIdx])) { my $iTestRunMin = defined($iModuleTestRun) ? $iModuleTestRun : (defined($$oTest{total}) ? 1 : -1); my $iTestRunMax = defined($iModuleTestRun) ? $iModuleTestRun : (defined($$oTest{total}) ? $$oTest{total} : -1); if (defined($$oTest{total}) && $iTestRunMax > $$oTest{total}) { confess &log(ERROR, "invalid run - must be >= 1 and <= $$oTest{total}") } for (my $iTestRunIdx = $iTestRunMin; $iTestRunIdx <= $iTestRunMax; $iTestRunIdx++) { my $iyThreadMax = [defined($iThreadMax) ? $iThreadMax : 1]; if (defined($$oTest{thread}) && $$oTest{thread} && !defined($iThreadMax) && $bFirstDbVersion) { $iyThreadMax = [1, 4]; } foreach my $iThreadTestMax (@{$iyThreadMax}) { my $oTestRun = { os => $strTestOS, module => $$oModule{name}, test => $$oTest{name}, run => $iTestRunIdx == -1 ? undef : $iTestRunIdx, thread => $iThreadTestMax, db => $iDbVersionIdx == -1 ? undef : ${$$oyVm{$strTestOS}{db}}[$iDbVersionIdx] }; push(@{$oyTestRun}, $oTestRun); $iTestsToRun++; } } $bFirstDbVersion = false; } } } } } } } if ($iTestsToRun == 0) { confess &log(ERROR, 'no tests were selected'); } my $iTestFail = 0; my $oyProcess = []; if (!$bDryRun) { for (my $iProcessIdx = 0; $iProcessIdx < 8; $iProcessIdx++) { # &log(INFO, "stop test-${iProcessIdx}"); push(@{$oyProcess}, undef); executeTest("docker rm -f test-${iProcessIdx}", {bSuppressError => true}); } } my $iTestIdx = 0; my $iProcessTotal; my $iTestMax = @{$oyTestRun}; my $lStartTime = time(); # foreach my $oTest (@{$oyTestRun}) do { do { $iProcessTotal = 0; for (my $iProcessIdx = 0; $iProcessIdx < $iProcessMax; $iProcessIdx++) { if (defined($$oyProcess[$iProcessIdx])) { my $oExecDone = $$oyProcess[$iProcessIdx]{exec}; my $strTestDone = $$oyProcess[$iProcessIdx]{test}; my $iTestDoneIdx = $$oyProcess[$iProcessIdx]{idx}; my $iExitStatus = $oExecDone->end(undef, $iProcessMax == 1); if (defined($iExitStatus)) { my $fTestElapsedTime = ceil((gettimeofday() - $$oyProcess[$iProcessIdx]{start_time}) * 100) / 100; if (!($iExitStatus == 0 || $iExitStatus == 255)) { &log(ERROR, "${strTestDone} (err${iExitStatus}-${fTestElapsedTime}s)" . (defined($oExecDone->{strOutLog}) ? ":\n\n" . trim($oExecDone->{strOutLog}) . "\n" : ''), undef, undef, 4); $iTestFail++; } else { &log( INFO, "${strTestDone} (${fTestElapsedTime}s)". ($bVmOut ? ":\n\n" . trim($oExecDone->{strOutLog}) . "\n" : ''), undef, undef, 4); } if (!$bNoCleanup) { executeTest("docker rm -f test-${iProcessIdx}"); } $$oyProcess[$iProcessIdx] = undef; } else { $iProcessTotal++; } } } if ($iProcessTotal == $iProcessMax) { waitHiRes(.1); } } while ($iProcessTotal == $iProcessMax); for (my $iProcessIdx = 0; $iProcessIdx < $iProcessMax; $iProcessIdx++) { if (!defined($$oyProcess[$iProcessIdx]) && $iTestIdx < @{$oyTestRun}) { my $oTest = $$oyTestRun[$iTestIdx]; $iTestIdx++; my $strTest = sprintf('P%0' . length($iProcessMax) . 'd-T%0' . length($iTestMax) . 'd/%0' . length($iTestMax) . "d - ", $iProcessIdx, $iTestIdx, $iTestMax) . "vm=$$oTest{os}, module=$$oTest{module}, test=$$oTest{test}" . (defined($$oTest{run}) ? ", run=$$oTest{run}" : '') . (defined($$oTest{thread}) ? ", thread-max=$$oTest{thread}" : '') . (defined($$oTest{db}) ? ", db=$$oTest{db}" : ''); my $strImage = 'test-' . $iProcessIdx; &log($bDryRun ? INFO : DEBUG, $strTest); if (!$bDryRun) { executeTest("docker run -itd -h $$oTest{os}-test --name=${strImage}" . " -v /backrest:/backrest backrest/$$oTest{os}-test"); } $strCommandLine =~ s/\-\-os\=\S*//g; $strCommandLine =~ s/\-\-test-path\=\S*//g; $strCommandLine =~ s/\-\-module\=\S*//g; $strCommandLine =~ s/\-\-test\=\S*//g; $strCommandLine =~ s/\-\-run\=\S*//g; $strCommandLine =~ s/\-\-db\-version\=\S*//g; my $strCommand = "docker exec -i -u vagrant ${strImage} $0 ${strCommandLine} --test-path=/home/vagrant/test" . " --vm=none --module=$$oTest{module} --test=$$oTest{test}" . (defined($$oTest{run}) ? " --run=$$oTest{run}" : '') . (defined($$oTest{thread}) ? " --thread-max=$$oTest{thread}" : '') . (defined($$oTest{db}) ? " --db-version=$$oTest{db}" : '') . " --no-cleanup"; &log(DEBUG, $strCommand); if (!$bDryRun) { my $fTestStartTime = gettimeofday(); my $oExec = new BackRestTest::Common::ExecuteTest($strCommand, {bSuppressError => true}); $oExec->begin(); my $oProcess = { exec => $oExec, test => $strTest, idx => $iTestIdx, start_time => $fTestStartTime }; $$oyProcess[$iProcessIdx] = $oProcess; } $iProcessTotal++; } } } while ($iProcessTotal > 0); &log(INFO, 'TESTS COMPLETED ' . ($iTestFail == 0 ? 'SUCCESSFULLY' : "WITH ${iTestFail} FAILURE(S)") . ' (' . (time() - $lStartTime) . 's)'); exit 0; } ################################################################################################################################ # Search for psql ################################################################################################################################ my @stryTestVersion; my @stryVersionSupport = versionSupport(); if (!defined($strPgSqlBin)) { # Distribution-specific paths where the PostgreSQL binaries may be located my @strySearchPath = ( '/usr/lib/postgresql/VERSION/bin', # Debian/Ubuntu '/usr/pgsql-VERSION/bin', # CentOS/RHEL/Fedora '/Library/PostgreSQL/VERSION/bin', # OSX '/usr/local/bin' # BSD ); foreach my $strSearchPath (@strySearchPath) { for (my $iVersionIdx = @stryVersionSupport - 1; $iVersionIdx >= 0; $iVersionIdx--) { if ($strDbVersion eq 'all' || $strDbVersion eq 'max' && @stryTestVersion == 0 || $strDbVersion eq $stryVersionSupport[$iVersionIdx]) { my $strVersionPath = $strSearchPath; $strVersionPath =~ s/VERSION/$stryVersionSupport[$iVersionIdx]/g; if (-e "${strVersionPath}/initdb") { &log(INFO, "FOUND pgsql-bin at ${strVersionPath}"); push @stryTestVersion, $strVersionPath; } } } } # Make sure at least one version of postgres was found @stryTestVersion > 0 or confess 'pgsql-bin was not defined and postgres could not be located automatically'; } else { push @stryTestVersion, $strPgSqlBin; } ################################################################################################################################ # Clean whitespace only if test.pl is being run from the test directory in the backrest repo ################################################################################################################################ # if (-e './test.pl' && -e '../bin/pg_backrest') # { # BackRestTestCommon_Execute( # "find .. -type f -not -path \"../.git/*\" -not -path \"*.DS_Store\" -not -path \"../test/test/*\" " . # "-not -path \"../test/data/*\" " . # "-exec sh -c 'for i;do echo \"\$i\" && sed 's/[[:space:]]*\$//' \"\$i\">/tmp/.\$\$ && cat /tmp/.\$\$ " . # "> \"\$i\";done' arg0 {} + > /dev/null", false, true); # } ################################################################################################################################ # Runs tests ################################################################################################################################ # &log(INFO, "Testing with test_path = " . BackRestTestCommon_TestPathGet() . ", host = {strHost}, user = {strUser}, " . # "group = {strGroup}"); my $iRun = 0; do { if (BackRestTestCommon_Setup($strExe, $strTestPath, $stryTestVersion[0], $iModuleTestRun, $bDryRun, $bNoCleanup, $bLogForce)) { &log(INFO, "TESTING psql-bin = $stryTestVersion[0]\n"); if ($bInfinite) { $iRun++; &log(INFO, "INFINITE - RUN ${iRun}\n"); } # if ($strModule eq 'all' || $strModule eq 'ini') # { # BackRestTestIni_Test($strModuleTest); # } if ($strModule eq 'all' || $strModule eq 'help') { BackRestTestHelp_Test($strModuleTest); } if ($strModule eq 'all' || $strModule eq 'config') { BackRestTestConfig_Test($strModuleTest); } if ($strModule eq 'all' || $strModule eq 'file') { BackRestTestFile_Test($strModuleTest); } if ($strModule eq 'all' || $strModule eq 'backup') { if (!defined($iThreadMax) || $iThreadMax == 1) { BackRestTestBackup_Test($strModuleTest, 1); } if (!defined($iThreadMax) || $iThreadMax > 1) { BackRestTestBackup_Test($strModuleTest, defined($iThreadMax) ? $iThreadMax : 4); } if (@stryTestVersion > 1 && ($strModuleTest eq 'all' || $strModuleTest eq 'full')) { for (my $iVersionIdx = 1; $iVersionIdx < @stryTestVersion; $iVersionIdx++) { BackRestTestCommon_Setup($strExe, $strTestPath, $stryTestVersion[$iVersionIdx], $iModuleTestRun, $bDryRun, $bNoCleanup); &log(INFO, "TESTING psql-bin = $stryTestVersion[$iVersionIdx] for backup/full\n"); BackRestTestBackup_Test('full', defined($iThreadMax) ? $iThreadMax : 4); } } } if ($strModule eq 'compare') { BackRestTestCompare_Test($strModuleTest); } } } while ($bInfinite); }; if ($@) { my $oMessage = $@; # If a backrest exception then return the code - don't confess if (blessed($oMessage) && $oMessage->isa('BackRest::Common::Exception')) { syswrite(*STDOUT, $oMessage->trace()); exit $oMessage->code(); } syswrite(*STDOUT, $oMessage); exit 250; } if (!$bDryRun) { &log(INFO, 'TESTS COMPLETED SUCCESSFULLY (DESPITE ANY ERROR MESSAGES YOU SAW)'); }