mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-10-26 08:55:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1191 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			1191 lines
		
	
	
		
			42 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| #!/usr/bin/perl
 | |
| # IBM(c) 2014 EPL license http://www.eclipse.org/legal/epl-v10.html
 | |
| 
 | |
| =head3  xCAT_plugin::energy
 | |
| 
 | |
|     This plugin module is used to handle the renergy command for:
 | |
|         FSP based Power 8 machine. 
 | |
|             1. mgt=fsp, mtm=(p8); 2. mgt=ipmi, arch=ppc64le;
 | |
| 
 | |
| =cut
 | |
| 
 | |
| package xCAT_plugin::energy;
 | |
| 
 | |
| BEGIN
 | |
| {
 | |
|     $::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
 | |
| }
 | |
| 
 | |
| use lib "$::XCATROOT/lib/perl";
 | |
| use strict;
 | |
| use warnings "all";
 | |
| use Getopt::Long;
 | |
| use IO::Socket;
 | |
| use Thread qw(yield);
 | |
| use POSIX "WNOHANG";
 | |
| use Storable qw(store_fd fd_retrieve);
 | |
| 
 | |
| use xCAT::Usage;
 | |
| use xCAT::CIMUtils;
 | |
| use xCAT::MsgUtils;
 | |
| use xCAT::Table;
 | |
| use xCAT::FSPUtils;
 | |
| use xCAT::NetworkUtils;
 | |
| 
 | |
| sub handled_commands {
 | |
|     return {
 | |
|         renergy => 'nodehm:mgt=ipmi|fsp',
 | |
|       }
 | |
| }
 | |
| 
 | |
| my $parent_fd;
 | |
| 
 | |
| # The hash includes all valid attribute for quering
 | |
| my %QUERY_ATTRS = (
 | |
|     'savingstatus'       => 1,
 | |
|     'dsavingstatus'      => 1,
 | |
|     'cappingstatus'      => 1,
 | |
|     'cappingmaxmin'      => 1,
 | |
|     'cappingvalue'       => 1,
 | |
|     'cappingsoftmin'     => 1,
 | |
|     'averageAC'          => 1,
 | |
|     'averageAChistory'   => 1,
 | |
|     'averageDC'          => 1,
 | |
|     'averageDChistory'   => 1,
 | |
|     'ambienttemp'        => 1,
 | |
|     'ambienttemphistory' => 1,
 | |
|     'exhausttemp'        => 1,
 | |
|     'exhausttemphistory' => 1,
 | |
|     'CPUspeed'           => 1,
 | |
|     'CPUspeedhistory'    => 1,
 | |
|     'fanspeed'           => 1,
 | |
|     'fanspeedhistory'    => 1,
 | |
|     'syssbpower'         => 1,
 | |
|     'sysIPLtime'         => 1,
 | |
| 
 | |
|     # for FFO, only supported when communicating to fsp directly
 | |
|     'ffoMin'        => 1,
 | |
|     'ffoVmin'       => 1,
 | |
|     'ffoTurbo'      => 1,
 | |
|     'ffoNorm'       => 1,
 | |
|     'fsavingstatus' => 1,
 | |
|     'ffovalue'      => 1,
 | |
| );
 | |
| 
 | |
| # The hash includes all valid attribute for writting
 | |
| my %SET_ATTRS = (
 | |
|     'savingstatus'  => 1,
 | |
|     'dsavingstatus' => 1,
 | |
|     'cappingstatus' => 1,
 | |
|     'cappingwatt'   => 1,
 | |
|     'cappingperc'   => 1,
 | |
| 
 | |
|     # for FFO
 | |
|     'fsavingstatus' => 1,
 | |
|     'ffovalue'      => 1,
 | |
| );
 | |
| 
 | |
| =head3  parse_args
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Parse the arguments from the command line of renergy command
 | |
|     ARGUMENTS:
 | |
|         The request hash from preprocess_request or process_request
 | |
|     RETURN
 | |
|         First element: rc: 0 -success; 1 - fail
 | |
|         Second element: 
 | |
|             1. a string: a message for display; 
 | |
|             2. a reference to a hash: {verbose}, {query_list} and {set_pair}.
 | |
|             
 | |
| =cut
 | |
| 
 | |
| sub parse_args {
 | |
|     my $request = shift;
 | |
| 
 | |
|     my $opt   = ();
 | |
|     my $cmd   = $request->{command}->[0];
 | |
|     my $args  = $request->{arg};
 | |
|     my $nodes = $request->{node};
 | |
| 
 | |
|     my @query_list = ();    # 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 $query_flag = ();    # Indicate there's param in the argv
 | |
| 
 | |
|     # set the usage subroutine
 | |
|     local *usage = sub {
 | |
|         my $add_msg      = shift;
 | |
|         my $usage_string = xCAT::Usage->getUsage($cmd);
 | |
|         if ($add_msg) {
 | |
|             return ("$add_msg\n" . $usage_string);
 | |
|         } else {
 | |
|             return ($usage_string);
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     # handle the arguments
 | |
|     if ($request->{arg}) {
 | |
|         @ARGV = @{ $request->{arg} };
 | |
| 
 | |
|         $Getopt::Long::ignorecase = 0;
 | |
|         Getopt::Long::Configure("bundling");
 | |
| 
 | |
|         if (!GetOptions('V' => \$::VERBOSE,
 | |
|                 'h|help'    => \$::HELP,
 | |
|                 'v|version' => \$::VERSION)) {
 | |
|             return (1, &usage());
 | |
|         }
 | |
|         if ($::HELP && $::VERSION) {
 | |
|             return (1, &usage());
 | |
|         }
 | |
|         if ($::HELP) {
 | |
|             return (0, &usage());
 | |
|         }
 | |
| 
 | |
|         if ($::VERSION) {
 | |
|             my $version_string = xCAT::Usage->getVersion('renergy');
 | |
|             return (0, $version_string);
 | |
|         }
 | |
| 
 | |
|         if ($::VERBOSE) {
 | |
|             $opt->{verbose} = 1;
 | |
|         }
 | |
|     } else {
 | |
|         return (1, &usage());
 | |
|     }
 | |
| 
 | |
|     # if the request has node range
 | |
|     if ($nodes) {
 | |
| 
 | |
|         # the default option is query all attributes
 | |
|         if ($#ARGV < 0) {
 | |
|             $ARGV[0] = "all";
 | |
|         }
 | |
| 
 | |
|         # Check the validity of the parameters of Query and Set
 | |
|         # Forbid to perform both query and set
 | |
|         foreach my $attr (@ARGV) {
 | |
|             my ($set_attr, $set_value) = split(/=/, $attr);
 | |
|             if (defined($set_value)) {
 | |
|                 if ($query_flag) {
 | |
|                     return (1, &usage("Cannot perform both Query and Set."));
 | |
|                 }
 | |
| 
 | |
|                 # make sure the attribute is valid
 | |
|                 if ($SET_ATTRS{$set_attr} != 1) {
 | |
|                     return (1, &usage("Invalid attribute."));
 | |
|                 }
 | |
| 
 | |
|                 if ($set_flag) {
 | |
|                     return (1, &usage("Only supports to perform one setting at invoke."));
 | |
|                 }
 | |
| 
 | |
|                 # make sure the value for attirbute is valid
 | |
|                 if (($set_attr eq "savingstatus" || $set_attr eq "fsavingstatus")
 | |
|                     && ($set_value ne "on" && $set_value ne "off")) {
 | |
|                     return (1, &usage("Incorrect Value"));
 | |
|                 } elsif ($set_attr eq "dsavingstatus"
 | |
|                     && ($set_value ne "off"
 | |
|                         && $set_value ne "on-norm" && $set_value ne "on-maxp")) {
 | |
|                     return (1, &usage("Incorrect Value"));
 | |
|                 } elsif ($set_attr eq "cappingstatus"
 | |
|                     && ($set_value ne "on" && $set_value ne "off")) {
 | |
|                     return (1, &usage("Incorrect Value"));
 | |
|                 } elsif (($set_attr eq "cappingwatt"
 | |
|                         || $set_attr eq "cappingperc" || $set_attr eq "ffovalue")
 | |
|                     && $set_value =~ /\D/) {
 | |
|                     return (1, &usage("Incorrect Value"));
 | |
|                 }
 | |
| 
 | |
|                 push @set_pair, $set_attr . "=" . $set_value;
 | |
|                 $set_flag = 1;
 | |
|             } else {
 | |
|                 if ($set_flag) {
 | |
|                     return (1, &usage("Cannot perform both Query and Set."));
 | |
|                 }
 | |
| 
 | |
|                 # replace the 'all' with all valid attribute
 | |
|                 if ($attr eq "all") {
 | |
|                     foreach my $q_attr (keys %QUERY_ATTRS) {
 | |
| 
 | |
|                         # do not include 'history' related attributes for all keyword
 | |
|                         if ($q_attr =~ /history$/) {
 | |
|                             next;
 | |
|                         }
 | |
|                         if (!grep (/^$q_attr$/, @query_list)) {
 | |
|                             push @query_list, $q_attr;
 | |
|                         }
 | |
|                     }
 | |
|                 } else {
 | |
| 
 | |
|                     # make sure the query attribute is valid
 | |
|                     if ($QUERY_ATTRS{$attr} != 1) {
 | |
|                         return (1, &usage("Invalid attribute."));
 | |
|                     }
 | |
|                     if (!grep (/^$attr$/, @query_list)) {
 | |
|                         push @query_list, $attr;
 | |
|                     }
 | |
|                 }
 | |
|                 $query_flag = 1;
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
| 
 | |
|         # no noderange, do nothing
 | |
|         return (1, &usage());
 | |
|     }
 | |
| 
 | |
|     if (@query_list) {
 | |
|         $opt->{'query_list'} = join(',', @query_list);
 | |
|     } elsif (@set_pair) {
 | |
|         $opt->{'set_pair'} = join(',', @set_pair);
 | |
|     }
 | |
| 
 | |
|     return (0, $opt);
 | |
| }
 | |
| 
 | |
| sub preprocess_request
 | |
| {
 | |
|     my $req      = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     # Exit if the packet has been preprocessed
 | |
|     if (defined($req->{_xcatpreprocessed}->[0]) && $req->{_xcatpreprocessed}->[0] == 1) { return [$req]; }
 | |
| 
 | |
| 
 | |
| 
 | |
|     # This plugin only handle the node which is 1. mgt=fsp, mtm=(p8); 2. mgt=ipmi, arch=ppc64le;
 | |
|     # otherwise, make other plugin to handle it
 | |
|     my (@mpnodes, @fspnodes, @bmcnodes, @nohandle);
 | |
|     xCAT::Utils->filter_nodes($req, \@mpnodes, \@fspnodes, \@bmcnodes, \@nohandle);
 | |
| 
 | |
|     # Find the nodes which are not handled by mp (@mpnodes), fsp (@fspnodes) and bmc (@bmcnods), and not in @nohandle.
 | |
|     # They are the one of p8_fsp nodes which should be handled by this plugin
 | |
|     my %tmphash = map { $_ => 1 } (@mpnodes, @fspnodes, @bmcnodes, @nohandle);
 | |
|     my @nodes = grep { not $tmphash{$_} } @{ $req->{node} };
 | |
| 
 | |
|     # build request array
 | |
|     my @requests;
 | |
|     if (@nodes) {
 | |
|         my ($rc, $args) = parse_args($req);
 | |
|         if ($rc) {
 | |
| 
 | |
|             # error or message display happens
 | |
|             xCAT::MsgUtils->message("E", { error => [$args], errorcode => [$rc] }, $callback);
 | |
|             return [];
 | |
|         } else {
 | |
|             unless (ref($args)) {
 | |
|                 xCAT::MsgUtils->message("I", { data => [$args] }, $callback);
 | |
|                 return [];
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         # do nothing if no query or setting required.
 | |
|         unless (defined($args->{query_list}) || defined($args->{set_pair})) {
 | |
|             return [];
 | |
|         }
 | |
|         my $sn = xCAT::ServiceNodeUtils->get_ServiceNode(\@nodes, 'xcat', 'MN');
 | |
| 
 | |
|         # Build each request for each service node
 | |
|         foreach my $snkey (keys %$sn) {
 | |
|             my $reqcopy = {%$req};
 | |
|             $reqcopy->{node}                   = $sn->{$snkey};
 | |
|             $reqcopy->{'_xcatdest'}            = $snkey;
 | |
|             $reqcopy->{_xcatpreprocessed}->[0] = 1;
 | |
|             if (defined($args->{verbose})) {
 | |
|                 $reqcopy->{verbose} = $args->{verbose};
 | |
|             }
 | |
|             if (defined($args->{query_list})) {
 | |
|                 $reqcopy->{query_list} = $args->{query_list};
 | |
|             }
 | |
|             if (defined($args->{set_pair})) {
 | |
|                 $reqcopy->{set_pair} = $args->{set_pair};
 | |
|             }
 | |
|             push @requests, $reqcopy;
 | |
|         }
 | |
| 
 | |
|         return \@requests;
 | |
|     }
 | |
| 
 | |
|     return [];
 | |
| }
 | |
| 
 | |
| 
 | |
| sub process_request {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
|     my $subreq   = shift;
 | |
| 
 | |
|     my $verbose;
 | |
| 
 | |
|     my $nodes = $request->{node};
 | |
|     my $args  = $request->{arg};
 | |
|     if (defined($request->{verbose})) {
 | |
|         $verbose = $request->{verbose};
 | |
|     }
 | |
| 
 | |
|     # get the password for the nodes
 | |
|     my $user_default     = "admin";
 | |
|     my $password_default = "admin";
 | |
| 
 | |
|     my $ipmi_tab = xCAT::Table->new('ipmi', -create => 0);
 | |
|     my $ipmi_hash;
 | |
|     if ($ipmi_tab) {
 | |
|         $ipmi_hash = $ipmi_tab->getNodesAttribs($request->{node}, [ 'bmc', 'username', 'password' ]);
 | |
|     }
 | |
| 
 | |
|     my $ppc_tab = xCAT::Table->new('ppc', -create => 0);
 | |
|     my $ppc_hash;
 | |
|     my @ppc_all_entry;
 | |
|     my $cec2fsp;
 | |
|     if ($ppc_tab) {
 | |
|         $ppc_hash = $ppc_tab->getNodesAttribs($request->{node}, [ 'hcp', 'nodetype' ]);
 | |
|     }
 | |
| 
 | |
|     my $nodehm_tab = xCAT::Table->new('nodehm', -create => 0);
 | |
|     my $nodehm_hash;
 | |
|     if ($nodehm_tab) {
 | |
|         $nodehm_hash = $nodehm_tab->getNodesAttribs($request->{node}, ['mgt']);
 | |
|     }
 | |
| 
 | |
|     # get the 'fsp' key from the passwd table
 | |
|     my $passwd_tab = xCAT::Table->new('passwd', -create => 0);
 | |
|     my $passwd_hash;
 | |
|     if ($passwd_tab) {
 | |
|         $passwd_hash = $passwd_tab->getAttribs({ key => 'fsp' }, qw(username password));
 | |
|     }
 | |
| 
 | |
|     my $ppcdirect_tab = xCAT::Table->new('ppcdirect', -create => 0);
 | |
| 
 | |
|     my $children;       # The number of child process
 | |
|     my %sp_children;    # Record the pid of child process
 | |
|     my $sub_fds = new IO::Select;  # Record the parent fd for each child process
 | |
| 
 | |
|     # Set the signal handler for ^c
 | |
|     $SIG{TERM} = $SIG{INT} = sub {
 | |
|         foreach (keys %sp_children) {
 | |
|             kill 2, $_;
 | |
|         }
 | |
|         $SIG{ALRM} = sub {
 | |
|             while (wait() > 0) {
 | |
|                 yield;
 | |
|             }
 | |
|             exit @_;
 | |
|         };
 | |
|         alarm(1);    # wait 1s for grace exit
 | |
|     };
 | |
| 
 | |
|     # Set the singal handler for child process finished it's work
 | |
|     $SIG{CHLD} = sub {
 | |
|         my $cpid;
 | |
|         while (($cpid = waitpid(-1, WNOHANG)) > 0) {
 | |
|             if ($sp_children{$cpid}) {
 | |
|                 delete $sp_children{$cpid};
 | |
|                 $children--;
 | |
|             }
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     # Do run each node
 | |
|     foreach my $node (@{ $request->{node} }) {
 | |
|         my $user     = $user_default;
 | |
|         my $password = $password_default;
 | |
|         my $hcp_ip;
 | |
| 
 | |
|         if (defined($nodehm_hash->{$node}->[0]->{mgt})) {
 | |
|             my $mgt = $nodehm_hash->{$node}->[0]->{mgt};
 | |
| 
 | |
|             if ($mgt eq 'fsp') {
 | |
| 
 | |
|                 # This is Power node which is running in PowerVM mode
 | |
|                 unless (@ppc_all_entry) {
 | |
|                     @ppc_all_entry = $ppc_tab->getAllNodeAttribs([ 'node', 'parent', 'hcp', 'nodetype' ]);
 | |
|                     foreach my $ppcentry (@ppc_all_entry) {
 | |
|                         if (defined($ppcentry->{parent}) && defined($ppcentry->{nodetype}) && $ppcentry->{nodetype} =~ /fsp/) {
 | |
|                             $cec2fsp->{ $ppcentry->{parent} } .= "$ppcentry->{node},";
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 $hcp_ip = $cec2fsp->{$node};
 | |
| 
 | |
|                 # Get the user/password for the node
 | |
|                 if ($ppcdirect_tab) {
 | |
|                     my $ppcdirect_hash = $ppcdirect_tab->getAttribs({ hcp => $node, username => $user }, 'password');
 | |
|                     if ($ppcdirect_hash) {
 | |
|                         $password = $ppcdirect_hash->{'password'};
 | |
|                     }
 | |
|                 }
 | |
|             } elsif ($mgt eq 'ipmi') {
 | |
|                 if (defined($ipmi_hash->{$node}->[0]->{bmc})) {
 | |
| 
 | |
|                     # This is a ipmi managed node. (should be a ppcle)
 | |
|                     $hcp_ip = $ipmi_hash->{$node}->[0]->{bmc};
 | |
| 
 | |
|                     # Get the passwd from passwd table for 'fsp' first, if not the default will be used
 | |
|                     if (defined($passwd_hash->{username}) && defined($passwd_hash->{password})) {
 | |
|                         $user     = $passwd_hash->{username};
 | |
|                         $password = $passwd_hash->{password};
 | |
|                     }
 | |
|                 } else {
 | |
|                     xCAT::MsgUtils->message("E", { data => ["$node: Missed attribute [bmc]."] }, $callback);
 | |
|                     return 1;
 | |
|                 }
 | |
|             } else {
 | |
|                 xCAT::MsgUtils->message("E", { data => ["$node: Support the valid mgt [fsp, ipmi]."] }, $callback);
 | |
|                 return 1;
 | |
|             }
 | |
|         } else {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: Missed important attributes [mgt] to know how to handle this node."] }, $callback);
 | |
|             return 1;
 | |
|         }
 | |
|         unless ($hcp_ip) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: Cannot find HCP"] }, $callback);
 | |
|             return 1;
 | |
|         }
 | |
| 
 | |
|         # fork a sub process to handle the communication with service processor
 | |
|         $children++;
 | |
|         my $cfd;
 | |
| 
 | |
|         # the $parent_fd will be used by &send_rep() to send response from child process to parent process
 | |
|         socketpair($parent_fd, $cfd, AF_UNIX, SOCK_STREAM, PF_UNSPEC) or die "socketpair: $!";
 | |
|         $cfd->autoflush(1);
 | |
|         $parent_fd->autoflush(1);
 | |
| 
 | |
|         my $child = xCAT::Utils->xfork;
 | |
|         if ($child == 0) {
 | |
|             close($cfd);
 | |
|             $0        = $0 . " for node [$node]";
 | |
|             $callback = \&send_rep;
 | |
|             foreach my $ip (split(',', $hcp_ip)) {
 | |
|                 unless ($ip) { next; }
 | |
|                 my $real_ip = xCAT::NetworkUtils->getipaddr($ip);
 | |
|                 unless ($real_ip) {
 | |
|                     xCAT::MsgUtils->message("E", { error => ["$node: Cannot get ip for $ip"], errorcode => [1] }, $callback);
 | |
|                     next;
 | |
|                 }
 | |
|                 my %args = (
 | |
|                     node     => $node,
 | |
|                     ip       => $real_ip,
 | |
|                     port     => '5989',
 | |
|                     method   => 'POST',
 | |
|                     user     => $user,
 | |
|                     password => $password);
 | |
| 
 | |
|                 if ($verbose) {
 | |
|                     $args{verbose}  = 1;
 | |
|                     $args{callback} = $callback;
 | |
|                     xCAT::MsgUtils->message("I", { data => ["$node: Access hcp [$ip], user [$user], passowrd [$password]"] }, $callback);
 | |
|                 }
 | |
| 
 | |
|                 # call the cim utils to connect to cim server
 | |
|                 my $ret = run_cim($request, $callback, \%args);
 | |
| 
 | |
|                 # 0 - success; 1 - cim error; 10 - ip is not pingable; 11 - this ip is a standby fsp
 | |
|                 unless ($ret == 10 || $ret == 11) {
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|             exit(0);
 | |
|         } else {
 | |
| 
 | |
|             # in the main process, record the created child process and add parent fd for the child process to an IO:Select object
 | |
|             # the main process will check all the parent fd and receive response
 | |
|             $sp_children{$child} = 1;
 | |
|             close($parent_fd);
 | |
|             $sub_fds->add($cfd);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # receive data from child processes
 | |
|     while ($sub_fds->count > 0 or $children > 0) {
 | |
|         forward_data($callback, $sub_fds);
 | |
|     }
 | |
|     while (forward_data($callback, $sub_fds)) { }
 | |
| }
 | |
| 
 | |
| =head3 send_rep 
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Send date from forked child process to parent process.
 | |
|         This subroutine will be replace the original $callback in the forked child process
 | |
| 
 | |
|     ARGUMENTS:
 | |
|         $resp - The response which generated in xCAT::Utils->message();
 | |
| 
 | |
| =cut
 | |
| 
 | |
| sub send_rep {
 | |
|     my $resp = shift;
 | |
| 
 | |
|     unless ($resp) { return; }
 | |
|     store_fd($resp, $parent_fd);
 | |
| }
 | |
| 
 | |
| =head3 forward_data 
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Receive data from forked child process and call the original $callback to forward data to xcat client
 | |
| 
 | |
| =cut
 | |
| 
 | |
| sub forward_data {
 | |
|     my $callback  = shift;
 | |
|     my $fds       = shift;
 | |
|     my @ready_fds = $fds->can_read(1);
 | |
|     my $rfh;
 | |
|     my $rc = @ready_fds;
 | |
|     foreach $rfh (@ready_fds) {
 | |
|         my $data;
 | |
|         my $responses;
 | |
|         eval {
 | |
|             $responses = fd_retrieve($rfh);
 | |
|         };
 | |
|         if ($@ and $@ =~ /^Magic number checking on storable file/) { #this most likely means we ran over the end of available input
 | |
|             $fds->remove($rfh);
 | |
|             close($rfh);
 | |
|         } else {
 | |
|             eval { print $rfh "ACK\n"; }; #Ignore ack loss due to child giving up and exiting, we don't actually explicitly care about the acks
 | |
|             $callback->($responses);
 | |
|         }
 | |
|     }
 | |
|     yield;    #Try to avoid useless iterations as much as possible
 | |
|     return $rc;
 | |
| }
 | |
| 
 | |
| 
 | |
| =head3 query_pum 
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Query the attribute for instance of FipS_PUMService class
 | |
| 
 | |
|     ARGUMENTS:
 | |
|         $http_params - refer to the HTTP_PARAMS in xCAT::CIMUtils.pm
 | |
| 
 | |
|     RETURN
 | |
|         $ret - return code and messages
 | |
|         $pum - a hash includes all the attributes
 | |
|         $namepath - the name path of pum instance
 | |
| 
 | |
| =cut
 | |
| 
 | |
| sub query_pum
 | |
| {
 | |
|     my $http_params = shift;
 | |
| 
 | |
|     my %cimargs = (classname => 'FipS_PUMService');
 | |
|     my ($ret, $value, $namepath) = xCAT::CIMUtils->enum_instance($http_params, \%cimargs);
 | |
|     if ($ret->{rc}) {
 | |
|         return ($ret);
 | |
|     }
 | |
| 
 | |
|     # parse the return xml to get all the property of pum instance
 | |
|     my $pum;
 | |
|     if ($value && @$value) {
 | |
|         my $instance = $$value[0];
 | |
|         foreach my $pname (keys %{ $instance->{property} }) {
 | |
|             $pum->{$pname} = $instance->{property}->{$pname}->{value};
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return ($ret, $pum, $namepath);
 | |
| }
 | |
| 
 | |
| =head3 query_cec_drawer 
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Query the attribute for instance of FipS_CECDrawer class
 | |
| 
 | |
|     ARGUMENTS:
 | |
|         $http_params - refer to the HTTP_PARAMS in xCAT::CIMUtils.pm
 | |
| 
 | |
|     RETURN
 | |
|         $ret - return code and messages
 | |
|         $cec_drawer - a hash includes all the attributes
 | |
| 
 | |
| =cut
 | |
| 
 | |
| sub query_cec_drawer
 | |
| {
 | |
|     my $http_params = shift;
 | |
| 
 | |
|     my %cimargs = (classname => 'FipS_CECDrawer');
 | |
|     my ($ret, $value, $namepath) = xCAT::CIMUtils->enum_instance($http_params, \%cimargs);
 | |
|     if ($ret->{rc}) {
 | |
|         return ($ret);
 | |
|     }
 | |
| 
 | |
|     # parse the return xml to get all the property of pum instance
 | |
|     my $cec_drawer;
 | |
|     if ($value && @$value) {
 | |
|         my $instance = $$value[0];
 | |
|         foreach my $pname (keys %{ $instance->{property} }) {
 | |
|             $cec_drawer->{$pname} = $instance->{property}->{$pname}->{value};
 | |
|         }
 | |
|     } else {
 | |
|         return ({ rc => 1, msg => "Cannot find instance for FipS_CECDrawer class" });
 | |
|     }
 | |
| 
 | |
|     return ($ret, $cec_drawer, $namepath);
 | |
| }
 | |
| 
 | |
| =head3 query_metric
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Query the attribute for instance of FipS_*metricValue class
 | |
| 
 | |
|     ARGUMENTS:
 | |
|         $http_params - refer to the HTTP_PARAMS in xCAT::CIMUtils.pm
 | |
|         $option - the specified operation
 | |
| 
 | |
|     RETURN
 | |
|         $ret - return code and messages
 | |
|         $array - a hash includes all the attributes
 | |
| 
 | |
| =cut
 | |
| 
 | |
| sub query_metric
 | |
| {
 | |
|     my $http_params     = shift;
 | |
|     my $classname       = shift;
 | |
|     my $matching_string = shift;
 | |
|     my $value_unit      = shift;
 | |
|     if (!defined($value_unit)) {
 | |
|         $value_unit = 1;
 | |
|     }
 | |
|     my %cimargs = (classname => "$classname");
 | |
|     my ($ret, $value) = xCAT::CIMUtils->enum_instance($http_params, \%cimargs);
 | |
|     if ($ret->{rc}) {
 | |
|         return $ret;
 | |
|     }
 | |
|     my ($matching_key, $matching_value) = split /:/, $matching_string;
 | |
|     my %instances_hash = ();
 | |
|     foreach my $instance (@$value) {
 | |
|         my $instance_element = undef;
 | |
|         my $timestamp        = undef;
 | |
|         if (defined($instance->{property}->{$matching_key}) and $instance->{property}->{$matching_key}->{value} !~ /$matching_value/) {
 | |
|             next;
 | |
|         }
 | |
|         if (defined($instance->{property}->{InstanceID})) {
 | |
|             $instance_element = $instance->{property}->{InstanceID}->{value};
 | |
|             $instance_element =~ s/ .*$//;
 | |
|         }
 | |
|         if (!defined($instance_element)) {
 | |
|             next;
 | |
|         }
 | |
| 
 | |
|         if (defined($instance->{property}->{MeasuredElementName})) {
 | |
|             $instances_hash{$instance_element}->{MeasuredElementName} = $instance->{property}->{MeasuredElementName}->{value};
 | |
|         } else {
 | |
|             next;
 | |
|         }
 | |
| 
 | |
|         if (defined($instance->{property}->{TimeStamp})) {
 | |
|             $timestamp = $instance->{property}->{TimeStamp}->{value};
 | |
|             $timestamp =~ s/\..*$//;
 | |
|         }
 | |
|         if (defined($instance->{property}->{MetricValue})) {
 | |
|             if (defined($timestamp)) {
 | |
|                 $instances_hash{$instance_element}->{MetricValue}->{$timestamp} = $instance->{property}->{MetricValue}->{value} / $value_unit;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return ($ret, \%instances_hash);
 | |
| }
 | |
| 
 | |
| =head3 query_tmp
 | |
|     DESCRIPTION:
 | |
|         Require the input and output temperature
 | |
| =cut 
 | |
| 
 | |
| sub query_tmp
 | |
| {
 | |
|     &query_metric(@_, "FipS_ThermalMetricValue", "InstanceID:InletAirTemp|ExhaustAirTemp", 100);
 | |
| }
 | |
| 
 | |
| =head3 query_cpuspeed
 | |
|     DESCRIPTION:
 | |
|         Require the cpuspeed history
 | |
| =cut 
 | |
| 
 | |
| sub query_cpuspeed
 | |
| {
 | |
|     &query_metric(@_, "FipS_CPUUsageMetricValue", "InstanceID:AvgCPUUsage");
 | |
| }
 | |
| 
 | |
| =head3 query_fanspeed
 | |
|     DESCRIPTION:
 | |
|         Require the fanspeed history
 | |
| =cut 
 | |
| 
 | |
| sub query_fanspeed
 | |
| {
 | |
|     &query_metric(@_, "FipS_FanSpeedMetricValue", "InstanceID:FansSpeed");
 | |
| }
 | |
| 
 | |
| =head3 query_powermetric
 | |
|     DESCRIPTION:
 | |
|         Require the AC and DC power comsume history
 | |
| =cut 
 | |
| 
 | |
| sub query_powermetric
 | |
| {
 | |
|     my $http_params = shift;
 | |
|     $http_params->{timeout} = 500;
 | |
|     my ($ret, $return_hash) = &query_metric($http_params, "FipS_PowerMetricValue", "InstanceID:AvgInputPwr");
 | |
|     if ($ret->{rc}) {
 | |
|         return $ret;
 | |
|     }
 | |
|     my %instances = ();
 | |
|     foreach my $ins (keys %$return_hash) {
 | |
|         if ($return_hash->{$ins}->{MeasuredElementName} =~ /Power Supply/) {
 | |
|             foreach my $timestamp (keys %{ $return_hash->{$ins}->{MetricValue} }) {
 | |
|                 if (!exists($instances{"averageAC"}->{MetricValue}->{$timestamp})) {
 | |
|                     $instances{"averageAC"}->{MetricValue}->{$timestamp} = $return_hash->{$ins}->{MetricValue}->{$timestamp};
 | |
|                 } else {
 | |
|                     $instances{"averageAC"}->{MetricValue}->{$timestamp} += $return_hash->{$ins}->{MetricValue}->{$timestamp};
 | |
|                 }
 | |
|             }
 | |
|         } else {
 | |
|             $instances{"averageDC"}->{MetricValue} = $return_hash->{$ins}->{MetricValue};
 | |
|         }
 | |
|     }
 | |
|     return ($ret, \%instances);
 | |
| }
 | |
| 
 | |
| 
 | |
| 
 | |
| =head3  run_cim
 | |
| 
 | |
|     DESCRIPTION:
 | |
|         Handle the Query and Setting of Energy via CIM
 | |
|         
 | |
|     ARGUMENTS:
 | |
|         $request
 | |
|         $callback
 | |
|         $http_params - refer to the HTTP_PARAMS in xCAT::CIMUtils.pm
 | |
|         
 | |
|     RETURN
 | |
|         First element: rc: 0 -success; 1 - cim error; 10 - ip is not pingable; 11 - this ip is a standby fsp
 | |
|         
 | |
| =cut
 | |
| 
 | |
| sub run_cim
 | |
| {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $http_params = shift;
 | |
|     my $node        = $http_params->{node};
 | |
| 
 | |
|     my $output;
 | |
|     my $verbose;
 | |
|     my $query_list;
 | |
|     my $set_pair;
 | |
| 
 | |
|     if (defined($request->{verbose})) {
 | |
|         $verbose = $request->{verbose};
 | |
|     }
 | |
|     if (defined($request->{query_list})) {
 | |
|         $query_list = $request->{query_list};
 | |
|     }
 | |
|     if (defined($request->{set_pair})) {
 | |
|         $set_pair = $request->{set_pair};
 | |
|     }
 | |
| 
 | |
|     # Try to connect CIM Server to Enumerate CEC object;
 | |
|     # If cannot access this ip, return 10 for caller to connect to the next ip
 | |
|     my $cimargs = {
 | |
|         classname => 'fips_cec',
 | |
|     };
 | |
|     $http_params->{timeout} = 5;
 | |
|     my ($ret, $value) = xCAT::CIMUtils->enum_instance($http_params, $cimargs);
 | |
|     if ($ret->{rc}) {
 | |
|         if ($ret->{msg} =~ /(Couldn't connect to server)|(Can't connect to)/) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: Couldn't connect to server [$http_params->{ip}]."] }, $callback);
 | |
|             return 10;
 | |
|         } else {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return 1;
 | |
|         }
 | |
|     }
 | |
|     delete $http_params->{timeout};
 | |
| 
 | |
|     # ======start to handle the query and setting======
 | |
| 
 | |
|     # Pre-query some specific instances since some instances are common for multiple energy attributes
 | |
|     #my $query_pum_capabilities_flag;    # set to query the instance for [FipS_PUMServiceCapabilities]
 | |
|     my $query_pum_flag;     # set to query the instance for [FipS_PUMService]
 | |
|     my $query_pum_value;    # the pre-queried PUM instance
 | |
|     my $namepath_pum;       # the name path of PUM instance
 | |
| 
 | |
|     my $query_drawer_flag;  # set to query the instance for [FipS_CECDrawer]
 | |
|     my $query_drawer_value; # the pre-queried cec drawer instance
 | |
| 
 | |
|     my %query_return_hash = (); # the hash store the returned hashes for query functions
 | |
|     my $query_tmp_value;        # the requrest for FipS_ThermalMetricValue class
 | |
|     my $query_cpuspeed_value;   # the request for FipS_CPUUsageMetricValue class
 | |
|     my $query_fanspeed_value;   # the request for FipS_FanSpeedMetricValue class
 | |
|     my $query_powermetric_value;   # the request for FipS_PowerMetricValue class
 | |
|     if ($query_list) {
 | |
|         foreach my $attr (split(',', $query_list)) {
 | |
|             if ($attr =~ /^(savingstatus|dsavingstatus|fsavingstatus|ffoMin|ffoVmin|ffoTurbo|ffoNorm|ffovalue)$/) {
 | |
|                 $query_pum_flag = 1;
 | |
|             } elsif ($attr =~ /^(syssbpower|sysIPLtime)$/) {
 | |
|                 $query_drawer_flag = 1;
 | |
|             } elsif ($attr =~ /^(ambienttemp|exhausttemp)/) {
 | |
|                 $query_tmp_value = 1;
 | |
|             } elsif ($attr =~ /^CPUspeed/) {
 | |
|                 $query_cpuspeed_value = 1;
 | |
|             } elsif ($attr =~ /^fanspeed/) {
 | |
|                 $query_fanspeed_value = 1;
 | |
|             } elsif ($attr =~ /^(averageAC|averageDC)/) {
 | |
|                 $query_powermetric_value = 1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if ($set_pair) {
 | |
|         my ($set_name, $set_value) = split('=', $set_pair);
 | |
|         if ($set_name =~ /^(savingstatus|dsavingstatus|fsavingstatus|ffovalue)$/) {
 | |
|             $query_pum_flag = 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # query the pre required instances
 | |
|     if ($query_pum_flag) {
 | |
|         ($ret, $query_pum_value, $namepath_pum) = query_pum($http_params);
 | |
|         if ($ret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($ret->{rc});
 | |
|         }
 | |
|     }
 | |
|     if ($query_drawer_flag) {
 | |
|         ($ret, $query_drawer_value) = query_cec_drawer($http_params);
 | |
|         if ($ret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($ret->{rc});
 | |
|         }
 | |
|     }
 | |
|     if ($query_powermetric_value) {
 | |
|         my ($tmpret, $tmpvalue) = query_powermetric($http_params);
 | |
|         if ($tmpret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($tmpret->{rc});
 | |
|         }
 | |
|         $query_return_hash{query_powermetric} = $tmpvalue;
 | |
|     }
 | |
|     if ($query_fanspeed_value) {
 | |
|         $http_params->{timeout} = 200;
 | |
|         my ($tmpret, $tmpvalue) = query_fanspeed($http_params);
 | |
|         if ($tmpret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($tmpret->{rc});
 | |
|         }
 | |
|         $query_return_hash{query_fanspeed} = $tmpvalue;
 | |
|     }
 | |
|     if ($query_cpuspeed_value) {
 | |
|         my ($tmpret, $tmpvalue) = query_cpuspeed($http_params);
 | |
|         if ($tmpret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($tmpret->{rc});
 | |
|         }
 | |
|         $query_return_hash{query_cpuspeed} = $tmpvalue;
 | |
|     }
 | |
|     if ($query_tmp_value) {
 | |
|         $http_params->{timeout} = 200;
 | |
|         my ($tmpret, $tmpvalue) = query_tmp($http_params);
 | |
|         if ($tmpret->{rc}) {
 | |
|             xCAT::MsgUtils->message("E", { data => ["$node: $ret->{msg}"] }, $callback);
 | |
|             return ($tmpret->{rc});
 | |
|         }
 | |
|         $query_return_hash{query_tmp} = $tmpvalue;
 | |
|     }
 | |
| 
 | |
|     # perform the query request
 | |
|     if ($query_list) {
 | |
|         foreach my $attr (split(',', $query_list)) {
 | |
| 
 | |
|             # Query the power saving status
 | |
|             if ($attr =~ /^(savingstatus|dsavingstatus|fsavingstatus)$/) {
 | |
|                 if ($query_pum_flag) {
 | |
|                     if (defined($query_pum_value->{PowerUtilizationMode})) {
 | |
| 
 | |
|                         # 2 = None; 3 = Dynamic; 4 = Static; 32768 = Dynamic Favor Performance; 32769 = FFO
 | |
|                         if ($query_pum_value->{PowerUtilizationMode} eq "2") {
 | |
|                             push @{ $output->{$node} }, "$attr: off";
 | |
|                         } elsif ($query_pum_value->{PowerUtilizationMode} eq "3") {
 | |
|                             if ($attr eq "dsavingstatus") {
 | |
|                                 push @{ $output->{$node} }, "$attr: on-norm";
 | |
|                             } else {
 | |
|                                 push @{ $output->{$node} }, "$attr: off";
 | |
|                             }
 | |
|                         } elsif ($query_pum_value->{PowerUtilizationMode} eq "4") {
 | |
|                             if ($attr eq "savingstatus") {
 | |
|                                 push @{ $output->{$node} }, "$attr: on";
 | |
|                             } else {
 | |
|                                 push @{ $output->{$node} }, "$attr: off";
 | |
|                             }
 | |
|                         } elsif ($query_pum_value->{PowerUtilizationMode} eq "32768") {
 | |
|                             if ($attr eq "dsavingstatus") {
 | |
|                                 push @{ $output->{$node} }, "$attr: on-maxp";
 | |
|                             } else {
 | |
|                                 push @{ $output->{$node} }, "$attr: off";
 | |
|                             }
 | |
|                         } elsif ($query_pum_value->{PowerUtilizationMode} eq "32769") {
 | |
|                             if ($attr eq "fsavingstatus") {
 | |
|                                 push @{ $output->{$node} }, "$attr: on";
 | |
|                             } else {
 | |
|                                 push @{ $output->{$node} }, "$attr: off";
 | |
|                             }
 | |
|                         }
 | |
|                     } else {
 | |
|                         push @{ $output->{$node} }, "$attr: na";
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "$attr: na";
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             # Query the FFO settings
 | |
|             if ($attr =~ /^(ffoMin|ffoVmin|ffoTurbo|ffoNorm|ffovalue)$/) {
 | |
|                 if ($query_pum_flag) {
 | |
|                     if (defined($query_pum_value->{FixedFrequencyPoints}) && defined($query_pum_value->{FixedFrequencyPointValues})) {
 | |
|                         my @ffo_point = split(',', $query_pum_value->{FixedFrequencyPoints});
 | |
|                         my @ffo_point_value = split(',', $query_pum_value->{FixedFrequencyPointValues});
 | |
|                         foreach my $index (0 .. $#ffo_point) {
 | |
|                             if ($ffo_point[$index] eq '2' && $attr eq 'ffoNorm') { # Norminal
 | |
|                                 push @{ $output->{$node} }, "$attr: $ffo_point_value[$index] MHZ";
 | |
|                             } elsif ($ffo_point[$index] eq '3' && $attr eq 'ffoTurbo') { # Turbo
 | |
|                                 push @{ $output->{$node} }, "$attr: $ffo_point_value[$index] MHZ";
 | |
|                             } elsif ($ffo_point[$index] eq '4' && $attr eq 'ffoVmin') { # Vmin
 | |
|                                 push @{ $output->{$node} }, "$attr: $ffo_point_value[$index] MHZ";
 | |
|                             } elsif ($ffo_point[$index] eq '5' && $attr eq 'ffoMin') { # Min
 | |
|                                 push @{ $output->{$node} }, "$attr: $ffo_point_value[$index] MHZ";
 | |
|                             }
 | |
|                         }
 | |
|                     } else {
 | |
|                         push @{ $output->{$node} }, "$attr: na";
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "$attr: na";
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             # Query the FFO Value
 | |
|             if ($attr eq 'ffovalue') {
 | |
|                 if ($query_pum_flag) {
 | |
|                     if (defined($query_pum_value->{FixedFrequencyOverrideFreq})) {
 | |
|                         if ($query_pum_value->{FixedFrequencyOverrideFreq} eq '4294967295') {
 | |
|                             push @{ $output->{$node} }, "$attr: 0 MHZ";
 | |
|                         } else {
 | |
|                             push @{ $output->{$node} }, "$attr: $query_pum_value->{FixedFrequencyOverrideFreq} MHZ";
 | |
|                         }
 | |
|                     } else {
 | |
|                         push @{ $output->{$node} }, "$attr: na";
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "$attr: na";
 | |
|                 }
 | |
| 
 | |
|             }
 | |
| 
 | |
|             # Query the attribute sysIPLtime and syssbpower
 | |
|             if ($attr =~ /^(syssbpower|sysIPLtime)$/) {
 | |
|                 if ($query_drawer_flag) {
 | |
|                     if (defined($query_drawer_value->{AverageTimeToIPL}) && $attr eq "sysIPLtime") {
 | |
|                         push @{ $output->{$node} }, "$attr: $query_drawer_value->{AverageTimeToIPL} S";
 | |
|                     } elsif (defined($query_drawer_value->{StandbyPowerUtilization}) && $attr eq "syssbpower") {
 | |
|                         push @{ $output->{$node} }, "$attr: $query_drawer_value->{StandbyPowerUtilization} W";
 | |
|                     } else {
 | |
|                         push @{ $output->{$node} }, "$attr: na";
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "$attr: na";
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if ($attr =~ /^ambienttemp/) {
 | |
|                 my $tmphash = $query_return_hash{query_tmp};
 | |
|                 foreach my $ins (keys %$tmphash) {
 | |
|                     if ($ins =~ /InletAirTemp/) {
 | |
|                         my @times = sort keys %{ $tmphash->{$ins}->{MetricValue} };
 | |
|                         if ($attr eq "ambienttemp") {
 | |
| 
 | |
|                             #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$times[-1]}";
 | |
|                             push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$times[-1]} C";
 | |
|                         } else {
 | |
|                             foreach my $time (@times) {
 | |
| 
 | |
|                                 #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$time}: $time";
 | |
|                                 push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$time} C: $time";
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             } elsif ($attr =~ /^exhausttemp/) {
 | |
|                 my $tmphash = $query_return_hash{query_tmp};
 | |
|                 foreach my $ins (keys %$tmphash) {
 | |
|                     if ($ins =~ /ExhaustAirTemp/) {
 | |
|                         my @times = sort keys %{ $tmphash->{$ins}->{MetricValue} };
 | |
|                         if ($attr eq "exhausttemp") {
 | |
| 
 | |
|                             #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$times[-1]}";
 | |
|                             push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$times[-1]} C";
 | |
|                         } else {
 | |
|                             foreach my $time (@times) {
 | |
| 
 | |
|                                 #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$time}: $time";
 | |
|                                 push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$time} C: $time";
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             } elsif ($attr =~ /^CPUspeed/) {
 | |
|                 my $tmphash = $query_return_hash{query_cpuspeed};
 | |
|                 foreach my $ins (keys %$tmphash) {
 | |
|                     if ($ins =~ /AvgCPUUsage/) {
 | |
|                         my @times = sort keys %{ $tmphash->{$ins}->{MetricValue} };
 | |
|                         if ($attr eq "CPUspeed") {
 | |
| 
 | |
|                             #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$times[-1]}";
 | |
|                             push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$times[-1]} MHZ";
 | |
|                         } else {
 | |
|                             foreach my $time (@times) {
 | |
| 
 | |
|                                 #push @{$output->{$node}}, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$time}: $time";
 | |
|                                 push @{ $output->{$node} }, "$attr: $tmphash->{$ins}->{MetricValue}->{$time} MHZ: $time";
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             } elsif ($attr =~ /^fanspeed/) {
 | |
|                 my $tmphash = $query_return_hash{query_fanspeed};
 | |
|                 foreach my $ins (keys %$tmphash) {
 | |
|                     if ($ins =~ /FansSpeed/) {
 | |
|                         my @times = sort keys %{ $tmphash->{$ins}->{MetricValue} };
 | |
|                         if ($attr eq "fanspeed") {
 | |
|                             push @{ $output->{$node} }, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$times[-1]} RPM";
 | |
|                         } else {
 | |
|                             foreach my $time (@times) {
 | |
|                                 push @{ $output->{$node} }, "$attr ($tmphash->{$ins}->{MeasuredElementName}): $tmphash->{$ins}->{MetricValue}->{$time} RPM: $time";
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 $query_fanspeed_value = 1;
 | |
|             } elsif ($attr =~ /^(averageAC|averageDC)/) {
 | |
|                 my $tmpattr = $1;
 | |
|                 my $tmphash = $query_return_hash{query_powermetric};
 | |
|                 my @times = sort keys %{ $tmphash->{$tmpattr}->{MetricValue} };
 | |
|                 if ($attr =~ /history$/) {
 | |
|                     foreach my $time (@times) {
 | |
|                         push @{ $output->{$node} }, "$attr: $tmphash->{$tmpattr}->{MetricValue}->{$time} W: $time";
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "$attr: $tmphash->{$attr}->{MetricValue}->{$times[-1]} W";
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # Perform the setting request
 | |
|     if ($set_pair) {
 | |
|         my ($set_name, $set_value) = split('=', $set_pair);
 | |
|         if ($set_name =~ /^(savingstatus|dsavingstatus|fsavingstatus|ffovalue)$/) {
 | |
|             if ($query_pum_flag) {
 | |
|                 if (defined($query_pum_value->{PowerUtilizationMode})) {
 | |
| 
 | |
|                     # set the power saving value
 | |
|                     my $ps_value;
 | |
|                     if ($set_name eq "savingstatus") {
 | |
|                         if ($set_value eq 'on') {
 | |
|                             $ps_value = '4';
 | |
|                         } elsif ($set_value eq 'off') {
 | |
|                             $ps_value = '2';
 | |
|                         }
 | |
|                     } elsif ($set_name eq "dsavingstatus") {
 | |
|                         if ($set_value eq 'on-norm') {
 | |
|                             $ps_value = '3';
 | |
|                         } elsif ($set_value eq 'on-maxp') {
 | |
|                             $ps_value = '32768';
 | |
|                         } elsif ($set_value eq 'off') {
 | |
|                             $ps_value = '2';
 | |
|                         }
 | |
|                     } elsif ($set_name eq "fsavingstatus") {
 | |
|                         if ($set_value eq 'on') {
 | |
|                             $ps_value = '32769';
 | |
|                         } elsif ($set_value eq 'off') {
 | |
|                             $ps_value = '2';
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if ($set_name eq "ffovalue") {
 | |
| 
 | |
|                         # set ffo value
 | |
|                         $cimargs = {
 | |
|                             propertyname  => 'FixedFrequencyOverrideFreq',
 | |
|                             propertyvalue => $set_value,
 | |
|                             namepath      => $namepath_pum,
 | |
|                         };
 | |
|                         $ret = xCAT::CIMUtils->set_property($http_params, $cimargs);
 | |
|                         if ($ret->{rc}) {
 | |
|                             push @{ $output->{$node} }, "Set $set_name failed. [$ret->{msg}]";
 | |
|                         } else {
 | |
|                             push @{ $output->{$node} }, "Set $set_name succeeded";
 | |
|                         }
 | |
|                     } else {
 | |
| 
 | |
|                         # set the power saving
 | |
|                         if ($ps_value eq $query_pum_value->{PowerUtilizationMode}) { # do nothing if it already was the correct status
 | |
|                             push @{ $output->{$node} }, "Set $set_name succeeded";
 | |
|                         } else {
 | |
| 
 | |
|                             # perform the setting
 | |
|                             $cimargs = {
 | |
|                                 propertyname  => 'PowerUtilizationMode',
 | |
|                                 propertyvalue => $ps_value,
 | |
|                                 namepath      => $namepath_pum,
 | |
|                             };
 | |
|                             $ret = xCAT::CIMUtils->set_property($http_params, $cimargs);
 | |
|                             if ($ret->{rc}) {
 | |
|                                 push @{ $output->{$node} }, "Set $set_name failed. [$ret->{msg}]";
 | |
|                             } else {
 | |
|                                 push @{ $output->{$node} }, "Set $set_name succeeded";
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 } else {
 | |
|                     push @{ $output->{$node} }, "Set $set_name failed";
 | |
|                 }
 | |
|             } else {
 | |
|                 push @{ $output->{$node} }, "Set $set_name failed";
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     # Display the output
 | |
|     foreach my $node (keys(%{$output})) {
 | |
|         my @newoutput;
 | |
|         if ($query_list && $query_list !~ /history/) {
 | |
|             @newoutput = sort (@{ $output->{$node} });
 | |
|         } else {
 | |
|             @newoutput = @{ $output->{$node} };
 | |
|         }
 | |
|         foreach (@newoutput) {
 | |
|             my $rsp;
 | |
|             $rsp->{node}->[0]->{name} = $node;
 | |
|             push @{ $rsp->{node}->[0]->{data}->[0]->{contents} }, $_;
 | |
| 
 | |
|             xCAT::MsgUtils->message("N", $rsp, $callback);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| 1;
 |