2009-06-17 06:57:37 +00:00
|
|
|
#!/usr/bin/env perl
|
|
|
|
#IBM(c) 2007 EPL license http://www.eclipse.org/legal/epl-v10.html
|
|
|
|
package xCAT_monitoring::rrdutil;
|
|
|
|
use strict;
|
|
|
|
use IO::Socket;
|
|
|
|
BEGIN
|
|
|
|
{
|
|
|
|
$::XCATROOT = $ENV{'XCATROOT'} ? $ENV{'XCATROOT'} : '/opt/xcat';
|
|
|
|
}
|
|
|
|
|
|
|
|
#Modules to use:
|
|
|
|
use lib "$::XCATROOT/lib/perl";
|
|
|
|
use xCAT::Utils;
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub start_RRD_server
|
|
|
|
#Description:
|
2009-07-04 03:59:42 +00:00
|
|
|
# FOR AIX:
|
2009-06-17 06:57:37 +00:00
|
|
|
# add "rrdsrv $port/tcp #RRD server" to /etc/services
|
|
|
|
# add "rrdsrv stream tcp no wait root /usr/bin/rrdtool rrdtool - $dir"
|
|
|
|
# to /etc/inetd.conf
|
|
|
|
# restart inetd
|
2009-07-04 03:59:42 +00:00
|
|
|
# FOR LINUX:
|
|
|
|
# add rrdsrv to xinetd
|
|
|
|
# restart xinetd
|
2009-06-17 06:57:37 +00:00
|
|
|
#Input:
|
|
|
|
# $port Port number of RRD server,
|
|
|
|
# $dir directory to save *.rrd
|
|
|
|
#return:
|
|
|
|
# 0 success
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
sub start_RRD_server
|
|
|
|
{
|
|
|
|
my ($port, $dir) = @_;
|
2009-07-04 03:59:42 +00:00
|
|
|
if(xCAT::Utils->isAIX()){
|
|
|
|
my $cmd = undef;
|
|
|
|
my @old = ();
|
|
|
|
my @new = ();
|
|
|
|
my $offset = 0;
|
|
|
|
my $found = 0;
|
|
|
|
@old = xCAT::Utils->runcmd("cat /etc/services", -2);
|
|
|
|
push @new, "rrdsrv $port/tcp #RRD server";
|
|
|
|
foreach (@old) {
|
|
|
|
if ($_ =~ /rrdsrv/){
|
|
|
|
if(!$found){
|
|
|
|
splice(@old, $offset, 1, @new);
|
|
|
|
$found = 1;
|
|
|
|
} else {
|
|
|
|
splice(@old, $offset, 1);
|
|
|
|
}
|
2009-07-03 11:13:07 +00:00
|
|
|
} else {
|
2009-07-04 03:59:42 +00:00
|
|
|
$offset++;
|
2009-07-03 11:13:07 +00:00
|
|
|
}
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
2009-07-03 11:13:07 +00:00
|
|
|
|
2009-07-04 03:59:42 +00:00
|
|
|
if(!$found){
|
|
|
|
push @old, @new;
|
|
|
|
}
|
2009-07-03 11:13:07 +00:00
|
|
|
|
2009-07-04 03:59:42 +00:00
|
|
|
open FILE, ">/etc/services.new" or return -1;
|
|
|
|
foreach (@old){
|
|
|
|
print FILE "$_\n" or return -1;
|
|
|
|
}
|
|
|
|
close FILE or return -1;
|
|
|
|
$cmd = "mv -f /etc/services.new /etc/services";
|
2009-06-17 06:57:37 +00:00
|
|
|
xCAT::Utils->runcmd($cmd, -2);
|
2009-07-04 03:59:42 +00:00
|
|
|
|
|
|
|
if(! -d $dir){
|
|
|
|
$cmd = "mkdir -p $dir";
|
|
|
|
xCAT::Utils->runcmd($cmd, -2);
|
|
|
|
} else {
|
|
|
|
$cmd = "rm -rf $dir/*";
|
|
|
|
xCAT::Utils->runcmd($cmd, -2);
|
|
|
|
}
|
|
|
|
@old = ();
|
|
|
|
@new = ();
|
|
|
|
@old = xCAT::Utils->runcmd("cat /etc/inetd.conf", -2);
|
|
|
|
$offset = 0;
|
|
|
|
$found = 0;
|
|
|
|
push @new, "rrdsrv stream tcp nowait root /usr/bin/rrdtool rrdtool - $dir";
|
|
|
|
foreach (@old) {
|
|
|
|
if ($_ =~ /rrdsrv/){
|
|
|
|
if(!$found){
|
|
|
|
splice(@old, $offset, 1, @new);
|
|
|
|
$found = 1;
|
|
|
|
} else {
|
|
|
|
splice(@old, $offset, 1);
|
|
|
|
}
|
2009-07-03 11:13:07 +00:00
|
|
|
} else {
|
2009-07-04 03:59:42 +00:00
|
|
|
$offset++;
|
2009-07-03 11:13:07 +00:00
|
|
|
}
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
2009-07-04 03:59:42 +00:00
|
|
|
if(!$found){
|
|
|
|
push @old, @new;
|
|
|
|
}
|
|
|
|
open FILE, ">/etc/inetd.conf.new" or return -1;
|
|
|
|
foreach (@old){
|
|
|
|
print FILE "$_\n" or return -1;
|
|
|
|
}
|
|
|
|
close FILE or return -1;
|
|
|
|
xCAT::Utils->runcmd("mv -f /etc/inetd.conf.new /etc/inetd.conf", -2);
|
2009-06-17 06:57:37 +00:00
|
|
|
|
|
|
|
xCAT::Utils->runcmd("stopsrc -s inetd", 0);
|
|
|
|
xCAT::Utils->runcmd("startsrc -s inetd", 0);
|
2009-07-04 03:59:42 +00:00
|
|
|
} elsif (xCAT::Utils->isLinux()){
|
|
|
|
if(-e "/etc/xinetd.d/rrdsrv"){
|
|
|
|
xCAT::Utils->runcmd("mv -f /etc/xinetd.d/rrdsrv /etc/xinetd.d/.rrdsrv.xcatbak",0)
|
|
|
|
}
|
|
|
|
open FILE, ">/etc/xinetd.d/rrdsrv" or return -1;
|
|
|
|
print FILE "# This is the configuration for the tcp/stream rrdsrv service.\n\n";
|
|
|
|
print FILE "service rrdsrv\n";
|
|
|
|
print FILE "{\n";
|
|
|
|
print FILE "\tdisable = no\n";
|
|
|
|
print FILE "\tport = 13900\n";
|
|
|
|
print FILE "\ttype = UNLISTED\n";
|
|
|
|
print FILE "\twait = no\n";
|
|
|
|
print FILE "\tsocket_type = stream\n";
|
|
|
|
print FILE "\tprotocol = tcp\n";
|
|
|
|
print FILE "\tuser = root\n";
|
|
|
|
print FILE "\tserver = /usr/bin/rrdtool\n";
|
|
|
|
print FILE "\tserver_args = - /var/rrd\n";
|
|
|
|
print FILE "}\n";
|
|
|
|
close FILE;
|
2014-06-10 13:11:07 +00:00
|
|
|
#xCAT::Utils->runcmd("service xinetd restart", 0);
|
|
|
|
xCAT::Utils->restartservice("xinetd");
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub stop_RRD_server
|
|
|
|
#Description:
|
2009-07-04 03:59:42 +00:00
|
|
|
# FOR AIX:
|
2009-06-17 06:57:37 +00:00
|
|
|
# remove "rrdsrv $port/tcp #RRD server" from /etc/services
|
|
|
|
# remove "rrdsrv stream tcp no wait root /usr/bin/rrdtool rrdtool - $dir"
|
|
|
|
# from /etc/inetd.conf
|
|
|
|
# restart inetd
|
2009-07-04 03:59:42 +00:00
|
|
|
# FOR LINUX:
|
|
|
|
# remove from xinetd
|
|
|
|
# restart xinetd
|
2009-06-17 06:57:37 +00:00
|
|
|
#Input:
|
|
|
|
# None
|
|
|
|
#return:
|
|
|
|
# 0 success
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
sub stop_RRD_server
|
|
|
|
{
|
2009-07-04 03:59:42 +00:00
|
|
|
if(xCAT::Utils->isAIX()){
|
|
|
|
my @old = ();
|
|
|
|
my $offset = 0;
|
|
|
|
@old = xCAT::Utils->runcmd("cat /etc/services", -2);
|
|
|
|
foreach (@old) {
|
|
|
|
if ($_ =~ /rrdsrv/){
|
|
|
|
splice(@old, $offset, 1);
|
|
|
|
} else {
|
|
|
|
$offset++;
|
|
|
|
}
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
2009-07-04 03:59:42 +00:00
|
|
|
open FILE, ">/etc/services.new" or return -1;
|
|
|
|
foreach (@old){
|
|
|
|
print FILE "$_\n" or return -1;
|
|
|
|
}
|
|
|
|
close FILE or return -1;
|
|
|
|
xCAT::Utils->runcmd("mv -f /etc/services.new /etc/services", -1);
|
2009-06-17 06:57:37 +00:00
|
|
|
|
2009-07-04 03:59:42 +00:00
|
|
|
@old = ();
|
|
|
|
@old = xCAT::Utils->runcmd("cat /etc/inetd.conf", -1);
|
|
|
|
$offset = 0;
|
|
|
|
foreach (@old) {
|
|
|
|
if ($_ =~ /rrdsrv/){
|
|
|
|
splice(@old, $offset, 1);
|
|
|
|
} else {
|
|
|
|
$offset++;
|
|
|
|
}
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
2009-07-04 03:59:42 +00:00
|
|
|
open FILE, ">/etc/inetd.conf.new" or return -1;
|
|
|
|
foreach (@old){
|
|
|
|
print FILE "$_\n" or return -1;
|
|
|
|
}
|
|
|
|
close FILE or return -1;
|
|
|
|
xCAT::Utils->runcmd("mv -f /etc/inetd.conf.new /etc/inetd.conf", -2);
|
2009-06-17 06:57:37 +00:00
|
|
|
xCAT::Utils->runcmd("stopsrc -s inetd", 0);
|
|
|
|
xCAT::Utils->runcmd("startsrc -s inetd", 0);
|
2009-07-04 03:59:42 +00:00
|
|
|
} elsif (xCAT::Utils->isLinux()){
|
|
|
|
if(-e "/etc/xinetd.d/.rrdsrv.xcatbak"){
|
|
|
|
xCAT::Utils->runcmd("mv -f /etc/xinetd.d/.rrdsrv.xcatbak /etc/xinetd.d/rrdsrv", 0);
|
|
|
|
} else {
|
|
|
|
xCAT::Utils->runcmd("rm -f /etc/xinetd.d/rrdsrv", 0);
|
|
|
|
}
|
2014-06-10 13:11:07 +00:00
|
|
|
#xCAT::Utils->runcmd("service xinetd restart", 0);
|
|
|
|
xCAT::Utils->restartservice("xinetd");
|
2009-06-17 06:57:37 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub runrrdcmd_remote
|
|
|
|
#Description:
|
|
|
|
# Run the given RRD cmd on remote RRD server,
|
|
|
|
# and return the output in an array.
|
|
|
|
#Input:
|
|
|
|
# $cmd RRD command
|
|
|
|
# $host IP or hostname of RRD server
|
|
|
|
# $port Port number of RRD server,
|
|
|
|
#return:
|
|
|
|
# output of command
|
|
|
|
################################################
|
|
|
|
sub runrrdcmd_remote
|
|
|
|
{
|
|
|
|
my($cmd, $host, $port) = @_;
|
|
|
|
|
|
|
|
my $socket = IO::Socket::INET->new
|
|
|
|
(PeerAddr=>$host,
|
|
|
|
PeerPort=>$port,
|
|
|
|
Proto=>'tcp',
|
|
|
|
Type=>SOCK_STREAM);
|
|
|
|
|
|
|
|
if(! $socket){
|
|
|
|
print "ERROR: to connect with $host:$port\n";
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
print $socket "$cmd\n";
|
|
|
|
my $output = [];
|
|
|
|
my $line = undef;
|
|
|
|
while($line = <$socket>){
|
|
|
|
push @$output, $line;
|
|
|
|
if($line =~ /(OK|ERROR)/){
|
|
|
|
last;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
print $socket "quit";
|
|
|
|
close($socket);
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub RRD_create
|
|
|
|
#Description:
|
|
|
|
# RRD_create will overwrite a RRdb if it already exists,
|
|
|
|
#Input:
|
|
|
|
# $rrd filename of RRD to create
|
|
|
|
# $sum if $sum != 0 to create a database file for SUMMARY information
|
|
|
|
# $step the base interval in seconds with which data will be fed into the RRD
|
|
|
|
# $start_time the time in seconds when the first value should be added to the RRD
|
|
|
|
# $ds_type ds-name:GAUGE | COUNTER | DERIVE | ABSOLUTE
|
|
|
|
# $data_source the IP or hostname of remote host
|
|
|
|
#return:
|
|
|
|
# 0 sucess
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
sub RRD_create
|
|
|
|
{
|
|
|
|
my ($rrd, $sum, $step, $start_time, $ds_type, $data_source) = @_;
|
|
|
|
my $output = [];
|
|
|
|
my $heartbeat = 8 * $step;
|
|
|
|
my $cmd = "create $rrd --start $start_time --step $step DS:sum:$ds_type:$heartbeat:U:U";
|
|
|
|
if($sum){
|
|
|
|
$cmd = $cmd." DS:num:$ds_type:$heartbeat:U:U";
|
|
|
|
}
|
|
|
|
#TODO: Specified custom RR archives here?
|
|
|
|
$cmd = $cmd." RRA:AVERAGE:0.5:1:244 RRA:AVERAGE:0.5:24:244 RRA:AVERAGE:0.5:168:244 RRA:AVERAGE:0.5:672:244 RRA:AVERAGE:0.5:5760:374";
|
|
|
|
if(defined($data_source)){
|
|
|
|
$output = &runrrdcmd_remote($cmd, $data_source, 13900);
|
|
|
|
} else {
|
|
|
|
@$output = xCAT::Utils->runcmd("rrdtool $cmd", 0);
|
|
|
|
}
|
|
|
|
my $line = pop(@$output);
|
|
|
|
if($line =~ /ERROR/){
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub RRD_update
|
|
|
|
#Description:
|
|
|
|
# RRD_update
|
|
|
|
#Input:
|
|
|
|
# $rrd filename of RRD to update
|
|
|
|
# $sum sum of all numeric metrics
|
|
|
|
# $num number of all numeric metrics, should be null for a host metrics
|
|
|
|
# $process_time update time
|
|
|
|
# $data_source the IP or hostname of remote host
|
|
|
|
#return:
|
|
|
|
# 0 sucess
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
sub RRD_update
|
|
|
|
{
|
|
|
|
my ($rrd, $sum, $num, $process_time, $data_source) = @_;
|
|
|
|
my $output = [];
|
|
|
|
my $cmd = "update $rrd";
|
|
|
|
if($num ne "null"){
|
|
|
|
$cmd = $cmd." $process_time:$sum:$num";
|
|
|
|
} else {
|
|
|
|
$cmd = $cmd." $process_time:$sum";
|
|
|
|
}
|
|
|
|
if(defined($data_source)){
|
|
|
|
$output = &runrrdcmd_remote($cmd, $data_source, 13900);
|
|
|
|
} else {
|
|
|
|
@$output = xCAT::Utils->runcmd("rrdtool $cmd", 0);
|
|
|
|
}
|
|
|
|
my $line = pop(@$output);
|
|
|
|
if($line =~ /ERROR/){
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub RRD_fetch
|
|
|
|
#Description:
|
|
|
|
# RRD_fetch
|
|
|
|
#Input:
|
|
|
|
# $rrd filename of RRD to update
|
|
|
|
# $start_time start of time series
|
|
|
|
# $end_time end of time series
|
|
|
|
# $data_source the IP or hostname of remote host
|
|
|
|
#return:
|
|
|
|
# 0 sucess
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
sub RRD_fetch
|
|
|
|
{
|
|
|
|
my ($rrd, $start_time, $end_time, $data_source) = @_;
|
|
|
|
my $output = [];
|
|
|
|
my $resolution = undef;
|
|
|
|
my $cmd = undef;
|
|
|
|
$cmd = "fetch $rrd AVERAGE -s $start_time -e $end_time";
|
|
|
|
if(defined($data_source)){
|
|
|
|
$output = &runrrdcmd_remote($cmd, $data_source, 13900);
|
|
|
|
} else {
|
|
|
|
@$output = xCAT::Utils->runcmd("rrdtool $cmd", 0);
|
|
|
|
}
|
|
|
|
return $output;
|
|
|
|
}
|
|
|
|
|
|
|
|
################################################
|
|
|
|
#sub push_data_to_rrd
|
|
|
|
#Description:
|
|
|
|
# push_data_to_rrd
|
|
|
|
#Input:
|
|
|
|
# $rrd filename of RRD to update
|
|
|
|
# $sum sum of all numeric metrics
|
|
|
|
# $num number of all numeric metrics, should be null for a host metrics
|
|
|
|
# $step the base interval in seconds with which data will be fed into the RRD
|
|
|
|
# $process_time update time
|
|
|
|
# $ds_type ds-name:GAUGE | COUNTER | DERIVE | ABSOLUTE
|
|
|
|
# $data_source the IP or hostname of remote host
|
|
|
|
#return:
|
|
|
|
# 0 sucess
|
|
|
|
# !0 fail
|
|
|
|
################################################
|
|
|
|
|
|
|
|
sub push_data_to_rrd
|
|
|
|
{
|
|
|
|
my $ret = 0;
|
|
|
|
my($rrd, $sum, $num, $step, $process_time, $ds_type, $data_source) = @_;
|
|
|
|
my $summary = $num eq 'null' ? 0 : 1;
|
|
|
|
if(! -f $rrd){
|
|
|
|
$ret = RRD_create($rrd, $summary, $step, $process_time-$step, $ds_type, $data_source);
|
|
|
|
if($ret != 0){
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$ret = RRD_update($rrd, $sum, $num, $process_time, $data_source);
|
|
|
|
return $ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
1;
|