2009-09-21 11:30:05 +00:00
# IBM(c) 2009 EPL license http://www.eclipse.org/legal/epl-v10.html
2009-09-21 06:33:27 +00:00
package xCAT::PPCenergy ;
2012-09-12 19:24:22 +00:00
BEGIN
{
$ ::XCATROOT = $ ENV { 'XCATROOT' } ? $ ENV { 'XCATROOT' } : '/opt/xcat' ;
}
use lib "$::XCATROOT/lib/perl" ;
2009-09-21 06:33:27 +00:00
use strict ;
use Getopt::Long ;
use xCAT::Usage ;
use xCAT::NodeRange ;
2011-01-26 13:27:25 +00:00
use xCAT::DBobjUtils ;
2012-05-03 04:58:16 +00:00
use xCAT::FSPUtils ;
2012-08-09 03:48:50 +00:00
use xCAT::TableUtils qw( get_site_Master ) ;
2009-09-21 06:33:27 +00:00
% ::QUERY_ATTRS = (
'savingstatus' = > 1 ,
2010-01-30 14:35:01 +00:00
'dsavingstatus' = > 1 ,
2009-09-21 06:33:27 +00:00
'cappingstatus' = > 1 ,
'cappingmaxmin' = > 1 ,
'cappingvalue' = > 1 ,
'cappingsoftmin' = > 1 ,
'averageAC' = > 1 ,
'averageDC' = > 1 ,
'ambienttemp' = > 1 ,
'exhausttemp' = > 1 ,
2010-01-30 14:35:01 +00:00
'CPUspeed' = > 1 ,
'syssbpower' = > 1 ,
'sysIPLtime' = > 1 ,
2011-06-27 15:10:15 +00:00
# for FFO, only supported when communicating to fsp directly
'ffoMin' = > 1 ,
'ffoVmin' = > 1 ,
'ffoTurbo' = > 1 ,
'ffoNorm' = > 1 ,
'fsavingstatus' = > 1 ,
'ffovalue' = > 1 ,
2010-01-30 14:35:01 +00:00
) ;
2009-09-21 06:33:27 +00:00
% ::SET_ATTRS = (
'savingstatus' = > 1 ,
2010-01-30 14:35:01 +00:00
'dsavingstatus' = > 1 ,
2009-09-21 06:33:27 +00:00
'cappingstatus' = > 1 ,
'cappingwatt' = > 1 ,
'cappingperc' = > 1 ,
2011-06-27 15:10:15 +00:00
# for FFO
'fsavingstatus' = > 1 ,
'ffovalue' = > 1 ,
2009-09-21 06:33:27 +00:00
) ;
$ ::CIM_CLIENT_PATH = "$::XCATROOT/sbin/xCAT_cim_client" ;
# Parse the arguments of the command line for renergy command
sub parse_args {
my $ request = shift ;
my % opt = ( ) ;
my $ cmd = $ request - > { command } ;
my $ args = $ request - > { arg } ;
my $ nodes = $ request - > { node } ;
my $ query_attrs = ( ) ; # The attributes list for query operation
my $ set_pair = ( ) ; # The attribute need to be set. e.g. savingstatus=on
my $ set_flag = ( ) ; # Indicate there's setting param in the argv
my $ argv_flag = ( ) ; # Indicate there's param in the argv
my @ notfspnodes = ( ) ; # The nodes list which are not fsp
# set the usage subroutine
local * usage = sub {
my $ usage_string = xCAT::Usage - > getUsage ( $ cmd ) ;
2010-01-30 14:35:01 +00:00
return ( [ $ _ [ 0 ] , $ usage_string ] ) ;
2009-09-21 06:33:27 +00:00
} ;
if ( $ request - > { arg } ) {
@ ARGV = @ { $ request - > { arg } } ;
$ Getopt:: Long:: ignorecase = 0 ;
Getopt::Long:: Configure ( "bundling" ) ;
if ( $ nodes ) {
if ( ! GetOptions ( 'V' = > \ $ ::VERBOSE ) ) {
return ( & usage ( ) ) ;
}
if ( $ ::VERBOSE ) {
$ opt { verbose } = 1 ;
}
if ( $# ARGV < 0 ) {
return ( & usage ( ) ) ;
}
# Check the validity of the parameters of Query and Set
foreach my $ attr ( @ ARGV ) {
my ( $ set_attr , $ set_value ) = split ( /=/ , $ attr ) ;
2010-01-30 14:35:01 +00:00
if ( defined ( $ set_value ) ) {
2009-09-21 06:33:27 +00:00
if ( $ argv_flag ) {
return ( & usage ( ) ) ;
}
if ( $ ::SET_ATTRS { $ set_attr } != 1 ) {
return ( & usage ( ) ) ;
}
2011-06-27 15:10:15 +00:00
if ( ( $ set_attr eq "savingstatus" || $ set_attr eq "fsavingstatus" )
2009-09-21 06:33:27 +00:00
&& ( $ set_value ne "on" && $ set_value ne "off" ) ) {
return ( & usage ( ) ) ;
2010-01-30 14:35:01 +00:00
} elsif ( $ set_attr eq "dsavingstatus"
&& ( $ set_value ne "off"
&& $ set_value ne "on-norm" && $ set_value ne "on-maxp" ) ) {
return ( & usage ( ) ) ;
2009-09-21 06:33:27 +00:00
} elsif ( $ set_attr eq "cappingstatus"
&& ( $ set_value ne "on" && $ set_value ne "off" ) ) {
return ( & usage ( ) ) ;
} elsif ( ( $ set_attr eq "cappingwatt"
2011-06-27 15:10:15 +00:00
|| $ set_attr eq "cappingperc" || $ set_attr eq "ffovalue" )
2009-09-21 06:33:27 +00:00
&& $ set_value =~ /\D/ ) {
return ( & usage ( ) ) ;
}
$ set_pair = $ set_attr . "=" . $ set_value ;
$ set_flag = 1 ;
} else {
if ( $ set_flag ) {
return ( & usage ( ) ) ;
}
}
$ argv_flag = 1 ;
}
if ( ! $ set_flag ) {
my @ query_list = @ ARGV ;
2010-01-30 14:35:01 +00:00
2009-09-21 06:33:27 +00:00
if ( $ query_list [ 0 ] eq "all" and $# query_list == 0 ) {
2010-01-30 14:35:01 +00:00
$ query_attrs = "all" ;
2009-09-21 06:33:27 +00:00
} else {
my @ no_dup_query_list = ( ) ;
foreach my $ q_attr ( @ query_list ) {
chomp ( $ q_attr ) ;
2010-01-30 14:35:01 +00:00
2009-09-21 06:33:27 +00:00
if ( $ ::QUERY_ATTRS { $ q_attr } != 1 ) {
return ( & usage ( ) ) ;
}
2009-09-21 11:30:05 +00:00
if ( ! grep ( /^$q_attr$/ , @ no_dup_query_list ) ) {
2009-09-21 06:33:27 +00:00
push @ no_dup_query_list , $ q_attr ;
}
}
$ query_attrs = join ( ',' , @ no_dup_query_list ) ;
}
}
} else {
# If has not nodes, the -h or -v option must be input
if ( ! GetOptions ( 'h|help' = > \ $ ::HELP ,
'v|version' = > \ $ ::VERSION ) ) {
return ( & usage ( ) ) ;
}
if ( ! ( $ ::HELP || $ ::VERSION ) ) {
return ( & usage ( ) ) ;
}
if ( $ ::HELP ) {
return ( & usage ( ) ) ;
}
if ( $ ::VERSION ) {
my $ version_string = xCAT::Usage - > getVersion ( 'renergy' ) ;
return ( [ $ _ [ 0 ] , $ version_string ] ) ;
}
2010-01-30 14:35:01 +00:00
if ( scalar ( @ ARGV ) ) {
return ( & usage ( ) ) ;
}
2009-09-21 06:33:27 +00:00
}
} else {
return ( & usage ( ) ) ;
}
2011-01-26 13:27:25 +00:00
# Check whether the hardware type of nodes are fsp or cec
2011-06-07 08:25:56 +00:00
#my $nodetype_tb = xCAT::Table->new('nodetype');
#unless ($nodetype_tb) {
# return ([undef, "Error: Cannot open the nodetype table"]);
#}
#my $nodetype_v = $nodetype_tb->getNodesAttribs($nodes, ['nodetype']);
2012-02-10 10:59:00 +00:00
my $ nodetyperef = xCAT::DBobjUtils - > getnodetype ( $ nodes , "ppc" ) ;
2011-06-07 08:25:56 +00:00
my $ i = 0 ;
2009-09-21 06:33:27 +00:00
foreach my $ node ( @ { $ nodes } ) {
2012-05-02 02:34:07 +00:00
if ( $$ nodetyperef { $ node } ne 'fsp' &&
$$ nodetyperef { $ node } ne 'cec' ) {
2009-09-21 06:33:27 +00:00
push @ notfspnodes , $ node ;
}
2011-06-07 08:25:56 +00:00
$ i + + ;
2009-09-21 06:33:27 +00:00
}
2011-06-07 08:25:56 +00:00
#$nodetype_tb->close();
2009-09-21 06:33:27 +00:00
if ( @ notfspnodes ) {
2011-01-26 13:27:25 +00:00
return ( [ undef , "Error: The hardware type of following nodes are not fsp or cec: " . join ( ',' , @ notfspnodes ) ] ) ;
2009-09-21 06:33:27 +00:00
}
if ( $ query_attrs ) {
$ opt { 'query' } = $ query_attrs ;
} elsif ( $ set_pair ) {
$ opt { 'set' } = $ set_pair ;
}
$ request - > { method } = $ cmd ;
return ( \ % opt ) ;
}
# Handle the energy query and setting work
sub renergy {
my $ request = shift ;
my $ hcphost = shift ;
my $ nodehash = shift ;
my @ return_msg = ( ) ;
my $ opt = $ request - > { 'opt' } ;
my $ verbose = $ opt - > { 'verbose' } ;
# Get the CEC
my ( $ node , $ attrs ) = %$ nodehash ;
my $ cec_name = @$ attrs [ 2 ] ;
2011-01-26 13:27:25 +00:00
my $ hw_type = @$ attrs [ 4 ] ;
2009-09-21 06:33:27 +00:00
2011-01-26 13:27:25 +00:00
if ( ! $ cec_name ) {
return ( [ [ $ node , "ERROR: Cannot find the cec name, check the attributes: vpd.serial, vpd.mtm." , 1 ] ] ) ;
2009-09-21 06:33:27 +00:00
}
# Check the existence of cim client
if ( ( ! - f $ ::CIM_CLIENT_PATH )
|| ( ! - x $ ::CIM_CLIENT_PATH ) ) {
2010-06-18 12:40:14 +00:00
return ( [ [ $ node , "ERROR: Cannot find the Energy Management Plugin for xCAT [$::CIM_CLIENT_PATH] or it's NOT executable. Please install the xCAT-cimclient package correctly. Get more information from man page of renergy command." , 1 ] ] ) ;
2009-09-21 06:33:27 +00:00
}
my $ verb_arg = "" ;
if ( $ verbose ) {
$ verb_arg = "-V" ;
}
2011-01-26 13:27:25 +00:00
# get the user and passwd for hcp: hmc, fsp, cec
2012-02-10 10:59:00 +00:00
my $ hcp_type = xCAT::DBobjUtils - > getnodetype ( $ hcphost , "ppc" ) ;
2011-01-26 13:27:25 +00:00
my $ user ;
my $ password ;
2011-01-27 09:09:18 +00:00
if ( $ hcp_type eq "hmc" ) {
2011-01-26 13:27:25 +00:00
( $ user , $ password ) = xCAT::PPCdb:: credentials ( $ hcphost , $ hcp_type ) ;
} else {
( $ user , $ password ) = xCAT::PPCdb:: credentials ( $ hcphost , $ hcp_type , 'HMC' ) ;
}
my $ fsps ; #The node of fsp that belong to the cec
2011-02-18 02:10:25 +00:00
my @ hcps_ip ;
if ( $ hcp_type ne "hmc" && $ hw_type eq "cec" ) {
2011-01-26 13:27:25 +00:00
$ fsps = xCAT::DBobjUtils - > getchildren ( $ node ) ;
if ( ! defined ( $ fsps ) ) {
return ( [ [ $ node , "Failed to get the FSPs for the cec $hcphost." , - 1 ] ] ) ;
}
2012-08-09 03:48:50 +00:00
#my $hcp_ip = xCAT::NetworkUtils::getNodeIPaddress($hcphost);
2012-05-02 06:30:04 +00:00
my $ hcp_ip = xCAT::FSPUtils:: getIPaddress ( $ request , $ hw_type , $ hcphost ) ;
2011-12-15 07:28:02 +00:00
if ( ! defined ( $ hcp_ip ) or ( $ hcp_ip == - 3 ) ) {
return ( [ [ $ node , "Failed to get IP address for $hcphost." , - 1 ] ] ) ;
}
2011-02-18 02:10:25 +00:00
push @ hcps_ip , split ( ',' , $ hcp_ip ) ;
2011-01-26 13:27:25 +00:00
my $ fsp_node = $$ fsps [ 0 ] ;
( $ user , $ password ) = xCAT::PPCdb:: credentials ( $ fsp_node , "fsp" , 'HMC' ) ;
if ( ! $ password ) {
return ( [ [ $ node , "Cannot get password of userid 'HMC'. Please check table 'ppchcp' or 'ppcdirect'." , - 1 ] ] ) ;
}
2011-02-18 02:10:25 +00:00
} else {
# for the case that hcp is hmc or fsp
2012-08-09 03:48:50 +00:00
push @ hcps_ip , xCAT::NetworkUtils:: getNodeIPaddress ( $ hcphost ) ;
2011-02-18 02:10:25 +00:00
}
2011-01-26 13:27:25 +00:00
if ( ! $ user || ! $ password ) {
return ( [ [ $ node , "Cannot get user:password for the node. Please check table 'ppchcp' or 'ppcdirect'." , - 1 ] ] ) ;
}
2012-04-24 01:50:26 +00:00
# secure passwd in verbose mode
my $ tmpv = $ verbose ;
$ verbose = 0 ;
2009-09-21 06:33:27 +00:00
if ( $ verbose ) {
2011-02-18 02:10:25 +00:00
push @ return_msg , [ $ node , "Attributes of $node:\n User=$user\n Password=$password\n CEC=$cec_name\n nodetype=$hw_type\n inithcp=$hcphost\n hcps=@hcps_ip\n hcptype=$hcp_type" , 0 ] ;
2009-09-21 06:33:27 +00:00
}
2012-04-24 01:50:26 +00:00
$ verbose = $ tmpv ;
2009-09-21 06:33:27 +00:00
2011-01-26 13:27:25 +00:00
my @ pingable_hcp ;
2012-09-12 19:24:22 +00:00
if ( - x '/usr/bin/nmap' or - x '/usr/local/bin/nmap' ) { #use nmap
#print "use nmap\n";
if ( $ verbose ) {
push @ return_msg , [ $ node , "Checking ping status using nmap for @hcps_ip" , 0 ] ;
}
my % deadnodes ;
foreach ( @ hcps_ip ) {
$ deadnodes { $ _ } = 1 ;
}
open ( NMAP , "nmap -PE --system-dns --send-ip -sP " . join ( ' ' , @ hcps_ip ) . " 2> /dev/null|" ) or die ( "Cannot open nmap pipe: $!" ) ;
my $ node1 ;
my $ msg1 ;
while ( <NMAP> ) {
#print "$_\n";
if ( /Host (.*) \((.*)\) appears to be up/ ) {
$ node1 = $ 2 ;
unless ( $ deadnodes { $ node1 } ) {
foreach ( keys % deadnodes ) {
if ( $ node1 =~ /^$_\./ ) {
$ node1 = $ _ ;
last ;
}
}
}
delete $ deadnodes { $ node1 } ;
if ( $ verbose ) {
push @ return_msg , [ $ node , $ _ , 0 ] ;
}
push ( @ pingable_hcp , $ node1 ) ;
} elsif ( /Nmap scan report for ([^ ]*) \((.*)\)/ ) {
$ node1 = $ 2 ;
$ msg1 = $ _ ;
} elsif ( /Host is up./ ) {
unless ( $ deadnodes { $ node1 } ) {
foreach ( keys % deadnodes ) {
if ( $ node1 =~ /^$_\./ ) {
$ node1 = $ _ ;
last ;
}
}
}
delete $ deadnodes { $ node1 } ;
if ( $ verbose ) {
push @ return_msg , [ $ node , "$msg1$_" , 0 ] ;
}
push ( @ pingable_hcp , $ node1 ) ;
}
}
} else {
#use fping
#print "use fping\n";
my $ master = xCAT::TableUtils - > get_site_Master ( ) ;
my $ masterip = xCAT::NetworkUtils - > getipaddr ( $ master ) ;
if ( $ masterip =~ /:/ ) { #IPv6, needs fping6 support
if ( ! - x '/usr/bin/fping6' )
{
push @ return_msg , [ $ node , "fping6 is not availabe for IPv6 ping." , - 1 ] ;
return \ @ return_msg ;
}
open ( FPING , "fping6 " . join ( ' ' , @ hcps_ip ) . " 2>&1 |" ) or die ( "Cannot open fping pipe: $!" ) ;
} else {
open ( FPING , "fping " . join ( ' ' , @ hcps_ip ) . " 2>&1 |" ) or die ( "Cannot open fping pipe: $!" ) ;
}
while ( <FPING> ) {
if ( $ verbose ) {
push @ return_msg , [ $ node , $ _ , 0 ] ;
}
if ( $ _ =~ /is alive/ ) {
s/ is alive// ;
push @ pingable_hcp , $ _ ;
}
}
2011-01-26 13:27:25 +00:00
}
if ( ! @ pingable_hcp ) {
2011-02-18 02:10:25 +00:00
push @ return_msg , [ $ node , "No hcp can be pinged." , - 1 ] ;
return \ @ return_msg ;
2011-01-26 13:27:25 +00:00
}
2011-02-18 02:10:25 +00:00
2011-01-26 13:27:25 +00:00
# try the ip of hcp one by one
2011-02-18 02:10:25 +00:00
my @ lastnoerr_msg ;
my @ noerr_msg ;
my @ last_msg ;
2011-01-26 13:27:25 +00:00
foreach my $ hcp ( @ pingable_hcp ) {
2011-02-18 02:10:25 +00:00
push @ noerr_msg , @ lastnoerr_msg ;
@ lastnoerr_msg = ( ) ;
@ last_msg = ( ) ;
2011-01-26 13:27:25 +00:00
# Generate the url path for CIM communication
2011-02-18 02:10:25 +00:00
chomp ( $ hcp ) ;
2011-01-26 13:27:25 +00:00
my $ url_path = "https://" . "$user" . ":" . "$password" . "\@" . "$hcp" . ":5989" ;
2013-01-15 06:11:21 +00:00
# Not display password in command outupt.
my $ display_url_path = "https://" . "$user" . ":" . "xxxxxx" . "\@" . "$hcp" . ":5989" ;
2011-01-26 13:27:25 +00:00
# Execute the request
my $ cmd = "" ;
2013-01-15 06:11:21 +00:00
my $ display_cmd = "" ;
2011-01-26 13:27:25 +00:00
if ( $ opt - > { 'set' } ) {
$ cmd = "$::CIM_CLIENT_PATH $verb_arg -u $url_path -n $cec_name -o $opt->{'set'}" ;
2013-01-15 06:11:21 +00:00
$ display_cmd = "$::CIM_CLIENT_PATH $verb_arg -u $display_url_path -n $cec_name -o $opt->{'set'}" ;
2011-01-26 13:27:25 +00:00
} elsif ( $ opt - > { 'query' } ) {
$ cmd = "$::CIM_CLIENT_PATH $verb_arg -u $url_path -n $cec_name -o $opt->{'query'}" ;
2013-01-15 06:11:21 +00:00
$ display_cmd = "$::CIM_CLIENT_PATH $verb_arg -u $display_url_path -n $cec_name -o $opt->{'query'}" ;
2011-01-26 13:27:25 +00:00
}
if ( $ verbose ) {
2013-01-15 06:11:21 +00:00
push @ noerr_msg , [ $ node , "Run following command: $display_cmd" , 0 ] ;
2011-01-26 13:27:25 +00:00
}
# Disable the CHID signal before run the command. Otherwise the
# $? value of `$cmd` will come from handler of CHID signal
2011-04-14 09:31:23 +00:00
$ SIG { CHLD } = 'DEFAULT' ;
2011-01-26 13:27:25 +00:00
# Call the xCAT_cim_client to query or set the energy capabilities
$ cmd . = " 2>&1" ;
2012-04-24 01:50:26 +00:00
# secure passwd in verbose mode
my $ tmpv = $ ::VERBOSE ;
$ ::VERBOSE = 0 ;
2011-01-27 09:09:18 +00:00
my @ result = xCAT::Utils - > runcmd ( "$cmd" , - 1 ) ;
2012-04-24 01:50:26 +00:00
$ ::VERBOSE = $ tmpv ;
2011-01-26 13:27:25 +00:00
foreach my $ line ( @ result ) {
chomp ( $ line ) ;
if ( $ line =~ /^\s*$/ ) {
next ;
}
2011-02-18 02:10:25 +00:00
push @ lastnoerr_msg , [ $ node , $ line , 0 ] ;
push @ last_msg , [ $ node , $ line , $ ::RUNCMD_RC ] ;
2009-09-21 06:33:27 +00:00
}
2011-01-26 13:27:25 +00:00
if ( ! $ ::RUNCMD_RC ) {
last ;
}
}
2011-02-18 02:10:25 +00:00
# only display the correct msg when getting correct result from one fsp
if ( $ ::RUNCMD_RC || $ verbose ) {
push @ return_msg , @ noerr_msg ;
}
push @ return_msg , @ last_msg ;
2009-09-21 06:33:27 +00:00
return \ @ return_msg ;
}
2011-02-18 02:10:25 +00:00
2009-09-21 06:33:27 +00:00
1 ;