# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html #------------------------------------------------------- =head1 xCAT plugin package to handle pdu Supported command: rpower rinv rvitals =cut #------------------------------------------------------- package xCAT_plugin::pdu; BEGIN { $::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : -d '/opt/xcat' ? '/opt/xcat' : '/usr'; } use lib "$::XCATROOT/lib/perl"; use xCAT::Table; use xCAT::Utils; use xCAT::FifoPipe; use xCAT::MsgUtils; use xCAT::State; use xCAT::SvrUtils; use xCAT::Usage; use xCAT::NodeRange; use Data::Dumper; use Getopt::Long; use File::Path; use Term::ANSIColor; use Time::Local; use strict; use Class::Struct; use XML::Simple; use Storable qw(dclone); use SNMP; use Expect; use Net::Ping; my $VERBOSE = 0; my %allerrornodes = (); my $callback; my $pdutab; my $pduhash; #------------------------------------------------------- =head3 handled_commands Return list of commands handled by this plugin =cut #------------------------------------------------------- sub handled_commands { return { rpower => ["nodehm:mgt","pduoutlet:pdu=\.\*"], rinv => ["nodehm:mgt"], rvitals => ["nodehm:mgt"], nodeset => ["nodehm:mgt"], rspconfig => ["nodehm:mgt"], pdudiscover => "pdu", }; } sub pdu_usage { my ($callback, $command) = @_; my $usagemsg = "Usage: The following commands support both type of PDUs : pdudiscover [|--range ipranges] [-r|-x|-z] [-w] [-V|--verbose] [--setup] rpower pdunodes [off|on|stat|reset] rinv pdunodes rvitals pdunodes The following commands support IR PDU with pdutype=irpdu : rpower computenodes [pduoff|pduon|pdustat|pdustatus|pdureset] rspconfig irpdunode [hostname=|ip=|gateway=|mask=] The following commands support CR PDU with pdutype=crpdu : rpower pdunodes relay=[1|2|3] [on|off] rspconfig pdunodes sshcfg rspconfig pdunodes snmpcfg rspconfig pdunode [hostname=|ip=|mask=] \n"; if ($callback) { my $rsp = {}; $rsp->{data}->[0] = $usagemsg; xCAT::MsgUtils->message("I", $rsp, $callback); } else { xCAT::MsgUtils->message("I", $usagemsg); } return; } #-------------------------------------------------------------------------------- =head3 preprocess_request Parse the arguments and display the usage or the version string. =cut #-------------------------------------------------------------------------------- sub preprocess_request { my $req = shift; if ($req->{_xcatpreprocessed}->[0] == 1) { return [$req]; } my $callback=shift; my @requests; my $command = $req->{command}->[0]; my $noderange = $req->{node}; #Should be arrayref my $extrargs = $req->{arg}; my @exargs=($req->{arg}); if (ref($extrargs)) { @exargs=@$extrargs; } my $usage_string=xCAT::Usage->parseCommand($command, @exargs); if ($usage_string) { &pdu_usage($callback, $command); return 1; } if ((!$noderange) && ($command ne "pdudiscover") ){ &pdu_usage($callback, $command); return; } my @result = (); my $mncopy = {%$req}; push @result, $mncopy; return \@result; } #------------------------------------------------------- =head3 process_request Process the command. =cut #------------------------------------------------------- sub process_request { my $request = shift; my $callback = shift; my $subreq = shift; my $command = $request->{command}->[0]; my $noderange = $request->{node}; #Should be arrayref my $extrargs = $request->{arg}; my @exargs = ($request->{arg}); if (ref($extrargs)) { @exargs = @$extrargs; } # get all entries from pdu table my @attrs=(); my $schema = xCAT::Table->getTableSchema('pdu'); my $desc = $schema->{descriptions}; foreach my $c (@{ $schema->{cols} }) { push @attrs, $c; } $pdutab = xCAT::Table->new('pdu'); if ($pdutab) { $pduhash = $pdutab->getAllNodeAttribs(\@attrs, 1); } if( $command eq "rinv") { #for higher performance, handle node in batch return showMFR($noderange, $callback); }elsif ($command eq "rvitals") { return showMonitorData($noderange, $callback); }elsif ($command eq "rpower") { my $subcmd = $exargs[0]; my $subcmd2 = $exargs[1]; if (($subcmd eq 'pduoff') || ($subcmd eq 'pduon') || ($subcmd eq 'pdustat')|| ($subcmd eq 'pdureset') ){ #if one day, pdu node have pdu attribute, handle in this section too return powerpduoutlet($noderange, $subcmd, $callback); } else { #------------------------------------------- #there are 2 cases will enter this block #one is if node's mgt is pdu #another is if node has pdu attribute but mgt isn't pdu #if the node has pdu attribute but mgt isn't pdu, #should do nothing for this node, let other plugin to hanle this node #------------------------------------------- my @allpdunodes=(); my $nodehm = xCAT::Table->new('nodehm'); my $nodehmhash = $nodehm->getNodesAttribs($noderange, ['mgt']); foreach my $node (@$noderange) { if($nodehmhash->{$node}->[0]->{mgt} eq 'pdu'){ push @allpdunodes, $node; } } if(@allpdunodes) { if ( ($subcmd =~ /relay/) || ($subcmd2 =~ /relay/) ) { process_powerrelay($request,$subreq,\@allpdunodes,$callback); } elsif(($subcmd eq 'on') || ($subcmd eq 'off') || ($subcmd eq 'stat') || ($subcmd eq 'state') || ($subcmd eq 'reset') ){ return powerpdu(\@allpdunodes, $subcmd, $callback); } else { my $pdunode = join (",", @allpdunodes); $callback->({ errorcode => [1],error => "The option $subcmd is not support for pdu node(s) $pdunode."}); &pdu_usage($callback, $command); } } } }elsif($command eq "rspconfig") { my $subcmd = $exargs[0]; if ($subcmd eq 'sshcfg') { process_sshcfg($noderange, $subcmd, $callback); }elsif ($subcmd eq 'snmpcfg') { process_snmpcfg($noderange, $subcmd, $callback); }elsif ($subcmd =~ /ip|gateway|netmask|hostname/) { process_netcfg($request, $subreq, $subcmd, $callback); } else { $callback->({ errorcode => [1],error => "The input $command $subcmd is not support for pdu"}); &pdu_usage($callback, $command); } }elsif($command eq "pdudiscover") { process_pdudiscover($request, $subreq, $callback); }elsif($command eq "nodeset") { $callback->({ errorcode => [1],error => "The input $command is not support for pdu"}); &pdu_usage($callback, $command); }else{ #reserve for other new command in future } return; } #------------------------------------------------------- =head3 fill_outletcount Get outlet count for IR PDU. =cut #------------------------------------------------------- sub fill_outletCount { my $session = shift; my $pdu = shift; my $callback = shift; my $outletoid = ".1.3.6.1.4.1.2.6.223.8.2.1.0"; my $pdutab = xCAT::Table->new('pdu'); my $count = $session->get("$outletoid"); if ($count) { $pdutab->setNodeAttribs($pdu, {outlet => $count}); } else { xCAT::SvrUtils::sendmsg("Invalid Outlet number ", $callback,$pdu); } return $count; } #------------------------------------------------------- =head3 powerpdu Process power command (stat/off/on) for pdu/pdus =cut #------------------------------------------------------- sub powerpdu { my $noderange = shift; my $subcmd = shift; my $callback = shift; if (($subcmd eq "stat") || ($subcmd eq "state")){ return powerstat($noderange, $callback); } foreach my $node (@$noderange) { if ($pduhash->{$node}->[0]->{pdutype} eq 'crpdu') { process_relay($node,$subcmd,$callback,1,3); next; } my $session = connectTopdu($node,$callback); if (!$session) { $callback->({ errorcode => [1],error => "Couldn't connect to $node"}); next; } my $count = $pduhash->{$node}->[0]->{outlet}; unless ($count) { $count = fill_outletCount($session, $node, $callback); } my $value; my $statstr; if ($subcmd eq "off") { $value = 0; $statstr = "off"; } elsif ( $subcmd eq "on") { $value = 1; $statstr = "on"; } else { $value = 2; $statstr = "reset"; } for (my $outlet =1; $outlet <= $count; $outlet++) { outletpower($session, $outlet, $value); if ($session->{ErrorStr}) { $callback->({ errorcode => [1],error => "Failed to get outlet status for $node"}); } else { my $output = " outlet $outlet is $statstr"; xCAT::SvrUtils::sendmsg($output, $callback, $node, %allerrornodes); } } } } #------------------------------------------------------- =head3 powerpduoutlet Process power command (pdustat/pduoff/pduon) for compute nodes, the pdu attribute needs to be set =cut #------------------------------------------------------- sub powerpduoutlet { my $noderange = shift; my $subcmd = shift; my $callback = shift; my $output; my $value; my $statstr; my $tmpnodestr = join(",", @$noderange); my $nodetab = xCAT::Table->new('pduoutlet'); my $nodepdu = $nodetab->getNodesAttribs($noderange,['pdu']); foreach my $node (@$noderange) { # the pdu attribute needs to be set if(! $nodepdu->{$node}->[0]->{pdu}){ $callback->({ error => "$node: without pdu attribute"}); next; } my @pdus = split /,/, $nodepdu->{$node}->[0]->{pdu}; foreach my $pdu_outlet (@pdus) { my ($pdu, $outlet) = split /:/, $pdu_outlet; if ($pduhash->{$pdu}->[0]->{pdutype} eq 'crpdu') { $callback->({ error => "$node: This command doesn't supports CONSTELLATION PDU with pdutype=crpdu for $pdu"}); next; } my $session = connectTopdu($pdu,$callback); if (!$session) { $callback->({ errorcode => [1],error => "$node: Couldn't connect to $pdu"}); next; } my $count = $pduhash->{$pdu}->[0]->{outlet}; unless ($count) { $count = fill_outletCount($session, $pdu, $callback); } if ($outlet > $count ) { $callback->({ error => "$node: $pdu outlet number $outlet is invalid"}); next; } my $cmd; if ($subcmd eq "pdustat") { $statstr=outletstat($session, $outlet); } elsif ($subcmd eq "pduoff") { $value = 0; $statstr = "off"; outletpower($session, $outlet, $value); } elsif ($subcmd eq "pduon") { $value = 1; $statstr = "on"; outletpower($session, $outlet, $value); } elsif ($subcmd eq "pdureset") { $value = 2; $statstr = "reset"; outletpower($session, $outlet, $value); } else { $callback->({ error => "$subcmd is not support"}); } if ($session->{ErrorStr}) { $callback->({ errorcode => [1],error => "$node: $pdu outlet $outlet has error = $session->{ErrorStr}"}); } else { $output = "$pdu operational state for outlet $outlet is $statstr"; xCAT::SvrUtils::sendmsg($output, $callback, $node, %allerrornodes); } } } } #------------------------------------------------------- =head3 outletpower Process power command for one pdu outlet, =cut #------------------------------------------------------- sub outletpower { my $session = shift; my $outlet = shift; my $value = shift; my $oid = ".1.3.6.1.4.1.2.6.223.8.2.2.1.11"; my $type = "INTEGER"; if ($session->{newmib}) { $oid = ".1.3.6.1.4.1.2.6.223.8.2.2.1.13"; } my $varbind = new SNMP::Varbind([ $oid, $outlet, $value, $type ]); return $session->set($varbind); } #------------------------------------------------------- =head3 powerstat Process command to query status of pdu =cut #------------------------------------------------------- sub powerstat { my $noderange = shift; my $callback = shift; my $output; foreach my $pdu (@$noderange) { if ($pduhash->{$pdu}->[0]->{pdutype} eq 'crpdu') { my $snmpversion = $pduhash->{$pdu}->[0]->{snmpversion}; my $snmpcmd; if ($snmpversion =~ /3/) { my $snmpuser = $pduhash->{$pdu}->[0]->{snmpuser}; my $seclevel = $pduhash->{$pdu}->[0]->{seclevel}; if ((defined $snmpuser) && (defined $seclevel)) { my $authtype = $pduhash->{$pdu}->[0]->{authtype}; if (!defined $authtype) { $authtype="MD5"; } my $authkey = $pduhash->{$pdu}->[0]->{authkey}; my $privtype = $pduhash->{$pdu}->[0]->{privtype}; if (!defined $privtype) { $privtype="DES"; } my $privkey = $pduhash->{$pdu}->[0]->{privkey}; if (!defined $privkey) { if (defined $authkey) { $privkey=$authkey; } } if ($seclevel eq "authNoPriv") { $snmpcmd = "snmpwalk -v3 -u $snmpuser -a $authtype -A $authkey -l $seclevel"; } elsif ($seclevel eq "authPriv") { $snmpcmd = "snmpwalk -v3 -u $snmpuser -a $authtype -A $authkey -l $seclevel -x $privtype -X $privkey"; } else { #default to notAuthNoPriv $snmpcmd = "snmpwalk -v3 -u $snmpuser -l $seclevel"; } } else { xCAT::SvrUtils::sendmsg("ERROR: No snmpuser or Security level defined for snmpV3 configuration", $callback,$pdu); xCAT::SvrUtils::sendmsg(" use chdef command to add pdu snmpV3 attributes to pdu table", $callback,$pdu); xCAT::SvrUtils::sendmsg(" ex: chdef coral-pdu snmpversion=3, snmpuser=admin, authtype=MD5 authkey=password1 privtype=DES privkey=password2 seclevel=authPriv", $callback,$pdu); xCAT::SvrUtils::sendmsg(" then run 'rspconfig $pdu snmpcfg' command ", $callback,$pdu); next; } } else { # use default value $snmpcmd = "snmpwalk -v3 -u admin -a MD5 -A password1 -l authPriv -x DES -X password2"; } for (my $relay = 1; $relay <= 3; $relay++) { relaystat($pdu, $relay, $snmpcmd, $callback); } next; } my $session = connectTopdu($pdu,$callback); if (!$session) { $callback->({ errorcode => [1],error => "Couldn't connect to $pdu"}); next; } my $count = $pduhash->{$pdu}->[0]->{outlet}; unless ($count) { $count = fill_outletCount($session, $pdu, $callback); } for (my $outlet =1; $outlet <= $count; $outlet++) { my $statstr = outletstat($session, $outlet); my $msg = " operational state for the outlet $outlet is $statstr"; xCAT::SvrUtils::sendmsg($msg, $callback, $pdu, %allerrornodes); } } } #------------------------------------------------------- =head3 outletstat Process command to query status of one pdu outlet ibmPduOutletState defined from mib file off(0) on(1) cycling(2) delaySwitch10(3) delaySwitch30(4) delaySwitch60(5) =cut #------------------------------------------------------- sub outletstat { my $session = shift; my $outlet = shift; my $oid = ".1.3.6.1.4.1.2.6.223.8.2.2.1.11"; my $output; my $statstr; if ($session->{newmib}) { $oid = ".1.3.6.1.4.1.2.6.223.8.2.2.1.13"; } $output = $session->get("$oid.$outlet"); if ($output eq 0) { $statstr = "off"; } elsif ($output eq 1) { $statstr = "on"; } elsif ($output eq 2) { $statstr = "cycling"; } elsif ($output eq 3) { $statstr = "delaySwitch10"; } elsif ($output eq 4) { $statstr = "delaySwitch30"; } elsif ($output eq 5) { $statstr = "delaySwitch60"; } else { $statstr = "$output(unknown state)" ; } return $statstr; } #------------------------------------------------------- =head3 connectTopdu connect pdu via snmp session =cut #------------------------------------------------------- sub connectTopdu { my $pdu = shift; my $callback = shift; #get community string from pdu table if defined, #otherwise, use default my $community; if ($pduhash->{$pdu}->[0]->{community}) { $community = $pduhash->{$pdu}->[0]->{community}; } else { $community = "public"; } my $snmpver = "1"; my $session; my $msg = "connectTopdu"; my $versionoid = ".1.3.6.1.4.1.2.6.223.7.3.0"; $session = new SNMP::Session( DestHost => $pdu, Version => $snmpver, Community => $community, UseSprintValue => 1, ); unless ($session) { return; } my $varbind = new SNMP::Varbind([ $versionoid, '' ]); my $pduversion = $session->get($varbind); if ($session->{ErrorStr}) { return; } $session->{newmib} = 0; if ($pduversion =~ /sLEN/) { $session->{newmib} = 1; } return $session; } #------------------------------------------------------- =head3 process_netcfg Config hostname of PDU Config ip/netmask of PDU via PduManager command PduManager is a tool for CoralPdu to manager the PDU. * /dev/shm/bin/PduManager -h '-i' set PDU system IP '-n' set system ip netmask. e.g.:PduManager -i xxx.xxx.xxx.xxx -n xxx.xxx.xxx.xxx example: rspconfig coralpdu hostname=coralpdu rspconfig coralpdu ip=1.1.1.1 netmask=255.0.0.0 =cut #------------------------------------------------------- sub process_netcfg { my $request = shift; my $subreq = shift; my $subcmd = shift; my $callback = shift; my $hostname; my $ip; my $netmask; my $args; my $exp; my $errstr; my $extrargs = $request->{arg}; my @exargs = ($request->{arg}); if (ref($extrargs)) { @exargs = @$extrargs; } my $nodes = $request->{node}; my $node_number = @$nodes; if ($node_number gt "1") { xCAT::SvrUtils::sendmsg("Can not configure more than 1 nodes", $callback); return; } my $pdu = @$nodes[0]; my $rsp = {}; my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodesAttribs($nodes,['ip','otherinterfaces']); my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; unless ($pduhash->{$pdu}->[0]->{pdutype} eq "crpdu") { netcfg_for_irpdu($pdu, $static_ip, $discover_ip, $request, $subreq, $callback); return; } # connect to PDU my $username = $pduhash->{$pdu}->[0]->{username}; my $password = $pduhash->{$pdu}->[0]->{password}; ($exp, $errstr) = session_connect($static_ip, $discover_ip,$username,$password); if (defined $errstr) { xCAT::SvrUtils::sendmsg("Failed to connect", $callback,$pdu); return; } foreach my $cmd (@exargs) { my ($key, $value) = split(/=/, $cmd); if ($key =~ /hostname/) { $hostname = $value; my ($ret, $err) = session_exec($exp, "echo $hostname > /etc/hostname;/etc/init.d/hostname.sh"); if (defined $err) { xCAT::SvrUtils::sendmsg("Failed to set hostname", $callback); } }elsif ($key =~ /ip/) { $ip = $value; } elsif ($key =~ /netmask/) { $netmask = $value; } else { xCAT::SvrUtils::sendmsg("rspconfig $cmd is not support yet, ignored", $callback); } } $args = "/dev/shm/bin/PduManager "; my $opt; if ($ip) { $opt = "-i $ip "; } if ($netmask) { $opt = $opt . "-n $netmask"; } if ($opt) { my $dshcmd = $args . $opt ; my ($ret, $err) = session_exec($exp, $dshcmd); if (defined $err) { #session will be hung if ip address changed my $p = Net::Ping->new(); if ( ($p->ping($ip)) && ($err =~ /TIMEOUT/) ) { xCAT::SvrUtils::sendmsg("$ip is reachable", $callback); } else { xCAT::SvrUtils::sendmsg("Failed to run $dshcmd, error=$err", $callback); return; } } xCAT::SvrUtils::sendmsg("$dshcmd ran successfully", $callback); xCAT::Utils->runxcmd({ command => ['chdef'], arg => ['-t','node','-o',$pdu,"ip=$ip","otherinterfaces="] }, $subreq, 0, 1); xCAT::Utils->runxcmd({ command => ['makehosts'], node => [$pdu] }, $subreq, 0, 1); } if (defined $exp) { $exp->hard_close(); } } #------------------------------------------------------- =head3 process_sshcfg Config passwordless for coralpdu example: rspconfig coralpdu sshcfg =cut #------------------------------------------------------- sub process_sshcfg { my $noderange = shift; my $subcmd = shift; my $callback = shift; my $keyfile = "/root/.ssh/id_rsa.pub"; my $rootkey = `cat /root/.ssh/id_rsa.pub`; my $cmd; my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodesAttribs($noderange,['ip','otherinterfaces']); foreach my $pdu (@$noderange) { unless ($pduhash->{$pdu}->[0]->{pdutype} eq "crpdu") { xCAT::SvrUtils::sendmsg("This command only supports CONSTELLATION PDU with pdutype=crpdu", $callback,$pdu); next; } my $msg = " process_sshcfg"; xCAT::SvrUtils::sendmsg($msg, $callback, $pdu, %allerrornodes); #remove old host key from /root/.ssh/known_hosts $cmd = "ssh-keygen -R $pdu"; xCAT::Utils->runcmd($cmd, 0); my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; my $username = $pduhash->{$pdu}->[0]->{username}; my $password = $pduhash->{$pdu}->[0]->{password}; my ($exp, $errstr) = session_connect($static_ip, $discover_ip,$username,$password); if (!defined $exp) { $msg = " Failed to connect $errstr"; xCAT::SvrUtils::sendmsg($msg, $callback, $pdu, %allerrornodes); next; } my $ret; my $err; ($ret, $err) = session_exec($exp, "mkdir -p /home/root/.ssh"); ($ret, $err) = session_exec($exp, "chmod 700 /home/root/.ssh"); ($ret, $err) = session_exec($exp, "echo \"$rootkey\" >/home/root/.ssh/authorized_keys"); ($ret, $err) = session_exec($exp, "chmod 644 /home/root/.ssh/authorized_keys"); $exp->hard_close(); } return; } #------------------------------------------------------- =head3 session_connect open a expect session and connect to CR PDU. =cut #------------------------------------------------------- sub session_connect { my $static_ip = shift; my $discover_ip = shift; my $userid = shift; my $password = shift; #default password for coral pdu if (!defined $userid) { $userid = "root"; } if (!defined $password) { $password = "password8"; } my $timeout = 30; my $ssh_ip; my $p = Net::Ping->new(); if ($p->ping($static_ip)) { $ssh_ip = $static_ip; } elsif ($p->ping($discover_ip)) { $ssh_ip = $discover_ip; } else { return(undef, " is not reachable\n"); } my $ssh = Expect->new; my $command = 'ssh'; my @parameters = ($userid . "@" . $ssh_ip); $ssh->debug(0); $ssh->log_stdout(0); # suppress stdout output.. unless ($ssh->spawn($command, @parameters)) { my $err = $!; $ssh->soft_close(); my $rsp; return(undef, "unable to run command $command $err\n"); } $ssh->expect($timeout, [ "-re", qr/WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED/, sub {die "WARNING: REMOTE HOST IDENTIFICATION HAS CHANGED!\n"; } ], [ "-re", qr/\(yes\/no\)\?\s*$/, sub { $ssh->send("yes\n"); exp_continue; } ], [ "-re", qr/ password:/, sub {$ssh->send("$password\n"); exp_continue; } ], [ "-re", qr/:~\$/, sub { $ssh->send("sudo su\n"); exp_continue; } ], [ "-re", qr/.*#/, sub { $ssh->clear_accum(); } ], [ timeout => sub { die "No login.\n"; } ] ); $ssh->clear_accum(); return ($ssh); } #------------------------------------------------------- =head3 session_exec execute command to CR PDU. =cut #------------------------------------------------------- sub session_exec { my $exp = shift; my $cmd = shift; my $timeout = shift; my $prompt = shift; $timeout = 30 unless defined $timeout; $prompt = qr/.*#/ unless defined $prompt; $exp->clear_accum(); $exp->send("$cmd\n"); my ($mpos, $merr, $mstr, $mbmatch, $mamatch) = $exp->expect(6, "-re", $prompt); if (defined $merr) { return(undef,$merr); } return($mbmatch); } #----------------------------------------------------------------- =head3 process_pdudiscover Discover the pdu for a given range of DHCP ip address it will call switchdiscover command with -s snmp --pdu options example: pdudiscover --range iprange -w =cut #------------------------------------------------------------------ sub process_pdudiscover { my $request = shift; my $sub_req = shift; my $callback = shift; my $extrargs = $request->{arg}; my @exargs = ($request->{arg}); if (ref($extrargs)) { @exargs=@$extrargs; } #check case in GetOptions $Getopt::Long::ignorecase = 0; Getopt::Long::Configure( "bundling" ); Getopt::Long::Configure("no_pass_through"); my %opt; if (!GetOptions( \%opt, qw(h|help V|verbose x z w r range=s setup))) { my $usage_string = xCAT::Usage->getUsage($request->{command}->[0]); $callback->({ data => $usage_string }); return; } push @exargs, "-s snmp --pdu"; my $cmd = "switchdiscover @exargs"; my $result = xCAT::Utils->runcmd($cmd, 0); my $rsp = {}; push @{ $rsp->{data} }, "$result"; xCAT::MsgUtils->message("I", $rsp, $callback); } #------------------------------------------------------- =head3 showMFR show MFR information of PDU via PduManager command PduManager is a tool for CoralPdu to manager the PDU. * /dev/shm/bin/PduManager -h '-m' show MFR info example: rinv coralpdu =cut #------------------------------------------------------- sub showMFR { my $noderange = shift; my $callback = shift; my $output; my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodesAttribs($noderange,['ip','otherinterfaces']); foreach my $pdu (@$noderange) { unless ($pduhash->{$pdu}->[0]->{pdutype} eq "crpdu") { rinv_for_irpdu($pdu, $callback); next; } # connect to PDU my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; my $username = $pduhash->{$pdu}->[0]->{username}; my $password = $pduhash->{$pdu}->[0]->{password}; my ($exp, $errstr) = session_connect($static_ip, $discover_ip,$username,$password); if (defined $errstr) { xCAT::SvrUtils::sendmsg("Failed to connect: $errstr", $callback); } my ($ret, $err) = session_exec($exp, "/dev/shm/bin/PduManager -m"); if (defined $err) { xCAT::SvrUtils::sendmsg("Failed to list MFR information: $err", $callback); } if (defined $ret) { foreach my $line (split /[\r\n]+/, $ret) { if ($line) { $line = join(' ',split(' ',$line)); xCAT::SvrUtils::sendmsg("$line", $callback,$pdu); } } } $exp->hard_close(); } } sub rinv_for_irpdu { my $pdu = shift; my $callback = shift; my $output; my $session = connectTopdu($pdu,$callback); if (!$session) { $callback->({ errorcode => [1],error => "Couldn't connect to $pdu"}); next; } #ibmPduSoftwareVersion $output = $session->get(".1.3.6.1.4.1.2.6.223.7.3.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Software Version: $output", $callback,$pdu); } #ibmPduMachineType $output = $session->get(".1.3.6.1.4.1.2.6.223.7.4.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Machine Type: $output", $callback,$pdu); } #ibmPduModelNumber $output = $session->get(".1.3.6.1.4.1.2.6.223.7.5.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Model Number: $output", $callback,$pdu); } #ibmPduPartNumber $output = $session->get(".1.3.6.1.4.1.2.6.223.7.6.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Part Number: $output", $callback,$pdu); } #ibmPduName $output = $session->get(".1.3.6.1.4.1.2.6.223.7.7.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Name: $output", $callback,$pdu); } #ibmPduSerialNumber $output = $session->get(".1.3.6.1.4.1.2.6.223.7.9.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Serial Number: $output", $callback,$pdu); } #ibmPduDescription $output = $session->get(".1.3.6.1.4.1.2.6.223.7.10.0"); if ($output) { xCAT::SvrUtils::sendmsg("PDU Description: $output", $callback,$pdu); } } #------------------------------------------------------- =head3 showMonitorData Show realtime monitor data(input voltage, current, power) of PDU via PduManager command PduManager is a tool for CoralPdu to manager the PDU. * /dev/shm/bin/PduManager -h '-d' show realtime monitor data(input voltage, current, power) example: rvitals coralpdu =cut #------------------------------------------------------- sub showMonitorData { my $noderange = shift; my $callback = shift; my $output; my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodesAttribs($noderange,['ip','otherinterfaces']); foreach my $pdu (@$noderange) { unless ($pduhash->{$pdu}->[0]->{pdutype} eq "crpdu") { my $session = connectTopdu($pdu,$callback); if (!$session) { $callback->({ errorcode => [1],error => "Couldn't connect to $pdu"}); next; } my $count = $pduhash->{$pdu}->[0]->{outlet}; unless ($count) { $count = fill_outletCount($session, $pdu, $callback); } if ($count > 0) { rvitals_for_irpdu($pdu, $count, $session, $callback); } next; } # connect to PDU my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; my $username = $pduhash->{$pdu}->[0]->{username}; my $password = $pduhash->{$pdu}->[0]->{password}; my ($exp, $errstr) = session_connect($static_ip, $discover_ip,$username,$password); my $ret; my $err; ($ret, $err) = session_exec($exp, "/dev/shm/bin/PduManager -d"); if (defined $err) { xCAT::SvrUtils::sendmsg("Failed to show monitor data: $err", $callback); } if (defined $ret) { foreach my $line (split /[\r\n]+/, $ret) { if ($line) { $line = join(' ',split(' ',$line)); xCAT::SvrUtils::sendmsg("$line", $callback,$pdu); } } } $exp->hard_close(); } } sub rvitals_for_irpdu { my $pdu = shift; my $count = shift; my $session = shift; my $callback = shift; my $output; #ibmPduVoltageWarning: (voltageNormal(0),voltageOutOfRange(1)) my $voltagewarning = ".1.3.6.1.4.1.2.6.223.0.1.1.7.0"; $output = $session->get("$voltagewarning"); xCAT::SvrUtils::sendmsg("Voltage Warning: $output", $callback,$pdu); # get power info for each outlet # starts oid .2.6.223.8.2.2.1.7 to .2.6.223.8.2.2.1.14 #ibmPduOutletCurrent my $outletcurrent = ".1.3.6.1.4.1.2.6.223.8.2.2.1.7"; #ibmPduOutletMaxCapacity my $outletmaxcap = ".1.3.6.1.4.1.2.6.223.8.2.2.1.8"; #ibmPduOutletCurrentThresholdWarning my $currentthrewarning = ".1.3.6.1.4.1.2.6.223.8.2.2.1.9"; #ibmPduOutletCurrentThresholdCritical my $currentthrecrit = ".1.3.6.1.4.1.2.6.223.8.2.2.1.10"; #ibmPduOutletLastPowerReading my $lastpowerreading = ".1.3.6.1.4.1.2.6.223.8.2.2.1.13"; for (my $outlet = 1; $outlet <= $count; $outlet++) { $output = $session->get("$outletcurrent.$outlet"); xCAT::SvrUtils::sendmsg("outlet $outlet Current: $output mA", $callback,$pdu); $output = $session->get("$outletmaxcap.$outlet"); xCAT::SvrUtils::sendmsg("outlet $outlet Max Capacity of the current: $output mA", $callback,$pdu); $output = $session->get("$currentthrewarning.$outlet"); xCAT::SvrUtils::sendmsg("outlet $outlet Current Threshold Warning: $output mA", $callback,$pdu); $output = $session->get("$currentthrecrit.$outlet"); xCAT::SvrUtils::sendmsg("outlet $outlet Current Threshold Critical: $output mA", $callback,$pdu); $output = $session->get("$lastpowerreading.$outlet"); xCAT::SvrUtils::sendmsg("outlet $outlet Last Power Reading: $output Watts", $callback,$pdu); } } #------------------------------------------------------- =head3 relaystat process individual relay stat for CR PDU. The OID for 3 relay: 1.3.6.1.4.1.2.6.262.15.2.13 1.3.6.1.4.1.2.6.262.15.2.14 1.3.6.1.4.1.2.6.262.15.2.15 =cut #------------------------------------------------------- sub relaystat { my $pdu = shift; my $relay = shift; my $snmpcmd = shift; my $callback = shift; my $relayoid = $relay + 12; #default pdu snmpv3, won't show up for snmpv1 my $cmd = "$snmpcmd $pdu 1.3.6.1.4.1.2.6.262.15.2.$relayoid"; my $result = xCAT::Utils->runcmd($cmd, 0); my ($msg,$stat) = split /: /, $result; if ($stat eq "1" ) { xCAT::SvrUtils::sendmsg(" relay $relay is on", $callback, $pdu, %allerrornodes); } elsif ( $stat eq "0" ) { xCAT::SvrUtils::sendmsg(" relay $relay is off", $callback, $pdu, %allerrornodes); } else { xCAT::SvrUtils::sendmsg(" relay $relay is $stat=unknown", $callback, $pdu, %allerrornodes); } return; } #------------------------------------------------------- =head3 process_powerrelay process relay action for CR PDU. =cut #------------------------------------------------------- sub process_powerrelay { my $request = shift; my $subreq = shift; my $subcmd = shift; my $callback = shift; my $relay; my $action; my $extrargs = $request->{arg}; my @exargs = ($request->{arg}); if (ref($extrargs)) { @exargs = @$extrargs; } my $nodes = $request->{node}; foreach my $cmd (@exargs) { if ($cmd =~ /=/ ) { my ($key, $value) = split(/=/, $cmd); $relay = $value; } else { $action = $cmd; } } if ( (defined $relay) && (defined $action) ) { my $relay_count = 1; foreach my $pdu (@$nodes) { process_relay($pdu, $action, $callback, $relay, $relay_count); } } else { xCAT::SvrUtils::sendmsg(" This command is not support, please define relay number and action", $callback); } } #------------------------------------------------------- =head3 process_relay process relay action for CR PDU. =cut #------------------------------------------------------- sub process_relay { my $pdu = shift; my $subcmd = shift; my $callback = shift; my $relay_num = shift; my $relay_count = shift; if ( !defined $relay_count ) { $relay_num = 1; $relay_count = 3; } my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodeAttribs($pdu,['ip','otherinterfaces']); my $username = $pduhash->{$pdu}->[0]->{username}; my $passwd = $pduhash->{$pdu}->[0]->{password}; # connect to PDU my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; my ($session, $errstr) = session_connect($static_ip, $discover_ip,$username,$passwd); my $ret; my $err; my $statestr; for (my $i = 0; $i < $relay_count; $i++) { my $relay = $relay_num; xCAT::SvrUtils::sendmsg(" power $subcmd for relay $relay_num", $callback,$pdu); if ($subcmd eq "off") { relay_action($session, $pdu, $relay, "OFF", $callback); } elsif ( $subcmd eq "on") { relay_action($session, $pdu, $relay, "ON", $callback); } elsif ( $subcmd eq "reset") { relay_action($session, $pdu, $relay, "OFF", $callback); relay_action($session, $pdu, $relay, "ON", $callback); } else { xCAT::SvrUtils::sendmsg(" subcmd $subcmd is not support", $callback,$pdu); } $relay_num++; } $session->hard_close(); } #------------------------------------------------------- =head3 realy_action process individual relay action for CR PDU. =cut #------------------------------------------------------- sub relay_action { my $session = shift; my $pdu = shift; my $relay = shift; my $action = shift; my $callback = shift; my ($ret, $err) = session_exec($session, "/dev/shm/bin/PduManager -r $relay -v $action"); if (defined $err) { xCAT::SvrUtils::sendmsg("Failed to process relay action: $err", $callback); } if (defined $ret) { xCAT::SvrUtils::sendmsg("$ret", $callback,$pdu); } } #------------------------------------------------------- =head3 process_snmpcfg config snmp and snmpv3 for CR PDU. =cut #------------------------------------------------------- sub process_snmpcfg { my $noderange = shift; my $subcmd = shift; my $callback = shift; my $snmp_conf="/etc/snmp/snmpd.conf"; my $xCATSettingsSTART="xCAT settings START"; my $xCATSettingsEND="xCAT settings END"; my $xCATSettingsInfo="Entries between the START and END lines will be replaced each time by xCAT command"; my $nodetab = xCAT::Table->new('hosts'); my $nodehash = $nodetab->getNodesAttribs($noderange,['ip','otherinterfaces']); foreach my $pdu (@$noderange) { unless ($pduhash->{$pdu}->[0]->{pdutype} eq "crpdu") { xCAT::SvrUtils::sendmsg("This command only supports CONSTELLATION PDU with pdutype=crpdu", $callback,$pdu); next; } my $community = $pduhash->{$pdu}->[0]->{community}; my $snmpversion = $pduhash->{$pdu}->[0]->{snmpversion}; my $snmpuser = $pduhash->{$pdu}->[0]->{snmpuser}; my $authtype = $pduhash->{$pdu}->[0]->{authtype}; if (!defined $authtype) { $authtype="MD5"; } my $authkey = $pduhash->{$pdu}->[0]->{authkey}; my $privtype = $pduhash->{$pdu}->[0]->{privtype}; if (!defined $privtype) { $privtype="DES"; } my $privkey = $pduhash->{$pdu}->[0]->{privkey}; if (!defined $privkey) { if (defined $authkey) { $privkey=$authkey; } } my $seclevel = $pduhash->{$pdu}->[0]->{seclevel}; # connect to PDU my $static_ip = $nodehash->{$pdu}->[0]->{ip}; my $discover_ip = $nodehash->{$pdu}->[0]->{otherinterfaces}; my $username = $pduhash->{$pdu}->[0]->{username}; my $password = $pduhash->{$pdu}->[0]->{password}; my ($exp, $errstr) = session_connect($static_ip, $discover_ip,$username,$password); my $ret; my $err; ($ret, $err) = session_exec($exp, "sed -i '/$xCATSettingsSTART/,/$xCATSettingsEND/ d' $snmp_conf"); ($ret, $err) = session_exec($exp, "echo '# $xCATSettingsSTART' >> $snmp_conf"); ($ret, $err) = session_exec($exp, "echo '# $xCATSettingsInfo' >> $snmp_conf"); if (defined $community) { ($ret, $err) = session_exec($exp, "echo 'com2sec readwrite default $community' >> $snmp_conf"); } #set snmpv3 configuration if ($snmpversion =~ /3/) { if ((defined $snmpuser) && (defined $seclevel)) { my $msg1; if ($seclevel eq "authNoPriv") { $msg1 = "createUser $snmpuser $authtype $authkey"; } elsif ($seclevel eq "authPriv") { $msg1 = "createUser $snmpuser $authtype $authkey $privtype $privkey"; } else { #default to notAuthNoPriv $msg1 = "createUser $snmpuser"; } my $msg2 = "rwuser $snmpuser $seclevel .1.3.6.1.4.1.2.6.262"; ($ret, $err) = session_exec($exp, "sed -i '/\"$snmpuser\"/ d' /var/lib/net-snmp/snmpd.conf"); ($ret, $err) = session_exec($exp, "echo $msg1 >> $snmp_conf"); ($ret, $err) = session_exec($exp, "echo $msg2 >> $snmp_conf"); } else { xCAT::SvrUtils::sendmsg("Need to define user name and security level for snmpv3 configuration", $callback); } } ($ret, $err) = session_exec($exp, "echo '# $xCATSettingsEND' >> $snmp_conf"); #need to restart snmpd after config file changes ($ret, $err) = session_exec($exp, "ps | grep snmpd | grep -v grep | awk '{ print $1}' | xargs kill -9"); ($ret, $err) = session_exec($exp, "/usr/sbin/snmpd -Lsd -Lf /dev/null -p /var/run/snmpd"); if (defined $err) { xCAT::SvrUtils::sendmsg("Failed to configure snmp : $err", $callback); } $exp->hard_close(); } } #------------------------------------------------------- =head3 netcfg_for_irpdu change hostname and network setting for IR PDU. =cut #------------------------------------------------------- sub netcfg_for_irpdu { my $pdu = shift; my $static_ip = shift; my $discover_ip = shift; my $request = shift; my $subreq = shift; my $callback = shift; my $hostname; my $ip; my $gateway; my $netmask; my $extrargs = $request->{arg}; my @exargs = ($request->{arg}); if (ref($extrargs)) { @exargs = @$extrargs; } #get user/password from pdu table if defined #default password for irpdu my $passwd = "1001"; my $username = "ADMIN"; if ($pduhash->{$pdu}->[0]->{username}) { $username = $pduhash->{$pdu}->[0]->{username}; } if ($pduhash->{$pdu}->[0]->{password}) { $passwd = $pduhash->{$pdu}->[0]->{password}; } my $timeout = 10; my $send_change = "N"; my $login_ip; # somehow, only system command works for checking if irpdu is pingable # Net::Ping Module and xCAT::NetworkUtils::isPingable both are not working if (system("ping -c 2 $static_ip") == 0 ) { $login_ip = $static_ip; } elsif (system("ping -c 2 $discover_ip") == 0) { $login_ip = $discover_ip; } else { xCAT::SvrUtils::sendmsg(" is not reachable", $callback,$pdu); return; } foreach my $cmd (@exargs) { my ($key, $value) = split(/=/, $cmd); if ($key =~ /hostname/) { $hostname = $value; xCAT::SvrUtils::sendmsg("change pdu hostname to $hostname", $callback); } if ($key =~ /ip/) { $ip = $value; $send_change = "Y"; xCAT::SvrUtils::sendmsg("change ip address for $pdu to $ip", $callback); } if ($key =~ /gateway/) { $gateway = $value; $send_change = "Y"; xCAT::SvrUtils::sendmsg("change gateway for $pdu to $gateway", $callback); } if ($key =~ /netmask/) { $netmask = $value; $send_change = "Y"; xCAT::SvrUtils::sendmsg("change netmask for $pdu to $netmask", $callback); } } my $login_cmd = "telnet $login_ip\r"; my $user_prompt = " Login: "; my $pwd_prompt = "Password: "; my $pdu_prompt = "Please Enter Your Selection => "; my $send_zero = 0; my $send_one = 1; my $send_two = 2; my $mypdu = new Expect; $mypdu->debug(0); $mypdu->log_stdout(0); # suppress stdout output.. unless ($mypdu->spawn($login_cmd)) { $mypdu->soft_close(); xCAT::SvrUtils::sendmsg("Unable to run $login_cmd", $callback); return; } my @result = $mypdu->expect( $timeout, [ $user_prompt, sub { $mypdu->clear_accum(); $mypdu->send("$username\r"); $mypdu->clear_accum(); $mypdu->exp_continue(); } ], [ $pwd_prompt, sub { $mypdu->clear_accum(); $mypdu->send("$passwd\r"); $mypdu->clear_accum(); $mypdu->exp_continue(); } ], [ $pdu_prompt, sub { $mypdu->clear_accum(); $mypdu->send("$send_one\r"); $mypdu->send("$send_one\r"); #change hostname $mypdu->send("$send_one\r"); $mypdu->send("$hostname\r"); $mypdu->send("$send_zero\r"); #change network setting $mypdu->send("$send_two\r"); $mypdu->send("$send_one\r"); $mypdu->send("$ip\r"); $mypdu->send("$gateway\r"); $mypdu->send("$netmask\r"); $mypdu->send("$send_change\r"); # go back Previous Menu $mypdu->send("$send_zero\r"); $mypdu->send("$send_zero\r"); } ], ); if (defined($result[1])) { my $errmsg = $result[1]; $mypdu->soft_close(); xCAT::SvrUtils::sendmsg("Failed expect command: $errmsg", $callback,$pdu); return; } $mypdu->soft_close(); xCAT::SvrUtils::sendmsg("hostname or network setting changed, update node definition ", $callback); xCAT::Utils->runxcmd({ command => ['chdef'], arg => ['-t','node','-o',$pdu,"otherinterfaces="] }, $subreq, 0, 1); if ( (defined $ip) and ($static_ip ne $ip) ) { xCAT::Utils->runxcmd({ command => ['chdef'], arg => ['-t','node','-o',$pdu,"ip=$ip",'status=configured'] }, $subreq, 0, 1); xCAT::Utils->runxcmd({ command => ['makehosts'], node => [$pdu] }, $subreq, 0, 1); } return; } 1;