mirror of
https://github.com/xcat2/xcat-core.git
synced 2025-10-24 07:55:27 +00:00
2744 lines
81 KiB
Perl
2744 lines
81 KiB
Perl
# IBM(c) 2011 EPL license http://www.eclipse.org/legal/epl-v10.html
|
|
#-------------------------------------------------------
|
|
|
|
=head 1
|
|
|
|
xCAT plugin to handle xCAT UI commands
|
|
|
|
=cut
|
|
|
|
#-------------------------------------------------------
|
|
|
|
package xCAT_plugin::web;
|
|
use strict;
|
|
require xCAT::Utils;
|
|
require xCAT::MsgUtils;
|
|
require xCAT::DBobjUtils;
|
|
require IO::Socket::INET;
|
|
use Getopt::Long;
|
|
use Data::Dumper;
|
|
use LWP::Simple;
|
|
use xCAT::Table;
|
|
use xCAT::NodeRange;
|
|
use xCAT::TableUtils;
|
|
require XML::Parser;
|
|
|
|
sub handled_commands {
|
|
return { webrun => "web" };
|
|
}
|
|
|
|
sub process_request {
|
|
my $request = shift;
|
|
my $callback = shift;
|
|
my $sub_req = shift;
|
|
my %authorized_cmds = (
|
|
'update' => \&web_update,
|
|
'lscondition' => \&web_lscond,
|
|
'lsresponse' => \&web_lsresp,
|
|
'lscondresp' => \&web_lscondresp,
|
|
'mkcondresp' => \&web_mkcondresp,
|
|
'startcondresp' => \&web_startcondresp,
|
|
'stopcondresp' => \&web_stopcondresp,
|
|
'lsevent' => \&web_lsevent,
|
|
'unlock' => \&web_unlock,
|
|
'unlockbyip' => \&web_unlockByIP,
|
|
'unlockshow' => \&web_unlockShow,
|
|
'rmcstart' => \&web_rmcmonStart,
|
|
'rmcshow' => \&web_rmcmonShow,
|
|
'gangliaconf' => \&web_gangliaconf,
|
|
'gangliastart' => \&web_gangliastart,
|
|
'gangliastop' => \&web_gangliastop,
|
|
'gangliastatus' => \&web_gangliastatus,
|
|
'gangliacheck' => \&web_gangliacheck,
|
|
'installganglia' => \&web_installganglia,
|
|
'mkcondition' => \&web_mkcondition,
|
|
'monls' => \&web_monls,
|
|
'dynamiciprange' => \&web_dynamiciprange,
|
|
'discover' => \&web_discover,
|
|
'updatevpd' => \&web_updatevpd,
|
|
'writeconfigfile' => \&web_writeconfigfile,
|
|
'createimage' => \&web_createimage,
|
|
'provision' => \&web_provision,
|
|
'summary' => \&web_summary,
|
|
'gangliashow' => \&web_gangliaShow,
|
|
'gangliacurrent' => \&web_gangliaLatest,
|
|
'rinstall' => \&web_rinstall,
|
|
'addnode' => \&web_addnode,
|
|
'graph' => \&web_graphinfo,
|
|
'getdefaultuserentry' => \&web_getdefaultuserentry,
|
|
'getzdiskinfo' => \&web_getzdiskinfo,
|
|
'passwd' => \&web_passwd,
|
|
'policy' => \&web_policy,
|
|
'deleteuser' => \&web_deleteuser,
|
|
'mkzprofile' => \&web_mkzprofile,
|
|
'rmzprofile' => \&web_rmzprofile,
|
|
'mkippool' => \&web_mkippool,
|
|
'rmippool' => \&web_rmippool,
|
|
'lsippool' => \&web_lsippool,
|
|
'updateosimage' => \&web_updateosimage,
|
|
'rmosimage' => \&web_rmosimage,
|
|
'updategroup' => \&web_updategroup,
|
|
'rmgroup' => \&web_rmgroup,
|
|
'framesetup' => \&web_framesetup,
|
|
'cecsetup' => \&web_cecsetup,
|
|
'deletefile' => \&web_deletefile,
|
|
'createfolder' => \&web_createfolder,
|
|
'getrepospace' => \&web_getrepospace,
|
|
'verifynode' => \&web_verifynode,
|
|
);
|
|
|
|
# Check whether the request is authorized or not
|
|
@_ = split ' ', $request->{arg}->[0];
|
|
my $cmd = $_[0];
|
|
if (grep { $_ eq $cmd } keys %authorized_cmds) {
|
|
my $func = $authorized_cmds{$cmd};
|
|
$func->($request, $callback, $sub_req);
|
|
}
|
|
else {
|
|
$callback->(
|
|
{ error => "$cmd is not authorized!\n", errorcode => [1] });
|
|
}
|
|
}
|
|
|
|
sub web_lsevent {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my @ret = `$request->{arg}->[0]`;
|
|
|
|
# Please refer the manpage for the output format of lsevent
|
|
my $data = [];
|
|
my $record = '';
|
|
my $i = 0;
|
|
my $j = 0;
|
|
|
|
foreach my $item (@ret) {
|
|
if ($item ne "\n") {
|
|
chomp $item;
|
|
my ($key, $value) = split("=", $item);
|
|
if ($j < 2) {
|
|
$record .= $value . ';';
|
|
}
|
|
else {
|
|
$record .= $value;
|
|
}
|
|
|
|
$j++;
|
|
if ($j == 3) {
|
|
$i++;
|
|
$j = 0;
|
|
push(@$data, $record);
|
|
$record = '';
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
$callback->({ data => $data });
|
|
}
|
|
|
|
sub web_mkcondresp {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $conditionName = $request->{arg}->[1];
|
|
my $temp = $request->{arg}->[2];
|
|
my $cmd = '';
|
|
my @resp = split(':', $temp);
|
|
|
|
# Create new associations
|
|
if (1 < length(@resp[0])) {
|
|
$cmd = substr(@resp[0], 1);
|
|
$cmd =~ s/,/ /;
|
|
$cmd = 'mkcondresp ' . $conditionName . ' ' . $cmd;
|
|
my $retInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
}
|
|
|
|
# Delete old associations
|
|
if (1 < length(@resp[1])) {
|
|
$cmd = substr(@resp[1], 1);
|
|
$cmd =~ s/,/ /;
|
|
$cmd = 'rmcondresp ' . $conditionName . ' ' . $cmd;
|
|
my $retInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
}
|
|
|
|
# There is no output for mkcondresp
|
|
$cmd = 'startcondresp ' . $conditionName;
|
|
my $refInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
$callback->({ data => "Success." });
|
|
}
|
|
|
|
sub web_startcondresp {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $conditionName = $request->{arg}->[1];
|
|
my $cmd = 'startcondresp "' . $conditionName . '"';
|
|
my $retInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
$callback->(
|
|
{ data => 'start monitor "' . $conditionName . '" Successful.' });
|
|
}
|
|
|
|
sub web_stopcondresp {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $conditionName = $request->{arg}->[1];
|
|
my $cmd = 'stopcondresp "' . $conditionName . '"';
|
|
my $retInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
$callback->(
|
|
{ data => 'stop monitor "' . $conditionName . '" Successful.' });
|
|
}
|
|
|
|
sub web_lscond {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodeRange = $request->{arg}->[1];
|
|
my $names = '';
|
|
|
|
# List all the conditions for all lpars in this group
|
|
if ($nodeRange) {
|
|
my @nodes = xCAT::NodeRange::noderange($nodeRange);
|
|
my %tempHash;
|
|
my $nodeCount = @nodes;
|
|
|
|
# No node in this group
|
|
if (1 > $nodeCount) {
|
|
return;
|
|
}
|
|
|
|
# No conditions return
|
|
my $tempCmd = 'lscondition -d :' . join(',', @nodes);
|
|
my $retInfo = xCAT::Utils->runcmd($tempCmd, -1, 1);
|
|
if (1 > @$retInfo) {
|
|
return;
|
|
}
|
|
|
|
shift @$retInfo;
|
|
shift @$retInfo;
|
|
foreach my $line (@$retInfo) {
|
|
my @temp = split(':', $line);
|
|
$tempHash{ @temp[0] }++;
|
|
}
|
|
|
|
foreach my $name (keys(%tempHash)) {
|
|
if ($nodeCount == $tempHash{$name}) {
|
|
$names = $names . $name . ';';
|
|
}
|
|
}
|
|
}
|
|
|
|
# Only list the conditions on local
|
|
else {
|
|
my $retInfo = xCAT::Utils->runcmd('lscondition -d', -1, 1);
|
|
if (2 > @$retInfo) {
|
|
return;
|
|
}
|
|
|
|
shift @$retInfo;
|
|
shift @$retInfo;
|
|
foreach my $line (@$retInfo) {
|
|
my @temp = split(':', $line);
|
|
$names = $names . @temp[0] . ':' . substr(@temp[2], 1, 3) . ';';
|
|
}
|
|
}
|
|
|
|
if ('' eq $names) {
|
|
return;
|
|
}
|
|
|
|
$names = substr($names, 0, (length($names) - 1));
|
|
$callback->({ data => $names });
|
|
}
|
|
|
|
sub web_mkcondition {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
if ('change' eq $request->{arg}->[1]) {
|
|
my @nodes;
|
|
my $conditionName = $request->{arg}->[2];
|
|
my $groupName = $request->{arg}->[3];
|
|
|
|
my $retInfo =
|
|
xCAT::Utils->runcmd('nodels ' . $groupName . " ppc.nodetype", -1,
|
|
1);
|
|
foreach my $line (@$retInfo) {
|
|
my @temp = split(':', $line);
|
|
if (@temp[1] !~ /lpar/) {
|
|
$callback->(
|
|
{
|
|
data =>
|
|
'Error : only the compute nodes\' group could select.'
|
|
}
|
|
);
|
|
return;
|
|
}
|
|
|
|
push(@nodes, @temp[0]);
|
|
}
|
|
|
|
xCAT::Utils->runcmd('chcondition -n ' + join(',', @nodes) + '-m m ' +
|
|
$conditionName);
|
|
$callback->({ data => 'Change scope success.' });
|
|
}
|
|
|
|
}
|
|
|
|
sub web_lsresp {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $names = '';
|
|
my @temp;
|
|
my $retInfo = xCAT::Utils->runcmd('lsresponse -d', -1, 1);
|
|
|
|
shift @$retInfo;
|
|
shift @$retInfo;
|
|
foreach my $line (@$retInfo) {
|
|
@temp = split(':', $line);
|
|
$names = $names . @temp[0] . ';';
|
|
}
|
|
|
|
$names = substr($names, 0, (length($names) - 1));
|
|
$callback->({ data => $names });
|
|
}
|
|
|
|
sub web_lscondresp {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $names = '';
|
|
my @temp;
|
|
|
|
# If there is a condition name, then we only show the condition linked associations
|
|
if ($request->{arg}->[1]) {
|
|
my $cmd = 'lscondresp -d ' . $request->{arg}->[1];
|
|
my $retInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
if (2 > @$retInfo) {
|
|
$callback->({ data => '' });
|
|
return;
|
|
}
|
|
|
|
shift @$retInfo;
|
|
shift @$retInfo;
|
|
for my $line (@$retInfo) {
|
|
@temp = split(':', $line);
|
|
$names = $names . @temp[1] . ';';
|
|
}
|
|
}
|
|
|
|
$names = substr($names, 0, (length($names) - 1));
|
|
$callback->({ data => $names });
|
|
}
|
|
|
|
sub web_update {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $os = "unknown";
|
|
my $rpmNames = $request->{arg}->[1];
|
|
my $repository = $request->{arg}->[2];
|
|
my $fileHandle;
|
|
my $cmd;
|
|
my $returnInfo;
|
|
my $webpageContent = undef;
|
|
my $remoteRpmFilePath = undef;
|
|
my $localRpmFilePath = undef;
|
|
|
|
if (xCAT::Utils->isLinux()) {
|
|
$os = xCAT::Utils->osver();
|
|
|
|
# SUSE Linux
|
|
if ($os =~ /sles.*/) {
|
|
$rpmNames =~ s/,/ /g;
|
|
|
|
# Create zypper command
|
|
$cmd = "zypper -n -p " . $repository . " update " . $rpmNames;
|
|
}
|
|
|
|
# Red Hat
|
|
else {
|
|
|
|
# Check the yum config file, and detect if it exists
|
|
if (-e "/tmp/xCAT_update.yum.conf") {
|
|
unlink("/tmp/xCAT_update.yum.conf");
|
|
}
|
|
|
|
# Create file, return error if failed
|
|
unless (open($fileHandle, '>>', "/tmp/xCAT_update.yum.conf")) {
|
|
$callback->(
|
|
{ error => "Created temp file error!\n", errorcode => [1] }
|
|
);
|
|
return;
|
|
}
|
|
|
|
# Write the RPM path into config file
|
|
print $fileHandle "[xcat_temp_update]\n";
|
|
print $fileHandle "name=temp prepository\n";
|
|
$repository = "baseurl=" . $repository . "\n";
|
|
print $fileHandle $repository;
|
|
print $fileHandle "enabled=1\n";
|
|
print $fileHandle "gpgcheck=0\n";
|
|
close($fileHandle);
|
|
|
|
# Use system to run the command: yum -y -c config-file update rpm-names
|
|
$rpmNames =~ s/,/ /g;
|
|
$cmd = "yum -y -c /tmp/xCAT_update.yum.conf update " . $rpmNames . " 2>&1";
|
|
}
|
|
|
|
# Run the command and return the result
|
|
$returnInfo = readpipe($cmd);
|
|
$callback->({ info => $returnInfo });
|
|
}
|
|
|
|
# AIX
|
|
else {
|
|
|
|
# Open the RPM path and read the page's content
|
|
$webpageContent = LWP::Simple::get($repository);
|
|
unless (defined($webpageContent)) {
|
|
$callback->({
|
|
error => "open $repository error, please check!!",
|
|
errorcode => [1]
|
|
});
|
|
return;
|
|
}
|
|
|
|
# Must support updating several RPM
|
|
foreach (split(/,/, $rpmNames)) {
|
|
|
|
# Find out RPMs corresponding RPM HREF on the web page
|
|
$webpageContent =~ m/href="($_-.*?[ppc64|noarch].rpm)/i;
|
|
unless (defined($1)) {
|
|
next;
|
|
}
|
|
$remoteRpmFilePath = $repository . $1;
|
|
$localRpmFilePath = '/tmp/' . $1;
|
|
|
|
# Download RPM package to temp
|
|
unless (-e $localRpmFilePath) {
|
|
$cmd = "wget -O " . $localRpmFilePath . " " . $remoteRpmFilePath;
|
|
if (0 != system($cmd)) {
|
|
$returnInfo = $returnInfo . "update " . $_ . " failed: cannot download the RPM\n";
|
|
$callback->({ error => $returnInfo, errorcode => [1] });
|
|
return;
|
|
}
|
|
}
|
|
|
|
# Update RPM by RPM packages
|
|
$cmd = "rpm -U " . $localRpmFilePath . " 2>&1";
|
|
$returnInfo = $returnInfo . readpipe($cmd);
|
|
}
|
|
|
|
$callback->({ info => $returnInfo });
|
|
}
|
|
}
|
|
|
|
sub web_unlock {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $node = $request->{arg}->[1];
|
|
my $password = $request->{arg}->[2];
|
|
|
|
# Unlock a node by setting up the SSH keys
|
|
my $out = `DSH_REMOTE_PASSWORD=$password /opt/xcat/bin/xdsh $node -K 2>&1`;
|
|
|
|
$callback->({ data => $out });
|
|
}
|
|
|
|
sub web_unlockByIP {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $node = $request->{arg}->[1];
|
|
my $password = $request->{arg}->[2];
|
|
my $ip = $request->{arg}->[3];
|
|
|
|
# Unlock a node by setting up the SSH keys
|
|
my $out = `DSH_REMOTE_PASSWORD=$password /opt/xcat/bin/xdsh $node -K --ip $ip 2>&1`;
|
|
|
|
$callback->({ data => $out });
|
|
}
|
|
|
|
sub web_unlockShow {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $node = $request->{arg}->[1];
|
|
my $show = $request->{arg}->[2];
|
|
|
|
# Unlock a node by setting up the SSH keys
|
|
my $out = `/opt/xcat/bin/xdsh $node -K --show $show 2>&1`;
|
|
|
|
$callback->({ data => $out });
|
|
}
|
|
|
|
sub web_gangliastatus {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
my $out = `/opt/xcat/bin/xdsh $nr "service gmond status"`;
|
|
|
|
# Parse output, and use $callback to send back to the web interface
|
|
# Output looks like:
|
|
# node_1: Checking for gmond: ..running
|
|
# node_2: Checking for gmond: ..running
|
|
my @lines = split('\n', $out);
|
|
my $line;
|
|
my $status;
|
|
foreach $line (@lines) {
|
|
if ($line =~ m/running/i) {
|
|
$status = 'on';
|
|
}
|
|
else {
|
|
$status = 'off';
|
|
}
|
|
|
|
@_ = split(': ', $line);
|
|
$callback->({
|
|
node => [ {
|
|
name => [ $_[0] ], # Node name
|
|
data => [$status] # Output
|
|
} ]
|
|
});
|
|
}
|
|
}
|
|
|
|
sub web_gangliaconf() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
|
|
my $info;
|
|
my $output;
|
|
|
|
# Add gangliamon to the monitoring table (if not already)
|
|
$output = `/opt/xcat/bin/monadd gangliamon`;
|
|
|
|
# Run the ganglia configuration script on node
|
|
if ($nr) {
|
|
$output = `/opt/xcat/bin/moncfg gangliamon $nr -r`;
|
|
}
|
|
else {
|
|
|
|
# If no node range is given, then assume all nodes
|
|
|
|
# Handle localhost (this needs to be 1st)
|
|
$output = `/opt/xcat/bin/moncfg gangliamon`;
|
|
|
|
# Handle remote nodes
|
|
$output .= `/opt/xcat/bin/moncfg gangliamon -r`;
|
|
}
|
|
|
|
my @lines = split('\n', $output);
|
|
foreach (@lines) {
|
|
if ($_) {
|
|
$info .= ($_ . "\n");
|
|
}
|
|
}
|
|
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_gangliastart() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
|
|
my $info;
|
|
my $output;
|
|
|
|
# Add gangliamon to the monitoring table (if not already)
|
|
$output = `/opt/xcat/bin/monadd gangliamon`;
|
|
|
|
# Start the gmond daemon on node
|
|
if ($nr) {
|
|
$output = `/opt/xcat/bin/moncfg gangliamon $nr -r`;
|
|
$output .= `/opt/xcat/bin/monstart gangliamon $nr -r`;
|
|
}
|
|
else {
|
|
|
|
# If no node range is given, then assume all nodes
|
|
|
|
# Handle localhost (this needs to be 1st)
|
|
$output = `/opt/xcat/bin/moncfg gangliamon`;
|
|
|
|
# Handle remote nodes
|
|
$output .= `/opt/xcat/bin/moncfg gangliamon -r`;
|
|
|
|
# Handle localhost (this needs to be 1st)
|
|
$output .= `/opt/xcat/bin/monstart gangliamon`;
|
|
|
|
# Handle remote nodes
|
|
$output .= `/opt/xcat/bin/monstart gangliamon -r`;
|
|
}
|
|
|
|
my @lines = split('\n', $output);
|
|
foreach (@lines) {
|
|
if ($_) {
|
|
$info .= ($_ . "\n");
|
|
}
|
|
}
|
|
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_gangliastop() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
|
|
my $info;
|
|
my $output;
|
|
|
|
# Stop the gmond daemon on node
|
|
if ($nr) {
|
|
$output = `/opt/xcat/bin/monstop gangliamon $nr -r`;
|
|
}
|
|
else {
|
|
|
|
# If no node range is given, then assume all nodes
|
|
|
|
# Handle localhost (this needs to be 1st)
|
|
$output = `/opt/xcat/bin/monstop gangliamon`;
|
|
|
|
# Handle remote nodes
|
|
$output .= `/opt/xcat/bin/monstop gangliamon -r`;
|
|
}
|
|
|
|
my @lines = split('\n', $output);
|
|
foreach (@lines) {
|
|
if ($_) {
|
|
$info .= ($_ . "\n");
|
|
}
|
|
}
|
|
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_gangliacheck() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
if (!$nr) {
|
|
$nr = '';
|
|
}
|
|
|
|
# Check if ganglia RPMs are installed
|
|
my $info;
|
|
my $info = `/opt/xcat/bin/xdsh $nr "rpm -q ganglia-gmond libganglia libconfuse"`;
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_installganglia() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get node range
|
|
my $nr = $request->{arg}->[1];
|
|
my @nodes = split(',', $nr);
|
|
|
|
# Loop through each node
|
|
my $info;
|
|
my $tab;
|
|
my $attrs;
|
|
my $osType;
|
|
my $dir;
|
|
my $pkglist;
|
|
my $defaultDir;
|
|
|
|
foreach (@nodes) {
|
|
|
|
# Get os, arch, profile, and provmethod
|
|
$tab = xCAT::Table->new('nodetype');
|
|
$attrs =
|
|
$tab->getNodeAttribs($_, [ 'os', 'arch', 'profile', 'provmethod' ]);
|
|
|
|
# If any attributes are missing, skip
|
|
if (!$attrs->{'os'}
|
|
|| !$attrs->{'arch'}
|
|
|| !$attrs->{'profile'}
|
|
|| !$attrs->{'provmethod'}) {
|
|
$callback->({ info => "$_: (Error) Missing attribute (os, arch, profile, or provmethod) in nodetype table" });
|
|
next;
|
|
}
|
|
|
|
# Get the right OS type
|
|
if ($attrs->{'os'} =~ /fedora/) {
|
|
$osType = 'fedora';
|
|
} elsif ($attrs->{'os'} =~ /rh/
|
|
|| $attrs->{'os'} =~ /rhel/
|
|
|| $attrs->{'os'} =~ /rhels/) {
|
|
$osType = 'rh';
|
|
} elsif ($attrs->{'os'} =~ /sles/) {
|
|
$osType = 'sles';
|
|
}
|
|
|
|
# Assume /install/post/otherpkgs/<os>/<arch>/ directory is created
|
|
# If Ganglia RPMs (ganglia-gmond-*, libconfuse-*, and libganglia-*) are not in directory
|
|
$dir = "/install/post/otherpkgs/$attrs->{'os'}/$attrs->{'arch'}/";
|
|
if (!(`test -e $dir/ganglia-gmond-* && echo 'File exists'`
|
|
&& `test -e $dir/libconfuse-* && echo 'File exists'`
|
|
&& `test -e $dir/libganglia-* && echo 'File exists'`
|
|
)) {
|
|
|
|
# Skip
|
|
$callback->({ info => "$_: (Error) Missing Ganglia RPMs under $dir" });
|
|
next;
|
|
}
|
|
|
|
# Find pkglist directory
|
|
$dir = "/install/custom/$attrs->{'provmethod'}/$osType";
|
|
if (!(`test -d $dir && echo 'Directory exists'`)) {
|
|
|
|
# Create pkglist directory
|
|
`mkdir -p $dir`;
|
|
}
|
|
|
|
# Find pkglist file
|
|
# Ganglia RPM names should be added to /install/custom/<inst_type>/<ostype>/<profile>.<os>.<arch>.otherpkgs.pkglist
|
|
$pkglist = "$attrs->{'profile'}.$attrs->{'os'}.$attrs->{'arch'}.otherpkgs.pkglist";
|
|
if (!(`test -e $dir/$pkglist && echo 'File exists'`)) {
|
|
|
|
# Copy default otherpkgs.pkglist
|
|
$defaultDir = "/opt/xcat/share/xcat/$attrs->{'provmethod'}/$osType";
|
|
if (`test -e $defaultDir/$pkglist && echo 'File exists'`) {
|
|
|
|
# Copy default pkglist
|
|
`cp $defaultDir/$pkglist $dir/$pkglist`;
|
|
} else {
|
|
|
|
# Create pkglist
|
|
`touch $dir/$pkglist`;
|
|
}
|
|
|
|
# Add Ganglia RPMs to pkglist
|
|
`echo ganglia-gmond >> $dir/$pkglist`;
|
|
`echo libconfuse >> $dir/$pkglist`;
|
|
`echo libganglia >> $dir/$pkglist`;
|
|
}
|
|
|
|
# Check if libapr1 is installed
|
|
$info = `xdsh $_ "rpm -qa libapr1"`;
|
|
if (!($info =~ /libapr1/)) {
|
|
$callback->(
|
|
{ info => "$_: (Error) libapr1 package not installed" });
|
|
next;
|
|
}
|
|
|
|
# Install Ganglia RPMs using updatenode
|
|
$callback->({ info => "$_: Installing Ganglia..." });
|
|
$info = `/opt/xcat/bin/updatenode $_ -S`;
|
|
$callback->({ info => "$info" });
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
sub web_gangliaShow {
|
|
|
|
# Get ganglia data from RRD file
|
|
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodename = $request->{arg}->[1];
|
|
my $timeRange = 'now-1h';
|
|
my $resolution = 60;
|
|
my $metric = $request->{arg}->[3];
|
|
my @nodes = ();
|
|
my $retStr = '';
|
|
my $runInfo;
|
|
my $cmd = '';
|
|
my $dirname = '/var/lib/ganglia/rrds/__SummaryInfo__/';
|
|
|
|
# Get the summary for this grid (the meaning of grid is referenced from Ganglia)
|
|
if ('_grid_' ne $nodename) {
|
|
$dirname = '/var/lib/ganglia/rrds/' . $nodename . '/';
|
|
}
|
|
|
|
if ('hour' eq $request->{arg}->[2]) {
|
|
$timeRange = 'now-1h';
|
|
$resolution = 60;
|
|
} elsif ('day' eq $request->{arg}->[2]) {
|
|
$timeRange = 'now-1d';
|
|
$resolution = 1800;
|
|
}
|
|
|
|
if ('_summary_' eq $metric) {
|
|
my @metricArray = (
|
|
'load_one', 'cpu_num', 'cpu_idle', 'mem_free',
|
|
'mem_total', 'disk_total', 'disk_free', 'bytes_in',
|
|
'bytes_out'
|
|
);
|
|
|
|
my $filename = '';
|
|
my $step = 1;
|
|
my $index = 0;
|
|
my $size = 0;
|
|
foreach my $tempmetric (@metricArray) {
|
|
my $temp = '';
|
|
my $line = '';
|
|
$retStr .= $tempmetric . ':';
|
|
$filename = $dirname . $tempmetric . '.rrd';
|
|
$cmd = "rrdtool fetch $filename -s $timeRange -r $resolution AVERAGE";
|
|
$runInfo = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
if (scalar(@$runInfo) < 3) {
|
|
$callback->({ data => 'error.' });
|
|
return;
|
|
}
|
|
|
|
# Delete the first 2 lines
|
|
shift(@$runInfo);
|
|
shift(@$runInfo);
|
|
|
|
# We only support 60 lines for one metric, in order to reduce the data load for web GUI
|
|
$size = scalar(@$runInfo);
|
|
if ($size > 60) {
|
|
$step = int($size / 60) + 1;
|
|
}
|
|
|
|
if (($tempmetric eq 'cpu_idle') && ('_grid_' eq $nodename)) {
|
|
my $cpuidle = 0;
|
|
my $cpunum = 0;
|
|
for ($index = 0 ; $index < $size ; $index += $step) {
|
|
if ($runInfo->[$index] =~ /^(\S+): (\S+) (\S+)/) {
|
|
my $timestamp = $1;
|
|
my $value = $2;
|
|
my $valuenum = $3;
|
|
if ((lc($value) =~ /nanq/) || (lc($value) =~ /nan/)) {
|
|
|
|
# The rrdtool fetch last line is always NaN, so no need to add into the return string
|
|
if ($index == ($size - 1)) {
|
|
next;
|
|
}
|
|
|
|
$temp .= $timestamp . ',0,';
|
|
} else {
|
|
$cpuidle = sprintf "%.2f", $value;
|
|
$cpunum = sprintf "%.2f", $valuenum;
|
|
$temp .= $timestamp . ',' . (sprintf "%.2f", $cpuidle / $cpunum) . ',';
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
for ($index = 0 ; $index < $size ; $index += $step) {
|
|
if ($runInfo->[$index] =~ /^(\S+): (\S+).*/) {
|
|
my $timestamp = $1;
|
|
my $value = $2;
|
|
if ((lc($value) =~ /nanq/) || (lc($value) =~ /nan/)) {
|
|
|
|
# The rrdtool fetch last line is always NaN, so no need to add into the return string
|
|
if ($index == ($size - 1)) {
|
|
next;
|
|
}
|
|
|
|
$temp .= $timestamp . ',0,';
|
|
} else {
|
|
$temp .= $timestamp . ',' . (sprintf "%.2f", $2) . ',';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$retStr .= substr($temp, 0, -1) . ';';
|
|
}
|
|
|
|
$retStr = substr($retStr, 0, -1);
|
|
$callback->({ data => $retStr });
|
|
return;
|
|
}
|
|
}
|
|
|
|
my $ganglia_return_flag = 0;
|
|
my %gangliaHash;
|
|
my $gangliaclustername;
|
|
my $ganglianodename;
|
|
|
|
sub web_gangliaLatest {
|
|
|
|
# Use socket to connect ganglia port to get the latest value/status
|
|
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $type = $request->{arg}->[1];
|
|
my $groupname = '';
|
|
my $xmlparser;
|
|
my $telnetcmd = '';
|
|
my $connect;
|
|
my $xmloutput = '';
|
|
my $tmpFilename = '/tmp/gangliadata';
|
|
|
|
$ganglia_return_flag = 0;
|
|
$gangliaclustername = '';
|
|
$ganglianodename = '';
|
|
undef(%gangliaHash);
|
|
|
|
if ($request->{arg}->[2]) {
|
|
$groupname = $request->{arg}->[2];
|
|
}
|
|
if ('grid' eq $type) {
|
|
$xmlparser = XML::Parser->new(
|
|
Handlers => {
|
|
Start => \&web_gangliaGridXmlStart,
|
|
End => \&web_gangliaXmlEnd
|
|
});
|
|
$telnetcmd = "/?filter=summary\n";
|
|
$tmpFilename = '/tmp/gangliagriddata';
|
|
} elsif ('node' eq $type) {
|
|
$xmlparser = XML::Parser->new(
|
|
Handlers => {
|
|
Start => \&web_gangliaNodeXmlStart,
|
|
End => \&web_gangliaXmlEnd
|
|
});
|
|
$telnetcmd = "/\n";
|
|
$tmpFilename = '/tmp/ganglianodedata';
|
|
}
|
|
|
|
# Use socket to telnet 127.0.0.1:8652 (Ganglia's interactive port)
|
|
$connect = IO::Socket::INET->new('127.0.0.1:8652');
|
|
unless ($connect) {
|
|
$callback->({ 'data' => 'error: connect local port failed.' });
|
|
return;
|
|
}
|
|
|
|
print $connect $telnetcmd;
|
|
open(TEMPFILE, '>' . $tmpFilename);
|
|
while (<$connect>) {
|
|
print TEMPFILE $_;
|
|
}
|
|
|
|
close($connect);
|
|
close(TEMPFILE);
|
|
|
|
$xmlparser->parsefile($tmpFilename);
|
|
|
|
if ('grid' eq $type) {
|
|
web_gangliaGridLatest($callback);
|
|
} elsif ('node' eq $type) {
|
|
web_gangliaNodeLatest($callback, $groupname);
|
|
}
|
|
return;
|
|
}
|
|
|
|
sub web_gangliaGridLatest {
|
|
|
|
# Create return data for grid current status
|
|
|
|
my $callback = shift;
|
|
my $retStr = '';
|
|
my $timestamp = time();
|
|
my $metricname = '';
|
|
my @metricArray = (
|
|
'load_one', 'cpu_num', 'mem_total', 'mem_free',
|
|
'disk_total', 'disk_free', 'bytes_in', 'bytes_out'
|
|
);
|
|
|
|
if ($gangliaHash{'cpu_idle'}) {
|
|
my $sum = $gangliaHash{'cpu_idle'}->{'SUM'};
|
|
my $num = $gangliaHash{'cpu_idle'}->{'NUM'};
|
|
$retStr .= 'cpu_idle:'
|
|
. $timestamp . ','
|
|
. (sprintf("%.2f", $sum / $num)) . ';';
|
|
}
|
|
|
|
foreach $metricname (@metricArray) {
|
|
if ($gangliaHash{$metricname}) {
|
|
$retStr .=
|
|
$metricname . ':'
|
|
. $timestamp . ','
|
|
. $gangliaHash{$metricname}->{'SUM'} . ';';
|
|
}
|
|
}
|
|
|
|
$retStr = substr($retStr, 0, -1);
|
|
$callback->({ data => $retStr });
|
|
}
|
|
|
|
sub web_gangliaNodeLatest {
|
|
|
|
# Create return data for node current status
|
|
|
|
my ($callback, $groupname) = @_;
|
|
my $node = '';
|
|
my $retStr = '';
|
|
my $timestamp = time() - 180;
|
|
my @nodes;
|
|
|
|
# Get all nodes by group
|
|
if ($groupname) {
|
|
@nodes = xCAT::NodeRange::noderange($groupname, 1);
|
|
} else {
|
|
@nodes = xCAT::DBobjUtils->getObjectsOfType('node');
|
|
}
|
|
|
|
foreach $node (@nodes) {
|
|
|
|
# If the node has Ganglia
|
|
if ($gangliaHash{$node}) {
|
|
my $lastupdate = $gangliaHash{$node}->{'timestamp'};
|
|
|
|
# Cannot get monitor data for too long
|
|
if ($lastupdate < $timestamp) {
|
|
$retStr .= $node . ':ERROR,Can not get monitor data more than 3 minutes!;';
|
|
next;
|
|
}
|
|
|
|
if ($gangliaHash{$node}->{'load_one'} >
|
|
$gangliaHash{$node}->{'cpu_num'}) {
|
|
$retStr .= $node . ':WARNING,';
|
|
} else {
|
|
$retStr .= $node . ':NORMAL,';
|
|
}
|
|
|
|
$retStr .= $gangliaHash{$node}->{'path'} . ';';
|
|
} else {
|
|
$retStr .= $node . ':UNKNOWN,;';
|
|
}
|
|
}
|
|
|
|
$retStr = substr($retStr, 0, -1);
|
|
$callback->({ data => $retStr });
|
|
}
|
|
|
|
sub web_gangliaXmlEnd {
|
|
|
|
# XML parser end function, do noting here
|
|
}
|
|
|
|
sub web_gangliaGridXmlStart {
|
|
|
|
# XML parser start function
|
|
|
|
my ($parseinst, $elementname, %attrs) = @_;
|
|
my $metricname = '';
|
|
|
|
# Only parse grid information
|
|
if ($ganglia_return_flag) {
|
|
return;
|
|
}
|
|
|
|
if ('METRICS' eq $elementname) {
|
|
$metricname = $attrs{'NAME'};
|
|
$gangliaHash{$metricname}->{'SUM'} = $attrs{'SUM'};
|
|
$gangliaHash{$metricname}->{'NUM'} = $attrs{'NUM'};
|
|
} elsif ('CLUSTER' eq $elementname) {
|
|
$ganglia_return_flag = 1;
|
|
return;
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
|
|
sub web_gangliaNodeXmlStart {
|
|
|
|
# XML parser start function for node current status
|
|
|
|
my ($parseinst, $elementname, %attrs) = @_;
|
|
my $metricname = '';
|
|
|
|
# Save cluster name
|
|
if ('CLUSTER' eq $elementname) {
|
|
$gangliaclustername = $attrs{'NAME'};
|
|
return;
|
|
} elsif ('HOST' eq $elementname) {
|
|
if ($attrs{'NAME'} =~ /(\S+?)\.(.*)/) {
|
|
$ganglianodename = $1;
|
|
} else {
|
|
$ganglianodename = $attrs{'NAME'};
|
|
}
|
|
|
|
$gangliaHash{$ganglianodename}->{'path'} =
|
|
$gangliaclustername . '/' . $attrs{'NAME'};
|
|
$gangliaHash{$ganglianodename}->{'timestamp'} = $attrs{'REPORTED'};
|
|
} elsif ('METRIC' eq $elementname) {
|
|
$metricname = $attrs{'NAME'};
|
|
if (('load_one' eq $metricname) || ('cpu_num' eq $metricname)) {
|
|
$gangliaHash{$ganglianodename}->{$metricname} = $attrs{'VAL'};
|
|
}
|
|
}
|
|
}
|
|
|
|
sub web_rmcmonStart {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodeRange = $request->{arg}->[1];
|
|
my $table;
|
|
my $retData = "";
|
|
my $output;
|
|
|
|
# Check running status
|
|
$table = xCAT::Table->new('monitoring');
|
|
my $rmcWorkingStatus = $table->getAttribs({ name => 'rmcmon' }, 'disable');
|
|
$table . close();
|
|
|
|
# RMC monitoring is running so return
|
|
if ($rmcWorkingStatus) {
|
|
if ($rmcWorkingStatus->{disable} =~ /0|No|no|NO|N|n/) {
|
|
$callback->({ info => 'RMC Monitoring is running now.' });
|
|
return;
|
|
}
|
|
}
|
|
|
|
$retData .= "RMC is not running, start it now.\n";
|
|
|
|
# Check monsetting table to see if rmc's montype contains performance
|
|
$table = xCAT::Table->new('monsetting');
|
|
my $rmcmonType =
|
|
$table->getAttribs({ name => 'rmcmon', key => 'montype' }, 'value');
|
|
$table . close();
|
|
|
|
# RMC monitoring is not configured right, we should configure it again
|
|
# There is no rmcmon in monsetting table
|
|
if (!$rmcmonType) {
|
|
$output = xCAT::Utils->runcmd('monadd rmcmon -s [montype=perf]', -1, 1);
|
|
foreach (@$output) {
|
|
$retData .= ($_ . "\n");
|
|
}
|
|
|
|
$retData .= "Adding rmcmon to the monsetting table complete.\n";
|
|
}
|
|
|
|
# Configure before but there is no performance monitoring, so change the table
|
|
else {
|
|
if (!($rmcmonType->{value} =~ /perf/)) {
|
|
$output = xCAT::Utils->runcmd('chtab name=rmcmon,key=montype monsetting.value=perf', -1, 1);
|
|
foreach (@$output) {
|
|
$retData .= ($_ . "\n");
|
|
}
|
|
|
|
$retData .= "Change the rmcmon configure in monsetting table finish.\n";
|
|
}
|
|
}
|
|
|
|
# Run the rmccfg command to add all nodes into local RMC configuration
|
|
$output = xCAT::Utils->runcmd("moncfg rmcmon $nodeRange", -1, 1);
|
|
foreach (@$output) {
|
|
$retData .= ($_ . "\n");
|
|
}
|
|
|
|
# Run the rmccfg command to add all nodes into remote RMC configuration
|
|
$output = xCAT::Utils->runcmd("moncfg rmcmon $nodeRange -r", -1, 1);
|
|
foreach (@$output) {
|
|
$retData .= ($_ . "\n");
|
|
}
|
|
|
|
# Start the RMC monitor
|
|
$output = xCAT::Utils->runcmd("monstart rmcmon", -1, 1);
|
|
foreach (@$output) {
|
|
$retData .= ($_ . "\n");
|
|
}
|
|
|
|
$callback->({ info => $retData });
|
|
return;
|
|
}
|
|
|
|
sub web_rmcmonShow() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodeRange = $request->{arg}->[1];
|
|
my $attr = $request->{arg}->[2];
|
|
my @nodes;
|
|
my $retInfo;
|
|
my $retHash = {};
|
|
my $output;
|
|
my $temp = "";
|
|
|
|
# Only get the system RMC info
|
|
if ('summary' eq $nodeRange) {
|
|
$output = xCAT::Utils->runcmd("monshow rmcmon -s -t 60 -o p -a " . $attr, -1, 1);
|
|
foreach $temp (@$output) {
|
|
|
|
# The attribute name
|
|
if ($temp =~ /Pct/) {
|
|
$temp =~ s/ //g;
|
|
|
|
# The first one
|
|
if ("" eq $retInfo) {
|
|
$retInfo .= ($temp . ':');
|
|
} else {
|
|
$retInfo =~ s/,$/;/;
|
|
$retInfo .= ($temp . ':');
|
|
}
|
|
|
|
next;
|
|
}
|
|
|
|
# The content of the attribute
|
|
$temp =~ m/\s+(\d+\.\d{4})/;
|
|
if (defined($1)) {
|
|
$retInfo .= ($1 . ',');
|
|
}
|
|
}
|
|
|
|
# Return the RMC info
|
|
$retInfo =~ s/,$//;
|
|
$callback->({ info => $retInfo });
|
|
return;
|
|
}
|
|
|
|
if ('compute' eq $nodeRange) {
|
|
my $node;
|
|
|
|
@nodes = xCAT::NodeRange::noderange($nodeRange);
|
|
for $node (@nodes) {
|
|
if (-e "/var/rrd/$node") {
|
|
push(@{ $retHash->{node} }, { name => $node, data => 'OK' });
|
|
} else {
|
|
push(@{ $retHash->{node} }, { name => $node, data => 'UNKNOWN' });
|
|
}
|
|
}
|
|
|
|
$callback->($retHash);
|
|
return;
|
|
}
|
|
|
|
my $attrName = "";
|
|
my @attrs = split(/,/, $attr);
|
|
for $attrName (@attrs) {
|
|
my @attrValue = ();
|
|
$output = xCAT::Utils->runcmd("rrdtool fetch /var/rrd/${nodeRange}/${attrName}.rrd -r 60 -s e-1h AVERAGE", -1, 1);
|
|
foreach (@$output) {
|
|
$temp = $_;
|
|
if ($temp eq '') {
|
|
next;
|
|
}
|
|
|
|
if (lc($temp) =~ /[nanq|nan]/) {
|
|
next;
|
|
}
|
|
|
|
if ($temp =~ /^(\d+): (\S+) (\S+)/) {
|
|
push(@attrValue, (sprintf "%.2f", $2));
|
|
}
|
|
}
|
|
|
|
if (scalar(@attrValue) > 1) {
|
|
push(@{ $retHash->{node} }, { name => $attrName, data => join(',', @attrValue) });
|
|
} else {
|
|
$retHash->{node} = { name => $attrName, data => '' };
|
|
last;
|
|
}
|
|
}
|
|
|
|
$callback->($retHash);
|
|
}
|
|
|
|
sub web_monls() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $retInfo = xCAT::Utils->runcmd("monls", -1, 1);
|
|
my $ret = '';
|
|
foreach my $line (@$retInfo) {
|
|
my @temp = split(/\s+/, $line);
|
|
$ret .= @temp[0];
|
|
if ('not-monitored' eq @temp[1]) {
|
|
$ret .= ':Off;';
|
|
} else {
|
|
$ret .= ':On;';
|
|
}
|
|
}
|
|
|
|
if ('' eq $ret) {
|
|
return;
|
|
}
|
|
|
|
$ret = substr($ret, 0, length($ret) - 1);
|
|
$callback->({ data => $ret });
|
|
}
|
|
|
|
sub web_dynamiciprange {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $iprange = $request->{arg}->[1];
|
|
|
|
open(TEMPFILE, '>/tmp/iprange.conf');
|
|
print TEMPFILE "xcat-service-lan:\n";
|
|
print TEMPFILE "dhcp-dynamic-range = " . $iprange . "\n";
|
|
close(TEMPFILE);
|
|
|
|
# Run xcatsetup command to change the dynamic IP range
|
|
xCAT::Utils->runcmd("xcatsetup /tmp/iprange.conf", -1, 1);
|
|
unlink('/tmp/iprange.conf');
|
|
xCAT::Utils->runcmd("makedhcp -n", -1, 1);
|
|
|
|
# Restart the DHCP server
|
|
if (xCAT::Utils->isLinux()) {
|
|
|
|
# xCAT::Utils->runcmd("service dhcpd restart", -1, 1);
|
|
} else {
|
|
|
|
# xCAT::Utils->runcmd("startsrc -s dhcpsd", -1, 1);
|
|
}
|
|
}
|
|
|
|
sub web_discover {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $type = uc($request->{arg}->[1]);
|
|
|
|
my $retStr = '';
|
|
my $retInfo = xCAT::Utils->runcmd("lsslp -m -s $type 2>/dev/null | grep -i $type | awk '{print \$1\":\" \$2\"-\"\$3}'", -1, 1);
|
|
if (scalar(@$retInfo) < 1) {
|
|
$retStr = 'Error: Can not discover frames in cluster!';
|
|
} else {
|
|
foreach my $line (@$retInfo) {
|
|
$retStr .= $line . ';';
|
|
}
|
|
|
|
$retStr = substr($retStr, 0, -1);
|
|
}
|
|
|
|
$callback->({ data => $retStr });
|
|
}
|
|
|
|
sub web_updatevpd {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $harwareMtmsPair = $request->{arg}->[1];
|
|
my @hardware = split(/:/, $harwareMtmsPair);
|
|
|
|
my $vpdtab = xCAT::Table->new('vpd');
|
|
unless ($vpdtab) {
|
|
return;
|
|
}
|
|
|
|
foreach my $hard (@hardware) {
|
|
|
|
# The sequence must be object name, mtm, serial
|
|
my @temp = split(/,/, $hard);
|
|
$vpdtab->setAttribs({ 'node' => @temp[0] }, { 'serial' => @temp[2], 'mtm' => @temp[1] });
|
|
}
|
|
|
|
$vpdtab->close();
|
|
}
|
|
|
|
sub web_writeconfigfile {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $filename = $request->{arg}->[1];
|
|
my $content = $request->{arg}->[2];
|
|
|
|
open(TEMPFILE, '>' . $filename);
|
|
print TEMPFILE $content;
|
|
|
|
close(TEMPFILE);
|
|
return;
|
|
}
|
|
|
|
sub web_createimage {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $ostype = $request->{arg}->[1];
|
|
my $osarch = lc($request->{arg}->[2]);
|
|
my $profile = $request->{arg}->[3];
|
|
my $bootif = $request->{arg}->[4];
|
|
my $imagetype = lc($request->{arg}->[5]);
|
|
my @softArray;
|
|
my $netdriver = '';
|
|
my $installdir = xCAT::TableUtils->getInstallDir();
|
|
my $tempos = $ostype;
|
|
$tempos =~ s/[0-9\.]//g;
|
|
my $CONFILE;
|
|
my $archFlag = 0;
|
|
my $ret = '';
|
|
my $cmdPath = '';
|
|
|
|
if ($request->{arg}->[6]) {
|
|
@softArray = split(',', $request->{arg}->[6]);
|
|
|
|
# Check the custom package, if the directory does not exist, create the directory first
|
|
if (-e "$installdir/custom/netboot/$ostype/") {
|
|
|
|
# The path exist, so archive all file under this path
|
|
opendir(TEMPDIR, "$installdir/custom/netboot/$ostype/");
|
|
my @fileArray = readdir(TEMPDIR);
|
|
closedir(TEMPDIR);
|
|
if (2 < scalar(@fileArray)) {
|
|
$archFlag = 1;
|
|
unless (-e "/tmp/webImageArch/") {
|
|
system("mkdir -p /tmp/webImageArch/");
|
|
}
|
|
|
|
system("mv $installdir/custom/netboot/$ostype/*.* /tmp/webImageArch/");
|
|
} else {
|
|
$archFlag = 0;
|
|
}
|
|
} else {
|
|
|
|
# No need to archive
|
|
$archFlag = 0;
|
|
system("mkdir -p $installdir/custom/netboot/$ostype/");
|
|
}
|
|
|
|
# Write pkglist
|
|
open($CONFILE, ">$installdir/custom/netboot/$ostype/$profile.pkglist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/IBMhpc.$ostype.ppc64.pkglist# \n";
|
|
close($CONFILE);
|
|
|
|
# Write otherpkglist
|
|
open($CONFILE, ">$installdir/custom/netboot/$ostype/$profile.otherpkgs.pkglist");
|
|
print $CONFILE "\n";
|
|
close($CONFILE);
|
|
|
|
# Write exlist for stateless
|
|
open($CONFILE, ">$installdir/custom/netboot/$ostype/$profile.exlist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/IBMhpc.$ostype.$osarch.exlist#\n";
|
|
close($CONFILE);
|
|
|
|
# Write postinstall
|
|
open($CONFILE, ">$installdir/custom/netboot/$ostype/$profile.postinstall");
|
|
print $CONFILE "/opt/xcat/share/xcat/IBMhpc/IBMhpc.$tempos.postinstall \$1 \$2 \$3 \$4 \$5 \n";
|
|
close($CONFILE);
|
|
|
|
for my $soft (@softArray) {
|
|
$soft = lc($soft);
|
|
if ('gpfs' eq $soft) {
|
|
web_gpfsConfigure($ostype, $profile, $osarch, $installdir);
|
|
} elsif ('rsct' eq $soft) {
|
|
web_rsctConfigure($ostype, $profile, $osarch, $installdir);
|
|
} elsif ('pe' eq $soft) {
|
|
web_peConfigure($ostype, $profile, $osarch, $installdir);
|
|
} elsif ('essl' eq $soft) {
|
|
web_esslConfigure($ostype, $profile, $osarch, $installdir);
|
|
} elsif ('ganglia' eq $soft) {
|
|
web_gangliaConfig($ostype, $profile, $osarch, 'netboot', $installdir);
|
|
}
|
|
}
|
|
|
|
system("chmod 755 $installdir/custom/netboot/$ostype/*.*");
|
|
}
|
|
|
|
if ($bootif =~ /hf/i) {
|
|
$netdriver = 'hf_if';
|
|
} else {
|
|
$netdriver = 'ibmveth';
|
|
}
|
|
|
|
if ($tempos =~ /rh/i) {
|
|
$cmdPath = "/opt/xcat/share/xcat/netboot/rh";
|
|
} else {
|
|
$cmdPath = "/opt/xcat/share/xcat/netboot/sles";
|
|
}
|
|
|
|
# For stateless only run packimage
|
|
if ('stateless' eq $imagetype) {
|
|
my $retInfo = xCAT::Utils->runcmd("${cmdPath}/genimage -i $bootif -n $netdriver -o $ostype -p $profile", -1, 1);
|
|
$ret = join("\n", @$retInfo);
|
|
|
|
if ($::RUNCMD_RC) {
|
|
web_restoreChange($request->{arg}->[6], $archFlag, $imagetype, $ostype, $installdir);
|
|
$callback->({ data => $ret });
|
|
return;
|
|
}
|
|
|
|
$ret .= "\n";
|
|
my $retInfo = xCAT::Utils->runcmd("packimage -o $ostype -p $profile -a $osarch", -1, 1);
|
|
$ret .= join("\n", @$retInfo);
|
|
} else {
|
|
|
|
# For statelist we should check the litefile table
|
|
# Step 1: Save the old litefile table content into litefilearchive.csv
|
|
system('tabdump litefile > /tmp/litefilearchive.csv');
|
|
|
|
# Step 2: Write the new litefile.csv for this lite image
|
|
open($CONFILE, ">/tmp/litefile.csv");
|
|
print $CONFILE "#image,file,options,comments,disable\n";
|
|
print $CONFILE '"ALL","/etc/lvm/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/ntp.conf","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/resolv.conf","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/sysconfig/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/yp.conf","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/ssh/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/var/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/tmp/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/root/.ssh/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/opt/xcat/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/xcatpost/","tmpfs",,' . "\n";
|
|
|
|
if ('rhels' eq $tempos) {
|
|
print $CONFILE '"ALL","/etc/adjtime","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/securetty","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/rsyslog.conf","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/rsyslog.conf.XCATORIG","tmpfs",,'
|
|
. "\n";
|
|
print $CONFILE '"ALL","/etc/udev/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/ntp.conf.predhclient","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/resolv.conf.predhclient","tmpfs",,'
|
|
. "\n";
|
|
} else {
|
|
print $CONFILE '"ALL","/etc/ntp.conf.org","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/syslog-ng/","tmpfs",,' . "\n";
|
|
print $CONFILE '"ALL","/etc/fstab","tmpfs",,' . "\n";
|
|
}
|
|
close($CONFILE);
|
|
|
|
# Write the HPC software litefile into temp litefile.csv
|
|
for my $soft (@softArray) {
|
|
$soft = lc($soft);
|
|
if (-e "/opt/xcat/share/xcat/IBMhpc/$soft/litefile.csv") {
|
|
system("grep '^[^#]' /opt/xcat/share/xcat/IBMhpc/$soft/litefile.csv >> /tmp/litefile.csv");
|
|
}
|
|
}
|
|
|
|
system("tabrestore /tmp/litefile.csv");
|
|
|
|
# Create the image
|
|
my $retInfo = xCAT::Utils->runcmd("${cmdPath}/genimage -i $bootif -n $netdriver -o $ostype -p $profile", -1, 1);
|
|
$ret = join("\n", @$retInfo);
|
|
if ($::RUNCMD_RC) {
|
|
web_restoreChange($request->{arg}->[6], $archFlag, $imagetype, $ostype, $installdir);
|
|
$callback->({ data => $ret });
|
|
return;
|
|
}
|
|
|
|
$ret .= "\n";
|
|
my $retInfo = xCAT::Utils->runcmd("liteimg -o $ostype -p $profile -a $osarch", -1, 1);
|
|
$ret .= join("\n", @$retInfo);
|
|
}
|
|
|
|
web_restoreChange($request->{arg}->[6], $archFlag, $imagetype, $ostype, $installdir);
|
|
$callback->({ data => $ret });
|
|
return;
|
|
}
|
|
|
|
sub web_gpfsConfigure {
|
|
my ($ostype, $profile, $osarch, $installdir) = @_;
|
|
my $CONFILE;
|
|
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/gpfs");
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.otherpkgs.pkglist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/gpfs/gpfs.otherpkgs.pkglist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.exlist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/gpfs/gpfs.exlist#\n";
|
|
close($CONFILE);
|
|
|
|
system('cp /opt/xcat/share/xcat/IBMhpc/gpfs/gpfs_mmsdrfs $installdir/postscripts/gpfs_mmsdrfs');
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.postinstall");
|
|
print $CONFILE "NODESETSTATE=genimage installroot=\$1 /opt/xcat/share/xcat/IBMhpc/gpfs/gpfs_updates\n";
|
|
print $CONFILE "installroot=\$1 $installdir/postscripts/gpfs_mmsdrfs\n";
|
|
close($CONFILE);
|
|
}
|
|
|
|
sub web_rsctConfigure {
|
|
my ($ostype, $profile, $osarch, $installdir) = @_;
|
|
my $CONFILE;
|
|
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/rsct");
|
|
|
|
if ($ostype =~ /sles/i) {
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.pkglist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/rsct/rsct.pkglist# \n";
|
|
close($CONFILE);
|
|
}
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.exlist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/rsct/rsct.exlist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.postinstall");
|
|
print $CONFILE "installroot=\$1 rsctdir=$installdir/post/otherpkgs/rhels6/ppc64/rsct NODESETSTATE=genimage /opt/xcat/share/xcat/IBMhpc/rsct/rsct_install\n";
|
|
close($CONFILE);
|
|
}
|
|
|
|
sub web_peConfigure {
|
|
my ($ostype, $profile, $osarch, $installdir) = @_;
|
|
my $CONFILE;
|
|
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/pe");
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/compilers");
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.pkglist");
|
|
if ($ostype =~ /rh/i) {
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/pe/pe.$ostype.pkglist#\n";
|
|
} else {
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/compilers/compilers.pkglist#\n";
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/pe/pe.pkglist#\n";
|
|
}
|
|
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.otherpkgs.pkglist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/pe/pe.otherpkgs.pkglist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.exlist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/compilers/compilers.exlist#\n";
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/pe/pe.exlist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.postinstall");
|
|
print $CONFILE "installroot=\$1 NODESETSTATE=genimage /opt/xcat/share/xcat/IBMhpc/compilers/compilers_license";
|
|
print $CONFILE "installroot=\$1 pedir=$installdir/post/otherpkgs/rhels6/ppc64/pe NODESETSTATE=genimage /opt/xcat/share/xcat/IBMhpc/pe/pe_install";
|
|
close($CONFILE);
|
|
}
|
|
|
|
sub web_esslConfigure {
|
|
my ($ostype, $profile, $osarch, $installdir) = @_;
|
|
my $CONFILE;
|
|
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/essl");
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.pkglist");
|
|
if ($ostype =~ /rh/i) {
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/compilers/compilers.rhels6.pkglist#\n";
|
|
} else {
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/essl/essl.pkglist#\n";
|
|
}
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.otherpkgs.pkglist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/essl/essl.otherpkgs.pkglist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.exlist");
|
|
print $CONFILE "#INCLUDE:/opt/xcat/share/xcat/IBMhpc/essl/essl.exlist#\n";
|
|
close($CONFILE);
|
|
|
|
open($CONFILE, ">>$installdir/custom/netboot/$ostype/$profile.postinstall");
|
|
print $CONFILE, "installroot=\$1 essldir=$installdir/post/otherpkgs/rhels6/ppc64/essl NODESETSTATE=genimage /opt/xcat/share/xcat/IBMhpc/essl/essl_install";
|
|
close($CONFILE);
|
|
}
|
|
|
|
sub web_gangliaConfig {
|
|
my ($ostype, $profile, $osarch, $provtype, $installdir) = @_;
|
|
my $CONFILE;
|
|
|
|
system("createrepo $installdir/post/otherpkgs/$ostype/$osarch/ganglia");
|
|
|
|
open($CONFILE, ">>$installdir/custom/$provtype/$ostype/$profile.otherpkgs.pkglist");
|
|
print $CONFILE "#created by xCAT Web Gui.\n";
|
|
print $CONFILE "ganglia/ganglia\n";
|
|
print $CONFILE "ganglia/ganglia-gmond\n";
|
|
print $CONFILE "ganglia/ganglia-gmetad\n";
|
|
print $CONFILE "ganglia/rrdtool\n";
|
|
close($CONFILE);
|
|
}
|
|
|
|
sub web_gangliaRpmCheck {
|
|
my ($ostype, $profile, $osarch, $installdir) = @_;
|
|
my @rpmnames = ("rrdtool", "ganglia", "ganglia-gmond", "ganglia-gmetad");
|
|
my %temphash;
|
|
my $rpmdir = "$installdir/post/otherpkgs/$ostype/$osarch/ganglia";
|
|
my $errorstr = '';
|
|
unless (-e $rpmdir) {
|
|
return "Put rrdtool,ganglia,ganglia-gmond,ganglia-gmetad rpms into $rpmdir.";
|
|
}
|
|
|
|
opendir(DIRHANDLE, $rpmdir);
|
|
foreach my $filename (readdir(DIRHANDLE)) {
|
|
if ($filename =~ /(\D+)-(\d+)\..*\.rpm$/) {
|
|
$temphash{$1} = 1;
|
|
}
|
|
}
|
|
closedir(DIRHANDLE);
|
|
|
|
# Check if all RPMs are in the array
|
|
foreach (@rpmnames) {
|
|
unless ($temphash{$_}) {
|
|
$errorstr .= $_ . ',';
|
|
}
|
|
}
|
|
|
|
if ($errorstr) {
|
|
$errorstr = substr($errorstr, 0, -1);
|
|
return "Put $errorstr rpms into $rpmdir.";
|
|
} else {
|
|
return "";
|
|
}
|
|
}
|
|
|
|
sub web_restoreChange {
|
|
my ($software, $archFlag, $imagetype, $ostype, $installdir) = @_;
|
|
|
|
# Recover all file in the $installdir/custom/netboot/$ostype/
|
|
if ($software) {
|
|
system("rm -f $installdir/custom/netboot/$ostype/*.*");
|
|
}
|
|
|
|
if ($archFlag) {
|
|
system("mv /tmp/webImageArch/*.* $installdir/custom/netboot/$ostype/");
|
|
}
|
|
|
|
# Recover the litefile table for statelite image
|
|
if ('statelite' == $imagetype) {
|
|
system("rm -r /tmp/litefile.csv ; mv /tmp/litefilearchive.csv /tmp/litefile.csv ; tabrestore /tmp/litefile.csv");
|
|
}
|
|
}
|
|
|
|
sub web_provision_preinstall {
|
|
my ($ostype, $profile, $arch, $installdir, $softwarenames) = @_;
|
|
my $checkresult = '';
|
|
my $errorstr = '';
|
|
my @software = split(',', $softwarenames);
|
|
my $softwarenum = scalar(@software);
|
|
|
|
if (-e "$installdir/custom/install/$ostype/") {
|
|
opendir(DIRHANDLE, "$installdir/custom/install/$ostype/");
|
|
foreach my $filename (readdir(DIRHANDLE)) {
|
|
if ('.' eq $filename || '..' eq $filename) {
|
|
next;
|
|
}
|
|
|
|
$filename = "$installdir/custom/install/$ostype/" . $filename;
|
|
if ($filename =~ /(.*)\.guibak$/) {
|
|
if ($softwarenum < 1) {
|
|
system("mv $filename $1");
|
|
}
|
|
next;
|
|
}
|
|
|
|
`/bin/grep 'xCAT Web Gui' $filename`;
|
|
if ($?) {
|
|
|
|
# Backup the original config file
|
|
if ($softwarenum > 0) {
|
|
system("mv $filename ${filename}.guibak");
|
|
}
|
|
} else {
|
|
unlink($filename);
|
|
}
|
|
}
|
|
|
|
closedir(DIRHANDLE);
|
|
} else {
|
|
`mkdir -p $installdir/custom/install/$ostype -m 0755`;
|
|
}
|
|
|
|
if ($softwarenum < 1) {
|
|
return '';
|
|
}
|
|
|
|
foreach (@software) {
|
|
if ('ganglia' eq $_) {
|
|
$checkresult = web_gangliaRpmCheck($ostype, $profile, $arch, $installdir);
|
|
}
|
|
|
|
if ($checkresult) {
|
|
$errorstr .= $checkresult . "\n";
|
|
}
|
|
}
|
|
|
|
if ($errorstr) {
|
|
return $errorstr;
|
|
}
|
|
|
|
foreach (@software) {
|
|
if ('ganglia' eq $_) {
|
|
web_gangliaConfig($ostype, $profile, $arch, 'install', $installdir);
|
|
}
|
|
}
|
|
return '';
|
|
}
|
|
|
|
sub web_provision {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodes = $request->{arg}->[1];
|
|
my $imageName = $request->{arg}->[2];
|
|
my ($arch, $inic, $pnic, $master, $tftp, $nfs) = split(/,/, $request->{arg}->[3]);
|
|
my $line = '';
|
|
my %imageattr;
|
|
my $retinfo = xCAT::Utils->runcmd("lsdef -t osimage -l $imageName", -1, 1);
|
|
my $installdir = xCAT::TableUtils->getInstallDir();
|
|
|
|
# Parse output, get the OS name and type
|
|
foreach $line (@$retinfo) {
|
|
if ($line =~ /(\w+)=(\S*)/) {
|
|
$imageattr{$1} = $2;
|
|
}
|
|
}
|
|
|
|
# Check the output
|
|
unless ($imageattr{'osname'}) {
|
|
web_infomsg("Image infomation error. Check the image first.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
if ('install' eq $imageattr{'provmethod'}) {
|
|
my $prepareinfo = web_provision_preinstall($imageattr{'osvers'}, $imageattr{'profile'}, $arch, $installdir, $request->{arg}->[4]);
|
|
if ($prepareinfo) {
|
|
web_infomsg("$prepareinfo \nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if ($imageattr{'osname'} =~ /aix/i) {
|
|
web_provisionaix($nodes, $imageName, $imageattr{'nimtype'}, $inic, $pnic, $master, $tftp, $nfs, $callback);
|
|
} else {
|
|
web_provisionlinux(
|
|
$nodes, $arch,
|
|
$imageattr{'osvers'}, $imageattr{'provmethod'},
|
|
$imageattr{'profile'}, $inic,
|
|
$pnic, $master,
|
|
$tftp, $nfs,
|
|
$callback
|
|
);
|
|
}
|
|
}
|
|
|
|
sub web_provisionlinux {
|
|
my ($nodes, $arch, $os, $provmethod, $profile, $inic, $pnic, $master, $tftp, $nfs, $callback) = @_;
|
|
my $outputMessage = '';
|
|
my $retvalue = 0;
|
|
my $netboot = '';
|
|
|
|
if ($arch =~ /ppc/i) {
|
|
$netboot = 'yaboot';
|
|
} elsif ($arch =~ /x.*86/i) {
|
|
$netboot = 'xnba';
|
|
}
|
|
|
|
$outputMessage =
|
|
"Do provison : $nodes \n"
|
|
. " Arch:$arch\n OS:$os\n Provision:$provmethod\n Profile:$profile\n Install NIC:$inic\n Primary NIC:$pnic\n"
|
|
. " xCAT Master:$master\n TFTP Server:$tftp\n NFS Server:$nfs\n Netboot:$netboot\n";
|
|
|
|
web_infomsg($outputMessage, $callback);
|
|
|
|
# Change the node attribute
|
|
my $cmd = "chdef -t node -o $nodes arch=$arch os=$os provmethod=$provmethod profile=$profile installnic=$inic tftpserver=$tftp nfsserver=$nfs netboot=$netboot" . " xcatmaster=$master primarynic=$pnic";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Configure nodes' attributes error.\nProvision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
$cmd = "makedhcp $nodes";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Make DHCP error.\nProvision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
# Restart DHCP
|
|
$cmd = "service dhcpd restart";
|
|
web_runcmd($cmd, $callback);
|
|
|
|
# Conserver
|
|
$cmd = "makeconservercf $nodes";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Configure conserver error.\nProvision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
# For system x, should configure boot sequence first
|
|
if ($arch =~ /x.*86/i) {
|
|
$cmd = "rbootseq $nodes net,hd";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Set boot sequence error.\nProvision stop.",
|
|
$callback);
|
|
return;
|
|
}
|
|
}
|
|
|
|
# Nodeset
|
|
$cmd = "nodeset $nodes $provmethod";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) { web_infomsg("Set nodes provision method error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
# Reboot the node fro provision
|
|
if ($arch =~ /ppc/i) {
|
|
$cmd = "rnetboot $nodes";
|
|
} else {
|
|
$cmd = "rpower $nodes boot";
|
|
}
|
|
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Boot nodes error.\nProvision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
# Provision complete
|
|
web_infomsg("Provision on $nodes success.\nProvision stop.");
|
|
}
|
|
|
|
sub web_provisionaix {
|
|
my (
|
|
$nodes, $imagename, $nimtype, $inic, $pnic,
|
|
$master, $tftp, $nfs, $callback
|
|
) = @_;
|
|
my $outputMessage = '';
|
|
my $retinfo;
|
|
my %nimhash;
|
|
my $line;
|
|
my @updatenodes;
|
|
my @addnodes;
|
|
my $cmd = '';
|
|
|
|
# Set attributes
|
|
$cmd = "chdef -t node -o $nodes installnic=$inic tftpserver=$tftp nfsserver=$nfs xcatmaster=$master primarynic=$pnic";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Change nodes' attributes error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
# Get all NIM resource to filter nodes
|
|
$retinfo = xCAT::Utils->runcmd("lsnim -c machines", -1, 1);
|
|
foreach $line (@$retinfo) {
|
|
if ($line =~ /(\S+)\s+\S+/) {
|
|
$nimhash{$1} = 1;
|
|
}
|
|
}
|
|
|
|
foreach my $node (split(/,/, $nodes)) {
|
|
if ($nimhash{$node}) {
|
|
push(@updatenodes, $node);
|
|
} else {
|
|
push(@addnodes, $node);
|
|
}
|
|
}
|
|
|
|
if (0 < scalar(@addnodes)) {
|
|
$cmd = "xcat2nim -t node -o " . join(",", @addnodes);
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("xcat2nim command error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (0 < scalar(@updatenodes)) {
|
|
$cmd = "xcat2nim -u -t node -o " . join(",", @updatenodes);
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("xcat2nim command error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
}
|
|
|
|
$cmd = "makeconservercf $nodes";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Configure conserver error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
if ($nimtype =~ /diskless/) {
|
|
$cmd = "mkdsklsnode -i $imagename $nodes";
|
|
} else {
|
|
$cmd = "nimnodeset -i $imagename $nodes";
|
|
}
|
|
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Set node install method error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
$cmd = "rnetboot $nodes";
|
|
web_runcmd($cmd, $callback);
|
|
if ($::RUNCMD_RC) {
|
|
web_infomsg("Reboot nodes error.\nprovision stop.", $callback);
|
|
return;
|
|
}
|
|
|
|
web_infomsg("Provision on $nodes success.\nprovision stop.");
|
|
}
|
|
|
|
sub web_runcmd {
|
|
my $cmd = shift;
|
|
my $callback = shift;
|
|
my $showstr = "\n" . $cmd . "\n";
|
|
|
|
web_infomsg($showstr, $callback);
|
|
|
|
my $retvalue = xCAT::Utils->runcmd($cmd, -1, 1);
|
|
$showstr = join("\n", @$retvalue);
|
|
$showstr .= "\n";
|
|
|
|
web_infomsg($showstr, $callback);
|
|
}
|
|
|
|
sub web_infomsg {
|
|
my $msg = shift;
|
|
my $callback = shift;
|
|
my %rsp;
|
|
|
|
push @{ $rsp{info} }, $msg;
|
|
xCAT::MsgUtils->message('I', \%rsp, $callback);
|
|
return;
|
|
}
|
|
|
|
sub web_summary {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $groupName = $request->{arg}->[1];
|
|
my @nodes;
|
|
my $nodetypeTab;
|
|
my $nodelistTab;
|
|
my $attrs;
|
|
my %oshash;
|
|
my %archhash;
|
|
my %provhash;
|
|
my %typehash;
|
|
my %statushash;
|
|
my $retHash = {};
|
|
my $temp;
|
|
|
|
if (defined($groupName)) {
|
|
@nodes = xCAT::NodeRange::noderange($groupName);
|
|
} else {
|
|
@nodes = xCAT::DBobjUtils->getObjectsOfType('node');
|
|
}
|
|
|
|
$nodetypeTab = xCAT::Table->new('nodetype');
|
|
unless ($nodetypeTab) {
|
|
return;
|
|
}
|
|
|
|
$nodelistTab = xCAT::Table->new('nodelist');
|
|
unless ($nodelistTab) {
|
|
return;
|
|
}
|
|
|
|
$attrs = $nodetypeTab->getNodesAttribs(\@nodes, [ 'os', 'arch', 'provmethod', 'nodetype' ]);
|
|
unless ($attrs) {
|
|
return;
|
|
}
|
|
|
|
while (my ($key, $value) = each(%{$attrs})) {
|
|
web_attrcount($value->[0]->{'os'}, \%oshash);
|
|
web_attrcount($value->[0]->{'arch'}, \%archhash);
|
|
web_attrcount($value->[0]->{'provmethod'},, \%provhash);
|
|
web_attrcount($value->[0]->{'nodetype'},, \%typehash);
|
|
}
|
|
|
|
$attrs = $nodelistTab->getNodesAttribs(\@nodes, ['status']);
|
|
while (my ($key, $value) = each(%{$attrs})) {
|
|
web_attrcount($value->[0]->{'status'}, \%statushash);
|
|
}
|
|
|
|
# Status
|
|
$temp = '';
|
|
while (my ($key, $value) = each(%statushash)) {
|
|
$temp .= ($key . ':' . $value . ';');
|
|
}
|
|
$temp = substr($temp, 0, -1);
|
|
push(@{ $retHash->{'data'} }, 'Status=' . $temp);
|
|
|
|
# OS
|
|
$temp = '';
|
|
while (my ($key, $value) = each(%oshash)) {
|
|
$temp .= ($key . ':' . $value . ';');
|
|
}
|
|
$temp = substr($temp, 0, -1);
|
|
push(@{ $retHash->{'data'} }, 'Operating System=' . $temp);
|
|
|
|
# Architecture
|
|
$temp = '';
|
|
while (my ($key, $value) = each(%archhash)) {
|
|
$temp .= ($key . ':' . $value . ';');
|
|
}
|
|
$temp = substr($temp, 0, -1);
|
|
push(@{ $retHash->{'data'} }, 'Architecture=' . $temp);
|
|
|
|
# Provision method
|
|
$temp = '';
|
|
while (my ($key, $value) = each(%provhash)) {
|
|
$temp .= ($key . ':' . $value . ';');
|
|
}
|
|
$temp = substr($temp, 0, -1);
|
|
push(@{ $retHash->{'data'} }, 'Provision Method=' . $temp);
|
|
|
|
# Nodetype
|
|
$temp = '';
|
|
while (my ($key, $value) = each(%typehash)) {
|
|
$temp .= ($key . ':' . $value . ';');
|
|
}
|
|
$temp = substr($temp, 0, -1);
|
|
push(@{ $retHash->{'data'} }, 'Node Type=' . $temp);
|
|
|
|
# Return data
|
|
$callback->($retHash);
|
|
}
|
|
|
|
sub web_attrcount {
|
|
my ($key, $container) = @_;
|
|
unless (defined($key)) {
|
|
$key = 'unknown';
|
|
}
|
|
|
|
if ($container->{$key}) {
|
|
$container->{$key}++;
|
|
} else {
|
|
$container->{$key} = 1;
|
|
}
|
|
}
|
|
|
|
sub web_rinstall {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $os = $request->{arg}->[1];
|
|
my $profile = $request->{arg}->[2];
|
|
my $arch = $request->{arg}->[3];
|
|
my $node = $request->{arg}->[4];
|
|
|
|
# Begin installation
|
|
my $out = `rinstall -o $os -p $profile -a $arch $node`;
|
|
|
|
$callback->({ data => $out });
|
|
}
|
|
|
|
sub web_addnode {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodetype = $request->{arg}->[1];
|
|
my @tempArray = split(',', $request->{arg}->[2]);
|
|
|
|
my $hcpname = shift(@tempArray);
|
|
if ('node' ne $nodetype) {
|
|
my $username = $tempArray[0];
|
|
my $passwd = $tempArray[1];
|
|
my $ip = $tempArray[2];
|
|
`/bin/grep '$hcpname' /etc/hosts`;
|
|
if ($?) {
|
|
open(OUTPUTFILE, '>>/etc/hosts');
|
|
print OUTPUTFILE "$ip $hcpname\n";
|
|
close(OUTPUTFILE);
|
|
}
|
|
|
|
if ('hmc' eq $nodetype) {
|
|
`/opt/xcat/bin/chdef -t node -o $hcpname username=$username password=$passwd mgt=hmc nodetype=$nodetype ip=$ip groups=all`;
|
|
} else {
|
|
`/opt/xcat/bin/chdef -t node -o $hcpname username=$username password=$passwd mgt=blade mpa=$hcpname nodetype=$nodetype id=0 groups=mm,all`;
|
|
}
|
|
return;
|
|
}
|
|
|
|
my %temphash;
|
|
my $writeflag = 0;
|
|
my $line = '';
|
|
|
|
# Save all nodes into a hash
|
|
foreach (@tempArray) {
|
|
$temphash{$_} = 1;
|
|
}
|
|
|
|
for (my $i = 0 ; $i < scalar(@tempArray) ; $i = $i + 2) {
|
|
$temphash{ $tempArray[$i] } = $tempArray[ $i + 1 ];
|
|
}
|
|
|
|
`/opt/xcat/bin/rscan $hcpname -z > /tmp/rscanall.tmp`;
|
|
|
|
unless (-e '/tmp/rscanall.tmp') {
|
|
return;
|
|
}
|
|
|
|
open(INPUTFILE, '/tmp/rscanall.tmp');
|
|
open(OUTPUTFILE, '>/tmp/webrscan.tmp');
|
|
while ($line = <INPUTFILE>) {
|
|
if ($line =~ /(\S+):$/) {
|
|
if ($temphash{$1}) {
|
|
$writeflag = 1;
|
|
print OUTPUTFILE $temphash{$1} . ":\n";
|
|
} else {
|
|
$writeflag = 0;
|
|
}
|
|
} else {
|
|
if ($writeflag) {
|
|
print OUTPUTFILE $line;
|
|
}
|
|
}
|
|
}
|
|
|
|
close(INPUTFILE);
|
|
close(OUTPUTFILE);
|
|
unlink('/tmp/rscanall.tmp');
|
|
|
|
`cat /tmp/webrscan.tmp | chdef -z`;
|
|
unlink('/tmp/webrscan.tmp');
|
|
}
|
|
|
|
sub web_graphinfo {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $nodetypeTab;
|
|
my @nodes;
|
|
my @parray;
|
|
my @bladearray;
|
|
my @xarray;
|
|
my %phash;
|
|
my %bladehash;
|
|
my %xhash;
|
|
my @unsupportarray;
|
|
my @missinfoarray;
|
|
my $result;
|
|
my $pretstr = '';
|
|
my $bladeretstr = '';
|
|
my $xretstr = '';
|
|
my $unsupretstr = '';
|
|
my $missretstr = '';
|
|
|
|
@nodes = xCAT::DBobjUtils->getObjectsOfType('node');
|
|
|
|
$nodetypeTab = xCAT::Table->new('nodetype');
|
|
unless ($nodetypeTab) {
|
|
return;
|
|
}
|
|
|
|
# Get all nodetype and seperate nodes into different group
|
|
$result = $nodetypeTab->getNodesAttribs(\@nodes, ['nodetype']);
|
|
while (my ($key, $value) = each(%$result)) {
|
|
my $temptype = $value->[0]->{'nodetype'};
|
|
if ($temptype =~ /(ppc|lpar|cec|frame)/i) {
|
|
push(@parray, $key);
|
|
} elsif ($temptype =~ /blade/i) {
|
|
push(@bladearray, $key);
|
|
} elsif ($temptype =~ /osi/i) {
|
|
push(@xarray, $key);
|
|
} else {
|
|
push(@unsupportarray, $key);
|
|
}
|
|
}
|
|
$nodetypeTab->close();
|
|
|
|
# Get all information for System p node
|
|
if (scalar(@parray) > 0) {
|
|
my $ppctab = xCAT::Table->new('ppc');
|
|
|
|
$result = $ppctab->getNodesAttribs(\@parray, ['parent']);
|
|
my $typehash = xCAT::DBobjUtils->getnodetype(\@parray);
|
|
foreach (@parray) {
|
|
my $value = $result->{$_};
|
|
if ($value->[0]) {
|
|
$phash{$_} = $$typehash{$_} . ':' . $value->[0]->{'parent'} . ':';
|
|
} else {
|
|
$phash{$_} = $$typehash{$_} . '::';
|
|
}
|
|
}
|
|
$ppctab->close();
|
|
|
|
undef @parray;
|
|
@parray = keys %phash;
|
|
}
|
|
if (scalar(@parray) > 0) {
|
|
|
|
# mtm
|
|
my $vpdtab = xCAT::Table->new('vpd');
|
|
$result = $vpdtab->getNodesAttribs(\@parray, ['mtm']);
|
|
foreach (@parray) {
|
|
my $value = $result->{$_};
|
|
$phash{$_} = $phash{$_} . $value->[0]->{'mtm'} . ':';
|
|
}
|
|
$vpdtab->close();
|
|
|
|
# Status
|
|
my $nodelisttab = xCAT::Table->new('nodelist');
|
|
$result = $nodelisttab->getNodesAttribs(\@parray, ['status']);
|
|
foreach (@parray) {
|
|
my $value = $result->{$_};
|
|
$phash{$_} = $phash{$_} . $value->[0]->{'status'};
|
|
}
|
|
$nodelisttab->close();
|
|
|
|
while (my ($key, $value) = each(%phash)) {
|
|
$pretstr = $pretstr . $key . ':' . $value . ';';
|
|
}
|
|
}
|
|
|
|
# Get all information for blade node
|
|
if (scalar(@bladearray) > 0) {
|
|
my $mptab = xCAT::Table->new('mp');
|
|
$result = $mptab->getNodesAttribs(\@bladearray, [ 'mpa', 'id' ]);
|
|
foreach (@bladearray) {
|
|
my $value = $result->{$_};
|
|
if ($value->[0]->{'mpa'}) {
|
|
$bladehash{$_} = 'blade:' . $value->[0]->{'mpa'} . ':' . $value->[0]->{'id'} . ':';
|
|
} else {
|
|
push(@missinfoarray, $_);
|
|
}
|
|
}
|
|
|
|
$mptab->close();
|
|
|
|
undef @bladearray;
|
|
@bladearray = keys %bladehash;
|
|
}
|
|
|
|
if (scalar(@bladearray) > 0) {
|
|
|
|
# Status
|
|
my $nodelisttab = xCAT::Table->new('nodelist');
|
|
$result = $nodelisttab->getNodesAttribs(\@bladearray, ['status']);
|
|
foreach (@bladearray) {
|
|
my $value = $result->{$_};
|
|
$bladehash{$_} = $bladehash{$_} . $value->[0]->{'status'};
|
|
}
|
|
$nodelisttab->close();
|
|
while (my ($key, $value) = each(%bladehash)) {
|
|
$bladeretstr = $bladeretstr . $key . ':' . $value . ';';
|
|
}
|
|
}
|
|
|
|
# Get all information for System x node
|
|
if (scalar(@xarray) > 0) {
|
|
|
|
# Rack and unit
|
|
my $nodepostab = xCAT::Table->new('nodepos');
|
|
$result = $nodepostab->getNodesAttribs(\@xarray, [ 'rack', 'u' ]);
|
|
foreach (@xarray) {
|
|
my $value = $result->{$_};
|
|
if ($value->[0]->{'rack'}) {
|
|
$xhash{$_} = 'systemx:' . $value->[0]->{'rack'} . ':' . $value->[0]->{'u'} . ':';
|
|
} else {
|
|
push(@missinfoarray, $_);
|
|
}
|
|
}
|
|
|
|
$nodepostab->close();
|
|
|
|
undef @xarray;
|
|
@xarray = keys %xhash;
|
|
}
|
|
|
|
if (scalar(@xarray) > 0) {
|
|
|
|
# mtm
|
|
my $vpdtab = xCAT::Table->new('vpd');
|
|
$result = $vpdtab->getNodesAttribs(\@xarray, ['mtm']);
|
|
foreach (@xarray) {
|
|
my $value = $result->{$_};
|
|
$xhash{$_} = $xhash{$_} . $value->[0]->{'mtm'} . ':';
|
|
}
|
|
$vpdtab->close();
|
|
|
|
# Status
|
|
my $nodelisttab = xCAT::Table->new('nodelist');
|
|
$result = $nodelisttab->getNodesAttribs(\@xarray, ['status']);
|
|
foreach (@xarray) {
|
|
my $value = $result->{$_};
|
|
$xhash{$_} = $xhash{$_} . $value->[0]->{'status'};
|
|
}
|
|
|
|
while (my ($key, $value) = each(%xhash)) {
|
|
$xretstr = $xretstr . $key . ':' . $value . ';';
|
|
}
|
|
}
|
|
|
|
@missinfoarray = (@missinfoarray, @unsupportarray);
|
|
foreach (@missinfoarray) {
|
|
$missretstr = $missretstr . $_ . ':linux:other;';
|
|
}
|
|
|
|
# Combine all information into a string
|
|
my $retstr = $pretstr . $bladeretstr . $xretstr . $missretstr;
|
|
if ($retstr) {
|
|
$retstr = substr($retstr, 0, -1);
|
|
}
|
|
|
|
$callback->({ data => $retstr });
|
|
}
|
|
|
|
sub web_getdefaultuserentry {
|
|
|
|
# Get default user entry
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $profile = $request->{arg}->[1];
|
|
|
|
if (!$profile) {
|
|
$profile = 'default';
|
|
}
|
|
|
|
my $entry;
|
|
if (!(`test -e /var/opt/xcat/profiles/$profile.direct && echo 'File exists'`)) {
|
|
$entry = `cat /var/opt/xcat/profiles/default.direct`;
|
|
} else {
|
|
$entry = `cat /var/opt/xcat/profiles/$profile.direct`;
|
|
}
|
|
|
|
$callback->({ data => $entry });
|
|
}
|
|
|
|
sub web_passwd() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get current and new passwords
|
|
my $user = $request->{arg}->[1];
|
|
my $password = $request->{arg}->[2];
|
|
|
|
# Generate encrypted password
|
|
my $random = rand(10000000);
|
|
my $encrypted = `perl -e "print crypt($password, $random)"`;
|
|
|
|
# Save in xCAT passwd table
|
|
`/opt/xcat/sbin/chtab username=$user passwd.key=xcat passwd.password=$encrypted`;
|
|
|
|
my $info = "User password successfully updated";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_policy() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get user attributes
|
|
my $priority = $request->{arg}->[1];
|
|
my $args = $request->{arg}->[2];
|
|
|
|
# Save in xCAT passwd and policy tables
|
|
my $out = `/opt/xcat/sbin/chtab priority=$priority $args`;
|
|
|
|
my $info = "User policy successfully updated";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_deleteuser() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get user attributes
|
|
my $user = $request->{arg}->[1];
|
|
my @users = split(',', $user);
|
|
|
|
# Delete user from xCAT passwd and policy tables
|
|
foreach (@users) {
|
|
`/opt/xcat/sbin/chtab -d username=$_ passwd`;
|
|
`/opt/xcat/sbin/chtab -d name=$_ policy`;
|
|
}
|
|
|
|
my $info = "User successfully deleted";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
sub web_getzdiskinfo() {
|
|
|
|
# Get default disk info
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $profile = $request->{arg}->[1];
|
|
|
|
if (!$profile) {
|
|
$profile = 'default';
|
|
}
|
|
|
|
my $info;
|
|
if (!(`test -e /var/opt/xcat/profiles/$profile.conf && echo 'File exists'`)) {
|
|
$info = `cat /var/opt/xcat/profiles/default.conf`;
|
|
} else {
|
|
$info = `cat /var/opt/xcat/profiles/$profile.conf`;
|
|
}
|
|
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_mkzprofile() {
|
|
|
|
# Create default profile
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $profile = $request->{arg}->[1];
|
|
my $pool = $request->{arg}->[2];
|
|
my $size = $request->{arg}->[3];
|
|
|
|
# Create profile under /var/opt/xcat/profiles
|
|
`mkdir -p /var/opt/xcat/profiles`;
|
|
my $var = "";
|
|
`echo "# Configuration for virtual machines" > /var/opt/xcat/profiles/$profile.conf`;
|
|
$var = $profile . "_diskpool";
|
|
`echo "$var=$pool" >> /var/opt/xcat/profiles/$profile.conf`;
|
|
$var = $profile . "_eckd_size";
|
|
`echo "$var=$size" >> /var/opt/xcat/profiles/$profile.conf`;
|
|
|
|
# Move directory entry into /var/opt/xcat/profiles from /var/tmp
|
|
`mv /var/tmp/$profile.direct /var/opt/xcat/profiles`;
|
|
|
|
my $info = "Profile successfully created/updated";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_rmzprofile() {
|
|
|
|
# Delete default profile
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $profile = $request->{arg}->[1];
|
|
my @profiles = split(',', $profile);
|
|
|
|
# Delete profile under /var/opt/xcat/profiles
|
|
foreach (@profiles) {
|
|
`rm /var/opt/xcat/profiles/$_.conf`;
|
|
`rm /var/opt/xcat/profiles/$_.direct`;
|
|
}
|
|
|
|
my $info = "Profile successfully deleted";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_mkippool() {
|
|
|
|
# Create group IP pool
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $group = $request->{arg}->[1];
|
|
|
|
# Move directory entry into /var/opt/xcat/ippool from /var/tmp
|
|
`mkdir -p /var/opt/xcat/ippool`;
|
|
`mv /var/tmp/$group.pool /var/opt/xcat/ippool`;
|
|
|
|
my $info = "IP pool successfully created/updated";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_rmippool() {
|
|
|
|
# Delete group IP pool
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $group = $request->{arg}->[1];
|
|
my @groups = split(',', $group);
|
|
|
|
# Delete IP pool under /var/opt/xcat/ippool
|
|
foreach (@groups) {
|
|
`rm -rf /var/opt/xcat/ippool/$_.pool`;
|
|
}
|
|
|
|
my $info = "IP pool successfully deleted";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_lsippool() {
|
|
|
|
# List IP pool
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get profile
|
|
my $group = $request->{arg}->[1];
|
|
|
|
# IP pool contained in /var/opt/xcat/ippool where a file exists per group
|
|
my $entries;
|
|
if (!(`test -e /var/opt/xcat/ippool/$group.pool && echo Exists`)) {
|
|
$entries = "No IP pool found!";
|
|
} else {
|
|
|
|
# List IP pool under /var/opt/xcat/ippool
|
|
$entries = `cat /var/opt/xcat/ippool/$group.pool`;
|
|
}
|
|
|
|
$callback->({ info => $entries });
|
|
}
|
|
|
|
sub web_updateosimage() {
|
|
|
|
# Add OS image to xCAT table
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
my $name = $request->{arg}->[1];
|
|
my $type = $request->{arg}->[2];
|
|
my $arch = $request->{arg}->[3];
|
|
my $osName = $request->{arg}->[4];
|
|
my $osVersion = $request->{arg}->[5];
|
|
my $profile = $request->{arg}->[6];
|
|
my $provMethod = $request->{arg}->[7];
|
|
my $comments = $request->{arg}->[8];
|
|
|
|
`/opt/xcat/sbin/chtab -d imagename=$name osimage`;
|
|
`/opt/xcat/sbin/chtab osimage.imagename=$name osimage.imagetype=$type osimage.osarch=$arch osimage.osname=$osName osimage.osvers=$osVersion osimage.profile=$profile osimage.provmethod=$provMethod osimage.comments=$comments`;
|
|
my $info = "Image successfully updated";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_rmosimage() {
|
|
|
|
# Delete OS image from xCAT table
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
my $name = $request->{arg}->[1];
|
|
my @names = split(',', $name);
|
|
|
|
# Delete user from xCAT passwd and policy tables
|
|
foreach (@names) {
|
|
`/opt/xcat/sbin/chtab -d imagename=$_ osimage`;
|
|
}
|
|
|
|
my $info = "Image successfully deleted";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_updategroup() {
|
|
|
|
# Add group to xCAT table
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
my $name = $request->{arg}->[1];
|
|
my $ip = $request->{arg}->[2];
|
|
$ip =~ s/'//g;
|
|
|
|
my $hostnames = $request->{arg}->[3];
|
|
$hostnames =~ s/'//g;
|
|
|
|
my $comments = $request->{arg}->[4];
|
|
$comments =~ s/'//g;
|
|
|
|
`/opt/xcat/sbin/chtab -d node=$name hosts`;
|
|
`/opt/xcat/sbin/chtab node=$name hosts.ip="$ip" hosts.hostnames="$hostnames" hosts.comments="$comments"`;
|
|
|
|
my $info = "Group successfully updated";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_rmgroup() {
|
|
|
|
# Delete group from xCAT table
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
my $name = $request->{arg}->[1];
|
|
my @names = split(',', $name);
|
|
|
|
# Delete user from xCAT passwd and policy tables
|
|
foreach (@names) {
|
|
`/opt/xcat/sbin/chtab -d node=$_ hosts`;
|
|
`rm -rf /var/opt/xcat/ippool/$_.pool`;
|
|
}
|
|
|
|
my $info = "Group successfully deleted";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_framesetup() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $adminpasswd = $request->{arg}->[1];
|
|
my $generalpasswd = $request->{arg}->[2];
|
|
my $hmcpasswd = $request->{arg}->[3];
|
|
my $configphase = $request->{arg}->[4];
|
|
my @tempnode = 'bpa';
|
|
|
|
if ($configphase == 1) {
|
|
|
|
#run makedhcp
|
|
xCAT::Utils->runcmd('makedhcp bpa', -1, 1);
|
|
sleep(10);
|
|
|
|
#run makehosts
|
|
xCAT::Utils->runcmd('makehosts bpa', -1, 1);
|
|
$callback->({ info => 'FRAMEs DHCP, DNS configured.' });
|
|
} elsif ($configphase == 2) {
|
|
|
|
#run chtab command
|
|
xCAT::Utils->runcmd('chtab key=bpa,username=HMC passwd.password=' . $hmcpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('chtab key=bpa,username=admin passwd.password=' . $adminpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('chtab key=bpa,username=general passwd.password=' . $generalpasswd, -1, 1);
|
|
|
|
#mkhwconn
|
|
xCAT::Utils->runcmd('mkhwconn frame -t', -1, 1);
|
|
|
|
#rspconfig
|
|
xCAT::Utils->runcmd('rspconfig frame general_passwd=general,' . $generalpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('rspconfig frame admin_passwd=admin,' . $adminpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('rspconfig frame HMC_passwd=,' . $hmcpasswd, -1, 1);
|
|
|
|
$callback->({ info => 'Hardware connection and configure password created.' });
|
|
}
|
|
}
|
|
|
|
sub web_cecsetup() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
my $adminpasswd = $request->{arg}->[1];
|
|
my $generalpasswd = $request->{arg}->[2];
|
|
my $hmcpasswd = $request->{arg}->[3];
|
|
my $configphase = $request->{arg}->[4];
|
|
my @tempnode = 'bpa';
|
|
|
|
if ($configphase == 1) {
|
|
|
|
# Run makedhcp
|
|
xCAT::Utils->runcmd('makedhcp fsp', -1, 1);
|
|
sleep(10);
|
|
|
|
# Run makehosts
|
|
xCAT::Utils->runcmd('makehosts fsp', -1, 1);
|
|
$callback->({ info => 'CEC DHCP, DNS configured.' });
|
|
} elsif ($configphase == 2) {
|
|
|
|
# Run chtab command
|
|
xCAT::Utils->runcmd('chtab key=fsp,username=HMC passwd.password=' . $hmcpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('chtab key=fsp,username=admin passwd.password=' . $adminpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('chtab key=fsp,username=general passwd.password=' . $generalpasswd, -1, 1);
|
|
|
|
# Run mkhwconn
|
|
xCAT::Utils->runcmd('mkhwconn cec -t', -1, 1);
|
|
|
|
# Run rspconfig
|
|
xCAT::Utils->runcmd('rspconfig cec general_passwd=general,' . $generalpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('rspconfig cec admin_passwd=admin,' . $adminpasswd, -1, 1);
|
|
xCAT::Utils->runcmd('rspconfig cec HMC_passwd=,' . $hmcpasswd, -1, 1);
|
|
|
|
$callback->({ info => 'Hardware connection and configure password created.' });
|
|
}
|
|
}
|
|
|
|
sub web_deletefile() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get file location
|
|
my $file = $request->{arg}->[1];
|
|
|
|
my $info;
|
|
if (substr($file, 0, 9) ne "/install/") {
|
|
$info = "(Error) Files are restricted to those in /install";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
unless (-e $file) {
|
|
$info = "(Error) File does not exist";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
# Escape spaces
|
|
$file =~ s/ /\\ /g;
|
|
my $out = `/bin/rm -rf $file`;
|
|
|
|
$info = "File successfully deleted";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_createfolder() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Get folder location
|
|
my $folder = $request->{arg}->[1];
|
|
|
|
my $info;
|
|
if (substr($folder, 0, 9) ne "/install/") {
|
|
$info = "(Error) Folders are restricted to those in /install";
|
|
$callback->({ info => $info });
|
|
return;
|
|
}
|
|
|
|
# Escape spaces
|
|
$folder =~ s/ /\\ /g;
|
|
my $out = `/bin/mkdir -p $folder`;
|
|
$out = `chown apache:apache $folder`;
|
|
|
|
$info = "Folder successfully created";
|
|
$callback->({ info => $info });
|
|
}
|
|
|
|
sub web_getrepospace() {
|
|
my ($request, $callback, $sub_req) = @_;
|
|
|
|
# Query repository space. Output is: size, used, available, used %, mount
|
|
# remove trailing / from /install so this query also works on CMO which uses cmo-data on /data
|
|
my $space = `/bin/df -h /install | egrep -i "/install"`;
|
|
$space =~ s/\s+/ /g;
|
|
$space =~ s/\s*$//;
|
|
$space =~ s/^\s*//;
|
|
|
|
$callback->({ info => $space });
|
|
}
|
|
|
|
sub web_verifynode() {
|
|
my ( $request, $callback, $sub_req ) = @_;
|
|
my $cmdOpts = '';
|
|
my $out;
|
|
|
|
# Loop to handle all options passed. We don't know how many they will pass
|
|
# so we look for 'end' to signify the end. We set a loop variable to 500 which
|
|
# we know is much larger than the expected number of arguments. This prevents
|
|
# the loop from going on forever should there be an error on the javascript side
|
|
# an the 'end' was not passed.
|
|
for ( my $i = 0 ; $i < 500 ; $i++ ) {
|
|
if ( $request->{arg}->[$i] ne 'end' ) {
|
|
$cmdOpts = "$cmdOpts $request->{arg}->[$i]";
|
|
} else {
|
|
last;
|
|
}
|
|
}
|
|
|
|
$out = `/opt/xcat/bin/verifynode $cmdOpts`;
|
|
|
|
$callback->( { info => $out });
|
|
}
|
|
|
|
1;
|