Monitorix/lib/Monitorix.pm

304 lines
6.8 KiB
Perl
Raw Normal View History

2013-01-04 08:27:05 +00:00
#
# Monitorix - A lightweight system monitoring tool.
#
# Copyright (C) 2005-2013 by Jordi Sanfeliu <jordi@fibranet.cat>
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along
# with this program; if not, write to the Free Software Foundation, Inc.,
# 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
#
package Monitorix;
use strict;
use warnings;
use Exporter 'import';
2013-01-31 17:48:29 +00:00
use POSIX qw(setuid setgid setsid);
our @EXPORT = qw(logger trim max to_fahrenheit httpd_setup get_nvidia_data flush_accounting_rules);
2013-01-04 08:27:05 +00:00
sub logger {
my ($msg) = @_;
$msg = localtime() . " - " . $msg;
print("$msg\n");
}
sub trim {
my $str = shift;
if($str) {
$str =~ s/^\s+//;
$str =~ s/\s+$//;
return $str;
}
}
2013-01-28 12:18:03 +00:00
sub max {
my ($max, @args) = @_;
foreach(@args) {
2013-01-28 12:18:03 +00:00
$max = $_ if $_ > $max;
}
return $max;
}
sub to_fahrenheit {
my ($config, $celsius) = @_;
$celsius = $celsius || 0;
if(lc($config->{temperatures_scale}) eq "f") {
return ($celsius * (9 / 5)) + 32;
}
return $celsius;
}
2013-01-31 17:48:29 +00:00
sub httpd_setup {
my ($config, $debug) = @_;
my $pid;
2013-02-07 14:57:42 +00:00
my (undef, undef, $uid) = getpwnam($config->{httpd_builtin}->{user});
my (undef, undef, $gid) = getgrnam($config->{httpd_builtin}->{group});
my $port = $config->{httpd_builtin}->{port};
if(!defined($uid)) {
2013-03-15 06:57:09 +00:00
logger("ERROR: invalid user defined for the built-in HTTP server.");
2013-02-07 14:57:42 +00:00
return;
}
if(!defined($gid)) {
2013-03-15 06:57:09 +00:00
logger("ERROR: invalid group defined for the built-in HTTP server.");
2013-02-07 14:57:42 +00:00
return;
}
if(!defined($port)) {
2013-03-15 06:57:09 +00:00
logger("ERROR: invalid port defined for the built-in HTTP server.");
2013-02-07 14:57:42 +00:00
return;
}
2013-01-31 17:48:29 +00:00
if($pid = fork()) {
$config->{httpd_pid} = $pid;
return; # parent returns
}
# create the HTTPd logfile
open(OUT, ">> " . $config->{httpd_builtin}->{log_file});
close(OUT);
chown($uid, $gid, $config->{httpd_builtin}->{log_file});
2013-01-31 17:48:29 +00:00
setgid($gid);
setuid($uid);
setsid();
$SIG{$_} = 'DEFAULT' for keys %SIG; # reset all sighandlers
$0 = "monitorix-httpd listening on $port"; # change process' name
chdir($config->{base_dir});
my $server = HTTPServer->new($port);
2013-02-11 11:36:31 +00:00
$server->run();
exit(0);
2013-01-31 17:48:29 +00:00
}
2013-01-04 08:27:05 +00:00
sub get_nvidia_data {
my $myself = (caller(0))[3];
2013-01-04 08:27:05 +00:00
my ($gpu) = @_;
my $total = 0;
my $used = 0;
my $mem = 0;
my $cpu = 0;
my $temp = 0;
my $check_mem = 0;
my $check_cpu = 0;
my $check_temp = 0;
my $l;
my @data = ();
if(open(IN, "nvidia-smi -q -i $gpu -d MEMORY,UTILIZATION,TEMPERATURE |")) {
@data = <IN>;
close(IN);
} else {
logger("$myself: ERROR: 'nvidia-smi' command is not installed.");
}
2013-01-04 08:27:05 +00:00
for($l = 0; $l < scalar(@data); $l++) {
if($data[$l] =~ /Memory Usage/) {
$check_mem = 1;
next;
}
if($check_mem) {
if($data[$l] =~ /Total/) {
my (undef, $tmp) = split(':', $data[$l]);
if($tmp eq "\n") {
$l++;
$tmp = $data[$l];
}
my ($value, undef) = split(' ', $tmp);
$value =~ s/[-]/./;
$value =~ s/[^0-9.]//g;
if(int($value) > 0) {
$total = int($value);
}
}
if($data[$l] =~ /Used/) {
my (undef, $tmp) = split(':', $data[$l]);
if($tmp eq "\n") {
$l++;
$tmp = $data[$l];
}
my ($value, undef) = split(' ', $tmp);
$value =~ s/[-]/./;
$value =~ s/[^0-9.]//g;
if(int($value) > 0) {
$used = int($value);
}
$check_mem = 0;
}
}
if($data[$l] =~ /Utilization/) {
$check_cpu = 1;
next;
}
if($check_cpu) {
if($data[$l] =~ /Gpu/) {
my (undef, $tmp) = split(':', $data[$l]);
if($tmp eq "\n") {
$l++;
$tmp = $data[$l];
}
my ($value, undef) = split(' ', $tmp);
$value =~ s/[-]/./;
$value =~ s/[^0-9.]//g;
if(int($value) > 0) {
$cpu = int($value);
}
}
if($data[$l] =~ /Memory/) {
my (undef, $tmp) = split(':', $data[$l]);
if($tmp eq "\n") {
$l++;
$tmp = $data[$l];
}
my ($value, undef) = split(' ', $tmp);
$value =~ s/[-]/./;
$value =~ s/[^0-9.]//g;
if(int($value) > 0) {
$mem = int($value);
}
}
$check_cpu = 0;
}
if($data[$l] =~ /Temperature/) {
$check_temp = 1;
next;
}
if($check_temp) {
if($data[$l] =~ /Gpu/) {
my (undef, $tmp) = split(':', $data[$l]);
if($tmp eq "\n") {
$l++;
$tmp = $data[$l];
}
my ($value, undef) = split(' ', $tmp);
$value =~ s/[-]/./;
$value =~ s/[^0-9.]//g;
if(int($value) > 0) {
$temp = int($value);
}
}
$check_temp = 0;
}
}
# NVIDIA driver v285.+ not supported (needs new output parsing).
# This is to avoid a divide by zero message.
if($total) {
$mem = ($used * 100) / $total;
} else {
$mem = $used = $total = 0;
}
return join(" ", $mem, $cpu, $temp);
}
# flushes out all Monitorix iptables/ipfw rules
sub flush_accounting_rules {
my ($config, $debug) = @_;
if($config->{os} eq "Linux") {
my $num = 0;
logger("Flushing out iptables rules.") if $debug;
{
2013-01-04 08:27:05 +00:00
my @names;
if(open(IN, "iptables -nxvL INPUT --line-numbers |")) {
my @rules;
while(<IN>) {
my ($rule, undef, undef, $name) = split(' ', $_);
if($name =~ /monitorix_IN/ || /monitorix_OUT/ || /monitorix_nginx_IN/) {
push(@rules, $rule);
push(@names, $name);
}
2013-01-04 08:27:05 +00:00
}
close(IN);
@rules = reverse(@rules);
foreach(@rules) {
system("iptables -D INPUT $_");
$num++;
2013-01-04 08:27:05 +00:00
}
}
if(open(IN, "iptables -nxvL OUTPUT --line-numbers |")) {
my @rules;
while(<IN>) {
my ($rule, undef, undef, $name) = split(' ', $_);
if($name =~ /monitorix_IN/ || /monitorix_OUT/ || /monitorix_nginx_IN/) {
push(@rules, $rule);
}
}
close(IN);
@rules = reverse(@rules);
foreach(@rules) {
system("iptables -D OUTPUT $_");
$num++;
}
2013-01-04 08:27:05 +00:00
}
foreach(@names) {
system("iptables -X $_");
}
}
if(open(IN, "iptables -nxvL FORWARD --line-numbers |")) {
my @rules;
my @names;
while(<IN>) {
my ($rule, undef, undef, $name) = split(' ', $_);
if($name =~ /monitorix_daily_/ || /monitorix_total_/) {
push(@rules, $rule);
push(@names, $name);
}
}
close(IN);
@rules = reverse(@rules);
foreach(@rules) {
system("iptables -D FORWARD $_");
$num++;
}
foreach(@names) {
system("iptables -F $_");
system("iptables -X $_");
}
}
2013-01-04 08:27:05 +00:00
logger("$num iptables rules have been flushed.") if $debug;
}
if(grep {$_ eq $config->{os}} ("FreeBSD", "OpenBSD", "NetBSD")) {
logger("Flushing out ipfw rules.") if $debug;
system("ipfw delete $config->{port}->{rule} 2>/dev/null");
system("ipfw delete $config->{nginx}->{rule} 2>/dev/null");
}
}
1;