|
|
|
#!/usr/bin/perl
|
|
|
|
#
|
|
|
|
# This adds a node (or gets a node to join to a new/rebuilt peer or added DR) to an Anvil!.
|
|
|
|
#
|
|
|
|
# Exit codes;
|
|
|
|
# 0 = Normal exit.
|
|
|
|
# 1 = Failed to connect to any database.
|
|
|
|
# 2 = Failed to load/parse the manifest.
|
|
|
|
# 3 = Failed to change the host name.
|
|
|
|
# 4 = Failed to reconnect to any database after the network was reconfigured
|
|
|
|
# 5 = Problem parsing job data or loading manifest or anvil data using job data.
|
|
|
|
# 6 = A pcs call failed.
|
|
|
|
# 7 = No job was found to run.
|
|
|
|
#
|
|
|
|
# TODO:
|
|
|
|
# - Check to see if this is a cluster node and/or running VMs, and if so, refuse to run.
|
|
|
|
#
|
|
|
|
|
|
|
|
use strict;
|
|
|
|
use warnings;
|
|
|
|
use Anvil::Tools;
|
|
|
|
use Data::Dumper;
|
|
|
|
use String::ShellQuote;
|
|
|
|
|
|
|
|
my $THIS_FILE = ($0 =~ /^.*\/(.*)$/)[0];
|
|
|
|
my $running_directory = ($0 =~ /^(.*?)\/$THIS_FILE$/)[0];
|
|
|
|
if (($running_directory =~ /^\./) && ($ENV{PWD}))
|
|
|
|
{
|
|
|
|
$running_directory =~ s/^\./$ENV{PWD}/;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Turn off buffering so that the pinwheel will display while waiting for the SSH call(s) to complete.
|
|
|
|
$| = 1;
|
|
|
|
|
|
|
|
my $anvil = Anvil::Tools->new();
|
|
|
|
|
|
|
|
# Read switches (target ([user@]host[:port]) and the file with the target's password. If the password is
|
|
|
|
# passed directly, it will be used. Otherwise, the password will be read from the database.
|
|
|
|
$anvil->Get->switches;
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 2, key => "log_0115", variables => { program => $THIS_FILE }});
|
|
|
|
|
|
|
|
$anvil->Database->connect();
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 2, key => "log_0132"});
|
|
|
|
if (not $anvil->data->{sys}{database}{connections})
|
|
|
|
{
|
|
|
|
# No databases, update the job, sleep for a bit and then exit. The daemon will pick it up and try
|
|
|
|
# again after we exit.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0129"});
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 1});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Get the job details
|
|
|
|
load_job($anvil);
|
|
|
|
|
|
|
|
# Update the user passwords
|
|
|
|
update_passwords($anvil);
|
|
|
|
|
|
|
|
# Check if we need to change any IPs or our hostname.
|
|
|
|
check_local_network($anvil);
|
|
|
|
|
|
|
|
# (wait for out peer and) Configure pacemaker
|
|
|
|
configure_pacemaker($anvil);
|
|
|
|
|
|
|
|
# Configure DRBD
|
|
|
|
configure_drbd($anvil);
|
|
|
|
|
|
|
|
update_progress($anvil, 100, "job_0128");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0128"});
|
|
|
|
$anvil->nice_exit({exit_code => 0});
|
|
|
|
|
|
|
|
#############################################################################################################
|
|
|
|
# Functions #
|
|
|
|
#############################################################################################################
|
|
|
|
|
|
|
|
# Configure DRBD
|
|
|
|
sub configure_drbd
|
|
|
|
{
|
|
|
|
my ($anvil) = @_;
|
|
|
|
|
|
|
|
### TODO: See if there is a hardware RAID controller and, if so, auto-enable
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 1), "job_0126");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0126"});
|
|
|
|
my $updated = $anvil->DRBD->update_global_common({
|
|
|
|
usage_count => $anvil->data->{sys}{privacy}{strong} ? 0 : 1,
|
|
|
|
});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { updated => $updated }});
|
|
|
|
if ($updated)
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 1), "job_0127");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0127"});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 1), "job_0128");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0128"});
|
|
|
|
}
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Update the passwords for user accounts.
|
|
|
|
sub update_passwords
|
|
|
|
{
|
|
|
|
my ($anvil) = @_;
|
|
|
|
|
|
|
|
my $machine = $anvil->data->{sys}{machine};
|
|
|
|
my $manifest_uuid = $anvil->data->{sys}{manifest_uuid};
|
|
|
|
my $anvil_name = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{name};
|
|
|
|
my $anvil_uuid = $anvil->data->{sys}{anvil_uuid};
|
|
|
|
my $new_password = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_password};
|
|
|
|
my $users = $machine =~ /dr\d+/ ? ["root", "admin"] : ["root", "admin", "hacluster"];
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
machine => $machine,
|
|
|
|
manifest_uuid => $manifest_uuid,
|
|
|
|
anvil_name => $anvil_name,
|
|
|
|
anvil_uuid => $anvil_uuid,
|
|
|
|
new_password => $anvil->Log->is_secure($new_password),
|
|
|
|
}});
|
|
|
|
|
|
|
|
foreach my $user (@{$users})
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { user => $user }});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0093,!!user!".$user."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0093", variables => { user => $user }});
|
|
|
|
my ($return_code) = $anvil->System->change_shell_user_password({
|
|
|
|
debug => 3,
|
|
|
|
user => $user,
|
|
|
|
new_password => $new_password,
|
|
|
|
});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
}
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
# (wait for out peer and) Configure pacemaker. If this is a DR host, this is skipped.
|
|
|
|
sub configure_pacemaker
|
|
|
|
{
|
|
|
|
my ($anvil) = @_;
|
|
|
|
|
|
|
|
my $machine = $anvil->data->{sys}{machine};
|
|
|
|
my $manifest_uuid = $anvil->data->{sys}{manifest_uuid};
|
|
|
|
|
|
|
|
### TODO: Move these to variables in the 'sys' hash
|
|
|
|
my $anvil_name = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{name};
|
|
|
|
my $anvil_uuid = $anvil->data->{sys}{anvil_uuid};
|
|
|
|
my $host_name = $anvil->Get->host_name;
|
|
|
|
my $new_password = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_password};
|
|
|
|
my $node1_host_uuid = $anvil->data->{sys}{node1_host_uuid} = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_node1_host_uuid};
|
|
|
|
my $node1_host_name = $anvil->data->{hosts}{host_uuid}{$node1_host_uuid}{host_name};
|
|
|
|
$node1_host_name =~ s/\..*$//;
|
|
|
|
my $node2_host_uuid = $anvil->data->{sys}{node2_host_uuid} = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_node2_host_uuid};
|
|
|
|
my $node2_host_name = $anvil->data->{hosts}{host_uuid}{$node2_host_uuid}{host_name};
|
|
|
|
$node2_host_name =~ s/\..*$//;
|
|
|
|
my $peer_host_name = $anvil->Get->host_uuid() eq $node1_host_uuid ? $node2_host_name : $node1_host_name;
|
|
|
|
my $peer_host_uuid = $anvil->Get->host_uuid() eq $node1_host_uuid ? $node2_host_uuid : $node1_host_uuid;
|
|
|
|
my $escaped_password = shell_quote($new_password);
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
machine => $machine,
|
|
|
|
anvil_uuid => $anvil_uuid,
|
|
|
|
anvil_name => $anvil_name,
|
|
|
|
host_name => $host_name,
|
|
|
|
manifest_uuid => $manifest_uuid,
|
|
|
|
node1_host_uuid => $node1_host_uuid,
|
|
|
|
node1_host_name => $node1_host_name,
|
|
|
|
node2_host_uuid => $node2_host_uuid,
|
|
|
|
node2_host_name => $node2_host_name,
|
|
|
|
peer_host_name => $peer_host_name,
|
|
|
|
peer_host_uuid => $peer_host_uuid,
|
|
|
|
new_password => $anvil->Log->is_secure($new_password),
|
|
|
|
escaped_password => $anvil->Log->is_secure($escaped_password),
|
|
|
|
}});
|
|
|
|
|
|
|
|
# If this is a DR box, we don't use pacemaker.
|
|
|
|
if ($machine =~ /dr\d+/)
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 10), "job_0096");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0096"});
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
### Run on both nodes.
|
|
|
|
# Enable pcsd and start the pcsd daemon.
|
|
|
|
my ($return_code) = $anvil->System->enable_daemon({daemon => "pcsd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
($return_code) = $anvil->System->start_daemon({daemon => "pcsd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0094,!!daemon!pcsd.service!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0094", variables => { daemon => "pcsd.service" }});
|
|
|
|
|
|
|
|
# Enable libvirtd and start the libvirtd daemon.
|
|
|
|
($return_code) = $anvil->System->enable_daemon({daemon => "libvirtd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
($return_code) = $anvil->System->start_daemon({daemon => "libvirtd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0094,!!daemon!libvirtd.service!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0094", variables => { daemon => "libvirtd.service" }});
|
|
|
|
|
|
|
|
# Disabled and stop the drbd daemon.
|
|
|
|
($return_code) = $anvil->System->disable_daemon({daemon => "drbd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
($return_code) = $anvil->System->stop_daemon({daemon => "drbd.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0095,!!daemon!drbd.service!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0095", variables => { daemon => "drbd.service" }});
|
|
|
|
|
|
|
|
# Disabled and stop the ksm and ksmtuned daemon.
|
|
|
|
($return_code) = $anvil->System->disable_daemon({daemon => "ksm.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
($return_code) = $anvil->System->stop_daemon({daemon => "ksmtuned.service"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { return_code => $return_code }});
|
|
|
|
$return_code = undef;
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0095,!!daemon!ksm.service!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0095", variables => { daemon => "drbd.service" }});
|
|
|
|
|
|
|
|
# If there is no corosync.conf, see if the peer has it. If so, copy it. If not, we'll initialize the
|
|
|
|
# cluster shortly.
|
|
|
|
if (not -e $anvil->data->{path}{configs}{'corosync.conf'})
|
|
|
|
{
|
|
|
|
my $corosync_conf = $anvil->Storage->read_file({
|
|
|
|
file => $anvil->data->{path}{configs}{'corosync.conf'},
|
|
|
|
target => $peer_host_name,
|
|
|
|
});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { corosync_conf => $corosync_conf }});
|
|
|
|
if ($corosync_conf ne "!!error!!")
|
|
|
|
{
|
|
|
|
# Write the file out.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0100"});
|
|
|
|
$anvil->Storage->write_file({
|
|
|
|
body => $corosync_conf,
|
|
|
|
file => $anvil->data->{path}{configs}{'corosync.conf'},
|
|
|
|
user => "root",
|
|
|
|
group => "root",
|
|
|
|
mode => "0644",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Node 1 initializes, node 2 waits.
|
|
|
|
if ($machine eq "node2")
|
|
|
|
{
|
|
|
|
my $start_time = 0;
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0103");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0103"});
|
|
|
|
|
|
|
|
# We loop until the peer finishes or the peer's job hit's 100.
|
|
|
|
my $tried_starting = 0;
|
|
|
|
my $both_online = 0;
|
|
|
|
until($both_online)
|
|
|
|
{
|
|
|
|
if (-e $anvil->data->{path}{configs}{'corosync.conf'})
|
|
|
|
{
|
|
|
|
if (not $start_time)
|
|
|
|
{
|
|
|
|
# Corosync is configured, we'll wait up to two minutes and then try
|
|
|
|
# joining the cluster ourselves.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0108"});
|
|
|
|
$start_time = time + 120;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { start_time => $start_time }});
|
|
|
|
}
|
|
|
|
elsif ((time > $start_time) && (not $tried_starting))
|
|
|
|
{
|
|
|
|
# We've waited a minute, time to try starting the cluster.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0106"});
|
|
|
|
my $cluster_started = $anvil->Cluster->start_cluster({debug => 2, all => 1});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { cluster_started => $cluster_started }});
|
|
|
|
|
|
|
|
# Mark that weve tried to start.
|
|
|
|
$tried_starting = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { tried_starting => $tried_starting }});
|
|
|
|
}
|
|
|
|
|
|
|
|
my $problem = $anvil->Cluster->parse_cib({debug => 3});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { problem => $problem }});
|
|
|
|
if (not $problem)
|
|
|
|
{
|
|
|
|
# See if both nodes are online.
|
|
|
|
my $node1_ready = $anvil->Cluster->check_node_status({node_name => $node1_host_name});
|
|
|
|
my $node2_ready = $anvil->Cluster->check_node_status({node_name => $node2_host_name});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
node1_ready => $node1_ready,
|
|
|
|
node2_ready => $node2_ready,
|
|
|
|
}});
|
|
|
|
if (($node1_ready) && ($node2_ready))
|
|
|
|
{
|
|
|
|
$both_online = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { both_online => $both_online }});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0104");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0104"});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Not online yet, wait a bit.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0105", variables => {
|
|
|
|
node1_name => $node1_host_name,
|
|
|
|
node1_ready => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{in_ccm},
|
|
|
|
node1_in_ccm => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{crmd},
|
|
|
|
node1_crmd => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{'join'},
|
|
|
|
node1_join => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{ready},
|
|
|
|
node2_name => $node2_host_name,
|
|
|
|
node2_ready => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{in_ccm},
|
|
|
|
node2_in_ccm => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{crmd},
|
|
|
|
node2_crmd => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{'join'},
|
|
|
|
node2_join => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{ready},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# corosync.conf doesn't exist yet.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0107"});
|
|
|
|
}
|
|
|
|
sleep 5 if not $both_online;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# We're node 1, proceed with cluster setup.
|
|
|
|
my $waiting = 1;
|
|
|
|
my $warning_printed = 0;
|
|
|
|
while($waiting)
|
|
|
|
{
|
|
|
|
# Try to authenticate against the peer.
|
|
|
|
my $auth_shell_call = $anvil->data->{path}{exe}{pcs}." host auth ".$node1_host_name." ".$node2_host_name." -u hacluster -p ".$escaped_password;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, secure => 1, list => { auth_shell_call => $auth_shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, secure => 1, shell_call => $auth_shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
if (not $warning_printed)
|
|
|
|
{
|
|
|
|
# Update the job
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0097");
|
|
|
|
$warning_printed = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { warning_printed => $warning_printed }});
|
|
|
|
}
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0097"});
|
|
|
|
sleep 5;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# We're good.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0098");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0098"});
|
|
|
|
$waiting = 0;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { waiting => $waiting }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# If there is no corosync.conf, see if the peer has it. If so, copy it. If not, initialize
|
|
|
|
# the cluster.
|
|
|
|
if (not -e $anvil->data->{path}{configs}{'corosync.conf'})
|
|
|
|
{
|
|
|
|
# There's no cluster yet, initialize it.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0099,!!anvil_name!".$anvil_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0099", variables => { anvil_name => $anvil_name }});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." cluster setup ".$anvil_name." ".$node1_host_name." ".$node2_host_name;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong
|
|
|
|
update_progress($anvil, 100, "job_0101,!!error!".$output."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 5});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# If we can parse the CIB, then pcsd is running.
|
|
|
|
my $problem = $anvil->Cluster->parse_cib({debug => 3});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { problem => $problem }});
|
|
|
|
if ($problem)
|
|
|
|
{
|
|
|
|
# Start the cluster.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0102");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0102"});
|
|
|
|
|
|
|
|
my $cluster_started = $anvil->Cluster->start_cluster({debug => 3, all => 1});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { cluster_started => $cluster_started }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Now wait for both nodes to come online.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0109");
|
|
|
|
my $both_online = 0;
|
|
|
|
my $start_again = time + 60;
|
|
|
|
until ($both_online)
|
|
|
|
{
|
|
|
|
### TODO: If we're waiting more that five minutes, call 'pcs cluster start --all' again.
|
|
|
|
my $problem = $anvil->Cluster->parse_cib({debug => 3});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { problem => $problem }});
|
|
|
|
if (not $problem)
|
|
|
|
{
|
|
|
|
# See if both nodes are online.
|
|
|
|
my $node1_ready = $anvil->Cluster->check_node_status({node_name => $node1_host_name});
|
|
|
|
my $node2_ready = $anvil->Cluster->check_node_status({node_name => $node2_host_name});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
node1_ready => $node1_ready,
|
|
|
|
node2_ready => $node2_ready,
|
|
|
|
}});
|
|
|
|
if (($node1_ready) && ($node2_ready))
|
|
|
|
{
|
|
|
|
$both_online = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { both_online => $both_online }});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0104");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0104"});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Not online yet, wait a bit.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0105", variables => {
|
|
|
|
node1_name => $node1_host_name,
|
|
|
|
node1_in_ccm => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{in_ccm},
|
|
|
|
node1_crmd => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{crmd},
|
|
|
|
node1_join => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{'join'},
|
|
|
|
node1_ready => $anvil->data->{cib}{parsed}{data}{node}{$node1_host_name}{node_state}{ready},
|
|
|
|
node2_name => $node2_host_name,
|
|
|
|
node2_in_ccm => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{in_ccm},
|
|
|
|
node2_crmd => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{crmd},
|
|
|
|
node2_join => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{'join'},
|
|
|
|
node2_ready => $anvil->data->{cib}{parsed}{data}{node}{$node2_host_name}{node_state}{ready},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (time > $start_again)
|
|
|
|
{
|
|
|
|
# Call cluster start again.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0272"});
|
|
|
|
$start_again = time + 60;
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." cluster start --all";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
start_again => $start_again,
|
|
|
|
shell_call => $shell_call,
|
|
|
|
}});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
sleep 5 if not $both_online;
|
|
|
|
}
|
|
|
|
|
|
|
|
### Setup fencing!
|
|
|
|
$anvil->Striker->get_fence_data();
|
|
|
|
|
|
|
|
# IPMI first, if applicable.
|
|
|
|
my $something_changed = {};
|
|
|
|
my $fence_order = {};
|
|
|
|
my $node1_use_delay = 0;
|
|
|
|
my $node2_use_delay = 0;
|
|
|
|
my $fence_devices = {};
|
|
|
|
foreach my $node ("node1", "node2")
|
|
|
|
{
|
|
|
|
my $node_name = $node eq "node1" ? $node1_host_name : $node2_host_name;
|
|
|
|
my $host_uuid = $node eq "node1" ? $node1_host_uuid : $node2_host_uuid;
|
|
|
|
my $host_ipmi = $anvil->data->{hosts}{host_uuid}{$host_uuid}{host_ipmi};
|
|
|
|
my $ipmi_stonith_name = "ipmilan_".$node;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
node_name => $node_name,
|
|
|
|
host_uuid => $host_uuid,
|
|
|
|
host_ipmi => $host_ipmi =~ /passw/ ? $anvil->Log->is_secure($host_ipmi) : $host_ipmi,
|
|
|
|
ipmi_stonith_name => $ipmi_stonith_name,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0115,!!node!".$node_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0115", variables => { node => $node_name }});
|
|
|
|
|
|
|
|
# This will store the fence level order. If something changes
|
|
|
|
$fence_order->{$node_name} = [];
|
|
|
|
|
|
|
|
# This will switch to '1' if something changed, triggering a reconfig of the fencing levels.
|
|
|
|
$something_changed->{$node_name} = 0;
|
|
|
|
|
|
|
|
# Does this stonith method already exist?
|
|
|
|
my $create_entry = 0;
|
|
|
|
my $delete_old = 0;
|
|
|
|
my $pcs_add_command = "";
|
|
|
|
if ($host_ipmi)
|
|
|
|
{
|
|
|
|
push @{$fence_order->{$node_name}}, "fence_ipmilan";
|
|
|
|
$fence_devices->{$node_name}{fence_ipmilan} = [$ipmi_stonith_name];
|
|
|
|
|
|
|
|
# The --action switch needs to be 'pcmk_off_action' in pcs, so we convert it here.
|
|
|
|
$host_ipmi =~ s/--action status//;
|
|
|
|
$host_ipmi =~ s/--action/--pcmk_off_action/;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
host_ipmi => $host_ipmi =~ /passw/ ? $anvil->Log->is_secure($host_ipmi) : $host_ipmi,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# We have IPMI, so we also want fence_delay for this node.
|
|
|
|
if ($node eq "node1")
|
|
|
|
{
|
|
|
|
$node1_use_delay = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { node1_use_delay => $node1_use_delay }});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$node2_use_delay = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { node2_use_delay => $node2_use_delay }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# If we're here, break up the command and turn it into the pcs call.
|
|
|
|
my $old_switches = {};
|
|
|
|
my ($fence_agent, $arguments) = ($host_ipmi =~ /^\/.*\/(.*?)\s+(.*)$/);
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
fence_agent => $fence_agent,
|
|
|
|
arguments => $arguments =~ /passw/ ? $anvil->Log->is_secure($arguments) : $arguments,
|
|
|
|
}});
|
|
|
|
|
|
|
|
$pcs_add_command = $anvil->data->{path}{exe}{pcs}." stonith create ".$ipmi_stonith_name." ".$fence_agent." pcmk_host_list=\"".$node_name."\" ";
|
|
|
|
my $switches = $anvil->System->parse_arguments({arguments => $arguments});
|
|
|
|
foreach my $switch (sort {$a cmp $b} keys %{$switches})
|
|
|
|
{
|
|
|
|
# Ignore 'delay', we handle that in Cluster->set_delay(); Also,
|
|
|
|
# convert '#!SET!#' to 'true'.
|
|
|
|
my $value = $switches->{$switch};
|
|
|
|
$value =~ s/"/\\"/g;
|
|
|
|
$value =~ s/#!SET!#/true/g;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
switch => $switch,
|
|
|
|
value => $value,
|
|
|
|
}});
|
|
|
|
next if $anvil->data->{fence_data}{$fence_agent}{switch}{$switch}{name} eq "delay";
|
|
|
|
next if $anvil->data->{fence_data}{$fence_agent}{switch}{$switch}{name} eq "action";
|
|
|
|
|
|
|
|
# Find the argument=value version.
|
|
|
|
my $argument = $anvil->data->{fence_data}{$fence_agent}{switch}{$switch}{name};
|
|
|
|
$pcs_add_command .= $argument."=\"".$value."\" ";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
argument => $argument,
|
|
|
|
value => $argument =~ /passw/ ? $anvil->Log->is_secure($value) : $value,
|
|
|
|
pcs_add_command => $pcs_add_command =~ /passw/ ? $anvil->Log->is_secure($pcs_add_command) : $pcs_add_command,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Store this to see if it's different from what's already in the CIB.
|
|
|
|
$old_switches->{$argument} = $value;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
"old_switches->{$argument}" => $old_switches->{$argument},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
$pcs_add_command .= "op monitor interval=\"60\"";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
pcs_add_command => $pcs_add_command =~ /passw/ ? $anvil->Log->is_secure($pcs_add_command) : $pcs_add_command,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# If there's an entry in the CIB, so if it's different somehow
|
|
|
|
if (exists $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$ipmi_stonith_name})
|
|
|
|
{
|
|
|
|
foreach my $argument (sort {$a cmp $b} keys %{$anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$ipmi_stonith_name}{argument}})
|
|
|
|
{
|
|
|
|
next if $argument eq "delay";
|
|
|
|
next if $argument eq "action";
|
|
|
|
my $old_entry = $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$ipmi_stonith_name}{argument}{$argument}{value};
|
|
|
|
my $new_entry = exists $old_switches->{$argument} ? $old_switches->{$argument} : "";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's1:argument' => $argument,
|
|
|
|
's2:old_entry' => $old_entry,
|
|
|
|
's3:new_entry' => $new_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if ($old_entry ne $new_entry)
|
|
|
|
{
|
|
|
|
# Changed, delete and recreate.
|
|
|
|
$delete_old = 1;
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0117");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0117"});
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Delete the old switch.
|
|
|
|
delete $old_switches->{$argument};
|
|
|
|
}
|
|
|
|
|
|
|
|
# Are there any old switches left?
|
|
|
|
my $old_switch_count = keys %{$old_switches};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
old_switch_count => $old_switch_count,
|
|
|
|
}});
|
|
|
|
if ((not $delete_old) && ($old_switch_count))
|
|
|
|
{
|
|
|
|
# Delete and recreate.
|
|
|
|
$delete_old = 1;
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0117");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0117"});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# No existing entry, add a new one.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0116");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0116"});
|
|
|
|
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { create_entry => $create_entry }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif (exists $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$ipmi_stonith_name})
|
|
|
|
{
|
|
|
|
# There was an existing fence config, but there's no entry in 'host_ipmi'.
|
|
|
|
# Remove the stonith entry.
|
|
|
|
$delete_old = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { delete_old => $delete_old }});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0118");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0118"});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Process the IPMI entry.
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
if ($delete_old)
|
|
|
|
{
|
|
|
|
# Delete
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0119,!!device!".$ipmi_stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0119", variables => { device => $ipmi_stonith_name }});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." stonith delete ".$ipmi_stonith_name;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
|
|
|
|
$something_changed->{$node_name} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "something_changed->{$node_name}" => $something_changed->{$node_name} }});
|
|
|
|
}
|
|
|
|
if ($create_entry)
|
|
|
|
{
|
|
|
|
# Create.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0120,!!device!".$ipmi_stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0120", variables => { device => $ipmi_stonith_name }});
|
|
|
|
|
|
|
|
my $shell_call = $pcs_add_command;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
|
|
|
|
$something_changed->{$node_name} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { "something_changed->{$node_name}" => $something_changed->{$node_name} }});
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
### Now any other fence devices.
|
|
|
|
foreach my $device (sort {$a cmp $b} keys %{$anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$node}{fence}})
|
|
|
|
{
|
|
|
|
my $delete_old = 0;
|
|
|
|
my $create_entry = 0;
|
|
|
|
my $old_switches = {};
|
|
|
|
my $fence_uuid = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{fences}{$device}{uuid};
|
|
|
|
my $fence_name = $anvil->data->{fences}{fence_uuid}{$fence_uuid}{fence_name};
|
|
|
|
my $fence_arguments = $anvil->data->{fences}{fence_uuid}{$fence_uuid}{fence_arguments};
|
|
|
|
my $fence_agent = $anvil->data->{fences}{fence_uuid}{$fence_uuid}{fence_agent};
|
|
|
|
my $stonith_name = ($fence_agent =~ /^fence_(.*)$/)[0]."_".$node."_".$fence_name;
|
|
|
|
my $port = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$node}{fence}{$device}{port};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
device => $device,
|
|
|
|
fence_uuid => $fence_uuid,
|
|
|
|
fence_name => $fence_name,
|
|
|
|
fence_arguments => $fence_arguments =~ /passw/ ? $anvil->Log->is_secure($fence_arguments) : $fence_arguments,
|
|
|
|
stonith_name => $stonith_name,
|
|
|
|
port => $port,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# We use this to tell if there are two or more entries per agent. If there
|
|
|
|
# are, we link them later when setting up the fence levels.
|
|
|
|
if (not exists $fence_devices->{$node_name}{$fence_agent})
|
|
|
|
{
|
|
|
|
push @{$fence_order->{$node_name}}, $fence_agent;
|
|
|
|
$fence_devices->{$node_name}{$fence_agent} = [];
|
|
|
|
}
|
|
|
|
push @{$fence_devices->{$node_name}{$fence_agent}}, $stonith_name;
|
|
|
|
|
|
|
|
# Fence arguments use 'action', but pcs deprecated it in favour of 'pcmk_off_action', so rename it.
|
|
|
|
$fence_arguments =~ s/action=/pcmk_off_action=/;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
fence_arguments => $fence_arguments =~ /passw/ ? $anvil->Log->is_secure($fence_arguments) : $fence_arguments,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Build the pcs command
|
|
|
|
my $pcs_add_command = $anvil->data->{path}{exe}{pcs}." stonith create ".$stonith_name." ".$fence_agent." pcmk_host_list=\"".$node_name."\" ".$fence_arguments." ";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
pcs_add_command => $pcs_add_command =~ /passw/ ? $anvil->Log->is_secure($pcs_add_command) : $pcs_add_command,
|
|
|
|
}});
|
|
|
|
while ($fence_arguments =~ /=/)
|
|
|
|
{
|
|
|
|
# Ignore 'delay', we handle that in Cluster->set_delay();
|
|
|
|
my $pair = ($fence_arguments =~ /(\S*?=".*?")/)[0];
|
|
|
|
$fence_arguments =~ s/$pair//;
|
|
|
|
$fence_arguments =~ s/^\s+//;
|
|
|
|
$fence_arguments =~ s/\s+$//;
|
|
|
|
my ($argument, $value) = ($pair =~ /(.*)="(.*)"/);
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:fence_arguments' => $fence_arguments,
|
|
|
|
's2:pair' => $pair =~ /passw/ ? $anvil->Log->is_secure($pair) : $pair,
|
|
|
|
's3:argument' => $argument,
|
|
|
|
's4:value' => $argument =~ /passw/ ? $anvil->Log->is_secure($value) : $value,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Ignore 'delay', we handle that in Cluster->set_delay();
|
|
|
|
if (($argument ne "pcmk_off_action") &&
|
|
|
|
(exists $anvil->data->{fence_data}{$fence_agent}{switch}{$argument}{name}) &&
|
|
|
|
($anvil->data->{fence_data}{$fence_agent}{switch}{$argument}{name} eq "delay"))
|
|
|
|
{
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Store this to see if it's different from what's already in the CIB.
|
|
|
|
$old_switches->{$argument} = $value;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"old_switches->{$argument}" => $old_switches->{$argument},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
if ($port)
|
|
|
|
{
|
|
|
|
$port =~ s/"/\\"/g;
|
|
|
|
$pcs_add_command .= "port=\"".$port."\" ";
|
|
|
|
$old_switches->{port} = $port;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
pcs_add_command => $pcs_add_command =~ /passw/ ? $anvil->Log->is_secure($pcs_add_command) : $pcs_add_command,
|
|
|
|
"old_switches->{port}" => $old_switches->{port},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
$pcs_add_command .= "op monitor interval=\"60\"";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
pcs_add_command => $pcs_add_command =~ /passw/ ? $anvil->Log->is_secure($pcs_add_command) : $pcs_add_command,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Does this device exist already?
|
|
|
|
if (exists $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$stonith_name})
|
|
|
|
{
|
|
|
|
foreach my $argument (sort {$a cmp $b} keys %{$anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$stonith_name}{argument}})
|
|
|
|
{
|
|
|
|
next if $argument eq "delay";
|
|
|
|
my $old_entry = $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$stonith_name}{argument}{$argument}{value};
|
|
|
|
my $new_entry = exists $old_switches->{$argument} ? $old_switches->{$argument} : "";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:argument' => $argument,
|
|
|
|
's2:old_entry' => $old_entry,
|
|
|
|
's3:new_entry' => $new_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if ($old_entry ne $new_entry)
|
|
|
|
{
|
|
|
|
# Changed, delete and recreate.
|
|
|
|
$delete_old = 1;
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0121,!!device!".$stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0121", variables => { device => $stonith_name }});
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Delete the old switch.
|
|
|
|
delete $old_switches->{$argument};
|
|
|
|
}
|
|
|
|
|
|
|
|
# Are there any old switches left?
|
|
|
|
my $old_switch_count = keys %{$old_switches};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { old_switch_count => $old_switch_count }});
|
|
|
|
if ((not $delete_old) && ($old_switch_count))
|
|
|
|
{
|
|
|
|
# Delete and recreate.
|
|
|
|
$delete_old = 1;
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0121,!!device!".$stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0121", variables => { device => $stonith_name }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# No existing entry, add a new one.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0122,!!device!".$stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0122", variables => { device => $stonith_name }});
|
|
|
|
|
|
|
|
$create_entry = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { create_entry => $create_entry }});
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
delete_old => $delete_old,
|
|
|
|
create_entry => $create_entry,
|
|
|
|
}});
|
|
|
|
if ($delete_old)
|
|
|
|
{
|
|
|
|
# Delete
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0119,!!device!".$stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0119", variables => { device => $stonith_name }});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." stonith delete ".$stonith_name;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
|
|
|
|
$something_changed->{$node_name} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "something_changed->{$node_name}" => $something_changed->{$node_name} }});
|
|
|
|
}
|
|
|
|
if ($create_entry)
|
|
|
|
{
|
|
|
|
# Create.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0120,!!device!".$stonith_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0120", variables => { device => $stonith_name }});
|
|
|
|
|
|
|
|
my $shell_call = $pcs_add_command;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
|
|
|
|
$something_changed->{$node_name} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "something_changed->{$node_name}" => $something_changed->{$node_name} }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
### If we had a fence_ipmilan entry, add a 'fence_delay' entry, if needed.
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
node => $node,
|
|
|
|
node1_use_delay => $node1_use_delay,
|
|
|
|
node2_use_delay => $node2_use_delay,
|
|
|
|
}});
|
|
|
|
if ((($node eq "node1") && ($node1_use_delay)) or
|
|
|
|
(($node eq "node2") && ($node2_use_delay)))
|
|
|
|
{
|
|
|
|
my $stonith_name = "delay_".$node;
|
|
|
|
push @{$fence_order->{$node_name}}, "fence_delay";
|
|
|
|
$fence_devices->{$node_name}{fence_delay} = [$stonith_name];
|
|
|
|
|
|
|
|
# Add the fence delay if it doesn't exist yet.
|
|
|
|
if (not exists $anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{device}{$stonith_name})
|
|
|
|
{
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." stonith create ".$stonith_name." fence_delay pcmk_host_list=\"".$node_name."\" wait=\"60\" op monitor interval=\"60\"";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Setup fence levels.
|
|
|
|
foreach my $node_name (sort {$a cmp $b} keys %{$fence_order})
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { "something_changed->{$node_name}" => $something_changed->{$node_name} }});
|
|
|
|
if ($something_changed->{$node_name})
|
|
|
|
{
|
|
|
|
# Update our view of the cluster.
|
|
|
|
my $problem = $anvil->Cluster->parse_cib({debug => 2});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { problem => $problem }});
|
|
|
|
|
|
|
|
# Delete any existing fence levels
|
|
|
|
if (exists $anvil->data->{cib}{parsed}{data}{node}{$node_name})
|
|
|
|
{
|
|
|
|
foreach my $index (sort {$a cmp $b} keys %{$anvil->data->{cib}{parsed}{data}{node}{$node_name}{fencing}{order}})
|
|
|
|
{
|
|
|
|
# pcs stonith level delete <index> <target>
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." stonith level delete ".$index." ".$node_name;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Create the new fence levels
|
|
|
|
my $i = 1;
|
|
|
|
foreach my $fence_agent (@{$fence_order->{$node_name}})
|
|
|
|
{
|
|
|
|
my $devices = "";
|
|
|
|
foreach my $device (sort {$a cmp $b} @{$fence_devices->{$node_name}{$fence_agent}})
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { device => $device }});
|
|
|
|
$devices .= $device.",";
|
|
|
|
}
|
|
|
|
$devices =~ s/,$//;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { devices => $devices }});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." stonith level add ".$i." ".$node_name." ".$devices;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Enable fencing and set the retry to INFINITY, if needed.
|
|
|
|
$anvil->data->{cib}{parsed}{data}{stonith}{'max-attempts'} = "" if not defined $anvil->data->{cib}{parsed}{data}{stonith}{'max-attempts'};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"cib::parsed::data::stonith::max-attempts" => $anvil->data->{cib}{parsed}{data}{stonith}{'max-attempts'},
|
|
|
|
}});
|
|
|
|
if ($anvil->data->{cib}{parsed}{data}{stonith}{'max-attempts'} ne "INFINITY")
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0124");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0124"});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." property set stonith-max-attempts=INFINITY";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"cib::parsed::data::stonith::enabled" => $anvil->data->{cib}{parsed}{data}{stonith}{enabled},
|
|
|
|
}});
|
|
|
|
if (not $anvil->data->{cib}{parsed}{data}{stonith}{enabled})
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0125");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0125"});
|
|
|
|
|
|
|
|
my $shell_call = $anvil->data->{path}{exe}{pcs}." property set stonith-enabled=true";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { shell_call => $shell_call }});
|
|
|
|
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $shell_call});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
if ($return_code)
|
|
|
|
{
|
|
|
|
# Something went wrong.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "error_0138", variables => {
|
|
|
|
shell_call => $shell_call,
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "error_0138,!!shell_call!".$shell_call."!!,!!output!".$output."!!,!!return_code!".$return_code."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 6});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Check if we need to change any IPs or our hostname.
|
|
|
|
sub check_local_network
|
|
|
|
{
|
|
|
|
my ($anvil) = @_;
|
|
|
|
|
|
|
|
# What host name and IP(s) should I have?
|
|
|
|
my $local_host = $anvil->Get->short_host_name();
|
|
|
|
my $machine = $anvil->data->{sys}{machine};
|
|
|
|
my $manifest_uuid = $anvil->data->{sys}{manifest_uuid};
|
|
|
|
|
|
|
|
my $domain = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{domain};
|
|
|
|
my $old_host_name = $anvil->Get->host_name;
|
|
|
|
my $new_host_name = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$machine}{name};
|
|
|
|
if ($domain)
|
|
|
|
{
|
|
|
|
$new_host_name = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$machine}{name}.".".$domain;
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's2:domain' => $domain,
|
|
|
|
's3:old_host_name' => $old_host_name,
|
|
|
|
's4:new_host_name' => $new_host_name,
|
|
|
|
}});
|
|
|
|
|
|
|
|
$anvil->data->{sys}{host_name} = $new_host_name;
|
|
|
|
|
|
|
|
# If the hostname isn't the same, change it.
|
|
|
|
if ($old_host_name ne $new_host_name)
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0061,!!host_name!".$new_host_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0061", variables => { host_name => $new_host_name }});
|
|
|
|
my ($now_host_name) = $anvil->System->host_name({
|
|
|
|
debug => 2,
|
|
|
|
set => $new_host_name,
|
|
|
|
});
|
|
|
|
if ($now_host_name eq $new_host_name)
|
|
|
|
{
|
|
|
|
# Success!
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0045");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0045"});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Failed
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "job_0063", variables => {
|
|
|
|
host_name => $new_host_name,
|
|
|
|
current_host_name => $now_host_name,
|
|
|
|
}});
|
|
|
|
update_progress($anvil, 0, "job_0063,!!host_name!".$new_host_name."!!,!!current_host_name!".$now_host_name."!!");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 3});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# No need to change
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0077,!!host_name!".$new_host_name."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0077", variables => { host_name => $new_host_name }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Read the local network manager data.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0080");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0080"});
|
|
|
|
$anvil->Network->read_nmcli({debug => 2});
|
|
|
|
|
|
|
|
# Now check IP addresses.
|
|
|
|
$anvil->Network->get_ips({debug => 2});
|
|
|
|
|
|
|
|
# This will be set when the first IFN with a gateway is set.
|
|
|
|
my $default_gateway_interface = "";
|
|
|
|
foreach my $in_iface (sort {$a cmp $b} keys %{$anvil->data->{network}{$local_host}{interface}})
|
|
|
|
{
|
|
|
|
if ($anvil->data->{network}{$local_host}{interface}{$in_iface}{default_gateway})
|
|
|
|
{
|
|
|
|
$default_gateway_interface = $in_iface;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { default_gateway_interface => $default_gateway_interface }});
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
my $dns = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{networks}{dns};
|
|
|
|
my $mtu = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{networks}{mtu};
|
|
|
|
my $ntp = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{networks}{ntp};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
dns => $dns,
|
|
|
|
mtu => $mtu,
|
|
|
|
ntp => $ntp,
|
|
|
|
}});
|
|
|
|
|
|
|
|
### TODO: sorting the array seems inconsistent, so sorting in a hash
|
|
|
|
# The DNS are comma-separated lists, that may or may not have spaces and may or may not be in
|
|
|
|
# alphabetical order. To properly compare, we'll rebuild the CSV string of the current and desired
|
|
|
|
# DNS settings.
|
|
|
|
my @dns_array = split/,/, $dns;
|
|
|
|
my $wanted_dns_hash = {};
|
|
|
|
foreach my $this_dns (sort {$a cmp $b} @dns_array)
|
|
|
|
{
|
|
|
|
$this_dns = $anvil->Words->clean_spaces({ string => $this_dns });
|
|
|
|
$wanted_dns_hash->{$this_dns} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"wanted_dns_hash->${this_dns}" => $wanted_dns_hash->{$this_dns},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
my $cleaned_wanted_dns = "";
|
|
|
|
foreach my $wanted_dns (sort {$a cmp $b} keys %{$wanted_dns_hash})
|
|
|
|
{
|
|
|
|
$cleaned_wanted_dns .= $wanted_dns.",";
|
|
|
|
}
|
|
|
|
$cleaned_wanted_dns =~ s/,$//;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { cleaned_wanted_dns => $cleaned_wanted_dns }});
|
|
|
|
|
|
|
|
# If any network interface is changed, we'll write out the config file then, when done, disconnect
|
|
|
|
# from the database, restart networking and then reconnect before moving on.
|
|
|
|
my $restart_interfaces = [];
|
|
|
|
|
|
|
|
foreach my $network (sort {$a cmp $b} keys %{$anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$machine}{network}})
|
|
|
|
{
|
|
|
|
my $ip = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{machine}{$machine}{network}{$network}{ip};
|
|
|
|
my $subnet = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{networks}{name}{$network}{subnet};
|
|
|
|
my $cidr = $anvil->Convert->cidr({subnet_mask => $subnet});
|
|
|
|
my $gateway = $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}{parsed}{networks}{name}{$network}{gateway};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's1:network' => $network,
|
|
|
|
's2:ip' => $ip,
|
|
|
|
's3:subnet' => $subnet,
|
|
|
|
's4:cidr' => $cidr,
|
|
|
|
's5:gateway' => $gateway,
|
|
|
|
}});
|
|
|
|
foreach my $in_iface (sort {$a cmp $b} keys %{$anvil->data->{network}{$local_host}{interface}})
|
|
|
|
{
|
|
|
|
# Only one interface will start with the network name and have an IP address.
|
|
|
|
next if $in_iface !~ /^${network}_/;
|
|
|
|
next if not $anvil->data->{network}{$local_host}{interface}{$in_iface}{ip};
|
|
|
|
my $current_ip = $anvil->data->{network}{$local_host}{interface}{$in_iface}{ip};
|
|
|
|
my $current_subnet = $anvil->data->{network}{$local_host}{interface}{$in_iface}{subnet_mask};
|
|
|
|
my $current_gateway = $anvil->data->{network}{$local_host}{interface}{$in_iface}{gateway};
|
|
|
|
my $current_dns = $anvil->data->{network}{$local_host}{interface}{$in_iface}{dns};
|
|
|
|
my $current_mtu = $anvil->data->{network}{$local_host}{interface}{$in_iface}{mtu};
|
|
|
|
my $mac_address = $anvil->data->{network}{$local_host}{interface}{$in_iface}{mac_address};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:in_iface' => $in_iface,
|
|
|
|
's2:current_ip' => $current_ip,
|
|
|
|
's4:current_subnet' => $current_subnet,
|
|
|
|
's5:current_gateway' => $current_gateway,
|
|
|
|
's6:current_dns' => $current_dns,
|
|
|
|
's7:current_mtu' => $current_mtu,
|
|
|
|
}});
|
|
|
|
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0081,!!name!".$in_iface."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0081", variables => { name => $in_iface }});
|
|
|
|
|
|
|
|
if ((not $default_gateway_interface) && ($in_iface =~ /^ifn/) && ($gateway))
|
|
|
|
{
|
|
|
|
# No existing default gateway, but this is the first IFN we've seen with a
|
|
|
|
# gateway defined, so we'll use this one.
|
|
|
|
$default_gateway_interface = $in_iface;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { default_gateway_interface => $default_gateway_interface }});
|
|
|
|
}
|
|
|
|
|
|
|
|
my $change = 0;
|
|
|
|
my $current_dns_hash = {};
|
|
|
|
my @current_dns_array = split/,/, $current_dns;
|
|
|
|
foreach my $this_current_dns (sort {$a cmp $b} @current_dns_array)
|
|
|
|
{
|
|
|
|
$this_current_dns = $anvil->Words->clean_spaces({ string => $this_current_dns });
|
|
|
|
$current_dns_hash->{$this_current_dns} = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"current_dns_hash->::${this_current_dns}" => $current_dns_hash->{$this_current_dns},
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
my $cleaned_current_dns = "";
|
|
|
|
foreach my $current_dns (sort {$a cmp $b} keys %{$current_dns_hash})
|
|
|
|
{
|
|
|
|
$cleaned_current_dns .= $current_dns.",";
|
|
|
|
}
|
|
|
|
$cleaned_current_dns =~ s/,$//;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
cleaned_wanted_dns => $cleaned_wanted_dns,
|
|
|
|
cleaned_current_dns => $cleaned_current_dns,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if (($current_ip ne $ip) or ($current_subnet ne $subnet))
|
|
|
|
{
|
|
|
|
# IP / subnet changed.
|
|
|
|
#print "IP: .... [".$current_ip."] -> [".$ip."]\n";
|
|
|
|
#print "Subnet: [".$current_subnet."] -> [".$subnet."]\n";
|
|
|
|
$change = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { change => $change }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# gateway?
|
|
|
|
if ($current_gateway ne $gateway)
|
|
|
|
{
|
|
|
|
#print "Gateway: [".$current_gateway."] -> [".$gateway."]\n";
|
|
|
|
$change = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { change => $change }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Check DNS only if this is the default gateway interface.
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
in_iface => $in_iface,
|
|
|
|
default_gateway_interface => $default_gateway_interface,
|
|
|
|
cleaned_current_dns => $cleaned_current_dns,
|
|
|
|
}});
|
|
|
|
if ($in_iface eq $default_gateway_interface)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
cleaned_current_dns => $cleaned_current_dns,
|
|
|
|
cleaned_wanted_dns => $cleaned_wanted_dns,
|
|
|
|
}});
|
|
|
|
if ($cleaned_wanted_dns ne $cleaned_current_dns)
|
|
|
|
{
|
|
|
|
$change = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { change => $change }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($cleaned_current_dns)
|
|
|
|
{
|
|
|
|
# Remove the old DNS entries.
|
|
|
|
$change = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { change => $change }});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (not $change)
|
|
|
|
{
|
|
|
|
# No change
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0082");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0082"});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Update the config.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0078,!!interface!".$in_iface."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0078", variables => { interface => $in_iface }});
|
|
|
|
|
|
|
|
my $interface_uuid = $anvil->data->{nmcli}{$local_host}{device_to_uuid}{$in_iface};
|
|
|
|
my $filename = $anvil->data->{nmcli}{$local_host}{uuid}{$interface_uuid}{filename};
|
|
|
|
my $interface_name = $anvil->data->{nmcli}{$local_host}{uuid}{$interface_uuid}{name};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:in_iface' => $in_iface,
|
|
|
|
's2:interface_name' => $interface_name,
|
|
|
|
's3:filename' => $filename,
|
|
|
|
's4:interface_uuid' => $interface_uuid,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Record the name to restart
|
|
|
|
push @{$restart_interfaces}, $interface_name;
|
|
|
|
|
|
|
|
# Read in the file. We'll update the date stamp and change the variables as
|
|
|
|
# needed and write it back out.
|
|
|
|
my $ip_seen = 0;
|
|
|
|
my $subnet_seen = 0;
|
|
|
|
my $gateway_seen = 0;
|
|
|
|
my $dns_seen = 0;
|
|
|
|
my $mtu_seen = 0;
|
|
|
|
my $boot_proto_seen = 0;
|
|
|
|
my $defroute_seen = 0;
|
|
|
|
my $say_default = $default_gateway_interface eq $in_iface ? "YES" : "NO";
|
|
|
|
my $new_config = "";
|
|
|
|
my $old_config = $anvil->Storage->read_file({file => $filename});
|
|
|
|
foreach my $line (split/\n/, $old_config)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { line => $line }});
|
|
|
|
if ($line =~ /^BOOTPROTO="(.*?)"/)
|
|
|
|
{
|
|
|
|
my $boot_protocol = $1;
|
|
|
|
$boot_proto_seen = 1;
|
|
|
|
$new_config .= "BOOTPROTO=\"none\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:boot_proto_seen' => $boot_proto_seen,
|
|
|
|
's2:boot_protocol' => $boot_protocol,
|
|
|
|
's3:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
if ($boot_protocol eq "dhcp")
|
|
|
|
{
|
|
|
|
# Inject the network config.
|
|
|
|
if (not $ip_seen)
|
|
|
|
{
|
|
|
|
$ip_seen = 1;
|
|
|
|
$new_config .= "IPADDR=\"".$ip."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:ip_seen' => $boot_proto_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
if (not $subnet_seen)
|
|
|
|
{
|
|
|
|
if ($cidr)
|
|
|
|
{
|
|
|
|
$new_config .= "PREFIX=\"".$cidr."\"\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$new_config .= "NETMASK=\"".$subnet."\"\n";
|
|
|
|
}
|
|
|
|
$subnet_seen = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:subnet_seen' => $subnet_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
if (($gateway) && (not $gateway_seen))
|
|
|
|
{
|
|
|
|
$gateway_seen = 1;
|
|
|
|
$new_config .= "GATEWAY=\"".$gateway."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:gateway_seen' => $gateway_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
if (not $defroute_seen)
|
|
|
|
{
|
|
|
|
$defroute_seen = 1;
|
|
|
|
$new_config .= "DEFROUTE=\"".$say_default."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:defroute_seen' => $defroute_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
if (($default_gateway_interface eq $in_iface) && ($cleaned_wanted_dns))
|
|
|
|
{
|
|
|
|
if (not $dns_seen)
|
|
|
|
{
|
|
|
|
my $i = 1;
|
|
|
|
foreach my $this_dns (split/,/, $cleaned_wanted_dns)
|
|
|
|
{
|
|
|
|
$new_config .= "DNS".$i."=\"".$this_dns."\"\n";
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
$dns_seen = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:dns_seen' => $dns_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($line =~ /^IPADDR=".*?"/)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { ip_seen => $ip_seen }});
|
|
|
|
if (not $ip_seen)
|
|
|
|
{
|
|
|
|
$ip_seen = 1;
|
|
|
|
$new_config .= "IPADDR=\"".$ip."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:ip_seen' => $boot_proto_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif (($line =~ /^PREFIX=".*?"/) or ($line =~ /^NETMASK=".*?"/))
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { subnet_seen => $subnet_seen }});
|
|
|
|
if (not $subnet_seen)
|
|
|
|
{
|
|
|
|
$subnet_seen = 1;
|
|
|
|
if ($cidr)
|
|
|
|
{
|
|
|
|
$new_config .= "PREFIX=\"".$cidr."\"\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$new_config .= "NETMASK=\"".$subnet."\"\n";
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:subnet_seen' => $subnet_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($line =~ /^GATEWAY=".*"/)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { gateway_seen => $gateway_seen }});
|
|
|
|
if ($in_iface ne $default_gateway_interface)
|
|
|
|
{
|
|
|
|
# This interface shouldn't have DNS.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0111", variables => {
|
|
|
|
interface => $in_iface,
|
|
|
|
gateway_line => $line,
|
|
|
|
}});
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
if (not $gateway_seen)
|
|
|
|
{
|
|
|
|
$gateway_seen = 1;
|
|
|
|
$new_config .= "GATEWAY=\"".$gateway."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:gateway_seen' => $gateway_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { gateway_seen => $gateway_seen }});
|
|
|
|
}
|
|
|
|
elsif ($line =~ /^DNS\d+=".*"/)
|
|
|
|
{
|
|
|
|
# The first time we see a DNS entry, we inject the DNS we've
|
|
|
|
# got and then ignore the rest.
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { dns_seen => $dns_seen }});
|
|
|
|
if ($in_iface ne $default_gateway_interface)
|
|
|
|
{
|
|
|
|
# This interface shouldn't have DNS.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0110", variables => {
|
|
|
|
interface => $in_iface,
|
|
|
|
dns_line => $line,
|
|
|
|
}});
|
|
|
|
next;
|
|
|
|
}
|
|
|
|
if (not $dns_seen)
|
|
|
|
{
|
|
|
|
|
|
|
|
# If there's no DNS, this will do nothing.
|
|
|
|
my $i = 1;
|
|
|
|
foreach my $this_dns (split/,/, $cleaned_wanted_dns)
|
|
|
|
{
|
|
|
|
$new_config .= "DNS".$i."=\"".$this_dns."\"\n";
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
$dns_seen = 1;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:dns_seen' => $dns_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
elsif ($line =~ /^MTU=".*"/)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { mtu_seen => $mtu_seen }});
|
|
|
|
if (not $mtu_seen)
|
|
|
|
{
|
|
|
|
$mtu_seen = 1;
|
|
|
|
$new_config .= "MTU=\"".$mtu."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:mtu_seen' => $mtu_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { mtu_seen => $mtu_seen }});
|
|
|
|
}
|
|
|
|
elsif ($line =~ /^DEFROUTE=".*"/)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { defroute_seen => $defroute_seen }});
|
|
|
|
if (not $defroute_seen)
|
|
|
|
{
|
|
|
|
$defroute_seen = 1;
|
|
|
|
$new_config .= "DEFROUTE=\"".$say_default."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
's1:defroute_seen' => $defroute_seen,
|
|
|
|
's2:new_config' => $new_config,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { defroute_seen => $defroute_seen }});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$new_config .= $line."\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { defroute_seen => $defroute_seen }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Add any values I've not yet seen.
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
ip_seen => $ip_seen,
|
|
|
|
subnet_seen => $subnet_seen,
|
|
|
|
gateway_seen => $gateway_seen,
|
|
|
|
dns_seen => $dns_seen,
|
|
|
|
mtu_seen => $mtu_seen,
|
|
|
|
defroute_seen => $defroute_seen,
|
|
|
|
}});
|
|
|
|
if (not $ip_seen)
|
|
|
|
{
|
|
|
|
$new_config .= "IPADDR=\"".$ip."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
if (not $subnet_seen)
|
|
|
|
{
|
|
|
|
if ($cidr)
|
|
|
|
{
|
|
|
|
$new_config .= "PREFIX=\"".$cidr."\"\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$new_config .= "NETMASK=\"".$subnet."\"\n";
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
if ($in_iface eq $default_gateway_interface)
|
|
|
|
{
|
|
|
|
if (not $gateway_seen)
|
|
|
|
{
|
|
|
|
$new_config .= "GATEWAY=\"".$gateway."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
if (not $dns_seen)
|
|
|
|
{
|
|
|
|
# If there's no DNS, this will do nothing.
|
|
|
|
my $i = 1;
|
|
|
|
foreach my $this_dns (split/,/, $cleaned_wanted_dns)
|
|
|
|
{
|
|
|
|
$new_config .= "DNS".$i."=\"".$this_dns."\"\n";
|
|
|
|
$i++;
|
|
|
|
}
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (not $mtu_seen)
|
|
|
|
{
|
|
|
|
$new_config .= "MTU=\"".$mtu."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
if (not $defroute_seen)
|
|
|
|
{
|
|
|
|
$new_config .= "DEFROUTE=\"".$say_default."\"\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Write out the new file.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0083,!!file!".$filename."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0083", variables => { file => $filename }});
|
|
|
|
$anvil->Storage->write_file({
|
|
|
|
debug => 2,
|
|
|
|
file => $filename,
|
|
|
|
body => $new_config,
|
|
|
|
user => "root",
|
|
|
|
group => "root",
|
|
|
|
mode => "0644",
|
|
|
|
overwrite => 1,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# If there are any entries in '$restart_interfaces', restart
|
|
|
|
my $restart_interface_count = @{$restart_interfaces};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { restart_interface_count => $restart_interface_count }});
|
|
|
|
if ($restart_interface_count)
|
|
|
|
{
|
|
|
|
# Disconnect from the database, as we're about to tear down our connection.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0079"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0079");
|
|
|
|
$anvil->Database->disconnect();
|
|
|
|
|
|
|
|
# Tell nmcli to re-read the config files.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "log_0463"});
|
|
|
|
my ($output, $return_code) = $anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{nmcli}." connection reload"});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
output => $output,
|
|
|
|
return_code => $return_code,
|
|
|
|
}});
|
|
|
|
|
|
|
|
foreach my $interface_name (sort {$a cmp $b} @{$restart_interfaces})
|
|
|
|
{
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "log_0462", variables => { interface => $interface_name }});
|
|
|
|
$anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{nmcli}." connection down \"".$interface_name."\""});
|
|
|
|
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "log_0464", variables => { interface => $interface_name }});
|
|
|
|
$anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{nmcli}." connection up \"".$interface_name."\""});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Wait for a DB connection. We'll wait up to 130 seconds (updelay is 120 seconds, plus a small buffer).
|
|
|
|
my $wait_until = time + 130;
|
|
|
|
until ($anvil->data->{sys}{database}{connections})
|
|
|
|
{
|
|
|
|
$anvil->refresh();
|
|
|
|
$anvil->Database->connect();
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "log_0132"});
|
|
|
|
if (not $anvil->data->{sys}{database}{connections})
|
|
|
|
{
|
|
|
|
if (time > $wait_until)
|
|
|
|
{
|
|
|
|
# Failed to reconnect, exit.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 0, 'print' => 1, priority => "err", key => "error_0107"});
|
|
|
|
$anvil->nice_exit({exit_code => 4});
|
|
|
|
}
|
|
|
|
|
|
|
|
# No databases, sleep and then try again.
|
|
|
|
sleep 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
# reload the job data.
|
|
|
|
load_job($anvil);
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0084"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0084");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Remove virbr0 if it exists.
|
|
|
|
if (exists $anvil->data->{network}{$local_host}{interface}{virbr0})
|
|
|
|
{
|
|
|
|
# Remove the NAT'ed bridge
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0085"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0085");
|
|
|
|
|
|
|
|
$anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{virsh}." net-destroy default"});
|
|
|
|
$anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{virsh}." net-undefine default "});
|
|
|
|
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0034"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0034");
|
|
|
|
}
|
|
|
|
|
|
|
|
# Update network view
|
|
|
|
$anvil->Network->read_nmcli({debug => 3});
|
|
|
|
$anvil->Network->get_ips({debug => 3});
|
|
|
|
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0086"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0086");
|
|
|
|
|
|
|
|
# Update MTUs (running interface and config) if needed.
|
|
|
|
foreach my $in_iface (sort {$a cmp $b} keys %{$anvil->data->{network}{$local_host}{interface}})
|
|
|
|
{
|
|
|
|
# Only one interface will start with the network name and have an IP address.
|
|
|
|
my $current_mtu = $anvil->data->{network}{$local_host}{interface}{$in_iface}{mtu};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's1:in_iface' => $in_iface,
|
|
|
|
's2:current_mtu' => $current_mtu,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if (($mtu) && ($current_mtu eq $mtu))
|
|
|
|
{
|
|
|
|
# It's fine
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0087,!!interface!".$in_iface."!!,!!mtu!".$mtu."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 3, key => "job_0087", variables => {
|
|
|
|
interface => $in_iface,
|
|
|
|
mtu => $mtu,
|
|
|
|
}});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
# Change the MTU both on the running interface and in the config file.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0088,!!interface!".$in_iface."!!,!!old_mtu!".$current_mtu."!!,!!mtu!".$mtu."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0088", variables => {
|
|
|
|
interface => $in_iface,
|
|
|
|
old_mtu => $current_mtu,
|
|
|
|
mtu => $mtu,
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Change the live MTU.
|
|
|
|
$anvil->System->call({debug => 3, shell_call => $anvil->data->{path}{exe}{ip}." link set ".$in_iface." mtu ".$mtu});
|
|
|
|
|
|
|
|
# Now update the config file.
|
|
|
|
my $interface_uuid = $anvil->data->{nmcli}{$local_host}{device_to_uuid}{$in_iface};
|
|
|
|
my $filename = $anvil->data->{nmcli}{$local_host}{uuid}{$interface_uuid}{filename};
|
|
|
|
my $interface_name = $anvil->data->{nmcli}{$local_host}{uuid}{$interface_uuid}{name};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's1:interface_name' => $interface_name,
|
|
|
|
's2:filename' => $filename,
|
|
|
|
's3:interface_uuid' => $interface_uuid,
|
|
|
|
}});
|
|
|
|
|
|
|
|
my $mtu_seen = 0;
|
|
|
|
my $new_config = "";
|
|
|
|
my $old_config = $anvil->Storage->read_file({file => $filename});
|
|
|
|
foreach my $line (split/\n/, $old_config)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { line => $line }});
|
|
|
|
if ($line =~ /^MTU=".*?"/)
|
|
|
|
{
|
|
|
|
$mtu_seen = 1;
|
|
|
|
$new_config .= "MTU=\"".$mtu."\"\n";
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$new_config .= $line."\n";
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { new_config => $new_config }});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (not $mtu_seen)
|
|
|
|
{
|
|
|
|
# Inject the MTU variable
|
|
|
|
$new_config .= "MTU=\"".$mtu."\"\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
# Write out the new file.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0083,!!file!".$filename."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0083", variables => { file => $filename }});
|
|
|
|
$anvil->Storage->write_file({
|
|
|
|
debug => 2,
|
|
|
|
file => $filename,
|
|
|
|
body => $new_config,
|
|
|
|
user => "root",
|
|
|
|
group => "root",
|
|
|
|
mode => "0644",
|
|
|
|
overwrite => 1,
|
|
|
|
});
|
|
|
|
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, level => 1, 'print' => 1, key => "job_0034"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0034");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Update NTP if set and needed.
|
|
|
|
if ($ntp)
|
|
|
|
{
|
|
|
|
# Break up the NTP servers into a list, we'll set to '1' the ones we find.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0089");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0089"});
|
|
|
|
my $new_ntp = {};
|
|
|
|
foreach my $ntp_server (split/,/, $ntp)
|
|
|
|
{
|
|
|
|
$ntp_server = $anvil->Words->clean_spaces({ string => $ntp_server });
|
|
|
|
$new_ntp->{$ntp_server} = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
# Call chrony to see what servers are setup already.
|
|
|
|
my $change = 0;
|
|
|
|
my $new_config = "";
|
|
|
|
my $old_config = $anvil->Storage->read_file({file => $anvil->data->{path}{data}{'chrony.conf'}});
|
|
|
|
foreach my $line (split/\n/, $old_config)
|
|
|
|
{
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => { line => $line }});
|
|
|
|
$new_config .= $line."\n";
|
|
|
|
if ($line =~ /^Server (.*)$/)
|
|
|
|
{
|
|
|
|
# This is a source.
|
|
|
|
my $source = $1;
|
|
|
|
$new_ntp->{$source} = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Are any sources not enabled?
|
|
|
|
foreach my $source (sort {$a cmp $b} keys %{$new_ntp})
|
|
|
|
{
|
|
|
|
if (not $new_ntp->{$source})
|
|
|
|
{
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0090,!!server!".$source."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0090", variables => { server => $source }});
|
|
|
|
$change = 1;
|
|
|
|
$new_config .= "Server ".$source."\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($change)
|
|
|
|
{
|
|
|
|
# Write out the updated file.
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0055,!!file!".$anvil->data->{path}{data}{'chrony.conf'}."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0055", variables => { file => $anvil->data->{path}{data}{'chrony.conf'} }});
|
|
|
|
$anvil->Storage->write_file({
|
|
|
|
debug => 2,
|
|
|
|
file => $anvil->data->{path}{data}{'chrony.conf'},
|
|
|
|
body => $new_config,
|
|
|
|
user => "root",
|
|
|
|
group => "root",
|
|
|
|
mode => "0644",
|
|
|
|
overwrite => 1,
|
|
|
|
});
|
|
|
|
|
|
|
|
# Restart the daemon.
|
|
|
|
my $daemon = "chronyd";
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0091,!!daemon!".$daemon."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 1, key => "job_0091", variables => { daemon => $daemon }});
|
|
|
|
$anvil->System->restart_daemon({
|
|
|
|
debug => 3,
|
|
|
|
daemon => $daemon,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# Update the hosts file.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 3, key => "job_0112"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0112");
|
|
|
|
$anvil->System->update_hosts({debug => 3});
|
|
|
|
|
|
|
|
# Configure SSH by adding ours and our peer's SSH keys to ~/.ssh/known_hosts
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 3, key => "job_0113"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0113");
|
|
|
|
$anvil->System->check_ssh_keys({debug => 3});
|
|
|
|
|
|
|
|
# Setup IPMI, if needed.
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 3, key => "job_0114"});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 2), "job_0114");
|
|
|
|
$anvil->System->configure_ipmi({debug => 3, manifest_uuid => $manifest_uuid});
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load the job details.
|
|
|
|
sub load_job
|
|
|
|
{
|
|
|
|
my ($anvil) = @_;
|
|
|
|
|
|
|
|
# See if we can find the job details. This method checks for the 'job-uuid' switch if it was used.
|
|
|
|
$anvil->data->{switches}{'job-uuid'} = "" if not exists $anvil->data->{switches}{'job-uuid'};
|
|
|
|
$anvil->data->{jobs}{job_uuid} = "" if not exists $anvil->data->{jobs}{job_uuid};
|
|
|
|
|
|
|
|
$anvil->Job->get_job_details({debug => 3});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
"jobs::job_uuid" => $anvil->data->{jobs}{job_uuid},
|
|
|
|
}});
|
|
|
|
|
|
|
|
if (not $anvil->data->{jobs}{job_uuid})
|
|
|
|
{
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "job_0130"});
|
|
|
|
sleep 1;
|
|
|
|
$anvil->nice_exit({exit_code => 7});
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->data->{job}{progress} = 0;
|
|
|
|
update_progress($anvil, $anvil->data->{job}{progress}, "clear");
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 1), "job_0074,!!job-uuid!".$anvil->data->{switches}{'job-uuid'}."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0074", variables => { 'job-uuid' => $anvil->data->{switches}{'job-uuid'} }});
|
|
|
|
|
|
|
|
my ($machine, $manifest_uuid, $anvil_uuid) = ($anvil->data->{jobs}{job_data} =~ /as_machine=(.*?),manifest_uuid=(.*?),anvil_uuid=(.*?)$/);
|
|
|
|
$machine = "" if not defined $machine;
|
|
|
|
$manifest_uuid = "" if not defined $manifest_uuid;
|
|
|
|
$anvil_uuid = "" if not defined $anvil_uuid;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
machine => $machine,
|
|
|
|
manifest_uuid => $manifest_uuid,
|
|
|
|
anvil_uuid => $anvil_uuid,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if ((not $machine) or
|
|
|
|
(not $manifest_uuid) or
|
|
|
|
(not $anvil_uuid))
|
|
|
|
{
|
|
|
|
# Terminate the job entirely, it's likely an unrecoverable problem.
|
|
|
|
update_progress($anvil, 100, "job_0092,!!job-uuid!".$anvil->data->{switches}{'job-uuid'}."!!,!!raw!".$anvil->data->{jobs}{job_data}."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "job_0092", variables => {
|
|
|
|
'job-uuid' => $anvil->data->{switches}{'job-uuid'},
|
|
|
|
raw => $anvil->data->{jobs}{job_data},
|
|
|
|
}});
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 5});
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->data->{sys}{machine} = $machine;
|
|
|
|
$anvil->data->{sys}{manifest_uuid} = $manifest_uuid;
|
|
|
|
$anvil->data->{sys}{anvil_uuid} = $anvil_uuid;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 2, list => {
|
|
|
|
"sys::machine" => $anvil->data->{sys}{machine},
|
|
|
|
"sys::manifest_uuid" => $anvil->data->{sys}{manifest_uuid},
|
|
|
|
"sys::anvil_uuid" => $anvil->data->{sys}{anvil_uuid},
|
|
|
|
}});
|
|
|
|
|
|
|
|
# Load in the host info and the manifest.
|
|
|
|
$anvil->Database->get_hosts();
|
|
|
|
my $problem = $anvil->Striker->load_manifest({manifest_uuid => $anvil->data->{sys}{manifest_uuid}});
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
manifest_uuid => $anvil->data->{sys}{manifest_uuid},
|
|
|
|
problem => $problem,
|
|
|
|
}});
|
|
|
|
|
|
|
|
if ($problem)
|
|
|
|
{
|
|
|
|
# Something went wrong
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 0, priority => "err", key => "warning_0046", variables => { uuid => $anvil->data->{sys}{manifest_uuid} }});
|
|
|
|
update_progress($anvil, 0, "job_0076");
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 2});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load the manifest and anvil data.
|
|
|
|
$anvil->Database->get_anvils();
|
|
|
|
if ((not exists $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}) or (not exists $anvil->data->{manifests}{manifest_uuid}{$manifest_uuid}))
|
|
|
|
{
|
|
|
|
# Terminate the job entirely, it's likely an unrecoverable problem.
|
|
|
|
update_progress($anvil, 100, "job_0092,!!job-uuid!".$anvil->data->{switches}{'job-uuid'}."!!,!!raw!".$anvil->data->{jobs}{job_data}."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0092", variables => {
|
|
|
|
'job-uuid' => $anvil->data->{switches}{'job-uuid'},
|
|
|
|
raw => $anvil->data->{jobs}{job_data},
|
|
|
|
}});
|
|
|
|
sleep 2;
|
|
|
|
$anvil->nice_exit({exit_code => 5});
|
|
|
|
}
|
|
|
|
|
|
|
|
# Load the anvil
|
|
|
|
$anvil->data->{sys}{node1_host_uuid} = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_node1_host_uuid};
|
|
|
|
$anvil->data->{sys}{node2_host_uuid} = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_node2_host_uuid};
|
|
|
|
$anvil->data->{sys}{dr1_host_uuid} = $anvil->data->{anvils}{anvil_uuid}{$anvil_uuid}{anvil_dr1_host_uuid};
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
"sys::node1_host_uuid" => $anvil->data->{sys}{node1_host_uuid},
|
|
|
|
"sys::node2_host_uuid" => $anvil->data->{sys}{node2_host_uuid},
|
|
|
|
"sys::dr1_host_uuid" => $anvil->data->{sys}{dr1_host_uuid},
|
|
|
|
}});
|
|
|
|
update_progress($anvil, ($anvil->data->{job}{progress} += 1), "job_0075,!!machine!".$anvil->data->{sys}{machine}."!!,!!manifest_uuid!".$anvil->data->{sys}{manifest_uuid}."!!");
|
|
|
|
$anvil->Log->entry({source => $THIS_FILE, line => __LINE__, 'print' => 1, level => 2, key => "job_0075", variables => {
|
|
|
|
machine => $anvil->data->{sys}{machine},
|
|
|
|
manifest_uuid => $anvil->data->{sys}{manifest_uuid},
|
|
|
|
}});
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
# If this is being called as a job, this will allow the progress to be updated.
|
|
|
|
sub update_progress
|
|
|
|
{
|
|
|
|
my ($anvil, $progress, $message) = @_;
|
|
|
|
$anvil->Log->variables({source => $THIS_FILE, line => __LINE__, level => 3, list => {
|
|
|
|
's1:progress' => $progress,
|
|
|
|
's2:message' => $message,
|
|
|
|
}});
|
|
|
|
|
|
|
|
$progress = 98 if $progress > 100;
|
|
|
|
if (not $anvil->data->{switches}{'job-uuid'})
|
|
|
|
{
|
|
|
|
return(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
$anvil->Job->update_progress({
|
|
|
|
debug => 3,
|
|
|
|
progress => $progress,
|
|
|
|
message => $message,
|
|
|
|
job_uuid => $anvil->data->{switches}{'job-uuid'},
|
|
|
|
});
|
|
|
|
|
|
|
|
return(0);
|
|
|
|
}
|