2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2016-06-24 14:12:58 +02:00
|
|
|
# ContainerTest.pm - Build containers for testing and documentation
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2016-06-24 14:12:58 +02:00
|
|
|
package pgBackRestTest::Common::ContainerTest;
|
2015-12-29 20:57:10 +02:00
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# Perl includes
|
|
|
|
####################################################################################################################################
|
|
|
|
use strict;
|
|
|
|
use warnings FATAL => qw(all);
|
|
|
|
use Carp qw(confess longmess);
|
2016-10-01 19:39:44 +02:00
|
|
|
use English '-no_match_vars';
|
2015-12-29 20:57:10 +02:00
|
|
|
|
|
|
|
use Cwd qw(abs_path);
|
2018-07-21 23:02:42 +02:00
|
|
|
use Digest::SHA qw(sha1_hex);
|
2016-01-09 15:21:53 +02:00
|
|
|
use Exporter qw(import);
|
|
|
|
our @EXPORT = qw();
|
2015-12-29 20:57:10 +02:00
|
|
|
use File::Basename qw(dirname);
|
|
|
|
use Getopt::Long qw(GetOptions);
|
|
|
|
|
2020-03-10 21:41:56 +02:00
|
|
|
use pgBackRestDoc::Common::Log;
|
|
|
|
use pgBackRestDoc::Common::String;
|
2020-03-10 23:57:02 +02:00
|
|
|
use pgBackRestDoc::ProjectInfo;
|
2020-03-10 21:12:44 +02:00
|
|
|
|
2016-04-14 15:30:54 +02:00
|
|
|
use pgBackRestTest::Common::ExecuteTest;
|
|
|
|
use pgBackRestTest::Common::VmTest;
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# User/group definitions
|
|
|
|
####################################################################################################################################
|
2016-10-01 19:39:44 +02:00
|
|
|
use constant TEST_USER => getpwuid($UID) . '';
|
|
|
|
push @EXPORT, qw(TEST_USER);
|
|
|
|
use constant TEST_USER_ID => $UID;
|
2021-05-22 15:30:54 +02:00
|
|
|
push @EXPORT, qw(TEST_USER_ID);
|
2019-01-30 17:03:17 +02:00
|
|
|
use constant TEST_GROUP => getgrgid((getpwnam(TEST_USER))[3]) . '';
|
|
|
|
push @EXPORT, qw(TEST_GROUP);
|
|
|
|
use constant TEST_GROUP_ID => getgrnam(TEST_GROUP) . '';
|
2021-05-22 15:30:54 +02:00
|
|
|
push @EXPORT, qw(TEST_GROUP_ID);
|
2016-06-12 15:00:16 +02:00
|
|
|
|
2017-08-08 22:26:44 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Cert file constants
|
|
|
|
####################################################################################################################################
|
|
|
|
use constant CERT_FAKE_PATH => '/etc/fake-cert';
|
|
|
|
use constant CERT_FAKE_CA => CERT_FAKE_PATH . '/ca.crt';
|
|
|
|
push @EXPORT, qw(CERT_FAKE_CA);
|
2017-08-08 23:15:01 +02:00
|
|
|
use constant CERT_FAKE_SERVER => CERT_FAKE_PATH . '/server.crt';
|
|
|
|
push @EXPORT, qw(CERT_FAKE_SERVER);
|
|
|
|
use constant CERT_FAKE_SERVER_KEY => CERT_FAKE_PATH . '/server.key';
|
|
|
|
push @EXPORT, qw(CERT_FAKE_SERVER_KEY);
|
2017-08-08 22:26:44 +02:00
|
|
|
|
2016-10-01 19:39:44 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
# Container Debug - speeds container debugging by splitting each section into a separate intermediate container
|
2016-10-01 19:39:44 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
use constant CONTAINER_DEBUG => false;
|
2016-10-01 19:39:44 +02:00
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Store cache container checksums
|
|
|
|
####################################################################################################################################
|
|
|
|
my $hContainerCache;
|
|
|
|
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
# Container repo - defines the Docker repository where the containers will be located
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
sub containerRepo
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2018-11-25 02:05:03 +02:00
|
|
|
return PROJECT_EXE . qw(/) . 'test';
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
push @EXPORT, qw(containerRepo);
|
2015-12-29 20:57:10 +02:00
|
|
|
|
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
# Section header
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
sub sectionHeader
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
return (CONTAINER_DEBUG ? "\n\nRUN \\\n" : " && \\\n\n");
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
|
|
|
|
2016-04-14 00:47:29 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Write the Docker container
|
|
|
|
####################################################################################################################################
|
|
|
|
sub containerWrite
|
|
|
|
{
|
2017-06-09 23:51:41 +02:00
|
|
|
my $oStorageDocker = shift;
|
2016-04-14 00:47:29 +02:00
|
|
|
my $strTempPath = shift;
|
2016-06-12 15:00:16 +02:00
|
|
|
my $strOS = shift;
|
|
|
|
my $strTitle = shift;
|
|
|
|
my $strImageParent = shift;
|
2016-04-14 00:47:29 +02:00
|
|
|
my $strImage = shift;
|
2017-06-24 16:59:00 +02:00
|
|
|
my $strCopy = shift;
|
2016-06-12 15:00:16 +02:00
|
|
|
my $strScript = shift;
|
2016-04-14 00:47:29 +02:00
|
|
|
my $bForce = shift;
|
2016-06-12 15:00:16 +02:00
|
|
|
|
2017-02-21 15:59:23 +02:00
|
|
|
my $strTag = containerRepo() . ":${strImage}";
|
2016-06-12 15:00:16 +02:00
|
|
|
|
|
|
|
$strScript =
|
|
|
|
"# ${strTitle} Container\n" .
|
2018-07-21 23:02:42 +02:00
|
|
|
"FROM ${strImageParent}" .
|
2018-11-22 01:13:37 +02:00
|
|
|
(defined($strCopy) ? "\n\n${strCopy}" : '') .
|
2018-07-21 23:02:42 +02:00
|
|
|
(defined($strScript) && $strScript ne ''?
|
|
|
|
"\n\nRUN echo '" . (CONTAINER_DEBUG ? 'DEBUG' : 'OPTIMIZED') . " BUILD'" . $strScript : '');
|
|
|
|
|
|
|
|
# Search for the image in the cache
|
|
|
|
my $strScriptSha1;
|
|
|
|
my $bCached = false;
|
|
|
|
|
|
|
|
if ($strImage =~ /\-base$/)
|
|
|
|
{
|
|
|
|
$strScriptSha1 = sha1_hex($strScript);
|
|
|
|
|
|
|
|
foreach my $strBuild (reverse(keys(%{$hContainerCache})))
|
|
|
|
{
|
2021-05-03 22:31:27 +02:00
|
|
|
if (defined($hContainerCache->{$strBuild}{hostArch()}{$strOS}) &&
|
|
|
|
$hContainerCache->{$strBuild}{hostArch()}{$strOS} eq $strScriptSha1)
|
2018-07-21 23:02:42 +02:00
|
|
|
{
|
|
|
|
&log(INFO, "Using cached ${strTag}-${strBuild} image (${strScriptSha1}) ...");
|
|
|
|
|
|
|
|
$strScript =
|
|
|
|
"# ${strTitle} Container\n" .
|
|
|
|
"FROM ${strTag}-${strBuild}";
|
|
|
|
|
|
|
|
$bCached = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$bCached)
|
|
|
|
{
|
|
|
|
&log(INFO, "Building ${strTag} image" . (defined($strScriptSha1) ? " (${strScriptSha1})" : '') . ' ...');
|
|
|
|
}
|
2016-04-14 00:47:29 +02:00
|
|
|
|
|
|
|
# Write the image
|
2017-06-09 23:51:41 +02:00
|
|
|
$oStorageDocker->put("${strTempPath}/${strImage}", trim($strScript) . "\n");
|
2016-04-14 00:47:29 +02:00
|
|
|
executeTest('docker build' . (defined($bForce) && $bForce ? ' --no-cache' : '') .
|
2017-02-21 15:59:23 +02:00
|
|
|
" -f ${strTempPath}/${strImage} -t ${strTag} ${strTempPath}",
|
2016-04-14 00:47:29 +02:00
|
|
|
{bSuppressStdErr => true});
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
# User/group creation
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
sub groupCreate
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
|
|
|
my $strOS = shift;
|
2017-06-24 16:59:00 +02:00
|
|
|
my $strName = shift;
|
|
|
|
my $iId = shift;
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2021-05-03 22:31:27 +02:00
|
|
|
return "groupadd -f -g${iId} ${strName}";
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
sub userCreate
|
2016-06-12 15:00:16 +02:00
|
|
|
{
|
|
|
|
my $strOS = shift;
|
2017-06-24 16:59:00 +02:00
|
|
|
my $strName = shift;
|
|
|
|
my $iId = shift;
|
2016-06-12 15:00:16 +02:00
|
|
|
my $strGroup = shift;
|
|
|
|
|
|
|
|
my $oVm = vmGet();
|
|
|
|
|
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
return "adduser -g${strGroup} -u${iId} -n ${strName}";
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
|
|
|
elsif ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_DEBIAN)
|
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
return "adduser --uid=${iId} --ingroup=${strGroup} --disabled-password --gecos \"\" ${strName}";
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
confess &log(ERROR, "unable to create user for os '${strOS}'");
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
|
|
|
|
2017-04-10 18:31:30 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
# Setup SSH
|
2017-04-10 18:31:30 +02:00
|
|
|
####################################################################################################################################
|
2017-06-24 16:59:00 +02:00
|
|
|
sub sshSetup
|
2017-04-10 18:31:30 +02:00
|
|
|
{
|
|
|
|
my $strOS = shift;
|
2017-06-24 16:59:00 +02:00
|
|
|
my $strUser = shift;
|
|
|
|
my $strGroup = shift;
|
|
|
|
my $bControlMaster = shift;
|
2017-04-10 18:31:30 +02:00
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
my $strUserPath = $strUser eq 'root' ? "/${strUser}" : "/home/${strUser}";
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
my $strScript = sectionHeader() .
|
|
|
|
"# Setup SSH\n" .
|
2018-07-21 23:02:42 +02:00
|
|
|
" mkdir ${strUserPath}/.ssh && \\\n" .
|
|
|
|
" echo '-----BEGIN RSA PRIVATE KEY-----' > ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'MIICXwIBAAKBgQDR0yJsZW5d5LcqteiOtv8d+FFeFFHDPI0VTcTOdMn1iDiIP1ou' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'X3Q2OyNjsBaDbsRJd+sp9IRq1LKX3zsBcgGZANwm0zduuNEPEU94ajS/uRoejIqY' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo '/XkKOpnEF6ZbQ2S7TaE4sWeGLvba7kUFs0QTOO+N+nV2dMbdqZf6C8lazwIDAQAB' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'AoGBAJXa6xzrnFVmwgK5BKzYuX/YF5TPgk2j80ch0ct50buQXH/Cb0/rUH5i4jWS' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'T6Hy/DFUehnuzpvV6O9auTOhDs3BhEKFRuRLn1nBwTtZny5Hh+cw7azUCEHFCJlz' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'makCrVbgawtno6oU/pFgQm1FcxD0f+Me5ruNcLHqUZsPQwkRAkEA+8pG+ckOlz6R' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'AJLIHedmfcrEY9T7sfdo83bzMOz8H5soUUP4aOTLJYCla1LO7JdDnXMGo0KxaHBP' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'l8j5zDmVewJBANVVPDJr1w37m0FBi37QgUOAijVfLXgyPMxYp2uc9ddjncif0063' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo '0Wc0FQefoPszf3CDrHv/RHvhHq97jXDwTb0CQQDgH83NygoS1r57pCw9chzpG/R0' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'aMEiSPhCvz757fj+qT3aGIal2AJ7/2c/gRZvwrWNETZ3XIZOUKqIkXzJLPjBAkEA' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'wnP799W2Y8d4/+VX2pMBkF7lG7sSviHEq1sP2BZtPBRQKSQNvw3scM7XcGh/mxmY' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'yx0qpqfKa8SKbNgI1+4iXQJBAOlg8MJLwkUtrG+p8wf69oCuZsnyv0K6UMDxm6/8' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'cbvfmvODulYFaIahaqHWEZoRo5CLYZ7gN43WHPOrKxdDL78=' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo '-----END RSA PRIVATE KEY-----' >> ${strUserPath}/.ssh/id_rsa && \\\n" .
|
|
|
|
" echo 'ssh-rsa AAAAB3NzaC1yc2EAAAADAQABAAAAgQDR0yJsZW5d5LcqteiOtv8d+FFeFFHDPI0VTcTOdMn1iDiIP1ouX3Q2OyNjsBaDbsRJd+sp9I" .
|
|
|
|
"Rq1LKX3zsBcgGZANwm0zduuNEPEU94ajS/uRoejIqY/XkKOpnEF6ZbQ2S7TaE4sWeGLvba7kUFs0QTOO+N+nV2dMbdqZf6C8lazw== " .
|
|
|
|
"user\@pgbackrest-test' > ${strUserPath}/.ssh/authorized_keys && \\\n" .
|
|
|
|
" echo 'Host *' > ${strUserPath}/.ssh/config && \\\n" .
|
|
|
|
" echo ' StrictHostKeyChecking no' >> ${strUserPath}/.ssh/config && \\\n";
|
2017-04-10 18:31:30 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
if ($bControlMaster)
|
2017-04-10 18:31:30 +02:00
|
|
|
{
|
|
|
|
$strScript .=
|
2018-07-21 23:02:42 +02:00
|
|
|
" echo ' ControlMaster auto' >> ${strUserPath}/.ssh/config && \\\n" .
|
|
|
|
" echo ' ControlPath /tmp/\%r\@\%h:\%p' >> ${strUserPath}/.ssh/config && \\\n" .
|
|
|
|
" echo ' ControlPersist 30' >> ${strUserPath}/.ssh/config && \\\n";
|
2017-04-10 18:31:30 +02:00
|
|
|
}
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
$strScript .=
|
2018-07-21 23:02:42 +02:00
|
|
|
" chown -R ${strUser}:${strGroup} ${strUserPath}/.ssh && \\\n" .
|
|
|
|
" chmod 700 ${strUserPath}/.ssh && \\\n" .
|
|
|
|
" chmod 600 ${strUserPath}/.ssh/*";
|
2017-06-24 16:59:00 +02:00
|
|
|
|
2017-04-10 18:31:30 +02:00
|
|
|
return $strScript;
|
|
|
|
}
|
|
|
|
|
2017-08-08 22:26:44 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Cert Setup
|
|
|
|
####################################################################################################################################
|
|
|
|
sub certSetup
|
|
|
|
{
|
2018-11-22 01:13:37 +02:00
|
|
|
my $strOS = shift;
|
|
|
|
|
|
|
|
my $strScript =
|
2017-08-08 22:26:44 +02:00
|
|
|
sectionHeader() .
|
|
|
|
"# Generate fake certs\n" .
|
|
|
|
" mkdir -p -m 755 /etc/fake-cert && \\\n" .
|
|
|
|
" cd /etc/fake-cert && \\\n" .
|
|
|
|
" openssl genrsa -out ca.key 2048 && \\\n" .
|
|
|
|
" openssl req -new -x509 -extensions v3_ca -key ca.key -out ca.crt -days 99999 \\\n" .
|
|
|
|
" -subj \"/C=US/ST=Country/L=City/O=Organization/CN=pgbackrest.org\" && \\\n" .
|
|
|
|
" openssl genrsa -out server.key 2048 && \\\n" .
|
|
|
|
" openssl req -new -key server.key -out server.csr \\\n" .
|
|
|
|
" -subj \"/C=US/ST=Country/L=City/O=Organization/CN=*.pgbackrest.org\" && \\\n" .
|
|
|
|
" openssl x509 -req -in server.csr -CA ca.crt -CAkey ca.key -CAcreateserial -out server.crt -days 99999 \\\n" .
|
2018-06-06 21:52:28 +02:00
|
|
|
" -sha256 && \\\n" .
|
2018-11-22 01:13:37 +02:00
|
|
|
" chmod 644 /etc/fake-cert/* && \\\n";
|
|
|
|
|
|
|
|
my $rhVm = vmGet();
|
|
|
|
|
|
|
|
if ($rhVm->{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
|
|
|
$strScript .=
|
|
|
|
" cp /etc/fake-cert/pgbackrest-test-ca.crt /etc/pki/ca-trust/source/anchors && \\\n" .
|
|
|
|
" update-ca-trust extract";
|
|
|
|
}
|
|
|
|
elsif ($rhVm->{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_DEBIAN)
|
|
|
|
{
|
|
|
|
$strScript .=
|
|
|
|
" cp /etc/fake-cert/pgbackrest-test-ca.crt /usr/local/share/ca-certificates && \\\n" .
|
|
|
|
" update-ca-certificates";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
confess &log(ERROR, "unable to install certificate for $rhVm->{$strOS}{&VM_OS_BASE}");
|
|
|
|
}
|
|
|
|
|
|
|
|
return $strScript;
|
2017-08-08 22:26:44 +02:00
|
|
|
}
|
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Entry point setup
|
|
|
|
####################################################################################################################################
|
|
|
|
sub entryPointSetup
|
|
|
|
{
|
|
|
|
my $strOS = shift;
|
|
|
|
|
|
|
|
my $strScript =
|
|
|
|
"\n\n# Start SSH when container starts\n" .
|
|
|
|
'ENTRYPOINT ';
|
|
|
|
|
|
|
|
my $oVm = vmGet();
|
|
|
|
|
|
|
|
if ($oVm->{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL || $strOS eq VM_U12)
|
|
|
|
{
|
|
|
|
$strScript .= '/usr/sbin/sshd -D';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .= 'service ssh restart && bash';
|
|
|
|
}
|
|
|
|
|
|
|
|
return $strScript;
|
|
|
|
}
|
|
|
|
|
2015-12-29 20:57:10 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# Build containers
|
|
|
|
####################################################################################################################################
|
2016-01-09 15:21:53 +02:00
|
|
|
sub containerBuild
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2017-06-09 23:51:41 +02:00
|
|
|
my $oStorageDocker = shift;
|
2016-04-14 00:47:29 +02:00
|
|
|
my $strVm = shift;
|
|
|
|
my $bVmForce = shift;
|
|
|
|
|
2016-01-09 15:21:53 +02:00
|
|
|
# Create temp path
|
2020-03-14 21:40:37 +02:00
|
|
|
my $strTempPath = $oStorageDocker->pathGet('test/result/docker');
|
2017-06-09 23:51:41 +02:00
|
|
|
$oStorageDocker->pathCreate($strTempPath, {strMode => '0770', bIgnoreExists => true, bCreateParent => true});
|
2016-01-09 15:21:53 +02:00
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
# Load container definitions from yaml
|
2021-03-08 23:01:05 +02:00
|
|
|
require YAML::XS;
|
|
|
|
YAML::XS->import(qw(Load));
|
2018-07-21 23:02:42 +02:00
|
|
|
|
|
|
|
$hContainerCache = Load(${$oStorageDocker->get($oStorageDocker->pathGet('test/container.yaml'))});
|
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
# Remove old images on force
|
|
|
|
if ($bVmForce)
|
|
|
|
{
|
2017-02-21 15:59:23 +02:00
|
|
|
my $strRegExp = '^' . containerRepo();
|
2016-06-12 15:00:16 +02:00
|
|
|
|
|
|
|
if ($strVm ne 'all')
|
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
$strRegExp .= "\:${strVm}-";
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
executeTest("rm -f ${strTempPath}/" . ($strVm eq 'all' ? '*' : "${strVm}-*"));
|
|
|
|
imageRemove("${strRegExp}.*");
|
|
|
|
}
|
|
|
|
|
2017-06-09 23:51:41 +02:00
|
|
|
# VM Images
|
|
|
|
################################################################################################################################
|
2016-05-24 14:17:13 +02:00
|
|
|
my $oVm = vmGet();
|
2016-01-11 02:30:51 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
foreach my $strOS ($strVm eq 'all' ? VM_LIST : ($strVm))
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
my $oOS = $oVm->{$strOS};
|
2016-06-12 15:00:16 +02:00
|
|
|
my $bDocBuild = false;
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
# Deprecated OSs can only be used to build packages
|
|
|
|
my $bDeprecated = $oVm->{$strOS}{&VM_DEPRECATED} ? true : false;
|
|
|
|
|
2015-12-29 20:57:10 +02:00
|
|
|
# Base image
|
|
|
|
###########################################################################################################################
|
2016-06-12 15:00:16 +02:00
|
|
|
my $strImageParent = "$$oVm{$strOS}{&VM_IMAGE}";
|
|
|
|
my $strImage = "${strOS}-base";
|
2018-07-21 23:02:42 +02:00
|
|
|
my $strCopy = undef;
|
2017-06-24 16:59:00 +02:00
|
|
|
|
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
|
|
|
my $strScript = sectionHeader() .
|
2019-07-03 04:20:35 +02:00
|
|
|
"# Install packages\n";
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
2020-12-02 23:23:05 +02:00
|
|
|
if ($strOS eq VM_CO7)
|
2019-07-03 04:20:35 +02:00
|
|
|
{
|
|
|
|
$strScript .=
|
2021-01-14 00:32:42 +02:00
|
|
|
" yum -y install centos-release-scl-rh epel-release && \\\n";
|
2019-07-03 04:20:35 +02:00
|
|
|
}
|
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
$strScript .=
|
2017-06-24 16:59:00 +02:00
|
|
|
" yum -y update && \\\n" .
|
2019-07-03 04:20:35 +02:00
|
|
|
" yum -y install openssh-server openssh-clients wget sudo valgrind git \\\n" .
|
|
|
|
" perl perl-Digest-SHA perl-DBD-Pg perl-YAML-LibYAML openssl \\\n" .
|
2018-11-09 04:41:41 +02:00
|
|
|
" gcc make perl-ExtUtils-MakeMaker perl-Test-Simple openssl-devel perl-ExtUtils-Embed rpm-build \\\n" .
|
2020-12-02 23:23:05 +02:00
|
|
|
" zlib-devel libxml2-devel lz4-devel lz4 bzip2-devel bzip2 perl-JSON-PP";
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
2017-06-24 16:59:00 +02:00
|
|
|
else
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2016-06-12 15:00:16 +02:00
|
|
|
$strScript .=
|
2018-06-06 21:52:28 +02:00
|
|
|
" export DEBCONF_NONINTERACTIVE_SEEN=true DEBIAN_FRONTEND=noninteractive && \\\n" .
|
2017-06-24 16:59:00 +02:00
|
|
|
" apt-get update && \\\n" .
|
2019-07-03 04:20:35 +02:00
|
|
|
" apt-get -y install openssh-server wget sudo gcc make valgrind git \\\n" .
|
|
|
|
" libdbd-pg-perl libhtml-parser-perl libssl-dev libperl-dev \\\n" .
|
2018-07-21 23:02:42 +02:00
|
|
|
" libyaml-libyaml-perl tzdata devscripts lintian libxml-checker-perl txt2man debhelper \\\n" .
|
2020-05-05 22:49:01 +02:00
|
|
|
" libppi-html-perl libtemplate-perl libtest-differences-perl zlib1g-dev libxml2-dev pkg-config \\\n" .
|
|
|
|
" libbz2-dev bzip2";
|
2017-06-24 16:59:00 +02:00
|
|
|
|
2018-02-15 00:19:54 +02:00
|
|
|
if ($strOS eq VM_U12)
|
2017-06-24 16:59:00 +02:00
|
|
|
{
|
2018-02-15 00:19:54 +02:00
|
|
|
$strScript .= ' libperl5.14';
|
2017-06-24 16:59:00 +02:00
|
|
|
}
|
2019-04-24 19:23:32 +02:00
|
|
|
else
|
|
|
|
{
|
2020-03-10 20:45:27 +02:00
|
|
|
$strScript .= ' libjson-pp-perl liblz4-dev liblz4-tool';
|
2019-04-24 19:23:32 +02:00
|
|
|
}
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
|
|
|
|
2020-05-04 21:25:27 +02:00
|
|
|
# Add zst command-line tool and development libs when available
|
|
|
|
if (vmWithZst($strOS))
|
|
|
|
{
|
|
|
|
if ($oVm->{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
|
|
|
$strScript .= ' zstd libzstd-devel';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .= ' zstd libzstd-dev';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-23 02:25:49 +02:00
|
|
|
# If no specific version of lcov is requested then install the default package
|
|
|
|
if (!defined($oVm->{$strOS}{&VMDEF_LCOV_VERSION}))
|
|
|
|
{
|
|
|
|
$strScript .= ' lcov';
|
|
|
|
}
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Regenerate SSH keys\n" .
|
|
|
|
" rm -f /etc/ssh/ssh_host_rsa_key* && \\\n" .
|
2017-02-21 15:59:23 +02:00
|
|
|
" ssh-keygen -t rsa -b 1024 -f /etc/ssh/ssh_host_rsa_key";
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_DEBIAN)
|
|
|
|
{
|
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Fix root tty\n" .
|
|
|
|
" sed -i 's/^mesg n/tty -s \\&\\& mesg n/g' /root/.profile";
|
2016-07-26 22:28:43 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Suppress dpkg interactive output\n" .
|
|
|
|
" rm /etc/apt/apt.conf.d/70debconf";
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
if ($strOS eq VM_U12)
|
|
|
|
{
|
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Create run directory required by SSH (not created automatically on Ubuntu 12.04)\n" .
|
|
|
|
" mkdir -p /var/run/sshd";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-08 22:26:44 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
2018-11-22 01:13:37 +02:00
|
|
|
my $strCertPath = 'test/certificate';
|
|
|
|
my $strCertName = 'pgbackrest-test';
|
|
|
|
|
|
|
|
$strCopy = '# Copy Test Certificates';
|
|
|
|
|
|
|
|
foreach my $strFile ('-ca.crt', '.crt', '.key')
|
|
|
|
{
|
|
|
|
$oStorageDocker->copy("${strCertPath}/${strCertName}${strFile}", "${strTempPath}/${strCertName}${strFile}");
|
|
|
|
$strCopy .= "\nCOPY ${strCertName}${strFile} " . CERT_FAKE_PATH . "/${strCertName}${strFile}";
|
|
|
|
}
|
|
|
|
|
|
|
|
$strScript .= certSetup($strOS);
|
2017-08-08 22:26:44 +02:00
|
|
|
|
2019-10-12 17:24:21 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
2019-11-23 02:25:49 +02:00
|
|
|
if (defined($oVm->{$strOS}{&VMDEF_LCOV_VERSION}))
|
|
|
|
{
|
|
|
|
my $strLCovVersion = $oVm->{$strOS}{&VMDEF_LCOV_VERSION};
|
|
|
|
my $strLCovPath = "/root/lcov-${strLCovVersion}";
|
2019-10-12 17:24:21 +02:00
|
|
|
|
2019-11-23 02:25:49 +02:00
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Build lcov ${strLCovVersion}\n" .
|
|
|
|
" wget -q -O - https://github.com/linux-test-project/lcov/releases/download/v${strLCovVersion}/" .
|
|
|
|
"lcov-${strLCovVersion}.tar.gz | tar zx -C /root && \\\n" .
|
|
|
|
" make -C ${strLCovPath} install && \\\n" .
|
|
|
|
" rm -rf ${strLCovPath}";
|
|
|
|
}
|
2019-10-12 17:24:21 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
|
|
|
if (!$bDeprecated)
|
2015-12-29 20:57:10 +02:00
|
|
|
{
|
2017-06-24 16:59:00 +02:00
|
|
|
$strScript .= sectionHeader() .
|
2018-07-21 23:02:42 +02:00
|
|
|
"# Install PostgreSQL packages\n";
|
2017-06-24 16:59:00 +02:00
|
|
|
|
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
2019-07-03 04:20:35 +02:00
|
|
|
$strScript .=
|
|
|
|
" rpm --import http://yum.postgresql.org/RPM-GPG-KEY-PGDG && \\\n";
|
|
|
|
|
2020-12-02 23:23:05 +02:00
|
|
|
if ($strOS eq VM_CO7)
|
2017-06-24 16:59:00 +02:00
|
|
|
{
|
|
|
|
$strScript .=
|
2018-07-21 23:02:42 +02:00
|
|
|
" rpm -ivh \\\n" .
|
2021-05-03 22:31:27 +02:00
|
|
|
" https://download.postgresql.org/pub/repos/yum/reporpms/EL-7-" . hostArch() . "/" .
|
2019-07-25 20:50:02 +02:00
|
|
|
"pgdg-redhat-repo-latest.noarch.rpm && \\\n";
|
2017-06-24 16:59:00 +02:00
|
|
|
}
|
2020-05-07 17:06:56 +02:00
|
|
|
elsif ($strOS eq VM_F32)
|
2019-10-09 21:03:03 +02:00
|
|
|
{
|
|
|
|
$strScript .=
|
|
|
|
" rpm -ivh \\\n" .
|
2021-05-03 22:31:27 +02:00
|
|
|
" https://download.postgresql.org/pub/repos/yum/reporpms/F-32-" . hostArch() . "/" .
|
2019-10-09 21:03:03 +02:00
|
|
|
"pgdg-fedora-repo-latest.noarch.rpm && \\\n";
|
|
|
|
}
|
2019-07-25 20:50:02 +02:00
|
|
|
|
|
|
|
$strScript .= " yum -y install postgresql-devel";
|
2017-06-24 16:59:00 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .=
|
|
|
|
" echo 'deb http://apt.postgresql.org/pub/repos/apt/ " .
|
2021-05-24 23:17:03 +02:00
|
|
|
$$oVm{$strOS}{&VM_OS_REPO} . '-pgdg main' . ($strOS eq VM_U18 ? ' 14' : '') .
|
2018-06-05 14:59:17 +02:00
|
|
|
"' >> /etc/apt/sources.list.d/pgdg.list && \\\n" .
|
2017-06-24 16:59:00 +02:00
|
|
|
" wget --quiet -O - https://www.postgresql.org/media/keys/ACCC4CF8.asc | apt-key add - && \\\n" .
|
2018-06-06 21:52:28 +02:00
|
|
|
" apt-get update && \\\n" .
|
2019-07-25 20:50:02 +02:00
|
|
|
" apt-get install -y postgresql-common libpq-dev && \\\n" .
|
2018-06-06 21:52:28 +02:00
|
|
|
" sed -i 's/^\\#create\\_main\\_cluster.*\$/create\\_main\\_cluster \\= false/' " .
|
|
|
|
"/etc/postgresql-common/createcluster.conf";
|
2017-06-24 16:59:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($oOS->{&VM_DB}) && @{$oOS->{&VM_DB}} > 0)
|
|
|
|
{
|
2018-06-06 21:52:28 +02:00
|
|
|
$strScript .= sectionHeader() .
|
2018-07-21 23:02:42 +02:00
|
|
|
"# Install PostgreSQL\n";
|
2017-06-24 16:59:00 +02:00
|
|
|
|
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
|
|
|
$strScript .= " yum -y install";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .= " apt-get install -y";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Construct list of databases to install
|
|
|
|
foreach my $strDbVersion (@{$oOS->{&VM_DB}})
|
|
|
|
{
|
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
|
|
|
my $strDbVersionNoDot = $strDbVersion;
|
|
|
|
$strDbVersionNoDot =~ s/\.//;
|
|
|
|
|
|
|
|
$strScript .= " postgresql${strDbVersionNoDot}-server";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .= " postgresql-${strDbVersion}";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
2017-06-24 16:59:00 +02:00
|
|
|
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
2019-07-03 04:20:35 +02:00
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_DEBIAN)
|
2017-06-24 16:59:00 +02:00
|
|
|
{
|
2019-07-03 04:20:35 +02:00
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Cleanup\n";
|
|
|
|
|
|
|
|
$strScript .= " apt-get clean";
|
2017-06-24 16:59:00 +02:00
|
|
|
}
|
2016-06-12 15:00:16 +02:00
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
containerWrite(
|
|
|
|
$oStorageDocker, $strTempPath, $strOS, 'Base', $strImageParent, $strImage, $strCopy, $strScript, $bVmForce);
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
# Test image
|
|
|
|
########################################################################################################################
|
|
|
|
if (!$bDeprecated)
|
2016-01-23 01:01:21 +02:00
|
|
|
{
|
2017-02-21 15:59:23 +02:00
|
|
|
$strImageParent = containerRepo() . ":${strOS}-base";
|
2017-06-24 16:59:00 +02:00
|
|
|
$strImage = "${strOS}-test";
|
2016-01-11 02:30:51 +02:00
|
|
|
|
2019-07-05 22:55:17 +02:00
|
|
|
$strCopy = undef;
|
|
|
|
$strScript = '';
|
2016-06-12 15:00:16 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
#---------------------------------------------------------------------------------------------------------------------------
|
2018-07-21 23:02:42 +02:00
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Create banner to make sure pgBackRest ignores it\n" .
|
|
|
|
" echo '***********************************************' > /etc/issue.net && \\\n" .
|
|
|
|
" echo 'Sample banner to make sure banners are skipped.' >> /etc/issue.net && \\\n" .
|
|
|
|
" echo '' >> /etc/issue.net && \\\n" .
|
|
|
|
" echo 'More banner after a blank line.' >> /etc/issue.net && \\\n" .
|
|
|
|
" echo '***********************************************' >> /etc/issue.net && \\\n" .
|
|
|
|
" echo 'Banner /etc/issue.net' >> /etc/ssh/sshd_config";
|
|
|
|
|
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Create test user\n" .
|
|
|
|
' ' . groupCreate($strOS, TEST_GROUP, TEST_GROUP_ID) . " && \\\n" .
|
|
|
|
' ' . userCreate($strOS, TEST_USER, TEST_USER_ID, TEST_GROUP) . " && \\\n" .
|
|
|
|
' mkdir -m 750 /home/' . TEST_USER . "/test && \\\n" .
|
|
|
|
' chown ' . TEST_USER . ':' . TEST_GROUP . ' /home/' . TEST_USER . "/test";
|
|
|
|
|
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Configure sudo\n";
|
|
|
|
|
|
|
|
if ($$oVm{$strOS}{&VM_OS_BASE} eq VM_OS_BASE_RHEL)
|
|
|
|
{
|
|
|
|
$strScript .=
|
2020-05-07 13:38:28 +02:00
|
|
|
# Don't allow sudo to disable core dump (suppresses errors, see https://github.com/sudo-project/sudo/issues/42)
|
|
|
|
" echo \"Set disable_coredump false\" >> /etc/sudo.conf && \\\n" .
|
2018-07-21 23:02:42 +02:00
|
|
|
" echo '%" . TEST_GROUP . " ALL=(ALL) NOPASSWD: ALL' > /etc/sudoers.d/" . TEST_GROUP . " && \\\n" .
|
|
|
|
" sed -i 's/^Defaults requiretty\$/\\# Defaults requiretty/' /etc/sudoers";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strScript .=
|
|
|
|
" echo '%" . TEST_GROUP . " ALL=(ALL) NOPASSWD: ALL' >> /etc/sudoers";
|
|
|
|
}
|
|
|
|
|
|
|
|
$strScript .=
|
|
|
|
sshSetup($strOS, TEST_USER, TEST_GROUP, $$oVm{$strOS}{&VM_CONTROL_MASTER});
|
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
$strScript .= sectionHeader() .
|
|
|
|
"# Make " . TEST_USER . " home dir readable\n" .
|
|
|
|
' chmod g+r,g+x /home/' . TEST_USER;
|
2017-04-10 18:31:30 +02:00
|
|
|
|
2018-07-21 23:02:42 +02:00
|
|
|
$strScript .= entryPointSetup($strOS);
|
|
|
|
|
2017-06-09 23:51:41 +02:00
|
|
|
containerWrite(
|
2017-06-24 16:59:00 +02:00
|
|
|
$oStorageDocker, $strTempPath, $strOS, 'Test', $strImageParent, $strImage, $strCopy, $strScript, $bVmForce);
|
2016-01-23 01:01:21 +02:00
|
|
|
}
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
2017-06-24 16:59:00 +02:00
|
|
|
|
|
|
|
&log(INFO, "Build Complete");
|
2016-06-12 15:00:16 +02:00
|
|
|
}
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
push @EXPORT, qw(containerBuild);
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# containerRemove
|
|
|
|
#
|
|
|
|
# Remove containers that match a regexp.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub containerRemove
|
|
|
|
{
|
|
|
|
my $strRegExp = shift;
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
foreach my $strContainer (sort(split("\n", trim(executeTest('docker ps -a --format "{{.Names}}"')))))
|
|
|
|
{
|
|
|
|
if ($strContainer =~ $strRegExp)
|
|
|
|
{
|
|
|
|
executeTest("docker rm -f ${strContainer}", {bSuppressError => true});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-12-29 20:57:10 +02:00
|
|
|
|
2016-10-01 19:39:44 +02:00
|
|
|
push @EXPORT, qw(containerRemove);
|
|
|
|
|
2016-06-12 15:00:16 +02:00
|
|
|
####################################################################################################################################
|
|
|
|
# imageRemove
|
|
|
|
#
|
|
|
|
# Remove images that match a regexp.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub imageRemove
|
|
|
|
{
|
|
|
|
my $strRegExp = shift;
|
2016-01-23 01:01:21 +02:00
|
|
|
|
2017-06-24 16:59:00 +02:00
|
|
|
foreach my $strImage (sort(split("\n", trim(executeTest('docker images --format "{{.Repository}}:{{.Tag}}"')))))
|
2016-06-12 15:00:16 +02:00
|
|
|
{
|
|
|
|
if ($strImage =~ $strRegExp)
|
|
|
|
{
|
|
|
|
&log(INFO, "Removing ${strImage} image...");
|
2017-06-24 16:59:00 +02:00
|
|
|
executeTest("docker rmi -f ${strImage}");
|
2016-01-23 01:01:21 +02:00
|
|
|
}
|
2015-12-29 20:57:10 +02:00
|
|
|
}
|
|
|
|
}
|
2016-01-09 15:21:53 +02:00
|
|
|
|
|
|
|
1;
|