# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html package xCAT::PPCcfg; use strict; use Getopt::Long; use xCAT::PPCcli qw(SUCCESS EXPECT_ERROR RC_ERROR NR_ERROR); use xCAT::Usage; use Storable qw(freeze thaw); use POSIX "WNOHANG"; use xCAT::NetworkUtils; use xCAT::MsgUtils qw(verbose_message); use LWP; use HTTP::Cookies; ########################################## # Globals ########################################## my %rspconfig = ( sshcfg => \&sshcfg, frame => \&frame, hostname => \&hostname ); my %rsp_result; my $start; ########################################################################## # Parse the command line for options and operands ########################################################################## sub parse_args { my $request = shift; my $command = $request->{command}; my $args = $request->{arg}; my %opt = (); my %cmds = (); my @fsp = ( "memdecfg", "decfg", "procdecfg", "iocap", "time", "date", "autopower", "sysdump", "spdump", "network", "HMC_passwd", "admin_passwd", "general_passwd", "*_passwd", "hostname", "resetnet", "dev", "celogin1" ); my @bpa = ( "frame", "password", "newpassword", "HMC_passwd", "admin_passwd", "general_passwd", "*_passwd", "hostname", "resetnet", "dev", "celogin1" ); my @ppc = ( "sshcfg" ); my %rsp = ( cec=> \@fsp, frame=>\@bpa, fsp => \@fsp, bpa => \@bpa, ivm => \@ppc, hmc => \@ppc ); ############################################# # Get support command list ############################################# #my $typetab = xCAT::Table->new( 'nodetype' ); #my $nodes = $request->{node}; #foreach (@$nodes) { # if ( defined( $typetab )) { # my ($ent) = $typetab->getAttribs({ node=>$_},'nodetype'); # if ( defined($ent) ) { # $request->{hwtype} = $ent->{nodetype}; # last; # } # # } # #} my $nodes = $request->{node}; my $typehash = xCAT::DBobjUtils->getnodetype($nodes); foreach my $nn (@$nodes) { $request->{hwtype} = $$typehash{$nn}; last if ($request->{hwtype}); } my $supported = $rsp{$request->{hwtype}}; ############################################# # Responds with usage statement ############################################# local *usage = sub { my $usage_string = xCAT::Usage->getUsage($command); return( [$_[0], $usage_string] ); }; ############################################# # Process command-line arguments ############################################# if ( !defined( $args )) { return(usage( "No command specified" )); } ############################################# # Checks case in GetOptions, allows opts # to be grouped (e.g. -vx), and terminates # at the first unrecognized option. ############################################# @ARGV = @$args; $Getopt::Long::ignorecase = 0; Getopt::Long::Configure( "bundling" ); $request->{method} = undef; if ( !GetOptions( \%opt, qw(V|verbose resetnet))) { return( usage() ); } #################################### # Check for "-" with no option #################################### if ( grep(/^-$/, @ARGV )) { return(usage( "Missing option: -" )); } #################################### # Check for "=" with no argument #################################### if (my ($c) = grep(/=$/, @ARGV )) { return(usage( "Missing argument: $c" )); } #################################### # Check for unsupported commands #################################### foreach my $arg ( @ARGV ) { my ($command,$value) = split( /=/, $arg ); if ( !grep( /^$command$/, @$supported) and !$opt{resetnet}) { return(usage( "Invalid command: $arg" )); } if ( exists( $cmds{$command} )) { return(usage( "Command multiple times: $command" )); } $cmds{$command} = $value; } #################################### # Check command arguments #################################### foreach ( keys %cmds ) { if ( $cmds{$_} ) { my $result = parse_option( $request, $_, $cmds{$_} ); if ( $result ) { return( usage($result) ); } } elsif ($_ =~ /_passwd$/) { return( usage("No argument specified for '$_'")); } } { if ($request->{dev} eq '1' && $request->{other} eq '1') { return ( usage("Invalid command arrays")); } # my $result = parse_dev_option( $request, \%cmds); # if ($result) { # return ( usage($result)); # } } #################################### # Return method to invoke #################################### if ( $request->{hwtype} =~ /(^hmc|ivm)$/ ) { $request->{method} = "cfg"; return( \%opt ); } #################################### # Return method to invoke #################################### if ( exists($cmds{frame}) or exists($cmds{hostname}) ) { $request->{hcp} = "hmc"; $request->{method} = "cfg"; return( \%opt ); } #################################### # Return method to invoke #################################### if ( $opt{resetnet} ) { $request->{hcp} = "hmc"; $request->{method} = "resetnet"; return( \%opt ); } #################################### # Return method to invoke #################################### if ( exists($cmds{HMC_passwd}) or exists($cmds{general_passwd}) or exists($cmds{admin_passwd}) or exists($cmds{"*_passwd"}) ) { $request->{hcp} = "hmc"; $request->{method} = "passwd"; return( \%opt ); } $request->{method} = \%cmds; return( \%opt ); } sub parse_dev_option{ my $req = shift; my $cmds = shift; foreach my $cmd (keys %$cmds) { if ( $cmd =~ /^(dev|celogin1)$/ ) { if ($cmds->{$cmd} and ($cmds->{$cmd} !~ /^(enable|disable)$/i) ) { return( "Invalid argument ".$cmds->{$cmd}." for ".$cmd ); } $req->{dev} = 1; } else { $req->{other} = 1; } } if ($req->{dev} eq '1' && $req->{other} eq '1') { return ("Invalid command arrays"); } return undef; } ########################################################################## # Parse the command line optional arguments ########################################################################## sub parse_option { my $request = shift; my $command = shift; my $value = shift; #################################### # Set/get time #################################### if ( $command =~ /^time$/ ) { if ( $value !~ /^([0-1]?[0-9]|2[0-3]):(0?[0-9]|[1-5][0-9]):(0?[0-9]|[1-5][0-9])$/){ return( "Invalid time format '$value'" ); } } #################################### # Set/get date #################################### if ( $command =~ /^date$/ ) { if ( $value !~ /^(0?[1-9]|1[012])-(0?[1-9]|[12][0-9]|3[01])-(20[0-9]{2})$/){ return( "Invalid date format '$value'" ); } } #################################### # Set/get options #################################### if ( $command =~ /^(autopower|iocap|sshcfg)$/ ) { if ( $value !~ /^(enable|disable)$/i ) { return( "Invalid argument '$value'" ); } } #################################### # Deconfiguration policy #################################### if ( $command =~ /^decfg$/ ) { if ( $value !~ /^(enable|disable):.*$/i ) { return( "Invalid argument '$value'" ); } } #################################### # Processor deconfiguration #################################### if ( $command =~ /^procdecfg$/ ) { if ( $value !~ /^(configure|deconfigure):\d+:(all|[\d,]+)$/i ) { return( "Invalid argument '$value'" ); } } ################################ # Memory deconfiguration ################################ elsif ( $command =~ /^memdecfg$/ ) { if ($value !~/^(configure|deconfigure):\d+:(unit|bank):(all|[\d,]+)$/i){ return( "Invalid argument '$value'" ); } } if ( $command eq 'network'){ my ( $adapter_name, $ip, $host, $gateway, $netmask) = split /,/, $value; return ( "Network interface name is required") if ( ! $adapter_name); return ( "Invalide network interface name $adapter_name") if ( $adapter_name !~ /^eth\d$/); return undef if ( $ip eq '*'); return ( "Invalid IP address format") if ( $ip and $ip !~ /\d+\.\d+\.\d+\.\d+/); return ( "Invalid netmask format") if ( $netmask and $netmask !~ /\d+\.\d+\.\d+\.\d+/); } if ( $command eq 'frame' ){ if ( $value !~ /^\d+$/i && $value ne '*' ) { return( "Invalid frame number '$value'" ); } } if ( $command eq 'admin_passwd' or $command eq 'general_passwd' or $command eq '*_passwd' ){ my ($passwd,$newpasswd) = split /,/, $value; if ( !$passwd or !$newpasswd) { return( "Current password and new password couldn't be empty" ); } } if ( $command eq 'HMC_passwd' ) { my ($passwd,$newpasswd) = split /,/, $value; if ( !$newpasswd ) { return( "New password couldn't be empty for user 'HMC'" ); } } if ( $command eq 'dev' or $command eq 'celogin1' ) { if ($value !~ /^(enable|disable)$/i ) { return( "Invalid argument '$value'" ); } $request->{dev} = 1; } else { $request->{other} = 1; } return undef; } ########################################################################## # Update passwords for different users on FSP/BPA ########################################################################## sub passwd { my $request = shift; my $hash = shift; my $exp = shift; my $args = $request->{arg}; my $result; my $users; foreach my $arg ( @$args ) { my ($user,$value) = split /=/, $arg; my ($passwd,$newpasswd) = split /,/, $value; $user =~ s/_passwd$//; $user =~ s/^HMC$/access/g; if ( $user eq "*" ) { push @$users, "access"; push @$users, "admin"; push @$users, "general"; } else { push @$users, $user; } foreach my $usr ( @$users ) { while ( my ($cec,$h) = each(%$hash) ) { while ( my ($node,$d) = each(%$h) ) { my $type = @$d[4]; xCAT::MsgUtils->verbose_message($request, "rspconfig :modify password of $usr for node:$node."); my $data = xCAT::PPCcli::chsyspwd( $exp, $usr, $type, $cec, $passwd, $newpasswd ); my $Rc = shift(@$data); my $usr_back = $usr; $usr_back =~ s/^access$/HMC/g; push @$result, [$node,"$usr_back: @$data[0]",$Rc]; ################################## # Write the new password to table ################################## if ( $Rc == SUCCESS ) { xCAT::MsgUtils->verbose_message($request, "rspconfig :update xCATdb for node:$node,ID:$usr_back."); xCAT::PPCdb::update_credentials( $node, $type, $usr_back, $newpasswd ); } } } } } return( [@$result] ); } ########################################################################## # Handles all PPC rspconfig commands ########################################################################## sub cfg { my $request = shift; my $hash = shift; my $exp = shift; my $args = $request->{arg}; my $result; foreach ( @$args ) { ################################## # Ignore switches in command-line ################################## unless ( /^-/ ) { my ($cmd,$value) = split /=/; no strict 'refs'; $result = $rspconfig{$cmd}( $request, $exp, $value, $hash ); use strict; } } return( $result ); } ########################################################################## # Enables/disables/displays SSH access to HMC/IVM ########################################################################## sub sshcfg { my $request = shift; my $exp = shift; my $mode = shift; my $server = @$exp[3]; my $userid = @$exp[4]; my $fname = ((xCAT::Utils::isAIX()) ? "/.ssh/":"/root/.ssh/")."id_rsa.pub"; my $auth = "/home/$userid/.ssh/authorized_keys2"; ##################################### # Get SSH key on Management Node ##################################### unless ( open(RSAKEY,"<$fname") ) { return( [[$server,"Error opening '$fname'",RC_ERROR]] ); } my ($sshkey) = ; close(RSAKEY); ##################################### # userid@host not found in key file ##################################### if ( $sshkey !~ /\s+(\S+\@\S+$)/ ) { return( [[$server,"Cannot find userid\@host in '$fname'",RC_ERROR]] ); } my $logon = $1; ##################################### # Determine if SSH is enabled ##################################### if ( !defined( $mode )) { xCAT::MsgUtils->verbose_message($request, "rspconfig :check sshcfg for user:$logon on node:$server."); my $result = xCAT::PPCcli::send_cmd( $exp, "cat $auth" ); my $Rc = shift(@$result); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { return( [[$server,@$result[0],$Rc]] ); } ################################# # Find logon in key file ################################# foreach ( @$result ) { if ( /$logon$/ ) { return( [[$server,"enabled",SUCCESS]] ); } } return( [[$server,"disabled",SUCCESS]] ); } ##################################### # Enable/disable SSH ##################################### xCAT::MsgUtils->verbose_message($request, "rspconfig :sshcfg $mode for user:$logon on node:$server."); my $result = xCAT::PPCcli::mkauthkeys( $exp, $mode, $logon, $sshkey ); my $Rc = shift(@$result); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { return( [[$server,@$result[0],$Rc]] ); } return( [[$server,lc($mode."d"),SUCCESS]] ); } sub frame { my $request = shift; my $exp = shift; my $value = shift; my $hash = shift; my $arg = $request->{arg}; foreach ( @$arg ) { my $result; my $Rc; my $data; my ($cmd, $value) = split /=/, $_; if ( $cmd ne "frame" ) { return( [[@$exp[2],"Multiple option $cmd and frame is not accepted",SUCCESS]] ); } ################################# # Open xCAT database to sync with # the frame number between hcp # and database ################################# my $tab = xCAT::Table->new( "ppc" ); while ( my ($cec,$h) = each(%$hash) ) { while ( my ($node,$d) = each(%$h) ) { if ( !defined($value) ) { ################################# # Get frame number ################################# xCAT::MsgUtils->verbose_message($request, "rspconfig :get frame_num for node:$node."); $data = xCAT::PPCcli::lssyscfg( $exp, @$d[4], @$d[2], 'frame_num' ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { return( [[$node,@$data[0],$Rc]] ); } push @$result, [$node,@$data[0],SUCCESS]; ################################# # Set frame number to database ################################# $tab->setNodeAttribs( $node, { id=>@$data[0] } ); } elsif ( $value eq '*' ) { ################################# # Set frame number # Read the settings from database ################################# my $ent=$tab->getNodeAttribs( $node,['id'] ); ################################# # Return error ################################# if ( !defined($ent) or !defined($ent->{id}) ) { return( [[$node,"Cannot find frame num in database",RC_ERROR]] ); } xCAT::MsgUtils->verbose_message($request, "rspconfig :set frame_num=".$ent->{id}." for node:$node."); $data = xCAT::PPCcli::chsyscfg( $exp, "bpa", $d, "frame_num=".$ent->{id} ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { return( [[$node,@$data[0],$Rc]] ); } push @$result, [$node,@$data[0],SUCCESS]; } else { ################################# # Set frame number # Read the frame number from opt ################################# xCAT::MsgUtils->verbose_message($request, "rspconfig :set frame_num=$value for node:$node."); $data = xCAT::PPCcli::chsyscfg( $exp, "bpa", $d, "frame_num=$value" ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { return( [[$node,@$data[0],$Rc]] ); } push @$result, [$node,@$data[0],SUCCESS]; ################################# # Set frame number to database ################################# xCAT::MsgUtils->verbose_message($request, "rspconfig : set frame_num, update node:$node attr id=$value."); $tab->setNodeAttribs( $node, { id=>$value } ); } } return( [@$result] ); } } } sub hostname { my $request = shift; my $exp = shift; my $value = shift; my $hash = shift; my $arg = $request->{arg}; my $result; foreach ( @$arg ) { my $data; my $Rc; my ($cmd, $value) = split /=/, $_; if ( $cmd ne "hostname" ) { return( [[@$exp[2],"Multiple option $cmd and hostname is not accepted",SUCCESS]] ); } while ( my ($cec,$h) = each(%$hash) ) { while ( my ($node,$d) = each(%$h) ) { if ( !defined($value) ) { ################################# # Get system name ################################# xCAT::MsgUtils->verbose_message($request, "rspconfig :get system name for node:$node."); $data = xCAT::PPCcli::lssyscfg( $exp, @$d[4], @$d[2], 'name' ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { push @$result, [$node,@$data[0],$Rc]; } push @$result, [$node,@$data[0],SUCCESS]; } elsif ( $value eq '*' ) { xCAT::MsgUtils->verbose_message($request, "rspconfig :set system name:$node for node:$node."); $data = xCAT::PPCcli::chsyscfg( $exp, @$d[4], $d, "new_name=$node" ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { push @$result, [$node,@$data[0],$Rc]; } push @$result, [$node,@$data[0],SUCCESS]; } else { xCAT::MsgUtils->verbose_message($request, "rspconfig :set system name:$value for node:$node."); $data = xCAT::PPCcli::chsyscfg( $exp, @$d[4], $d, "new_name=$value" ); $Rc = shift(@$data); ################################# # Return error ################################# if ( $Rc != SUCCESS ) { push @$result, [$node,@$data[0],$Rc]; } push @$result, [$node,@$data[0],SUCCESS]; } } } } return( [@$result] ); } ########################################################################## # Do resetnet public entry ########################################################################## sub resetnet { my $request = shift; doresetnet($request); return 0; } ########################################################################## # Reset the network interfraces if necessary ########################################################################## sub doresetnet { my $req = shift; my %iphash; my $targets; my $result; my %grouphash; my %oihash; my %machash; my %vpdhash; unless ($req) { send_msg( $req, 1, "request is empty, return" ); return; } ########################################### # prepare to reset network ########################################### xCAT::MsgUtils->verbose_message($req, "rspconfig :do resetnet begin to phase nodes"); my $hoststab = xCAT::Table->new( 'hosts' ); if ( !$hoststab ) { send_msg( $req, 1, "Error open hosts table" ); return; } else { my @hostslist = $hoststab->getAllNodeAttribs(['node','otherinterfaces']); foreach my $otherentry ( @hostslist) { $oihash{$otherentry->{node}} = $otherentry->{otherinterfaces}; } } my $mactab = xCAT::Table->new( 'mac' ); if ( !$mactab ) { send_msg( $req, 1, "Error open mac table" ); return; }else{ my @maclist = $mactab->getAllNodeAttribs(['node','mac']); foreach my $macentry (@maclist) { $machash{$macentry->{node}} = $macentry->{mac}; } } $mactab = (); my $vpdtab = xCAT::Table->new( 'vpd' ); if ( !$vpdtab ) { send_msg( $req, 1, "Error open vpd table" ); return; } else { my @vpdlist = $vpdtab->getAllNodeAttribs(['node','mtm','serial','side']); foreach my $vpdentry (@vpdlist) { if ($vpdentry->{side} =~ /(\w)\-\w/) { my $side = $1; $vpdhash{$vpdentry->{node}} = $vpdentry->{mtm}."*".$vpdentry->{serial}."*".$side; } } } $vpdtab = (); unless ( $req->{node} ) { send_msg( $req, 0, "no node specified" ); return; } ########################################### # Process nodes and get network information ########################################### my $nodetype = $req->{hwtype}; if ( $nodetype =~ /^(cec|frame)$/ ) { # this brunch is just for the xcat 2.6(or 2.6+) database foreach my $nn ( @{ $req->{node}} ) { my $cnodep = xCAT::DBobjUtils->getchildren($nn); $nodetype = ( $nodetype =~ /^frame$/i ) ? "bpa" : "fsp"; if ($cnodep) { foreach my $cnode (@$cnodep) { my $ip = xCAT::NetworkUtils::getNodeIPaddress( $cnode ); my $oi = $oihash{$cnode}; if(!defined $ip) { send_msg($req, "doresetnet: can't get $cnode ip"); next; } if(!defined $oi) { send_msg($req, "doresetnet: can't get $cnode hosts.otherinterfaces"); next; } if ( exists($oihash{$cnode}) and $ip eq $oihash{$cnode}) { send_msg( $req, 0, "$cnode: same ip address, skipping $nn network reset" ); } elsif( ! exists $machash{$cnode}){ send_msg( $req, 0, "$cnode: no mac defined, skipping $nn network reset" ); } else { $iphash{$cnode}{sip} = $ip; $iphash{$cnode}{tip} = $oihash{$cnode}; if(exists $grouphash{$vpdhash{$cnode}}) { $grouphash{$vpdhash{$cnode}} .= ",$cnode"; } else { $grouphash{$vpdhash{$cnode}} = "$cnode"; } $targets->{$nodetype}->{$ip}->{'args'} = "0.0.0.0,$cnode"; $targets->{$nodetype}->{$ip}->{'mac'} = $machash{$cnode}; $targets->{$nodetype}->{$ip}->{'name'} = $cnode; $targets->{$nodetype}->{$ip}->{'ip'} = $ip; $targets->{$nodetype}->{$ip}->{'type'} = $nodetype; my %netinfo = xCAT::DBobjUtils->getNetwkInfo( [$ip] ); $targets->{$nodetype}->{$ip}->{'args'} .= ",$netinfo{$ip}{'gateway'},$netinfo{$ip}{'mask'}"; #xCAT::MsgUtils->verbose_message($req, "doresetnet: get node $cnode info $targets->{$nodetype}->{$ip}->{'args'}, ip is $ip"); $targets->{$nodetype}->{$oi}->{'args'} = "0.0.0.0,$cnode"; $targets->{$nodetype}->{$oi}->{'mac'} = $machash{$cnode}; $targets->{$nodetype}->{$oi}->{'name'} = $cnode; $targets->{$nodetype}->{$oi}->{'ip'} = $oi; $targets->{$nodetype}->{$oi}->{'type'} = $nodetype; %netinfo = xCAT::DBobjUtils->getNetwkInfo( [$oi] ); $targets->{$nodetype}->{$oi}->{'args'} .= ",$netinfo{$oi}{'gateway'},$netinfo{$oi}{'mask'}"; #xCAT::MsgUtils->verbose_message($req, "doresetnet: get node $cnode info $targets->{$nodetype}->{$oi}->{'args'}, oi is $oi"); } } } else { send_msg( $req, 1, "Can't get the fsp/bpa nodes for the $nn" ); return; } } # this brunch is just for the xcat 2.5(or 2.5-) databse } elsif ( $nodetype =~ /^(fsp|bpa)$/ ) { foreach my $nn ( @{ $req->{node}} ) { my $ip = xCAT::NetworkUtils::getNodeIPaddress( $nn ); if(!defined $ip) { send_msg($req, "doresetnet: can't get $nn ip"); next; } if(!exists $oihash{$nn}) { send_msg($req, "doresetnet: can't get $nn hosts.otherinterfaces"); next; } my $oi = $oihash{$nn}; if( exists($oihash{$nn}) and $ip eq $oihash{$nn}) { send_msg( $req, 0, "$nn: same ip address, skipping network reset" ); } elsif (!exists $machash{$nn}){ send_msg( $req, 0, "$nn: no mac defined, skipping network reset" ); } else { $iphash{$nn}{sip} = $ip; $iphash{$nn}{tip} = $oihash{$nn}; if(exists $grouphash{$vpdhash{$nn}}) { $grouphash{$vpdhash{$nn}} .= ",$nn"; } else { $grouphash{$vpdhash{$nn}} = "$nn"; } $targets->{$nodetype}->{$ip}->{'args'} = "0.0.0.0,$nn"; $targets->{$nodetype}->{$ip}->{'mac'} = $machash{$nn}; $targets->{$nodetype}->{$ip}->{'name'} = $nn; $targets->{$nodetype}->{$ip}->{'ip'} = $ip; $targets->{$nodetype}->{$ip}->{'type'} = $nodetype; my %netinfo = xCAT::DBobjUtils->getNetwkInfo( [$ip] ); $targets->{$nodetype}->{$ip}->{'args'} .= ",$netinfo{$ip}{'gateway'},$netinfo{$ip}{'mask'}"; #xCAT::MsgUtils->verbose_message($req, "doresetnet: get node $nn info $targets->{$nodetype}->{$ip}->{'args'},ip is $ip"); $targets->{$nodetype}->{$oi}->{'args'} = "0.0.0.0,$nn"; $targets->{$nodetype}->{$oi}->{'mac'} = $machash{$nn}; $targets->{$nodetype}->{$oi}->{'name'} = $nn; $targets->{$nodetype}->{$oi}->{'ip'} = $oi; $targets->{$nodetype}->{$oi}->{'type'} = $nodetype; %netinfo = xCAT::DBobjUtils->getNetwkInfo( [$oi] ); $targets->{$nodetype}->{$oi}->{'args'} .= ",$netinfo{$oi}{'gateway'},$netinfo{$oi}{'mask'}"; #xCAT::MsgUtils->verbose_message($req, "doresetnet: get node $nn info $targets->{$nodetype}->{$oi}->{'args'}, oi is $oi"); } } } elsif ( !$nodetype ){ send_msg( $req, 0, "no nodetype defined, skipping network reset" ); return; } else { send_msg( $req, 0, "$nodetype not supported, skipping network reset" ); return; } unless (%grouphash) { send_msg( $req, 0, "Failed to group the nodes, skipping network reset" ); return; } ########################################### # Update target hardware w/discovery info ########################################### my %rsp_dev = get_rsp_dev( $req, $targets); ###################################################### # Start to reset network. Fork one process per BPA/FSP ###################################################### %oihash = (); %machash = (); %vpdhash = (); send_msg( $req, 0, "\nStart to reset network..\n" ); $start = Time::HiRes::gettimeofday(); my $children = 0; $SIG{CHLD} = sub { while (waitpid(-1, WNOHANG) > 0) { $children--; } }; my $fds = new IO::Select; my $callback = $req->{callback}; foreach my $node ( keys %grouphash) { my %iphashfornode; my $gc = $grouphash{$node}; my %rsp_devfornode; foreach my $tn (split /,/, $gc) { $iphashfornode{$tn} = $iphash{$tn}; for my $ti (keys %{$iphash{$tn}}){ my $tip = $iphash{$tn}{$ti}; $rsp_devfornode{$tip} = $rsp_dev{$tip}; } } xCAT::MsgUtils->verbose_message($req, "========> begin to fork process for node $node"); ###################################################### # Begin fork ###################################################### my $pipe; my $rspdevref = \%rsp_devfornode; my $grouphashref = $gc; my $iphashref = \%iphashfornode; my $result; my @data = ("RSPCONFIG6sK4ci"); ####################################### # Pipe childs output back to parent ####################################### my $parent; my $child; pipe $parent, $child; my $pid = xCAT::Utils->xfork(); if ( !defined($pid) ) { ################################### # Fork error ################################### send_msg( $req, 1, "Fork error: $!" ); return undef; } elsif ( $pid == 0 ) { ################################### # Child process, clear memory first ################################### %rsp_dev = (); %grouphash = (); %iphash = (); close( $parent ); $req->{pipe} = $child; my @ns = split /,/, $grouphashref; my $ips ; my $retrytime = 0; while (scalar(@ns)) { foreach my $fspport (@ns) { my $ip = ${$iphashref->{$fspport}}{tip}; my $rc = system("ping -q -n -c 1 -w 1 $ip > /dev/null"); if ($rc != 0) { $ip = ${$iphashref->{$fspport}}{sip}; $rc = system("ping -q -n -c 1 -w 1 $ip > /dev/null"); if ($rc != 0) { xCAT::MsgUtils->verbose_message($req, "$fspport: Ping failed for both ips: ${$iphashref->{$fspport}}{tip}, ${$iphashref->{$fspport}}{sip}. Need to retry for fsp $fspport. Retry $retrytime"); # record verbose info, retry 10 times for the noping situation $retrytime++; if ($retrytime > 5) { while (@ns) { my $n1 = shift @ns; push @data, $n1; push @data, 1; xCAT::MsgUtils->verbose_message($req, "$n1: Ping failed for both ips: ${$iphashref->{$n1}}{tip}, ${$iphashref->{$n1}}{sip}. Need to retry for fsp $n1. return"); } } } else { xCAT::MsgUtils->verbose_message($req, "$fspport: has got new ip $ip"); my @temp; foreach my $tn (@ns){ push @temp, $tn unless ($tn eq $fspport); } @ns = @temp; push @data, $fspport; push @data, 0; } } else { xCAT::MsgUtils->verbose_message($req, "$fspport: Begin to use ip $ip to loggin"); $result = invoke_cmd( $req, $ip, $rspdevref); xCAT::MsgUtils->verbose_message($req, "$fspport: resetnet result is @$result[0]"); if (@$result[0] == 0) { my @temp; foreach my $tn (@ns){ push @temp, $tn unless ($tn eq $fspport); } @ns = @temp; push @data, $fspport; push @data, 0; #push @data, @$result[2]; xCAT::MsgUtils->verbose_message($req, "$fspport: reset successfully for node $fspport"); } else { xCAT::MsgUtils->verbose_message($req, "$fspport: reset fail for node $fspport, retrying..."); } } } # retry time less than 1 min. my $elapsed = Time::HiRes::gettimeofday() - $start; if ($elapsed > 60) { while (@ns) { my $n1 = shift @ns; push @data, $n1; push @data, 1; xCAT::MsgUtils->verbose_message($req, "$n1: ASMI retry failed, return."); } } } #################################### # Pass result array back to parent #################################### my $out = $req->{pipe}; print $out freeze( \@data ); print $out "\nENDOFFREEZE6sK4ci\n"; exit(0); } else { ################################### # Parent process ################################### close( $child ); $pipe = $parent ; } if ( $pipe ) { $fds->add( $pipe ); $children++; } } ############################################# # Process responses from children ############################################# while ( $children > 0 ) { child_response( $callback, $fds ); } while (child_response($callback,$fds)) {} my $elapsed = Time::HiRes::gettimeofday() - $start; my $msg = sprintf( "Total rspconfig Time: %.3f sec\n", $elapsed ); xCAT::MsgUtils->verbose_message($req, $msg); my @failed_node; my @succeed_node; foreach my $ip ( keys %rsp_result ) { ################################# # Error logging on ################################# my $result = $rsp_result{$ip}; my $Rc = shift(@$result); if ( $Rc != SUCCESS ) { push @failed_node, $ip; } else { push @succeed_node, $ip; } } ########################################### # Print the result ########################################### $result = "\nReset network failed nodes:\n"; foreach my $fspport ( @failed_node ) { $result .= $fspport.","; } $result .= "\nReset network succeed nodes:\n"; foreach my $fspport ( @succeed_node ) { $result .= $fspport.","; xCAT::MsgUtils->verbose_message( $req, "$fspport: Set the otherinterfaces value of $fspport to $iphash{$fspport}{sip} "); $hoststab->setNodeAttribs( $fspport ,{otherinterfaces=> $iphash{$fspport}{sip}} ); } $result .= "\nReset network finished.\n"; send_msg( $req, 0, $result ); return undef; } ############################################# # Get rsp devices and their logon info ############################################# sub get_rsp_dev { my $request = shift; my $targets = shift; my $mm = $targets->{'mm'} ? $targets->{'mm'} : {}; my $hmc = $targets->{'hmc'} ? $targets->{'hmc'}: {}; my $fsp = $targets->{'fsp'} ? $targets->{'fsp'}: {}; my $bpa = $targets->{'bpa'} ? $targets->{'bpa'}: {}; if (%$mm) { my $bladeuser = 'USERID'; my $bladepass = 'PASSW0RD'; #if ( $verbose ) { # trace( $request, "telneting to management-modules....." ); #} ############################################# # Check passwd table for userid/password ############################################# my $passtab = xCAT::Table->new('passwd'); if ( $passtab ) { #my ($ent) = $passtab->getAttribs({key=>'blade'},'username','password'); my $ent = $passtab->getNodeAttribs('blade', ['username','password']); if ( defined( $ent )) { $bladeuser = $ent->{username}; $bladepass = $ent->{password}; } } ############################################# # Get userid/password ############################################# my $mpatab = xCAT::Table->new('mpa'); for my $nd ( keys %$mm ) { my $user = $bladeuser; my $pass = $bladepass; if ( defined( $mpatab )) { #my ($ent) = $mpatab->getAttribs({mpa=>$_},'username','password'); my $ent = $mpatab->getNodeAttribs($nd, ['username','password']); if ( defined( $ent->{password} )) { $pass = $ent->{password}; } if ( defined( $ent->{username} )) { $user = $ent->{username}; } } $mm->{$nd}->{username} = $user; $mm->{$nd}->{password} = $pass; } } if (%$hmc ) { ############################################# # Get HMC userid/password ############################################# foreach ( keys %$hmc ) { ( $hmc->{$_}->{username}, $hmc->{$_}->{password}) = xCAT::PPCdb::credentials( $hmc->{$_}->{name}, lc($hmc->{$_}->{'type'}), "hscroot" ); xCAT::MsgUtils->verbose_message( $request, "user/passwd for $_ is $hmc->{$_}->{username} $hmc->{$_}->{password}"); } } if ( %$fsp) { ############################################# # Get FSP userid/password ############################################# foreach ( keys %$fsp ) { ( $fsp->{$_}->{username}, $fsp->{$_}->{password}) = xCAT::PPCdb::credentials( $fsp->{$_}->{name}, lc($fsp->{$_}->{'type'}), "admin"); xCAT::MsgUtils->verbose_message( $request, "user/passwd for $_ is $fsp->{$_}->{username} $fsp->{$_}->{password}"); } } if ( %$bpa) { ############################################# # Get BPA userid/password ############################################# foreach ( keys %$bpa ) { ( $bpa->{$_}->{username}, $bpa->{$_}->{password}) = xCAT::PPCdb::credentials( $bpa->{$_}->{name}, lc($bpa->{$_}->{'type'}), "admin"); xCAT::MsgUtils->verbose_message( $request, "user/passwd for $_ is $bpa->{$_}->{username} $bpa->{$_}->{password}"); } } return (%$mm,%$hmc,%$fsp,%$bpa); } ########################################################################## # Run the forked command and send reply to parent ########################################################################## sub invoke_cmd { my $request = shift; my $ip = shift; my $args = shift; ######################################## # Telnet (rspconfig) command ######################################## my $target_dev = $args->{$ip}; ######################################## # check args ######################################## unless ($target_dev){ send_msg( $request, 1, "invoke_cmd: Can't get the device information about the target $ip" ); return; } my @cmds; my $result; if ($target_dev->{'type'} eq 'mm') { @cmds = ( "snmpcfg=enable", "sshcfg=enable", "network_reset=$target_dev->{args}" ); xCAT::MsgUtils->verbose_message($request, "rspconfig :doresetnet run xCAT_plugin::blade::clicmds for node:$target_dev->{name},ip:$ip."); $result = xCAT_plugin::blade::clicmds( $ip, $target_dev->{username}, $target_dev->{password}, 0, @cmds ); } elsif($target_dev->{'type'} eq 'hmc') { @cmds = ("network_reset=$target_dev->{args}"); xCAT::MsgUtils->verbose_message($request, "sshcmds on hmc $ip"); xCAT::MsgUtils->verbose_message($request, "rspconfig :doresetnet run xCAT::PPC::sshcmds_on_hmc for node:$target_dev->{name},ip:$ip."); $result = xCAT::PPC::sshcmds_on_hmc( $ip, $target_dev->{username}, $target_dev->{password}, @cmds ); } else #The rest must be fsp or bpa { @cmds = ("network=$ip,$target_dev->{args}"); xCAT::MsgUtils->verbose_message($request, "rspconfig :doresetnet run xCAT::PPC::updconf_in_asm for node:$target_dev->{name},ip:$ip."); $result = updconf_in_asm( $ip, $target_dev, @cmds ); } return $result; } ########################################################################## # Invokes the callback with the specified message ########################################################################## sub send_msg { my $request = shift; my $ecode = shift; my %output; ################################################# # Called from child process - send to parent ################################################# if ( exists( $request->{pipe} )) { my $out = $request->{pipe}; $output{errorcode} = $ecode; $output{data} = \@_; print $out freeze( [\%output] ); print $out "\nENDOFFREEZE6sK4ci\n"; } ################################################# # Called from parent - invoke callback directly ################################################# elsif ( exists( $request->{callback} )) { my $callback = $request->{callback}; $output{errorcode} = $ecode; $output{data} = \@_; $callback->( \%output ); } } ########################################################################## # Collect output from the child processes ########################################################################## sub child_response { my $callback = shift; my $fds = shift; my @ready_fds = $fds->can_read(1); foreach my $rfh (@ready_fds) { my $data = <$rfh>; ################################# # Read from child process ################################# if ( defined( $data )) { while ($data !~ /ENDOFFREEZE6sK4ci/) { $data .= <$rfh>; } my $responses = thaw($data); ############################# # rspconfig results ############################# if ( @$responses[0] =~ /^RSPCONFIG6sK4ci$/ ) { shift @$responses; my $ip = @$responses[0]; my @rsp1 = (@$responses[1]); $rsp_result{$ip} = \@rsp1; $ip = @$responses[2]; if ($ip) { my @rsp2 = (@$responses[3]); $rsp_result{$ip} = \@rsp2; } next; } ############################# # Message or verbose trace ############################# foreach ( @$responses ) { $callback->( $_ ); } next; } ################################# # Done - close handle ################################# $fds->remove($rfh); close($rfh); } } ########################################################################## # Logon through remote FSP HTTP-interface ########################################################################## sub connect { my $req = shift; my $server = shift; my $verbose = $req->{verbose}; my $timeout = $req->{fsptimeout}; my $lwp_log; ################################## # Use timeout from site table ################################## if ( !$timeout ) { $timeout = 30; } ################################## # Get userid/password ################################## my $cred = $req->{$server}{cred}; ################################## # Redirect STDERR to variable ################################## if ( $verbose ) { close STDERR; if ( !open( STDERR, '>', \$lwp_log )) { return( "Unable to redirect STDERR: $!" ); } } $IO::Socket::SSL::VERSION = undef; eval { require Net::SSL }; ################################## # Turn on tracing ################################## if ( $verbose ) { LWP::Debug::level( '+' ); } ################################## # Create cookie ################################## my $cookie = HTTP::Cookies->new(); $cookie->set_cookie( 0,'asm_session','0','cgi-bin','','443',0,0,3600,0 ); ################################## # Create UserAgent ################################## my $ua = LWP::UserAgent->new(); ################################## # Set options ################################## my $url = "https://$server/cgi-bin/cgi?form=2"; xCAT::MsgUtils->verbose_message($req, "rspconfig :url for connect is $url \n"); $ua->cookie_jar( $cookie ); $ua->timeout( $timeout ); ################################## # Submit logon ################################## my $res = $ua->post( $url, [ user => @$cred[0], password => @$cred[1], lang => "0", submit => "Log in" ] ); ################################## # Logon failed ################################## if ( !$res->is_success() ) { return( $lwp_log.$res->status_line ); } ################################## # To minimize number of GET/POSTs, # if we successfully logon, we should # get back a valid cookie: # Set-Cookie: asm_session=3038839768778613290 # ################################## if ( $res->as_string =~ /Set-Cookie: asm_session=(\d+)/ ) { ############################## # Successful logon.... # Return: # UserAgent # Server hostname # UserId # Redirected STDERR/STDOUT ############################## return( $ua, $server, @$cred[0], \$lwp_log ); } ############################## # Logon error ############################## $res = $ua->get( $url ); if ( !$res->is_success() ) { return( $lwp_log.$res->status_line ); } ############################## # Check for specific failures ############################## if ( $res->content =~ /(Invalid user ID or password|Too many users)/i ) { return( $lwp_log.$1 . ". Please check node attribute hcp and its password settings."); } return( $lwp_log."Logon failure" ); } ########################################################################## # logon asm and update configuration ########################################################################## sub updconf_in_asm { my $ip = shift; my $target_dev = shift; my @cmds = @_; eval { require xCAT::PPCfsp }; if ( $@ ) { return ([1,@cmds]); } my %handled; for my $cmd (@cmds) { if ( $cmd =~ /(.+?)=(.*)/) { my ($command,$value) = ($1,$2); $handled{$command} = $value; } } my %request = ( ppcretry => 1, verbose => 0, ppcmaxp => 64, ppctimeout => 0, fsptimeout => 0, ppcretry => 3, maxssh => 8, arg => \@cmds, method => \%handled, command => 'rspconfig', hwtype => lc($target_dev->{'type'}), ); my $valid_ip; my @exp; foreach my $individual_ip ( split /,/, $ip ) { ################################ # Get userid and password ################################ my @cred = ($target_dev->{'username'},$target_dev->{'password'}); $request{$individual_ip}{cred} = \@cred; $request{node} = [$individual_ip]; @exp = xCAT::PPCcfg::connect(\%request, $individual_ip); #################################### # Successfully connected #################################### if ( ref($exp[0]) eq "LWP::UserAgent" ) { $valid_ip = $individual_ip; last; } } #################################### # Error connecting #################################### if ( ref($exp[0]) ne "LWP::UserAgent" ) { return ([1,@cmds]); } my $result = xCAT::PPCfsp::handler( $valid_ip, \%request, \@exp ); my $RC = shift( @$result); my @data; push @data, @$result[0]; return ([0, undef, \@data]); } ########################################################################## # PPing nodes befor loggin the node while doing resetnet ########################################################################## #sub pingnodes { # my $ips = shift; # my @ping = split /,/, $ips; # #my @ping = `pping $ips`; # foreach my $res (@ping) { # #if ($res =~ /(\w+)\:\s+ping/i) { # # my $ip = $1; # # return $ip # #} # my $rc = system("ping -q -n -c 1 -w 1 $res > /dev/null"); # if ($rc == 0) { # return $res; # } # } # return undef; #} 1;