2015-11-22 16:44:01 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# DOC EXECUTE MODULE
|
|
|
|
####################################################################################################################################
|
|
|
|
package BackRestDoc::Common::DocExecute;
|
|
|
|
use parent 'BackRestDoc::Common::DocRender';
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings FATAL => qw(all);
|
|
|
|
use Carp qw(confess);
|
2017-03-16 19:50:59 -04:00
|
|
|
use English '-no_match_vars';
|
2015-11-22 16:44:01 -05:00
|
|
|
|
|
|
|
use Exporter qw(import);
|
|
|
|
our @EXPORT = qw();
|
|
|
|
use Storable qw(dclone);
|
|
|
|
|
2016-09-06 09:44:50 -04:00
|
|
|
use pgBackRest::Common::Exception;
|
2016-04-14 09:30:54 -04:00
|
|
|
use pgBackRest::Common::Log;
|
|
|
|
use pgBackRest::Common::String;
|
2016-06-02 09:32:56 -04:00
|
|
|
use pgBackRest::Version;
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2017-11-26 17:45:00 -05:00
|
|
|
use pgBackRestBuild::Config::Data;
|
|
|
|
|
|
|
|
use BackRestDoc::Common::DocManifest;
|
|
|
|
|
2016-04-14 09:30:54 -04:00
|
|
|
use pgBackRestTest::Common::ExecuteTest;
|
|
|
|
use pgBackRestTest::Common::HostTest;
|
2016-06-24 08:12:58 -04:00
|
|
|
use pgBackRestTest::Common::HostGroupTest;
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2017-03-16 19:50:59 -04:00
|
|
|
####################################################################################################################################
|
|
|
|
# User that's building the docs
|
|
|
|
####################################################################################################################################
|
2017-06-27 22:27:48 -04:00
|
|
|
use constant DOC_USER => getpwuid($UID) eq 'root' ? 'ubuntu' : getpwuid($UID) . '';
|
2017-03-16 19:50:59 -04:00
|
|
|
|
2017-11-26 17:45:00 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# Generate indexed defines
|
|
|
|
####################################################################################################################################
|
|
|
|
my $rhConfigDefineIndex = cfgDefine();
|
|
|
|
|
|
|
|
foreach my $strKey (sort(keys(%{$rhConfigDefineIndex})))
|
|
|
|
{
|
|
|
|
# Build options for all possible db configurations
|
|
|
|
if (defined($rhConfigDefineIndex->{$strKey}{&CFGDEF_PREFIX}) &&
|
2018-02-03 18:27:38 -05:00
|
|
|
$rhConfigDefineIndex->{$strKey}{&CFGDEF_PREFIX} eq CFGDEF_PREFIX_PG)
|
2017-11-26 17:45:00 -05:00
|
|
|
{
|
|
|
|
my $strPrefix = $rhConfigDefineIndex->{$strKey}{&CFGDEF_PREFIX};
|
|
|
|
|
2018-02-03 18:27:38 -05:00
|
|
|
for (my $iIndex = 1; $iIndex <= CFGDEF_INDEX_PG; $iIndex++)
|
2017-11-26 17:45:00 -05:00
|
|
|
{
|
|
|
|
my $strKeyNew = "${strPrefix}${iIndex}" . substr($strKey, length($strPrefix));
|
|
|
|
|
|
|
|
$rhConfigDefineIndex->{$strKeyNew} = dclone($rhConfigDefineIndex->{$strKey});
|
|
|
|
|
2018-02-03 18:27:38 -05:00
|
|
|
$rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_INDEX_TOTAL} = CFGDEF_INDEX_PG;
|
2017-11-26 17:45:00 -05:00
|
|
|
$rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_INDEX} = $iIndex - 1;
|
|
|
|
|
2018-02-03 18:27:38 -05:00
|
|
|
# Options indexed > 1 are never required
|
|
|
|
if ($iIndex != 1)
|
2017-11-26 17:45:00 -05:00
|
|
|
{
|
|
|
|
$rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_REQUIRED} = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_DEPEND}) &&
|
|
|
|
defined($rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_DEPEND}{&CFGDEF_DEPEND_OPTION}))
|
|
|
|
{
|
|
|
|
$rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_DEPEND}{&CFGDEF_DEPEND_OPTION} =
|
|
|
|
"${strPrefix}${iIndex}" .
|
|
|
|
substr(
|
|
|
|
$rhConfigDefineIndex->{$strKeyNew}{&CFGDEF_DEPEND}{&CFGDEF_DEPEND_OPTION},
|
|
|
|
length($strPrefix));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete($rhConfigDefineIndex->{$strKey});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$rhConfigDefineIndex->{$strKey}{&CFGDEF_INDEX} = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# CONSTRUCTOR
|
|
|
|
####################################################################################################################################
|
|
|
|
sub new
|
|
|
|
{
|
|
|
|
my $class = shift; # Class name
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strType,
|
|
|
|
$oManifest,
|
|
|
|
$strRenderOutKey,
|
|
|
|
$bExe
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-05-26 09:09:42 -04:00
|
|
|
__PACKAGE__ . '->new', \@_,
|
2015-11-22 16:44:01 -05:00
|
|
|
{name => 'strType'},
|
|
|
|
{name => 'oManifest'},
|
|
|
|
{name => 'strRenderOutKey'},
|
|
|
|
{name => 'bExe'}
|
|
|
|
);
|
|
|
|
|
|
|
|
# Create the class hash
|
2017-07-26 10:22:22 -04:00
|
|
|
my $self = $class->SUPER::new($strType, $oManifest, $bExe, $strRenderOutKey);
|
2015-11-22 16:44:01 -05:00
|
|
|
bless $self, $class;
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($self->{oSource}{hyCache}))
|
|
|
|
{
|
|
|
|
$self->{bCache} = true;
|
|
|
|
$self->{iCacheIdx} = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$self->{bCache} = false;
|
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
$self->{bExe} = $bExe;
|
|
|
|
|
2016-11-07 17:06:35 +02:00
|
|
|
$self->{iCmdLineLen} = $self->{oDoc}->paramGet('cmd-line-len', false, 80);
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'self', value => $self}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
####################################################################################################################################
|
|
|
|
# executeKey
|
|
|
|
#
|
|
|
|
# Get a unique key for the execution step to determine if the cache is valid.
|
|
|
|
####################################################################################################################################
|
|
|
|
sub executeKey
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strHostName,
|
|
|
|
$oCommand,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->executeKey', \@_,
|
|
|
|
{name => 'strHostName', trace => true},
|
|
|
|
{name => 'oCommand', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
# Add user to command
|
2018-06-10 14:13:56 -04:00
|
|
|
my $bUserForce = $oCommand->paramTest('user-force', 'y') ? true : false;
|
2016-06-02 09:32:56 -04:00
|
|
|
my $strCommand = $self->{oManifest}->variableReplace(trim($oCommand->fieldGet('exe-cmd')));
|
2018-06-10 14:13:56 -04:00
|
|
|
my $strUser = $self->{oManifest}->variableReplace($oCommand->paramGet('user', false, DOC_USER));
|
|
|
|
$strCommand = ($strUser eq DOC_USER || $bUserForce ? '' : ('sudo ' . ($strUser eq 'root' ? '' : "-u $strUser "))) . $strCommand;
|
2016-06-02 09:32:56 -04:00
|
|
|
|
|
|
|
# Format and split command
|
|
|
|
$strCommand =~ s/[ ]*\n[ ]*/ \\\n /smg;
|
2016-11-07 17:06:35 +02:00
|
|
|
$strCommand =~ s/ \\\@ \\//smg;
|
2016-06-02 09:32:56 -04:00
|
|
|
my @stryCommand = split("\n", $strCommand);
|
|
|
|
|
|
|
|
my $hCacheKey =
|
|
|
|
{
|
|
|
|
host => $strHostName,
|
|
|
|
cmd => \@stryCommand,
|
|
|
|
output => JSON::PP::false,
|
|
|
|
};
|
|
|
|
|
2018-06-10 14:13:56 -04:00
|
|
|
$$hCacheKey{'run-as-user'} = $bUserForce ? $strUser : undef;
|
|
|
|
|
2016-06-03 20:07:28 -04:00
|
|
|
if (defined($oCommand->fieldGet('exe-cmd-extra', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{'cmd-extra'} = $oCommand->fieldGet('exe-cmd-extra');
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($oCommand->paramGet('err-expect', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{'err-expect'} = $oCommand->paramGet('err-expect');
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($oCommand->paramTest('output', 'y') || $oCommand->paramTest('show', 'y') || $oCommand->paramTest('variable-key'))
|
|
|
|
{
|
|
|
|
$$hCacheKey{'output'} = JSON::PP::true;
|
|
|
|
}
|
|
|
|
|
2018-06-10 14:13:56 -04:00
|
|
|
$$hCacheKey{'load-env'} = $oCommand->paramTest('load-env', 'n') ? JSON::PP::false : JSON::PP::true;
|
|
|
|
$$hCacheKey{'bash-wrap'} = $oCommand->paramTest('bash-wrap', 'n') ? JSON::PP::false : JSON::PP::true;
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($oCommand->fieldGet('exe-highlight', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{'output'} = JSON::PP::true;
|
|
|
|
$$hCacheKey{highlight}{'filter'} = $oCommand->paramTest('filter', 'n') ? JSON::PP::false : JSON::PP::true;
|
|
|
|
$$hCacheKey{highlight}{'filter-context'} = $oCommand->paramGet('filter-context', false, 2);
|
|
|
|
|
|
|
|
my @stryHighlight;
|
|
|
|
$stryHighlight[0] = $self->{oManifest}->variableReplace($oCommand->fieldGet('exe-highlight'));
|
|
|
|
|
|
|
|
$$hCacheKey{highlight}{list} = \@stryHighlight;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'hExecuteKey', value => $hCacheKey, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# execute
|
|
|
|
####################################################################################################################################
|
|
|
|
sub execute
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oSection,
|
|
|
|
$strHostName,
|
|
|
|
$oCommand,
|
2016-06-02 09:32:56 -04:00
|
|
|
$iIndent,
|
|
|
|
$bCache,
|
2018-06-12 13:43:15 -04:00
|
|
|
$bShow,
|
2015-11-22 16:44:01 -05:00
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-05-26 09:09:42 -04:00
|
|
|
__PACKAGE__ . '->execute', \@_,
|
2015-11-22 16:44:01 -05:00
|
|
|
{name => 'oSection'},
|
|
|
|
{name => 'strHostName'},
|
|
|
|
{name => 'oCommand'},
|
2018-06-12 13:43:15 -04:00
|
|
|
{name => 'iIndent', optional => true, default => 1},
|
|
|
|
{name => 'bCache', optional => true, default => true},
|
|
|
|
{name => 'bShow', optional => true, default => true},
|
2015-11-22 16:44:01 -05:00
|
|
|
);
|
|
|
|
|
|
|
|
# Working variables
|
2016-06-02 09:32:56 -04:00
|
|
|
my $hCacheKey = $self->executeKey($strHostName, $oCommand);
|
|
|
|
my $strCommand = join("\n", @{$$hCacheKey{cmd}});
|
2015-11-22 16:44:01 -05:00
|
|
|
my $strOutput;
|
|
|
|
|
2018-06-12 13:43:15 -04:00
|
|
|
if ($bShow && $self->{bExe} && $self->isRequired($oSection))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
# Make sure that no lines are greater than 80 chars
|
|
|
|
foreach my $strLine (split("\n", $strCommand))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-11-07 17:06:35 +02:00
|
|
|
if (length(trim($strLine)) > $self->{iCmdLineLen})
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-11-07 17:06:35 +02:00
|
|
|
confess &log(ERROR,
|
|
|
|
"command has a line > $self->{iCmdLineLen} characters:\n${strCommand}\noffending line: ${strLine}");
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
&log(DEBUG, (' ' x $iIndent) . "execute: $strCommand");
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2018-11-29 14:45:15 -05:00
|
|
|
if ($self->{oManifest}->variableReplace($oCommand->paramGet('skip', false, 'n')) ne 'y' ||
|
|
|
|
$oCommand->paramGet('pre', false, 'n') eq 'y' && $self->{oManifest}->{bPre})
|
2016-06-02 09:32:56 -04:00
|
|
|
{
|
|
|
|
if ($self->{bExe} && $self->isRequired($oSection))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
my ($bCacheHit, $strCacheType, $hCacheKey, $hCacheValue) = $self->cachePop('exe', $hCacheKey);
|
|
|
|
|
|
|
|
if ($bCacheHit)
|
|
|
|
{
|
|
|
|
$strOutput = defined($$hCacheValue{output}) ? join("\n", @{$$hCacheValue{output}}) : undef;
|
|
|
|
}
|
|
|
|
else
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
|
|
|
# Check that the host is valid
|
|
|
|
my $oHost = $self->{host}{$strHostName};
|
|
|
|
|
|
|
|
if (!defined($oHost))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, "cannot execute on host ${strHostName} because the host does not exist");
|
|
|
|
}
|
|
|
|
|
2016-06-03 20:07:28 -04:00
|
|
|
my $oExec = $oHost->execute(
|
|
|
|
$strCommand . (defined($$hCacheKey{'cmd-extra'}) ? ' ' . $$hCacheKey{'cmd-extra'} : ''),
|
|
|
|
{iExpectedExitStatus => $$hCacheKey{'err-expect'},
|
|
|
|
bSuppressError => $oCommand->paramTest('err-suppress', 'y'),
|
2018-06-10 14:13:56 -04:00
|
|
|
iRetrySeconds => $oCommand->paramGet('retry', false)}, $hCacheKey->{'run-as-user'},
|
|
|
|
{bLoadEnv => $hCacheKey->{'load-env'}, bBashWrap => $hCacheKey->{'bash-wrap'}});
|
2015-11-22 16:44:01 -05:00
|
|
|
$oExec->begin();
|
|
|
|
$oExec->end();
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($oExec->{strOutLog}) && $oExec->{strOutLog} ne '')
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
|
|
|
$strOutput = $oExec->{strOutLog};
|
|
|
|
|
|
|
|
# Trim off extra linefeeds before and after
|
|
|
|
$strOutput =~ s/^\n+|\n$//g;
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($$hCacheKey{'err-expect'}) && defined($oExec->{strErrorLog}) && $oExec->{strErrorLog} ne '')
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
$strOutput .= $oExec->{strErrorLog};
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
|
|
|
|
if ($$hCacheKey{output} && defined($$hCacheKey{highlight}) && $$hCacheKey{highlight}{filter} && defined($strOutput))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
my $strHighLight = @{$$hCacheKey{highlight}{list}}[0];
|
2015-11-22 16:44:01 -05:00
|
|
|
|
|
|
|
if (!defined($strHighLight))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'filter requires highlight definition: ' . $strCommand);
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
my $iFilterContext = $$hCacheKey{highlight}{'filter-context'};
|
2015-11-22 16:44:01 -05:00
|
|
|
|
|
|
|
my @stryOutput = split("\n", $strOutput);
|
|
|
|
undef($strOutput);
|
|
|
|
# my $iFiltered = 0;
|
|
|
|
my $iLastOutput = -1;
|
|
|
|
|
|
|
|
for (my $iIndex = 0; $iIndex < @stryOutput; $iIndex++)
|
|
|
|
{
|
|
|
|
if ($stryOutput[$iIndex] =~ /$strHighLight/)
|
|
|
|
{
|
|
|
|
# Determine the first line to output
|
|
|
|
my $iFilterFirst = $iIndex - $iFilterContext;
|
|
|
|
|
|
|
|
# Don't go past the beginning
|
|
|
|
$iFilterFirst = $iFilterFirst < 0 ? 0 : $iFilterFirst;
|
|
|
|
|
|
|
|
# Don't repeat lines that have already been output
|
|
|
|
$iFilterFirst = $iFilterFirst <= $iLastOutput ? $iLastOutput + 1 : $iFilterFirst;
|
|
|
|
|
|
|
|
# Determine the last line to output
|
|
|
|
my $iFilterLast = $iIndex + $iFilterContext;
|
|
|
|
|
|
|
|
# Don't got past the end
|
|
|
|
$iFilterLast = $iFilterLast >= @stryOutput ? @stryOutput -1 : $iFilterLast;
|
|
|
|
|
|
|
|
# Mark filtered lines if any
|
|
|
|
if ($iFilterFirst > $iLastOutput + 1)
|
|
|
|
{
|
|
|
|
my $iFiltered = $iFilterFirst - ($iLastOutput + 1);
|
|
|
|
|
|
|
|
if ($iFiltered > 1)
|
|
|
|
{
|
|
|
|
$strOutput .= (defined($strOutput) ? "\n" : '') .
|
|
|
|
" [filtered ${iFiltered} lines of output]";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$iFilterFirst -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Output the lines
|
|
|
|
for (my $iOutputIndex = $iFilterFirst; $iOutputIndex <= $iFilterLast; $iOutputIndex++)
|
|
|
|
{
|
|
|
|
$strOutput .= (defined($strOutput) ? "\n" : '') . $stryOutput[$iOutputIndex];
|
|
|
|
}
|
|
|
|
|
|
|
|
$iLastOutput = $iFilterLast;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (@stryOutput - 1 > $iLastOutput + 1)
|
|
|
|
{
|
|
|
|
my $iFiltered = (@stryOutput - 1) - ($iLastOutput + 1);
|
|
|
|
|
|
|
|
if ($iFiltered > 1)
|
|
|
|
{
|
|
|
|
$strOutput .= (defined($strOutput) ? "\n" : '') .
|
|
|
|
" [filtered ${iFiltered} lines of output]";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-04-10 17:23:39 -04:00
|
|
|
$strOutput .= (defined($strOutput) ? "\n" : '') . $stryOutput[-1];
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
|
|
|
|
if (!$$hCacheKey{output})
|
|
|
|
{
|
|
|
|
$strOutput = undef;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($strOutput))
|
|
|
|
{
|
|
|
|
my @stryOutput = split("\n", $strOutput);
|
|
|
|
$$hCacheValue{output} = \@stryOutput;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($bCache)
|
|
|
|
{
|
|
|
|
$self->cachePush($strCacheType, $hCacheKey, $hCacheValue);
|
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
|
|
|
|
# Output is assigned to a var
|
|
|
|
if ($oCommand->paramTest('variable-key'))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
$self->{oManifest}->variableSet($oCommand->paramGet('variable-key'), trim($strOutput), true);
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
elsif ($$hCacheKey{output})
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
$strOutput = 'Output suppressed for testing';
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
# Default variable output when it was not set by execution
|
|
|
|
if ($oCommand->paramTest('variable-key') && !defined($self->{oManifest}->variableGet($oCommand->paramGet('variable-key'))))
|
|
|
|
{
|
|
|
|
$self->{oManifest}->variableSet($oCommand->paramGet('variable-key'), '[Test Variable]', true);
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strCommand', value => $strCommand, trace => true},
|
|
|
|
{name => 'strOutput', value => $strOutput, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# configKey
|
|
|
|
####################################################################################################################################
|
|
|
|
sub configKey
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oConfig,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->hostKey', \@_,
|
|
|
|
{name => 'oConfig', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
my $hCacheKey =
|
|
|
|
{
|
|
|
|
host => $self->{oManifest}->variableReplace($oConfig->paramGet('host')),
|
|
|
|
file => $self->{oManifest}->variableReplace($oConfig->paramGet('file')),
|
|
|
|
};
|
|
|
|
|
|
|
|
if ($oConfig->paramTest('reset', 'y'))
|
|
|
|
{
|
|
|
|
$$hCacheKey{reset} = JSON::PP::true;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Add all options to the key
|
|
|
|
my $strOptionTag = $oConfig->nameGet() eq 'backrest-config' ? 'backrest-config-option' : 'postgres-config-option';
|
|
|
|
|
|
|
|
foreach my $oOption ($oConfig->nodeList($strOptionTag))
|
|
|
|
{
|
|
|
|
my $hOption = {};
|
|
|
|
|
|
|
|
if ($oOption->paramTest('remove', 'y'))
|
|
|
|
{
|
|
|
|
$$hOption{remove} = JSON::PP::true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (defined($oOption->valueGet(false)))
|
|
|
|
{
|
|
|
|
$$hOption{value} = $self->{oManifest}->variableReplace($oOption->valueGet());
|
|
|
|
}
|
|
|
|
|
|
|
|
my $strKey = $self->{oManifest}->variableReplace($oOption->paramGet('key'));
|
|
|
|
|
|
|
|
if ($oConfig->nameGet() eq 'backrest-config')
|
|
|
|
{
|
|
|
|
my $strSection = $self->{oManifest}->variableReplace($oOption->paramGet('section'));
|
|
|
|
|
|
|
|
$$hCacheKey{option}{$strSection}{$strKey} = $hOption;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$$hCacheKey{option}{$strKey} = $hOption;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'hCacheKey', value => $hCacheKey, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# backrestConfig
|
|
|
|
####################################################################################################################################
|
|
|
|
sub backrestConfig
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oSection,
|
|
|
|
$oConfig,
|
|
|
|
$iDepth
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-05-26 09:09:42 -04:00
|
|
|
__PACKAGE__ . '->backrestConfig', \@_,
|
2015-11-22 16:44:01 -05:00
|
|
|
{name => 'oSection'},
|
|
|
|
{name => 'oConfig'},
|
|
|
|
{name => 'iDepth'}
|
|
|
|
);
|
|
|
|
|
|
|
|
# Working variables
|
2016-06-02 09:32:56 -04:00
|
|
|
my $hCacheKey = $self->configKey($oConfig);
|
|
|
|
my $strFile = $$hCacheKey{file};
|
|
|
|
my $strConfig = undef;
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
&log(DEBUG, (' ' x $iDepth) . 'process backrest config: ' . $$hCacheKey{file});
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($self->{bExe} && $self->isRequired($oSection))
|
|
|
|
{
|
2018-11-24 20:31:35 -05:00
|
|
|
# Load module dynamically
|
|
|
|
require pgBackRest::Common::Ini;
|
|
|
|
pgBackRest::Common::Ini->import();
|
|
|
|
|
2018-11-24 19:05:03 -05:00
|
|
|
my ($bCacheHit, $strCacheType, $hCacheKey, $hCacheValue) = $self->cachePop('cfg-' . PROJECT_EXE, $hCacheKey);
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($bCacheHit)
|
|
|
|
{
|
|
|
|
$strConfig = defined($$hCacheValue{config}) ? join("\n", @{$$hCacheValue{config}}) : undef;
|
|
|
|
}
|
|
|
|
else
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
|
|
|
# Check that the host is valid
|
|
|
|
my $strHostName = $self->{oManifest}->variableReplace($oConfig->paramGet('host'));
|
|
|
|
my $oHost = $self->{host}{$strHostName};
|
|
|
|
|
|
|
|
if (!defined($oHost))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, "cannot configure backrest on host ${strHostName} because the host does not exist");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Reset all options
|
|
|
|
if ($oConfig->paramTest('reset', 'y'))
|
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
delete(${$self->{config}}{$strHostName}{$$hCacheKey{file}})
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
foreach my $oOption ($oConfig->nodeList('backrest-config-option'))
|
|
|
|
{
|
2016-11-07 17:06:35 +02:00
|
|
|
my $strSection = $self->{oManifest}->variableReplace($oOption->paramGet('section'));
|
|
|
|
my $strKey = $self->{oManifest}->variableReplace($oOption->paramGet('key'));
|
2015-11-22 16:44:01 -05:00
|
|
|
my $strValue;
|
|
|
|
|
|
|
|
if (!$oOption->paramTest('remove', 'y'))
|
|
|
|
{
|
|
|
|
$strValue = $self->{oManifest}->variableReplace(trim($oOption->valueGet(false)));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($strValue))
|
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
delete(${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}{$strKey});
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (keys(%{${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}}) == 0)
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
delete(${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection});
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
&log(DEBUG, (' ' x ($iDepth + 1)) . "reset ${strSection}->${strKey}");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-04-14 09:30:54 -04:00
|
|
|
# Make sure the specified option exists
|
2016-08-24 12:39:27 -04:00
|
|
|
# ??? This is too simplistic to handle new indexed options. The check below works for now but it would be good
|
|
|
|
# ??? to bring back more sophisticated checking in the future.
|
2017-11-26 17:45:00 -05:00
|
|
|
# if (!defined($rhConfigDefineIndex->{$strKey}))
|
2016-08-24 12:39:27 -04:00
|
|
|
# {
|
|
|
|
# confess &log(ERROR, "option ${strKey} does not exist");
|
|
|
|
# }
|
2016-04-14 09:30:54 -04:00
|
|
|
|
|
|
|
# If this option is a hash and the value is already set then append to the array
|
2017-11-26 17:45:00 -05:00
|
|
|
if (defined($rhConfigDefineIndex->{$strKey}) &&
|
|
|
|
$rhConfigDefineIndex->{$strKey}{&CFGDEF_TYPE} eq CFGDEF_TYPE_HASH &&
|
2016-06-02 09:32:56 -04:00
|
|
|
defined(${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}{$strKey}))
|
2016-04-14 09:30:54 -04:00
|
|
|
{
|
|
|
|
my @oValue = ();
|
2016-06-02 09:32:56 -04:00
|
|
|
my $strHashValue = ${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}{$strKey};
|
2016-04-14 09:30:54 -04:00
|
|
|
|
|
|
|
# If there is only one key/value
|
|
|
|
if (ref(\$strHashValue) eq 'SCALAR')
|
|
|
|
{
|
|
|
|
push(@oValue, $strHashValue);
|
|
|
|
}
|
|
|
|
# Else if there is an array of values
|
|
|
|
else
|
|
|
|
{
|
|
|
|
@oValue = @{$strHashValue};
|
|
|
|
}
|
|
|
|
|
|
|
|
push(@oValue, $strValue);
|
2016-06-02 09:32:56 -04:00
|
|
|
${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}{$strKey} = \@oValue;
|
2016-04-14 09:30:54 -04:00
|
|
|
}
|
|
|
|
# else just set the value
|
|
|
|
else
|
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
${$self->{config}}{$strHostName}{$$hCacheKey{file}}{$strSection}{$strKey} = $strValue;
|
2016-04-14 09:30:54 -04:00
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
&log(DEBUG, (' ' x ($iDepth + 1)) . "set ${strSection}->${strKey} = ${strValue}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-03-16 19:50:59 -04:00
|
|
|
my $strLocalFile = '/home/' . DOC_USER . '/data/pgbackrest.conf';
|
2015-11-22 16:44:01 -05:00
|
|
|
|
|
|
|
# Save the ini file
|
2017-06-09 17:51:41 -04:00
|
|
|
$self->{oManifest}->storage()->put($strLocalFile, iniRender($self->{config}{$strHostName}{$$hCacheKey{file}}, true));
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2017-02-10 10:22:05 -05:00
|
|
|
$oHost->copyTo(
|
|
|
|
$strLocalFile, $$hCacheKey{file},
|
|
|
|
$self->{oManifest}->variableReplace($oConfig->paramGet('owner', false, 'postgres:postgres')), '640');
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-10-05 09:09:30 -04:00
|
|
|
# Remove the log-console-stderr option before pushing into the cache
|
|
|
|
# ??? This is not very pretty and should be replaced with a general way to hide config options
|
|
|
|
my $oConfigClean = dclone($self->{config}{$strHostName}{$$hCacheKey{file}});
|
2017-08-25 16:47:47 -04:00
|
|
|
delete($$oConfigClean{&CFGDEF_SECTION_GLOBAL}{&CFGOPT_LOG_LEVEL_STDERR});
|
|
|
|
delete($$oConfigClean{&CFGDEF_SECTION_GLOBAL}{&CFGOPT_LOG_TIMESTAMP});
|
2018-06-12 13:43:15 -04:00
|
|
|
delete($$oConfigClean{&CFGDEF_SECTION_GLOBAL}{'repo1-s3-verify-ssl'});
|
2016-10-05 09:09:30 -04:00
|
|
|
|
2017-08-25 16:47:47 -04:00
|
|
|
if (keys(%{$$oConfigClean{&CFGDEF_SECTION_GLOBAL}}) == 0)
|
2016-10-05 09:09:30 -04:00
|
|
|
{
|
2017-08-25 16:47:47 -04:00
|
|
|
delete($$oConfigClean{&CFGDEF_SECTION_GLOBAL});
|
2016-10-05 09:09:30 -04:00
|
|
|
}
|
|
|
|
|
2017-06-09 17:51:41 -04:00
|
|
|
$self->{oManifest}->storage()->put("${strLocalFile}.clean", iniRender($oConfigClean, true));
|
2016-10-05 09:09:30 -04:00
|
|
|
|
|
|
|
# Push config file into the cache
|
2017-06-09 17:51:41 -04:00
|
|
|
$strConfig = ${$self->{oManifest}->storage()->get("${strLocalFile}.clean")};
|
2016-10-05 09:09:30 -04:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
my @stryConfig = undef;
|
|
|
|
|
|
|
|
if (trim($strConfig) ne '')
|
|
|
|
{
|
|
|
|
@stryConfig = split("\n", $strConfig);
|
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
$$hCacheValue{config} = \@stryConfig;
|
|
|
|
$self->cachePush($strCacheType, $hCacheKey, $hCacheValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strConfig = 'Config suppressed for testing';
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strFile', value => $strFile, trace => true},
|
|
|
|
{name => 'strConfig', value => $strConfig, trace => true},
|
|
|
|
{name => 'bShow', value => $oConfig->paramTest('show', 'n') ? false : true, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# postgresConfig
|
|
|
|
####################################################################################################################################
|
|
|
|
sub postgresConfig
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oSection,
|
|
|
|
$oConfig,
|
|
|
|
$iDepth
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-05-26 09:09:42 -04:00
|
|
|
__PACKAGE__ . '->postgresConfig', \@_,
|
2015-11-22 16:44:01 -05:00
|
|
|
{name => 'oSection'},
|
|
|
|
{name => 'oConfig'},
|
|
|
|
{name => 'iDepth'}
|
|
|
|
);
|
|
|
|
|
|
|
|
# Working variables
|
2016-06-02 09:32:56 -04:00
|
|
|
my $hCacheKey = $self->configKey($oConfig);
|
|
|
|
my $strFile = $$hCacheKey{file};
|
2015-11-22 16:44:01 -05:00
|
|
|
my $strConfig;
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($self->{bExe} && $self->isRequired($oSection))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
my ($bCacheHit, $strCacheType, $hCacheKey, $hCacheValue) = $self->cachePop('cfg-postgresql', $hCacheKey);
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($bCacheHit)
|
|
|
|
{
|
|
|
|
$strConfig = defined($$hCacheValue{config}) ? join("\n", @{$$hCacheValue{config}}) : undef;
|
|
|
|
}
|
|
|
|
else
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
|
|
|
# Check that the host is valid
|
|
|
|
my $strHostName = $self->{oManifest}->variableReplace($oConfig->paramGet('host'));
|
|
|
|
my $oHost = $self->{host}{$strHostName};
|
|
|
|
|
|
|
|
if (!defined($oHost))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, "cannot configure postgres on host ${strHostName} because the host does not exist");
|
|
|
|
}
|
|
|
|
|
2017-03-16 19:50:59 -04:00
|
|
|
my $strLocalFile = '/home/' . DOC_USER . '/data/postgresql.conf';
|
2016-06-02 09:32:56 -04:00
|
|
|
$oHost->copyFrom($$hCacheKey{file}, $strLocalFile);
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (!defined(${$self->{'pg-config'}}{$strHostName}{$$hCacheKey{file}}{base}) && $self->{bExe})
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2017-06-09 17:51:41 -04:00
|
|
|
${$self->{'pg-config'}}{$strHostName}{$$hCacheKey{file}}{base} =
|
|
|
|
${$self->{oManifest}->storage()->get($strLocalFile)};
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
my $oConfigHash = $self->{'pg-config'}{$strHostName}{$$hCacheKey{file}};
|
2015-11-22 16:44:01 -05:00
|
|
|
my $oConfigHashNew;
|
|
|
|
|
|
|
|
if (!defined($$oConfigHash{old}))
|
|
|
|
{
|
|
|
|
$oConfigHashNew = {};
|
|
|
|
$$oConfigHash{old} = {}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$oConfigHashNew = dclone($$oConfigHash{old});
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
&log(DEBUG, (' ' x $iDepth) . 'process postgres config: ' . $$hCacheKey{file});
|
2015-11-22 16:44:01 -05:00
|
|
|
|
|
|
|
foreach my $oOption ($oConfig->nodeList('postgres-config-option'))
|
|
|
|
{
|
|
|
|
my $strKey = $oOption->paramGet('key');
|
|
|
|
my $strValue = $self->{oManifest}->variableReplace(trim($oOption->valueGet()));
|
|
|
|
|
|
|
|
if ($strValue eq '')
|
|
|
|
{
|
|
|
|
delete($$oConfigHashNew{$strKey});
|
|
|
|
|
|
|
|
&log(DEBUG, (' ' x ($iDepth + 1)) . "reset ${strKey}");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$$oConfigHashNew{$strKey} = $strValue;
|
|
|
|
&log(DEBUG, (' ' x ($iDepth + 1)) . "set ${strKey} = ${strValue}");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Generate config text
|
|
|
|
foreach my $strKey (sort(keys(%$oConfigHashNew)))
|
|
|
|
{
|
|
|
|
if (defined($strConfig))
|
|
|
|
{
|
|
|
|
$strConfig .= "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$strConfig .= "${strKey} = $$oConfigHashNew{$strKey}";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Save the conf file
|
|
|
|
if ($self->{bExe})
|
|
|
|
{
|
2017-06-09 17:51:41 -04:00
|
|
|
$self->{oManifest}->storage()->put($strLocalFile, $$oConfigHash{base} .
|
2015-11-22 16:44:01 -05:00
|
|
|
(defined($strConfig) ? "\n# pgBackRest Configuration\n${strConfig}\n" : ''));
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
$oHost->copyTo($strLocalFile, $$hCacheKey{file}, 'postgres:postgres', '640');
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
$$oConfigHash{old} = $oConfigHashNew;
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
my @stryConfig = undef;
|
|
|
|
|
|
|
|
if (trim($strConfig) ne '')
|
|
|
|
{
|
|
|
|
@stryConfig = split("\n", $strConfig);
|
|
|
|
}
|
|
|
|
|
|
|
|
$$hCacheValue{config} = \@stryConfig;
|
|
|
|
$self->cachePush($strCacheType, $hCacheKey, $hCacheValue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$strConfig = 'Config suppressed for testing';
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'strFile', value => $strFile, trace => true},
|
|
|
|
{name => 'strConfig', value => $strConfig, trace => true},
|
|
|
|
{name => 'bShow', value => $oConfig->paramTest('show', 'n') ? false : true, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
####################################################################################################################################
|
|
|
|
# hostKey
|
|
|
|
####################################################################################################################################
|
|
|
|
sub hostKey
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oHost,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->hostKey', \@_,
|
|
|
|
{name => 'oHost', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
my $hCacheKey =
|
|
|
|
{
|
|
|
|
name => $self->{oManifest}->variableReplace($oHost->paramGet('name')),
|
|
|
|
image => $self->{oManifest}->variableReplace($oHost->paramGet('image')),
|
|
|
|
};
|
|
|
|
|
2018-02-21 18:10:53 -05:00
|
|
|
if (defined($oHost->paramGet('id', false)))
|
|
|
|
{
|
|
|
|
$hCacheKey->{id} = $self->{oManifest}->variableReplace($oHost->paramGet('id'));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$hCacheKey->{id} = $hCacheKey->{name};
|
|
|
|
}
|
|
|
|
|
2017-02-10 10:22:05 -05:00
|
|
|
if (defined($oHost->paramGet('option', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{option} = $self->{oManifest}->variableReplace($oHost->paramGet('option'));
|
|
|
|
}
|
|
|
|
|
2018-05-25 13:42:09 -04:00
|
|
|
if (defined($oHost->paramGet('param', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{param} = $self->{oManifest}->variableReplace($oHost->paramGet('param'));
|
|
|
|
}
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if (defined($oHost->paramGet('os', false)))
|
|
|
|
{
|
|
|
|
$$hCacheKey{os} = $self->{oManifest}->variableReplace($oHost->paramGet('os'));
|
|
|
|
}
|
|
|
|
|
2018-05-25 13:42:09 -04:00
|
|
|
$$hCacheKey{'update-hosts'} = $oHost->paramTest('update-hosts', 'n') ? JSON::PP::false : JSON::PP::true;
|
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'hCacheKey', value => $hCacheKey, trace => true}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# cachePop
|
|
|
|
####################################################################################################################################
|
|
|
|
sub cachePop
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strCacheType,
|
|
|
|
$hCacheKey,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->hostKey', \@_,
|
|
|
|
{name => 'strCacheType', trace => true},
|
|
|
|
{name => 'hCacheKey', trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
my $bCacheHit = false;
|
|
|
|
my $oCacheValue = undef;
|
|
|
|
|
|
|
|
if ($self->{bCache})
|
|
|
|
{
|
|
|
|
my $oJSON = JSON::PP->new()->canonical()->allow_nonref();
|
|
|
|
# &log(WARN, "checking cache for\ncurrent key: " . $oJSON->encode($hCacheKey));
|
|
|
|
|
|
|
|
my $hCache = ${$self->{oSource}{hyCache}}[$self->{iCacheIdx}];
|
|
|
|
|
|
|
|
if (!defined($hCache))
|
|
|
|
{
|
2016-09-06 09:44:50 -04:00
|
|
|
confess &log(ERROR, 'unable to get index from cache', ERROR_FILE_INVALID);
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($$hCache{key}))
|
|
|
|
{
|
2016-09-06 09:44:50 -04:00
|
|
|
confess &log(ERROR, 'unable to get key from cache', ERROR_FILE_INVALID);
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!defined($$hCache{type}))
|
|
|
|
{
|
2016-09-06 09:44:50 -04:00
|
|
|
confess &log(ERROR, 'unable to get type from cache', ERROR_FILE_INVALID);
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($$hCache{type} ne $strCacheType)
|
|
|
|
{
|
2016-09-06 09:44:50 -04:00
|
|
|
confess &log(ERROR, 'types do not match, cache is invalid', ERROR_FILE_INVALID);
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($oJSON->encode($$hCache{key}) ne $oJSON->encode($hCacheKey))
|
|
|
|
{
|
|
|
|
confess &log(ERROR,
|
|
|
|
"keys at index $self->{iCacheIdx} do not match, cache is invalid." .
|
|
|
|
"\ncache key: " . $oJSON->encode($$hCache{key}) .
|
2016-09-06 09:44:50 -04:00
|
|
|
"\ncurrent key: " . $oJSON->encode($hCacheKey), ERROR_FILE_INVALID);
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
$bCacheHit = true;
|
|
|
|
$oCacheValue = $$hCache{value};
|
|
|
|
$self->{iCacheIdx}++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if ($self->{oManifest}{bCacheOnly})
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'Cache only operation forced by --cache-only option');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
{name => 'bCacheHit', value => $bCacheHit, trace => true},
|
|
|
|
{name => 'strCacheType', value => $strCacheType, trace => true},
|
|
|
|
{name => 'hCacheKey', value => $hCacheKey, trace => true},
|
|
|
|
{name => 'oCacheValue', value => $oCacheValue, trace => true},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
####################################################################################################################################
|
|
|
|
# cachePush
|
|
|
|
####################################################################################################################################
|
|
|
|
sub cachePush
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$strType,
|
|
|
|
$hCacheKey,
|
|
|
|
$oCacheValue,
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
|
|
|
__PACKAGE__ . '->hostKey', \@_,
|
|
|
|
{name => 'strType', trace => true},
|
|
|
|
{name => 'hCacheKey', trace => true},
|
|
|
|
{name => 'oCacheValue', required => false, trace => true},
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($self->{bCache})
|
|
|
|
{
|
|
|
|
confess &log(ASSERT, "cachePush should not be called when cache is already present");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Create the cache entry
|
|
|
|
my $hCache =
|
|
|
|
{
|
|
|
|
key => $hCacheKey,
|
|
|
|
type => $strType,
|
|
|
|
};
|
|
|
|
|
|
|
|
if (defined($oCacheValue))
|
|
|
|
{
|
|
|
|
$$hCache{value} = $oCacheValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
push @{$self->{oSource}{hyCache}}, $hCache;
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn($strOperation);
|
|
|
|
}
|
|
|
|
|
2015-11-22 16:44:01 -05:00
|
|
|
####################################################################################################################################
|
|
|
|
# sectionChildProcesss
|
|
|
|
####################################################################################################################################
|
|
|
|
sub sectionChildProcess
|
|
|
|
{
|
|
|
|
my $self = shift;
|
|
|
|
|
|
|
|
# Assign function parameters, defaults, and log debug info
|
|
|
|
my
|
|
|
|
(
|
|
|
|
$strOperation,
|
|
|
|
$oSection,
|
|
|
|
$oChild,
|
|
|
|
$iDepth
|
|
|
|
) =
|
|
|
|
logDebugParam
|
|
|
|
(
|
2016-05-26 09:09:42 -04:00
|
|
|
__PACKAGE__ . '->sectionChildProcess', \@_,
|
2015-11-22 16:44:01 -05:00
|
|
|
{name => 'oSection'},
|
|
|
|
{name => 'oChild'},
|
|
|
|
{name => 'iDepth'}
|
|
|
|
);
|
|
|
|
|
|
|
|
&log(DEBUG, (' ' x ($iDepth + 1)) . 'process child: ' . $oChild->nameGet());
|
|
|
|
|
|
|
|
# Execute a command
|
|
|
|
if ($oChild->nameGet() eq 'host-add')
|
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($self->{bExe} && $self->isRequired($oSection))
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2016-06-02 09:32:56 -04:00
|
|
|
my ($bCacheHit, $strCacheType, $hCacheKey, $hCacheValue) = $self->cachePop('host', $self->hostKey($oChild));
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
if ($bCacheHit)
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
2018-02-21 18:10:53 -05:00
|
|
|
$self->{oManifest}->variableSet('host-' . $hCacheKey->{id} . '-ip', $hCacheValue->{ip}, true);
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
2016-06-02 09:32:56 -04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (defined($self->{host}{$$hCacheKey{name}}))
|
|
|
|
{
|
|
|
|
confess &log(ERROR, 'cannot add host ${strName} because the host already exists');
|
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-24 08:12:58 -04:00
|
|
|
executeTest("rm -rf ~/data/$$hCacheKey{name}");
|
|
|
|
executeTest("mkdir -p ~/data/$$hCacheKey{name}/etc");
|
|
|
|
|
2018-11-29 14:45:15 -05:00
|
|
|
my $strHost = $hCacheKey->{name};
|
|
|
|
my $strImage = $hCacheKey->{image};
|
|
|
|
|
|
|
|
# Determine if a pre-built image should be created
|
|
|
|
if (defined($self->preExecute($strHost)))
|
|
|
|
{
|
|
|
|
my $strPreImage = "${strImage}-${strHost}";
|
|
|
|
my $strFrom = $strImage;
|
|
|
|
|
|
|
|
&log(INFO, "Build vm '${strPreImage}' from '${strFrom}'");
|
|
|
|
|
|
|
|
my $strCommandList;
|
|
|
|
|
|
|
|
# Add all pre commands
|
|
|
|
foreach my $oExecute ($self->preExecute($strHost))
|
|
|
|
{
|
|
|
|
my $hExecuteKey = $self->executeKey($strHost, $oExecute);
|
|
|
|
my $strCommand =
|
|
|
|
join("\n", @{$hExecuteKey->{cmd}}) .
|
|
|
|
(defined($hExecuteKey->{'cmd-extra'}) ? ' ' . $hExecuteKey->{'cmd-extra'} : '');
|
|
|
|
|
|
|
|
if (defined($strCommandList))
|
|
|
|
{
|
|
|
|
$strCommandList .= "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
$strCommandList .= "RUN ${strCommand}";
|
|
|
|
|
|
|
|
&log(DETAIL, " Pre command $strCommand");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Build container
|
|
|
|
my $strDockerfile = $self->{oManifest}{strDocPath} . "/output/doc-host.dockerfile";
|
|
|
|
|
|
|
|
$self->{oManifest}{oStorage}->put(
|
|
|
|
$strDockerfile,
|
|
|
|
"FROM ${strFrom}\n\n" . trim($self->{oManifest}->variableReplace($strCommandList)) . "\n");
|
|
|
|
executeTest("docker build -f ${strDockerfile} -t ${strPreImage} " . $self->{oManifest}{oStorage}->pathGet());
|
|
|
|
|
|
|
|
# Use the pre-built image
|
|
|
|
$strImage = $strPreImage;
|
|
|
|
}
|
|
|
|
|
2016-06-24 08:12:58 -04:00
|
|
|
my $oHost = new pgBackRestTest::Common::HostTest(
|
2018-11-29 14:45:15 -05:00
|
|
|
$$hCacheKey{name}, "doc-$$hCacheKey{name}", $strImage,
|
2017-06-27 22:27:48 -04:00
|
|
|
$self->{oManifest}->variableReplace($oChild->paramGet('user')), $$hCacheKey{os},
|
|
|
|
defined($oChild->paramGet('mount', false)) ?
|
|
|
|
[$self->{oManifest}->variableReplace($oChild->paramGet('mount'))] : undef,
|
2018-05-25 13:42:09 -04:00
|
|
|
$$hCacheKey{option}, $$hCacheKey{param});
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
$self->{host}{$$hCacheKey{name}} = $oHost;
|
2018-02-21 18:10:53 -05:00
|
|
|
$self->{oManifest}->variableSet('host-' . $hCacheKey->{id} . '-ip', $oHost->{strIP}, true);
|
2016-06-02 09:32:56 -04:00
|
|
|
$$hCacheValue{ip} = $oHost->{strIP};
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2017-02-16 17:18:48 -05:00
|
|
|
# Add to the host group
|
|
|
|
my $oHostGroup = hostGroupGet();
|
2018-05-25 13:42:09 -04:00
|
|
|
$oHostGroup->hostAdd($oHost, {bUpdateHosts => $$hCacheKey{'update-hosts'}});
|
2017-02-16 17:18:48 -05:00
|
|
|
|
|
|
|
# Execute initialize commands
|
2016-11-07 17:06:35 +02:00
|
|
|
foreach my $oExecute ($oChild->nodeList('execute', false))
|
2016-06-02 09:32:56 -04:00
|
|
|
{
|
2018-06-12 13:43:15 -04:00
|
|
|
$self->execute(
|
|
|
|
$oSection, $$hCacheKey{name}, $oExecute, {iIndent => $iDepth + 1, bCache => false, bShow => false});
|
2016-06-02 09:32:56 -04:00
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
|
2016-06-02 09:32:56 -04:00
|
|
|
$self->cachePush($strCacheType, $hCacheKey, $hCacheValue);
|
|
|
|
}
|
2015-11-22 16:44:01 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
# Skip children that have already been processed and error on others
|
2016-05-26 10:34:10 -04:00
|
|
|
elsif ($oChild->nameGet() ne 'title')
|
2015-11-22 16:44:01 -05:00
|
|
|
{
|
|
|
|
confess &log(ASSERT, 'unable to process child type ' . $oChild->nameGet());
|
|
|
|
}
|
|
|
|
|
|
|
|
# Return from function and log return values if any
|
|
|
|
return logDebugReturn
|
|
|
|
(
|
|
|
|
$strOperation
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|