2008-09-08 18:54:30 +00:00
#!/usr/bin/env perl
# IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
package xCAT_plugin::updatenode ;
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
BEGIN
{
2009-11-12 20:13:34 +00:00
$ ::XCATROOT = $ ENV { 'XCATROOT' } ? $ ENV { 'XCATROOT' } : '/opt/xcat' ;
2008-09-08 18:54:30 +00:00
}
use lib "$::XCATROOT/lib/perl" ;
use xCAT::Table ;
use xCAT::Schema ;
use Data::Dumper ;
use xCAT::Utils ;
2012-02-19 16:09:21 +00:00
use xCAT::SvrUtils ;
use xCAT::Usage ;
2012-08-29 10:21:39 +00:00
use Storable qw( dclone ) ;
2012-08-09 04:07:40 +00:00
use xCAT::TableUtils ;
use xCAT::ServiceNodeUtils ;
2010-05-10 09:21:56 +00:00
use xCAT::NetworkUtils ;
2009-11-12 20:13:34 +00:00
use xCAT::InstUtils ;
2012-09-27 13:51:15 +00:00
use xCAT::CFMUtils ;
2008-09-08 18:54:30 +00:00
use Getopt::Long ;
use xCAT::GlobalDef ;
use Sys::Hostname ;
2009-11-12 20:13:34 +00:00
use File::Basename ;
2008-09-25 03:04:56 +00:00
use xCAT::GlobalDef ;
use xCAT_monitoring::monitorctrl ;
2009-11-12 20:13:34 +00:00
use Socket ;
2012-08-29 10:21:39 +00:00
use strict ;
2010-07-01 19:55:11 +00:00
my $ CALLBACK ;
my $ RERUNPS4SECURITY ;
2008-09-08 18:54:30 +00:00
1 ;
#-------------------------------------------------------------------------------
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
= head1 xCAT_plugin:updatenode
= head2 Package Description
xCAT plug - in module . It handles the updatenode command .
= cut
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
#------------------------------------------------------------------------------
#--------------------------------------------------------------------------------
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
= head3 handled_commands
It returns a list of commands handled by this plugin .
Arguments:
none
Returns:
a list of commands .
= cut
2009-11-12 20:13:34 +00:00
#------------------------------------------------------------------------------
2008-09-08 18:54:30 +00:00
sub handled_commands
{
2009-11-12 20:13:34 +00:00
return {
2012-08-29 10:21:39 +00:00
updatenode = > "updatenode" ,
updatenodestat = > "updatenode" ,
updatemynodestat = > "updatenode" ,
2010-11-24 07:07:56 +00:00
updatenodeappstat = > "updatenode"
2009-11-12 20:13:34 +00:00
} ;
2008-09-08 18:54:30 +00:00
}
#-------------------------------------------------------
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
= head3 preprocess_request
Check and setup for hierarchy
= cut
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
#-------------------------------------------------------
sub preprocess_request
{
my $ request = shift ;
my $ callback = shift ;
2009-11-12 20:13:34 +00:00
$ ::subreq = shift ;
2010-03-25 18:15:06 +00:00
# needed for runcmd output
$ ::CALLBACK = $ callback ;
2009-11-12 20:13:34 +00:00
my $ command = $ request - > { command } - > [ 0 ] ;
2009-07-15 19:12:07 +00:00
if ( $ request - > { _xcatpreprocessed } - > [ 0 ] == 1 ) { return [ $ request ] ; }
2009-11-12 20:13:34 +00:00
my @ requests = ( ) ;
2008-09-08 18:54:30 +00:00
if ( $ command eq "updatenode" )
{
2009-11-12 20:13:34 +00:00
return & preprocess_updatenode ( $ request , $ callback , $ ::subreq ) ;
}
elsif ( $ command eq "updatenodestat" )
{
return [ $ request ] ;
}
2012-08-24 17:12:57 +00:00
elsif ( $ command eq "updatemynodestat" )
{
return [ $ request ] ;
}
2010-11-24 07:07:56 +00:00
elsif ( $ command eq "updatenodeappstat" )
{
return [ $ request ] ;
}
2009-11-12 20:13:34 +00:00
else
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] = "Unsupported command: $command." ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2008-09-25 03:04:56 +00:00
}
2008-09-08 18:54:30 +00:00
}
2009-11-12 20:13:34 +00:00
#-----------------------------------------------------------------------------
2008-09-08 18:54:30 +00:00
= head3 process_request
2009-11-12 20:13:34 +00:00
It processes the updatenode command .
2008-09-08 18:54:30 +00:00
Arguments:
request - - a hash table which contains the command name and the arguments .
callback - - a callback pointer to return the response to .
Returns:
2009-11-12 20:13:34 +00:00
0 - for success . The output is returned through the callback pointer .
1 - for error . The error messages are returns through the
callback pointer .
2008-09-08 18:54:30 +00:00
= cut
2009-11-12 20:13:34 +00:00
#------------------------------------------------------------------------------
2008-09-08 18:54:30 +00:00
sub process_request
{
my $ request = shift ;
my $ callback = shift ;
2009-11-12 20:13:34 +00:00
$ ::subreq = shift ;
2010-03-25 18:15:06 +00:00
# needed for runcmd output
$ ::CALLBACK = $ callback ;
2009-11-12 20:13:34 +00:00
my $ command = $ request - > { command } - > [ 0 ] ;
my $ localhostname = hostname ( ) ;
if ( $ command eq "updatenode" )
{
return updatenode ( $ request , $ callback , $ ::subreq ) ;
}
elsif ( $ command eq "updatenodestat" )
{
return updatenodestat ( $ request , $ callback ) ;
}
2012-08-24 17:12:57 +00:00
elsif ( $ command eq "updatemynodestat" )
{
2012-08-29 10:21:39 +00:00
delete $ request
- > { node } ; #the restricted form of this command must be forbidden from specifying other nodes, only can set it's own value
2012-08-24 17:12:57 +00:00
return updatenodestat ( $ request , $ callback ) ;
}
2010-11-24 07:07:56 +00:00
elsif ( $ command eq "updatenodeappstat" )
{
return updatenodeappstat ( $ request , $ callback ) ;
}
2009-11-12 20:13:34 +00:00
else
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] = "$localhostname: Unsupported command: $command." ;
$ callback - > ( $ rsp ) ;
return 1 ;
}
return 0 ;
2008-09-08 18:54:30 +00:00
}
2009-11-12 20:13:34 +00:00
#-----------------------------------------------------------------------------
2008-09-08 18:54:30 +00:00
= head3 preprocess_updatenode
This function checks for the syntax of the updatenode command
2009-11-12 20:13:34 +00:00
and distributes the command to the right server .
2008-09-08 18:54:30 +00:00
Arguments:
2009-11-12 20:13:34 +00:00
request - the request .
2008-09-08 18:54:30 +00:00
callback - the pointer to the callback function .
2009-11-12 20:13:34 +00:00
subreq - the sub request
2008-09-08 18:54:30 +00:00
Returns:
A pointer to an array of requests .
= cut
2009-11-12 20:13:34 +00:00
#------------------------------------------------------------------------------
sub preprocess_updatenode
{
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
my $ args = $ request - > { arg } ;
my @ requests = ( ) ;
2012-08-09 04:07:40 +00:00
my $ installdir = xCAT::TableUtils - > getInstallDir ( ) ;
2010-02-24 08:23:36 +00:00
2009-11-12 20:13:34 +00:00
# subroutine to display the usage
sub updatenode_usage
{
2012-08-29 10:21:39 +00:00
my $ cb = shift ;
my $ rsp = { } ;
2010-03-21 13:11:47 +00:00
my $ usage_string = xCAT::Usage - > getUsage ( "updatenode" ) ;
2012-08-29 10:21:39 +00:00
push @ { $ rsp - > { data } } , $ usage_string ;
2010-03-21 13:11:47 +00:00
2009-11-12 20:13:34 +00:00
$ cb - > ( $ rsp ) ;
}
@ ARGV = ( ) ;
if ( $ args )
{
@ ARGV = @ { $ args } ;
}
# parse the options
Getopt::Long:: Configure ( "bundling" ) ;
Getopt::Long:: Configure ( "no_pass_through" ) ;
if (
! GetOptions (
2012-08-29 10:21:39 +00:00
'A|updateallsw' = > \ $ ::ALLSW ,
'c|cmdlineonly' = > \ $ ::CMDLINE ,
'd=s' = > \ $ ::ALTSRC ,
'h|help' = > \ $ ::HELP ,
'v|version' = > \ $ ::VERSION ,
'V|verbose' = > \ $ ::VERBOSE ,
'F|sync' = > \ $ ::FILESYNC ,
'f|snsync' = > \ $ ::SNFILESYNC ,
'S|sw' = > \ $ ::SWMAINTENANCE ,
's|sn' = > \ $ ::SETSERVER ,
'P|scripts:s' = > \ $ ::RERUNPS ,
'k|security' = > \ $ ::SECURITY ,
'o|os:s' = > \ $ ::OS ,
2009-11-12 20:13:34 +00:00
)
)
{
& updatenode_usage ( $ callback ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
# display the usage if -h or --help is specified
if ( $ ::HELP )
{
& updatenode_usage ( $ callback ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
# display the version statement if -v or --verison is specified
if ( $ ::VERSION )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] = xCAT::Utils - > Version ( ) ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
2010-03-21 13:11:47 +00:00
# -c must work with -S for AIX node
2012-08-29 10:21:39 +00:00
if ( $ ::CMDLINE && ! $ ::SWMAINTENANCE )
{
2012-08-22 19:05:10 +00:00
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"If you specify the -c flag you must specify the -S flag" ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2010-03-21 13:11:47 +00:00
}
2012-08-29 10:21:39 +00:00
2010-03-21 13:11:47 +00:00
# -s must work with -P or -S or --security
2012-08-29 10:21:39 +00:00
if ( $ ::SETSERVER && ! ( $ ::SWMAINTENANCE || $ ::RERUNPS || $ ::SECURITY ) )
{
2012-08-22 19:05:10 +00:00
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" If you specify the - s flag you must specify either the - S or - k or - P
flags " ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2010-03-21 13:11:47 +00:00
}
2012-08-29 10:21:39 +00:00
2011-05-13 13:29:05 +00:00
# -f or -F not both
2012-08-29 10:21:39 +00:00
if ( ( $ ::FILESYNC ) && ( $ ::SNFILESYNC ) )
{
my $ rsp = { } ;
2012-08-22 19:05:10 +00:00
$ rsp - > { data } - > [ 0 ] = "You can not specify both the -f and -F flags." ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2011-05-13 13:29:05 +00:00
}
2010-03-21 13:11:47 +00:00
# --security cannot work with -S -P -F
2012-08-29 10:21:39 +00:00
if ( $ ::SECURITY
&& ( $ ::SWMAINTENANCE || $ ::RERUNPS || defined ( $ ::RERUNPS ) ) )
{
2012-08-22 19:05:10 +00:00
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"If you use the -k flag, you cannot specify the -S,-P or -F flags." ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2010-03-21 13:11:47 +00:00
}
2012-09-06 19:23:22 +00:00
# the -P flag is omitted when only postscripts are specified,
2009-11-12 20:13:34 +00:00
# so if there are parameters without any flags, it may mean
2012-09-06 19:23:22 +00:00
# to re-run the postscripts. Except for the -k flag
2009-11-12 20:13:34 +00:00
if ( @ ARGV )
{
# we have one or more operands on the cmd line
2012-08-29 10:21:39 +00:00
if (
$# ARGV == 0
&& ! (
$ ::FILESYNC
|| $ ::SNFILESYNC
|| $ ::SWMAINTENANCE
|| defined ( $ ::RERUNPS )
|| $ ::SECURITY
)
)
2009-11-12 20:13:34 +00:00
{
# there is only one operand
# if it doesn't contain an = sign then it must be postscripts
if ( ! ( $ ARGV [ 0 ] =~ /=/ ) )
{
$ ::RERUNPS = $ ARGV [ 0 ] ;
2010-03-21 13:11:47 +00:00
$ ARGV [ 0 ] = "" ;
2009-11-12 20:13:34 +00:00
}
2010-06-07 12:04:48 +00:00
2009-11-12 20:13:34 +00:00
}
}
else
{
2012-08-29 10:21:39 +00:00
2011-05-13 13:29:05 +00:00
# if not syncing Service Node
2012-08-29 10:21:39 +00:00
if ( ! ( $ ::SNFILESYNC ) )
{
2009-11-12 20:13:34 +00:00
2012-08-29 10:21:39 +00:00
# no flags and no operands, set defaults
if (
! (
$ ::FILESYNC
|| $ ::SWMAINTENANCE
|| defined ( $ ::RERUNPS )
|| $ ::SECURITY
)
)
{
$ ::FILESYNC = 1 ;
$ ::SWMAINTENANCE = 1 ;
$ ::RERUNPS = "" ;
}
}
2010-03-21 13:11:47 +00:00
}
2009-11-12 20:13:34 +00:00
my $ nodes = $ request - > { node } ;
if ( ! $ nodes )
{
2012-08-22 19:05:10 +00:00
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"A noderange is required for the updatenode command." ;
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
2012-08-22 19:05:10 +00:00
if ( $ ::SECURITY )
{
# check to see if the Management Node is in the noderange and
# if it is abort
my $ mname = xCAT::Utils - > noderangecontainsMn ( @$ nodes ) ;
if ( $ mname )
{ # MN in the nodelist
my $ rsp = { } ;
$ rsp - > { error } - > [ 0 ] =
"You must not run -K option against the Management Node:$mname." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback , 1 ) ;
return ;
}
2012-09-06 19:23:22 +00:00
# now build a list of all service nodes that are either in the
# noderange or a service node of a node in the noderange
# and update there ssh keys and credentials
# get computenodes and servicenodes from the noderange
my @ SN ;
my @ CN ;
xCAT::ServiceNodeUtils - > getSNandCPnodes ( \ @$ nodes , \ @ SN , \ @ CN ) ;
$ ::NODEOUT = ( ) ;
& update_SN_security ( $ request , $ callback , $ subreq , \ @ SN ) ;
# are there compute nodes, then we want to change the request to
# just update the compute nodes
if ( scalar ( @ CN ) ) {
$ request - > { node } = \ @ CN ;
$ request - > { noderange } = \ @ CN ;
$ ::RERUNPS = "remoteshell" ;
} else { # no more nodes
return ;
}
2012-08-22 19:05:10 +00:00
}
2012-08-29 10:21:39 +00:00
#
2010-03-25 18:15:06 +00:00
# process @ARGV
2009-11-12 20:13:34 +00:00
#
2010-03-25 18:15:06 +00:00
# the first arg should be a noderange - the other should be attr=val
# - put attr=val operands in %attrvals hash
2008-09-08 18:54:30 +00:00
2009-11-12 20:13:34 +00:00
my % attrvals ;
2012-08-29 10:21:39 +00:00
if ( $ ::SWMAINTENANCE )
{
2010-03-21 13:11:47 +00:00
while ( my $ a = shift ( @ ARGV ) )
2009-11-12 20:13:34 +00:00
{
2010-03-21 13:11:47 +00:00
if ( $ a =~ /=/ )
2009-11-12 20:13:34 +00:00
{
2012-08-29 10:21:39 +00:00
2010-03-21 13:11:47 +00:00
# if it has an "=" sign its an attr=val - we hope
my ( $ attr , $ value ) = $ a =~ /^\s*(\S+?)\s*=\s*(\S*.*)$/ ;
2012-08-29 10:21:39 +00:00
2010-03-21 13:11:47 +00:00
if ( ! defined ( $ attr ) || ! defined ( $ value ) )
{
my $ rsp ;
$ rsp - > { data } - > [ 0 ] = "Incorrect \'attr=val\' pair - $a\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ ::callback ) ;
return 3 ;
}
2012-08-29 10:21:39 +00:00
2010-03-21 13:11:47 +00:00
# put attr=val in hash
$ attrvals { $ attr } = $ value ;
2009-11-12 20:13:34 +00:00
}
}
}
my @ nodes = @$ nodes ;
my $ postscripts ;
2012-08-29 10:21:39 +00:00
# Handle updating operating system
if ( defined ( $ ::OS ) )
{
my $ reqcopy = { %$ request } ;
2010-07-12 00:40:50 +00:00
$ reqcopy - > { os } - > [ 0 ] = "yes" ;
push @ requests , $ reqcopy ;
2012-08-29 10:21:39 +00:00
return \ @ requests ;
2010-07-12 00:40:50 +00:00
}
2012-08-29 10:21:39 +00:00
# handle the validity of postscripts
2009-11-12 20:13:34 +00:00
if ( defined ( $ ::RERUNPS ) )
{
if ( $ ::RERUNPS eq "" )
{
$ postscripts = "" ;
}
else
{
$ postscripts = $ ::RERUNPS ;
2012-08-29 10:21:39 +00:00
my @ posts = split ( ',' , $ postscripts ) ;
2010-03-21 13:11:47 +00:00
2009-11-12 20:13:34 +00:00
foreach ( @ posts )
{
2012-08-29 10:21:39 +00:00
my @ aa = split ( ' ' , $ _ ) ;
2011-06-15 14:09:46 +00:00
if ( ! - e "$installdir/postscripts/$aa[0]" )
2009-11-12 20:13:34 +00:00
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
2011-06-15 14:09:46 +00:00
"The postcript $installdir/postscripts/$aa[0] does not exist." ;
2009-11-12 20:13:34 +00:00
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
}
}
}
2012-08-29 10:21:39 +00:00
# If -F or -f option specified, sync files to the noderange or their
2011-05-13 13:29:05 +00:00
# service nodes.
2010-03-21 13:11:47 +00:00
# Note: This action only happens on MN, since xdcp, xdsh handles the
# hierarchical scenario inside
2009-11-12 20:13:34 +00:00
if ( $ ::FILESYNC )
{
my $ reqcopy = { %$ request } ;
$ reqcopy - > { FileSyncing } - > [ 0 ] = "yes" ;
push @ requests , $ reqcopy ;
}
2012-08-29 10:21:39 +00:00
if ( $ ::SNFILESYNC ) # either sync service node
2011-05-13 13:29:05 +00:00
{
my $ reqcopy = { %$ request } ;
$ reqcopy - > { SNFileSyncing } - > [ 0 ] = "yes" ;
push @ requests , $ reqcopy ;
}
2009-11-12 20:13:34 +00:00
2012-09-25 18:19:37 +00:00
# If -F or -f then, call CFMUtils to check if any PCM CFM data is to be
# built for the node. This will also create the synclists attribute in
# the osimage for each node in the noderange
if ( ( $ ::FILESYNC ) || ( $ ::SNFILESYNC ) ) {
# determine the list of osimages names in the noderange to pass into
# the CFMUtils
my @ imagenames = xCAT::TableUtils - > getimagenames ( \ @ nodes ) ;
# Now here we will call CFMUtils
2012-09-27 13:51:15 +00:00
$ ::CALLBACK = $ callback ;
my $ rc = 0 ;
2012-10-09 11:17:17 +00:00
$ rc = xCAT::CFMUtils - > updateCFMSynclistFile ( \ @ imagenames ) ;
2012-09-27 13:51:15 +00:00
if ( $ rc != 0 ) {
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"The call to CFMUtils to build synclist returned an errorcode=$rc." ;
$ callback - > ( $ rsp ) ;
return ;
}
2012-09-25 18:19:37 +00:00
}
2012-08-29 10:21:39 +00:00
# if not -S or -P or --security
2010-03-21 13:11:47 +00:00
unless ( defined ( $ ::SWMAINTENANCE ) || defined ( $ ::RERUNPS ) || $ ::SECURITY )
2010-01-27 10:23:03 +00:00
{
return \ @ requests ;
}
2009-11-12 20:13:34 +00:00
# - need to consider the mixed cluster case
2012-09-10 07:55:07 +00:00
# - can't depend on the os of the MN - need to split out the AIX nodes
2012-08-29 10:21:39 +00:00
my ( $ rc , $ AIXnodes , $ Linuxnodes ) =
2012-09-10 07:55:07 +00:00
xCAT::InstUtils - > getOSnodes ( $ nodes ) ;
2009-11-12 20:13:34 +00:00
my @ aixnodes = @$ AIXnodes ;
2012-08-29 10:21:39 +00:00
2009-11-12 20:13:34 +00:00
# for AIX nodes we need to copy software to SNs first - if needed
2010-03-25 18:15:06 +00:00
my ( $ imagedef , $ updateinfo ) ;
2009-11-12 20:13:34 +00:00
if ( defined ( $ ::SWMAINTENANCE ) && scalar ( @ aixnodes ) )
{
( $ rc , $ imagedef , $ updateinfo ) =
& doAIXcopy ( $ callback , \ % attrvals , $ AIXnodes , $ subreq ) ;
if ( $ rc != 0 )
{
2012-08-29 10:21:39 +00:00
2009-11-18 06:51:59 +00:00
# Do nothing when doAIXcopy failed
return undef ;
2009-11-12 20:13:34 +00:00
}
}
2012-08-29 10:21:39 +00:00
# when specified -S or -P or --security
# find service nodes for requested nodes
# build an individual request for each service node
2012-08-09 04:07:40 +00:00
my $ sn = xCAT::ServiceNodeUtils - > get_ServiceNode ( \ @ nodes , "xcat" , "MN" ) ;
2009-11-12 20:13:34 +00:00
if ( $ ::ERROR_RC )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not get list of xCAT service nodes." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
2012-08-29 10:21:39 +00:00
return ;
2009-11-12 20:13:34 +00:00
}
2012-09-06 19:23:22 +00:00
# Get the MN names
2012-08-29 10:21:39 +00:00
my @ MNip = xCAT::NetworkUtils - > determinehostname ;
my @ sns = ( ) ;
foreach my $ s ( keys %$ sn )
{
my @ tmp_a = split ( ',' , $ s ) ;
foreach my $ s1 ( @ tmp_a )
{
if ( ! grep ( /^$s1$/ , @ MNip ) )
{
push @ sns , $ s1 ;
}
}
2010-03-21 13:11:47 +00:00
}
2012-08-29 10:21:39 +00:00
2012-09-06 19:23:22 +00:00
# build each request for each node
2009-11-12 20:13:34 +00:00
foreach my $ snkey ( keys %$ sn )
{
2012-08-29 10:21:39 +00:00
my @ tmp_a = split ( ',' , $ snkey ) ;
foreach my $ s1 ( @ tmp_a )
{
2012-09-08 10:11:32 +00:00
# TODO fix the error handling, right now cause inf loop
# when getdata and updatenode_cb are registered LKV
# if ( $::SECURITY
# && !(grep /^$s1$/, @::good_sns) # is it good
# && !(grep /^$s1$/, @MNip)) # is the MN
# {
# my $rsp;
# push @{$rsp->{data}},
# "The security update for service node $snkey encountered error, update security for following nodes will be skipped: @{$sn->{$snkey}}";
# xCAT::MsgUtils->message("E", $rsp, $callback);
# next;
# }
2012-08-29 10:21:39 +00:00
# remove the service node which have been handled before
2012-09-06 19:23:22 +00:00
#if ($::SECURITY && (grep /^$s1$/, @MNip))
#{
# delete @{$sn->{$snkey}}[@sns];
# if (scalar(@{$sn->{$snkey}}) == 0)
# {
# next;
# }
#}
2012-08-29 10:21:39 +00:00
}
2012-09-06 19:23:22 +00:00
# build request
2012-08-29 10:21:39 +00:00
2009-11-12 20:13:34 +00:00
my $ reqcopy = { %$ request } ;
$ reqcopy - > { node } = $ sn - > { $ snkey } ;
$ reqcopy - > { '_xcatdest' } = $ snkey ;
$ reqcopy - > { _xcatpreprocessed } - > [ 0 ] = 1 ;
if ( defined ( $ ::SWMAINTENANCE ) )
{
2012-09-10 07:55:07 +00:00
$ reqcopy - > { swmaintenance } - > [ 0 ] = "yes" ;
2012-08-29 10:21:39 +00:00
2012-09-10 07:55:07 +00:00
# send along the update info and osimage defs
if ( $ imagedef )
2012-08-29 10:21:39 +00:00
{
2012-09-10 07:55:07 +00:00
xCAT::InstUtils - > taghash ( $ imagedef ) ;
$ reqcopy - > { imagedef } = [ $ imagedef ] ;
2009-11-12 20:13:34 +00:00
}
2012-09-10 07:55:07 +00:00
if ( $ updateinfo )
2012-08-29 10:21:39 +00:00
{
2012-09-10 07:55:07 +00:00
xCAT::InstUtils - > taghash ( $ updateinfo ) ;
$ reqcopy - > { updateinfo } = [ $ updateinfo ] ;
2009-11-12 20:13:34 +00:00
}
}
if ( defined ( $ ::RERUNPS ) )
{
$ reqcopy - > { rerunps } - > [ 0 ] = "yes" ;
$ reqcopy - > { postscripts } = [ $ postscripts ] ;
2012-08-29 10:21:39 +00:00
if ( defined ( $ ::SECURITY ) )
{
2010-03-21 13:11:47 +00:00
$ reqcopy - > { rerunps4security } - > [ 0 ] = "yes" ;
}
}
2012-08-29 10:21:39 +00:00
if ( defined ( $ ::SECURITY ) )
{
2010-03-21 13:11:47 +00:00
$ reqcopy - > { security } - > [ 0 ] = "yes" ;
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
2010-07-12 00:40:50 +00:00
#
# Handle updating OS
#
2012-08-29 10:21:39 +00:00
if ( defined ( $ ::OS ) )
{
$ reqcopy - > { os } - > [ 0 ] = "yes" ;
}
2009-11-12 20:13:34 +00:00
push @ requests , $ reqcopy ;
}
return \ @ requests ;
}
2008-09-08 18:54:30 +00:00
2012-08-29 10:21:39 +00:00
#-------------------------------------------------------------------------------
2012-09-06 19:23:22 +00:00
= head3 update_SN_security
process updatenode - k command
determine all the service nodes that must be processed from the
input noderange and then update the ssh keys and credentials
= cut
#-----------------------------------------------------------------------------
sub update_SN_security
{
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
my $ servicenodes = shift ;
my @ SN = @$ servicenodes ;
my $ nodes = $ request - > { node } ;
my @ nodes = @$ nodes ;
my $ sn = xCAT::ServiceNodeUtils - > get_ServiceNode ( \ @ nodes , "xcat" , "MN" ) ;
if ( $ ::ERROR_RC )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not get list of xCAT service nodes." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return ;
}
# take out the Management Node
my @ MNip = xCAT::NetworkUtils - > determinehostname ;
my @ sns = ( ) ;
foreach my $ s ( keys %$ sn )
{
my @ tmp_a = split ( ',' , $ s ) ;
foreach my $ s1 ( @ tmp_a )
{
if ( ! grep ( /^$s1$/ , @ MNip ) )
{
push @ sns , $ s1 ;
}
}
}
# now add any service nodes in the input noderange, we missed
foreach my $ sn ( @ SN ) {
if ( ! grep ( /^$sn$/ , @ sns ) )
{
push @ sns , $ sn ;
}
}
# if we have any service nodes to process
if ( scalar ( @ sns ) )
{
# setup the ssh keys on the service nodes
# run the postscripts: remoteshell, servicenode
# These are all servicenodes
$ ::RERUNPS = "remoteshell,servicenode" ;
my $ req_rs = { %$ request } ;
my $ ps ;
$ ps = $ ::RERUNPS ;
$ req_rs - > { rerunps } - > [ 0 ] = "yes" ;
$ req_rs - > { security } - > [ 0 ] = "yes" ;
$ req_rs - > { rerunps4security } - > [ 0 ] = "yes" ;
$ req_rs - > { node } = \ @ sns ;
$ req_rs - > { noderange } = \ @ sns ;
$ req_rs - > { postscripts } = [ $ ps ] ;
updatenode ( $ req_rs , $ callback , $ subreq ) ;
# parse the output of update security for sns
foreach my $ sn ( keys % { $ ::NODEOUT } )
{
if ( ! grep /^$sn$/ , @ sns )
{
next ;
}
if ( ( grep /ps ok/ , @ { $ ::NODEOUT - > { $ sn } } )
&& ( grep /ssh ok/ , @ { $ ::NODEOUT - > { $ sn } } ) )
{
push @ ::good_sns , $ sn ;
}
}
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Update security for following service nodes: @sns." ;
push @ { $ rsp - > { data } } ,
" Following service nodes have been updated successfully: @::good_sns" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
}
return ;
}
#-------------------------------------------------------------------------------
2012-08-29 10:21:39 +00:00
= head3 security_update_sshkeys
process updatenode - k command
the ssh keys on the service nodes and nodes
by calling xdsh - K
= cut
#-----------------------------------------------------------------------------
sub security_update_sshkeys
{
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
my $ nodes = shift ;
my @ nodes = @$ nodes ;
my $ localhostname = hostname ( ) ;
# remove the host key from known_hosts
xCAT::Utils - > runxcmd (
{
command = > [ 'makeknownhosts' ] ,
node = > \ @$ nodes ,
arg = > [ '-r' ] ,
} ,
$ subreq , 0 , 1
) ;
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" $localhostname: run makeknownhosts to clean known_hosts file for nodes: @$nodes" ;
$ callback - > ( $ rsp ) ;
}
# call the xdsh -K to set up the ssh keys
my @ envs = @ { $ request - > { environment } } ;
my @ args = ( "-K" ) ;
my $ res =
xCAT::Utils - > runxcmd (
{
command = > [ 'xdsh' ] ,
node = > \ @$ nodes ,
arg = > \ @ args ,
env = > \ @ envs ,
} ,
$ subreq , 0 , 1
) ;
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: xdsh -K. nodes = @$nodes, arguments = @args, env = @envs" ;
$ rsp - > { data } - > [ 1 ] =
" $localhostname: return messages of last command: @$res" ;
$ callback - > ( $ rsp ) ;
}
# parse the output of xdsh -K
my @ failednodes = @$ nodes ;
foreach my $ line ( @$ res )
{
chomp ( $ line ) ;
if ( $ line =~ /SSH setup failed for the following nodes: (.*)\./ )
{
@ failednodes = split ( /,/ , $ 1 ) ;
}
elsif ( $ line =~ /setup is complete/ )
{
@ failednodes = ( ) ;
}
}
my $ rsp = { } ;
foreach my $ node ( @$ nodes )
{
if ( grep /^$node$/ , @ failednodes )
{
push @ { $ rsp - > { data } } , "$node: Setup ssh keys failed." ;
}
else
{
push @ { $ rsp - > { data } } , "$node: Setup ssh keys has completed." ;
}
}
$ callback - > ( $ rsp ) ;
return ;
}
2010-03-21 13:11:47 +00:00
#--------------------------------------------------------------------------------
= head3 updatenode_cb
A callback function which is used to handle the output of updatenode function
when run updatenode - - secruity for service node inside
= cut
#-----------------------------------------------------------------------------
sub updatenode_cb
{
my $ resp = shift ;
# call the original callback function
$ ::CALLBACK - > ( $ resp ) ;
2012-08-29 10:21:39 +00:00
foreach my $ line ( @ { $ resp - > { data } } )
{
2010-03-21 13:11:47 +00:00
my $ node ;
my $ msg ;
2012-08-29 10:21:39 +00:00
if ( $ line =~ /(.*):(.*)/ )
{
2010-03-21 13:11:47 +00:00
$ node = $ 1 ;
2012-08-29 10:21:39 +00:00
$ msg = $ 2 ;
2010-03-21 13:11:47 +00:00
}
2012-09-06 19:23:22 +00:00
if ( $ msg =~ /Redeliver security files has completed/ )
2012-08-29 10:21:39 +00:00
{
2010-03-21 13:11:47 +00:00
push @ { $ ::NODEOUT - > { $ node } } , "ps ok" ;
2012-08-29 10:21:39 +00:00
}
elsif ( $ msg =~ /Setup ssh keys has completed/ )
{
2010-03-21 13:11:47 +00:00
push @ { $ ::NODEOUT - > { $ node } } , "ssh ok" ;
}
}
}
2008-09-08 18:54:30 +00:00
#--------------------------------------------------------------------------------
2009-11-12 20:13:34 +00:00
2008-09-08 18:54:30 +00:00
= head3 updatenode
This function implements the updatenode command .
Arguments:
request - the request .
callback - the pointer to the callback function .
2009-11-12 20:13:34 +00:00
subreq - the sub request
2008-09-08 18:54:30 +00:00
Returns:
2009-11-12 20:13:34 +00:00
0 - for success . The output is returned through the callback pointer .
1 - for error . The error messages are returned through the
callback pointer .
2008-09-08 18:54:30 +00:00
= cut
2009-11-12 20:13:34 +00:00
#-----------------------------------------------------------------------------
sub updatenode
{
my $ request = shift ;
my $ callback = shift ;
my $ subreq = shift ;
#print Dumper($request);
my $ nodes = $ request - > { node } ;
my $ localhostname = hostname ( ) ;
# in a mixed cluster we could potentially have both AIX and Linux
# nodes provided on the command line ????
my ( $ rc , $ AIXnodes , $ Linuxnodes ) = xCAT::InstUtils - > getOSnodes ( $ nodes ) ;
my $ args = $ request - > { arg } ;
@ ARGV = ( ) ;
if ( $ args )
{
@ ARGV = @ { $ args } ;
2009-08-24 09:53:55 +00:00
}
2010-02-24 08:23:36 +00:00
# Lookup Install dir location at this Mangment Node.
# XXX: Suppose that compute nodes has the same Install dir location.
2012-08-09 04:07:40 +00:00
my $ installdir = xCAT::TableUtils - > getInstallDir ( ) ;
2010-02-24 08:23:36 +00:00
2012-08-29 10:21:39 +00:00
#if the postscripts directory exists then make sure it is
# world readable and executable by root
2012-06-04 18:41:45 +00:00
my $ postscripts = "$installdir/postscripts" ;
2012-08-29 10:21:39 +00:00
if ( - e $ postscripts )
{
my $ cmd = "chmod -R u+x,a+r $postscripts" ;
xCAT::Utils - > runcmd ( $ cmd , 0 ) ;
my $ rsp = { } ;
if ( $ ::RUNCMD_RC != 0 )
{
$ rsp - > { data } - > [ 0 ] = "$cmd failed.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
2012-08-28 11:39:41 +00:00
2012-08-29 10:21:39 +00:00
}
2012-09-04 02:37:01 +00:00
#$cmd="cd $postscripts;tar -czf $postscripts.tgz ."; #print "cmd:$cmd\n";
#xCAT::Utils->runcmd($cmd, 0);
#$rsp = {};
#if ($::RUNCMD_RC != 0)
#{
# $rsp->{data}->[0] = "$cmd failed.\n";
# xCAT::MsgUtils->message("E", $rsp, $callback);
#}
2012-08-28 11:39:41 +00:00
2012-06-04 18:41:45 +00:00
}
2009-11-12 20:13:34 +00:00
# convert the hashes back to the way they were passed in
my $ flatreq = xCAT::InstUtils - > restore_request ( $ request , $ callback ) ;
my $ imgdefs ;
my $ updates ;
if ( $ flatreq - > { imagedef } )
{
$ imgdefs = $ flatreq - > { imagedef } ;
}
if ( $ flatreq - > { updateinfo } )
{
$ updates = $ flatreq - > { updateinfo } ;
}
# get the NIM primary server name
my $ nimprime = xCAT::InstUtils - > getnimprime ( ) ;
chomp $ nimprime ;
2012-08-29 10:21:39 +00:00
# parse the options
2009-11-12 20:13:34 +00:00
Getopt::Long:: Configure ( "bundling" ) ;
Getopt::Long:: Configure ( "no_pass_through" ) ;
if (
! GetOptions (
2012-08-29 10:21:39 +00:00
'A|updateallsw' = > \ $ ::ALLSW ,
'c|cmdlineonly' = > \ $ ::CMDLINE ,
'd=s' = > \ $ ::ALTSRC ,
'h|help' = > \ $ ::HELP ,
'v|version' = > \ $ ::VERSION ,
'V|verbose' = > \ $ ::VERBOSE ,
'F|sync' = > \ $ ::FILESYNC ,
'f|snsync' = > \ $ ::SNFILESYNC ,
'S|sw' = > \ $ ::SWMAINTENANCE ,
's|sn' = > \ $ ::SETSERVER ,
'P|scripts:s' = > \ $ ::RERUNPS ,
'k|security' = > \ $ ::SECURITY ,
'o|os:s' = > \ $ ::OS ,
2009-11-12 20:13:34 +00:00
)
)
{
}
#
# process @ARGV
#
2012-08-29 10:21:39 +00:00
# - put attr=val operands in %::attrres hash
2009-11-12 20:13:34 +00:00
while ( my $ a = shift ( @ ARGV ) )
{
if ( $ a =~ /=/ )
{
# if it has an "=" sign its an attr=val - we hope
my ( $ attr , $ value ) = $ a =~ /^\s*(\S+?)\s*=\s*(\S*.*)$/ ;
if ( ! defined ( $ attr ) || ! defined ( $ value ) )
{
my $ rsp ;
$ rsp - > { data } - > [ 0 ] = "Incorrect \'attr=val\' pair - $a\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ ::callback ) ;
return 3 ;
}
2012-08-29 10:21:39 +00:00
# put attr=val in hash
2009-11-12 20:13:34 +00:00
$ ::attrres { $ attr } = $ value ;
}
}
#
# handle file synchronization
#
2011-05-13 13:29:05 +00:00
if ( ( $ request - > { FileSyncing } && $ request - > { FileSyncing } - > [ 0 ] eq "yes" )
2012-08-29 10:21:39 +00:00
|| (
( $ request - > { SNFileSyncing }
&& $ request - > { SNFileSyncing } - > [ 0 ] eq "yes" ) ) )
2009-11-12 20:13:34 +00:00
{
my % syncfile_node = ( ) ;
my % syncfile_rootimage = ( ) ;
my $ node_syncfile = xCAT::SvrUtils - > getsynclistfile ( $ nodes ) ;
foreach my $ node ( @$ nodes )
{
my $ synclist = $$ node_syncfile { $ node } ;
if ( $ synclist )
{
2012-09-20 18:58:47 +00:00
# this can be a comma separated list of multiple
# syncfiles
my @ sl = split ( ',' , $ synclist ) ;
foreach my $ s ( @ sl ) {
push @ { $ syncfile_node { $ s } } , $ node ;
}
2009-11-12 20:13:34 +00:00
}
}
2012-08-29 10:21:39 +00:00
2009-11-12 20:13:34 +00:00
# Check the existence of the synclist file
2012-08-29 10:21:39 +00:00
if ( % syncfile_node )
{ # there are files to sync defined
foreach my $ synclist ( keys % syncfile_node )
{
if ( ! ( - r $ synclist ) )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"The Synclist file $synclist which specified for certain node does NOT existed." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
# Sync files to the target nodes
foreach my $ synclist ( keys % syncfile_node )
{
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
if ( $ request - > { FileSyncing } - > [ 0 ] eq "yes" )
{ # sync nodes
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: xdcp -F $synclist" ;
}
else
{ # sync SN
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: xdcp -s -F $synclist" ;
}
$ callback - > ( $ rsp ) ;
}
my $ args ;
my $ env ;
if ( $ request - > { FileSyncing } - > [ 0 ] eq "yes" )
{ # sync nodes
$ args = [ "-F" , "$synclist" ] ;
$ env = [ "DSH_RSYNC_FILE=$synclist" ] ;
}
else
{ # sync SN only
$ args = [ "-s" , "-F" , "$synclist" ] ;
$ env = [ "DSH_RSYNC_FILE=$synclist" , "RSYNCSNONLY=1" ] ;
}
$ subreq - > (
{
command = > [ 'xdcp' ] ,
node = > $ syncfile_node { $ synclist } ,
arg = > $ args ,
env = > $ env
} ,
$ callback
) ;
}
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] = "File synchronization has completed." ;
$ callback - > ( $ rsp ) ;
}
else
{ # no syncfiles defined
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"There were no syncfiles defined to process. File synchronization has completed." ;
$ callback - > ( $ rsp ) ;
2009-11-12 20:13:34 +00:00
}
}
2009-11-23 10:59:29 +00:00
if ( scalar ( @$ AIXnodes ) )
{
if ( xCAT::Utils - > isLinux ( ) )
{
2012-08-29 10:21:39 +00:00
2009-11-23 10:59:29 +00:00
# mixed cluster enviornment, Linux MN=>AIX node
# linux nfs client can not mount AIX nfs directory with default settings.
# settting nfs_use_reserved_ports=1 could solve the problem
2012-08-29 10:21:39 +00:00
my $ cmd = qq~nfso -o nfs_use_reserved_ports=1~ ;
2009-11-23 10:59:29 +00:00
my $ output =
2012-08-29 10:21:39 +00:00
xCAT::InstUtils - > xcmd ( $ callback , $ subreq , "xdsh" , $ AIXnodes , $ cmd ,
0 ) ;
2009-11-23 10:59:29 +00:00
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not set nfs_use_reserved_ports=1 on nodes. Error message is:\n" ;
push @ { $ rsp - > { data } } , "$output\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
}
}
2009-11-12 20:13:34 +00:00
#
# handle software updates
#
if ( $ request - > { swmaintenance } && $ request - > { swmaintenance } - > [ 0 ] eq "yes" )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Performing software maintenance operations. This could take a while.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
2012-08-29 10:21:39 +00:00
my ( $ rc , $ AIXnodes_nd , $ Linuxnodes_nd ) =
2012-09-10 07:55:07 +00:00
xCAT::InstUtils - > getOSnodes ( $ nodes ) ;
2010-01-27 10:23:03 +00:00
2012-08-29 10:21:39 +00:00
#
2010-03-25 18:15:06 +00:00
# do linux nodes
#
2010-01-27 10:23:03 +00:00
if ( scalar ( @$ Linuxnodes_nd ) )
2009-11-12 20:13:34 +00:00
{ # we have a list of linux nodes
my $ cmd ;
2012-08-29 10:21:39 +00:00
# get server names as known by the nodes
my % servernodes =
% { xCAT::InstUtils - > get_server_nodes ( $ callback , \ @$ Linuxnodes_nd ) } ;
# it's possible that the nodes could have diff server names
# do all the nodes for a particular server at once
foreach my $ snkey ( keys % servernodes )
{
my $ nodestring = join ( ',' , @ { $ servernodes { $ snkey } } ) ;
my $ cmd ;
if ( $ ::SETSERVER )
{
$ cmd =
"XCATBYPASS=Y $::XCATROOT/bin/xdsh $nodestring -s -v -e $installdir/postscripts/xcatdsklspost 2 -M $snkey ospkgs,otherpkgs 2>&1" ;
}
else
{
$ cmd =
"XCATBYPASS=Y $::XCATROOT/bin/xdsh $nodestring -s -v -e $installdir/postscripts/xcatdsklspost 2 -m $snkey ospkgs,otherpkgs 2>&1" ;
}
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: $cmd" ;
$ callback - > ( $ rsp ) ;
}
if ( $ cmd && ! open ( CMD , "$cmd |" ) )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"$localhostname: Cannot run command $cmd" ;
$ callback - > ( $ rsp ) ;
}
else
{
while ( <CMD> )
{
my $ rsp = { } ;
my $ output = $ _ ;
chomp ( $ output ) ;
$ output =~ s/\\cM// ;
if ( $ output =~ /returned from postscript/ )
{
$ output =~
s/returned from postscript/Running of Software Maintenance has completed./ ;
}
$ rsp - > { data } - > [ 0 ] = "$output" ;
$ callback - > ( $ rsp ) ;
}
close ( CMD ) ;
}
}
}
#
2010-03-25 18:15:06 +00:00
# do AIX nodes
#
2009-11-12 20:13:34 +00:00
2010-03-25 18:15:06 +00:00
if ( scalar ( @$ AIXnodes_nd ) )
2012-08-29 10:21:39 +00:00
{
2009-11-12 20:13:34 +00:00
# update the software on an AIX node
2012-08-29 10:21:39 +00:00
if (
& updateAIXsoftware (
$ callback , \ % ::attrres , $ imgdefs ,
$ updates , $ AIXnodes_nd , $ subreq
) != 0
)
{
2009-11-12 20:13:34 +00:00
# my $rsp;
# push @{$rsp->{data}}, "Could not update software for AIX nodes \'@$AIXnodes\'.";
# xCAT::MsgUtils->message("E", $rsp, $callback);;
return 1 ;
}
}
} # end sw maint section
2010-03-21 13:11:47 +00:00
#
# handle of setting up ssh keys
#
2012-08-29 10:21:39 +00:00
if ( $ request - > { security } && $ request - > { security } - > [ 0 ] eq "yes" )
{
# check to see if the Management Node is in the noderange and
# if it is abort
my $ mname = xCAT::Utils - > noderangecontainsMn ( @$ nodes ) ;
if ( $ mname )
{ # MN in the nodelist
2012-06-26 15:15:19 +00:00
my $ rsp = { } ;
$ rsp - > { error } - > [ 0 ] =
"You must not run -K option against the Management Node:$mname." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ ::CALLBACK , 1 ) ;
return ;
}
2012-08-30 12:41:20 +00:00
# setup the root ssh keys ( runs xdsh -k)
2012-08-29 10:21:39 +00:00
2012-08-30 12:41:20 +00:00
& security_update_sshkeys
( $ request , $ callback , $ subreq , \ @$ nodes ) ;
2010-03-21 13:11:47 +00:00
}
2009-11-12 20:13:34 +00:00
#
# handle the running of cust scripts
#
if ( $ request - > { rerunps } && $ request - > { rerunps } - > [ 0 ] eq "yes" )
{
2012-08-29 10:21:39 +00:00
my $ postscripts = "" ;
2010-03-21 13:11:47 +00:00
my $ orig_postscripts = "" ;
2009-11-12 20:13:34 +00:00
if ( ( $ request - > { postscripts } ) && ( $ request - > { postscripts } - > [ 0 ] ) )
{
2010-03-21 13:11:47 +00:00
$ orig_postscripts = $ request - > { postscripts } - > [ 0 ] ;
2009-11-12 20:13:34 +00:00
}
if ( scalar ( @$ Linuxnodes ) )
2012-08-29 10:21:39 +00:00
{
my $ DBname = xCAT::Utils - > get_DBName ;
2012-09-08 10:14:07 +00:00
$ postscripts = $ orig_postscripts ;
2012-08-29 10:21:39 +00:00
# we have Linux nodes
my $ cmd ;
# get server names as known by the nodes
my % servernodes =
% { xCAT::InstUtils - > get_server_nodes ( $ callback , \ @$ Linuxnodes ) } ;
# it's possible that the nodes could have diff server names
# do all the nodes for a particular server at once
foreach my $ snkey ( keys % servernodes )
{
my $ nodestring = join ( ',' , @ { $ servernodes { $ snkey } } ) ;
my $ args ;
2010-03-21 13:11:47 +00:00
my $ mode ;
2012-08-29 10:21:39 +00:00
if ( $ request - > { rerunps4security }
&& $ request - > { rerunps4security } - > [ 0 ] eq "yes" )
{
2010-03-21 13:11:47 +00:00
# for updatenode --security
$ mode = "5" ;
2012-08-29 10:21:39 +00:00
}
else
{
2010-03-21 13:11:47 +00:00
# for updatenode -P
2012-08-29 10:21:39 +00:00
$ mode = "1" ;
2010-03-21 13:11:47 +00:00
}
2011-05-13 13:29:05 +00:00
my $ args1 ;
2012-08-29 10:21:39 +00:00
if ( $ ::SETSERVER )
{
$ args1 = [
"-s" ,
"-v" ,
"-e" ,
"$installdir/postscripts/xcatdsklspost $mode -M $snkey '$postscripts'"
] ;
}
else
{
$ args1 = [
"-s" ,
"-v" ,
"-e" ,
"$installdir/postscripts/xcatdsklspost $mode -m $snkey '$postscripts'"
] ;
}
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: xdsh $nodestring "
. join ( ' ' , @$ args1 ) ;
$ callback - > ( $ rsp ) ;
}
$ CALLBACK = $ callback ;
if ( $ request - > { rerunps4security } )
{
$ RERUNPS4SECURITY = $ request - > { rerunps4security } - > [ 0 ] ;
}
else
{
$ RERUNPS4SECURITY = "" ;
}
$ subreq - > (
{
command = > [ "xdsh" ] ,
node = > $ servernodes { $ snkey } ,
arg = > $ args1 ,
_xcatpreprocessed = > [ 1 ]
} ,
\ & getdata
) ;
2010-07-01 19:55:11 +00:00
2012-08-29 10:21:39 +00:00
}
}
2009-11-12 20:13:34 +00:00
if ( scalar ( @$ AIXnodes ) )
2010-03-21 13:11:47 +00:00
{
2012-08-29 10:21:39 +00:00
# we have AIX nodes
2012-09-08 10:14:07 +00:00
$ postscripts = $ orig_postscripts ;
2012-08-29 10:21:39 +00:00
# need to pass the name of the server on the xcataixpost cmd line
# get server names as known by the nodes
my % servernodes =
% { xCAT::InstUtils - > get_server_nodes ( $ callback , \ @$ AIXnodes ) } ;
# it's possible that the nodes could have diff server names
# do all the nodes for a particular server at once
foreach my $ snkey ( keys % servernodes )
{
my $ nodestring = join ( ',' , @ { $ servernodes { $ snkey } } ) ;
my $ cmd ;
2010-03-21 13:11:47 +00:00
my $ mode ;
2012-08-29 10:21:39 +00:00
if ( $ request - > { rerunps4security }
&& $ request - > { rerunps4security } - > [ 0 ] eq "yes" )
{
2010-03-21 13:11:47 +00:00
# for updatenode --security
$ mode = "5" ;
2012-08-29 10:21:39 +00:00
}
else
{
2010-03-21 13:11:47 +00:00
# for updatenode -P
$ mode = "1" ;
}
2012-08-29 10:21:39 +00:00
if ( $ ::SETSERVER )
{
$ cmd =
"XCATBYPASS=Y $::XCATROOT/bin/xdsh $nodestring -s -v -e $installdir/postscripts/xcataixpost -M $snkey -c $mode '$postscripts' 2>&1" ;
}
else
{
$ cmd =
"XCATBYPASS=Y $::XCATROOT/bin/xdsh $nodestring -s -v -e $installdir/postscripts/xcataixpost -m $snkey -c $mode '$postscripts' 2>&1" ;
}
if ( $ ::VERBOSE )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
" $localhostname: Internal call command: $cmd" ;
$ callback - > ( $ rsp ) ;
}
if ( ! open ( CMD , "$cmd |" ) )
{
my $ rsp = { } ;
$ rsp - > { data } - > [ 0 ] =
"$localhostname: Cannot run command $cmd" ;
$ callback - > ( $ rsp ) ;
}
else
{
2010-03-21 13:11:47 +00:00
my $ rsp = { } ;
2012-08-29 10:21:39 +00:00
while ( <CMD> )
{
my $ output = $ _ ;
chomp ( $ output ) ;
$ output =~ s/\\cM// ;
if ( $ output =~ /returned from postscript/ )
{
$ output =~
s/returned from postscript/Running of postscripts has completed./ ;
}
if ( $ request - > { rerunps4security }
&& $ request - > { rerunps4security } - > [ 0 ] eq "yes" )
{
if ( $ output =~
/Running of postscripts has completed/ )
{
$ output =~
2012-09-06 19:23:22 +00:00
s/Running of postscripts has completed/Redeliver security files has completed/ ;
2010-03-21 13:11:47 +00:00
push @ { $ rsp - > { data } } , $ output ;
2012-08-29 10:21:39 +00:00
}
elsif ( $ output !~
/Running postscript|Error loading module/ )
{
2010-03-21 13:11:47 +00:00
push @ { $ rsp - > { data } } , $ output ;
}
}
2012-08-29 10:21:39 +00:00
elsif ( $ output !~ /Error loading module/ )
{
push @ { $ rsp - > { data } } , "$output" ;
}
}
close ( CMD ) ;
2010-03-21 13:11:47 +00:00
$ callback - > ( $ rsp ) ;
2012-08-29 10:21:39 +00:00
}
}
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
if ( $ request - > { rerunps4security }
&& $ request - > { rerunps4security } - > [ 0 ] eq "yes" )
{
2010-03-21 13:11:47 +00:00
# clean the know_hosts
2012-08-29 10:21:39 +00:00
xCAT::Utils - > runxcmd (
{
command = > [ 'makeknownhosts' ] ,
node = > \ @$ nodes ,
arg = > [ '-r' ] ,
} ,
$ subreq , 0 , 1
) ;
}
}
#
# Handle updating OS
#
if ( $ request - > { os } && $ request - > { os } - > [ 0 ] eq "yes" )
{
my $ os = $ ::OS ;
# Process ID for xfork()
my $ pid ;
# Child process IDs
my @ children ;
# Go through each node
foreach my $ node ( @$ nodes )
{
$ pid = xCAT::Utils - > xfork ( ) ;
# Parent process
if ( $ pid )
{
push ( @ children , $ pid ) ;
}
# Child process
elsif ( $ pid == 0 )
{
# Update OS
updateOS ( $ callback , $ node , $ os ) ;
# Exit process
exit ( 0 ) ;
}
else
{
2010-07-12 00:40:50 +00:00
2012-08-29 10:21:39 +00:00
# Ran out of resources
die "Error: Could not fork\n" ;
}
} # End of foreach
# Wait for all processes to end
foreach ( @ children )
{
waitpid ( $ _ , 0 ) ;
}
}
# finish clean up the tarred postscript file
2012-09-04 02:37:01 +00:00
#if (-e "$postscripts.tgz") {
# my $cmd="rm $postscripts.tgz"; #print "cmd:$cmd\n";
# xCAT::Utils->runcmd($cmd, 0);
# my $rsp = {};
# if ($::RUNCMD_RC != 0)
# {
# $rsp->{data}->[0] = "$cmd failed.\n";
# xCAT::MsgUtils->message("E", $rsp, $callback);
# }
#}
2009-11-12 20:13:34 +00:00
return 0 ;
}
2012-08-29 10:21:39 +00:00
#
# Handles the return from running xcatdsklspost
#
sub getdata
{
no strict ;
my $ response = shift ;
my $ rsp ;
foreach my $ type ( keys %$ response )
{
foreach my $ output ( @ { $ response - > { $ type } } )
{
chomp ( $ output ) ;
$ output =~ s/\\cM// ;
if ( $ output =~ /returned from postscript/ )
{
$ output =~
s/returned from postscript/Running of postscripts has completed./ ;
}
if ( $ RERUNPS4SECURITY && $ RERUNPS4SECURITY eq "yes" )
{
if ( $ output =~ /Running of postscripts has completed/ )
{
$ output =~
2012-09-06 19:23:22 +00:00
s/Running of postscripts has completed/Redeliver security files has completed/ ;
2012-08-29 10:21:39 +00:00
push @ { $ rsp - > { $ type } } , $ output ;
}
elsif ( $ output !~ /Running postscript|Error loading module/ )
{
push @ { $ rsp - > { $ type } } , "$output" ;
}
}
elsif ( $ output !~ /Error loading module/ )
{
push @ { $ rsp - > { $ type } } , "$output" ;
}
}
}
$ CALLBACK - > ( $ rsp ) ;
2010-07-01 19:55:11 +00:00
}
2010-03-25 18:15:06 +00:00
#-------------------------------------------------------------------------------
= head3 updatenodestat
Arguments:
Returns:
0 - for success .
1 - for error .
= cut
#-----------------------------------------------------------------------------
2009-11-12 20:13:34 +00:00
sub updatenodestat
{
my $ request = shift ;
my $ callback = shift ;
my @ nodes = ( ) ;
my @ args = ( ) ;
if ( ref ( $ request - > { node } ) )
{
@ nodes = @ { $ request - > { node } } ;
}
else
{
2012-08-29 10:21:39 +00:00
if ( $ request - > { node } ) { @ nodes = ( $ request - > { node } ) ; }
else
{ #client asking to update its own status...
unless ( ref $ request - > { username } )
{
return ;
} #TODO: log an attempt without credentials?
@ nodes = @ { $ request - > { username } } ;
2012-08-24 17:12:57 +00:00
}
2009-11-12 20:13:34 +00:00
}
if ( ref ( $ request - > { arg } ) )
{
@ args = @ { $ request - > { arg } } ;
}
else
{
@ args = ( $ request - > { arg } ) ;
}
if ( ( @ nodes > 0 ) && ( @ args > 0 ) )
{
my % node_status = ( ) ;
my $ stat = $ args [ 0 ] ;
2012-08-29 10:21:39 +00:00
unless ( $ ::VALID_STATUS_VALUES { $ stat } )
{
return ;
} #don't accept just any string, see GlobalDef for updates
2009-11-12 20:13:34 +00:00
$ node_status { $ stat } = [] ;
foreach my $ node ( @ nodes )
{
my $ pa = $ node_status { $ stat } ;
push ( @$ pa , $ node ) ;
}
xCAT_monitoring::monitorctrl:: setNodeStatusAttributes ( \ % node_status , 1 ) ;
}
return 0 ;
2008-09-08 18:54:30 +00:00
}
2009-11-12 20:13:34 +00:00
#-------------------------------------------------------------------------------
= head3 doAIXcopy
Copy software update files to SNs - if needed .
Arguments:
Returns:
errors:
0 - OK
1 - error
hash refs:
- osimage definitions
- node update information
Example
my ( $ rc , $ imagedef , $ updateinfo ) = & doAIXcopy ( $ callback , \ % attrvals ,
$ nodes , $ subreq ) ;
Comments:
- running on MN
= cut
#------------------------------------------------------------------------------
sub doAIXcopy
{
my $ callback = shift ;
my $ av = shift ;
my $ nodes = shift ;
my $ subreq = shift ;
my @ nodelist ; # node list
my % attrvals ; # cmd line attr=val pairs
if ( $ nodes )
{
@ nodelist = @$ nodes ;
}
if ( $ av )
{
% attrvals = % { $ av } ;
}
# get the NIM primary server name
my $ nimprime = xCAT::InstUtils - > getnimprime ( ) ;
chomp $ nimprime ;
my % nodeupdateinfo ;
#
# do we have to copy files to any SNs????
#
# get a list of service nodes for this node list
2012-08-09 04:07:40 +00:00
my $ sn = xCAT::ServiceNodeUtils - > get_ServiceNode ( \ @ nodelist , "xcat" , "MN" ) ;
2009-11-12 20:13:34 +00:00
if ( $ ::ERROR_RC )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not get list of xCAT service nodes." ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
# want list of remote service nodes - to copy files to
2010-03-25 18:15:06 +00:00
# get the ip of the NIM primary (normally the management node)
2010-05-10 09:21:56 +00:00
my $ ip = xCAT::NetworkUtils - > getipaddr ( $ nimprime ) ;
2009-11-12 20:13:34 +00:00
chomp $ ip ;
my ( $ p1 , $ p2 , $ p3 , $ p4 ) = split /\./ , $ ip ;
my @ SNlist ;
foreach my $ snkey ( keys %$ sn )
{
2010-05-10 09:21:56 +00:00
my $ sip = xCAT::NetworkUtils - > getipaddr ( $ snkey ) ;
2009-11-12 20:13:34 +00:00
chomp $ ip ;
2010-05-10 09:21:56 +00:00
if ( $ ip eq $ sip )
2009-11-12 20:13:34 +00:00
{
next ;
}
else
{
if ( ! grep ( /^$snkey$/ , @ SNlist ) )
{
push ( @ SNlist , $ snkey ) ;
}
}
}
# get a list of osimage names needed for the nodes
my $ nodetab = xCAT::Table - > new ( 'nodetype' ) ;
my $ images =
$ nodetab - > getNodesAttribs ( \ @ nodelist , [ 'node' , 'provmethod' , 'profile' ] ) ;
my @ imagenames ;
2010-12-20 09:16:29 +00:00
my @ noimage ;
2009-11-12 20:13:34 +00:00
foreach my $ node ( @ nodelist )
{
my $ imgname ;
if ( $ images - > { $ node } - > [ 0 ] - > { provmethod } )
{
$ imgname = $ images - > { $ node } - > [ 0 ] - > { provmethod } ;
}
elsif ( $ images - > { $ node } - > [ 0 ] - > { profile } )
{
$ imgname = $ images - > { $ node } - > [ 0 ] - > { profile } ;
}
2010-12-20 09:16:29 +00:00
2012-08-29 10:21:39 +00:00
if ( ! $ imgname )
{
2010-12-20 09:16:29 +00:00
push @ noimage , $ node ;
2012-08-29 10:21:39 +00:00
}
elsif ( ! grep ( /^$imgname$/ , @ imagenames ) )
2009-11-12 20:13:34 +00:00
{
push @ imagenames , $ imgname ;
}
$ nodeupdateinfo { $ node } { imagename } = $ imgname ;
}
$ nodetab - > close ;
2012-08-29 10:21:39 +00:00
if ( @ noimage )
{
2010-12-20 09:16:29 +00:00
my $ rsp ;
my $ allnodes = join ( ',' , @ noimage ) ;
2012-08-29 10:21:39 +00:00
push @ { $ rsp - > { data } } ,
"No osimage specified for the following nodes: $allnodes. You can try to run the nimnodeset command or set the profile|provmethod attributes manually." ;
2010-12-20 09:16:29 +00:00
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
2009-11-12 20:13:34 +00:00
my $ osimageonly = 0 ;
if ( ( ! $ attrvals { installp_bundle } && ! $ attrvals { otherpkgs } ) && ! $ ::CMDLINE )
{
# if nothing is provided on the cmd line and we don't set CMDLINE
2010-03-25 18:15:06 +00:00
# then we just use the osimage def - used for permanent updates
2009-11-12 20:13:34 +00:00
$ osimageonly = 1 ;
}
#
# get the osimage defs
#
my % imagedef ;
my @ pkglist ; # list of all software to go to SNs
my % bndloc ;
2011-05-13 13:29:05 +00:00
foreach my $ img ( @ imagenames )
2009-11-12 20:13:34 +00:00
{
my % objtype ;
$ objtype { $ img } = 'osimage' ;
% imagedef = xCAT::DBobjUtils - > getobjdefs ( \ % objtype , $ callback ) ;
2010-03-25 18:15:06 +00:00
if ( ! ( % imagedef ) )
2009-11-12 20:13:34 +00:00
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not get the xCAT osimage definition for \'$img\'.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
#
# if this is not a "standalone" type image then this is an error
#
if ( $ imagedef { $ img } { nimtype } ne "standalone" )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
2009-11-18 06:51:59 +00:00
"The osimage \'$img\' is not a standalone type. \nThe software maintenance function of updatenode command can only be used for standalone (diskfull) type nodes. \nUse the mknimimage comamand to update diskless osimages." ;
2009-11-12 20:13:34 +00:00
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
2012-08-29 10:21:39 +00:00
# if we're not using the os image
2009-11-12 20:13:34 +00:00
if ( $ osimageonly != 1 )
{
2012-08-29 10:21:39 +00:00
# set the imagedef to the cmd line values
2009-11-12 20:13:34 +00:00
if ( $ attrvals { installp_bundle } )
{
$ imagedef { $ img } { installp_bundle } = $ attrvals { installp_bundle } ;
}
else
{
$ imagedef { $ img } { installp_bundle } = "" ;
}
if ( $ attrvals { otherpkgs } )
{
$ imagedef { $ img } { otherpkgs } = $ attrvals { otherpkgs } ;
}
else
{
$ imagedef { $ img } { otherpkgs } = "" ;
}
}
if ( $ attrvals { installp_flags } )
{
$ imagedef { $ img } { installp_flags } = $ attrvals { installp_flags } ;
}
if ( $ attrvals { rpm_flags } )
{
$ imagedef { $ img } { rpm_flags } = $ attrvals { rpm_flags } ;
}
2012-08-29 10:21:39 +00:00
if ( $ attrvals { emgr_flags } )
{
$ imagedef { $ img } { emgr_flags } = $ attrvals { emgr_flags } ;
}
2009-11-12 20:13:34 +00:00
2010-03-25 18:15:06 +00:00
# get loc of software for node
2012-08-29 10:21:39 +00:00
if ( $ ::ALTSRC )
{
$ imagedef { $ img } { alt_loc } = $ ::ALTSRC ;
}
else
{
if ( $ imagedef { $ img } { lpp_source } )
{
$ imagedef { $ img } { lpp_loc } =
xCAT::InstUtils - > get_nim_attr_val ( $ imagedef { $ img } { lpp_source } ,
'location' , $ callback , $ nimprime , $ subreq ) ;
}
else
{
$ imagedef { $ img } { lpp_loc } = "" ;
next ;
}
}
if ( $ ::ALLSW )
{
# get a list of all the files in the location
# if its an alternate loc than just check that dir
# if it's an lpp_source than check both RPM and installp
my $ rpmloc ;
my $ instploc ;
my $ emgrloc ;
if ( $ ::ALTSRC )
{
# use same loc for everything
$ rpmloc = $ instploc = $ imagedef { $ img } { alt_loc } ;
}
else
{
# use specific lpp_source loc
$ rpmloc = "$imagedef{$img}{lpp_loc}/RPMS/ppc" ;
$ instploc = "$imagedef{$img}{lpp_loc}/installp/ppc" ;
$ emgrloc = "$imagedef{$img}{lpp_loc}/emgr/ppc" ;
}
# get installp filesets in this dir
my $ icmd =
qq~installp -L -d $instploc | /usr/bin/cut -f1 -d':' 2>/dev/null~ ;
my @ ilist = xCAT::Utils - > runcmd ( "$icmd" , - 1 ) ;
foreach my $ f ( @ ilist )
{
if ( ! grep ( /^$f$/ , @ pkglist ) )
{
push ( @ pkglist , $ f ) ;
}
}
# get epkg files
my $ ecmd = qq~/usr/bin/ls $emgrloc 2>/dev/null~ ;
my @ elist = xCAT::Utils - > runcmd ( "$ecmd" , - 1 ) ;
foreach my $ f ( @ elist )
{
if ( ( $ f =~ /epkg\.Z/ ) )
{
push ( @ pkglist , $ f ) ;
}
}
# get rpm packages
my $ rcmd = qq~/usr/bin/ls $rpmloc 2>/dev/null~ ;
my @ rlist = xCAT::Utils - > runcmd ( "$rcmd" , - 1 ) ;
foreach my $ f ( @ rlist )
{
if ( $ f =~ /\.rpm/ )
{
push ( @ pkglist , $ f ) ;
}
}
}
else
{
# use otherpkgs and or installp_bundle
# keep a list of packages from otherpkgs and bndls
if ( $ imagedef { $ img } { otherpkgs } )
{
foreach my $ pkg ( split ( /,/ , $ imagedef { $ img } { otherpkgs } ) )
{
my ( $ junk , $ pname ) ;
$ pname = $ pkg ;
if ( ! grep ( /^$pname$/ , @ pkglist ) )
{
push ( @ pkglist , $ pname ) ;
}
}
}
if ( $ imagedef { $ img } { installp_bundle } )
{
my @ bndlist = split ( /,/ , $ imagedef { $ img } { installp_bundle } ) ;
foreach my $ bnd ( @ bndlist )
{
my ( $ rc , $ list , $ loc ) =
xCAT::InstUtils - > readBNDfile ( $ callback , $ bnd , $ nimprime ,
$ subreq ) ;
foreach my $ pkg ( @$ list )
{
chomp $ pkg ;
if ( ! grep ( /^$pkg$/ , @ pkglist ) )
{
push ( @ pkglist , $ pkg ) ;
}
}
$ bndloc { $ bnd } = $ loc ;
}
}
}
2009-11-12 20:13:34 +00:00
# put array in string to pass along to SN
$ imagedef { $ img } { pkglist } = join ( ',' , @ pkglist ) ;
}
# if there are no SNs to update then return
if ( scalar ( @ SNlist ) == 0 )
{
return ( 0 , \ % imagedef , \ % nodeupdateinfo ) ;
}
2010-03-25 18:15:06 +00:00
# copy pkgs from location on nim prime to same loc on SN
2009-11-12 20:13:34 +00:00
foreach my $ snkey ( @ SNlist )
{
2012-08-29 10:21:39 +00:00
2009-11-12 20:13:34 +00:00
# copy files to SN from nimprime!!
# for now - assume nimprime is management node
foreach my $ img ( @ imagenames )
{
2012-08-29 10:21:39 +00:00
if ( ! $ ::ALTSRC )
{
# if lpp_source is not defined on SN then next
my $ scmd =
qq~/usr/sbin/lsnim -l $imagedef{$img}{lpp_source} 2>/dev/null~ ;
my $ out =
xCAT::InstUtils - > xcmd ( $ callback , $ subreq , "xdsh" , $ snkey ,
$ scmd , 0 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"The NIM lpp_source resource named $imagedef{$img}{lpp_source} is not defined on $snkey. Cannot copy software to $snkey.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
next ;
}
}
# get the dir names to copy to
my $ rpm_srcdir ;
my $ instp_srcdir ;
my $ emgr_srcdir ;
if ( $ ::ALTSRC )
{
$ rpm_srcdir = "$imagedef{$img}{alt_loc}" ;
$ instp_srcdir = "$imagedef{$img}{alt_loc}" ;
$ emgr_srcdir = "$imagedef{$img}{alt_loc}" ;
}
else
{
$ rpm_srcdir = "$imagedef{$img}{lpp_loc}/RPMS/ppc" ;
$ instp_srcdir = "$imagedef{$img}{lpp_loc}/installp/ppc" ;
$ emgr_srcdir = "$imagedef{$img}{lpp_loc}/emgr/ppc" ;
}
# make sure the dir exists on the service node
# also make sure it's writeable by all
my $ mkcmd =
qq~/usr/bin/mkdir -p $rpm_srcdir; chmod 777 $rpm_srcdir; /usr/bin/mkdir -p $instp_srcdir; chmod 777 $instp_srcdir; /usr/bin/mkdir -p $emgr_srcdir; chmod 777 $emgr_srcdir~ ;
my $ output =
xCAT::InstUtils - > xcmd ( $ callback , $ subreq , "xdsh" , $ snkey , $ mkcmd ,
0 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not create directories on $snkey.\n" ;
if ( $ ::VERBOSE )
{
push @ { $ rsp - > { data } } , "$output\n" ;
}
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
next ;
}
2009-11-12 20:13:34 +00:00
# copy all the packages
foreach my $ pkg ( @ pkglist )
{
2010-02-03 08:47:00 +00:00
my $ rcpargs ;
2009-11-12 20:13:34 +00:00
my $ srcfile ;
2012-02-15 20:13:51 +00:00
if ( ( $ pkg =~ /^R:/ ) || ( $ pkg =~ /\.rpm/ ) )
2009-11-12 20:13:34 +00:00
{
2012-08-29 10:21:39 +00:00
my ( $ junk , $ pname ) ;
if ( $ pkg =~ /:/ )
{
( $ junk , $ pname ) = split ( /:/ , $ pkg ) ;
}
else
{
$ pname = $ pkg ;
}
2009-11-12 20:13:34 +00:00
# use rpm location
2010-02-03 08:47:00 +00:00
$ rcpargs = [ "$rpm_srcdir/$pname" , "$rpm_srcdir" ] ;
2012-02-15 20:13:51 +00:00
2012-08-29 10:21:39 +00:00
}
elsif ( ( $ pkg =~ /^E:/ ) || ( $ pkg =~ /epkg\.Z/ ) )
{
2012-02-15 20:13:51 +00:00
2012-08-29 10:21:39 +00:00
my ( $ junk , $ pname ) ;
if ( $ pkg =~ /:/ )
{
2012-02-15 20:13:51 +00:00
( $ junk , $ pname ) = split ( /:/ , $ pkg ) ;
2012-08-29 10:21:39 +00:00
}
else
{
2012-02-15 20:13:51 +00:00
$ pname = $ pkg ;
}
$ rcpargs = [ "$emgr_srcdir/$pname" , "$emgr_srcdir" ] ;
2012-08-29 10:21:39 +00:00
}
2009-11-12 20:13:34 +00:00
else
{
my $ pname ;
my $ junk ;
if ( $ pkg =~ /:/ )
{
( $ junk , $ pname ) = split ( /:/ , $ pkg ) ;
}
else
{
$ pname = $ pkg ;
}
# use installp loc
2012-08-29 10:21:39 +00:00
my @ allfiles = glob "$instp_srcdir/$pname*" ;
2010-02-03 08:47:00 +00:00
my $ sourcefiles = "" ;
2012-08-29 10:21:39 +00:00
foreach my $ file ( @ allfiles )
{
$ sourcefiles . = "$file " ;
2010-02-03 08:47:00 +00:00
}
$ rcpargs = [ $ sourcefiles , "$instp_srcdir" ] ;
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Copying files to $snkey.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdcp" ] ,
node = > [ $ snkey ] ,
arg = > $ rcpargs
} ,
$ subreq , - 1 , 0
) ;
2010-02-03 08:47:00 +00:00
2009-11-12 20:13:34 +00:00
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not copy $pkg to $snkey.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
}
}
}
} # end - for each service node
return ( 0 , \ % imagedef , \ % nodeupdateinfo ) ;
2008-09-25 03:04:56 +00:00
}
2009-11-12 20:13:34 +00:00
#-------------------------------------------------------------------------------
= head3 updateAIXsoftware
Update the software on an xCAT AIX cluster node .
2008-09-25 03:04:56 +00:00
2009-11-12 20:13:34 +00:00
Arguments:
Returns:
0 - OK
1 - error
Example
2010-03-25 18:15:06 +00:00
if ( & updateAIXsoftware ( $ callback , \ % attrres , $ imgdefs , $ updates , $ nodes , $ subreq ) != 0 )
2009-11-12 20:13:34 +00:00
Comments:
= cut
#-------------------------------------------------------------------------------
sub updateAIXsoftware
{
my $ callback = shift ;
2012-08-29 10:21:39 +00:00
my $ attrs = shift ;
2009-11-12 20:13:34 +00:00
my $ imgdefs = shift ;
my $ updates = shift ;
my $ nodes = shift ;
my $ subreq = shift ;
2008-09-08 18:54:30 +00:00
2009-11-12 20:13:34 +00:00
my @ noderange = @$ nodes ;
my % attrvals ; # cmd line attr=val pairs
2011-05-13 13:29:05 +00:00
my % imagedefs ;
my % nodeupdateinfo ;
2009-11-12 20:13:34 +00:00
my @ pkglist ; # list of ALL software to install
2008-09-08 18:54:30 +00:00
2009-11-12 20:13:34 +00:00
# att=val - bndls, otherpakgs, flags
if ( $ attrs )
{
% attrvals = % { $ attrs } ;
}
if ( $ imgdefs )
{
% imagedefs = % { $ imgdefs } ;
}
if ( $ updates )
{
% nodeupdateinfo = % { $ updates } ;
}
my % bndloc ;
# get the server name for each node - as known by node
my $ noderestab = xCAT::Table - > new ( 'noderes' ) ;
my $ xcatmasters =
$ noderestab - > getNodesAttribs ( \ @ noderange , [ 'node' , 'xcatmaster' ] ) ;
# get the NIM primary server name
my $ nimprime = xCAT::InstUtils - > getnimprime ( ) ;
chomp $ nimprime ;
# if it's not the xcatmaster then default to the NIM primary
my % server ;
my @ servers ;
foreach my $ node ( @ noderange )
{
if ( $ xcatmasters - > { $ node } - > [ 0 ] - > { xcatmaster } )
{
$ server { $ node } = $ xcatmasters - > { $ node } - > [ 0 ] - > { xcatmaster } ;
}
else
{
$ server { $ node } = $ nimprime ;
}
if ( ! grep ( $ server { $ node } , @ servers ) )
{
push ( @ servers , $ server { $ node } ) ;
}
}
$ noderestab - > close ;
# sort nodes by image name so we can do bunch at a time
my % nodeoslist ;
foreach my $ node ( @ noderange )
{
2012-08-29 10:21:39 +00:00
push ( @ { $ nodeoslist { $ nodeupdateinfo { $ node } { imagename } } } , $ node ) ;
2009-11-12 20:13:34 +00:00
}
my $ error = 0 ;
my @ installp_files ; # list of tmp installp files created
2012-08-29 10:21:39 +00:00
my @ emgr_files ; # list of tmp emgr file created
2010-03-25 18:15:06 +00:00
2009-11-12 20:13:34 +00:00
foreach my $ img ( keys % imagedefs )
{
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
# set the location of the software
my $ pkgdir = "" ;
if ( $ ::ALTSRC )
{
$ pkgdir = $ ::ALTSRC ;
}
else
{
$ pkgdir = $ imagedefs { $ img } { lpp_loc } ;
}
my $ noinstp = 0 ;
my $ noemgr = 0 ;
chomp $ img ;
if ( $ img )
2009-11-12 20:13:34 +00:00
{
my @ nodes = @ { $ nodeoslist { $ img } } ;
2008-09-08 18:54:30 +00:00
2009-11-12 20:13:34 +00:00
# process the package list
2010-03-25 18:15:06 +00:00
# - split into rpm, emgr, and installp
2009-11-12 20:13:34 +00:00
# - remove leading prefix - if any
my @ rpm_pkgs ;
2012-08-29 10:21:39 +00:00
my @ emgr_pkgs ;
2009-11-12 20:13:34 +00:00
my @ installp_pkgs ;
2010-03-25 18:15:06 +00:00
@ pkglist = split ( /,/ , $ imagedefs { $ img } { pkglist } ) ;
2009-11-12 20:13:34 +00:00
if ( scalar ( @ pkglist ) )
{
foreach my $ p ( @ pkglist )
{
2012-08-29 10:21:39 +00:00
if ( ( $ p =~ /\.rpm/ ) || ( $ p =~ /^R:/ ) )
2009-11-12 20:13:34 +00:00
{
2012-08-29 10:21:39 +00:00
my ( $ junk , $ pname ) ;
if ( $ p =~ /:/ )
{
( $ junk , $ pname ) = split ( /:/ , $ p ) ;
}
else
{
$ pname = $ p ;
}
2009-11-12 20:13:34 +00:00
push @ rpm_pkgs , $ pname ;
2012-08-29 10:21:39 +00:00
}
elsif ( ( $ p =~ /epkg\.Z/ ) || ( $ p =~ /^E:/ ) )
{
2012-02-15 20:13:51 +00:00
2012-08-29 10:21:39 +00:00
my ( $ junk , $ pname ) ;
if ( $ p =~ /:/ )
{
2012-02-15 20:13:51 +00:00
( $ junk , $ pname ) = split ( /:/ , $ p ) ;
2012-08-29 10:21:39 +00:00
}
else
{
2012-02-15 20:13:51 +00:00
$ pname = $ p ;
}
push @ emgr_pkgs , $ pname ;
2012-08-29 10:21:39 +00:00
}
else
{
my ( $ junk , $ pname ) ;
if ( $ p =~ /:/ )
{
2009-11-12 20:13:34 +00:00
( $ junk , $ pname ) = split ( /:/ , $ p ) ;
2012-08-29 10:21:39 +00:00
}
else
{
2009-11-12 20:13:34 +00:00
$ pname = $ p ;
}
push @ installp_pkgs , $ pname ;
}
}
}
2012-08-29 10:21:39 +00:00
my $ thisdate = `date +%s` ;
2010-03-25 18:15:06 +00:00
2009-11-12 20:13:34 +00:00
#
2010-03-25 18:15:06 +00:00
# create tmp file for installp filesets
2009-11-12 20:13:34 +00:00
#
my $ installp_file_name = "installp_file-" . $ thisdate ;
chomp $ installp_file_name ;
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
my $ noinstallp = 0 ;
2009-11-12 20:13:34 +00:00
if ( scalar ( @ installp_pkgs ) )
{
2012-08-29 10:21:39 +00:00
if ( $ pkgdir )
{
if ( ! open ( INSTPFILE , ">/tmp/$installp_file_name" ) )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not open $installp_file_name.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
$ noinstp = 1 ;
}
else
{
foreach ( @ installp_pkgs )
{
print INSTPFILE $ _ . "\n" ;
}
close ( INSTPFILE ) ;
# add new file to list so it can be removed later
push @ installp_files , $ installp_file_name ;
# copy file to each lpp_source, make sure it's
# all readable and export the dir
if ( ( - e "/tmp/$installp_file_name" ) )
{
my $ icmd =
qq~cp /tmp/$installp_file_name $pkgdir; chmod 444 /$pkgdir/$installp_file_name~ ;
my $ output = xCAT::Utils - > runcmd ( "$icmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not copy /tmp/$installp_file_name.\n" ;
push @ { $ rsp - > { data } } , "$output\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
$ noinstp = 1 ;
}
}
}
}
} # end installp tmp file
2009-11-12 20:13:34 +00:00
#
2010-03-25 18:15:06 +00:00
# create tmp file for interim fix packages
2009-11-12 20:13:34 +00:00
#
2010-03-25 18:15:06 +00:00
my $ emgr_file_name = "emgr_file-" . $ thisdate ;
chomp $ emgr_file_name ;
if ( scalar ( @ emgr_pkgs ) )
2009-11-12 20:13:34 +00:00
{
2012-08-29 10:21:39 +00:00
if ( $ pkgdir )
{
if ( ! open ( EMGRFILE , ">/tmp/$emgr_file_name" ) )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not open $emgr_file_name.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
$ noemgr = 1 ;
}
else
{
foreach ( @ emgr_pkgs )
{
print EMGRFILE "./$_" . "\n" ;
}
close ( EMGRFILE ) ;
# add new file to list so it can be removed later
push @ emgr_files , $ emgr_file_name ;
# copy file to each package directory, make sure it's
# all readable and export ed
if ( ( - e "/tmp/$emgr_file_name" ) )
{
my $ icmd =
qq~cp /tmp/$emgr_file_name $pkgdir; chmod 444 $pkgdir/$emgr_file_name~ ;
my $ output = xCAT::Utils - > runcmd ( "$icmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not copy /tmp/$emgr_file_name.\n" ;
push @ { $ rsp - > { data } } , "$output\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
$ noemgr = 1 ;
}
}
}
}
} # end emgr tmp file
# make sure pkg dir is exported
if ( scalar ( @ pkglist ) )
{
my $ ecmd ;
my @ nfsv4 =
xCAT::TableUtils - > get_site_attribute ( "useNFSv4onAIX" ) ;
2011-10-24 11:14:13 +00:00
if ( $ nfsv4 [ 0 ] && ( $ nfsv4 [ 0 ] =~ /1|Yes|yes|YES|Y|y/ ) )
{
$ ecmd = qq~exportfs -i -o vers=4 $pkgdir~ ;
}
else
{
$ ecmd = qq~exportfs -i $pkgdir~ ;
}
2011-05-13 13:29:05 +00:00
my $ output = xCAT::Utils - > runcmd ( "$ecmd" , - 1 ) ;
2009-11-12 20:13:34 +00:00
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
2010-03-25 18:15:06 +00:00
push @ { $ rsp - > { data } } , "Could not export $pkgdir.\n" ;
2009-11-12 20:13:34 +00:00
push @ { $ rsp - > { data } } , "$output\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
2010-03-25 18:15:06 +00:00
$ error + + ;
next ;
2009-11-12 20:13:34 +00:00
}
}
#
# install sw on nodes
#
2012-08-29 10:21:39 +00:00
# $serv is the name of the nodes server as known by the node
if ( scalar ( @ pkglist ) )
{
foreach my $ serv ( @ servers )
{
# make sure the permissions are correct in the lpp_source
my $ chmcmd = qq~/bin/chmod -R +r $pkgdir~ ;
# if server is me then just do chmod
if ( xCAT::InstUtils - > is_me ( $ serv ) )
{
my @ result = xCAT::Utils - > runcmd ( "$chmcmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not set permissions for $pkgdir.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
}
else
{ # if server is remote then use xdsh
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdsh" ] ,
node = > [ $ serv ] ,
arg = > [ $ chmcmd ]
} ,
$ subreq , - 1 , 1
) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not set permissions for $pkgdir.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
}
# mount source dir to node
my $ mcmd ;
my @ nfsv4 =
xCAT::TableUtils - > get_site_attribute ( "useNFSv4onAIX" ) ;
if ( $ nfsv4 [ 0 ] && ( $ nfsv4 [ 0 ] =~ /1|Yes|yes|YES|Y|y/ ) )
{
$ mcmd =
qq~mkdir -m 644 -p /xcatmnt; mount -o vers=4 $serv:$pkgdir /xcatmnt~ ;
}
else
{
$ mcmd =
qq~mkdir -m 644 -p /xcatmnt; mount $serv:$pkgdir /xcatmnt~ ;
}
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running command: $mcmd\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
my $ output =
xCAT::Utils - > runxcmd (
{ command = > [ "xdsh" ] , node = > \ @ nodes , arg = > [ $ mcmd ] } ,
$ subreq , - 1 , 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not mount $pkgdir on nodes.\n" ;
foreach my $ o ( @$ output )
2010-03-25 18:15:06 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
2012-08-29 10:21:39 +00:00
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
next ;
}
}
}
2009-11-12 20:13:34 +00:00
# do installp first
# if we have installp filesets or other installp flags
2012-08-29 10:21:39 +00:00
# we may just get flags!
if (
(
( scalar ( @ installp_pkgs ) )
|| $ ::ALLSW
|| ( $ imagedefs { $ img } { installp_flags } )
)
&& ! $ noinstp
)
{
2009-11-12 20:13:34 +00:00
# - use installp with file
# set flags
my $ flags ;
if ( $ imagedefs { $ img } { installp_flags } )
{
$ flags = " " . $ imagedefs { $ img } { installp_flags } ;
}
else
{
$ flags = " -agQX " ;
}
# put together the installp command
my $ inpcmd = qq~/usr/sbin/installp ~ ;
# these installp flags can be used with -d
if ( $ flags =~ /l|L|i|A|a/ )
{
2012-08-29 10:21:39 +00:00
# if a specific dir was provided then use it
# otherwise use the installp dir in the lpp src
if ( $ ::ALTSRC )
{
$ inpcmd . = qq~-d /xcatmnt ~ ;
}
else
{
$ inpcmd . = qq~-d /xcatmnt/installp/ppc ~ ;
}
2009-11-12 20:13:34 +00:00
}
$ inpcmd . = qq~$flags ~ ;
# don't provide a list of filesets with these flags
2012-08-29 10:21:39 +00:00
if ( $ flags !~ /C|L|l/ )
{
if ( $ ::ALLSW )
{
2010-03-25 18:15:06 +00:00
# we want all sw installed
$ inpcmd . = qq~ all~ ;
2012-08-29 10:21:39 +00:00
}
elsif ( scalar ( @ installp_pkgs ) == 0 )
{
2010-03-25 18:15:06 +00:00
# there is no sw to install
2012-08-29 10:21:39 +00:00
$ noinstallp = 1 ;
}
else
{
2010-03-25 18:15:06 +00:00
# install what is in installp_pkgs
$ inpcmd . = qq~-f /xcatmnt/$installp_file_name~ ;
}
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
# - could just have installp flags by mistake -ugh!
# - but don't have fileset to install - so don't run
# installp - UNLESS the flags don't need filesets
if ( $ noinstallp == 0 )
{
2009-11-17 18:53:10 +00:00
2012-08-29 10:21:39 +00:00
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running: \'$inpcmd\'.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdsh" ] ,
node = > \ @ nodes ,
arg = > [ $ inpcmd ]
} ,
$ subreq , - 1 , 1
) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not run installp command.\n" ;
foreach my $ o ( @$ output )
2010-03-25 18:15:06 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
2012-08-29 10:21:39 +00:00
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
$ error + + ;
2010-10-05 19:09:24 +00:00
2012-08-29 10:21:39 +00:00
}
elsif ( $ ::VERBOSE )
{
my $ rsp ;
foreach my $ o ( @$ output )
2010-03-25 18:15:06 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
2012-08-29 10:21:39 +00:00
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
}
#
# - run updtvpkg to make sure installp software
# is registered with rpm
#
my $ upcmd = qq~/usr/sbin/updtvpkg~ ;
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running command: $upcmd\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdsh" ] ,
node = > \ @ nodes ,
arg = > [ $ upcmd ]
} ,
$ subreq , - 1 , 1
) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not run updtvpkg.\n" ;
foreach my $ o ( @$ output )
2010-03-25 18:15:06 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
2012-08-29 10:21:39 +00:00
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
}
2009-11-12 20:13:34 +00:00
}
2012-08-29 10:21:39 +00:00
#
# do any interim fixes using emgr
#
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
# if we have epkgs or emgr flags
# we may just get flags!
if (
(
( scalar ( @ emgr_pkgs ) )
|| $ ::ALLSW
|| ( $ imagedefs { $ img } { emgr_flags } )
)
&& ! $ noemgr
)
{
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
# if a specific dir was provided then use it
2012-02-15 20:13:51 +00:00
# otherwise use the rpm dir in the lpp src
my $ dir ;
2012-08-29 10:21:39 +00:00
if ( $ ::ALTSRC )
{
2012-02-15 20:13:51 +00:00
$ dir = "/xcatmnt" ;
2012-08-29 10:21:39 +00:00
}
else
{
2012-02-15 20:13:51 +00:00
$ dir = "/xcatmnt/emgr/ppc" ;
}
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
my $ emgrcmd = qq~cd $dir; /usr/sbin/emgr~ ;
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
if ( $ imagedefs { $ img } { emgr_flags } )
{
2012-02-15 20:13:51 +00:00
$ emgrcmd . = qq~ $imagedefs{$img}{emgr_flags}~ ;
}
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
if ( ( scalar ( @ emgr_pkgs ) ) || $ ::ALLSW )
{
# call emgr with -f filename
$ emgrcmd . = qq~ -f /xcatmnt/$emgr_file_name~ ;
}
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running: \'$emgrcmd\'.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdsh" ] ,
node = > \ @ nodes ,
arg = > [ $ emgrcmd ]
} ,
$ subreq , - 1 , 1
) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not run emgr command.\n" ;
foreach my $ o ( @$ output )
{
push @ { $ rsp - > { data } } , "$o" ;
}
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
$ error + + ;
}
elsif ( $ ::VERBOSE )
{
my $ rsp ;
foreach my $ o ( @$ output )
{
push @ { $ rsp - > { data } } , "$o" ;
}
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
}
#
# do RPMs
#
2010-03-25 18:15:06 +00:00
if ( scalar ( @ rpm_pkgs ) || $ ::ALLSW || ( $ imagedefs { $ img } { rpm_flags } ) )
2009-11-12 20:13:34 +00:00
{
# don't do rpms if these installp flags were specified
2012-08-29 10:21:39 +00:00
# check this ??????!!!!! - this check doesn't seem necessary?
2010-03-25 18:15:06 +00:00
#if ($imagedefs{$img}{installp_flags} !~ /C|L|l/)
2012-08-29 10:21:39 +00:00
if ( 1 )
2009-11-12 20:13:34 +00:00
{
# set flags
my $ flags ;
if ( $ imagedefs { $ img } { rpm_flags } )
{
$ flags = " " . $ imagedefs { $ img } { rpm_flags } ;
}
else
{
2012-08-29 10:21:39 +00:00
# use --replacepkgs so cmd won't fail if one is
# already installed
2009-11-12 20:13:34 +00:00
$ flags = " -Uvh --replacepkgs " ;
}
2012-08-29 10:21:39 +00:00
# if a specific dir was provided then use it
2010-03-25 18:15:06 +00:00
# otherwise use the rpm dir in the lpp src
2012-08-29 10:21:39 +00:00
my $ dir ;
if ( $ ::ALTSRC )
{
2010-03-25 18:15:06 +00:00
$ dir = "/xcatmnt" ;
2012-08-29 10:21:39 +00:00
}
else
{
2010-03-25 18:15:06 +00:00
$ dir = "/xcatmnt/RPMS/ppc" ;
2009-11-12 20:13:34 +00:00
}
2010-03-25 18:15:06 +00:00
my $ pkg_string = "" ;
2012-08-29 10:21:39 +00:00
if ( $ ::ALLSW )
{
$ pkg_string = " *.rpm " ;
}
else
{
foreach my $ pkg ( @ rpm_pkgs )
{
$ pkg_string . = " $pkg" ;
}
}
my $ rcmd ;
if ( scalar ( @ rpm_pkgs ) )
{
$ rcmd = qq~cd $dir; /usr/bin/rpm $flags $pkg_string ~ ;
}
else
{
$ rcmd = qq~/usr/bin/rpm $flags ~ ;
}
2009-11-12 20:13:34 +00:00
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running: \'$rcmd\'.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
2012-08-29 10:21:39 +00:00
my $ output =
xCAT::Utils - > runxcmd (
{ command = > [ "xdsh" ] , node = > \ @ nodes , arg = > [ $ rcmd ] } ,
$ subreq , - 1 , 1 ) ;
2010-03-25 18:15:06 +00:00
2012-08-29 10:21:39 +00:00
if ( ( $ ::RUNCMD_RC != 0 ) && ( ! $ ::ALLSW ) )
{
2009-11-12 20:13:34 +00:00
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not install RPMs.\n" ;
2012-08-29 10:21:39 +00:00
foreach my $ o ( @$ output )
2009-11-12 20:13:34 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
$ error + + ;
2012-08-29 10:21:39 +00:00
}
elsif ( $ ::VERBOSE )
{
# should we always give output??
# could get gobs of unwanted output in some cases
2009-11-12 20:13:34 +00:00
my $ rsp ;
2012-08-29 10:21:39 +00:00
push @ { $ rsp - > { data } } , "Command output:\n" ;
foreach my $ o ( @$ output )
2009-11-12 20:13:34 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
}
}
2012-08-29 10:21:39 +00:00
# unmount the src dir -
if ( scalar ( @ pkglist ) )
{
my $ ucmd = qq~umount -f /xcatmnt~ ;
if ( $ ::VERBOSE )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Running command: $ucmd\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
my $ output =
xCAT::Utils - > runxcmd (
{
command = > [ "xdsh" ] ,
node = > \ @ nodes ,
arg = > [ $ ucmd ]
} ,
$ subreq , - 1 , 1
) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not umount.\n" ;
foreach my $ o ( @$ output )
2010-03-25 18:15:06 +00:00
{
push @ { $ rsp - > { data } } , "$o" ;
}
2012-08-29 10:21:39 +00:00
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
next ;
}
}
2009-11-12 20:13:34 +00:00
}
}
# clean up files copied to lpp_source locations and
# unexport the lpp locations
foreach my $ img ( keys % imagedefs )
{
2010-03-25 18:15:06 +00:00
2009-11-12 20:13:34 +00:00
chomp $ img ;
2012-08-29 10:21:39 +00:00
my $ pkgdir ;
if ( $ ::ALTSRC )
{
$ pkgdir = $ ::ALTSRC ;
}
else
{
$ pkgdir = $ imagedefs { $ img } { lpp_loc } ;
}
if ( scalar ( @ pkglist ) )
{
# remove tmp installp files
foreach my $ file ( @ installp_files )
{
my $ rcmd ;
$ rcmd = qq~rm -f $pkgdir/$file; rm -f /tmp/$file~ ;
my $ output = xCAT::Utils - > runcmd ( "$rcmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not remove $imagedefs{$img}{lpp_loc}/$file.\n" ;
if ( $ ::VERBOSE )
{
push @ { $ rsp - > { data } } , "$output\n" ;
}
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
next ;
}
}
# remove tmp emgr files
foreach my $ file ( @ emgr_files )
{
my $ rcmd ;
$ rcmd = qq~rm -f $pkgdir/$file; rm -f /tmp/$file~ ;
my $ output = xCAT::Utils - > runcmd ( "$rcmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Could not remove $imagedefs{$img}{lpp_loc}/$file.\n" ;
if ( $ ::VERBOSE )
{
push @ { $ rsp - > { data } } , "$output\n" ;
}
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
next ;
}
}
# unexport lpp dirs
my $ ucmd = qq~exportfs -u -F $pkgdir~ ;
my $ output = xCAT::Utils - > runcmd ( "$ucmd" , - 1 ) ;
if ( $ ::RUNCMD_RC != 0 )
{
my $ rsp ;
push @ { $ rsp - > { data } } , "Could not unexport $pkgdir.\n" ;
if ( $ ::VERBOSE )
{
push @ { $ rsp - > { data } } , "$output\n" ;
}
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
$ error + + ;
next ;
}
}
}
2009-11-12 20:13:34 +00:00
if ( $ error )
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"One or more errors occured while updating node software.\n" ;
xCAT::MsgUtils - > message ( "E" , $ rsp , $ callback ) ;
return 1 ;
}
else
{
my $ rsp ;
push @ { $ rsp - > { data } } ,
"Cluster node software update commands have completed successfully.\n" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
return 0 ;
}
2009-12-01 18:21:30 +00:00
2010-07-12 00:40:50 +00:00
#-------------------------------------------------------
= head3 updateOS
Description : Update the node operating system
Arguments :
Returns : Nothing
Example : updateOS ( $ callback , $ nodes , $ os ) ;
= cut
#-------------------------------------------------------
2012-08-29 10:21:39 +00:00
sub updateOS
{
# Get inputs
my ( $ callback , $ node , $ os ) = @ _ ;
my $ rsp ;
# Get install directory
my $ installDIR = xCAT::TableUtils - > getInstallDir ( ) ;
# Get HTTP server
my $ http = xCAT::NetworkUtils - > my_ip_facing ( $ node ) ;
if ( ! $ http )
{
push @ { $ rsp - > { data } } , "$node: (Error) Missing HTTP server" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Get OS to update to
my $ update2os = $ os ;
push @ { $ rsp - > { data } } , "$node: Upgrading $node to $os" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Get the OS that is installed on the node
my $ arch = `ssh -o ConnectTimeout=5 $node "uname -m"` ;
chomp ( $ arch ) ;
my $ installOS ;
my $ version ;
# Red Hat Linux
if (
`ssh -o ConnectTimeout=5 $node "test -f /etc/redhat-release && echo 'redhat'"`
)
{
$ installOS = "rh" ;
chomp ( $ version =
`ssh $node "tr -d '.' < /etc/redhat-release" | head -n 1` ) ;
$ version =~ s/[^0-9]*([0-9]+).*/$1/ ;
}
# SUSE Linux
elsif (
`ssh -o ConnectTimeout=5 $node "test -f /etc/SuSE-release && echo 'SuSE'"`
)
{
$ installOS = "sles" ;
chomp ( $ version =
`ssh $node "tr -d '.' < /etc/SuSE-release" | head -n 1` ) ;
$ version =~ s/[^0-9]*([0-9]+).*/$1/ ;
}
# Everything else
else
{
$ installOS = "Unknown" ;
push @ { $ rsp - > { data } } , "$node: (Error) Linux distribution not supported" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Is the installed OS and the update to OS of the same distributor
if ( ! ( $ update2os =~ m/$installOS/i ) )
{
push @ { $ rsp - > { data } } ,
"$node: (Error) Cannot not update $installOS$version to $os. Linux distribution does not match" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Setup the repository for the node
my $ path ;
my $ out ;
if ( "$installOS$version" =~ m/sles10/i )
{
# SUSE repository path - http://10.1.100.1/install/sles10.3/s390x/1/
$ path = "http://$http$installDIR/$os/$arch/1/" ;
if ( ! ( - e "$installDIR/$os/$arch/1/" ) )
{
push @ { $ rsp - > { data } } ,
"$node: (Error) Missing install directory $installDIR/$os/$arch/1/" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Add installation source using rug
$ out = `ssh $node "rug sa -t zypp $path $os"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Subscribe to catalog
$ out = `ssh $node "rug sub $os"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Refresh services
$ out = `ssh $node "rug ref"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Update
$ out = `ssh $node "rug up -y"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
elsif ( "$installOS$version" =~ m/sles11/i )
{
# SUSE repository path - http://10.1.100.1/install/sles10.3/s390x/1/
$ path = "http://$http$installDIR/$os/$arch/1/" ;
if ( ! ( - e "$installDIR/$os/$arch/1/" ) )
{
push @ { $ rsp - > { data } } ,
"$node: (Error) Missing install directory $installDIR/$os/$arch/1/" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Add installation source using zypper
$ out = `ssh $node "zypper ar $path $installOS$version"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Refresh services
$ out = `ssh $node "zypper ref"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
# Update
$ out =
`ssh $node "zypper --non-interactive update --auto-agree-with-licenses"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
elsif ( "$installOS$version" =~ m/rh/i )
{
# Red Hat repository path - http://10.0.0.1/install/rhel5.4/s390x/Server/
$ path = "http://$http$installDIR/$os/$arch/Server/" ;
if ( ! ( - e "$installDIR/$os/$arch/Server/" ) )
{
push @ { $ rsp - > { data } } ,
"$node: (Error) Missing install directory $installDIR/$os/$arch/Server/" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
return ;
}
# Create a yum repository file
my $ exist =
`ssh $node "test -e /etc/yum.repos.d/$os.repo && echo 'File exists'"` ;
if ( ! $ exist )
{
$ out = `ssh $node "echo [$os] >> /etc/yum.repos.d/$os.repo"` ;
$ out =
`ssh $node "echo baseurl=$path >> /etc/yum.repos.d/$os.repo"` ;
$ out = `ssh $node "echo enabled=1 >> /etc/yum.repos.d/$os.repo"` ;
}
# Send over release key
my $ key = "$installDIR/$os/$arch/RPM-GPG-KEY-redhat-release" ;
my $ tmp = "/tmp/RPM-GPG-KEY-redhat-release" ;
my $ tgt = "root@" . $ node ;
$ out = `scp $key $tgt:$tmp` ;
# Import key
$ out = `ssh $node "rpm --import $tmp"` ;
# Upgrade
$ out = `ssh $node "yum -y upgrade"` ;
push @ { $ rsp - > { data } } , "$node: $out" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
else
{
push @ { $ rsp - > { data } } ,
"$node: (Error) Could not update operating system" ;
xCAT::MsgUtils - > message ( "I" , $ rsp , $ callback ) ;
}
return ;
2010-07-12 00:40:50 +00:00
}
2010-11-24 07:07:56 +00:00
#-------------------------------------------------------------------------------
= head3 updatenodeappstat
This subroutine is used to handle the messages reported by
HPCbootstatus postscript . Update appstatus node attribute .
Arguments:
Returns:
0 - for success .
1 - for error .
= cut
#-----------------------------------------------------------------------------
sub updatenodeappstat
{
my $ request = shift ;
my $ callback = shift ;
my @ nodes = ( ) ;
my @ args = ( ) ;
if ( ref ( $ request - > { node } ) )
{
@ nodes = @ { $ request - > { node } } ;
}
else
{
if ( $ request - > { node } ) { @ nodes = ( $ request - > { node } ) ; }
}
if ( ref ( $ request - > { arg } ) )
{
@ args = @ { $ request - > { arg } } ;
}
else
{
@ args = ( $ request - > { arg } ) ;
}
if ( ( @ nodes > 0 ) && ( @ args > 0 ) )
{
2012-08-29 10:21:39 +00:00
2010-11-24 07:07:56 +00:00
# format: apps=status
my $ appstat = $ args [ 0 ] ;
2012-08-29 10:21:39 +00:00
my ( $ apps , $ newstatus ) = split ( /=/ , $ appstat ) ;
2010-11-24 07:07:56 +00:00
2012-08-09 04:07:40 +00:00
xCAT::TableUtils - > setAppStatus ( \ @ nodes , $ apps , $ newstatus ) ;
2010-11-24 07:07:56 +00:00
2012-08-29 10:21:39 +00:00
}
2010-11-24 07:07:56 +00:00
return 0 ;
}