mirror of
				https://github.com/xcat2/xcat-core.git
				synced 2025-10-26 00:45:38 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			2005 lines
		
	
	
		
			67 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
			
		
		
	
	
			2005 lines
		
	
	
		
			67 KiB
		
	
	
	
		
			Perl
		
	
	
	
	
	
| #!/usr/bin/env perl
 | |
| # IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
 | |
| package xCAT_plugin::monctrlcmds;
 | |
| 
 | |
| BEGIN
 | |
| {
 | |
|     $::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
 | |
| }
 | |
| use lib "$::XCATROOT/lib/perl";
 | |
| use strict;
 | |
| use xCAT::NodeRange;
 | |
| use xCAT::Table;
 | |
| use xCAT::MsgUtils;
 | |
| use xCAT_monitoring::monitorctrl;
 | |
| use xCAT::Utils;
 | |
| use Sys::Hostname;
 | |
| use Data::Dumper;
 | |
| 
 | |
| 
 | |
| 1;
 | |
| 
 | |
| 
 | |
| 
 | |
| #-------------------------------------------------------------------------------
 | |
| 
 | |
| =head1  xCAT_plugin:monctrlcmds
 | |
| =head2    Package Description
 | |
|   xCAT monitoring control commands plugini module. This modules handles
 | |
|   monitoring related commands.
 | |
| =cut
 | |
| 
 | |
| #-------------------------------------------------------------------------------
 | |
| 
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   handled_commands
 | |
|       It returns a list of commands handled by this plugin.
 | |
|     Arguments:
 | |
|         none
 | |
|     Returns:
 | |
|         a list of commands.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub handled_commands {
 | |
|     return {
 | |
|         monstart => "monctrlcmds",
 | |
|         monstop  => "monctrlcmds",
 | |
|         monls    => "monctrlcmds",
 | |
|         monadd   => "monctrlcmds",
 | |
|         monrm    => "monctrlcmds",
 | |
|         moncfg   => "monctrlcmds",
 | |
|         mondecfg => "monctrlcmds",
 | |
|         monshow  => "monctrlcmds",
 | |
|       }
 | |
| }
 | |
| 
 | |
| #-------------------------------------------------------
 | |
| 
 | |
| =head3  preprocess_request
 | |
| 
 | |
|   Check and setup for hierarchy 
 | |
| 
 | |
| =cut
 | |
| 
 | |
| #-------------------------------------------------------
 | |
| sub preprocess_request
 | |
| {
 | |
|     my $req      = shift;
 | |
|     my $callback = shift;
 | |
|     my $command  = $req->{command}->[0];
 | |
| 
 | |
|     #  if ($req->{_xcatdest}) { return [$req]; }    #exit if preprocessed
 | |
|     if ($req->{_xcatpreprocessed}->[0] == 1) { return [$req]; }
 | |
| 
 | |
|     if ($req->{module}) { return [$req]; }
 | |
|     my $args = $req->{arg};
 | |
| 
 | |
|     my @requests = ();
 | |
| 
 | |
| 
 | |
|     if (($command eq "monstart") || ($command eq "monstop") || ($command eq "moncfg") || ($command eq "mondecfg") || ($command eq "monshow")) {
 | |
|         my @a_ret;    #(0, $modulename, $nodestatutmon, $scope, \@nodes)
 | |
|         if ($command eq "monstart") {
 | |
|             @a_ret = preprocess_monstart($args, $callback);
 | |
|         } elsif ($command eq "monstop") {
 | |
|             @a_ret = preprocess_monstop($args, $callback);
 | |
|         } elsif ($command eq "moncfg") {
 | |
|             @a_ret = preprocess_moncfg($args, $callback);
 | |
|         } elsif ($command eq "mondecfg") {
 | |
|             @a_ret = preprocess_mondecfg($args, $callback);
 | |
|         } elsif ($command eq "monshow") {
 | |
|             @a_ret = preprocess_monshow($args, $callback);
 | |
|         }
 | |
| 
 | |
|         if ($a_ret[0] != 0) {
 | |
|             $req = {};
 | |
|             return;
 | |
|         } else {
 | |
|             my $allnodes = $a_ret[4];
 | |
| 
 | |
|             #print "allnodes=@$allnodes\n";
 | |
|             my $pname       = $a_ret[1];
 | |
|             my $file_name   = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|             my $module_name = "xCAT_monitoring::$pname";
 | |
|             undef $SIG{CHLD};
 | |
|             if (($command eq "monshow") && (@$allnodes == 0) && ($a_ret[2] & 0x2 != 0)) {
 | |
|                 my $reqcopy = {%$req};
 | |
|                 push @{ $reqcopy->{module} }, $a_ret[1];
 | |
|                 push @{ $reqcopy->{priv} },   $a_ret[2];
 | |
|                 push @{ $reqcopy->{priv} },   $a_ret[3];
 | |
|                 push @{ $reqcopy->{priv} },   $a_ret[5];
 | |
|                 push @{ $reqcopy->{priv} },   $a_ret[6];
 | |
|                 push @{ $reqcopy->{priv} },   $a_ret[7];
 | |
|                 push @requests, $reqcopy;
 | |
|                 return \@requests;
 | |
|             }
 | |
| 
 | |
|             #initialize and start monitoring
 | |
|             no strict "refs";
 | |
|             my $mon_hierachy;
 | |
|             if (defined(${ $module_name . "::" }{getNodesMonServers})) {
 | |
|                 $mon_hierachy = ${ $module_name . "::" }{getNodesMonServers}->($allnodes, $callback);
 | |
|             } else {
 | |
|                 $mon_hierachy = xCAT_monitoring::monitorctrl->getNodeMonServerPair($allnodes, 1);
 | |
|             }
 | |
| 
 | |
|             #print Dumper($mon_hierachy);
 | |
| 
 | |
|             if (ref($mon_hierachy) eq 'ARRAY') {
 | |
|                 my $rsp2 = {};
 | |
|                 $rsp2->{data}->[0] = $mon_hierachy->[1];
 | |
|                 $callback->($rsp2);
 | |
|                 $req = {};
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
| 
 | |
|             my @mon_servers = keys(%$mon_hierachy);
 | |
|             my @hostinfo    = xCAT::NetworkUtils->determinehostname();
 | |
| 
 | |
|             #print "hostinfo=@hostinfo\n";
 | |
|             my $isSV   = xCAT::Utils->isServiceNode();
 | |
|             my %iphash = ();
 | |
|             foreach (@hostinfo) { $iphash{$_} = 1; }
 | |
|             if (!$isSV) { $iphash{'noservicenode'} = 1; }
 | |
| 
 | |
|             #check if we should also pass nodes that are managed by the sn to mn.
 | |
|             my $handleGrands = 0;
 | |
|             if (!$isSV) {
 | |
|                 if (defined(${ $module_name . "::" }{handleGrandChildren})) {
 | |
|                     $handleGrands = ${ $module_name . "::" }{handleGrandChildren}->();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             #print "handleGrands=$handleGrands\n";
 | |
| 
 | |
|             my $index          = 0;
 | |
|             my $reqcopy_grands = {%$req};
 | |
|             foreach my $sv_pair (@mon_servers) {
 | |
| 
 | |
|                 #service node come in pairs, the first one is the monserver adapter that facing the mn,
 | |
|                 # the second one is facing the cn. we use the first one here
 | |
|                 my @server_pair = split(':', $sv_pair);
 | |
|                 my $sv = $server_pair[0];
 | |
|                 my $sv1;
 | |
|                 if (@server_pair > 1) {
 | |
|                     $sv1 = $server_pair[1];
 | |
|                 }
 | |
|                 my $mon_nodes = $mon_hierachy->{$sv_pair};
 | |
|                 if ((!$mon_nodes) || (@$mon_nodes == 0)) { next; }
 | |
| 
 | |
|                 #print "sv=$sv, nodes=@$mon_nodes\n";
 | |
| 
 | |
|                 my $reqcopy = {%$req};
 | |
|                 if (!$iphash{$sv}) {
 | |
|                     if ($isSV) { next; } #if the command is issued on the monserver, only handle its children.
 | |
|                     else {
 | |
|                         if ($handleGrands) {
 | |
|                             $index++;
 | |
|                             $reqcopy_grands->{"grand_$index"} = "$sv,$sv1," . join(',', @$mon_nodes);
 | |
|                         }
 | |
|                     }
 | |
|                     $reqcopy->{'_xcatdest'} = $sv;
 | |
|                     $reqcopy->{_xcatpreprocessed}->[0] = 1;
 | |
|                     my $rsp2 = {};
 | |
|                     $rsp2->{data}->[0] = "sending request to $sv..., " . join(',', @$mon_nodes);
 | |
|                     $callback->($rsp2);
 | |
|                 }
 | |
| 
 | |
|                 push @{ $reqcopy->{module} }, $a_ret[1];
 | |
|                 if ($command eq "monshow") {
 | |
|                     push @{ $reqcopy->{priv} }, $a_ret[2];
 | |
|                     push @{ $reqcopy->{priv} }, $a_ret[3];
 | |
|                     push @{ $reqcopy->{priv} }, $a_ret[5];
 | |
|                     push @{ $reqcopy->{priv} }, $a_ret[6];
 | |
|                     push @{ $reqcopy->{priv} }, $a_ret[7];
 | |
|                 } else {
 | |
|                     push @{ $reqcopy->{nodestatmon} }, $a_ret[2];
 | |
|                     push @{ $reqcopy->{scope} },       $a_ret[3];
 | |
|                 }
 | |
|                 push @{ $reqcopy->{nodeinfo} }, join(',', @$mon_nodes);
 | |
|                 push @requests, $reqcopy;
 | |
|             }
 | |
| 
 | |
|             #add the a request for mn to handle all its grand children
 | |
|             if ($index > 0) {
 | |
|                 $reqcopy_grands->{grand_total} = $index;
 | |
|                 push @{ $reqcopy_grands->{module} }, $a_ret[1];
 | |
|                 if ($command eq "monshow") {
 | |
|                     push @{ $reqcopy_grands->{priv} }, $a_ret[2];
 | |
|                     push @{ $reqcopy_grands->{priv} }, $a_ret[3];
 | |
|                     push @{ $reqcopy_grands->{priv} }, $a_ret[5];
 | |
|                     push @{ $reqcopy_grands->{priv} }, $a_ret[6];
 | |
|                     push @{ $reqcopy_grands->{priv} }, $a_ret[7];
 | |
|                 } else {
 | |
|                     push @{ $reqcopy_grands->{nodestatmon} }, $a_ret[2];
 | |
|                     push @{ $reqcopy_grands->{scope} },       $a_ret[3];
 | |
|                 }
 | |
|                 push @requests, $reqcopy_grands;
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         my $reqcopy = {%$req};
 | |
|         push @requests, $reqcopy;
 | |
|     }
 | |
| 
 | |
|     return \@requests;
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   process_request
 | |
|       It processes the monitoring control commands.
 | |
|     Arguments:
 | |
|       request -- a hash table which contains the command name and the arguments.
 | |
|       callback -- a callback pointer to return the response to.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub process_request {
 | |
|     use Getopt::Long;
 | |
| 
 | |
|     # options can be bundled up like -vV
 | |
|     Getopt::Long::Configure("bundling");
 | |
|     $Getopt::Long::ignorecase = 0;
 | |
| 
 | |
|     #print "process_request get called\n";
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
|     my $command  = $request->{command}->[0];
 | |
|     my $args     = $request->{arg};
 | |
|     my $doreq    = shift;
 | |
| 
 | |
|     if ($command eq "monstart") {
 | |
|         return monstart($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "monstop") {
 | |
|         return monstop($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "monls") {
 | |
|         return monls($request, $callback, $doreq);
 | |
| 
 | |
|     }
 | |
|     elsif ($command eq "monadd") {
 | |
|         return monadd($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "monrm") {
 | |
|         return monrm($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "moncfg") {
 | |
|         return moncfg($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "mondecfg") {
 | |
|         return mondecfg($request, $callback, $doreq);
 | |
|     }
 | |
|     elsif ($command eq "monshow") {
 | |
|         return monshow($request, $callback, $doreq);
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "unsupported command: $command.";
 | |
|         $callback->($rsp);
 | |
|         return 1;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3  preprocess_monstart
 | |
|         This function handles the syntax checking for monstart command,
 | |
|      turn on the given monitoring plug-in to the 'monitoring' table.
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [noderange] [-r|--remote]        
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be invoked for monitoring the xCAT cluster.
 | |
|           noderange a range of nodes to be monitored. Default is all.
 | |
|           -r|--remote indicates that both monservers and the nodes need to be called to start
 | |
|              the monitoring. The defaults is monservers only.
 | |
|     Returns:
 | |
|         (0, $modulename, $nodestatutmon, $scope, \@nodes) for success. scope is the scope of the
 | |
|             actions. 1 means monervers only, 2 means both nodes and monservers.
 | |
|         (1, "") for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub preprocess_monstart
 | |
| {
 | |
|     my $args     = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     if (xCAT::Utils->isServiceNode()) {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "This command is not supported on a service node.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monstart_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monstart name [noderange] [-r|--remote]";
 | |
|         $rsp->{data}->[2] = "  monstart [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "     name is the name of the monitoring plug-in module to be invoked.";
 | |
|         $rsp->{data}->[4] = "       Use 'monls -a' command to list all the monitoring plug-in names.";
 | |
|         $rsp->{data}->[5] = "     noderange is a range of nodes to be monitored. The default is all nodes.";
 | |
|         $rsp->{data}->[6] = "      -r|--remote indicates that both monservers and the nodes need to be called\n       to start the monitoring. The default is monservers only.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     my $settings;
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'v|version' => \$::VERSION,
 | |
|             'r|remote'  => \$::REMOTE,))
 | |
|     {
 | |
|         &monstart_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monstart_usage($callback);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     my $pname = "";
 | |
|     my $scope = 0; #set it to 0 instead of 1 because it will be distributed to monservers.
 | |
|     my @nodes       = ();
 | |
|     my $nodestatmon = 0;
 | |
| 
 | |
|     if ($::REMOTE) { $scope = 2; }
 | |
| 
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &monstart_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
|     else {
 | |
|         #@product_names=split(/,/, $ARGV[0]);
 | |
|         $pname = $ARGV[0];
 | |
|         if (@ARGV > 1) {
 | |
|             my $noderange = $ARGV[1];
 | |
|             @nodes = noderange($noderange);
 | |
|             if (nodesmissed) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "Invalid nodes in noderange:" . join(',', nodesmissed);
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         if (!-e $file_name) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|             $callback->($rsp);
 | |
|             return (1, "");
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1, -autocommit => 1);
 | |
|     if ($table) {
 | |
|         my $found = 0;
 | |
|         my $tmp1  = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     $found = 1;
 | |
|                     if ($_->{disable} !~ /0|NO|No|no|N|n/) {
 | |
|                         my %key_col = (name    => $pname);
 | |
|                         my %tb_cols = (disable => "0");
 | |
|                         $table->setAttribs(\%key_col, \%tb_cols);
 | |
|                     }
 | |
|                     if ($_->{nodestatmon} =~ /1|Yes|yes|YES|Y|y/) { $nodestatmon = 1; }
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname has not been added to the monitoring table. Please run 'monadd' command to add.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return (1, "");
 | |
|         }
 | |
| 
 | |
|         $table->close();
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     return (0, $pname, $nodestatmon, $scope, \@nodes);
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monstart
 | |
|         This function calls moniutoring control to start the monitoring and node
 | |
|     status monitoring for the given plug-in module. 
 | |
|     Arguments:
 | |
|       request -- pointer to a hash with keys are command, module and nodestatmon.
 | |
|       callback - the pointer to the callback function.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monstart {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $pname       = $request->{module}->[0];
 | |
|     my $nodestatmon = $request->{nodestatmon}->[0];
 | |
|     my $scope       = $request->{scope}->[0];
 | |
|     my $nodeinfo    = $request->{nodeinfo}->[0];
 | |
| 
 | |
|     my $grands = {};
 | |
|     my $total  = 0;
 | |
|     if (exists($request->{grand_total})) {
 | |
|         $total = $request->{grand_total};
 | |
|     }
 | |
|     for (my $i = 1 ; $i <= $total ; $i++) {
 | |
|         if (exists($request->{"grand_$i"})) {
 | |
|             my $temp = $request->{"grand_$i"};
 | |
|             my @tmpnodes = split(',', $temp);
 | |
|             if (@tmpnodes > 2) {
 | |
|                 my $sv  = shift(@tmpnodes);
 | |
|                 my $sv1 = shift(@tmpnodes);
 | |
|                 $grands->{"$sv,$sv1"} = \@tmpnodes;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #print "-------grands" . Dumper($grands);
 | |
| 
 | |
| 
 | |
|     my @nodes = split(',', $nodeinfo);
 | |
| 
 | |
|     #print "monstart get called: pname=$pname\nnodestatmon=$nodestatmon\nnodeinfo=$nodeinfo\nscope=$scope\n";
 | |
| 
 | |
|     xCAT_monitoring::monitorctrl->startMonitoring([$pname], \@nodes, $scope, $callback, $grands);
 | |
| 
 | |
|     if ($nodestatmon) {
 | |
|         xCAT_monitoring::monitorctrl->startNodeStatusMonitoring($pname, \@nodes, $scope, $callback, $grands);
 | |
|     }
 | |
|     return;
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   preprocess_monstop
 | |
|         This function unregisters the given monitoring plug-in from the 'monitoring' table.
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [noderange] [-r|--remote]
 | |
|         name
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be stopped for monitoring the xCAT cluster.
 | |
|           noderange a range of nodes. Default is all.
 | |
|           -r|--remote indicates that both monservers and the nodes need to be called to stop
 | |
|              the monitoring. The defaults is monservers only.
 | |
|     Returns:
 | |
|         (0, $modulename, $nodestatutmon, $scope, \@nodes) for success. scope is the scope of the
 | |
|             actions. 1 means monervers only, 2 means both nodes and monservers.
 | |
|         (1, "") for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub preprocess_monstop
 | |
| {
 | |
|     my $args     = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     if (xCAT::Utils->isServiceNode()) {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "This command is not supported on a service node.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monstop_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monstop name [noderange] [-r|--remote]";
 | |
|         $rsp->{data}->[2] = "  monstop [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "      name is the name of the monitoring plug-in module registered in the monitoring table.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'r|remote'  => \$::REMOTE,
 | |
|             'v|version' => \$::VERSION,))
 | |
|     {
 | |
|         &monstop_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monstop_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     my $pname       = "";
 | |
|     my $scope       = 0;
 | |
|     my @nodes       = ();
 | |
|     my $nodestatmon = 0;
 | |
| 
 | |
|     if ($::REMOTE) { $scope = 2; }
 | |
| 
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &monstop_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
|     else {
 | |
|         $pname = $ARGV[0];
 | |
|         if (@ARGV > 1) {
 | |
|             my $noderange = $ARGV[1];
 | |
|             @nodes = noderange($noderange);
 | |
|             if (nodesmissed) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "Invalid nodes in noderange:" . join(',', nodesmissed);
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         if (!-e $file_name) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|             $callback->($rsp);
 | |
|             return (1, "");
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1, -autocommit => 1);
 | |
|     if ($table) {
 | |
|         my $found = 0;
 | |
|         my $tmp1  = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     $found = 1;
 | |
|                     if ($_->{disable} =~ /0|NO|No|no|N|n/) {
 | |
|                         my %key_col = (name    => $pname);
 | |
|                         my %tb_cols = (disable => "1");
 | |
|                         $table->setAttribs(\%key_col, \%tb_cols);
 | |
|                     }
 | |
|                     if ($_->{nodestatmon} =~ /1|Yes|yes|YES|Y|y/) { $nodestatmon = 1; }
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname cannot be found in the monitoring table.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return (1, "");
 | |
|         }
 | |
| 
 | |
|         $table->close();
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     return (0, $pname, $nodestatmon, $scope, \@nodes);
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monstop
 | |
|         This function calls moniutoring control to stop the monitoring and node
 | |
|     status monitoring for the given plug-in module. 
 | |
|     Arguments:
 | |
|       request -- pointer to a hash with keys are command, module and nodestatmon.
 | |
|       callback - the pointer to the callback function.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monstop {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $pname       = $request->{module}->[0];
 | |
|     my $nodestatmon = $request->{nodestatmon}->[0];
 | |
|     my $scope       = $request->{scope}->[0];
 | |
|     my $nodeinfo    = $request->{nodeinfo}->[0];
 | |
| 
 | |
| 
 | |
|     my $grands = {};
 | |
|     my $total  = 0;
 | |
|     if (exists($request->{grand_total})) {
 | |
|         $total = $request->{grand_total};
 | |
|     }
 | |
|     for (my $i = 1 ; $i <= $total ; $i++) {
 | |
|         if (exists($request->{"grand_$i"})) {
 | |
|             my $temp = $request->{"grand_$i"};
 | |
|             my @tmpnodes = split(',', $temp);
 | |
|             if (@tmpnodes > 2) {
 | |
|                 my $sv  = shift(@tmpnodes);
 | |
|                 my $sv1 = shift(@tmpnodes);
 | |
|                 $grands->{"$sv,$sv1"} = \@tmpnodes;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #print "-------grands" . Dumper($grands);
 | |
| 
 | |
| 
 | |
|     my @nodes = split(',', $nodeinfo);
 | |
| 
 | |
|     #print "monstop get called: pname=$pname\nnodestatmon=$nodestatmon\nnodeinfo=@nodes\nscope=$scope\n";
 | |
| 
 | |
|     if ($nodestatmon) {
 | |
|         xCAT_monitoring::monitorctrl->stopNodeStatusMonitoring($pname, \@nodes, $scope, $callback, $grands);
 | |
|     }
 | |
| 
 | |
|     xCAT_monitoring::monitorctrl->stopMonitoring([$pname], \@nodes, $scope, $callback, $grands);
 | |
| 
 | |
|     return;
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monls
 | |
|         This function list the monitoring plug-in module names, status and description. 
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         [name] [-a|all] [-d|--description]         
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monls {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
|     my $args     = $request->{arg};
 | |
|     my $doreq    = shift;
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monls_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monls name [-d|--description]";
 | |
|         $rsp->{data}->[2] = "  monls [-a|--all] [-d|--description]";
 | |
|         $rsp->{data}->[3] = "  monls [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[4] = "     name is the name of the monitoring plug-in module.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) {
 | |
|         @ARGV = @{$args};
 | |
|     }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'        => \$::HELP,
 | |
|             'v|version'     => \$::VERSION,
 | |
|             'a|all'         => \$::ALL,
 | |
|             'd|discription' => \$::DESC))
 | |
|     {
 | |
|         &monls_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monls_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     my $usetab     = 0;
 | |
|     my %names      = ();
 | |
|     my $plugin_dir = "$::XCATROOT/lib/perl/xCAT_monitoring";
 | |
|     if (@ARGV > 0)
 | |
|     {
 | |
|         $names{ $ARGV[0] } = 0;
 | |
|     }
 | |
|     else {
 | |
|         if ($::ALL) {
 | |
| 
 | |
|             #get all the module names from /opt/xcat/lib/perl/XCAT_monitoring directory
 | |
|             my @plugins = glob($plugin_dir . "/*.pm");
 | |
|             foreach (@plugins) {
 | |
|                 /.*\/([^\/]*).pm$/;
 | |
|                 $names{$1} = 0;
 | |
|             }
 | |
| 
 | |
|             # remove 2 files that are not plug-ins
 | |
|             delete($names{monitorctrl});
 | |
|             delete($names{montbhandler});
 | |
|         }
 | |
|         else {
 | |
|             $usetab = 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #get the list from the table
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1);
 | |
|     if ($table) {
 | |
|         my $tmp1 = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 my $pname = $_->{name};
 | |
|                 if (($usetab) || exists($names{$pname})) {
 | |
|                     $names{$pname} = 1;
 | |
| 
 | |
|                     #find out the monitoring plugin file and module name for the product
 | |
|                     my $rsp = {};
 | |
| 
 | |
|                     my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|                     my $module_name = "xCAT_monitoring::$pname";
 | |
| 
 | |
|                     #load the module in memory
 | |
|                     eval { require($file_name) };
 | |
|                     if ($@) {
 | |
|                         $rsp->{data}->[0] = "$pname: The file $file_name cannot be located or has compiling errors.";
 | |
|                         $callback->($rsp);
 | |
|                         next;
 | |
|                     } else {
 | |
|                         no strict "refs";
 | |
|                         if (!defined(${ $module_name . "::" }{start})) { next; }
 | |
|                     }
 | |
| 
 | |
|                     my $monnode = 0;
 | |
|                     my $disable = 1;
 | |
|                     if ($_->{nodestatmon} =~ /1|Yes|yes|YES|Y|y/) { $monnode = 1; }
 | |
|                     if ($_->{disable} =~ /0|NO|No|no|N|n/) { $disable = 0; }
 | |
|                     if ($disable) { $monnode = 0; }
 | |
|                     $rsp->{data}->[0] = "$pname\t\t" .
 | |
|                       ($disable ? "not-monitored"           : "monitored") .
 | |
|                       ($monnode ? "\tnode-status-monitored" : "");
 | |
|                     if ($::DESC) { getModuleDescription($rsp, $module_name); }
 | |
|                     $callback->($rsp);
 | |
|                 }
 | |
|             }    #foreach
 | |
|         }
 | |
|         $table->close();
 | |
|     }
 | |
| 
 | |
| 
 | |
|     #now handle the ones that are not in the table
 | |
|     foreach (keys(%names)) {
 | |
|         my $pname = $_;
 | |
|         if (!$names{$pname}) {
 | |
|             my $rsp = {};
 | |
| 
 | |
|             #find out the monitoring plugin file and module name for the product
 | |
|             my $file_name   = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|             my $module_name = "xCAT_monitoring::$pname";
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 $rsp->{data}->[0] = "$pname: The file $file_name cannot be located or has compiling errors.";
 | |
|                 $callback->($rsp);
 | |
|                 next;
 | |
|             } else {
 | |
|                 no strict "refs";
 | |
|                 if (!defined(${ $module_name . "::" }{start})) { next; }
 | |
|             }
 | |
|             $rsp->{data}->[0] = "$pname\t\tnot-monitored";
 | |
| 
 | |
|             if ($::DESC) {
 | |
|                 getModuleDescription($rsp, $module_name);
 | |
|             }
 | |
|             $callback->($rsp);
 | |
|         }
 | |
|     }
 | |
|     return;
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   getModuleDescription
 | |
|         This function gets description, postscripts and other info from the
 | |
|      the given monitoring plug_in and stored it in the given hash. 
 | |
|     Arguments:
 | |
|     Returns:
 | |
|         0 for success.
 | |
|         1. for unsuccess.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub getModuleDescription {
 | |
|     my $rsp         = shift;
 | |
|     my $module_name = shift;
 | |
|     no strict "refs";
 | |
| 
 | |
|     #description
 | |
|     if (defined(${ $module_name . "::" }{getDescription})) {
 | |
|         $rsp->{data}->[1] = ${ $module_name . "::" }{getDescription}->();
 | |
|     } else {
 | |
|         $rsp->{data}->[1] = "    No description available.";
 | |
|     }
 | |
| 
 | |
|     #postscripts
 | |
|     $rsp->{data}->[2] = "  Postscripts:\n";
 | |
|     if (defined(${ $module_name . "::" }{getPostscripts})) {
 | |
|         my $desc = ${ $module_name . "::" }{getPostscripts}->();
 | |
|         my @pn   = keys(%$desc);
 | |
| 
 | |
|         if (@pn > 0) {
 | |
|             foreach my $group (@pn) {
 | |
|                 $rsp->{data}->[2] .= "    $group: " . $desc->{$group};
 | |
|             }
 | |
|         } else { $rsp->{data}->[2] .= "    None"; }
 | |
|     } else { $rsp->{data}->[2] .= "    None"; }
 | |
| 
 | |
|     #support node status monitoring
 | |
|     $rsp->{data}->[3] = "  Support node status monitoring:\n";
 | |
|     my $snodestatusmon = 0;
 | |
|     if (defined(${ $module_name . "::" }{supportNodeStatusMon})) {
 | |
|         $snodestatusmon = ${ $module_name . "::" }{supportNodeStatusMon}->();
 | |
|     }
 | |
|     if   ($snodestatusmon) { $rsp->{data}->[3] .= "    Yes\n"; }
 | |
|     else                   { $rsp->{data}->[3] .= "    No\n"; }
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monadd
 | |
|         This function adds the given module name into the monitoring table and
 | |
|      sets the postsctipts in the postsctipts table. It also sets the given
 | |
|      settings into the monsetting table. 
 | |
|     Arguments:
 | |
|       request -- a hash table which contains the command name and the arguments.
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [-n|--nodestatmon] [-s|--settings ...]        
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be registered and invoked 
 | |
|               for monitoring the xCAT cluster.
 | |
|           -n|--nodestatmon  indicates that this plug-in will be used for feeding the node liveness
 | |
|               status to the xCAT nodelist table.  If not specified, the plug-in will not be used 
 | |
|               for feeding node status to xCAT. 
 | |
|           -s|--settings settings are used by the plug-in to customize it behavor.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monadd {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
|     my $args     = $request->{arg};
 | |
|     my $doreq    = shift;
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monadd_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monadd name [-n|--nodestatmon] [-s|--settings settings]";
 | |
|         $rsp->{data}->[2] = "  monadd [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "     name is the name of the monitoring plug-in module to be added.";
 | |
|         $rsp->{data}->[4] = "       Use 'monls -a' command to list all the monitoring plug-in names.";
 | |
|         $rsp->{data}->[5] = "     settings is used by the monitoring plug-in to customize its behavior.";
 | |
|         $rsp->{data}->[6] = "       Format: -s key1=value1 -s key2=value2 ... ";
 | |
|         $rsp->{data}->[7] = "       Please note that the square brackets are needed. ";
 | |
|         $rsp->{data}->[7] = "       Use 'monls name -d' command to look for the possible settings for a plug-in.";
 | |
|         $rsp->{data}->[8] = "  Example: monadd xcatmon -n -s ping-interval=10";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
|     my $settings;
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'        => \$::HELP,
 | |
|             'v|version'     => \$::VERSION,
 | |
|             'n|nodestatmon' => \$::NODESTATMON,
 | |
|             's|settings=s@' => \$settings))
 | |
|     {
 | |
|         &monadd_usage($callback);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monadd_usage($callback);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     #my @product_names;
 | |
|     my $pname;
 | |
|     my $nodestatmon = 0;
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &monadd_usage($callback);
 | |
|         return 1;
 | |
|     }
 | |
|     else {
 | |
|         #@product_names=split(/,/, $ARGV[0]);
 | |
|         $pname = $ARGV[0];
 | |
|         my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         if (!-e $file_name) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|             $callback->($rsp);
 | |
|             return 1;
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|                 $callback->($rsp);
 | |
|                 return 1;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1);
 | |
|     if ($table) {
 | |
| 
 | |
|         #my $tmp1=$table->getAllEntries("all");
 | |
|         #if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|         #  foreach(@$tmp1) {
 | |
|         #    my $name=$_->{name};
 | |
|         #    if ($name eq $pname) {
 | |
|         #      my $rsp={};
 | |
|         #      $rsp->{data}->[0]="$pname has already been added in the monitoring table.";
 | |
|         #      $callback->($rsp);
 | |
|         #      $table->close();
 | |
|         #      return 1;
 | |
|         #    }
 | |
|         #  }
 | |
|         #}
 | |
| 
 | |
|         my $module_name = "xCAT_monitoring::$pname";
 | |
| 
 | |
|         #check if the module suppors node status monitoring or not.
 | |
|         if ($::NODESTATMON) {
 | |
|             no strict "refs";
 | |
|             my $snodestatusmon = 0;
 | |
|             if (defined(${ $module_name . "::" }{supportNodeStatusMon})) {
 | |
|                 $snodestatusmon = ${ $module_name . "::" }{supportNodeStatusMon}->();
 | |
|             }
 | |
|             if (!$snodestatusmon) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "$pname does not support node status monitoring.";
 | |
|                 $callback->($rsp);
 | |
|                 $table->close();
 | |
|                 return 1;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         #update the monsetting table
 | |
|         if ($settings) {
 | |
|             my $table1 = xCAT::Table->new("monsetting", -create => 1, -autocommit => 1);
 | |
|             my %key_col1 = (name => $pname);
 | |
| 
 | |
|             #parse the settings. Setting format: key="value",key="value"....
 | |
|             foreach (@$settings) {
 | |
|                 if (/^\[(.*)\]$/) {    #backward compatible
 | |
|                     while (s/^\[([^\[\]\=]*)=([^\[\]]*)\](,)*//) {
 | |
|                         $key_col1{key} = $1;
 | |
|                         my %setting_hash = ();
 | |
|                         $setting_hash{value} = $2;
 | |
|                         $table1->setAttribs(\%key_col1, \%setting_hash);
 | |
|                     }
 | |
|                 } else {
 | |
|                     /^([^\=]*)=(.*)/;
 | |
|                     $key_col1{key} = $1;
 | |
|                     my %setting_hash = ();
 | |
|                     $setting_hash{value} = $2;
 | |
|                     $table1->setAttribs(\%key_col1, \%setting_hash);
 | |
|                 }
 | |
|             }
 | |
|             $table1->close();
 | |
|         }
 | |
| 
 | |
|         #update the monitoring table
 | |
|         my %key_col = (name => $pname);
 | |
|         my $nstat = 'N';
 | |
|         if ($::NODESTATMON) {
 | |
|             $nstat       = 'Y';
 | |
|             $nodestatmon = 1;
 | |
|         }
 | |
|         my %tb_cols = (nodestatmon => $nstat, disable => "1");
 | |
|         $table->setAttribs(\%key_col, \%tb_cols);
 | |
|         $table->close();
 | |
| 
 | |
|         #updating the postscript table
 | |
|         no strict "refs";
 | |
|         my $postscripts_h = {};
 | |
|         if (defined(${ $module_name . "::" }{getPostscripts})) {
 | |
|             my $postscripts_h = ${ $module_name . "::" }{getPostscripts}->();
 | |
|             my @pn            = keys(%$postscripts_h);
 | |
|             if (@pn > 0) {
 | |
|                 my $table2 = xCAT::Table->new("postscripts", -create => 1);
 | |
|                 if (!$table2) {
 | |
|                     my $rsp = {};
 | |
|                     $rsp->{data}->[0] = "Cannot open the postscripts table.\nFailed to set the postscripts for $pname.";
 | |
|                     $callback->($rsp);
 | |
|                     return 1;
 | |
|                 }
 | |
|                 foreach my $group (@pn) {
 | |
|                     my $posts = $postscripts_h->{$group};
 | |
|                     if ($posts) {
 | |
|                         (my $ref) = $table2->getAttribs({ node => $group }, 'postscripts');
 | |
|                         if ($ref and $ref->{postscripts}) {
 | |
|                             my @old_a = split(',', $ref->{postscripts});
 | |
|                             my @new_a = split(',', $posts);
 | |
|                             my %new_h = ();
 | |
|                             foreach my $new_tmp (@new_a) {
 | |
|                                 my $found = 0;
 | |
|                                 foreach my $old_tmp (@old_a) {
 | |
|                                     if ($old_tmp eq $new_tmp) { $found = 1; last; }
 | |
|                                 }
 | |
|                                 if (!$found) { $new_h{$new_tmp} = 1; }
 | |
|                             }
 | |
| 
 | |
|                             if (keys(%new_h) > 0) {
 | |
|                                 foreach (keys(%new_h)) { push(@old_a, $_); }
 | |
|                                 my $new_post = join(',', @old_a);
 | |
|                                 my %key_col2 = (node        => $group);
 | |
|                                 my %tb_cols2 = (postscripts => $new_post);
 | |
|                                 $table2->setAttribs(\%key_col2, \%tb_cols2);
 | |
|                             }
 | |
|                         } else {
 | |
|                             my %key_col2 = (node        => $group);
 | |
|                             my %tb_cols2 = (postscripts => $posts);
 | |
|                             $table2->setAttribs(\%key_col2, \%tb_cols2);
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 $table2->close();
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monrm
 | |
|       This function removes the given monitoring plug-in from the 'monitoring' table.
 | |
|     It also removed the postscritps for the module from the 'postscritps' table.
 | |
|     Arguments:
 | |
|       request -- a hash table which contains the command name and the arguments.
 | |
|       callback - the pointer to the callback function.
 | |
|        args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be stopped for monitoring the xCAT 
 | |
|               cluster if it is running and then removed from the monitoring table. 
 | |
|     Returns:
 | |
|         0 for success.
 | |
|         1 for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monrm {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
|     my $args     = $request->{arg};
 | |
|     my $doreq    = shift;
 | |
| 
 | |
|     if (xCAT::Utils->isServiceNode()) {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "This command is not supported on a service node.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monrm_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monrm name";
 | |
|         $rsp->{data}->[2] = "  monrm [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "      name is the name of the monitoring plug-in module registered in the monitoring table.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'v|version' => \$::VERSION,))
 | |
|     {
 | |
|         &monrm_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monrm_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     my $pname;
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &monrm_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
|     else {
 | |
|         $pname = $ARGV[0];
 | |
|     }
 | |
| 
 | |
|     my $disable = 1;
 | |
|     my $found   = 0;
 | |
|     my $table   = xCAT::Table->new("monitoring", -create => 1);
 | |
|     if ($table) {
 | |
|         my $tmp1 = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     if ($_->{disable} =~ /0|NO|No|no|N|n/) { $disable = 0; }
 | |
|                     $found = 1;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname is not in the monitoring talble.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return 0;
 | |
|         }
 | |
| 
 | |
|         if (!$disable) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "Please run command 'monstop $pname' to stop monitoring before running this command.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return 0;
 | |
|         }
 | |
| 
 | |
|         my %key_col = (name => $pname);
 | |
|         $table->delEntries(\%key_col);
 | |
|         $table->close();
 | |
| 
 | |
| 
 | |
|         #remove the postscripts for the module from the postscript table
 | |
|         no strict "refs";
 | |
|         my $file_name   = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         my $module_name = "xCAT_monitoring::$pname";
 | |
|         if (!-e $file_name) {
 | |
|             return 0;
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 return 0;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $postscripts_h = {};
 | |
|         if (defined(${ $module_name . "::" }{getPostscripts})) {
 | |
|             my $postscripts_h = ${ $module_name . "::" }{getPostscripts}->();
 | |
|             my @pn            = keys(%$postscripts_h);
 | |
|             if (@pn > 0) {
 | |
|                 my $table2 = xCAT::Table->new("postscripts", -create => 1);
 | |
|                 if (!$table2) {
 | |
|                     my $rsp = {};
 | |
|                     $rsp->{data}->[0] = "Cannot open the postscripts table.\nFailed to remove the postscripts for $pname.";
 | |
|                     $callback->($rsp);
 | |
|                     return 1;
 | |
|                 }
 | |
|                 foreach my $group (@pn) {
 | |
|                     my $posts = $postscripts_h->{$group};
 | |
|                     if ($posts) {
 | |
|                         (my $ref) = $table2->getAttribs({ node => $group }, 'postscripts');
 | |
|                         if ($ref and $ref->{postscripts}) {
 | |
|                             my @old_a = split(',', $ref->{postscripts});
 | |
|                             my @new_a = split(',', $posts);
 | |
|                             my %new_h = ();
 | |
|                             my @new_post_a = ();
 | |
|                             foreach my $old_tmp (@old_a) {
 | |
|                                 my $found = 0;
 | |
|                                 foreach my $new_tmp (@new_a) {
 | |
|                                     if ($old_tmp eq $new_tmp) { $found = 1; last; }
 | |
|                                 }
 | |
|                                 if (!$found) { push(@new_post_a, $old_tmp); }
 | |
|                             }
 | |
| 
 | |
|                             if (@new_post_a > 0) {
 | |
|                                 my $new_post = join(',', @new_post_a);
 | |
|                                 if ($new_post ne $ref->{postscripts}) {
 | |
|                                     my %key_col2 = (node        => $group);
 | |
|                                     my %tb_cols2 = (postscripts => $new_post);
 | |
|                                     $table2->setAttribs(\%key_col2, \%tb_cols2);
 | |
|                                 }
 | |
|                             } else {
 | |
|                                 my %key_col2 = (node => $group);
 | |
|                                 $table2->delEntries(\%key_col2);
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 $table2->close();
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Cannot open monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3  preprocess_moncfg
 | |
|         This function handles the syntax checking for moncfg command.
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [noderange] [-r|--remote]        
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be invoked for configuring the cluster to monitor the nodes.
 | |
|           noderange a range of nodes to be configured for. Default is all.
 | |
|           -r|--remote indicates that both monservers and the nodes need to configured.
 | |
|              The defaults is monservers only.
 | |
|     Returns:
 | |
|         (0, $modulename, $nodestatutmon, $scope, \@nodes) for success. scope is the scope of the
 | |
|             actions. 1 means monervers only, 2 means both nodes and monservers.
 | |
|         (1, "") for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub preprocess_moncfg
 | |
| {
 | |
|     my $args     = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub moncfg_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  moncfg name [noderange] [-r|--remote]";
 | |
|         $rsp->{data}->[2] = "  moncfg [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "     name is the name of the monitoring plug-in module to be invoked.";
 | |
|         $rsp->{data}->[4] = "        Use 'monls -a' command to list all the monitoring plug-in names.";
 | |
|         $rsp->{data}->[5] = "     noderange is a range of nodes to be configured for. The default is all nodes.";
 | |
|         $rsp->{data}->[6] = "      -r|--remote indicates that both monservers and the nodes need to be configured.\n       The default is monservers only.";
 | |
|         $rsp->{data}->[7] = "        The default is monservers only.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'v|version' => \$::VERSION,
 | |
|             'r|remote'  => \$::REMOTE,))
 | |
|     {
 | |
|         &moncfg_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &moncfg_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     my $pname       = "";
 | |
|     my $scope       = 0;
 | |
|     my @nodes       = ();
 | |
|     my $nodestatmon = 0;
 | |
| 
 | |
|     if ($::REMOTE) { $scope = 2; }
 | |
| 
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &moncfg_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
|     else {
 | |
|         $pname = $ARGV[0];
 | |
|         if (@ARGV > 1) {
 | |
|             my $noderange = $ARGV[1];
 | |
|             @nodes = noderange($noderange);
 | |
|             if (nodesmissed) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "Invalid nodes in noderange:" . join(',', nodesmissed);
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         if (!-e $file_name) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|             $callback->($rsp);
 | |
|             return (1, "");
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1, -autocommit => 1);
 | |
|     if ($table) {
 | |
|         my $found = 0;
 | |
|         my $tmp1  = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     $found = 1;
 | |
|                     if ($_->{nodestatmon} =~ /1|Yes|yes|YES|Y|y/) { $nodestatmon = 1; }
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname cannot be found in the monitoring table.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return (1, "");
 | |
|         }
 | |
| 
 | |
|         $table->close();
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     return (0, $pname, $nodestatmon, $scope, \@nodes);
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   moncfg
 | |
|       This function configures the cluster for the given nodes. It includes configuring 
 | |
|       and setting up the 3rd party monitoring software for monitoring the given nodes.  
 | |
|     Arguments:
 | |
|       request -- a hash table which contains the command name and the arguments.
 | |
|        callback -- the callback pointer for error and status displaying. It can be null.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub moncfg
 | |
| {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $pname       = $request->{module}->[0];
 | |
|     my $nodestatmon = $request->{nodestatmon}->[0];
 | |
|     my $scope       = $request->{scope}->[0];
 | |
|     my $nodeinfo    = $request->{nodeinfo}->[0];
 | |
| 
 | |
|     #print "---------monctrlcmnd::moncfg request=" . Dumper($request);
 | |
| 
 | |
|     my $grands = {};
 | |
|     my $total  = 0;
 | |
|     if (exists($request->{grand_total})) {
 | |
|         $total = $request->{grand_total};
 | |
|     }
 | |
|     for (my $i = 1 ; $i <= $total ; $i++) {
 | |
|         if (exists($request->{"grand_$i"})) {
 | |
|             my $temp = $request->{"grand_$i"};
 | |
|             my @tmpnodes = split(',', $temp);
 | |
|             if (@tmpnodes > 2) {
 | |
|                 my $sv  = shift(@tmpnodes);
 | |
|                 my $sv1 = shift(@tmpnodes);
 | |
|                 $grands->{"$sv,$sv1"} = \@tmpnodes;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #print "-------grands" . Dumper($grands);
 | |
| 
 | |
|     my @nodes = split(',', $nodeinfo);
 | |
| 
 | |
|     #print "moncfg get called: pname=$pname\nnodestatmon=$nodestatmon\nnodeinfo=@nodes\nscope=$scope\n";
 | |
| 
 | |
|     xCAT_monitoring::monitorctrl->config([$pname], \@nodes, $scope, $callback, $grands);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3  preprocess_mondecfg
 | |
|         This function handles the syntax checking for mondecfg command.
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [noderange] [-r|--remote]        
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. 
 | |
|               The specified plug-in will be invoked for deconfiguring the cluster to monitor the nodes.
 | |
|           noderange a range of nodes to be deconfigured for. Default is all.
 | |
|           -r|--remote indicates that both monservers and the nodes need to be deconfigured.
 | |
|              The defaults is monservers only.
 | |
|     Returns:
 | |
|         (0, $modulename, $nodestatutmon, $scope, \@nodes) for success. scope is the scope of the
 | |
|             actions. 1 means monervers only, 2 means both nodes and monservers.
 | |
|         (1, "") for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub preprocess_mondecfg
 | |
| {
 | |
|     my $args     = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub mondecfg_usage
 | |
|     {
 | |
|         my $cb  = shift;
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  mondecfg name [noderange] [-r|--remote]";
 | |
|         $rsp->{data}->[2] = "  mondecfg [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "     name is the name of the monitoring plug-in module to be invoked.";
 | |
|         $rsp->{data}->[4] = "        Use 'monls -a' command to list all the monitoring plug-in names.";
 | |
|         $rsp->{data}->[5] = "     noderange is a range of nodes to be deconfigured for.";
 | |
|         $rsp->{data}->[6] = "        The default is all nodes.";
 | |
|         $rsp->{data}->[7] = "      -r|--remote indicates that both monservers and the nodes need to be deconfigured.";
 | |
|         $rsp->{data}->[8] = "        The default is monservers only.";
 | |
|         $cb->($rsp);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'v|version' => \$::VERSION,
 | |
|             'r|remote'  => \$::REMOTE,))
 | |
|     {
 | |
|         &mondecfg_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &mondecfg_usage($callback);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     my $pname       = "";
 | |
|     my $scope       = 0;
 | |
|     my @nodes       = ();
 | |
|     my $nodestatmon = 0;
 | |
| 
 | |
|     if ($::REMOTE) { $scope = 2; }
 | |
| 
 | |
|     if (@ARGV < 1)
 | |
|     {
 | |
|         &mondecfg_usage($callback);
 | |
|         return (1, "");
 | |
|     }
 | |
|     else {
 | |
|         $pname = $ARGV[0];
 | |
|         if (@ARGV > 1) {
 | |
|             my $noderange = $ARGV[1];
 | |
|             @nodes = noderange($noderange);
 | |
|             if (nodesmissed) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "Invalid nodes in noderange:" . join(',', nodesmissed);
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|         if (!-e $file_name) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|             $callback->($rsp);
 | |
|             return (1, "");
 | |
|         } else {
 | |
| 
 | |
|             #load the module in memory
 | |
|             eval { require($file_name) };
 | |
|             if ($@) {
 | |
|                 my $rsp = {};
 | |
|                 $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|                 $callback->($rsp);
 | |
|                 return (1, "");
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1, -autocommit => 1);
 | |
|     if ($table) {
 | |
|         my $found = 0;
 | |
|         my $tmp1  = $table->getAllEntries("all");
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     $found = 1;
 | |
|                     if ($_->{nodestatmon} =~ /1|Yes|yes|YES|Y|y/) { $nodestatmon = 1; }
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname cannot be found in the monitoring table.";
 | |
|             $callback->($rsp);
 | |
|             $table->close();
 | |
|             return (1, "");
 | |
|         }
 | |
| 
 | |
|         $table->close();
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     return (0, $pname, $nodestatmon, $scope, \@nodes);
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   mondecfg
 | |
|       This function deconfigures the cluster for the given nodes. It includes deconfiguring 
 | |
|       and clearning up the 3rd party monitoring software for monitoring the given nodes.  
 | |
|     Arguments:
 | |
|        names -- a pointer to an  array of monitoring plug-in names. If non is specified,
 | |
|          all the plug-ins registered in the monitoring table will be notified.
 | |
|        p_nodes -- a pointer to an arrays of nodes to be removed from the monitoring domain. 
 | |
|                   none means all.
 | |
|        scope -- the action scope, it indicates the node type the action will take place.
 | |
|                 0 means localhost only. 
 | |
|                 1 means monserver only, 
 | |
|                 2 means both monservers and nodes, 
 | |
|        callback -- the callback pointer for error and status displaying. It can be null.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub mondecfg
 | |
| {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $pname       = $request->{module}->[0];
 | |
|     my $nodestatmon = $request->{nodestatmon}->[0];
 | |
|     my $scope       = $request->{scope}->[0];
 | |
|     my $nodeinfo    = $request->{nodeinfo}->[0];
 | |
| 
 | |
|     my $grands = {};
 | |
|     my $total  = 0;
 | |
|     if (exists($request->{grand_total})) {
 | |
|         $total = $request->{grand_total};
 | |
|     }
 | |
|     for (my $i = 1 ; $i <= $total ; $i++) {
 | |
|         if (exists($request->{"grand_$i"})) {
 | |
|             my $temp = $request->{"grand_$i"};
 | |
|             my @tmpnodes = split(',', $temp);
 | |
|             if (@tmpnodes > 2) {
 | |
|                 my $sv  = shift(@tmpnodes);
 | |
|                 my $sv1 = shift(@tmpnodes);
 | |
|                 $grands->{"$sv,$sv1"} = \@tmpnodes;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #print "-------grands" . Dumper($grands);
 | |
| 
 | |
|     my @nodes = split(',', $nodeinfo);
 | |
| 
 | |
|     #print "mondecfg get called: pname=$pname\nnodestatmon=$nodestatmon\nnodeinfo=@nodes\nscope=$scope\n";
 | |
| 
 | |
|     xCAT_monitoring::monitorctrl->deconfig([$pname], \@nodes, $scope, $callback, $grands);
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3  preprocess_monshow
 | |
|         This function handles the syntax checking for monshow command.
 | |
|     Arguments:
 | |
|       callback - the pointer to the callback function.
 | |
|       args - The format of the args is:
 | |
|         [-h|--help|-v|--version] or
 | |
|         name [noderange] [-s] [-t time] [-a attributes] [-w attr<operator>val [-w attr<operator>val] ...] [-o pe]        
 | |
|         where
 | |
|           name is the monitoring plug-in name. For example: rmcmon. Only for rmcmon currently.
 | |
|           noderange a range of nodes to be showed for. If omitted, the data for all the nodes will be displayed.
 | |
|           -s shows the summary data only
 | |
|           -t specify a range of time for the data, default is last 60 minutes
 | |
| 	  -a specifies a comma-separated list of attributes or metrics names. The default is all.
 | |
|     Returns:
 | |
|         (0, $modulename, $sum, $time, \@nodes, $attrs, $pe, $where) for success.
 | |
|         (1, "") for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub preprocess_monshow
 | |
| {
 | |
|     my $args     = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     # subroutine to display the usage
 | |
|     sub monshow_usage
 | |
|     {
 | |
|         my $cb    = shift;
 | |
|         my $error = shift;
 | |
|         my $rsp   = {};
 | |
|         $rsp->{data}->[0] = "Usage:";
 | |
|         $rsp->{data}->[1] = "  monshow name noderange [-s] [-t time] [-a attributes] [-w attr<operator>val[-w attr<operator>val ...]][-o pe]";
 | |
|         $rsp->{data}->[2] = "  monshow [-h|--help|-v|--version]";
 | |
|         $rsp->{data}->[3] = "     name is the name of the monitoring plug-in module to be invoked.";
 | |
|         $rsp->{data}->[4] = "     noderange is a list of nodes to be showed for. If omitted,";
 | |
|         $rsp->{data}->[5] = "        the data for all the nodes will be displayed.";
 | |
|         $rsp->{data}->[6] = "     -s shows the summary data.";
 | |
|         $rsp->{data}->[7] = "     -t specifies a range of time for the data, The default is last 60 minutes";
 | |
|         $rsp->{data}->[8] = "     -a specifies a comma-separated list of attributes or metrics names. The default is all.";
 | |
|         $rsp->{data}->[9] = "     -w specifies one or multiple selection string that can be used to select events.";
 | |
|         $rsp->{data}->[10] = "     -o specifies montype, it can be p, e or pe.";
 | |
|         $rsp->{data}->[11] = "        p means performance, e means events, default is e";
 | |
| 
 | |
|         #    $cb->($rsp);
 | |
|         xCAT::MsgUtils->message("D", $rsp, $callback, $error);
 | |
|     }
 | |
| 
 | |
|     @ARGV = ();
 | |
|     if ($args) { @ARGV = @{$args}; }
 | |
| 
 | |
|     # parse the options
 | |
|     if (!GetOptions(
 | |
|             'h|help'    => \$::HELP,
 | |
|             'v|version' => \$::VERSION,
 | |
|             's'         => \$::SUMMARY,
 | |
|             't=s'       => \$::TIME,
 | |
|             'a=s'       => \$::ATTRS,
 | |
|             'o=s'       => \$::PE,
 | |
|             'w=s@'      => \$::OPT_W))
 | |
|     {
 | |
|         &monshow_usage($callback, 1);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the usage if -h or --help is specified
 | |
|     if ($::HELP) {
 | |
|         &monshow_usage($callback, 0);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     # display the version statement if -v or --verison is specified
 | |
|     if ($::VERSION)
 | |
|     {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = xCAT::Utils->Version();
 | |
|         $callback->($rsp);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     my $pname = "";
 | |
|     my $sum   = 0;
 | |
|     my $time  = 60;
 | |
|     my @nodes = ();
 | |
|     my $attrs = undef;
 | |
|     my $pe    = 'e';
 | |
|     my $where = [];
 | |
| 
 | |
|     if (@ARGV < 1) {
 | |
|         &monshow_usage($callback, 1);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     if ($::SUMMARY) { $sum  = 1; }
 | |
|     if ($::TIME)    { $time = $::TIME; }
 | |
|     if ($::ATTRS) {
 | |
|         $attrs = $::ATTRS;
 | |
|     } else {
 | |
|         my $conftable = xCAT::Table->new('monsetting');
 | |
|         my @metrixconf = $conftable->getAttribs({ 'name' => 'rmcmon' }, ('key', 'value'));
 | |
|         foreach (@metrixconf) {
 | |
|             my $key   = $_->{key};
 | |
|             my $value = $_->{value};
 | |
|             my $temp  = undef;
 | |
|             if ($key =~ /^rmetrics/) {
 | |
|                 if ($value =~ /\]/) {
 | |
|                     ($temp, $value) = split /\]/, $value;
 | |
|                 }
 | |
|                 ($value, $temp) = split /:/, $value;
 | |
|                 if ($attrs) {
 | |
|                     $attrs = "$attrs,$value";
 | |
|                 } else {
 | |
|                     $attrs = $value;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     if ($::PE) { $pe = $::PE; }
 | |
|     if ($::OPT_W) {
 | |
|         $where = $::OPT_W;
 | |
|     }
 | |
| 
 | |
|     $pname = $ARGV[0];
 | |
| 
 | |
|     my $noderange = '';
 | |
|     if (@ARGV == 1) {
 | |
|         if ($sum) {
 | |
|             $sum |= 0x2;
 | |
|         }
 | |
|     } else {
 | |
|         $noderange = $ARGV[1];
 | |
|     }
 | |
| 
 | |
|     @nodes = noderange($noderange);
 | |
| 
 | |
|     if (xCAT::Utils->isMN() && nodesmissed) {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Invalid nodes in noderange:" . join(',', nodesmissed);
 | |
|         xCAT::MsgUtils->message("E", $rsp, $callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     my $file_name = "$::XCATROOT/lib/perl/xCAT_monitoring/$pname.pm";
 | |
|     if (!-e $file_name) {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "File $file_name does not exist.";
 | |
|         xCAT::MsgUtils->message("E", $rsp, $callback);
 | |
|         return (1, "");
 | |
|     } else {
 | |
| 
 | |
|         #load the module in memory
 | |
|         eval { require($file_name) };
 | |
|         if ($@) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "The file $file_name has compiling errors:\n$@\n";
 | |
|             xCAT::MsgUtils->message("E", $rsp, $callback);
 | |
|             return (1, "");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     my $table = xCAT::Table->new("monitoring", -create => 1, -autocommit => 1);
 | |
|     if ($table) {
 | |
|         my $found = 0;
 | |
|         my $tmp1  = $table->getAllEntries();
 | |
|         if (defined($tmp1) && (@$tmp1 > 0)) {
 | |
|             foreach (@$tmp1) {
 | |
|                 if ($pname eq $_->{name}) {
 | |
|                     $found = 1;
 | |
|                     last;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (!$found) {
 | |
|             my $rsp = {};
 | |
|             $rsp->{data}->[0] = "$pname cannot be found in the monitoring table.";
 | |
|             xCAT::MsgUtils->message("E", $rsp, $callback);
 | |
|             $table->close();
 | |
|             return (1, "");
 | |
|         }
 | |
| 
 | |
|         $table->close();
 | |
|     } else {
 | |
|         my $rsp = {};
 | |
|         $rsp->{data}->[0] = "Failed to open the monitoring table.";
 | |
|         xCAT::MsgUtils->message("E", $rsp, $callback);
 | |
|         return (1, "");
 | |
|     }
 | |
| 
 | |
|     return (0, $pname, $sum, $time, \@nodes, $attrs, $pe, $where);
 | |
| }
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| 
 | |
| =head3   monshow
 | |
|       This function configures the cluster performance for the given nodes.  
 | |
|     Arguments:
 | |
|       request -- a hash table which contains the command name and the arguments.
 | |
|       callback -- the callback pointer for error and status displaying. It can be null.
 | |
|     Returns:
 | |
|         0 for success. The output is returned through the callback pointer.
 | |
|         1. for unsuccess. The error messages are returns through the callback pointer.
 | |
| =cut
 | |
| 
 | |
| #--------------------------------------------------------------------------------
 | |
| sub monshow
 | |
| {
 | |
|     my $request  = shift;
 | |
|     my $callback = shift;
 | |
| 
 | |
|     my $pname    = $request->{module}->[0];
 | |
|     my $nodeinfo = $request->{nodeinfo}->[0];
 | |
|     my $sum      = $request->{priv}->[0];
 | |
|     my $time     = $request->{priv}->[1];
 | |
|     my $attrs    = $request->{priv}->[2];
 | |
|     my $pe       = $request->{priv}->[3];
 | |
|     my $where    = $request->{priv}->[4];
 | |
| 
 | |
|     my @nodes = split(',', $nodeinfo);
 | |
| 
 | |
|     xCAT_monitoring::monitorctrl->show([$pname], \@nodes, $sum, $time, $attrs, $pe, $where, $callback);
 | |
|     return 0;
 | |
| }
 | |
| 
 |