version 1.24, 2007/02/14 21:59:10 |
version 1.32, 2009/11/10 19:46:06 |
|
|
#!/usr/bin/perl -T |
#!/usr/bin/perl -T |
# $RedRiver: check_hw_sensors,v 1.23 2007/01/22 02:38:35 andrew Exp $ |
# $RedRiver: check_hw_sensors,v 1.31 2009/11/10 18:24:56 andrew Exp $ |
######################################################################## |
######################################################################## |
# check_hw_sensors *** A nagios check for OpenBSD hw.sensors |
# check_hw_sensors *** A nagios check for OpenBSD sysctl hw.sensors |
# |
# |
# 2006.05.01 #*#*# andrew fresh <andrew@mad-techies.org> |
# 2006.05.01 #*#*# andrew fresh <andrew@afresh1.com> |
######################################################################## |
######################################################################## |
# TODO: |
|
# Really need real documentation. |
|
# allow checking of hw.sensors on a remote host with ssh somehow |
|
######################################################################## |
|
use strict; |
use strict; |
use warnings; |
use warnings; |
|
|
|
|
|
|
use constant NAGIOS_OUTPUT => 1; |
use constant NAGIOS_OUTPUT => 1; |
|
|
|
my $License = <<'EOL'; |
|
Copyright (c) 2009 Andrew Fresh <andrew@afresh1.com> |
|
Permission to use, copy, modify, and distribute this software for any |
|
purpose with or without fee is hereby granted, provided that the above |
|
copyright notice and this permission notice appear in all copies. |
|
|
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES |
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF |
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR |
|
ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN |
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF |
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
|
EOL |
|
|
use POSIX; |
use POSIX; |
use Config; |
use Config; |
use lib "/usr/local/libexec/nagios"; |
use lib "/usr/local/libexec/nagios"; |
use utils qw($TIMEOUT %ERRORS &print_revision &support); |
use utils qw($TIMEOUT %ERRORS &support); |
|
|
use Getopt::Long; |
use Getopt::Long; |
Getopt::Long::Configure('bundling'); |
Getopt::Long::Configure('bundling'); |
|
|
my $PROGNAME = "check_hw_sensors"; |
my $PROGNAME = "check_hw_sensors"; |
|
|
my $SYSCTL = '/sbin/sysctl'; |
my $SYSCTL = '/sbin/sysctl'; |
my $GETCAP = '/usr/bin/getcap'; |
my $GETCAP = '/usr/bin/getcap'; |
my $BASE = 'hw.sensors'; |
my $BASE = 'hw.sensors'; |
my $DEFAULT_CONFIG = '/etc/sensorsd.conf'; |
my $DEFAULT_CONFIG = '/etc/sensorsd.conf'; |
my $OSVer = $Config{'osvers'} || 0; |
my $OSVer = $Config{'osvers'} || 0; |
|
|
my $state = 'UNKNOWN'; # tells whether the it is warning, critical, or OK |
my $state = 'UNKNOWN'; # tells whether the it is warning, critical, or OK |
my %states; # This stores the count of states; |
my %states; # This stores the count of states; |
my $filename; |
my $opt_V; |
|
my $opt_h; |
my $ignore_status; |
my $ignore_status; |
|
my $filename; |
my $sensor; |
my $sensor; |
my $warning; |
my $warning; |
my $critical; |
my $critical; |
my $opt_h; |
|
my $opt_V; |
|
|
|
my $CHECK_SENSOR = $BASE; |
my $CHECK_SENSOR = $BASE; |
my %CHECKS; |
my %CHECKS; |
my %SENSORS; |
my @SENSORS; |
|
|
#Option checking |
#Option checking |
my $status = GetOptions( |
my $status = GetOptions( |
"version|V" => \$opt_V, |
"version|V" => \$opt_V, |
"filename|f:s" => \$filename, |
"help|h" => \$opt_h, |
"ignore-status|i" => \$ignore_status, |
"ignore-status|i" => \$ignore_status, |
"sensor|s=s" => \$sensor, |
"filename|f:s" => \$filename, |
"warning|w=s" => \$warning, |
"sensor|s=s" => \$sensor, |
"critical|c=s" => \$critical, |
"warning|w=s" => \$warning, |
|
"critical|c=s" => \$critical, |
); |
); |
|
|
# set the default this way so it only happens if someone typed -f or --filename |
if ( $status == 0 ) { |
$filename = $DEFAULT_CONFIG if (defined $filename && $filename eq ''); |
print_help(); |
|
exit $ERRORS{'OK'}; |
# Stuff is output in this file by print_sensor() |
|
# http://www.openbsd.org/cgi-bin/cvsweb/src/sbin/sysctl/sysctl.c |
|
my @Type_Map = ( |
|
{ |
|
type => 'temp', |
|
regex => qr/\sdegC$/, |
|
}, |
|
{ |
|
type => 'fanrpm', |
|
regex => qr/\sRPM$/, |
|
}, |
|
{ |
|
type => 'volts_dc', |
|
regex => qr/\sV\sDC$/, |
|
}, |
|
{ |
|
type => 'amps', |
|
regex => qr/\sA$/, |
|
}, |
|
{ |
|
type => 'watthour', |
|
regex => qr/\sWh$/, |
|
}, |
|
{ |
|
type => 'amphour', |
|
regex => qr/\sAh$/, |
|
}, |
|
{ |
|
type => 'indicator', |
|
regex => qr/^(On|Off)$/, |
|
}, |
|
{ |
|
type => 'integer', |
|
regex => qr/\sraw$/, |
|
}, |
|
{ |
|
type => 'percent', |
|
regex => qr/\s\%$/, |
|
}, |
|
{ |
|
type => 'lux', |
|
regex => qr/\slx$/, |
|
}, |
|
{ |
|
type => 'drive', |
|
regex => qr/^drive\s/, |
|
}, |
|
{ |
|
type => 'timedelta', |
|
regex => qr/\ssecs$/, |
|
}, |
|
); |
|
|
|
if ($status == 0) { |
|
print_help() ; |
|
exit $ERRORS{'OK'}; |
|
} |
} |
|
|
if ($opt_V) { |
if ($opt_V) { |
print_revision($PROGNAME,'$Revision$ '); |
print_revision( $PROGNAME, '$Revision$ ' ); |
exit $ERRORS{'OK'}; |
exit $ERRORS{'OK'}; |
} |
} |
|
|
unless ( |
if ($opt_h) { |
( |
print_help(); |
defined $filename || |
exit $ERRORS{'OK'}; |
(not defined $ignore_status) || |
|
(defined $sensor && ($warning || $critical)) |
|
) && |
|
( (!defined $filename) || (!defined $sensor) ) |
|
) { |
|
print_help(); |
|
exit $ERRORS{'OK'}; |
|
} |
} |
|
|
|
# set the default this way so it only happens if someone typed -f or --filename |
|
$filename = $DEFAULT_CONFIG if ( defined $filename && $filename eq '' ); |
|
|
if (defined $sensor) { |
# Stuff is output in this file by print_sensor() |
if ($sensor !~ /^$BASE/) { |
# http://www.openbsd.org/cgi-bin/cvsweb/src/sbin/sysctl/sysctl.c |
$sensor = $BASE . '.' . $sensor; |
my @Type_Map = ( |
} |
{ type => 'temp', |
$CHECK_SENSOR = $sensor; |
regex => qr/\sdegC$/, |
|
}, |
|
{ type => 'fanrpm', |
|
regex => qr/\sRPM$/, |
|
}, |
|
{ type => 'volts_dc', |
|
regex => qr/\sV\sDC$/, |
|
}, |
|
{ type => 'amps', |
|
regex => qr/\sA$/, |
|
}, |
|
{ type => 'watthour', |
|
regex => qr/\sWh$/, |
|
}, |
|
{ type => 'amphour', |
|
regex => qr/\sAh$/, |
|
}, |
|
{ type => 'indicator', |
|
regex => qr/^(On|Off)$/, |
|
}, |
|
{ type => 'integer', |
|
regex => qr/\sraw$/, |
|
}, |
|
{ type => 'percent', |
|
regex => qr/\s\%$/, |
|
}, |
|
{ type => 'lux', |
|
regex => qr/\slx$/, |
|
}, |
|
{ type => 'drive', |
|
regex => qr/^drive\s/, |
|
}, |
|
{ type => 'timedelta', |
|
regex => qr/\ssecs$/, |
|
}, |
|
); |
|
|
$CHECKS{$sensor}{'warn'} = $warning if defined $warning; |
if ( defined $sensor ) { |
$CHECKS{$sensor}{'crit'} = $critical if defined $critical; |
if ( $sensor !~ /^$BASE/ ) { |
|
$sensor = $BASE . '.' . $sensor; |
|
} |
|
$CHECK_SENSOR = $sensor; |
|
|
} elsif (defined $filename) { |
$CHECKS{$sensor}{'warn'} = $warning; |
%CHECKS = parse_file($filename); |
$CHECKS{$sensor}{'crit'} = $critical; |
|
|
} |
} |
|
elsif ( defined $filename ) { |
|
%CHECKS = parse_file($filename); |
|
} |
|
|
open my $sysctl, "-|", $SYSCTL, $CHECK_SENSOR |
open my $sysctl, "-|", $SYSCTL, $CHECK_SENSOR |
or die "Couldn't open sysctl: $!"; |
or die "Couldn't open sysctl: $!"; |
while (<$sysctl>) { |
while (<$sysctl>) { |
#while (<>) { |
chomp; |
chomp; |
my ( $id, $output ) = split /=/xms; |
my ($id, $output) = split /=/; |
my @s = split /\./xms, $id; |
my @s = split /\./, $id; |
my @o = split /,\s*/xms, $output; |
my @o = split /,\s*/, $output; |
|
|
|
my ($type, $source, $descr, $data, $status); |
my ( $type, $source, $descr, $data, $status ); |
|
|
$source = $o[0]; |
$source = $o[0]; |
$descr = $o[1]; |
$descr = $o[1]; |
|
|
if (@s == 4) { # XXX This mebbe needs to end up $OSVer >= 4.1 |
if ( $OSVer >= 4.1 ) { |
$data = $o[0]; |
$data = $o[0]; |
if ($data =~ s/\s+\((.*)\).*$//) { |
if ( $data =~ s/\s+\((.*)\).*$//xms ) { |
$descr = $1; |
$descr = $1; |
} |
} |
$status = $o[1]; |
$status = $o[1]; |
($source, $type) = $id =~ /([^\.]+)\.([^\.]+)\d+$/; |
( $source, $type ) = $id =~ /([^\.]+)\.([^\.]+?)\d+$/xms; |
} elsif ($OSVer >= 4.0) { |
} |
$data = $o[2]; |
elsif ( $OSVer >= 4.0 ) { |
$status = $o[3]; |
$data = $o[2]; |
foreach my $t (@Type_Map) { |
$status = $o[3]; |
if ($data =~ /$t->{'regex'}/) { |
foreach my $t (@Type_Map) { |
$type = $t->{'type'}; |
if ( $data =~ /$t->{'regex'}/xms ) { |
last; |
$type = $t->{'type'}; |
} |
last; |
} |
} |
} else { |
} |
$data = $o[-1]; |
} |
$status = $o[2] if @o == 5; |
else { |
$type = $o[-2]; |
$data = $o[-1]; |
} |
$status = $o[2] if @o == 5; |
|
$type = $o[-2]; |
|
} |
|
|
$type ||= 'unknown'; |
$type ||= 'unknown'; |
|
|
$SENSORS{$id} = { |
push @SENSORS, |
id => $id, |
{ |
output => $output, |
id => $id, |
source => $source, |
output => $output, |
description => $descr, |
source => $source, |
status => $status, |
description => $descr, |
type => $type, |
status => $status, |
data => $data, |
type => $type, |
}; |
data => $data, |
|
}; |
} |
} |
close $sysctl; |
close $sysctl; |
|
|
sub as_if_numeric { |
foreach my $sensor (@SENSORS) { |
my $_a = $a; |
my ( $r, $data ); |
my $_b = $b; |
if ( exists $CHECKS{ $sensor->{id} } ) { |
$_a =~ s/\D//g; |
$r = check_sensor( $sensor, $CHECKS{ $sensor->{id} } ); |
$_b =~ s/\D//g; |
$data = $sensor->{id} . '=' . $sensor->{output}; |
$_a <=> $_b; |
} |
} |
elsif ( !$ignore_status && $sensor->{status} ) { |
|
$r = check_sensor( $sensor, { STATUS => 1 } ); |
|
$data = $sensor->{id} . '=' . $sensor->{output}; |
|
} |
|
else { |
|
|
foreach my $s (sort as_if_numeric keys %SENSORS) { |
# ignore this sensor, theoretically you could do the check and |
my ($r, $data); |
# that would show unknown sensors. |
if (exists $CHECKS{$s}) { |
} |
$r = check_sensor($SENSORS{$s}, $CHECKS{$s}); |
if ( defined $r ) { |
$data = $s . '=' . $SENSORS{$s}{'output'}; |
push @{ $states{$r} }, $data; |
} elsif (not $ignore_status) { |
} |
$r = check_sensor($SENSORS{$s}); |
|
$data = $s . '=' . $SENSORS{$s}{'output'}; |
|
} else { |
|
# ignore this sensor |
|
} |
|
next unless defined $r; |
|
push @{ $states{ $r } }, $data; |
|
} |
} |
|
|
$state = 'OK'; |
|
my $have_results = 0; |
my $have_results = 0; |
foreach my $error (sort { $ERRORS{$a} <=> $ERRORS{$b} } keys %ERRORS) { |
$state = 'OK'; |
if (exists $states{$error}) { |
foreach my $error ( sort { $ERRORS{$b} <=> $ERRORS{$a} } keys %ERRORS ) { |
$have_results++; |
if ( exists $states{$error} ) { |
$state = $error; |
$have_results++; |
} |
$state = $error if $ERRORS{$state} < $ERRORS{$error}; |
|
|
|
if (NAGIOS_OUTPUT) { |
|
print "$error (" . scalar( @{ $states{$error} } ) . ")"; |
|
if ( $error ne 'OK' ) { |
|
print '<br>'; |
|
print map {" - $_<br>"} @{ $states{$error} }; |
|
} |
|
} |
|
else { |
|
print "$error (" . scalar( @{ $states{$error} } ) . "):\n"; |
|
foreach ( @{ $states{$error} } ) { |
|
print " $_\n"; |
|
} |
|
} |
|
} |
} |
} |
foreach my $error (sort { $ERRORS{$b} <=> $ERRORS{$a} } keys %ERRORS) { |
if ( $have_results == 0 ) { |
if (exists $states{$error}) { |
print "No results found\n"; |
if (NAGIOS_OUTPUT) { |
|
print "$error (" . scalar(@{ $states{ $error } }) . ")"; |
|
unless ($error eq 'OK') { |
|
print '<br>'; |
|
print map {" - $_<br>"} @{ $states{ $error } }; |
|
} |
|
} else { |
|
print "$error (" . scalar(@{ $states{ $error } }) . "):\n"; |
|
foreach (@{ $states{ $error } }) { |
|
print " $_\n"; |
|
} |
|
} |
|
} |
|
} |
} |
if ($have_results == 0) { |
|
print "No results found\n"; |
|
} |
|
exit $ERRORS{$state}; |
exit $ERRORS{$state}; |
|
|
|
|
sub parse_file { |
sub parse_file { |
my $filename = shift; |
my $filename = shift; |
my %contents; |
my %contents; |
|
|
die "file '$filename' does not exist." unless -e $filename; |
|
|
|
open my $fh, '-|', $GETCAP, '-a', '-f', $filename |
die "file '$filename' does not exist." unless -e $filename; |
or die "Couldn't open FILE '$GETCAP -a -f $filename': $!"; |
|
LINE: while (<$fh>) { |
|
chomp; |
|
my ($key, @c) = split /\:/; |
|
foreach (@c) { |
|
my ($k, $v) = split /\=/; |
|
if (lc($k) eq 'ignore') { |
|
$contents{$key}{'IGNORE'} = 1; |
|
} elsif (lc($k) eq 'status') { |
|
$contents{$key}{'STATUS'} = 1; |
|
} else { |
|
$contents{$key}{$k} = $v; |
|
} |
|
} |
|
} |
|
close $fh; |
|
|
|
return %contents; |
open my $fh, '-|', $GETCAP, '-a', '-f', $filename |
|
or die "Couldn't open FILE '$GETCAP -a -f $filename': $!"; |
|
while (<$fh>) { |
|
chomp; |
|
my ( $key, @c ) = split /\:/; |
|
foreach (@c) { |
|
my ( $k, $v ) = split /\=/; |
|
if ( lc($k) eq 'ignore' ) { |
|
$contents{$key}{'IGNORE'} = 1; |
|
} |
|
elsif ( lc($k) eq 'status' ) { |
|
$contents{$key}{'STATUS'} = 1; |
|
} |
|
else { |
|
$contents{$key}{$k} = $v; |
|
} |
|
} |
|
} |
|
close $fh; |
|
|
|
return %contents; |
} |
} |
|
|
sub parse_check { |
sub parse_check { |
my $type = shift; |
my $check = shift; |
my $check = shift; |
|
|
|
return undef unless $check; |
return unless $check; |
return undef if $check->{'STATUS'}; |
return 'STATUS' if $check->{'STATUS'}; |
return 'IGNORE' if $check->{'IGNORE'}; |
return 'IGNORE' if $check->{'IGNORE'}; |
|
|
foreach my $code ('crit', 'warn') { |
foreach my $code ( 'crit', 'warn' ) { |
if (defined $check->{$code} && $check->{$code} =~ /:/) { |
if ( defined $check->{$code} && $check->{$code} =~ /:/ ) { |
if (my ($low, $high) = split /:/, $check->{$code}) { |
if ( my ( $low, $high ) = split /:/, $check->{$code} ) { |
$check->{$code . '.low'} = $low; |
$check->{ $code . '.low' } = $low if length $low; |
$check->{$code . '.high'} = $high; |
$check->{ $code . '.high' } = $high if length $high; |
} |
} |
delete $check->{$code}; |
delete $check->{$code}; |
} |
} |
|
|
foreach my $severity ('low', 'high') { |
foreach my $severity ( 'low', 'high' ) { |
if (defined $check->{$severity}) { |
if ( defined $check->{$severity} ) { |
$check->{ $code . '.' . $severity } = $check->{$severity} |
$check->{ $code . '.' . $severity } = $check->{$severity} |
unless defined $check->{ $code . '.' . $severity }; |
if !defined $check->{ $code . '.' . $severity }; |
} |
} |
} |
} |
no warnings 'uninitialized'; |
no warnings 'uninitialized'; |
$check->{$code} = [ split /,\s*/, $check->{$code} ]; |
$check->{$code} = [ split /,\s*/, $check->{$code} ]; |
} |
} |
|
|
return $check; |
return $check; |
} |
} |
|
|
sub check_sensor { |
sub check_sensor { |
my $sensor = shift; |
my $sensor = shift; |
my $check = shift; |
my $check = shift; |
my $result = 'UNKNOWN'; |
my $result = 'UNKNOWN'; |
my %errors = ( |
my %errors = ( |
'warn' => 'WARNING', |
'warn' => 'WARNING', |
'crit' => 'CRITICAL', |
'crit' => 'CRITICAL', |
); |
); |
|
|
return $result unless ref $sensor eq 'HASH'; |
return $result unless ref $sensor eq 'HASH'; |
$check = parse_check($sensor->{'type'}, $check) if $check; |
$check = parse_check($check) if $check; |
|
|
# It looks like doing this should be safe, from |
if ( !$check ) { |
# src/sbin/sysctl/sysctl.c |
return $result; |
return $sensor->{'status'} unless $check; |
} |
|
elsif ( $check eq 'STATUS' ) { |
|
|
return undef if $check eq 'IGNORE'; |
# It looks like doing this should be safe, from |
|
# src/sbin/sysctl/sysctl.c |
|
return ( $sensor->{'status'} || $result ); |
|
} |
|
elsif ( $check eq 'IGNORE' ) { |
|
return; |
|
} |
|
|
$result = 'OK'; |
foreach my $code ( 'warn', 'crit' ) { |
foreach my $code ('warn', 'crit') { |
if ( $sensor->{'type'} eq 'fan' |
if ( |
|| $sensor->{'type'} eq 'fanrpm' |
$sensor->{'type'} eq 'fan' || |
|| $sensor->{'type'} eq 'volt' |
$sensor->{'type'} eq 'fanrpm' || |
|| $sensor->{'type'} eq 'volts_dc' |
$sensor->{'type'} eq 'volt' || |
|| $sensor->{'type'} eq 'amps' |
$sensor->{'type'} eq 'volts_dc' || |
|| $sensor->{'type'} eq 'watthour' |
$sensor->{'type'} eq 'amps' || |
|| $sensor->{'type'} eq 'amphour' |
$sensor->{'type'} eq 'watthour' || |
|| $sensor->{'type'} eq 'integer' |
$sensor->{'type'} eq 'amphour' || |
|| $sensor->{'type'} eq 'raw' |
$sensor->{'type'} eq 'integer' || |
|| $sensor->{'type'} eq 'percent' |
$sensor->{'type'} eq 'raw' || |
|| $sensor->{'type'} eq 'lux' |
$sensor->{'type'} eq 'percent' || |
|| $sensor->{'type'} eq 'timedelta' ) |
$sensor->{'type'} eq 'lux' || |
{ |
$sensor->{'type'} eq 'timedelta' |
my $data = $sensor->{'data'}; |
) { |
$data =~ s/[^\d\.]//gxms; |
my $data = $sensor->{'data'}; |
if ( !length $data ) { |
$data =~ s/[^\d\.]//g; |
warn |
unless (length $data) { |
"INVALID DATA ($sensor->{'data'}) for '$sensor->{'id'}'\n"; |
warn "INVALID DATA ($sensor->{'data'}) for '$sensor->{'id'}'"; |
next; |
next; |
} |
} |
|
|
|
if ( |
if ( defined $check->{ $code . ".low" } |
defined $check->{$code . ".low"} || |
|| defined $check->{ $code . ".high" } ) |
defined $check->{$code . ".high"} |
{ |
) { |
if ( defined $check->{ $code . ".low" } ) { |
if (defined $check->{$code . ".low"}) { |
my $c = $check->{ $code . ".low" }; |
my $c = $check->{$code . ".low"}; |
$c =~ s/[^\d\.]//gxms; |
$c =~ s/[^\d\.]//g; |
|
|
|
unless (length $c) { |
|
warn "INVALID CHECK (" . $check->{$code . ".low"} . |
|
") for '$sensor->{'id'}:$code.low'"; |
|
next; |
|
} |
|
|
|
$result = $errors{$code} |
if ( !length $c ) { |
if ($c >= $data); |
warn "INVALID CHECK (" |
} |
. $check->{ $code . ".low" } |
if (defined $check->{$code . ".high"}) { |
. ") for '$sensor->{'id'}:$code.low'\n"; |
my $c = $check->{$code . ".high"}; |
next; |
$c =~ s/[^\d\.]//g; |
} |
unless (length $c) { |
|
warn "INVALID CHECK (" . $check->{$code . ".high"} . |
|
") for '$sensor->{'id'}:$code.high'"; |
|
next; |
|
} |
|
|
|
$result = $errors{$code} |
$result = $errors{$code} |
if ($c <= $data); |
if ( $c >= $data ); |
} |
} |
} elsif (@{ $check->{$code} }) { |
if ( defined $check->{ $code . ".high" } ) { |
my $matched = 0; |
my $c = $check->{ $code . ".high" }; |
foreach my $c (@{ $check->{$code} }) { |
$c =~ s/[^\d\.]//gxms; |
$c =~ s/[^\d\.]//g; |
if ( !length $c ) { |
unless (length $c) { |
warn "INVALID CHECK (" |
warn "INVALID CHECK (" . $c . |
. $check->{ $code . ".high" } |
") for '$sensor->{'id'}:$code'"; |
. ") for '$sensor->{'id'}:$code.high'\n"; |
next; |
next; |
} |
} |
|
|
if ($c eq $data) { |
$result = $errors{$code} |
$matched = 1; |
if ( $c <= $data ); |
last; |
} |
} |
$result = 'OK' if $result eq 'UNKNOWN'; |
} |
} |
$result = $errors{$code} unless $matched; |
elsif ( @{ $check->{$code} } ) { |
} |
my $matched = 0; |
|
foreach ( @{ $check->{$code} } ) { |
|
my $c = $_; |
|
$c =~ s/[^\d\.]//gxms; |
|
if ( !length $c ) { |
|
warn "INVALID CHECK (" . $_ |
|
. ") for '$sensor->{'id'}:$code'\n"; |
|
next; |
|
} |
|
|
} elsif ($sensor->{'type'} eq 'temp') { |
if ( $c eq $data ) { |
my ($degC, $degF) = split /\//, $sensor->{'data'}; |
$matched = 1; |
$degC =~ s/[^\d\.]//g; |
last; |
$degF ||= $degC * 9 / 5 + 32; |
} |
$degF =~ s/[^\d\.]//g; |
} |
if ( |
if ($matched) { |
defined $check->{$code . ".low"} || |
$result = 'OK' if $result eq 'UNKNOWN'; |
defined $check->{$code . ".high"} |
} |
) { |
else { |
if (defined $check->{$code . ".low"}) { |
$result = $errors{$code}; |
my $c = $check->{$code . ".low"}; |
} |
my $data = $degC; |
} |
|
} |
|
elsif ( $sensor->{'type'} eq 'temp' ) { |
|
my ( $degC, $degF ) = split /\//, $sensor->{'data'}; |
|
$degC =~ s/[^\d\.]//gxms; |
|
$degF ||= $degC * 9 / 5 + 32; |
|
$degF =~ s/[^\d\.]//gxms; |
|
if ( defined $check->{ $code . ".low" } |
|
|| defined $check->{ $code . ".high" } ) |
|
{ |
|
if ( defined $check->{ $code . ".low" } ) { |
|
my $c = $check->{ $code . ".low" }; |
|
my $data = $degC; |
|
|
$data = $degF if ($c =~ /F/i); |
$data = $degF if ( $c =~ /F/ixms ); |
unless (length $data) { |
if ( !length $data ) { |
warn "INVALID DATA (" . $sensor->{'data'} . |
warn "INVALID DATA (" |
") for '$sensor->{'id'}'"; |
. $sensor->{'data'} |
next; |
. ") for '$sensor->{'id'}'\n"; |
} |
next; |
|
} |
|
|
$c =~ s/[^\d\.]//g; |
$c =~ s/[^\d\.]//gxms; |
unless (length $c) { |
if ( !length $c ) { |
warn "INVALID CHECK (" . $check->{$code . ".low"} . |
warn "INVALID CHECK (" |
") for '$sensor->{'id'}':$code.low"; |
. $check->{ $code . ".low" } |
next; |
. ") for '$sensor->{'id'}':$code.low\n"; |
} |
next; |
|
} |
|
|
$result = $errors{$code} |
$result = $errors{$code} |
if ($c >= $data); |
if ( $c >= $data ); |
} |
} |
if (defined $check->{$code . ".high"}) { |
if ( defined $check->{ $code . ".high" } ) { |
my $c = $check->{$code . ".high"}; |
my $c = $check->{ $code . ".high" }; |
|
|
my $data = $degC; |
my $data = $degC; |
$data = $degF if ($c =~ /F/i); |
$data = $degF if ( $c =~ /F/ixms ); |
unless (length $data) { |
if ( !length $data ) { |
warn "INVALID DATA (" . $sensor->{'data'} . |
warn "INVALID DATA (" |
") for '$sensor->{'id'}'"; |
. $sensor->{'data'} |
next; |
. ") for '$sensor->{'id'}'\n"; |
} |
next; |
|
} |
|
|
$c =~ s/[^\d\.]//g; |
$c =~ s/[^\d\.]//gxms; |
unless (length $c) { |
if ( !length $c ) { |
warn "INVALID CHECK (" . $check->{$code . ".high"} . |
warn "INVALID CHECK (" |
") for '$sensor->{'id'}:$code.high'"; |
. $check->{ $code . ".high" } |
next; |
. ") for '$sensor->{'id'}:$code.high'\n"; |
} |
next; |
|
} |
|
|
$result = $errors{$code} |
$result = $errors{$code} |
if ($c <= $data); |
if ( $c <= $data ); |
} |
} |
} elsif (@{ $check->{$code} }) { |
$result = 'OK' if $result eq 'UNKNOWN'; |
|
} |
|
elsif ( @{ $check->{$code} } ) { |
|
my $matched = 0; |
|
foreach ( @{ $check->{$code} } ) { |
|
my $c = $_; |
|
my $data = $degC; |
|
|
my $matched = 0; |
$data = $degF if ( $c =~ /F/ixms ); |
foreach my $c (@{ $check->{$code} }) { |
if ( !length $data ) { |
my $data = $degC; |
warn "INVALID DATA (" |
|
. $sensor->{'data'} |
|
. ") for '$sensor->{'id'}'\n"; |
|
next; |
|
} |
|
|
$data = $degF if ($c =~ /F/i); |
$c =~ s/[^\d\.]//gxms; |
unless (length $data) { |
if ( !length $c ) { |
warn "INVALID DATA (" . $sensor->{'data'} . |
warn "INVALID CHECK (" . $_ |
") for '$sensor->{'id'}'"; |
. ") for '$sensor->{'id'}':$code\n"; |
next; |
next; |
} |
} |
|
|
$c =~ s/[^\d\.]//g; |
if ( $c eq $data ) { |
unless (length $c) { |
$matched = 1; |
warn "INVALID CHECK (" . $c . |
last; |
") for '$sensor->{'id'}':$code"; |
} |
next; |
} |
} |
if ($matched) { |
|
$result = 'OK' if $result eq 'UNKNOWN'; |
|
} |
|
else { |
|
$result = $errors{$code}; |
|
} |
|
} |
|
} |
|
elsif ($sensor->{'type'} eq 'drive' |
|
|| $sensor->{'type'} eq 'indicator' ) |
|
{ |
|
$sensor->{'data'} =~ s/^drive\s+//; |
|
if ( @{ $check->{$code} } ) { |
|
my $matched = 0; |
|
foreach ( @{ $check->{$code} } ) { |
|
if ( $_ eq $sensor->{'data'} ) { |
|
$matched = 1; |
|
last; |
|
} |
|
} |
|
if ($matched) { |
|
$result = 'OK' if $result eq 'UNKNOWN'; |
|
} |
|
else { |
|
$result = $errors{$code}; |
|
} |
|
} |
|
|
if ($c eq $data) { |
} |
$matched = 1; |
else { |
last; |
warn 'Unknown Sensor Type: ', |
} |
$sensor->{'id'}, |
} |
'=', |
$result = $errors{$code} unless $matched; |
$sensor->{'type'}, |
} |
"\n"; |
|
} |
|
|
} elsif ( |
} |
$sensor->{'type'} eq 'drive' || |
|
$sensor->{'type'} eq 'indicator' |
|
) { |
|
$sensor->{'data'} =~ s/^drive\s+//; |
|
if (@{ $check->{$code} }) { |
|
my $matched = 0; |
|
foreach (@{ $check->{$code} }) { |
|
if ($_ eq $sensor->{'data'}) { |
|
$matched = 1; |
|
last; |
|
} |
|
} |
|
$result = $errors{$code} unless $matched; |
|
} |
|
|
|
} else { |
return $result; |
print STDERR 'Unknown Sensor Type: ', |
|
$sensor->{'id'}, |
|
'=', |
|
$sensor->{'type'}, |
|
"\n"; |
|
$result = 'UNKNOWN'; |
|
} |
|
|
|
} |
|
|
|
return $result; |
|
} |
} |
|
|
sub print_help { |
sub print_help { |
print <<EOL; |
print <<"EOL"; |
$PROGNAME plugin for Nagios monitors sysctl hw.sensors on OpenBSD |
$PROGNAME - monitors sysctl hw.sensors on OpenBSD |
$PROGNAME [-i] (-f [<FILENAME>]|(-s <hw.sensors id> [-w limit] [-c limit])) |
$PROGNAME [-i] [-f [<FILENAME>]|(-s <hw.sensors id> [-w limit] [-c limit])] |
|
|
Usage: |
Usage: |
-i, --ignore-status |
-i, --ignore-status |
Don't check the status of sensors that report it. |
Don't automatically check the status of sensors that report it. |
-f, --filename=FILE |
-f, --filename=FILE |
FILE to load checks from (defaults to /etc/sensorsd.conf) |
FILE to load checks from (defaults to /etc/sensorsd.conf) |
-s, --sensor=ID |
-s, --sensor=ID |
ID of a single sensor. "-s 0" means hw.sensors.0. |
ID of a single sensor. "-s kate0.temp0" means hw.sensors.kate0.temp0 |
|
Overrides --filename. |
-w, --warning=RANGE or single ENTRY |
-w, --warning=RANGE or single ENTRY |
Exit with WARNING status if outside of RANGE or if != ENTRY |
Exit with WARNING status if outside of RANGE or if != ENTRY |
-c, --critical=RANGE or single ENTRY |
-c, --critical=RANGE or single ENTRY |
|
|
|
|
FILE is in the same format as sensorsd.conf(5) plus some additional |
FILE is in the same format as sensorsd.conf(5) plus some additional |
entries. These additional entries in the file are ignored by |
entries. These additional entries in the file are ignored by |
sensorsd(8). |
sensorsd(8) this means you can use the same config file for $PROGNAME |
|
as well as sensorsd(8). |
|
|
$PROGNAME understands the following entries: |
$PROGNAME understands the following entries: |
|
|
|
|
|
|
A RANGE is a low ENTRY and a high ENTRY separated by a colon (:). |
A RANGE is a low ENTRY and a high ENTRY separated by a colon (:). |
It can also be low: or :high with the other side left blank to only |
It can also be low: or :high with the other side left blank to only |
make the single check.. |
make the single check. |
|
|
An entry marked "ignore" will cause that sensor to be skipped. |
An entry marked "ignore" will cause that sensor to be skipped. |
Generally used with state checking of all sensors to ignore sensors you |
Generally used with state checking of all sensors to ignore sensors you |
|
|
|
|
EOL |
EOL |
|
|
print_revision($PROGNAME, '$Revision$'); |
print_revision( $PROGNAME, '$Revision$' ); |
|
|
|
print $License; |
|
} |
|
|
|
sub print_revision { |
|
my ( $prog, $rev ) = @_; |
|
$rev =~ s/^\D+([\d\.]+)\D+$/v$1/xms; |
|
|
|
print "$prog $rev\n"; |
} |
} |
|
|