1
0
mirror of https://github.com/opinkerfi/nagios-plugins.git synced 2024-11-22 02:13:44 +01:00
nagios-plugins/check_hpasm/check_hpasm

8391 lines
262 KiB
Plaintext
Raw Normal View History

2010-06-10 21:09:42 +02:00
#! /usr/bin/perl -w
package Nagios::MiniPlugin;
use strict;
use Getopt::Long qw(:config no_ignore_case bundling);
our @STATUS_CODES = qw(OK WARNING CRITICAL UNKNOWN DEPENDENT);
require Exporter;
our @ISA = qw(Exporter);
our @EXPORT = (@STATUS_CODES, qw(nagios_exit nagios_die check_messages));
our @EXPORT_OK = qw(%ERRORS);
use constant OK => 0;
use constant WARNING => 1;
use constant CRITICAL => 2;
use constant UNKNOWN => 3;
use constant DEPENDENT => 4;
our %ERRORS = (
'OK' => OK,
'WARNING' => WARNING,
'CRITICAL' => CRITICAL,
'UNKNOWN' => UNKNOWN,
'DEPENDENT' => DEPENDENT,
);
our %STATUS_TEXT = reverse %ERRORS;
sub new {
my $class = shift;
my %params = @_;
my $self = {
perfdata => [],
messages => {
ok => [],
warning => [],
critical => [],
unknown => [],
},
args => [],
opts => Nagios::MiniPlugin::Getopt->new(%params),
};
foreach (qw(shortname usage version url plugin blurb extra
license timeout)) {
$self->{$_} = $params{$_};
}
bless $self, $class;
}
sub add_arg {
my $self = shift;
$self->{opts}->add_arg(@_);
}
sub getopts {
my $self = shift;
$self->{opts}->getopts();
}
sub opts {
my $self = shift;
return $self->{opts};
}
sub add_message {
my $self = shift;
my ($code, @messages) = @_;
$code = (qw(ok warning critical unknown))[$code] if $code =~ /^\d+$/;
$code = lc $code;
push @{$self->{messages}->{$code}}, @messages;
}
sub add_perfdata {
my ($self, %args) = @_;
my $str = $args{label}.'='.$args{value};
if ($args{uom}) {
$str .= $args{uom};
}
if ($args{warning}) {
$str .= ';'.$args{warning};
}
if ($args{critical}) {
$str .= ';'.$args{critical};
}
push @{$self->{perfdata}}, $str;
}
sub check_messages {
my $self = shift;
my %args = @_;
# Add object messages to any passed in as args
for my $code (qw(critical warning unknown ok)) {
my $messages = $self->{messages}->{$code} || [];
if ($args{$code}) {
unless (ref $args{$code} eq 'ARRAY') {
if ($code eq 'ok') {
$args{$code} = [ $args{$code} ];
}
}
push @{$args{$code}}, @$messages;
} else {
$args{$code} = $messages;
}
}
my %arg = %args;
$arg{join} = ' ' unless defined $arg{join};
# Decide $code
my $code = OK;
$code ||= CRITICAL if @{$arg{critical}};
$code ||= WARNING if @{$arg{warning}};
$code ||= UNKNOWN if @{$arg{unknown}};
return $code unless wantarray;
# Compose message
my $message = '';
if ($arg{join_all}) {
$message = join( $arg{join_all},
map { @$_ ? join( $arg{'join'}, @$_) : () }
$arg{critical},
$arg{warning},
$arg{unknown},
$arg{ok} ? (ref $arg{ok} ? $arg{ok} : [ $arg{ok} ]) : []
);
}
else {
$message ||= join( $arg{'join'}, @{$arg{critical}} )
if $code == CRITICAL;
$message ||= join( $arg{'join'}, @{$arg{warning}} )
if $code == WARNING;
$message ||= join( $arg{'join'}, @{$arg{unknown}} )
if $code == UNKNOWN;
$message ||= ref $arg{ok} ? join( $arg{'join'}, @{$arg{ok}} ) : $arg{ok}
if $arg{ok};
}
return ($code, $message);
}
sub nagios_exit {
my $self = shift;
my ($code, $message, $arg) = @_;
$code = $ERRORS{$code} if defined $code && exists $ERRORS{$code};
$code = UNKNOWN unless defined $code && exists $STATUS_TEXT{$code};
$message = '' unless defined $message;
if (ref $message && ref $message eq 'ARRAY') {
$message = join(' ', map { chomp; $_ } @$message);
} else {
chomp $message;
}
my $output = "$STATUS_TEXT{$code}";
$output .= " - $message" if defined $message && $message ne '';
if (scalar (@{$self->{perfdata}})) {
$output .= " | ".join(" ", @{$self->{perfdata}});
}
$output .= "\n";
print $output;
exit $code;
}
package Nagios::MiniPlugin::Getopt;
use strict;
use File::Basename;
use Data::Dumper;
use Getopt::Long qw(:config no_ignore_case bundling);
# Standard defaults
my %DEFAULT = (
timeout => 15,
verbose => 0,
license =>
"This nagios plugin is free software, and comes with ABSOLUTELY NO WARRANTY.
It may be used, redistributed and/or modified under the terms of the GNU
General Public Licence (see http://www.fsf.org/licensing/licenses/gpl.txt).",
);
# Standard arguments
my @ARGS = ({
spec => 'usage|?',
help => "-?, --usage\n Print usage information",
}, {
spec => 'help|h',
help => "-h, --help\n Print detailed help screen",
}, {
spec => 'version|V',
help => "-V, --version\n Print version information",
}, {
#spec => 'extra-opts:s@',
#help => "--extra-opts=[<section>[@<config_file>]]\n Section and/or config_file from which to load extra options (may repeat)",
}, {
spec => 'timeout|t=i',
help => "-t, --timeout=INTEGER\n Seconds before plugin times out (default: %s)",
default => $DEFAULT{timeout},
}, {
spec => 'verbose|v+',
help => "-v, --verbose\n Show details for command-line debugging (can repeat up to 3 times)",
default => $DEFAULT{verbose},
},
);
# Standard arguments we traditionally display last in the help output
my %DEFER_ARGS = map { $_ => 1 } qw(timeout verbose);
sub _init
{
my $self = shift;
my %params = @_;
# Check params
my $plugin = basename($ENV{NAGIOS_PLUGIN} || $0);
#my %attr = validate( @_, {
my %attr = (
usage => 1,
version => 0,
url => 0,
plugin => { default => $plugin },
blurb => 0,
extra => 0,
'extra-opts' => 0,
license => { default => $DEFAULT{license} },
timeout => { default => $DEFAULT{timeout} },
);
# Add attr to private _attr hash (except timeout)
$self->{timeout} = delete $attr{timeout};
$self->{_attr} = { %attr };
foreach (keys %{$self->{_attr}}) {
if (exists $params{$_}) {
$self->{_attr}->{$_} = $params{$_};
} else {
$self->{_attr}->{$_} = $self->{_attr}->{$_}->{default}
if ref ($self->{_attr}->{$_}) eq 'HASH' &&
exists $self->{_attr}->{$_}->{default};
}
}
# Chomp _attr values
chomp foreach values %{$self->{_attr}};
# Setup initial args list
$self->{_args} = [ grep { exists $_->{spec} } @ARGS ];
$self
}
sub new
{
my $class = shift;
my $self = bless {}, $class;
$self->_init(@_);
}
sub add_arg {
my $self = shift;
my %arg = @_;
push (@{$self->{_args}}, \%arg);
}
sub getopts {
my $self = shift;
my %commandline = ();
my @params = map { $_->{spec} } @{$self->{_args}};
if (! GetOptions(\%commandline, @params)) {
$self->print_help();
exit 0;
} else {
no strict 'refs';
do { $self->print_help(); exit 0; } if $commandline{help};
do { $self->print_version(); exit 0 } if $commandline{version};
do { $self->print_usage(); exit 0 } if $commandline{usage};
foreach (map { $_->{spec} =~ /^([\w\-]+)/; $1; } @{$self->{_args}}) {
my $field = $_;
*{"$field"} = sub {
return $self->{opts}->{$field};
};
}
foreach (grep { exists $_->{default} } @{$self->{_args}}) {
$_->{spec} =~ /^([\w\-]+)/;
my $spec = $1;
$self->{opts}->{$spec} = $_->{default};
}
foreach (keys %commandline) {
$self->{opts}->{$_} = $commandline{$_};
}
}
}
sub get {
my $self = shift;
my $opt = shift;
return $self->{opts}->{$opt};
}
sub print_help {
my $self = shift;
$self->print_version();
printf "\n%s\n", $self->{_attr}->{license};
printf "\n%s\n\n", $self->{_attr}->{blurb};
$self->print_usage();
foreach (@{$self->{_args}}) {
printf " %s\n", $_->{help};
}
exit 0;
}
sub print_usage {
my $self = shift;
printf $self->{_attr}->{usage}, $self->{_attr}->{plugin};
print "\n";
}
sub print_version {
my $self = shift;
printf "%s %s", $self->{_attr}->{plugin}, $self->{_attr}->{version};
printf " [%s]", $self->{_attr}->{url} if $self->{_attr}->{url};
print "\n";
}
sub print_license {
my $self = shift;
printf "%s\n", $self->{_attr}->{license};
print "\n";
}
package SNMP::Utils;
use strict;
{
sub get_indices {
my $oids = shift;
my $entry = shift;
my $numindices = shift;
# find all oids beginning with $entry
# then skip one field for the sequence
# then read the next numindices fields
my $entrypat = $entry;
$entrypat =~ s/\./\\\./g;
my @indices = map {
/^$entrypat\.\d+\.(.*)/ && $1;
} grep {
/^$entrypat/
} keys %{$oids};
my %seen = ();
my @o = map {[split /\./]} sort grep !$seen{$_}++, @indices;
return @o;
}
sub get_size {
my $oids = shift;
my $entry = shift;
my $entrypat = $entry;
$entrypat =~ s/\./\\\./g;
my @entries = grep {
/^$entrypat/
} keys %{$oids};
return scalar(@entries);
}
sub get_object {
my $oids = shift;
my $object = shift;
my @indices = @_;
#my $oid = $object.'.'.join('.', @indices);
my $oid = $object;
$oid .= '.'.join('.', @indices) if (@indices);
return $oids->{$oid};
}
sub get_object_value {
my $oids = shift;
my $object = shift;
my $values = shift;
my @indices = @_;
my $key = get_object($oids, $object, @indices);
if (defined $key) {
return $values->{$key};
} else {
return undef;
}
}
#SNMP::Utils::counter([$idxs1, $idxs2], $idx1, $idx2),
# this flattens a n-dimensional array and returns the absolute position
# of the element at position idx1,idx2,...,idxn
# element 1,2 in table 0,0 0,1 0,2 1,0 1,1 1,2 2,0 2,1 2,2 is at pos 6
sub get_number {
my $indexlists = shift; #, zeiger auf array aus [1, 2]
my @element = @_;
my $dimensions = scalar(@{$indexlists->[0]});
my @sorted = ();
my $number = 0;
if ($dimensions == 1) {
@sorted =
sort { $a->[0] <=> $b->[0] } @{$indexlists};
} elsif ($dimensions == 2) {
@sorted =
sort { $a->[0] <=> $b->[0] || $a->[1] <=> $b->[1] } @{$indexlists};
} elsif ($dimensions == 3) {
@sorted =
sort { $a->[0] <=> $b->[0] ||
$a->[1] <=> $b->[1] ||
$a->[2] <=> $b->[2] } @{$indexlists};
}
foreach (@sorted) {
if ($dimensions == 1) {
if ($_->[0] == $element[0]) {
last;
}
} elsif ($dimensions == 2) {
if ($_->[0] == $element[0] && $_->[1] == $element[1]) {
last;
}
} elsif ($dimensions == 3) {
if ($_->[0] == $element[0] &&
$_->[1] == $element[1] &&
$_->[2] == $element[2]) {
last;
}
}
$number++;
}
return ++$number;
}
}
package HP::Proliant::Component::PowersupplySubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
condition => $params{condition},
status => $params{status},
powersupplies => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
return HP::Proliant::Component::PowersupplySubsystem::SNMP->new(%params);
} elsif ($self->{method} eq 'cli') {
return HP::Proliant::Component::PowersupplySubsystem::CLI->new(%params);
} else {
die "unknown method";
}
return $self;
}
sub check {
my $self = shift;
my $errorfound = 0;
$self->add_info('checking power supplies');
if (scalar (@{$self->{powersupplies}}) == 0) {
#$self->overall_check();
} else {
foreach (@{$self->{powersupplies}}) {
$_->check();
}
}
}
sub dump {
my $self = shift;
foreach (@{$self->{powersupplies}}) {
$_->dump();
}
}
package HP::Proliant::Component::PowersupplySubsystem::Powersupply;
our @ISA = qw(HP::Proliant::Component::PowersupplySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqHeFltTolPowerSupplyChassis => $params{cpqHeFltTolPowerSupplyChassis},
cpqHeFltTolPowerSupplyBay => $params{cpqHeFltTolPowerSupplyBay},
cpqHeFltTolPowerSupplyPresent => $params{cpqHeFltTolPowerSupplyPresent},
cpqHeFltTolPowerSupplyCondition => $params{cpqHeFltTolPowerSupplyCondition},
cpqHeFltTolPowerSupplyRedundant => $params{cpqHeFltTolPowerSupplyRedundant},
blacklisted => 0,
info => undef,
extendexinfo => undef,
};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('p', $self->{cpqHeFltTolPowerSupplyBay});
if ($self->{cpqHeFltTolPowerSupplyPresent} eq "present") {
if ($self->{cpqHeFltTolPowerSupplyCondition} ne "ok") {
if ($self->{cpqHeFltTolPowerSupplyCondition} eq "other") {
$self->add_info(sprintf "powersupply %d is missing",
$self->{cpqHeFltTolPowerSupplyBay});
} else {
$self->add_info(sprintf "powersupply %d needs attention (%s)",
$self->{cpqHeFltTolPowerSupplyBay},
$self->{cpqHeFltTolPowerSupplyCondition});
}
$self->add_message(CRITICAL, $self->{info});
} else {
$self->add_info(sprintf "powersupply %d is %s",
$self->{cpqHeFltTolPowerSupplyBay},
$self->{cpqHeFltTolPowerSupplyCondition});
}
$self->add_extendedinfo(sprintf "ps_%s=%s",
$self->{cpqHeFltTolPowerSupplyBay},
$self->{cpqHeFltTolPowerSupplyCondition});
} else {
$self->add_info(sprintf "powersupply %d is %s",
$self->{cpqHeFltTolPowerSupplyBay},
$self->{cpqHeFltTolPowerSupplyPresent});
$self->add_extendedinfo(sprintf "ps_%s=%s",
$self->{cpqHeFltTolPowerSupplyBay},
$self->{cpqHeFltTolPowerSupplyPresent});
}
}
sub dump {
my $self = shift;
printf "[PS_%s]\n", $self->{cpqHeFltTolPowerSupplyBay};
foreach (qw(cpqHeFltTolPowerSupplyBay cpqHeFltTolPowerSupplyChassis
cpqHeFltTolPowerSupplyPresent cpqHeFltTolPowerSupplyCondition
cpqHeFltTolPowerSupplyRedundant)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n\n", $self->{info};
}
package HP::Proliant::Component::PowersupplySubsystem::CLI;
our @ISA = qw(HP::Proliant::Component::PowersupplySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
powersupplies => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
my %tmpps = (
runtime => $self->{runtime},
cpqHeFltTolPowerSupplyChassis => 1,
);
my $inblock = 0;
foreach (grep(/^powersupply/, split(/\n/, $self->{rawdata}))) {
s/^powersupply\s*//g;
if (/^Power supply #(\d+)/) {
if ($inblock) {
$inblock = 0;
push(@{$self->{powersupplies}},
HP::Proliant::Component::PowersupplySubsystem::Powersupply->new(%tmpps));
%tmpps = (
runtime => $self->{runtime},
cpqHeFltTolPowerSupplyChassis => 1,
);
}
$tmpps{cpqHeFltTolPowerSupplyBay} = $1;
$inblock = 1;
} elsif (/\s*Present\s+:\s+(\w+)/) {
$tmpps{cpqHeFltTolPowerSupplyPresent} = lc $1 eq 'yes' ? 'present' :
lc $1 eq 'no' ? 'absent': 'other';
} elsif (/\s*Redundant\s*:\s+(\w+)/) {
$tmpps{cpqHeFltTolPowerSupplyRedundant} = lc $1 eq 'yes' ? 'redundant' :
lc $1 eq 'no' ? 'notRedundant' : 'other';
} elsif (/\s*Condition\s*:\s+(\w+)/) {
$tmpps{cpqHeFltTolPowerSupplyCondition} = lc $1;
} elsif (/\s*Power Supply not present/) {
$tmpps{cpqHeFltTolPowerSupplyPresent} = "absent";
$tmpps{cpqHeFltTolPowerSupplyCondition} = "other";
$tmpps{cpqHeFltTolPowerSupplyRedundant} = "notRedundant";
} elsif (/^\s*$/) {
if ($inblock) {
$inblock = 0;
push(@{$self->{powersupplies}},
HP::Proliant::Component::PowersupplySubsystem::Powersupply->new(%tmpps));
%tmpps = (
runtime => $self->{runtime},
cpqHeFltTolPowerSupplyChassis => 1,
);
}
}
}
if ($inblock) {
push(@{$self->{powersupplies}},
HP::Proliant::Component::PowersupplySubsystem::Powersupply->new(%tmpps));
%tmpps = (
runtime => $params{runtime},
);
}
}
package HP::Proliant::Component::PowersupplySubsystem::SNMP;
our @ISA = qw(HP::Proliant::Component::PowersupplySubsystem
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
powersupplies => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeFltTolPowerSupplyEntry => "1.3.6.1.4.1.232.6.2.9.3.1",
cpqHeFltTolPowerSupplyChassis => "1.3.6.1.4.1.232.6.2.9.3.1.1",
cpqHeFltTolPowerSupplyBay => "1.3.6.1.4.1.232.6.2.9.3.1.2",
cpqHeFltTolPowerSupplyPresent => "1.3.6.1.4.1.232.6.2.9.3.1.3",
cpqHeFltTolPowerSupplyCondition => "1.3.6.1.4.1.232.6.2.9.3.1.4",
cpqHeFltTolPowerSupplyRedundant => "1.3.6.1.4.1.232.6.2.9.3.1.9",
cpqHeFltTolPowerSupplyPresentValue => {
1 => "other",
2 => "absent",
3 => "present",
},
cpqHeFltTolPowerSupplyConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqHeFltTolPowerSupplyRedundantValue => {
1 => "other",
2 => "notRedundant",
3 => "redundant",
},
};
# INDEX { cpqHeFltTolPowerSupplyChassis, cpqHeFltTolPowerSupplyBay }
foreach ($self->get_entries($oids, 'cpqHeFltTolPowerSupplyEntry')) {
push(@{$self->{powersupplies}},
HP::Proliant::Component::PowersupplySubsystem::Powersupply->new(%{$_}));
}
$oids = {
cpqHePowerConvEntry => "1.3.6.1.4.1.232.6.2.13.3.1",
cpqHePowerConvChassis => "1.3.6.1.4.1.232.6.2.13.3.1",
cpqHePowerConvIndex => "1.3.6.1.4.1.232.6.2.13.3.1.2",
cpqHePowerConvPresent => "1.3.6.1.4.1.232.6.2.13.3.1.3",
cpqHePowerConvRedundant => "1.3.6.1.4.1.232.6.2.13.3.1.6",
cpqHePowerConvCondition => "1.3.6.1.4.1.232.6.2.13.3.1.8",
cpqHePowerConvPresentValue => {
1 => "other",
2 => "absent",
3 => "present",
},
cpqHePowerConvRedundantValue => {
1 => "other",
2 => "notRedundant",
3 => "redundant",
},
cpqHePowerConvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqHePowerConvHwLocation => "1.3.6.1.4.1.232.6.2.13.3.1.9",
};
# INDEX { cpqHePowerConvChassis cpqHePowerConvIndex }
#foreach ($self->get_entries($oids, 'cpqHePowerConvEntry')) {
#push(@{$self->{powersupplies}},
# HP::Proliant::Component::PowersupplySubsystem::Powersupply->new(%{$_}));
#}
# keine ahnung, was man damit machen kann
}
package HP::Proliant::Component::TemperatureSubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
################################## custom_thresholds
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
condition => $params{condition},
status => $params{status},
temperatures => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if ($params{runtime}->{options}->{customthresholds}) {
if (-f $params{runtime}->{options}->{customthresholds}) {
$params{runtime}->{options}->{customthresholds} =
do { local (@ARGV, $/) =
$params{runtime}->{options}->{customthresholds}; <> };
}
foreach my $ct_items
(split(/\//, $params{runtime}->{options}->{customthresholds})) {
if ($ct_items =~ /^(\d+):(\d+)$/) {
$params{runtime}->{options}->{thresholds}->{$1} = $2;
} else {
die sprintf "invalid threshold %s", $ct_items;
}
}
}
if ($self->{method} eq 'snmp') {
return HP::Proliant::Component::TemperatureSubsystem::SNMP->new(%params);
} elsif ($self->{method} eq 'cli') {
return HP::Proliant::Component::TemperatureSubsystem::CLI->new(%params);
} else {
die "unknown method";
}
return $self;
}
sub check {
my $self = shift;
my $errorfound = 0;
$self->add_info('checking temperatures');
if (scalar (@{$self->{temperatures}}) == 0) {
#$self->overall_check();
$self->add_info('no temperatures found');
} else {
foreach (sort { $a->{cpqHeTemperatureIndex} <=> $b->{cpqHeTemperatureIndex}}
@{$self->{temperatures}}) {
$_->check();
}
}
}
sub dump {
my $self = shift;
foreach (@{$self->{temperatures}}) {
$_->dump();
}
}
package HP::Proliant::Component::TemperatureSubsystem::Temperature;
our @ISA = qw(HP::Proliant::Component::TemperatureSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqHeTemperatureChassis => $params{cpqHeTemperatureChassis},
cpqHeTemperatureIndex => $params{cpqHeTemperatureIndex},
cpqHeTemperatureLocale => $params{cpqHeTemperatureLocale},
cpqHeTemperatureCelsius => $params{cpqHeTemperatureCelsius},
cpqHeTemperatureThresholdCelsius => $params{cpqHeTemperatureThresholdCelsius},
cpqHeTemperatureCondition => $params{cpqHeTemperatureCondition},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if ($params{runtime}->{options}->{celsius}) {
$self->{cpqHeTemperatureUnits} = 'C';
$self->{cpqHeTemperature} = $self->{cpqHeTemperatureCelsius};
$self->{cpqHeTemperatureThreshold} =
$self->{cpqHeTemperatureThresholdCelsius};
} else {
$self->{cpqHeTemperatureUnits} = 'F';
$self->{cpqHeTemperature} =
(($self->{cpqHeTemperatureCelsius} * 9) / 5) + 32;
$self->{cpqHeTemperatureThreshold} =
(($self->{cpqHeTemperatureThresholdCelsius} * 9) / 5) + 32;
}
my $index = $self->{cpqHeTemperatureIndex};
if (exists $params{runtime}->{options}->{thresholds}->{$index}) {
$self->{cpqHeTemperatureThreshold} =
$params{runtime}->{options}->{thresholds}->{$index};
}
return $self;
}
sub check {
my $self = shift;
$self->blacklist('t', $self->{cpqHeTemperatureIndex});
if ($self->{cpqHeTemperature} > $self->{cpqHeTemperatureThreshold}) {
$self->add_info(sprintf "%d %s temperature too high (%d%s, %d max)",
$self->{cpqHeTemperatureIndex}, $self->{cpqHeTemperatureLocale},
$self->{cpqHeTemperature}, $self->{cpqHeTemperatureUnits},
$self->{cpqHeTemperatureThreshold});
$self->add_message(CRITICAL, $self->{info});
} else {
$self->add_info(sprintf "%d %s temperature is %d%s (%d max)",
$self->{cpqHeTemperatureIndex}, $self->{cpqHeTemperatureLocale},
$self->{cpqHeTemperature}, $self->{cpqHeTemperatureUnits},
$self->{cpqHeTemperatureThreshold});
}
if ($self->{runtime}->{options}->{perfdata} == 2) {
$self->{runtime}->{plugin}->add_perfdata(
label => sprintf('temp_%s', $self->{cpqHeTemperatureIndex}),
value => $self->{cpqHeTemperature},
warning => $self->{cpqHeTemperatureThreshold},
critical => $self->{cpqHeTemperatureThreshold}
);
} elsif ($self->{runtime}->{options}->{perfdata} == 1) {
$self->{runtime}->{plugin}->add_perfdata(
label => sprintf('temp_%s_%s', $self->{cpqHeTemperatureIndex},
$self->{cpqHeTemperatureLocale}),
value => $self->{cpqHeTemperature},
warning => $self->{cpqHeTemperatureThreshold},
critical => $self->{cpqHeTemperatureThreshold}
);
}
$self->add_extendedinfo(sprintf "temp_%s=%d",
$self->{cpqHeTemperatureIndex},
$self->{cpqHeTemperature});
}
sub dump {
my $self = shift;
printf "[TEMP_%s]\n", $self->{cpqHeTemperatureIndex};
foreach (qw(cpqHeTemperatureChassis cpqHeTemperatureIndex
cpqHeTemperatureLocale cpqHeTemperatureCelsius cpqHeTemperatureThreshold
cpqHeTemperatureCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n\n", $self->{info};
}
package HP::Proliant::Component::TemperatureSubsystem::CLI;
our @ISA = qw(HP::Proliant::Component::TemperatureSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
temperatures => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
my $tempcnt = 1;
foreach (grep(/^temp/, split(/\n/, $params{rawdata}))) {
s/^temp\s*//g;
if (/^#(\d+)\s+([\w_\/\-#]+)\s+(\d+)C\/(\d+)F\s+(\d+)C\/(\d+)F/) {
my %params = ();
$params{runtime} = $self->{runtime};
$params{cpqHeTemperatureChassis} = 1;
$params{cpqHeTemperatureIndex} = $1;
$params{cpqHeTemperatureLocale} = lc $2;
$params{cpqHeTemperatureCelsius} = $3;
$params{cpqHeTemperatureThresholdCelsius} = $5;
$params{cpqHeTemperatureCondition} = 'unknown';
push(@{$self->{temperatures}},
HP::Proliant::Component::TemperatureSubsystem::Temperature->new(
%params));
} elsif (/^#(\d+)\s+([\w_\/\-#]+)\s+\-\s+(\d+)C\/(\d+)F/) {
# #3 CPU#2 - 0C/0F
$self->trace(2, sprintf "skipping temperature %s", $_);
} elsif (/^#(\d+)\s+([\w_\/\-#]+)\s+(\d+)C\/(\d+)F\s+\-/) {
# #3 CPU#2 0C/0F -
$self->trace(2, sprintf "skipping temperature %s", $_);
} elsif (/^#(\d+)\s+([\w_\/\-#]+)\s+\-\s+\-/) {
# #3 CPU#2 - -
$self->trace(2, sprintf "skipping temperature %s", $_);
} elsif (/^#(\d+)/) {
$self->trace(0, sprintf "send this to lausser: %s", $_);
}
}
}
package HP::Proliant::Component::TemperatureSubsystem::SNMP;
our @ISA = qw(HP::Proliant::Component::TemperatureSubsystem
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
temperatures => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeTemperatureEntry => "1.3.6.1.4.1.232.6.2.6.8.1",
cpqHeTemperatureChassis => "1.3.6.1.4.1.232.6.2.6.8.1.1",
cpqHeTemperatureIndex => "1.3.6.1.4.1.232.6.2.6.8.1.2",
cpqHeTemperatureLocale => "1.3.6.1.4.1.232.6.2.6.8.1.3",
cpqHeTemperatureCelsius => "1.3.6.1.4.1.232.6.2.6.8.1.4",
cpqHeTemperatureThresholdCelsius => "1.3.6.1.4.1.232.6.2.6.8.1.5",
cpqHeTemperatureCondition => "1.3.6.1.4.1.232.6.2.6.8.1.6",
cpqHeTemperatureLocaleValue => {
1 => "other",
2 => "unknown",
3 => "system",
4 => "systemBoard",
5 => "ioBoard",
6 => "cpu",
7 => "memory",
8 => "storage",
9 => "removableMedia",
10 => "powerSupply",
11 => "ambient",
12 => "chassis",
13 => "bridgeCard",
},
cpqHeTemperatureConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
}
};
# INDEX { cpqHeTemperatureChassis, cpqHeTemperatureIndex }
foreach ($self->get_entries($oids, 'cpqHeTemperatureEntry')) {
push(@{$self->{temperatures}},
HP::Proliant::Component::TemperatureSubsystem::Temperature->new(%{$_}));
}
}
package HP::Proliant::Component::CpuSubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
################################## scrapiron ##########
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
condition => $params{condition},
status => $params{status},
cpus => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
return HP::Proliant::Component::CpuSubsystem::SNMP->new(%params);
} elsif ($self->{method} eq 'cli') {
return HP::Proliant::Component::CpuSubsystem::CLI->new(%params);
} else {
die "unknown method";
}
return $self;
}
sub check {
my $self = shift;
my $errorfound = 0;
$self->add_info('checking cpus');
if (scalar (@{$self->{cpus}}) == 0) {
# sachen gibts.....
# $self->overall_check(); # sowas ist mir nur einmal untergekommen
} else {
foreach (@{$self->{cpus}}) {
$_->check();
}
}
}
sub num_cpus {
my $self = shift;
return scalar @{$self->{cpus}};
}
sub dump {
my $self = shift;
foreach (@{$self->{cpus}}) {
$_->dump();
}
}
package HP::Proliant::Component::CpuSubsystem::Cpu;
our @ISA = qw(HP::Proliant::Component::CpuSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqSeCpuSlot => $params{cpqSeCpuSlot},
cpqSeCpuUnitIndex => $params{cpqSeCpuUnitIndex},
cpqSeCpuName => $params{cpqSeCpuName},
cpqSeCpuStatus => $params{cpqSeCpuStatus},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('c', $self->{cpqSeCpuUnitIndex});
if ($self->{cpqSeCpuStatus} ne "ok") {
if ($self->{runtime}->{options}{scrapiron} &&
($self->{cpqSeCpuStatus} eq "unknown")) {
$self->add_info(sprintf "cpu %d probably ok (%s)",
$self->{cpqSeCpuUnitIndex}, $self->{cpqSeCpuStatus});
} else {
$self->add_info(sprintf "cpu %d needs attention (%s)",
$self->{cpqSeCpuUnitIndex}, $self->{cpqSeCpuStatus});
$self->add_message(CRITICAL, $self->{info});
}
} else {
$self->add_info(sprintf "cpu %d is %s",
$self->{cpqSeCpuUnitIndex}, $self->{cpqSeCpuStatus});
}
$self->add_extendedinfo(sprintf "cpu_%s=%s",
$self->{cpqSeCpuUnitIndex}, $self->{cpqSeCpuStatus});
}
sub dump {
my $self = shift;
printf "[CPU_%s]\n", $self->{cpqSeCpuUnitIndex};
foreach (qw(cpqSeCpuSlot cpqSeCpuUnitIndex cpqSeCpuName cpqSeCpuStatus)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n", $self->{info};
printf "\n";
}
package HP::Proliant::Component::CpuSubsystem::CLI;
our @ISA = qw(HP::Proliant::Component::CpuSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
cpus => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
my %tmpcpu = (
runtime => $params{runtime},
);
my $inblock = 0;
foreach (grep(/^server/, split(/\n/, $self->{rawdata}))) {
if (/Processor:\s+(\d+)/) {
$tmpcpu{cpqSeCpuUnitIndex} = $1;
$inblock = 1;
} elsif (/Name\s*:\s+(.+?)\s*$/) {
$tmpcpu{cpqSeCpuName} = $1;
} elsif (/Status\s*:\s+(.+?)\s*$/) {
$tmpcpu{cpqSeCpuStatus} = lc $1;
} elsif (/Socket\s*:\s+(.+?)\s*$/) {
$tmpcpu{cpqSeCpuSlot} = $1;
} elsif (/^server\s*$/) {
if ($inblock) {
$inblock = 0;
push(@{$self->{cpus}},
HP::Proliant::Component::CpuSubsystem::Cpu->new(%tmpcpu));
%tmpcpu = (
runtime => $params{runtime},
);
}
}
}
if ($inblock) {
push(@{$self->{cpus}},
HP::Proliant::Component::CpuSubsystem::Cpu->new(%tmpcpu));
}
}
package HP::Proliant::Component::CpuSubsystem::SNMP;
our @ISA = qw(HP::Proliant::Component::CpuSubsystem
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
cpus => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQSTDEQ-MIB
my $oids = {
cpqSeCpuEntry => '1.3.6.1.4.1.232.1.2.2.1.1',
cpqSeCpuUnitIndex => '1.3.6.1.4.1.232.1.2.2.1.1.1',
cpqSeCpuSlot => '1.3.6.1.4.1.232.1.2.2.1.1.2',
cpqSeCpuName => '1.3.6.1.4.1.232.1.2.2.1.1.3',
cpqSeCpuStatus => '1.3.6.1.4.1.232.1.2.2.1.1.6',
cpqSeCpuStatusValue => {
1 => "unknown",
2 => "ok",
3 => "degraded",
4 => "failed",
5 => "disabled",
},
};
# INDEX { cpqSeCpuUnitIndex }
foreach ($self->get_entries($oids, 'cpqSeCpuEntry')) {
push(@{$self->{cpus}},
HP::Proliant::Component::CpuSubsystem::Cpu->new(%{$_}));
}
}
package HP::Proliant::Component::FanSubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
################################## fan_redundancy ##########
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
condition => $params{condition},
status => $params{status},
fans => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
return HP::Proliant::Component::FanSubsystem::SNMP->new(%params);
} elsif ($self->{method} eq 'cli') {
return HP::Proliant::Component::FanSubsystem::CLI->new(%params);
} else {
die 'unknown method';
}
return $self;
}
sub check {
my $self = shift;
my $errorfound = 0;
$self->add_info('checking fans');
$self->blacklist('ff', '');
if (scalar (@{$self->{fans}}) == 0) {
$self->overall_check(); # sowas ist mir nur einmal untergekommen
# die maschine hatte alles in allem nur 2 oids (cpqHeFltTolFanChassis)
# SNMPv2-SMI::enterprises.232.6.2.6.7.1.1.0.1 = INTEGER: 0
# SNMPv2-SMI::enterprises.232.6.2.6.7.1.1.0.2 = INTEGER: 0
} else {
my $overallhealth = $self->overall_check();
foreach (@{$self->{fans}}) {
$_->{overallhealth} = $overallhealth;
$_->check();
}
}
}
sub dump {
my $self = shift;
foreach (@{$self->{fans}}) {
$_->dump();
}
}
sub get_fan_by_index {
my $self = shift;
my $index;
foreach (@{$self->{fans}}) {
return $_ if exists $_->{cpqHeFltTolFanIndex} &&
$_->{cpqHeFltTolFanIndex} == $index;
}
return undef;
}
package HP::Proliant::Component::FanSubsystem::Fan;
our @ISA = qw(HP::Proliant::Component::FanSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
if (exists $params{cpqHeFltTolFanRedundant}) {
return HP::Proliant::Component::FanSubsystem::Fan::FTol->new(%params);
} else {
return HP::Proliant::Component::FanSubsystem::Fan::Thermal->new(%params);
}
}
package HP::Proliant::Component::FanSubsystem::Fan::FTol;
our @ISA = qw(HP::Proliant::Component::FanSubsystem::Fan);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqHeFltTolFanChassis => $params{cpqHeFltTolFanChassis},
cpqHeFltTolFanIndex => $params{cpqHeFltTolFanIndex},
cpqHeFltTolFanLocale => $params{cpqHeFltTolFanLocale},
cpqHeFltTolFanPresent => $params{cpqHeFltTolFanPresent},
cpqHeFltTolFanType => $params{cpqHeFltTolFanType},
cpqHeFltTolFanSpeed => $params{cpqHeFltTolFanSpeed},
cpqHeFltTolFanRedundant => $params{cpqHeFltTolFanRedundant},
cpqHeFltTolFanRedundantPartner => $params{cpqHeFltTolFanRedundantPartner},
cpqHeFltTolFanCondition => $params{cpqHeFltTolFanCondition},
cpqHeFltTolFanPctMax => $params{cpqHeFltTolFanPctMax}, #!!!
cpqHeFltTolFanHotPlug => $params{cpqHeFltTolFanHotPlug}, #!!!
partner => $params{partner},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
if (($self->{cpqHeFltTolFanRedundant} eq 'redundant') &&
((! defined $self->{cpqHeFltTolFanRedundantPartner}) ||
(! $self->{cpqHeFltTolFanRedundantPartner}))) {
$self->{cpqHeFltTolFanRedundant} = 'notRedundant';
# cpqHeFltTolFanRedundantPartner=0: partner not avail
}
return $self;
}
sub check {
my $self = shift;
$self->blacklist('f', $self->{cpqHeFltTolFanIndex});
$self->add_info(sprintf 'fan %d is %s, speed is %s, pctmax is %s%%, '.
'location is %s, redundance is %s, partner is %s',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanPresent},
$self->{cpqHeFltTolFanSpeed}, $self->{cpqHeFltTolFanPctMax},
$self->{cpqHeFltTolFanLocale}, $self->{cpqHeFltTolFanRedundant},
$self->{cpqHeFltTolFanRedundantPartner});
$self->add_extendedinfo(sprintf 'fan_%s=%d%%',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanPctMax});
if ($self->{cpqHeFltTolFanPresent} eq 'present') {
if ($self->{cpqHeFltTolFanSpeed} eq 'high') {
$self->add_info(sprintf 'fan %d (%s) runs at high speed',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(CRITICAL, $self->{info});
} elsif ($self->{cpqHeFltTolFanSpeed} ne 'normal') {
$self->add_info(sprintf 'fan %d (%s) needs attention',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(CRITICAL, $self->{info});
}
if ($self->{cpqHeFltTolFanCondition} eq 'failed') {
$self->add_info(sprintf 'fan %d (%s) failed',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(CRITICAL, $self->{info});
} elsif ($self->{cpqHeFltTolFanCondition} eq 'degraded') {
$self->add_info(sprintf 'fan %d (%s) degraded',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(WARNING, $self->{info});
} elsif ($self->{cpqHeFltTolFanCondition} ne 'ok') {
$self->add_info(sprintf 'fan %d (%s) is not ok',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(WARNING, $self->{info});
}
if ($self->{cpqHeFltTolFanRedundant} eq 'notRedundant') {
# sieht so aus, als waere notRedundant und partner=0 normal z.b. dl360
# das duerfte der fall sein, wenn nur eine cpu verbaut wurde und
# statt einem redundanten paar nur dummies drinstecken.
# "This specifies if the fan is in a redundant configuration"
# notRedundant heisst also sowohl nicht redundant wegen ausfall
# des partners als auch von haus aus nicht redundant ausgelegt
if ($self->{cpqHeFltTolFanRedundantPartner}) {
# nicht redundant, hat aber einen partner. da muss man genauer
# hinschauen
#if (my $partner = $self->{partner}) {
#}
if ($self->{overallhealth}) {
# da ist sogar das system der meinung, dass etwas faul ist
if (! $self->{runtime}->{options}->{ignore_fan_redundancy}) {
$self->add_info(sprintf 'fan %d (%s) is not redundant',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(WARNING, $self->{info});
}
} else {
# das ist wohl so gewollt, dass einzelne fans eingebaut werden,
# obwohl redundante paerchen vorgesehen sind.
# scheint davon abzuhaengen, wieviele cpus geordert wurden.
}
}
} elsif ($self->{cpqHeFltTolFanRedundant} eq 'other') {
#seen on a dl320 g5p with bios from 2008.
# maybe redundancy is not supported at all
}
} elsif ($self->{cpqHeFltTolFanPresent} eq 'failed') { # from cli
$self->add_info(sprintf 'fan %d (%s) failed',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
$self->add_message(CRITICAL, $self->{info});
} elsif ($self->{cpqHeFltTolFanPresent} eq 'absent') {
$self->add_info(sprintf 'fan %d (%s) needs attention (is absent)',
$self->{cpqHeFltTolFanIndex}, $self->{cpqHeFltTolFanLocale});
# weiss nicht, ob absent auch kaputt bedeuten kann
# wenn nicht, dann wuerde man sich hier dumm und daemlich blacklisten
#$self->add_message(CRITICAL, $self->{info});
$self->add_message(WARNING, $self->{info}) if $self->{overallhealth};
}
if ($self->{runtime}->{options}->{perfdata}) {
$self->{runtime}->{plugin}->add_perfdata(
label => sprintf('fan_%s', $self->{cpqHeFltTolFanIndex}),
value => $self->{cpqHeFltTolFanPctMax},
uom => '%',
);
}
}
sub dump {
my $self = shift;
printf "[FAN_%s]\n", $self->{cpqHeFltTolFanIndex};
foreach (qw(cpqHeFltTolFanChassis cpqHeFltTolFanIndex cpqHeFltTolFanLocale
cpqHeFltTolFanPresent cpqHeFltTolFanType cpqHeFltTolFanSpeed
cpqHeFltTolFanRedundant cpqHeFltTolFanRedundantPartner
cpqHeFltTolFanCondition cpqHeFltTolFanHotPlug)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n", $self->{info};
printf "\n";
}
package HP::Proliant::Component::FanSubsystem::Fan::Thermal;
our @ISA = qw(HP::Proliant::Component::FanSubsystem::Fan);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqHeThermalFanIndex => $params{cpqHeThermalFanIndex},
cpqHeThermalFanRequired => $params{cpqHeThermalFanRequired},
cpqHeThermalFanPresent => $params{cpqHeThermalFanPresent},
cpqHeThermalFanCpuFan => $params{cpqHeThermalFanCpuFan},
cpqHeThermalFanStatus => $params{cpqHeThermalFanStatus},
cpqHeThermalFanHwLocation => $params{cpqHeThermalFanHwLocation},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
}
sub dump {
my $self = shift;
printf "[FAN_%s]\n", $self->{cpqHeThermalFanIndex};
foreach (qw(cpqHeThermalFanIndex cpqHeThermalFanRequired
cpqHeThermalFanPresent cpqHeThermalFanCpuFan cpqHeThermalFanStatus
cpqHeThermalFanHwLocation)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n", $self->{info};
printf "\n";
}
package HP::Proliant::Component::FanSubsystem::CLI;
our @ISA = qw(HP::Proliant::Component::FanSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
fans => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
# partner not available = cpqHeFltTolFanRedundantPartner=0
# cpqHeFltTolFanTypeValue = other
sub init {
my $self = shift;
my %params = @_;
my %tmpfan = ();
foreach (grep(/^fans/, split(/\n/, $self->{rawdata}))) {
s/^fans //g;
if (/^#(\d+)\s+([\w#_\/\-]+)\s+(\w+)\s+(\w+)\s+(FAILED|[N\/A\d]+)%*\s+([\w\/]+)\s+(FAILED|[N\/A\d]+)\s+(\w+)/) {
%tmpfan = (
cpqHeFltTolFanIndex => $1,
cpqHeFltTolFanLocale => lc $2,
cpqHeFltTolFanPresent => lc $3,
cpqHeFltTolFanSpeed => lc $4,
cpqHeFltTolFanPctMax => lc $5, # (FAILED|[N\/A\d]+)
cpqHeFltTolFanRedundant => lc $6,
cpqHeFltTolFanRedundantPartner => lc $7, # (FAILED|[N\/A\d]+)
cpqHeFltTolFanHotPlug => lc $8,
);
} elsif (/^#(\d+)\s+([\w#_\/\-]+?)(Yes|No|N\/A)\s+(\w+)\s+(FAILED|[N\/A\d]+)%*\s+([\w\/]+)\s+(FAILED|[N\/A\d]+)\s+(\w+)/) {
# #5 SCSI_BACKPLANE_ZONEYes NORMAL N/A ....
%tmpfan = (
cpqHeFltTolFanIndex => $1,
cpqHeFltTolFanLocale => lc $2,
cpqHeFltTolFanPresent => lc $3,
cpqHeFltTolFanSpeed => lc $4,
cpqHeFltTolFanPctMax => lc $5,
cpqHeFltTolFanRedundant => lc $6,
cpqHeFltTolFanRedundantPartner => lc $7,
cpqHeFltTolFanHotPlug => lc $8,
);
} elsif (/^#(\d+)\s+([\w#_\/\-]+)\s+[NOno]+\s/) {
# Fan is not installed. #2 CPU#2 No - - No N/A -
} elsif (/^#(\d+)/) {
main::contact_author("FAN", $_);
}
if (%tmpfan) {
$tmpfan{runtime} = $params{runtime};
$tmpfan{cpqHeFltTolFanChassis} = 1; # geht aus hpasmcli nicht hervor
$tmpfan{cpqHeFltTolFanType} = 'other';
if ($tmpfan{cpqHeFltTolFanPctMax} !~ /^\d+$/) {
if ($tmpfan{cpqHeFltTolFanSpeed} eq 'normal') {
$tmpfan{cpqHeFltTolFanPctMax} = 50;
} elsif ($tmpfan{cpqHeFltTolFanSpeed} eq 'high') {
$tmpfan{cpqHeFltTolFanPctMax} = 100;
} else {
$tmpfan{cpqHeFltTolFanPctMax} = 0;
}
}
if($tmpfan{cpqHeFltTolFanSpeed} eq 'failed') {
$tmpfan{cpqHeFltTolFanCondition} = 'failed';
} elsif($tmpfan{cpqHeFltTolFanSpeed} eq 'n/a') {
$tmpfan{cpqHeFltTolFanCondition} = 'other';
} else {
$tmpfan{cpqHeFltTolFanCondition} = 'ok';
}
$tmpfan{cpqHeFltTolFanRedundant} =
$tmpfan{cpqHeFltTolFanRedundant} eq 'yes' ? 'redundant' :
$tmpfan{cpqHeFltTolFanRedundant} eq 'no' ? 'notRedundant' : 'other';
$tmpfan{cpqHeFltTolFanPresent} =
$tmpfan{cpqHeFltTolFanPresent} eq 'yes' ? 'present' :
$tmpfan{cpqHeFltTolFanPresent} eq 'failed' ? 'present' :
$tmpfan{cpqHeFltTolFanPresent} eq 'no' ? 'absent' : 'other';
$tmpfan{cpqHeFltTolFanHotPlug} =
$tmpfan{cpqHeFltTolFanHotPlug} eq 'yes' ? 'hotPluggable' :
$tmpfan{cpqHeFltTolFanHotPlug} eq 'no' ? 'nonHotPluggable' : 'other';
push(@{$self->{fans}},
HP::Proliant::Component::FanSubsystem::Fan->new(%tmpfan));
%tmpfan = ();
}
}
}
sub overall_check {
my $self = shift;
# nix. nur wegen der gleichheit mit snmp
return 0;
}
package HP::Proliant::Component::FanSubsystem::SNMP;
our @ISA = qw(HP::Proliant::Component::FanSubsystem
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
fans => [],
he_fans => [],
th_fans => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->overall_init(%params);
$self->he_init(%params);
$self->te_init(%params);
$self->unite();
return $self;
}
sub overall_init {
my $self = shift;
my %params = @_;
my $snmpwalk = $params{rawdata};
# overall
my $cpqHeThermalSystemFanStatus = '1.3.6.1.4.1.232.6.2.6.4.0';
my $cpqHeThermalSystemFanStatusValue = {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
};
my $cpqHeThermalCpuFanStatus = '1.3.6.1.4.1.232.6.2.6.5.0';
my $cpqHeThermalCpuFanStatusValue = {
1 => 'other',
2 => 'ok',
4 => 'failed', # shutdown
};
$self->{sysstatus} = lc SNMP::Utils::get_object_value(
$snmpwalk, $cpqHeThermalSystemFanStatus,
$cpqHeThermalSystemFanStatusValue);
$self->{cpustatus} = lc SNMP::Utils::get_object_value(
$snmpwalk, $cpqHeThermalCpuFanStatus,
$cpqHeThermalCpuFanStatusValue);
}
sub te_init {
my $self = shift;
my %params = @_;
my $snmpwalk = $params{rawdata};
my $ignore_redundancy = $params{ignore_redundancy};
# cpqHeThermalFanTable
my $oids = {
cpqHeThermalFanEntry => '1.3.6.1.4.1.232.6.2.6.6.1',
cpqHeThermalFanIndex => '1.3.6.1.4.1.232.6.2.6.6.1.1',
cpqHeThermalFanRequired => '1.3.6.1.4.1.232.6.2.6.6.1.2',
cpqHeThermalFanPresent => '1.3.6.1.4.1.232.6.2.6.6.1.3',
cpqHeThermalFanCpuFan => '1.3.6.1.4.1.232.6.2.6.6.1.4',
cpqHeThermalFanStatus => '1.3.6.1.4.1.232.6.2.6.6.1.5',
cpqHeThermalFanHwLocation => '1.3.6.1.4.1.232.6.2.6.6.1.6',
cpqHeThermalFanRequiredValue => {
1 => 'other',
2 => 'nonRequired',
3 => 'required',
},
cpqHeThermalFanPresentValue => {
1 => 'other',
2 => 'absent',
3 => 'present',
},
cpqHeThermalFanCpuFanValue => {
1 => 'other',
2 => 'systemFan',
3 => 'cpuFan',
},
cpqHeThermalFanStatusValue => {
1 => 'other',
2 => 'ok',
4 => 'failed',
},
};
# INDEX { cpqHeThermalFanIndex }
foreach ($self->get_entries($oids, 'cpqHeThermalFanEntry')) {
next if ! $_->{cpqHeThermalFanPresent};
push(@{$self->{th_fans}},
HP::Proliant::Component::FanSubsystem::Fan->new(%{$_}));
}
}
sub he_init {
my $self = shift;
my %params = @_;
my $snmpwalk = $params{rawdata};
my $ignore_redundancy = $params{ignore_redundancy};
# cpqHeFltTolFanTable
my $oids = {
cpqHeFltTolFanEntry => '1.3.6.1.4.1.232.6.2.6.7.1',
cpqHeFltTolFanChassis => '1.3.6.1.4.1.232.6.2.6.7.1.1',
cpqHeFltTolFanIndex => '1.3.6.1.4.1.232.6.2.6.7.1.2',
cpqHeFltTolFanLocale => '1.3.6.1.4.1.232.6.2.6.7.1.3',
cpqHeFltTolFanPresent => '1.3.6.1.4.1.232.6.2.6.7.1.4',
cpqHeFltTolFanType => '1.3.6.1.4.1.232.6.2.6.7.1.5',
cpqHeFltTolFanSpeed => '1.3.6.1.4.1.232.6.2.6.7.1.6',
cpqHeFltTolFanRedundant => '1.3.6.1.4.1.232.6.2.6.7.1.7',
cpqHeFltTolFanRedundantPartner => '1.3.6.1.4.1.232.6.2.6.7.1.8',
cpqHeFltTolFanCondition => '1.3.6.1.4.1.232.6.2.6.7.1.9',
cpqHeFltTolFanHotPlug => '1.3.6.1.4.1.232.6.2.6.7.1.10',
cpqHeFltTolFanHwLocation => '1.3.6.1.4.1.232.6.2.6.7.1.11',
cpqHeFltTolFanCurrentSpeed => '1.3.6.1.4.1.232.6.2.6.7.1.12',
cpqHeFltTolFanLocaleValue => {
1 => "other",
2 => "unknown",
3 => "system",
4 => "systemBoard",
5 => "ioBoard",
6 => "cpu",
7 => "memory",
8 => "storage",
9 => "removableMedia",
10 => "powerSupply",
11 => "ambient",
12 => "chassis",
13 => "bridgeCard",
},
cpqHeFltTolFanPresentValue => {
1 => "other",
2 => "absent",
3 => "present",
},
cpqHeFltTolFanSpeedValue => {
1 => "other",
2 => "normal",
3 => "high",
},
cpqHeFltTolFanRedundantValue => {
1 => "other",
2 => "notRedundant",
3 => "redundant",
},
cpqHeFltTolFanTypeValue => {
1 => "other",
2 => "tachInput",
3 => "spinDetect",
},
cpqHeFltTolFanConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqHeFltTolFanHotPlugValue => {
1 => "other",
2 => "nonHotPluggable",
3 => "hotPluggable",
},
};
# INDEX { cpqHeFltTolFanChassis, cpqHeFltTolFanIndex }
foreach ($self->get_entries($oids, 'cpqHeFltTolFanEntry')) {
next if ! defined $_->{cpqHeFltTolFanIndex};
# z.b. USM65201WS hat nur solche fragmente. die werden erst gar nicht
# als fans akzeptiert. dafuer gibts dann die overall condition
# SNMPv2-SMI::enterprises.232.6.2.6.7.1.1.0.1 = INTEGER: 0
# SNMPv2-SMI::enterprises.232.6.2.6.7.1.1.0.2 = INTEGER: 0
$_->{cpqHeFltTolFanPctMax} = ($_->{cpqHeFltTolFanPresent} eq 'present') ?
50 : 0;
push(@{$self->{he_fans}},
HP::Proliant::Component::FanSubsystem::Fan->new(%{$_}));
}
}
sub unite {
my $self = shift;
my $tmpfans = {};
foreach (@{$self->{he_fans}}) {
$tmpfans->{$_->{cpqHeFltTolFanIndex}} = $_;
}
foreach (@{$self->{he_fans}}) {
if (exists $tmpfans->{$_->{cpqHeFltTolFanRedundantPartner}}) {
$_->{partner} = $tmpfans->{$_->{cpqHeFltTolFanRedundantPartner}};
} else {
$_->{partner} = undef;
}
}
@{$self->{fans}} = @{$self->{he_fans}};
}
sub overall_check {
my $self = shift;
my $result = 0;
if ($self->{sysstatus} && $self->{cpustatus}) {
if ($self->{sysstatus} eq 'degraded') {
$result = 1;
$self->add_message(WARNING,
sprintf 'system fan overall status is %s', $self->{sysstatus});
} elsif ($self->{sysstatus} eq 'failed') {
$result = 2;
$self->add_message(CRITICAL,
sprintf 'system fan overall status is %s', $self->{sysstatus});
}
if ($self->{cpustatus} eq 'degraded') {
$result = 1;
$self->add_message(WARNING,
sprintf 'cpu fan overall status is %s', $self->{cpustatus});
} elsif ($self->{cpustatus} eq 'failed') {
$result = 2;
$self->add_message(CRITICAL,
sprintf 'cpu fan overall status is %s', $self->{cpustatus});
}
$self->add_info(sprintf 'overall fan status: system=%s, cpu=%s',
$self->{sysstatus}, $self->{cpustatus});
} else {
$result = 0;
$self->add_info('this system seems to be water-cooled. no fans found');
}
return $result;
}
package HP::Proliant::Component::MemorySubsystem::CLI;
our @ISA = qw(HP::Proliant::Component::MemorySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
dimms => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init(%params);
return $self;
}
sub init {
my $self = shift;
my %params = @_;
$self->{dimms} = [];
my %tmpdimm = (
runtime => $params{runtime},
);
my $inblock = 0;
foreach (grep(/^dimm/, split(/\n/, $self->{rawdata}))) {
s/^dimm\s*$//g;
if (/Cartridge #:\s+(\d+)/ || /Processor #:\s+(\d+)/) {
# neuerdings (g6) tauchen hier prozessor- statt cartridge-angaben auf
$tmpdimm{cartridge} = $1;
$tmpdimm{board} = $1;
$inblock = 1;
} elsif (/Module #:\s+(\d+)/) {
$tmpdimm{module} = $1;
} elsif (/Present:\s+(\w+)/) {
$tmpdimm{status} = lc $1 eq 'yes' ? 'present' :
lc $1 eq 'no' ? 'notPresent' : 'other';
} elsif (/Status:\s+(.+?)\s*$/) {
$tmpdimm{condition} = lc $1 =~ /degraded/ ? 'degraded' :
lc $1 eq 'ok' ? 'ok' : lc $1 =~ /n\/a/ ? 'n/a' : 'other';
} elsif (/Size:\s+(\d+)\s*(.+?)\s*$/) {
$tmpdimm{size} = $1 * (lc $2 eq 'mb' ? 1024*1024 :
lc $2 eq 'gb' ? 1024*1024*1024 : 1);
} elsif (/^\s*$/) {
if ($inblock) {
$inblock = 0;
push(@{$self->{dimms}},
HP::Proliant::Component::MemorySubsystem::Dimm->new(%tmpdimm));
%tmpdimm = (
runtime => $params{runtime},
);
}
}
}
if ($inblock) {
push(@{$self->{dimms}},
HP::Proliant::Component::MemorySubsystem::Dimm->new(%tmpdimm));
}
}
sub is_faulty {
my $self = shift;
return 0; # cli hat so einen globalen status nicht
}
package HP::Proliant::Component::MemorySubsystem::SNMP;
our @ISA = qw(HP::Proliant::Component::MemorySubsystem
HP::Proliant::Component::SNMP);
use strict;
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
dimms => [],
si_dimms => [],
he_dimms => [],
h2_dimms => [],
he_cartridges => [],
h2_cartridges => [],
si_overall_condition => undef,
he_overall_condition => undef,
h2_overall_condition => undef,
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->si_init();
$self->he_init();
$self->he_cartridge_init();
$self->h2_init();
#$self->h2_cartridge_init();
$self->condense();
return $self;
}
sub si_init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqSiMemModuleEntry => '1.3.6.1.4.1.232.2.2.4.5.1',
cpqSiMemBoardIndex => '1.3.6.1.4.1.232.2.2.4.5.1.1',
cpqSiMemModuleIndex => '1.3.6.1.4.1.232.2.2.4.5.1.2',
cpqSiMemModuleSize => '1.3.6.1.4.1.232.2.2.4.5.1.3',
cpqSiMemModuleType => '1.3.6.1.4.1.232.2.2.4.5.1.4',
cpqSiMemECCStatus => '1.3.6.1.4.1.232.2.2.4.5.1.11',
cpqSiMemModuleHwLocation => '1.3.6.1.4.1.232.2.2.4.5.1.12',
cpqSiMemModuleTypeValue => {
1 => 'other',
2 => 'board',
3 => 'cpqSingleWidthModule',
4 => 'cpqDoubleWidthModule',
5 => 'simm',
6 => 'pcmcia',
7 => 'compaq-specific',
8 => 'dimm',
9 => 'smallOutlineDimm',
10 => 'rimm',
11 => 'srimm',
},
cpqSiMemECCStatusValue => {
0 => "n/a",
1 => "other",
2 => "ok",
3 => "degraded",
4 => "degradedModuleIndexUnknown",
34 => 'n/a', # es ist zum kotzen...
104 => 'n/a',
},
};
# INDEX { cpqSiMemBoardIndex, cpqSiMemModuleIndex }
foreach ($self->get_entries($oids, 'cpqSiMemModuleEntry')) {
$_->{cartridge} = $_->{cpqSiMemBoardIndex};
$_->{module} = $_->{cpqSiMemModuleIndex};
next if (! defined $_->{cartridge} || ! defined $_->{module});
$_->{size} = $_->{cpqSiMemModuleSize};
$_->{type} = $_->{cpqSiMemModuleType};
$_->{condition} = $_->{cpqSiMemECCStatus};
$_->{status} = ($_->{cpqSiMemModuleSize} > 0) ? 'present' : 'notPresent';
push(@{$self->{si_dimms}},
HP::Proliant::Component::MemorySubsystem::Dimm->new(%{$_})
);
}
my $cpqSiMemECCCondition = '1.3.6.1.4.1.232.2.2.4.15.0';
my $cpqSiMemECCConditionValue = {
1 => 'other',
2 => 'ok',
3 => 'degraded',
};
$self->{si_overall_condition} = SNMP::Utils::get_object_value(
$self->{rawdata}, $cpqSiMemECCCondition,
$cpqSiMemECCConditionValue);
$self->trace(2, sprintf 'overall si condition is %s',
$self->{si_overall_condition} || 'undefined');
}
sub he_init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeResMemModuleEntry => '1.3.6.1.4.1.232.6.2.14.11.1',
cpqHeResMemBoardIndex => '1.3.6.1.4.1.232.6.2.14.11.1.1',
cpqHeResMemModuleIndex => '1.3.6.1.4.1.232.6.2.14.11.1.2',
cpqHeResMemModuleStatus => '1.3.6.1.4.1.232.6.2.14.11.1.4',
cpqHeResMemModuleCondition => '1.3.6.1.4.1.232.6.2.14.11.1.5',
cpqHeResMemModuleStatusValue => {
1 => "other", # unknown or could not be determined
2 => "notPresent", # not present or un-initialized
3 => "present", # present but not in use
4 => "good", # present and in use. ecc threshold not exceeded
5 => "add", # added but not yet in use
6 => "upgrade", # upgraded but not yet in use
7 => "missing", # expected but missing
8 => "doesNotMatch", # does not match the other modules in the bank
9 => "notSupported", # module not supported
10 => "badConfig", # violates add/upgrade configuration
11 => "degraded", # ecc exceeds threshold
},
# condition = status of the correctable memory errors
cpqHeResMemModuleConditionValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "ok",
3 => "degraded",
},
};
my $tablesize = SNMP::Utils::get_size($snmpwalk,
$oids->{cpqHeResMemModuleEntry});
# INDEX { cpqHeResMemBoardIndex, cpqHeResMemModuleIndex }
foreach ($self->get_entries($oids, 'cpqHeResMemModuleEntry')) {
$_->{cartridge} = $_->{cpqHeResMemBoardIndex};
$_->{module} = $_->{cpqHeResMemModuleIndex};
$_->{present} = $_->{cpqHeResMemModuleStatus};
$_->{status} = $_->{cpqHeResMemModuleStatus};
$_->{condition} = $_->{cpqHeResMemModuleCondition};
if ((! defined $_->{module}) && ($_->{cartridge} == 0)) {
$_->{module} = $_->{index2}; # auf dem systemboard verbaut
}
push(@{$self->{he_dimms}},
HP::Proliant::Component::MemorySubsystem::Dimm->new(%{$_})
) unless (! defined $_->{cartridge} || ! defined $_->{module} ||
$tablesize == 1);
}
my $cpqHeResilientMemCondition = '1.3.6.1.4.1.232.6.2.14.4.0';
my $cpqHeResilientMemConditionValue = {
1 => 'other',
2 => 'ok',
3 => 'degraded',
};
$self->{he_overall_condition} = SNMP::Utils::get_object_value(
$self->{rawdata}, $cpqHeResilientMemCondition,
$cpqHeResilientMemConditionValue);
$self->trace(2, sprintf 'overall he condition is %s',
$self->{hei_overall_condition} || 'undefined');
}
sub he_cartridge_init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeResMemBoardEntry => '1.3.6.1.4.1.232.6.2.14.10.1',
cpqHeResMemBoardSlotIndex => '1.3.6.1.4.1.232.6.2.14.10.1.1',
cpqHeResMemBoardOnlineStatus => '1.3.6.1.4.1.232.6.2.14.10.1.2',
cpqHeResMemBoardErrorStatus => '1.3.6.1.4.1.232.6.2.14.10.1.3',
cpqHeResMemBoardNumSockets => '1.3.6.1.4.1.232.6.2.14.10.1.5',
cpqHeResMemBoardOsMemSize => '1.3.6.1.4.1.232.6.2.14.10.1.6',
cpqHeResMemBoardTotalMemSize => '1.3.6.1.4.1.232.6.2.14.10.1.7',
cpqHeResMemBoardCondition => '1.3.6.1.4.1.232.6.2.14.10.1.8',
# onlinestatus
cpqHeResMemBoardOnlineStatusValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "present",
3 => "absent",
},
cpqHeResMemBoardErrorStatusValue => {
1 => "other", #
2 => "noError", #
3 => "dimmEccError", #
4 => "unlockError", #
5 => "configError", #
6 => "busError", #
7 => "powerError", #
},
# condition = status of the correctable memory errors
cpqHeResMemBoardConditionValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "ok",
3 => "degraded",
},
};
my $tablesize = SNMP::Utils::get_size($snmpwalk,
$oids->{cpqHeResMemBoardEntry});
# INDEX { cpqHeResMemBoardIndex, cpqHeResMemBoardIndex }
foreach ($self->get_entries($oids, 'cpqHeResMemBoardEntry')) {
push(@{$self->{he_cartridges}},
HP::Proliant::Component::MemorySubsystem::Cartridge->new(%{$_})
) unless (! defined $_->{cpqHeResMemBoardSlotIndex} || $tablesize == 1);
}
}
sub h2_init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeResMem2ModuleEntry => '1.3.6.1.4.1.232.6.2.14.13.1',
cpqHeResMem2BoardNum => '1.3.6.1.4.1.232.6.2.14.13.1.2',
cpqHeResMem2ModuleNum => '1.3.6.1.4.1.232.6.2.14.13.1.5',
cpqHeResMem2ModuleStatus => '1.3.6.1.4.1.232.6.2.14.13.1.19',
cpqHeResMem2ModuleCondition => '1.3.6.1.4.1.232.6.2.14.13.1.20',
cpqHeResMem2ModuleSize => '1.3.6.1.4.1.232.6.2.14.13.1.6',
cpqHeResMem2ModuleStatusValue => {
1 => "other", # unknown or could not be determined
2 => "notPresent", # not present or un-initialized
3 => "present", # present but not in use
4 => "good", # present and in use. ecc threshold not exceeded
5 => "add", # added but not yet in use
6 => "upgrade", # upgraded but not yet in use
7 => "missing", # expected but missing
8 => "doesNotMatch", # does not match the other modules in the bank
9 => "notSupported", # module not supported
10 => "badConfig", # violates add/upgrade configuration
11 => "degraded", # ecc exceeds threshold
},
# condition = status of the correctable memory errors
cpqHeResMem2ModuleConditionValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "ok",
3 => "degraded",
},
};
# INDEX { cpqHeResMem2ModuleNum }
my $lastboard = 0;
my $lastmodule = 0;
my $myboard= 0;
my $hpboard = 0;
foreach (sort { $a->{index1} <=> $b->{index1} }
$self->get_entries($oids, 'cpqHeResMem2ModuleEntry')) {
$hpboard = $_->{cpqHeResMem2BoardNum};
# dass hier faelschlicherweise 0 zurueckkommt, wundert mich schon
# gar nicht mehr
$_->{module} = $_->{cpqHeResMem2ModuleNum};
if ($_->{module} < $lastmodule) {
# sieht so aus, als haette man es mit einem neuen board zu tun
# da hp zu bloed ist, selber hochzuzaehlen, muss ich das tun
$myboard++;
}
$lastmodule = $_->{cpqHeResMem2ModuleNum};
$_->{cartridge} = ($myboard != $hpboard) ? $myboard : $hpboard;
$_->{present} = $_->{cpqHeResMem2ModuleStatus};
$_->{status} = $_->{cpqHeResMem2ModuleStatus};
$_->{condition} = $_->{cpqHeResMem2ModuleCondition};
$_->{size} = $_->{cpqHeResMem2ModuleSize};
push(@{$self->{h2_dimms}},
HP::Proliant::Component::MemorySubsystem::Dimm->new(%{$_})
) unless (! defined $_->{cpqHeResMem2BoardNum} ||
! defined $_->{cpqHeResMem2ModuleNum});
}
}
sub h2_cartridge_init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my $oids = {
cpqHeResMem2BoardEntry => '1.3.6.1.4.1.232.6.2.14.12.1',
cpqHeResMem2BoardIndex => '1.3.6.1.4.1.232.6.2.14.12.1.1',
cpqHeResMem2BoardOnlineStatus => '1.3.6.1.4.1.232.6.2.14.12.1.5',
cpqHeResMem2BoardErrorStatus => '1.3.6.1.4.1.232.6.2.14.12.1.6',
cpqHeResMem2BoardNumSockets => '1.3.6.1.4.1.232.6.2.14.12.1.8',
cpqHeResMem2BoardOsMemSize => '1.3.6.1.4.1.232.6.2.14.12.1.9',
cpqHeResMem2BoardTotalMemSize => '1.3.6.1.4.1.232.6.2.14.12.1.10',
cpqHeResMem2BoardCondition => '1.3.6.1.4.1.232.6.2.14.12.1.11',
# onlinestatus
cpqHeResMem2BoardOnlineStatusValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "present",
3 => "absent",
},
cpqHeResMem2BoardErrorStatusValue => {
1 => "other", #
2 => "noError", #
3 => "dimmEccError", #
4 => "unlockError", #
5 => "configError", #
6 => "busError", #
7 => "powerError", #
},
# condition = status of the correctable memory errors
cpqHeResMem2BoardConditionValue => {
0 => "n/a", # this appears only with buggy firmwares.
# (only 1 module shows up)
1 => "other",
2 => "ok",
3 => "degraded",
},
};
my $tablesize = SNMP::Utils::get_size($snmpwalk,
$oids->{cpqHeResMemBoardEntry});
# INDEX { cpqHeResMem2BoardIndex, cpqHeResMem2BoardIndex }
foreach ($self->get_entries($oids, 'cpqHeResMem2BoardEntry')) {
push(@{$self->{h2_cartridges}},
HP::Proliant::Component::MemorySubsystem::Cartridge->new(%{$_})
) unless (! defined $_->{cpqHeRes2MemBoardIndex} || $tablesize == 1);
}
}
sub condense {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# wenn saemtliche dimms n/a sind
# wenn ignore dimms: ignoring %d dimms with status 'n/a'
# wenn buggyfirmware: ignoring %d dimms with status 'n/a' because of buggy firmware
# if buggy firmware : condition n/a ist normal
# ignore-dimms :
# es gibt si_dimms und he_dimms
my $si_dimms = scalar(@{$self->{si_dimms}});
my $he_dimms = scalar(@{$self->{he_dimms}});
my $h2_dimms = scalar(@{$self->{h2_dimms}});
$self->trace(2, sprintf "SI: %02d HE: %02d H2: %02d",
$si_dimms, $he_dimms, $h2_dimms)
if ($self->{runtime}->{options}->{verbose} >= 2);
foreach ($self->get_si_boards()) {
printf "SI%02d-> ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
foreach ($self->get_si_modules($_)) {
printf "%02d ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
}
printf "\n" if ($self->{runtime}->{options}->{verbose} >= 2);
}
foreach ($self->get_he_boards()) {
printf "HE%02d-> ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
foreach ($self->get_he_modules($_)) {
printf "%02d ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
}
printf "\n" if ($self->{runtime}->{options}->{verbose} >= 2);
}
foreach ($self->get_h2_boards()) {
printf "H2%02d-> ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
foreach ($self->get_h2_modules($_)) {
printf "%02d ", $_ if ($self->{runtime}->{options}->{verbose} >= 2);
}
printf "\n" if ($self->{runtime}->{options}->{verbose} >= 2);
}
if (($h2_dimms == 0) && ($he_dimms == 0) && ($si_dimms > 0)) {
printf "TYP1 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_si_with_si();
} elsif (($h2_dimms == 0) && ($he_dimms > 0) && ($si_dimms > 0)) {
printf "TYP2 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_si_with_he();
} elsif (($h2_dimms == 0) && ($he_dimms > 0) && ($si_dimms == 0)) {
printf "TYP3 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_he_with_he();
} elsif (($h2_dimms > 0) && ($he_dimms == 0) && ($si_dimms == 0)) {
printf "TYP4 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_h2_with_h2();
} elsif (($h2_dimms > 0) && ($he_dimms > 0) && ($si_dimms == 0)) {
printf "TYP5 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_he_with_h2();
} elsif (($h2_dimms > 0) && ($he_dimms == 0) && ($si_dimms > 0)) {
printf "TYP6 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_si_with_h2();
} elsif (($h2_dimms > 0) && ($he_dimms > 0) && ($si_dimms > 0)) {
if ($h2_dimms > $si_dimms) {
printf "TYP7 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_he_with_h2();
} else {
printf "TYP8 %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
@{$self->{dimms}} = $self->update_si_with_he();
}
} else {
printf "TYPX %s\n", $self->{runtime}->{product}
if ($self->{runtime}->{options}->{verbose} >= 2);
}
my $all_dimms = scalar(@{$self->{dimms}});
$self->trace(2, sprintf "ALL: %02d", $all_dimms);
}
sub dump {
my $self = shift;
if ($self->{runtime}->{options}->{verbose} > 2) {
printf "[SI]\n";
foreach (@{$self->{si_dimms}}) {
$_->dump();
}
printf "[HE]\n";
foreach (@{$self->{he_dimms}}) {
$_->dump();
}
printf "[H2]\n";
foreach (@{$self->{h2_dimms}}) {
$_->dump();
}
}
$self->SUPER::dump();
}
sub update_si_with_si {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my @dimms = ();
my $repaircondition = undef;
# wenn si keine statusinformationen liefert, dann besteht die chance
# dass ein undokumentiertes he-fragment vorliegt
# 1.3.6.1.4.1.232.6.2.14.11.1.1.0.<anzahl der dimms>
my $cpqHeResMemModuleEntry = "1.3.6.1.4.1.232.6.2.14.11.1";
if (SNMP::Utils::get_size($snmpwalk, $cpqHeResMemModuleEntry) == 1) {
$repaircondition = lc SNMP::Utils::get_object(
$snmpwalk, $cpqHeResMemModuleEntry.'.1.0.'.scalar(@{$self->{si_dimms}}));
# repaircondition 0 (ok) biegt alles wieder gerade
} else {
# anderer versuch
if ($self->{si_overall_condition} &&
$self->{si_overall_condition} eq 'ok') {
$repaircondition = 0;
}
}
foreach my $si_dimm (@{$self->{si_dimms}}) {
if (($si_dimm->{condition} eq 'n/a') ||
($si_dimm->{condition} eq 'other')) {
$si_dimm->{condition} = 'ok' if
(defined $repaircondition && $repaircondition == 0);
}
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $si_dimm->{runtime},
cartridge => $si_dimm->{cartridge},
module => $si_dimm->{module},
size => $si_dimm->{size},
status => $si_dimm->{status},
condition => $si_dimm->{condition},
));
}
return @dimms;
}
sub update_si_with_he {
my $self = shift;
my @dimms = ();
my $first_si_cartridge = ($self->get_si_boards())[0];
my $first_he_cartridge = ($self->get_he_boards())[0];
my $offset = 0;
if (scalar(@{$self->{si_dimms}}) == scalar(@{$self->{he_dimms}})) {
# aufpassen! sowas kann vorkommen: si cartridge 0...6, he cartridge 1...7
if ($first_si_cartridge != $first_he_cartridge) {
# README case 5
$offset = $first_si_cartridge - $first_he_cartridge;
}
} elsif ((scalar(@{$self->{si_dimms}}) > 1) &&
(scalar(@{$self->{he_dimms}}) == 1)) {
# siehe update_si_with_si. he-fragment
$self->update_si_with_si();
return;
} else {
# z.b. 4 si notpresent, 4 si present, 4 he
}
foreach my $si_dimm (@{$self->{si_dimms}}) {
if (($si_dimm->{condition} eq 'n/a') ||
($si_dimm->{condition} eq 'other')) {
if (my $he_dimm = $self->get_he_module(
$si_dimm->{cartridge} - $offset, $si_dimm->{module})) {
# vielleicht hat he mehr ahnung
$si_dimm->{condition} = $he_dimm->{condition};
if (($si_dimm->{condition} eq 'n/a') ||
($si_dimm->{condition} eq 'other')) {
# wenns immer noch kein brauchbares ergebnis gibt....
if ($self->{he_overall_condition} &&
$self->{he_overall_condition} eq 'ok') {
# wird schon stimmen...
$si_dimm->{condition} = 'ok';
} else {
# ansonsten stellen wir uns dumm
$si_dimm->{status} = 'notPresent';
}
}
} else {
# in dem fall zeigt si unbestueckte cartridges an
}
}
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $si_dimm->{runtime},
cartridge => $si_dimm->{cartridge},
module => $si_dimm->{module},
size => $si_dimm->{size},
status => $si_dimm->{status},
condition => $si_dimm->{condition},
));
}
return @dimms;
}
sub update_he_with_he {
my $self = shift;
my @dimms = ();
foreach my $he_dimm (@{$self->{he_dimms}}) {
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $he_dimm->{runtime},
cartridge => $he_dimm->{cartridge},
module => $he_dimm->{module},
size => $he_dimm->{size},
status => $he_dimm->{status},
condition => $he_dimm->{condition},
));
}
return @dimms;
}
sub update_si_with_h2 {
my $self = shift;
my @dimms = ();
my $first_si_cartridge = ($self->get_si_boards())[0];
my $first_h2_cartridge = ($self->get_h2_boards())[0];
my $offset = 0;
if (scalar(@{$self->{si_dimms}}) == scalar(@{$self->{h2_dimms}})) {
# aufpassen! sowas kann vorkommen: si cartridge 0...6, he cartridge 1...7
if ($first_si_cartridge != $first_h2_cartridge) {
# README case 5
$offset = $first_si_cartridge - $first_h2_cartridge;
}
} else {
# z.b. 4 si notpresent, 4 si present, 4 he
}
foreach my $si_dimm (@{$self->{si_dimms}}) {
if (($si_dimm->{condition} eq 'n/a') ||
($si_dimm->{condition} eq 'other')) {
if (my $h2_dimm = $self->get_h2_module(
$si_dimm->{cartridge} - $offset, $si_dimm->{module})) {
# vielleicht hat h2 mehr ahnung
$si_dimm->{condition} = $h2_dimm->{condition};
if (1) {
# ist zwar da, aber irgendwie auskonfiguriert
$si_dimm->{status} = 'notPresent' if $h2_dimm->{status} eq 'other';
}
} else {
# in dem fall zeigt si unbestueckte cartridges an
}
}
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $si_dimm->{runtime},
cartridge => $si_dimm->{cartridge},
module => $si_dimm->{module},
size => $si_dimm->{size},
status => $si_dimm->{status},
condition => $si_dimm->{condition},
));
}
return @dimms;
}
sub update_he_with_h2 {
my $self = shift;
my @dimms = ();
my $first_he_cartridge = ($self->get_he_boards())[0];
my $first_h2_cartridge = ($self->get_h2_boards())[0];
my $offset = 0;
# auch hier koennte sowas u.u.vorkommen: he cartridge 0..6, h2 cartridge 1..7
# ich habs zwar nie gesehen, aber wer weiss...
if ($first_h2_cartridge != $first_he_cartridge) {
$offset = $first_h2_cartridge - $first_he_cartridge;
}
foreach my $he_dimm (@{$self->{he_dimms}}) {
if (($he_dimm->{condition} eq 'n/a') ||
($he_dimm->{condition} eq 'other')) {
if (my $h2_dimm = $self->get_h2_module(
$he_dimm->{cartridge} + $offset, $he_dimm->{module})) {
# vielleicht hat h2 mehr ahnung
$he_dimm->{condition} = $h2_dimm->{condition};
if (1) {
# ist zwar da, aber irgendwie auskonfiguriert
$he_dimm->{status} = 'notPresent' if $h2_dimm->{status} eq 'other';
}
} else {
# in dem fall weiss he mehr als h2
}
}
if ($he_dimm->{size} == 0) {
if (my $h2_dimm = $self->get_h2_module(
$he_dimm->{cartridge} + $offset, $he_dimm->{module})) {
$he_dimm->{size} = $h2_dimm->{size};
# h2 beinhaltet eine size-oid
}
}
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $he_dimm->{runtime},
cartridge => $he_dimm->{cartridge},
module => $he_dimm->{module},
size => $he_dimm->{size},
status => $he_dimm->{status},
condition => $he_dimm->{condition},
));
}
return @dimms;
}
sub update_h2_with_h2 {
my $self = shift;
my @dimms = ();
foreach my $h2_dimm (@{$self->{h2_dimms}}) {
push(@dimms,
HP::Proliant::Component::MemorySubsystem::Dimm->new(
runtime => $h2_dimm->{runtime},
cartridge => $h2_dimm->{cartridge},
module => $h2_dimm->{module},
size => $h2_dimm->{size},
status => $h2_dimm->{status},
condition => $h2_dimm->{condition},
));
}
return @dimms;
}
sub is_faulty {
my $self = shift;
if (scalar(@{$self->{si_dimms}}) > 0 &&
scalar(@{$self->{he_dimms}}) > 0) {
return $self->si_is_faulty() || $self->he_is_faulty();
} elsif (scalar(@{$self->{si_dimms}}) > 0 &&
scalar(@{$self->{he_dimms}}) == 0) {
return $self->si_is_faulty();
} elsif (scalar(@{$self->{si_dimms}}) == 0 &&
scalar(@{$self->{he_dimms}}) > 0) {
return $self->he_is_faulty();
} else {
return 0;
}
}
sub si_is_faulty {
my $self = shift;
return ! defined $self->{si_overall_condition} ? 0 :
$self->{si_overall_condition} eq 'degraded' ? 1 : 0;
}
sub si_is_ok {
my $self = shift;
return ! defined $self->{si_overall_condition} ? 1 :
$self->{si_overall_condition} eq 'ok' ? 1 : 0;
}
sub he_is_faulty {
my $self = shift;
return ! defined $self->{he_overall_condition} ? 0 :
$self->{he_overall_condition} eq 'degraded' ? 1 : 0;
}
sub he_is_ok {
my $self = shift;
return ! defined $self->{he_overall_condition} ? 1 :
$self->{he_overall_condition} eq 'ok' ? 1 : 0;
}
sub get_si_boards {
my $self = shift;
my %found = ();
foreach (@{$self->{si_dimms}}) {
$found{$_->{cartridge}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_si_modules {
my $self = shift;
my $board = shift;
my %found = ();
foreach (grep { $_->{cartridge} == $board } @{$self->{si_dimms}}) {
$found{$_->{module}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_he_boards {
my $self = shift;
my %found = ();
foreach (@{$self->{he_dimms}}) {
$found{$_->{cartridge}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_he_modules {
my $self = shift;
my $board = shift;
my %found = ();
foreach (grep { $_->{cartridge} == $board } @{$self->{he_dimms}}) {
$found{$_->{module}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_he_module {
my $self = shift;
my $board = shift;
my $module = shift;
my $found = (grep { $_->{cartridge} == $board && $_->{module} == $module }
@{$self->{he_dimms}})[0];
return $found;
}
sub get_h2_boards {
my $self = shift;
my %found = ();
#
foreach (@{$self->{h2_dimms}}) {
$found{$_->{cartridge}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_h2_modules {
my $self = shift;
my $board = shift;
my %found = ();
foreach (grep { $_->{cartridge} == $board } @{$self->{h2_dimms}}) {
$found{$_->{module}} = 1;
}
return sort { $a <=> $b } keys %found;
}
sub get_h2_module {
my $self = shift;
my $board = shift;
my $module = shift;
my $found = (grep { $_->{cartridge} == $board && $_->{module} == $module }
@{$self->{h2_dimms}})[0];
return $found;
}
package HP::Proliant::Component::MemorySubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
condition => $params{condition},
status => $params{status},
blacklisted => 0,
info => undef,
extendedinfo => undef,
dimms => [],
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
return HP::Proliant::Component::MemorySubsystem::SNMP->new(%params);
} elsif ($self->{method} eq 'cli') {
return HP::Proliant::Component::MemorySubsystem::CLI->new(%params);
} else {
die "unknown method";
}
}
sub check {
my $self = shift;
my $errorfound = 0;
$self->add_info('checking memory');
foreach (@{$self->{dimms}}) {
$_->check(); # info ausfuellen
}
if ((scalar(grep {
$_->is_present() &&
($_->{condition} ne 'n/a' && $_->{condition} ne 'other' )
} @{$self->{dimms}})) != 0) {
foreach (@{$self->{dimms}}) {
if (($_->is_present()) && ($_->{condition} ne 'ok')) {
$_->add_message(CRITICAL, $_->{info});
$errorfound++;
}
}
} else {
if ($self->{runtime}->{options}->{ignore_dimms}) {
$self->add_message(OK,
sprintf "ignoring %d dimms with status 'n/a' ",
scalar(grep { ($_->is_present()) } @{$self->{dimms}}));
} elsif ($self->{runtime}->{options}->{buggy_firmware}) {
$self->add_message(OK,
sprintf "ignoring %d dimms with status 'n/a' because of buggy firmware",
scalar(grep { ($_->is_present()) } @{$self->{dimms}}));
} else {
$self->add_message(WARNING,
sprintf "status of all %d dimms is n/a (please upgrade firmware)",
scalar(grep { $_->is_present() } @{$self->{dimms}}));
$errorfound++;
}
}
foreach (@{$self->{dimms}}) {
printf "%s\n", $_->{info} if $self->{runtime}->{options}->{verbose} >= 2;
}
if (! $errorfound && $self->is_faulty()) {
#if ($self->is_faulty()) {
$self->add_message(WARNING,
sprintf 'overall memory error found');
}
}
sub dump {
my $self = shift;
printf "i dump the memory\n";
foreach (@{$self->{dimms}}) {
$_->dump();
}
}
package HP::Proliant::Component::MemorySubsystem::Dimm;
our @ISA = qw(HP::Proliant::Component::MemorySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cartridge => $params{cartridge},
module => $params{module},
size => $params{size} || 0,
status => $params{status},
condition => $params{condition},
type => $params{type},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->{name} = sprintf '%s:%s',
$self->{cartridge}, $self->{module};
$self->{location} = sprintf 'module %s @ cartridge %s',
$self->{module}, $self->{cartridge};
return $self;
}
sub check {
my $self = shift;
# check dient nur dazu, info und extended_info zu f<>llen
# die eigentliche bewertung findet eins h<>her statt
$self->blacklist('d', $self->{name});
if (($self->{status} eq 'present') || ($self->{status} eq 'good')) {
if ($self->{condition} eq 'other') {
$self->add_info(sprintf 'dimm %s (%s) is n/a',
$self->{name}, $self->{location});
} elsif ($self->{condition} ne 'ok') {
$self->add_info(
sprintf "dimm module %s (%s) needs attention (%s)",
$self->{name}, $self->{location}, $self->{condition});
} else {
$self->add_info(sprintf 'dimm module %s (%s) is %s',
$self->{name}, $self->{location}, $self->{condition});
}
} elsif ($self->{status} eq 'notPresent') {
$self->add_info(sprintf 'dimm module %s (%s) is not present',
$self->{name}, $self->{location});
} else {
$self->add_info(
sprintf "dimm module %s (%s) needs attention (%s)",
$self->{name}, $self->{location}, $self->{condition});
}
}
sub is_present {
my $self = shift;
my @signs_of_presence = (qw(present good add upgraded doesnotmatch
notsupported badconfig degraded));
return scalar(grep { $self->{status} eq $_ } @signs_of_presence);
}
sub dump {
my $self = shift;
#printf "[DIMM_%s_%s]\n", $self->{cartridge}, $self->{module};
#foreach (qw(cartridge module size status condition info)) {
# printf "%s: %s\n", $_, $self->{$_};
#}
#printf "status: %s\n", $self->{status} if exists $self->{status};
#printf "\n";
printf "car %02d mod %02d siz %.0f sta %-12s con %-10s typ %s\n",
$self->{cartridge}, $self->{module}, $self->{size},
$self->{status}, $self->{condition}, defined $self->{type} ? $self->{type} : "";
}
package HP::Proliant::Component::MemorySubsystem::Cartridge;
our @ISA = qw(HP::Proliant::Component::MemorySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
cpqHeResMemBoardSlotIndex => $params{cpqHeResMemBoardSlotIndex},
cpqHeResMemBoardOnlineStatus => $params{cpqHeResMemBoardOnlineStatus},
cpqHeResMemBoardErrorStatus => $params{cpqHeResMemBoardErrorStatus},
cpqHeResMemBoardNumSockets => $params{cpqHeResMemBoardNumSockets},
cpqHeResMemBoardOsMemSize => $params{cpqHeResMemBoardOsMemSize},
cpqHeResMemBoardTotalMemSize => $params{cpqHeResMemBoardTotalMemSize},
cpqHeResMemBoardCondition => $params{cpqHeResMemBoardCondition},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
return $self;
}
sub dump {
my $self = shift;
#printf "[CARTRIDGE_%s_%s]\n", $self->{cpqHeResMemBoardSlotIndex};
#foreach (qw(cpqHeResMemBoardSlotIndex cpqHeResMemBoardOnlineStatus
# cpqHeResMemBoardErrorStatus cpqHeResMemBoardNumSockets
# cpqHeResMemBoardOsMemSize cpqHeResMemBoardTotalMemSize
# cpqHeResMemBoardCondition)) {
# printf "%s: %s\n", $_, $self->{$_};
#}
#printf "\n";
}
package HP::Proliant::Component::SNMP;
sub get_entries {
my $self = shift;
my $oids = shift;
my $entry = shift;
my $snmpwalk = $self->{rawdata};
my @params = ();
my @indices = SNMP::Utils::get_indices($snmpwalk, $oids->{$entry});
foreach (@indices) {
my @idx = @{$_};
my %params = (
runtime => $self->{runtime},
);
my $maxdimension = scalar(@idx) - 1;
foreach my $idxnr (1..scalar(@idx)) {
$params{'index'.$idxnr} = $_->[$idxnr - 1];
}
foreach my $oid (keys %{$oids}) {
next if $oid =~ /Entry$/;
next if $oid =~ /Value$/;
if (exists $oids->{$oid.'Value'}) {
$params{$oid} = SNMP::Utils::get_object_value(
$snmpwalk, $oids->{$oid}, $oids->{$oid.'Value'}, @idx);
if (! defined $params{$oid}) {
my $numerical_value = SNMP::Utils::get_object(
$snmpwalk, $oids->{$oid}, @idx);
if (! defined $numerical_value) {
# maschine liefert schrott
$params{$oid} = 'value_unknown';
} else {
$params{$oid} = 'value_'.SNMP::Utils::get_object(
$snmpwalk, $oids->{$oid}, @idx);
}
}
} else {
$params{$oid} = SNMP::Utils::get_object(
$snmpwalk, $oids->{$oid}, @idx);
}
}
push(@params, \%params);
}
return @params;
}
sub mib {
my $self = shift;
my $mib = shift;
my $condition = {
0 => 'other',
1 => 'ok',
2 => 'degraded',
3 => 'failed',
};
my $MibRevMajor = $mib.'.1.0';
my $MibRevMinor = $mib.'.2.0';
my $MibRevCondition = $mib.'.3.0';
return (
$self->SNMP::Utils::get_object($self->{rawdata},
$MibRevMajor),
$self->SNMP::Utils::get_object($self->{rawdata},
$MibRevMinor),
$self->SNMP::Utils::get_object_value($self->{rawdata},
$MibRevCondition, $condition));
};
package HP::Proliant::Component::DiskSubsystem::Da::CLI;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $hpacucli = $self->{rawdata};
my $slot = 0;
my $type = "unkn";
my @lines = ();
my $thistype = 0;
my $tmpcntl = {};
my $tmpaccel = {};
my $tmpld = {};
my $tmppd = {};
my $cntlindex = 0;
my $ldriveindex = 0;
my $pdriveindex = 0;
my $incontroller = 0;
foreach (split(/\n/, $hpacucli)) {
next unless /^status/;
next if /^status\s*$/;
s/^status\s*//;
if (/([\s\w]+) in Slot\s+(\d+)/) {
$incontroller = 1;
$slot = $2;
$cntlindex++;
$tmpcntl->{$slot}->{cpqDaCntlrIndex} = $cntlindex;
$tmpcntl->{$slot}->{cpqDaCntlrModel} = $1;
$tmpcntl->{$slot}->{cpqDaCntlrSlot} = $slot;
} elsif (/Controller Status: (\w+)/) {
$tmpcntl->{$slot}->{cpqDaCntlrBoardCondition} = lc $1;
$tmpcntl->{$slot}->{cpqDaCntlrCondition} = lc $1;
} elsif (/Cache Status: (\w+)/) {
$tmpaccel->{$slot}->{cpqDaAccelCntlrIndex} = $cntlindex;
$tmpaccel->{$slot}->{cpqDaAccelSlot} = $slot;
$tmpaccel->{$slot}->{cpqDaAccelCondition} = lc $1;
$tmpaccel->{$slot}->{cpqDaAccelStatus} = 'enabled';
} elsif (/Battery.* Status: (\w+)/) {
# sowas gibts auch Battery/Capacitor Status: OK
$tmpaccel->{$slot}->{cpqDaAccelBattery} = lc $1;
} elsif (/^\s*$/) {
}
}
$cntlindex = 0;
$ldriveindex = 0;
$pdriveindex = 0;
foreach (split(/\n/, $hpacucli)) {
next unless /^config/;
next if /^config\s*$/;
s/^config\s*//;
if (/([\s\w]+) in Slot\s+(\d+)/) {
$slot = $2;
$cntlindex++;
$pdriveindex = 1;
} elsif (/logicaldrive\s+(.+?)\s+\((.*)\)/) {
# logicaldrive 1 (683.5 GB, RAID 5, OK)
# logicaldrive 1 (683.5 GB, RAID 5, OK)
# logicaldrive 2 (442 MB, RAID 1+0, OK)
$ldriveindex = $1;
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvCntlrIndex} = $cntlindex;
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvIndex} = $ldriveindex;
($tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvSize},
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvFaultTol},
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvCondition}) =
map { lc $_ } split(/,\s*/, $2);
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvStatus} =
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvCondition};
$tmpld->{$slot}->{$ldriveindex}->{cpqDaLogDrvPhyDrvIDs} = 'unknown';
} elsif (/physicaldrive\s+(.+?)\s+\((.*)\)/) {
# physicaldrive 2:0 (port 2:id 0 , Parallel SCSI, 36.4 GB, OK)
# physicaldrive 2I:1:6 (port 2I:box 1:bay 6, SAS, 146 GB, OK)
my $name = $1;
my($location, $type, $size, $status) = split(/,/, $2);
$status =~ s/^\s+//g;
$status =~ s/\s+$//g;
$status = lc $status;
my %location = ();
foreach (split(/:/, $location)) {
$location{$1} = $2 if /(\w+)\s+(\w+)/;
}
$location{box} ||= 0;
$location{id} ||= $pdriveindex;
$location{bay} ||= $location{id};
$tmppd->{$slot}->{$name}->{name} = lc $name;
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvCntlrIndex} = $cntlindex;
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvIndex} = $location{id};
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvBay} = $location{bay};
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvBusNumber} = $location{port};
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvSize} = $size;
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvStatus} = $status;
$tmppd->{$slot}->{$name}->{cpqDaPhyDrvCondition} = $status;
$tmppd->{$slot}->{$name}->{ldriveindex} = $ldriveindex || -1;
foreach (keys %{$tmppd->{$slot}->{$name}}) {
$tmppd->{$slot}->{$name}->{$_} =~ s/^\s+//g;
$tmppd->{$slot}->{$name}->{$_} =~ s/\s+$//g;
$tmppd->{$slot}->{$name}->{$_} = lc $tmppd->{$slot}->{$name}->{$_};
}
}
}
foreach my $slot (keys %{$tmpcntl}) {
if (exists $tmpcntl->{$slot}->{cpqDaCntlrModel} &&
! $self->identified($tmpcntl->{$slot}->{cpqDaCntlrModel})) {
delete $tmpcntl->{$slot};
delete $tmpaccel->{$slot};
delete $tmpld->{$slot};
delete $tmppd->{$slot};
}
}
#printf "%s\n", Data::Dumper::Dumper($tmpcntl);
#printf "%s\n", Data::Dumper::Dumper($tmpaccel);
#printf "%s\n", Data::Dumper::Dumper($tmpld);
#printf "%s\n", Data::Dumper::Dumper($tmppd);
foreach my $slot (sort {
$tmpcntl->{$a}->{cpqDaCntlrIndex} <=> $tmpcntl->{$b}->{cpqDaCntlrIndex}
}keys %{$tmpcntl}) {
$tmpcntl->{$slot}->{runtime} = $self->{runtime};
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Da::Controller->new(
%{$tmpcntl->{$slot}}));
}
foreach my $slot (sort {
$tmpaccel->{$a}->{cpqDaAccelCntlrIndex} <=> $tmpaccel->{$b}->{cpqDaAccelCntlrIndex}
} keys %{$tmpaccel}) {
$tmpaccel->{$slot}->{runtime} = $self->{runtime};
push(@{$self->{accelerators}},
HP::Proliant::Component::DiskSubsystem::Da::Accelerator->new(
%{$tmpaccel->{$slot}}));
}
foreach my $slot (keys %{$tmpld}) {
foreach my $ldriveindex (keys %{$tmpld->{$slot}}) {
$tmpld->{$slot}->{$ldriveindex}->{runtime} = $self->{runtime};
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Da::LogicalDrive->new(
%{$tmpld->{$slot}->{$ldriveindex}}));
}
foreach my $pdriveindex (sort {
(split ':', $a, 2)[0] cmp (split ':', $b, 2)[0] ||
(split ':', $a, 2)[1] cmp (split ':', $b, 2)[1] ||
(split ':', $a, 2)[2] <=> (split ':', $b, 2)[2]
} keys %{$tmppd->{$slot}}) {
$tmppd->{$slot}->{$pdriveindex}->{runtime} = $self->{runtime};
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Da::PhysicalDrive->new(
%{$tmppd->{$slot}->{$pdriveindex}}));
}
}
}
sub identified {
my $self = shift;
my $info = shift;
return 1 if $info =~ /Parallel SCSI/;
return 1 if $info =~ /Smart Array/; # Trond: works fine on E200i, P400, E400
#return 1 if $info =~ /Smart Array (5|6)/;
#return 1 if $info =~ /Smart Array P400i/; # snmp sagt Da, trotz SAS in cli
#return 1 if $info =~ /Smart Array P410i/; # dto
return 0;
}
package HP::Proliant::Component::DiskSubsystem::Da::SNMP;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQIDA-MIB
my $oids = {
cpqDaCntlrEntry => "1.3.6.1.4.1.232.3.2.2.1.1",
cpqDaCntlrIndex => "1.3.6.1.4.1.232.3.2.2.1.1.1",
cpqDaCntlrModel => "1.3.6.1.4.1.232.3.2.2.1.1.2",
cpqDaCntlrSlot => "1.3.6.1.4.1.232.3.2.2.1.1.5",
cpqDaCntlrCondition => "1.3.6.1.4.1.232.3.2.2.1.1.6",
cpqDaCntlrBoardCondition => "1.3.6.1.4.1.232.3.2.2.1.1.12",
cpqDaCntlrModelValue => {
1 => 'other',
2 => 'ida',
3 => 'idaExpansion',
4 => 'ida-2',
5 => 'smart',
6 => 'smart-2e',
7 => 'smart-2p',
8 => 'smart-2sl',
9 => 'smart-3100es',
10 => 'smart-3200',
11 => 'smart-2dh',
12 => 'smart-221',
13 => 'sa-4250es',
14 => 'sa-4200',
15 => 'sa-integrated',
16 => 'sa-431',
17 => 'sa-5300',
18 => 'raidLc2',
19 => 'sa-5i',
20 => 'sa-532',
21 => 'sa-5312',
22 => 'sa-641',
23 => 'sa-642',
24 => 'sa-6400',
25 => 'sa-6400em',
26 => 'sa-6i',
},
cpqDaCntlrConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqDaCntlrBoardConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqDaCntlrIndex }
foreach ($self->get_entries($oids, 'cpqDaCntlrEntry')) {
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Da::Controller->new(%{$_}));
}
$oids = {
cpqDaAccelEntry => "1.3.6.1.4.1.232.3.2.2.2.1",
cpqDaAccelCntlrIndex => "1.3.6.1.4.1.232.3.2.2.2.1.1",
cpqDaAccelStatus => "1.3.6.1.4.1.232.3.2.2.2.1.2",
cpqDaAccelSlot => "1.3.6.1.4.1.232.3.2.2.2.1.5",
cpqDaAccelBattery => "1.3.6.1.4.1.232.3.2.2.2.1.6",
cpqDaAccelCondition => "1.3.6.1.4.1.232.3.2.2.2.1.9",
cpqDaAccelBatteryValue => {
1 => 'other',
2 => 'ok',
3 => 'recharging',
4 => 'failed',
5 => 'degraded',
6 => 'notPresent',
},
cpqDaAccelConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqDaAccelStatusValue => {
1 => "other",
2 => "invalid",
3 => "enabled",
4 => "tmpDisabled",
5 => "permDisabled",
}
};
# INDEX { cpqDaAccelCntlrIndex }
foreach ($self->get_entries($oids, 'cpqDaAccelEntry')) {
push(@{$self->{accelerators}},
HP::Proliant::Component::DiskSubsystem::Da::Accelerator->new(%{$_}));
}
$oids = {
cpqDaLogDrvEntry => "1.3.6.1.4.1.232.3.2.3.1.1",
cpqDaLogDrvCntlrIndex => "1.3.6.1.4.1.232.3.2.3.1.1.1",
cpqDaLogDrvIndex => "1.3.6.1.4.1.232.3.2.3.1.1.2",
cpqDaLogDrvFaultTol => "1.3.6.1.4.1.232.3.2.3.1.1.3",
cpqDaLogDrvStatus => "1.3.6.1.4.1.232.3.2.3.1.1.4",
cpqDaLogDrvSize => "1.3.6.1.4.1.232.3.2.3.1.1.9",
cpqDaLogDrvPhyDrvIDs => "1.3.6.1.4.1.232.3.2.3.1.1.10",
cpqDaLogDrvCondition => "1.3.6.1.4.1.232.3.2.3.1.1.11",
cpqDaLogDrvPercentRebuild => "1.3.6.1.4.1.232.3.2.3.1.1.12",
cpqDaLogDrvFaultTolValue => {
1 => "other",
2 => "none",
3 => "mirroring",
4 => "dataGuard",
5 => "distribDataGuard",
7 => "advancedDataGuard",
},
cpqDaLogDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqDaLogDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "unconfigured",
5 => "recovering",
6 => "readyForRebuild",
7 => "rebuilding",
8 => "wrongDrive",
9 => "badConnect",
10 => "overheating",
11 => "shutdown",
12 => "expanding",
13 => "notAvailable",
14 => "queuedForExpansion",
},
};
# INDEX { cpqDaLogDrvCntlrIndex, cpqDaLogDrvIndex }
foreach ($self->get_entries($oids, 'cpqDaLogDrvEntry')) {
$_->{cpqDaLogDrvPhyDrvIDs} ||= 'empty';
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Da::LogicalDrive->new(%{$_}));
}
$oids = {
cpqDaPhyDrvEntry => "1.3.6.1.4.1.232.3.2.5.1.1",
cpqDaPhyDrvCntlrIndex => "1.3.6.1.4.1.232.3.2.5.1.1.1",
cpqDaPhyDrvIndex => "1.3.6.1.4.1.232.3.2.5.1.1.2",
cpqDaPhyDrvBay => "1.3.6.1.4.1.232.3.2.5.1.1.5",
cpqDaPhyDrvStatus => "1.3.6.1.4.1.232.3.2.5.1.1.6",
cpqDaPhyDrvSize => "1.3.6.1.4.1.232.3.2.5.1.1.9",
cpqDaPhyDrvCondition => "1.3.6.1.4.1.232.3.2.5.1.1.37",
cpqDaPhyDrvBusNumber => "1.3.6.1.4.1.232.3.2.5.1.1.50",
cpqDaPhyDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqDaPhyDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "predictiveFailure",
},
};
# INDEX { cpqDaPhyDrvCntlrIndex, cpqDaPhyDrvIndex }
foreach ($self->get_entries($oids, 'cpqDaPhyDrvEntry')) {
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Da::PhysicalDrive->new(%{$_}));
}
}
package HP::Proliant::Component::DiskSubsystem::Da;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
condition => undef,
blacklisted => 0,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Da::SNMP';
} else {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Da::CLI';
}
$self->init();
$self->assemble();
return $self;
}
sub check {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Da::Controller;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqDaCntlrIndex => $params{cpqDaCntlrIndex},
cpqDaCntlrSlot => $params{cpqDaCntlrSlot},
cpqDaCntlrModel => $params{cpqDaCntlrModel},
cpqDaCntlrCondition => $params{cpqDaCntlrCondition},
cpqDaCntlrBoardCondition => $params{cpqDaCntlrBoardCondition},
blacklisted => 0,
};
$self->{name} = $params{name} || $self->{cpqDaCntlrSlot};
$self->{controllerindex} = $self->{cpqDaCntlrIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
#$self->dumper($self);
$self->blacklist('daco', $self->{cpqDaCntlrIndex});
if ($self->{cpqDaCntlrCondition} eq 'other') {
if (scalar(@{$self->{physical_drives}})) {
$self->add_message(CRITICAL,
sprintf 'da controller %s in slot %s needs attention',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
$self->add_info(sprintf 'da controller %s in slot %s needs attention',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
} else {
$self->add_info(sprintf 'da controller %s in slot %s is ok and unused',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
$self->{blacklisted} = 1;
}
} elsif ($self->{cpqDaCntlrCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'da controller %s in slot %s needs attention',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
$self->add_info(sprintf 'da controller %s in slot %s needs attention',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
} else {
$self->add_info(sprintf 'da controller %s in slot %s is ok',
$self->{cpqDaCntlrIndex}, $self->{cpqDaCntlrSlot});
}
foreach (@{$self->{accelerators}}) {
$_->check();
}
foreach (@{$self->{logical_drives}}) {
$_->check();
}
foreach (@{$self->{physical_drives}}) {
$_->check();
}
foreach (@{$self->{spare_drives}}) {
$_->check();
}
}
sub dump {
my $self = shift;
printf "[DA_CONTROLLER_%s]\n", $self->{name};
foreach (qw(cpqDaCntlrSlot cpqDaCntlrIndex cpqDaCntlrCondition
cpqDaCntlrModel)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
foreach (@{$self->{accelerators}}) {
$_->dump();
}
foreach (@{$self->{logical_drives}}) {
$_->dump();
}
foreach (@{$self->{physical_drives}}) {
$_->dump();
}
foreach (@{$self->{spare_drives}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Da::Accelerator;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqDaAccelCntlrIndex => $params{cpqDaAccelCntlrIndex},
cpqDaAccelBattery => $params{cpqDaAccelBattery} || 'notPresent',
cpqDaAccelCondition => $params{cpqDaAccelCondition},
cpqDaAccelStatus => $params{cpqDaAccelStatus},
blacklisted => 0,
};
$self->{controllerindex} = $self->{cpqDaAccelCntlrIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('daac', $self->{cpqDaAccelCntlrIndex});
$self->add_info(sprintf 'controller accelerator is %s',
$self->{cpqDaAccelCondition});
if ($self->{cpqDaAccelStatus} ne "enabled") {
} elsif ($self->{cpqDaAccelCondition} ne "ok") {
$self->add_message(CRITICAL, "controller accelerator needs attention");
}
$self->blacklist('daacb', $self->{cpqDaAccelCntlrIndex});
$self->add_info(sprintf 'controller accelerator battery is %s',
$self->{cpqDaAccelBattery});
if ($self->{cpqDaAccelBattery} eq "notPresent") {
} elsif ($self->{cpqDaAccelBattery} eq "recharging") {
$self->add_message(WARNING, "controller accelerator battery recharging");
} elsif ($self->{cpqDaAccelBattery} ne "ok") {
# (other) failed degraded
$self->add_message(CRITICAL, "controller accelerator battery needs attention");
}
}
sub dump {
my $self = shift;
printf "[ACCELERATOR]\n";
foreach (qw(cpqDaAccelCntlrIndex cpqDaAccelBattery
cpqDaAccelStatus cpqDaAccelCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Da::LogicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqDaLogDrvIndex => $params{cpqDaLogDrvIndex},
cpqDaLogDrvCntlrIndex => $params{cpqDaLogDrvCntlrIndex},
cpqDaLogDrvSize => $params{cpqDaLogDrvSize},
cpqDaLogDrvFaultTol => $params{cpqDaLogDrvFaultTol},
cpqDaLogDrvPercentRebuild => $params{cpqDaLogDrvPercentRebuild},
cpqDaLogDrvStatus => $params{cpqDaLogDrvStatus},
cpqDaLogDrvCondition => $params{cpqDaLogDrvCondition},
cpqDaLogDrvPhyDrvIDs => $params{cpqDaLogDrvPhyDrvIDs},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqDaLogDrvCntlrIndex}.':'.$self->{cpqDaLogDrvIndex}; ##vorerst
$self->{controllerindex} = $self->{cpqDaLogDrvCntlrIndex};
if (! $self->{cpqDaLogDrvPercentRebuild} ||
$self->{cpqDaLogDrvPercentRebuild} == 4294967295) {
$self->{cpqDaLogDrvPercentRebuild} = 100;
}
return $self;
}
sub check {
my $self = shift;
$self->blacklist('dald', $self->{name});
$self->add_info(sprintf "logical drive %s is %s (%s)",
$self->{name}, $self->{cpqDaLogDrvStatus},
$self->{cpqDaLogDrvFaultTol});
if ($self->{cpqDaLogDrvCondition} ne "ok") {
if ($self->{cpqDaLogDrvStatus} =~
/rebuild|recovering|expanding|queued/) {
$self->add_message(WARNING,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqDaLogDrvStatus});
} else {
$self->add_message(CRITICAL,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqDaLogDrvStatus});
}
}
}
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cpqDaLogDrvCntlrIndex cpqDaLogDrvIndex cpqDaLogDrvSize
cpqDaLogDrvFaultTol cpqDaLogDrvStatus cpqDaLogDrvCondition
cpqDaLogDrvPercentRebuild cpqDaLogDrvPhyDrvIDs)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Da::PhysicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
name => $params{name},
cpqDaPhyDrvCntlrIndex => $params{cpqDaPhyDrvCntlrIndex},
cpqDaPhyDrvIndex => $params{cpqDaPhyDrvIndex},
cpqDaPhyDrvBay => $params{cpqDaPhyDrvBay},
cpqDaPhyDrvBusNumber => $params{cpqDaPhyDrvBusNumber},
cpqDaPhyDrvSize => $params{cpqDaPhyDrvSize},
cpqDaPhyDrvStatus => $params{cpqDaPhyDrvStatus},
cpqDaPhyDrvCondition => $params{cpqDaPhyDrvCondition},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqDaPhyDrvCntlrIndex}.':'.$self->{cpqDaPhyDrvIndex}; ##vorerst
$self->{controllerindex} = $self->{cpqDaPhyDrvCntlrIndex};
return $self;
}
sub check {
my $self = shift;
$self->blacklist('dapd', $self->{name});
$self->add_info(
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqDaPhyDrvCondition});
if ($self->{cpqDaPhyDrvCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqDaPhyDrvCondition});
}
}
sub dump {
my $self = shift;
printf "[PHYSICAL_DRIVE]\n";
foreach (qw(cpqDaPhyDrvCntlrIndex cpqDaPhyDrvIndex cpqDaPhyDrvBay
cpqDaPhyDrvBusNumber cpqDaPhyDrvSize cpqDaPhyDrvStatus
cpqDaPhyDrvCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Da::SpareDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Da);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cntrlcpqDaCntlrIndex cpqDaCntlrIndex size level status condition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Sas::CLI;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
}
package HP::Proliant::Component::DiskSubsystem::Sas::SNMP;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQSCSI-MIB
my $oids = {
cpqSasHbaEntry => "1.3.6.1.4.1.232.5.5.1.1.1",
cpqSasHbaIndex => "1.3.6.1.4.1.232.5.5.1.1.1.1",
cpqSasHbaLocation => "1.3.6.1.4.1.232.5.5.1.1.1.2",
cpqSasHbaSlot => "1.3.6.1.4.1.232.5.5.1.1.1.6",
cpqSasHbaStatus => "1.3.6.1.4.1.232.5.5.1.1.1.4",
cpqSasHbaStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
},
cpqSasHbaCondition => "1.3.6.1.4.1.232.5.5.1.1.1.5",
cpqSasHbaConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqSasHbaIndex }
foreach ($self->get_entries($oids, 'cpqSasHbaEntry')) {
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Sas::Controller->new(%{$_}));
}
$oids = {
cpqSasLogDrvEntry => "1.3.6.1.4.1.232.5.5.3.1.1",
cpqSasLogDrvHbaIndex => "1.3.6.1.4.1.232.5.5.3.1.1.1",
cpqSasLogDrvIndex => "1.3.6.1.4.1.232.5.5.3.1.1.2",
cpqSasLogDrvStatus => "1.3.6.1.4.1.232.5.5.3.1.1.4",
cpqSasLogDrvCondition => "1.3.6.1.4.1.232.5.5.3.1.1.5",
cpqSasLogDrvRebuildingPercent => "1.3.6.1.4.1.232.5.5.3.1.1.12",
cpqSasLogDrvRaidLevel => "1.3.6.1.4.1.232.5.5.3.1.1.3",
cpqSasLogDrvRaidLevelValue => {
1 => "other",
2 => "raid0",
3 => "raid1",
4 => "raid0plus1",
5 => "raid5",
6 => "raid15",
7 => "volume",
},
cpqSasLogDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqSasLogDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "rebuilding",
5 => "failed",
6 => "offline",
}
};
# INDEX { cpqSasLogDrvCntlrIndex, cpqSasLogDrvIndex }
foreach ($self->get_entries($oids, 'cpqSasLogDrvEntry')) {
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Sas::LogicalDrive->new(%{$_}));
}
$oids = {
cpqSasPhyDrvEntry => "1.3.6.1.4.1.232.5.5.2.1.1",
cpqSasPhyDrvHbaIndex => "1.3.6.1.4.1.232.5.5.2.1.1.1",
cpqSasPhyDrvIndex => "1.3.6.1.4.1.232.5.5.2.1.1.2",
cpqSasPhyDrvLocationString => "1.3.6.1.4.1.232.5.5.2.1.1.3",
cpqSasPhyDrvStatus => "1.3.6.1.4.1.232.5.5.2.1.1.5",
cpqSasPhyDrvSize => "1.3.6.1.4.1.232.5.5.2.1.1.8",
cpqSasPhyDrvCondition => "1.3.6.1.4.1.232.5.5.2.1.1.6",
cpqSasPhyDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqSasPhyDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "predictiveFailure",
4 => "offline",
5 => "failed",
6 => "missingWasOk",
7 => "missingWasPredictiveFailure",
8 => "missingWasOffline",
9 => "missingWasFailed",
},
};
# INDEX { cpqPhyLogDrvCntlrIndex, cpqSasPhyDrvIndex }
foreach ($self->get_entries($oids, 'cpqSasPhyDrvEntry')) {
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Sas::PhysicalDrive->new(%{$_}));
}
}
package HP::Proliant::Component::DiskSubsystem::Sas;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
controllers => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
condition => undef,
blacklisted => 0,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Sas::SNMP';
} else {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Sas::CLI';
}
$self->init();
$self->assemble();
return $self;
}
sub check {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Sas::Controller;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqSasHbaIndex => $params{cpqSasHbaIndex},
cpqSasHbaLocation => $params{cpqSasHbaLocation},
cpqSasHbaSlot => $params{cpqSasHbaSlot},
cpqSasHbaStatus => $params{cpqSasHbaStatus},
cpqSasHbaCondition => $params{cpqSasHbaCondition},
blacklisted => 0,
};
$self->{name} = $params{name} || $self->{cpqSasHbaSlot};
$self->{controllerindex} = $self->{cpqSasHbaIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqSasHbaCondition} eq 'other') {
if (scalar(@{$self->{physical_drives}})) {
$self->add_message(CRITICAL,
sprintf 'sas controller in slot %s needs attention',
$self->{cpqSasHbaSlot});
$self->add_info(sprintf 'sas controller in slot %s needs attention',
$self->{cpqSasHbaSlot});
} else {
$self->add_info(sprintf 'sas controller in slot %s is ok and unused',
$self->{cpqSasHbaSlot});
$self->{blacklisted} = 1;
}
} elsif ($self->{cpqSasHbaCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'sas controller in slot %s needs attention',
$self->{cpqSasHbaSlot});
$self->add_info(sprintf 'sas controller in slot %s needs attention',
$self->{cpqSasHbaSlot});
} else {
$self->add_info(sprintf 'sas controller in slot %s is ok',
$self->{cpqSasHbaSlot});
}
foreach (@{$self->{logical_drives}}) {
$_->check();
}
foreach (@{$self->{physical_drives}}) {
$_->check();
}
foreach (@{$self->{spare_drives}}) {
$_->check();
}
}
sub dump {
my $self = shift;
printf "[SAS_HBA%s]\n", $self->{name};
foreach (qw(cpqSasHbaSlot cpqSasHbaIndex cpqSasHbaCondition
cpqSasHbaStatus cpqSasHbaLocation)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
foreach (@{$self->{logical_drives}}) {
$_->dump();
}
foreach (@{$self->{physical_drives}}) {
$_->dump();
}
foreach (@{$self->{spare_drives}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Sas::LogicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqSasLogDrvHbaIndex => $params{cpqSasLogDrvHbaIndex},
cpqSasLogDrvIndex => $params{cpqSasLogDrvIndex},
cpqSasLogDrvStatus => $params{cpqSasLogDrvStatus},
cpqSasLogDrvCondition => $params{cpqSasLogDrvCondition},
cpqSasLogDrvRebuildingPercent => $params{cpqSasLogDrvRebuildingPercent},
cpqSasLogDrvRaidLevel => $params{cpqSasLogDrvRaidLevel},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqSasLogDrvHbaIndex}.':'.$self->{cpqSasLogDrvIndex}; ####vorerst
$self->{controllerindex} = $self->{cpqSasLogDrvHbaIndex};
if (! $self->{cpqSasLogDrvRebuildingPercent} ||
$self->{cpqSasLogDrvRebuildingPercent} == 4294967295) {
$self->{cpqSasLogDrvRebuildingPercent} = 100;
}
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqSasLogDrvCondition} ne "ok") {
if ($self->{cpqSasLogDrvStatus} =~
/rebuild|recovering|expanding|queued/) {
$self->add_message(WARNING,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqSasLogDrvStatus});
} else {
$self->add_message(CRITICAL,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqSasLogDrvStatus});
}
}
$self->add_info(
sprintf "logical drive %s is %s (%s)", $self->{name},
$self->{cpqSasLogDrvStatus}, $self->{cpqSasLogDrvRaidLevel});
}
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cpqSasLogDrvHbaIndex cpqSasLogDrvIndex cpqSasLogDrvRaidLevel
cpqSasLogDrvStatus cpqSasLogDrvCondition
cpqSasLogDrvRebuildingPercent)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Sas::PhysicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqSasPhyDrvHbaIndex => $params{cpqSasPhyDrvHbaIndex},
cpqSasPhyDrvIndex => $params{cpqSasPhyDrvIndex},
cpqSasPhyDrvLocationString => $params{cpqSasPhyDrvLocationString},
cpqSasPhyDrvStatus => $params{cpqSasPhyDrvStatus},
cpqSasPhyDrvSize => $params{cpqSasPhyDrvSize},
cpqSasPhyDrvCondition => $params{cpqSasPhyDrvCondition},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqSasPhyDrvHbaIndex}.':'.$self->{cpqSasPhyDrvIndex}; ####vorerst
$self->{controllerindex} = $self->{cpqSasPhyDrvHbaIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqSasPhyDrvCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqSasPhyDrvCondition});
}
$self->add_info(
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqSasPhyDrvCondition});
}
sub dump {
my $self = shift;
printf "[PHYSICAL_DRIVE]\n";
foreach (qw(cpqSasPhyDrvHbaIndex cpqSasPhyDrvIndex cpqSasPhyDrvLocationString
cpqSasPhyDrvSize cpqSasPhyDrvStatus cpqSasPhyDrvCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Sas::SpareDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Sas);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub dump {
my $self = shift;
printf "[SPARE_DRIVE]\n";
}
package HP::Proliant::Component::DiskSubsystem::Scsi::CLI;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
}
package HP::Proliant::Component::DiskSubsystem::Scsi::SNMP;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQSCSI-MIB
my $oids = {
cpqScsiCntlrEntry => '1.3.6.1.4.1.232.5.2.2.1.1',
cpqScsiCntlrIndex => '1.3.6.1.4.1.232.5.2.2.1.1.1',
cpqScsiCntlrBusIndex => '1.3.6.1.4.1.232.5.2.2.1.1.2',
cpqScsiCntlrSlot => '1.3.6.1.4.1.232.5.2.2.1.1.6',
cpqScsiCntlrStatus => '1.3.6.1.4.1.232.5.2.2.1.1.7',
cpqScsiCntlrCondition => '1.3.6.1.4.1.232.5.2.2.1.1.12',
cpqScsiCntlrHwLocation => '1.3.6.1.4.1.232.5.2.2.1.1.16',
cpqScsiCntlrStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
},
cpqScsiCntlrConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
}
};
# INDEX { cpqScsiCntlrIndex, cpqScsiCntlrBusIndex }
foreach ($self->get_entries($oids, 'cpqScsiCntlrEntry')) {
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Scsi::Controller->new(%{$_}));
}
$oids = {
cpqScsiLogDrvEntry => '1.3.6.1.4.1.232.5.2.3.1.1',
cpqScsiLogDrvCntlrIndex => '1.3.6.1.4.1.232.5.2.3.1.1.1',
cpqScsiLogDrvBusIndex => '1.3.6.1.4.1.232.5.2.3.1.1.2',
cpqScsiLogDrvIndex => '1.3.6.1.4.1.232.5.2.3.1.1.3',
cpqScsiLogDrvFaultTol => '1.3.6.1.4.1.232.5.2.3.1.1.4',
cpqScsiLogDrvStatus => '1.3.6.1.4.1.232.5.2.3.1.1.5',
cpqScsiLogDrvSize => '1.3.6.1.4.1.232.5.2.3.1.1.6',
cpqScsiLogDrvPhyDrvIDs => '1.3.6.1.4.1.232.5.2.3.1.1.7',
cpqScsiLogDrvCondition => '1.3.6.1.4.1.232.5.2.3.1.1.8',
cpqScsiLogDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "unconfigured",
5 => "recovering",
6 => "readyForRebuild",
7 => "rebuilding",
8 => "wrongDrive",
9 => "badConnect",
},
cpqScsiLogDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqScsiLogDrvFaultTolValue => {
1 => "other",
2 => "none",
3 => "mirroring",
4 => "dataGuard",
5 => "distribDataGuard",
},
};
# INDEX { cpqScsiLogDrvCntlrIndex, cpqScsiLogDrvBusIndex, cpqScsiLogDrvIndex }
foreach ($self->get_entries($oids, 'cpqScsiLogDrvEntry')) {
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Scsi::LogicalDrive->new(%{$_}));
}
$oids = {
cpqScsiPhyDrvEntry => '1.3.6.1.4.1.232.5.2.4.1.1',
cpqScsiPhyDrvCntlrIndex => '1.3.6.1.4.1.232.5.2.4.1.1.1',
cpqScsiPhyDrvBusIndex => '1.3.6.1.4.1.232.5.2.4.1.1.2',
cpqScsiPhyDrvIndex => '1.3.6.1.4.1.232.5.2.4.1.1.3',
cpqScsiPhyDrvStatus => '1.3.6.1.4.1.232.5.2.4.1.1.9',
cpqScsiPhyDrvSize => '1.3.6.1.4.1.232.5.2.4.1.1.7',
cpqScsiPhyDrvCondition => '1.3.6.1.4.1.232.5.2.4.1.1.26',
cpqScsiPhyDrvConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqScsiPhyDrvStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "notConfigured",
5 => "badCable",
6 => "missingWasOk",
7 => "missingWasFailed",
8 => "predictiveFailure",
9 => "missingWasPredictiveFailure",
10 => "offline",
11 => "missingWasOffline",
12 => "hardError",
},
};
# INDEX { cpqScsiPhyDrvCntlrIndex, cpqScsiPhyDrvBusIndex, cpqScsiPhyDrvIndex }
foreach ($self->get_entries($oids, 'cpqScsiPhyDrvEntry')) {
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Scsi::PhysicalDrive->new(%{$_}));
}
}
package HP::Proliant::Component::DiskSubsystem::Scsi;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
controllers => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
condition => undef,
blacklisted => 0,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Scsi::SNMP';
} else {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Scsi::CLI';
}
$self->init();
$self->assemble();
return $self;
}
sub check {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Scsi::Controller;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqScsiCntlrIndex => $params{cpqScsiCntlrIndex},
cpqScsiCntlrBusIndex => $params{cpqScsiCntlrBusIndex},
cpqScsiCntlrSlot => $params{cpqScsiCntlrSlot},
cpqScsiCntlrStatus => $params{cpqScsiCntlrStatus},
cpqScsiCntlrCondition => $params{cpqScsiCntlrCondition},
cpqScsiCntlrHwLocation => $params{cpqScsiCntlrHwLocation},
blacklisted => 0,
};
$self->{name} = $params{name} || $self->{cpqScsiCntlrIndex};
$self->{controllerindex} = $self->{cpqScsiCntlrIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqScsiCntlrCondition} eq 'other') {
if (scalar(@{$self->{physical_drives}})) {
$self->add_message(CRITICAL,
sprintf 'scsi controller in slot %s needs attention',
$self->{cpqScsiCntlrSlot});
$self->add_info(sprintf 'scsi controller in slot %s needs attention',
$self->{cpqScsiCntlrSlot});
} else {
$self->add_info(sprintf 'scsi controller in slot %s is ok and unused',
$self->{cpqScsiCntlrSlot});
$self->{blacklisted} = 1;
}
} elsif ($self->{cpqScsiCntlrCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'scsi controller in slot %s needs attention',
$self->{cpqScsiCntlrSlot});
$self->add_info(sprintf 'scsi controller in slot %s needs attention',
$self->{cpqScsiCntlrSlot});
} else {
$self->add_info(sprintf 'scsi controller in slot %s is ok',
$self->{cpqScsiCntlrSlot});
}
foreach (@{$self->{logical_drives}}) {
$_->check();
}
foreach (@{$self->{physical_drives}}) {
$_->check();
}
foreach (@{$self->{spare_drives}}) {
$_->check();
}
}
sub dump {
my $self = shift;
printf "[SCSI_CONTROLLER_%s]\n", $self->{name};
foreach (qw(cpqScsiCntlrIndex cpqScsiCntlrBusIndex cpqScsiCntlrSlot
cpqScsiCntlrStatus cpqScsiCntlrCondition cpqScsiCntlrHwLocation)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
foreach (@{$self->{logical_drives}}) {
$_->dump();
}
foreach (@{$self->{physical_drives}}) {
$_->dump();
}
foreach (@{$self->{spare_drives}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Scsi::LogicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqScsiLogDrvCntlrIndex => $params{cpqScsiLogDrvCntlrIndex},
cpqScsiLogDrvBusIndex => $params{cpqScsiLogDrvBusIndex},
cpqScsiLogDrvIndex => $params{cpqScsiLogDrvIndex},
cpqScsiLogDrvFaultTol => $params{cpqScsiLogDrvFaultTol},
cpqScsiLogDrvStatus => $params{cpqScsiLogDrvStatus},
cpqScsiLogDrvSize => $params{cpqScsiLogDrvSize},
cpqScsiLogDrvPhyDrvIDs => $params{cpqScsiLogDrvPhyDrvIDs},
cpqScsiLogDrvCondition => $params{cpqScsiLogDrvCondition},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqScsiLogDrvCntlrIndex}.':'.
$self->{cpqScsiLogDrvBusIndex}.':'.
$self->{cpqScsiLogDrvIndex}; ####vorerst
$self->{controllerindex} = $self->{cpqScsiLogDrvCntlrIndex};
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqScsiLogDrvCondition} ne "ok") {
if ($self->{cpqScsiLogDrvStatus} =~
/rebuild|recovering/) {
$self->add_message(WARNING,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqScsiLogDrvStatus});
} else {
$self->add_message(CRITICAL,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqScsiLogDrvStatus});
}
}
$self->add_info(
sprintf "logical drive %s is %s", $self->{name},
$self->{cpqScsiLogDrvStatus});
}
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cpqScsiLogDrvCntlrIndex cpqScsiLogDrvBusIndex cpqScsiLogDrvIndex
cpqScsiLogDrvFaultTol cpqScsiLogDrvStatus cpqScsiLogDrvSize
cpqScsiLogDrvPhyDrvIDs cpqScsiLogDrvCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Scsi::PhysicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqScsiPhyDrvCntlrIndex => $params{cpqScsiPhyDrvCntlrIndex},
cpqScsiPhyDrvBusIndex => $params{cpqScsiPhyDrvBusIndex},
cpqScsiPhyDrvIndex => $params{cpqScsiPhyDrvIndex},
cpqScsiPhyDrvStatus => $params{cpqScsiPhyDrvStatus},
cpqScsiPhyDrvSize => $params{cpqScsiPhyDrvSize},
cpqScsiPhyDrvCondition => $params{cpqScsiPhyDrvCondition},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqScsiPhyDrvCntlrIndex}.':'.$self->{cpqScsiPhyDrvIndex};
$self->{controllerindex} = $self->{cpqScsiPhyDrvCntlrIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqScsiPhyDrvCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqScsiPhyDrvCondition});
}
$self->add_info(
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqScsiPhyDrvCondition});
}
sub dump {
my $self = shift;
printf "[PHYSICAL_DRIVE]\n";
foreach (qw(cpqScsiPhyDrvCntlrIndex cpqScsiPhyDrvBusIndex cpqScsiPhyDrvIndex
cpqScsiPhyDrvStatus cpqScsiPhyDrvSize cpqScsiPhyDrvCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Scsi::SpareDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Scsi);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub dump {
my $self = shift;
printf "[SPARE_DRIVE]\n";
}
package HP::Proliant::Component::DiskSubsystem::Ide::CLI;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
}
package HP::Proliant::Component::DiskSubsystem::Ide::SNMP;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQIDE-MIB
my $oids = {
cpqIdeControllerEntry => '1.3.6.1.4.1.232.14.2.3.1.1',
cpqIdeControllerIndex => '1.3.6.1.4.1.232.14.2.3.1.1.1',
cpqIdeControllerOverallCondition => '1.3.6.1.4.1.232.14.2.3.1.1.2',
cpqIdeControllerModel => '1.3.6.1.4.1.232.14.2.3.1.1.3',
cpqIdeControllerSlot => '1.3.6.1.4.1.232.14.2.3.1.1.5',
cpqIdeControllerOverallConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqIdeControllerIndex }
foreach ($self->get_entries($oids, 'cpqIdeControllerEntry')) {
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Ide::Controller->new(%{$_}));
}
$oids = {
cpqIdeLogicalDriveEntry => '1.3.6.1.4.1.232.14.2.6.1.1',
cpqIdeLogicalDriveControllerIndex => '1.3.6.1.4.1.232.14.2.6.1.1.1',
cpqIdeLogicalDriveIndex => '1.3.6.1.4.1.232.14.2.6.1.1.2',
cpqIdeLogicalDriveRaidLevel => '1.3.6.1.4.1.232.14.2.6.1.1.3',
cpqIdeLogicalDriveCapacity => '1.3.6.1.4.1.232.14.2.6.1.1.4',
cpqIdeLogicalDriveStatus => '1.3.6.1.4.1.232.14.2.6.1.1.5',
cpqIdeLogicalDriveCondition => '1.3.6.1.4.1.232.14.2.6.1.1.6',
cpqIdeLogicalDriveDiskIds => '1.3.6.1.4.1.232.14.2.6.1.1.7',
cpqIdeLogicalDriveSpareIds => '1.3.6.1.4.1.232.14.2.6.1.1.9',
cpqIdeLogicalDriveRebuildingDisk => '1.3.6.1.4.1.232.14.2.6.1.1.10',
cpqIdeLogicalDriveRaidLevelValue => {
1 => "other",
2 => "raid0",
3 => "raid1",
4 => "raid0plus1",
},
cpqIdeLogicalDriveStatusValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "rebuilding",
5 => "failed",
},
cpqIdeLogicalDriveConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqIdeLogicalDriveControllerIndex, cpqIdeLogicalDriveIndex }
foreach ($self->get_entries($oids, 'cpqIdeLogicalDriveEntry')) {
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Ide::LogicalDrive->new(%{$_}));
}
$oids = {
cpqIdeAtaDiskEntry => '1.3.6.1.4.1.232.14.2.4.1.1',
cpqIdeAtaDiskControllerIndex => '1.3.6.1.4.1.232.14.2.4.1.1.1',
cpqIdeAtaDiskIndex => '1.3.6.1.4.1.232.14.2.4.1.1.2',
cpqIdeAtaDiskModel => '1.3.6.1.4.1.232.14.2.4.1.1.3',
cpqIdeAtaDiskStatus => '1.3.6.1.4.1.232.14.2.4.1.1.6',
cpqIdeAtaDiskCondition => '1.3.6.1.4.1.232.14.2.4.1.1.7',
cpqIdeAtaDiskCapacity => '1.3.6.1.4.1.232.14.2.4.1.1.8',
cpqIdeAtaDiskLogicalDriveMember => '1.3.6.1.4.1.232.14.2.4.1.1.13',
cpqIdeAtaDiskIsSpare => '1.3.6.1.4.1.232.14.2.4.1.1.14',
cpqIdeAtaDiskStatusValue => {
1 => "other",
2 => "ok",
3 => "smartError",
4 => "failed",
},
cpqIdeAtaDiskConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqIdeAtaDiskControllerIndex, cpqIdeAtaDiskIndex }
foreach ($self->get_entries($oids, 'cpqIdeAtaDiskEntry')) {
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Ide::PhysicalDrive->new(%{$_}));
}
}
package HP::Proliant::Component::DiskSubsystem::Ide;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
controllers => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
condition => undef,
blacklisted => 0,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Ide::SNMP';
} else {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Ide::CLI';
}
$self->init();
$self->assemble();
return $self;
}
sub check {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{controllers}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Ide::Controller;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqIdeControllerIndex => $params{cpqIdeControllerIndex},
cpqIdeControllerOverallCondition => $params{cpqIdeControllerOverallCondition},
cpqIdeControllerModel => $params{cpqIdeControllerModel},
cpqIdeControllerSlot => $params{cpqIdeControllerSlot}, # -1 ist sysboard?
blacklisted => 0,
};
$self->{name} = $params{name} || $self->{cpqIdeControllerIndex};
$self->{controllerindex} = $self->{cpqIdeControllerIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqIdeControllerOverallCondition} eq 'other') {
if (scalar(@{$self->{physical_drives}})) {
$self->add_message(CRITICAL,
sprintf 'ide controller %s in slot %s needs attention',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
$self->add_info(sprintf 'ide controller %s in slot %s needs attention',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
} else {
$self->add_info(sprintf 'ide controller %s in slot %s is ok and unused',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
$self->{blacklisted} = 1;
}
} elsif ($self->{cpqIdeControllerOverallCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'ide controller %s in slot %s needs attention',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
$self->add_info(sprintf 'ide controller %s in slot %s needs attention',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
} else {
$self->add_info(sprintf 'ide controller %s in slot %s is ok',
$self->{cpqIdeControllerIndex}, $self->{cpqIdeControllerSlot});
}
foreach (@{$self->{logical_drives}}) {
$_->check();
}
foreach (@{$self->{physical_drives}}) {
$_->check();
}
foreach (@{$self->{spare_drives}}) {
$_->check();
}
}
sub dump {
my $self = shift;
printf "[IDE_CONTROLLER_%s]\n", $self->{name};
foreach (qw(cpqIdeControllerIndex cpqIdeControllerSlot
cpqIdeControllerModel cpqIdeControllerOverallCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
foreach (@{$self->{logical_drives}}) {
$_->dump();
}
foreach (@{$self->{physical_drives}}) {
$_->dump();
}
foreach (@{$self->{spare_drives}}) {
$_->dump();
}
}
package HP::Proliant::Component::DiskSubsystem::Ide::LogicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqIdeLogicalDriveControllerIndex => $params{cpqIdeLogicalDriveControllerIndex},
cpqIdeLogicalDriveIndex => $params{cpqIdeLogicalDriveIndex},
cpqIdeLogicalDriveRaidLevel => $params{cpqIdeLogicalDriveRaidLevel},
cpqIdeLogicalDriveCapacity => $params{cpqIdeLogicalDriveCapacity},
cpqIdeLogicalDriveStatus => $params{cpqIdeLogicalDriveStatus},
cpqIdeLogicalDriveCondition => $params{cpqIdeLogicalDriveCondition},
cpqIdeLogicalDriveDiskIds => $params{cpqIdeLogicalDriveDiskIds},
cpqIdeLogicalDriveSpareIds => $params{cpqIdeLogicalDriveSpareIds},
cpqIdeLogicalDriveRebuildingDisk => $params{cpqIdeLogicalDriveRebuildingDisk},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqIdeLogicalDriveControllerIndex}.':'.
$self->{cpqIdeLogicalDriveIndex};
$self->{controllerindex} = $self->{cpqIdeLogicalDriveControllerIndex};
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqIdeLogicalDriveCondition} ne "ok") {
if ($self->{cpqIdeLogicalDriveStatus} =~
/rebuild/) {
$self->add_message(WARNING,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqIdeLogicalDriveStatus});
} else {
$self->add_message(CRITICAL,
sprintf "logical drive %s is %s",
$self->{name}, $self->{cpqIdeLogicalDriveStatus});
}
}
$self->add_info(
sprintf "logical drive %s is %s", $self->{name},
$self->{cpqIdeLogicalDriveStatus});
}
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cpqIdeLogicalDriveControllerIndex cpqIdeLogicalDriveIndex
cpqIdeLogicalDriveRaidLevel cpqIdeLogicalDriveCapacity
cpqIdeLogicalDriveDiskIds cpqIdeLogicalDriveSpareIds
cpqIdeLogicalDriveRebuildingDisk cpqIdeLogicalDriveStatus
cpqIdeLogicalDriveCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Ide::PhysicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqIdeAtaDiskControllerIndex => $params{cpqIdeAtaDiskControllerIndex},
cpqIdeAtaDiskIndex => $params{cpqIdeAtaDiskIndex},
cpqIdeAtaDiskModel => $params{cpqIdeAtaDiskModel},
cpqIdeAtaDiskStatus => $params{cpqIdeAtaDiskStatus},
cpqIdeAtaDiskCondition => $params{cpqIdeAtaDiskCondition},
cpqIdeAtaDiskCapacity => $params{cpqIdeAtaDiskCapacity},
cpqIdeAtaDiskLogicalDriveMember => $params{cpqIdeAtaDiskLogicalDriveMember},
cpqIdeAtaDiskIsSpare => $params{cpqIdeAtaDiskIsSpare},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqIdeAtaDiskControllerIndex}.':'.
$self->{cpqIdeAtaDiskIndex}; ####vorerst
$self->{controllerindex} = $self->{cpqIdeAtaDiskControllerIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqIdeAtaDiskCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqIdeAtaDiskCondition});
}
$self->add_info(
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqIdeAtaDiskCondition});
}
sub dump {
my $self = shift;
printf "[PHYSICAL_DRIVE]\n";
foreach (qw(cpqIdeAtaDiskControllerIndex cpqIdeAtaDiskIndex
cpqIdeAtaDiskModel cpqIdeAtaDiskCapacity cpqIdeAtaDiskLogicalDriveMember
cpqIdeAtaDiskStatus cpqIdeAtaDiskCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Ide::SpareDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Ide);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub dump {
my $self = shift;
printf "[SPARE_DRIVE]\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::CLI;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
# .....
$self->{global_status} =
HP::Proliant::Component::DiskSubsystem::Fca::GlobalStatus->new(
runtime => $self->{runtime},
cpqFcaMibCondition => 'n/a',
);
}
package HP::Proliant::Component::DiskSubsystem::Fca::SNMP;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca
HP::Proliant::Component::SNMP);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
blacklisted => 0,
};
bless $self, $class;
return $self;
}
sub init {
my $self = shift;
my $snmpwalk = $self->{rawdata};
# CPQFCA-MIB
my $oids = {
cpqFcaHostCntlrEntry => '1.3.6.1.4.1.232.16.2.7.1.1',
cpqFcaHostCntlrIndex => '1.3.6.1.4.1.232.16.2.7.1.1.1',
cpqFcaHostCntlrSlot => '1.3.6.1.4.1.232.16.2.7.1.1.2',
cpqFcaHostCntlrModel => '1.3.6.1.4.1.232.16.2.7.1.1.3',
cpqFcaHostCntlrStatus => '1.3.6.1.4.1.232.16.2.7.1.1.4',
cpqFcaHostCntlrCondition => '1.3.6.1.4.1.232.16.2.7.1.1.5',
cpqFcaHostCntlrOverallCondition => '1.3.6.1.4.1.232.16.2.7.1.1.8',
cpqFcaHostCntlrModelValue => {
1 => "other",
2 => "fchc-p",
3 => "fchc-e",
4 => "fchc64",
5 => "sa-sam",
6 => "fca-2101",
7 => "sw64-33",
8 => "fca-221x",
9 => "dpfcmc",
},
cpqFcaHostCntlrStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "shutdown",
5 => "loopDegraded",
6 => "loopFailed",
},
cpqFcaHostCntlrConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
cpqFcaHostCntlrOverallConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
}, # cntrl + alle associated storage boxes
};
# INDEX { cpqFcaHostCntlrIndex }
foreach ($self->get_entries($oids, 'cpqFcaHostCntlrEntry')) {
push(@{$self->{host_controllers}},
HP::Proliant::Component::DiskSubsystem::Fca::HostController->new(%{$_}));
}
$oids = {
cpqFcaCntlrEntry => '1.3.6.1.4.1.232.16.2.2.1.1',
cpqFcaCntlrBoxIndex => '1.3.6.1.4.1.232.16.2.2.1.1.1',
cpqFcaCntlrBoxIoSlot => '1.3.6.1.4.1.232.16.2.2.1.1.2',
cpqFcaCntlrModel => '1.3.6.1.4.1.232.16.2.2.1.1.3',
cpqFcaCntlrStatus => '1.3.6.1.4.1.232.16.2.2.1.1.5',
cpqFcaCntlrCondition => '1.3.6.1.4.1.232.16.2.2.1.1.6',
cpqFcaCntlrModelValue => {
1 => "other",
2 => "fibreArray",
3 => "msa1000",
4 => "smartArrayClusterStorage",
5 => "hsg80",
6 => "hsv110",
7 => "msa500g2",
8 => "msa20",
},
cpqFcaCntlrStatusValue => {
1 => "other",
2 => "ok",
3 => "failed",
4 => "offline",
4 => "redundantPathOffline",
},
cpqFcaCntlrConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqFcaCntlrBoxIndex, cpqFcaCntlrBoxIoSlot }
foreach ($self->get_entries($oids, 'cpqFcaCntlrEntry')) {
push(@{$self->{controllers}},
HP::Proliant::Component::DiskSubsystem::Fca::Controller->new(%{$_}));
}
$oids = {
cpqFcaAccelEntry => '1.3.6.1.4.1.232.16.2.2.2.1',
cpqFcaAccelBoxIndex => '1.3.6.1.4.1.232.16.2.2.2.1.1',
cpqFcaAccelBoxIoSlot => '1.3.6.1.4.1.232.16.2.2.2.1.2',
cpqFcaAccelStatus => '1.3.6.1.4.1.232.16.2.2.2.1.3',
cpqFcaAccelErrCode => '1.3.6.1.4.1.232.16.2.2.2.1.5',
cpqFcaAccelBatteryStatus => '1.3.6.1.4.1.232.16.2.2.2.1.6',
cpqFcaAccelCondition => '1.3.6.1.4.1.232.16.2.2.2.1.9',
cpqFcaAccelStatusValue => {
1 => "other",
2 => "invalid",
3 => "enabled",
4 => "tmpDisabled",
5 => "permDisabled",
},
cpqFcaAccelErrCodeValue => {
1 => 'other',
2 => 'invalid',
3 => 'badConfig',
4 => 'lowBattery',
5 => 'disableCmd',
6 => 'noResources',
7 => 'notConnected',
8 => 'badMirrorData',
9 => 'readErr',
10 => 'writeErr',
11 => 'configCmd',
12 => 'expandInProgress',
13 => 'snapshotInProgress',
14 => 'redundantLowBattery',
15 => 'redundantSizeMismatch',
16 => 'redundantCacheFailure',
17 => 'excessiveEccErrors',
19 => 'postEccErrors',
},
cpqFcaAccelBatteryStatusValue => {
1 => 'other',
2 => 'ok',
3 => 'recharging',
4 => 'failed',
5 => 'degraded',
6 => 'notPresent',
},
cpqFcaAccelConditionValue => {
1 => "other",
2 => "ok",
3 => "degraded",
4 => "failed",
},
};
# INDEX { cpqFcaAccelBoxIndex, cpqFcaAccelBoxIoSlot }
foreach ($self->get_entries($oids, 'cpqFcaAccelEntry')) {
push(@{$self->{accelerators}},
HP::Proliant::Component::DiskSubsystem::Fca::Accelerator->new(%{$_}));
}
$oids = {
cpqFcaLogDrvEntry => '1.3.6.1.4.1.232.16.2.3.1.1',
cpqFcaLogDrvBoxIndex => '1.3.6.1.4.1.232.16.2.3.1.1.1',
cpqFcaLogDrvIndex => '1.3.6.1.4.1.232.16.2.3.1.1.2',
cpqFcaLogDrvFaultTol => '1.3.6.1.4.1.232.16.2.3.1.1.3',
cpqFcaLogDrvStatus => '1.3.6.1.4.1.232.16.2.3.1.1.4',
cpqFcaLogDrvPercentRebuild => '1.3.6.1.4.1.232.16.2.3.1.1.6',
cpqFcaLogDrvSize => '1.3.6.1.4.1.232.16.2.3.1.1.9',
cpqFcaLogDrvPhyDrvIDs => '1.3.6.1.4.1.232.16.2.3.1.1.10',
cpqFcaLogDrvCondition => '1.3.6.1.4.1.232.16.2.3.1.1.11',
cpqFcaLogDrvFaultTolValue => {
1 => 'other',
2 => 'none',
3 => 'mirroring',
4 => 'dataGuard',
5 => 'distribDataGuard',
7 => 'advancedDataGuard',
},
cpqFcaLogDrvStatusValue => {
1 => 'other',
2 => 'ok',
3 => 'failed',
4 => 'unconfigured',
5 => 'recovering',
6 => 'readyForRebuild',
7 => 'rebuilding',
8 => 'wrongDrive',
9 => 'badConnect',
10 => 'overheating',
11 => 'shutdown',
12 => 'expanding',
13 => 'notAvailable',
14 => 'queuedForExpansion',
15 => 'hardError',
},
cpqFcaLogDrvConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
};
# INDEX { cpqFcaLogDrvBoxIndex, cpqFcaLogDrvIndex }
foreach ($self->get_entries($oids, 'cpqFcaLogDrvEntry')) {
push(@{$self->{logical_drives}},
HP::Proliant::Component::DiskSubsystem::Fca::LogicalDrive->new(%{$_}));
}
$oids = {
cpqFcaPhyDrvEntry => '1.3.6.1.4.1.232.16.2.5.1.1',
cpqFcaPhyDrvBoxIndex => '1.3.6.1.4.1.232.16.2.5.1.1.1',
cpqFcaPhyDrvIndex => '1.3.6.1.4.1.232.16.2.5.1.1.2',
cpqFcaPhyDrvModel => '1.3.6.1.4.1.232.16.2.5.1.1.3',
cpqFcaPhyDrvBay => '1.3.6.1.4.1.232.16.2.5.1.1.5',
cpqFcaPhyDrvStatus => '1.3.6.1.4.1.232.16.2.5.1.1.6',
cpqFcaPhyDrvCondition => '1.3.6.1.4.1.232.16.2.5.1.1.31',
cpqFcaPhyDrvSize => '1.3.6.1.4.1.232.16.2.5.1.1.38',
cpqFcaPhyDrvBusNumber => '1.3.6.1.4.1.232.16.2.5.1.1.42',
cpqFcaPhyDrvStatusValue => {
1 => 'other',
2 => 'unconfigured',
3 => 'ok',
4 => 'threshExceeded',
5 => 'predictiveFailure',
6 => 'failed',
},
cpqFcaPhyDrvConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
};
# INDEX { cpqFcaPhyDrvBoxIndex, cpqFcaPhyDrvIndex }
foreach ($self->get_entries($oids, 'cpqFcaPhyDrvEntry')) {
push(@{$self->{physical_drives}},
HP::Proliant::Component::DiskSubsystem::Fca::PhysicalDrive->new(%{$_}));
}
$oids = {
cpqFcaMibRevMajor => '1.3.6.1.4.1.232.16.1.1.0',
cpqFcaMibRevMinor => '1.3.6.1.4.1.232.16.1.2.0',
cpqFcaMibCondition => '1.3.6.1.4.1.232.16.1.3.0',
cpqFcaMibConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
};
$self->{global_status} =
HP::Proliant::Component::DiskSubsystem::Fca::GlobalStatus->new(
runtime => $self->{runtime},
cpqFcaMibCondition =>
SNMP::Utils::get_object_value($snmpwalk,
$oids->{cpqFcaMibCondition}, $oids->{cpqFcaMibConditionValue})
);
}
package HP::Proliant::Component::DiskSubsystem::Fca;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
host_controllers => [],
controllers => [],
accelerators => [],
physical_drives => [],
logical_drives => [],
spare_drives => [],
global_status => undef,
blacklisted => 0,
};
bless $self, $class;
if ($self->{method} eq 'snmp') {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Fca::SNMP';
} else {
bless $self, 'HP::Proliant::Component::DiskSubsystem::Fca::CLI';
}
$self->init();
$self->assemble();
return $self;
}
sub assemble {
my $self = shift;
$self->trace(3, sprintf "%s controllers und platten zusammenf<6E>hren",
ref($self));
$self->trace(3, sprintf "has %d host controllers",
scalar(@{$self->{host_controllers}}));
$self->trace(3, sprintf "has %d controllers",
scalar(@{$self->{controllers}}));
$self->trace(3, sprintf "has %d physical_drives",
scalar(@{$self->{physical_drives}}));
$self->trace(3, sprintf "has %d logical_drives",
scalar(@{$self->{logical_drives}}));
$self->trace(3, sprintf "has %d spare_drives",
scalar(@{$self->{spare_drives}}));
}
sub check {
my $self = shift;
foreach (@{$self->{host_controllers}}) {
$_->check();
}
foreach (@{$self->{controllers}}) {
$_->check();
}
foreach (@{$self->{accelerators}}) {
$_->check();
}
foreach (@{$self->{logical_drives}}) {
$_->check();
}
foreach (@{$self->{physical_drives}}) {
$_->check();
}
# wozu eigentlich?
#if (! $self->has_controllers()) {
#$self->{global_status}->check();
#}
}
sub dump {
my $self = shift;
foreach (@{$self->{host_controllers}}) {
$_->dump();
}
foreach (@{$self->{controllers}}) {
$_->dump();
}
foreach (@{$self->{accelerators}}) {
$_->dump();
}
foreach (@{$self->{logical_drives}}) {
$_->dump();
}
foreach (@{$self->{physical_drives}}) {
$_->dump();
}
#$self->{global_status}->dump();
}
package HP::Proliant::Component::DiskSubsystem::Fca::GlobalStatus;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaMibCondition => $params{cpqFcaMibCondition},
};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->{cpqFcaMibCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'fcal overall condition is %s', $self->{cpqFcaMibCondition});
}
$self->{info} =
sprintf 'fcal overall condition is %s', $self->{cpqFcaMibCondition};
}
sub dump {
my $self = shift;
printf "[FCAL]\n";
foreach (qw(cpqFcaMibCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::HostController;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaHostCntlrIndex => $params{cpqFcaHostCntlrIndex},
cpqFcaHostCntlrSlot => $params{cpqFcaHostCntlrSlot},
cpqFcaHostCntlrModel => $params{cpqFcaHostCntlrModel},
cpqFcaHostCntlrStatus => $params{cpqFcaHostCntlrStatus},
cpqFcaHostCntlrCondition => $params{cpqFcaHostCntlrCondition},
cpqFcaHostCntlrOverallCondition => $params{cpqFcaHostCntlrOverallCondition},
blacklisted => 0,
};
$self->{name} = $params{name} || $self->{cpqFcaHostCntlrIndex};
$self->{controllerindex} = $self->{cpqFcaHostCntlrIndex};
$self->{ident} = sprintf '%s in slot %s',
$self->{cpqFcaHostCntlrIndex}, $self->{cpqFcaHostCntlrSlot};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
if ($self->is_blacklisted('fcahc',
$self->{cpqFcaHostCntlrSlot}.'-'.$self->{cpqFcaHostCntlrIndex})) {
$self->add_info(sprintf 'fcal host controller %s is %s (blacklisted)',
$self->{ident}, $self->{cpqFcaHostCntlrCondition});
} elsif ($self->{cpqFcaHostCntlrCondition} eq 'other') {
$self->add_message(CRITICAL,
sprintf 'fcal host controller %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
$self->add_info(sprintf 'fcal host controller %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
} elsif ($self->{cpqFcaHostCntlrCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'fcal host controller %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
$self->add_info(sprintf 'fcal host controller %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
} else {
$self->add_info(sprintf 'fcal host controller %s is ok',
$self->{ident});
}
if ($self->is_blacklisted('fcahc',
$self->{cpqFcaHostCntlrSlot}.'-'.$self->{cpqFcaHostCntlrIndex})) {
} elsif ($self->{cpqFcaHostCntlrOverallCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'fcal host controller %s reports problems (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
$self->add_info(sprintf 'fcal host controller %s reports problems (%s)',
$self->{ident}, $self->{cpqFcaHostCntlrStatus});
}
}
sub dump {
my $self = shift;
printf "[FCAL_HOST_CONTROLLER_%s]\n", $self->{name};
foreach (qw(cpqFcaHostCntlrIndex cpqFcaHostCntlrSlot
cpqFcaHostCntlrModel cpqFcaHostCntlrStatus cpqFcaHostCntlrCondition
cpqFcaHostCntlrOverallCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::Controller;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaCntlrBoxIndex => $params{cpqFcaCntlrBoxIndex},
cpqFcaCntlrBoxIoSlot => $params{cpqFcaCntlrBoxIoSlot},
cpqFcaCntlrModel => $params{cpqFcaCntlrModel},
cpqFcaCntlrStatus => $params{cpqFcaCntlrStatus},
cpqFcaCntlrCondition => $params{cpqFcaCntlrCondition},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqFcaCntlrBoxIndex}.':'.$self->{cpqFcaCntlrBoxIoSlot};
$self->{controllerindex} = $self->{cpqFcaCntlrBoxIndex};
$self->{ident} = sprintf 'in box %s/slot %s',
$self->{cpqFcaCntlrBoxIndex}, $self->{cpqFcaCntlrBoxIoSlot};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('fcaco', $self->{name});
if ($self->{cpqFcaCntlrCondition} eq 'other') {
if (1) { # was ist mit phys. drives?
$self->add_message(CRITICAL,
sprintf 'fcal controller %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaCntlrStatus});
$self->add_info(sprintf 'fcal controller %s %s needs attention (%s)',
$self->{name}, $self->{ident}, $self->{cpqFcaCntlrStatus});
} else {
$self->add_info(sprintf 'fcal controller %s is ok and unused',
$self->{ident});
$self->{blacklisted} = 1;
}
} elsif ($self->{cpqFcaCntlrCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'fcal controller %s needs attention',
$self->{ident});
$self->add_info(sprintf 'fcal controller %s %s needs attention (%s)',
$self->{name}, $self->{ident}, $self->{cpqFcaCntlrCondition});
} else {
$self->add_info(sprintf 'fcal controller %s is ok',
$self->{ident});
}
}
sub dump {
my $self = shift;
printf "[FCAL_CONTROLLER_%s]\n", $self->{name};
foreach (qw(cpqFcaCntlrBoxIndex cpqFcaCntlrBoxIoSlot cpqFcaCntlrModel
cpqFcaCntlrStatus cpqFcaCntlrCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::Accelerator;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaAccelBoxIndex => $params{cpqFcaAccelBoxIndex},
cpqFcaAccelBoxIoSlot => $params{cpqFcaAccelBoxIoSlot},
cpqFcaAccelStatus => $params{cpqFcaAccelStatus},
cpqFcaAccelErrCode => $params{cpqFcaAccelErrCode},
cpqFcaAccelBatteryStatus => $params{cpqFcaAccelBatteryStatus},
cpqFcaAccelCondition => $params{cpqFcaAccelCondition},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqFcaAccelBoxIndex}.':'.$self->{cpqFcaAccelBoxIoSlot};
$self->{controllerindex} = $self->{cpqFcaAccelBoxIndex};
$self->{ident} = sprintf 'in box %s/slot %s',
$self->{cpqFcaAccelBoxIndex}, $self->{cpqFcaAccelBoxIoSlot};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
# !!! cpqFcaAccelStatus
$self->blacklist('fcaac', $self->{name});
if ($self->{cpqFcaAccelStatus} eq 'invalid') {
$self->add_info(sprintf 'fcal accelerator %s is not installed',
$self->{ident});
} elsif ($self->{cpqFcaAccelStatus} eq 'tmpDisabled') {
$self->add_info(sprintf 'fcal accelerator %s is temp disabled',
$self->{ident});
} elsif ($self->{cpqFcaAccelCondition} eq 'other') {
$self->add_message(CRITICAL,
sprintf 'fcal accelerator %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaAccelErrCode});
$self->add_info(sprintf 'fcal accelerator %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaAccelErrCode});
} elsif ($self->{cpqFcaAccelCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf 'fcal accelerator %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaAccelErrCode});
$self->add_info(sprintf 'fcal accelerator %s needs attention (%s)',
$self->{ident}, $self->{cpqFcaAccelErrCode});
} else {
$self->add_info(sprintf 'fcal accelerator %s is ok',
$self->{ident});
}
}
sub dump {
my $self = shift;
printf "[FCAL_ACCELERATOR_%s]\n", $self->{name};
foreach (qw(cpqFcaAccelBoxIndex cpqFcaAccelBoxIoSlot cpqFcaAccelStatus
cpqFcaAccelErrCode cpqFcaAccelBatteryStatus cpqFcaAccelCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::LogicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaLogDrvBoxIndex => $params{cpqFcaLogDrvBoxIndex},
cpqFcaLogDrvIndex => $params{cpqFcaLogDrvIndex},
cpqFcaLogDrvFaultTol => $params{cpqFcaLogDrvFaultTol},
cpqFcaLogDrvStatus => $params{cpqFcaLogDrvStatus},
cpqFcaLogDrvPercentRebuild => $params{cpqFcaLogDrvPercentRebuild},
cpqFcaLogDrvSize => $params{cpqFcaLogDrvSize},
cpqFcaLogDrvPhyDrvIDs => $params{cpqFcaLogDrvPhyDrvIDs},
cpqFcaLogDrvCondition => $params{cpqFcaLogDrvCondition},
blacklisted => 0,
};
bless $self, $class;
$self->{name} = $params{name} ||
$self->{cpqFcaLogDrvBoxIndex}.':'.
$self->{cpqFcaLogDrvIndex};
$self->{controllerindex} = $self->{cpqFcaLogDrvBoxIndex};
return $self;
}
sub check {
my $self = shift;
$self->blacklist('fcald', $self->{name});
if ($self->{cpqFcaLogDrvCondition} ne "ok") {
if ($self->{cpqFcaLogDrvStatus} =~
/rebuild|recovering|expand/) {
$self->add_message(WARNING,
sprintf "logical drive %s is %s (%s)",
$self->{name}, $self->{cpqFcaLogDrvStatus},
$self->{cpqFcaLogDrvFaultTol});
} else {
$self->add_message(CRITICAL,
sprintf "logical drive %s is %s (%s)",
$self->{name}, $self->{cpqFcaLogDrvStatus},
$self->{cpqFcaLogDrvFaultTol});
}
}
$self->add_info(
sprintf "logical drive %s is %s (%s)", $self->{name},
$self->{cpqFcaLogDrvStatus}, $self->{cpqFcaLogDrvFaultTol});
}
sub dump {
my $self = shift;
printf "[LOGICAL_DRIVE]\n";
foreach (qw(cpqFcaLogDrvBoxIndex cpqFcaLogDrvIndex cpqFcaLogDrvFaultTol
cpqFcaLogDrvStatus cpqFcaLogDrvPercentRebuild cpqFcaLogDrvSize
cpqFcaLogDrvPhyDrvIDs cpqFcaLogDrvCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::PhysicalDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
cpqFcaPhyDrvBoxIndex => $params{cpqFcaPhyDrvBoxIndex},
cpqFcaPhyDrvIndex => $params{cpqFcaPhyDrvIndex},
cpqFcaPhyDrvModel => $params{cpqFcaPhyDrvModel},
cpqFcaPhyDrvBay => $params{cpqFcaPhyDrvBay},
cpqFcaPhyDrvStatus => $params{cpqFcaPhyDrvStatus},
cpqFcaPhyDrvCondition => $params{cpqFcaPhyDrvCondition},
cpqFcaPhyDrvSize => $params{cpqFcaPhyDrvSize},
cpqFcaPhyDrvBusNumber => $params{cpqFcaPhyDrvBusNumber},
blacklisted => 0,
};
$self->{name} = $params{name} ||
$self->{cpqFcaPhyDrvBoxIndex}.':'.$self->{cpqFcaPhyDrvIndex}; ####vorerst
$self->{controllerindex} = $self->{cpqScsiPhyDrvCntlrIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('fcapd', $self->{name});
if ($self->{cpqFcaPhyDrvStatus} eq 'unconfigured') {
# not part of a logical drive
# condition will surely be other
} elsif ($self->{cpqFcaPhyDrvCondition} ne 'ok') {
$self->add_message(CRITICAL,
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqFcaPhyDrvStatus});
}
$self->add_info(
sprintf "physical drive %s is %s",
$self->{name}, $self->{cpqFcaPhyDrvStatus});
}
sub dump {
my $self = shift;
printf "[PHYSICAL_DRIVE]\n";
foreach (qw(cpqFcaPhyDrvBoxIndex cpqFcaPhyDrvIndex cpqFcaPhyDrvModel
cpqFcaPhyDrvBay cpqFcaPhyDrvStatus cpqFcaPhyDrvCondition
cpqFcaPhyDrvSize cpqFcaPhyDrvBusNumber)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::Proliant::Component::DiskSubsystem::Fca::SpareDrive;
our @ISA = qw(HP::Proliant::Component::DiskSubsystem::Fca);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub dump {
my $self = shift;
printf "[SPARE_DRIVE]\n";
}
package HP::Proliant::Component::DiskSubsystem;
our @ISA = qw(HP::Proliant::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
da_subsystem => undef,
sas_da_subsystem => undef,
ide_da_subsystem => undef,
fca_da_subsystem => undef,
scsi_da_subsystem => undef,
condition => $params{condition},
blacklisted => 0,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
$self->{da_subsystem} = HP::Proliant::Component::DiskSubsystem::Da->new(
runtime => $self->{runtime},
rawdata => $self->{rawdata},
method => $self->{method},
);
$self->{sas_subsystem} = HP::Proliant::Component::DiskSubsystem::Sas->new(
runtime => $self->{runtime},
rawdata => $self->{rawdata},
method => $self->{method},
);
$self->{scsi_subsystem} = HP::Proliant::Component::DiskSubsystem::Scsi->new(
runtime => $self->{runtime},
rawdata => $self->{rawdata},
method => $self->{method},
);
$self->{ide_subsystem} = HP::Proliant::Component::DiskSubsystem::Ide->new(
runtime => $self->{runtime},
rawdata => $self->{rawdata},
method => $self->{method},
);
$self->{fca_subsystem} = HP::Proliant::Component::DiskSubsystem::Fca->new(
runtime => $self->{runtime},
rawdata => $self->{rawdata},
method => $self->{method},
);
}
sub check {
my $self = shift;
$self->add_info('checking disk subsystem');
$self->{da_subsystem}->check();
$self->{sas_subsystem}->check();
$self->{scsi_subsystem}->check();
$self->{ide_subsystem}->check();
$self->{fca_subsystem}->check();
$self->disk_summary();
}
sub dump {
my $self = shift;
$self->{da_subsystem}->dump();
$self->{sas_subsystem}->dump();
$self->{scsi_subsystem}->dump();
$self->{ide_subsystem}->dump();
$self->{fca_subsystem}->dump();
}
sub disk_summary {
my $self = shift;
foreach my $subsys (qw(da sas scsi ide fca)) {
if (my $pd = $self->{$subsys.'_subsystem'}->has_physical_drives()) {
my $ld = $self->{$subsys.'_subsystem'}->has_logical_drives();
$self->add_summary(sprintf '%s: %d logical drives, %d physical drives',
$subsys, $ld, $pd);
}
}
}
sub assemble {
my $self = shift;
$self->trace(3, sprintf "%s controllers und platten zusammenf<6E>hren",
ref($self));
$self->trace(3, sprintf "has %d controllers",
scalar(@{$self->{controllers}}));
$self->trace(3, sprintf "has %d accelerators",
scalar(@{$self->{accelerators}})) if exists $self->{accelerators};
$self->trace(3, sprintf "has %d physical_drives",
scalar(@{$self->{physical_drives}}));
$self->trace(3, sprintf "has %d logical_drives",
scalar(@{$self->{logical_drives}}));
$self->trace(3, sprintf "has %d spare_drives",
scalar(@{$self->{spare_drives}}));
my $found = {
accelerators => {},
logical_drives => {},
physical_drives => {},
spare_drives => {},
};
# found->{komponente}->{controllerindex} ist ein array
# von teilen, die zu einem controller gehoeren
foreach my $item (qw(accelerators logical_drives physical_drives spare_drives)) {
foreach (@{$self->{$item}}) {
$found->{item}->{$_->{controllerindex}} = []
unless exists $found->{$item}->{$_->{controllerindex}};
push(@{$found->{$item}->{$_->{controllerindex}}}, $_);
}
}
foreach my $item (qw(accelerators logical_drives physical_drives spare_drives)) {
foreach (@{$self->{controllers}}) {
if (exists $found->{$item}->{$_->{controllerindex}}) {
$_->{$item} = $found->{$item}->{$_->{controllerindex}};
delete $found->{$item}->{$_->{controllerindex}};
} else {
$_->{$item} = []; # z.b. ein leerer controller: physical_drives = []
}
}
}
# was jetzt noch in $found uebrig ist, gehoert zu keinem controller
# d.h. komponenten mit ungueltigen cnrtlindex wurden gefunden
}
sub has_controllers {
my $self = shift;
return scalar(@{$self->{controllers}});
}
sub has_accelerators {
my $self = shift;
return exists $self->{accelerators} ? scalar(@{$self->{accelerators}}) : 0;
}
sub has_physical_drives {
my $self = shift;
return scalar(@{$self->{physical_drives}});
}
sub has_logical_drives {
my $self = shift;
return scalar(@{$self->{logical_drives}});
}
package HP::Proliant::Component;
our @ISA = qw(HP::Proliant);
package HP::Proliant;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
use Data::Dumper;
our @ISA = qw(HP::Server);
sub init {
my $self = shift;
$self->{components} = {
powersupply_subsystem => undef,
fan_subsystem => undef,
temperature_subsystem => undef,
cpu_subsystem => undef,
memory_subsystem => undef,
disk_subsystem => undef,
};
$self->{serial} = 'unknown';
$self->{product} = 'unknown';
$self->{romversion} = 'unknown';
$self->collect();
if (! $self->{runtime}->{plugin}->check_messages() &&
! exists $self->{noinst_hint}) {
$self->set_serial();
$self->check_for_buggy_firmware();
$self->analyze_cpus();
$self->analyze_powersupplies();
$self->analyze_fan_subsystem();
$self->analyze_temperatures();
$self->analyze_memory_subsystem();
$self->analyze_disk_subsystem();
$self->auto_blacklist();
$self->check_cpus();
$self->check_powersupplies();
$self->check_fan_subsystem();
$self->check_temperatures();
$self->check_memory_subsystem();
$self->check_disk_subsystem();
}
}
sub identify {
my $self = shift;
return sprintf "System: '%s', S/N: '%s', ROM: '%s'",
$self->{product}, $self->{serial}, $self->{romversion};
}
sub check_for_buggy_firmware {
my $self = shift;
my @buggyfirmwares = (
"P24 12/11/2001",
"P24 11/15/2002",
"D13 06/03/2003",
"D13 09/15/2004",
"P20 12/17/2002"
);
$self->{runtime}->{options}->{buggy_firmware} =
grep /^$self->{romversion}/, @buggyfirmwares;
}
sub dump {
my $self = shift;
printf STDERR "serial %s\n", $self->{serial};
printf STDERR "product %s\n", $self->{product};
printf STDERR "romversion %s\n", $self->{romversion};
printf STDERR "%s\n", Data::Dumper::Dumper($self->{components});
}
sub analyze_powersupplies {
my $self = shift;
$self->{components}->{powersupply_subsystem} =
HP::Proliant::Component::PowersupplySubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_fan_subsystem {
my $self = shift;
$self->{components}->{fan_subsystem} =
HP::Proliant::Component::FanSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_temperatures {
my $self = shift;
$self->{components}->{temperature_subsystem} =
HP::Proliant::Component::TemperatureSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_cpus {
my $self = shift;
$self->{components}->{cpu_subsystem} =
HP::Proliant::Component::CpuSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_memory_subsystem {
my $self = shift;
$self->{components}->{memory_subsystem} =
HP::Proliant::Component::MemorySubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_disk_subsystem {
my $self = shift;
$self->{components}->{disk_subsystem} =
HP::Proliant::Component::DiskSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub check_cpus {
my $self = shift;
$self->{components}->{cpu_subsystem}->check();
$self->{components}->{cpu_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_powersupplies {
my $self = shift;
$self->{components}->{powersupply_subsystem}->check();
$self->{components}->{powersupply_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_fan_subsystem {
my $self = shift;
$self->{components}->{fan_subsystem}->check();
$self->{components}->{fan_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_temperatures {
my $self = shift;
$self->{components}->{temperature_subsystem}->check();
$self->{components}->{temperature_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_memory_subsystem {
my $self = shift;
$self->{components}->{memory_subsystem}->check();
$self->{components}->{memory_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_disk_subsystem {
my $self = shift;
$self->{components}->{disk_subsystem}->check();
$self->{components}->{disk_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
# zum anhaengen an die normale ausgabe... da: 2 logical drives, 5 physical...
$self->{runtime}->{plugin}->add_message(OK,
$self->{components}->{disk_subsystem}->{summary})
if $self->{components}->{disk_subsystem}->{summary};
}
sub auto_blacklist() {
my $self = shift;
if ($self->{product} =~ /380 g6/) {
# http://bizsupport1.austin.hp.com/bc/docs/support/SupportManual/c01723408/c01723408.pdf seite 19
if ($self->{components}->{cpu_subsystem}->num_cpus() == 1) {
$self->add_blacklist('ff/f:5,6');
}
} elsif ($self->{product} =~ /380 g6/) {
# http://bizsupport1.austin.hp.com/bc/docs/support/SupportManual/c01704762/c01704762.pdf Fan 2 is only required when processor 2 is installed in the server.
}
}
package HP::Proliant::CLI;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
our @ISA = qw(HP::Proliant);
sub collect {
my $self = shift;
my $hpasmcli = undef;
if (($self->{runtime}->{plugin}->opts->hpasmcli) &&
(-f $self->{runtime}->{plugin}->opts->hpasmcli) &&
(! -x $self->{runtime}->{plugin}->opts->hpasmcli)) {
no strict 'refs';
open(BIRK, $self->{runtime}->{plugin}->opts->hpasmcli);
# all output in one file prefixed with server|powersupply|fans|temp|dimm
while(<BIRK>) {
chomp;
$self->{rawdata} .= $_."\n";
}
close BIRK;
# If you run this script and redirect it's output to a file
# you can use it for testing purposes with
# --hpasmcli <output>
# It must not be executable. (chmod 644)
my $diag = <<'EOEO';
hpasmcli=$(which hpasmcli)
hpacucli=$(which hpacucli)
for i in server powersupply fans temp dimm
do
$hpasmcli -s "show $i" | while read line
do
printf "%s %s\n" $i "$line"
done
done
if [ -x "$hpacucli" ]; then
for i in config status
do
$hpacucli ctrl all show $i | while read line
do
printf "%s %s\n" $i "$line"
done
done
fi
EOEO
} else {
#die "exec hpasmcli";
# alles einsammeln und in rawdata stecken
my $hpasmcli = undef;
$hpasmcli = $self->{runtime}->{plugin}->opts->hpasmcli ?
$self->{runtime}->{plugin}->opts->hpasmcli : '/sbin/hpasmcli';
# check if this exists at all
# descend the directory
if ($< != 0) {
$hpasmcli = "sudo ".$hpasmcli;
}
$self->trace(2, sprintf "calling %s\n", $hpasmcli);
$self->check_daemon();
if (! $self->{runtime}->{plugin}->check_messages()) {
$self->check_hpasm_client($hpasmcli);
if (! $self->{runtime}->{plugin}->check_messages()) {
foreach my $component (qw(server fans temp dimm powersupply)) {
if (open HPASMCLI, "$hpasmcli -s \"show $component\"|") {
my @output = <HPASMCLI>;
close HPASMCLI;
$self->{rawdata} .= join("\n", map {
$component.' '.$_;
} @output);
}
}
if ($self->{runtime}->{options}->{hpacucli}) {
#1 oder 0. pfad selber finden
my $hpacucli = undef;
if (-e '/usr/sbin/hpacucli') {
$hpacucli = '/usr/sbin/hpacucli';
} elsif (-e '/usr/local/sbin/hpacucli') {
$hpacucli = '/usr/local/sbin/hpacucli';
} else {
$hpacucli = $hpasmcli;
$hpacucli =~ s/^sudo\s*//;
$hpacucli =~ s/hpasmcli/hpacucli/;
$hpacucli = -e $hpacucli ? $hpacucli : undef;
}
if ($hpacucli) {
if ($< != 0) {
$hpacucli = "sudo ".$hpacucli;
}
$self->trace(2, sprintf "calling %s\n", $hpacucli);
$self->check_hpacu_client($hpacucli);
if (! $self->{runtime}->{plugin}->check_messages()) {
if (open HPACUCLI, "$hpacucli ctrl all show config 2>&1|") {
my @output = <HPACUCLI>;
close HPACUCLI;
$self->{rawdata} .= join("\n", map {
'config '.$_;
} @output);
}
if (open HPACUCLI, "$hpacucli ctrl all show status 2>&1|") {
my @output = <HPACUCLI>;
close HPACUCLI;
$self->{rawdata} .= join("\n", map {
'status '.$_;
} @output);
}
}
} else {
if ($self->{runtime}->{options}->{noinstlevel} eq 'ok') {
$self->add_message(OK,
'hpacucli is not installed. let\'s hope the best...');
} else {
$self->add_message(
uc $self->{runtime}->{options}->{noinstlevel},
'hpacucli is not installed.');
}
}
}
}
}
}
}
sub check_daemon {
my $self = shift;
my $multiproc_os_signatures_files = {
'/etc/SuSE-release' => 'VERSION\s*=\s*8',
'/etc/trustix-release' => '.*',
'/etc/redhat-release' => '.*Pensacola.*',
'/etc/debian_version' => '3\.1',
'/etc/issue' => '.*Kernel 2\.4\.9-vmnix2.*', # VMware ESX Server 2.5.4
};
if (open PS, "/bin/ps -e -ocmd|") {
my $numprocs = 0;
my $numcliprocs = 0;
my @procs = <PS>;
close PS;
$numprocs = grep /hpasm.*d$/, map { (split /\s+/, $_)[0] } @procs;
$numcliprocs = grep /hpasmcli/, grep !/check_hpasm/, @procs;
if (! $numprocs ) {
$self->add_message(CRITICAL, 'hpasmd needs to be restarted');
} elsif ($numprocs > 1) {
my $known = 0;
foreach my $osfile (keys %{$multiproc_os_signatures_files}) {
if (-f $osfile) {
open OSSIG, $osfile;
if (grep /$multiproc_os_signatures_files->{$osfile}/, <OSSIG>) {
$known = 1;
}
close OSSIG;
}
}
if (! $known) {
$self->add_message(UNKNOWN, 'multiple hpasmd procs');
}
}
if ($numcliprocs == 1) {
$self->add_message(UNKNOWN, 'another hpasmdcli is running');
} elsif ($numcliprocs > 1) {
$self->add_message(UNKNOWN, 'hanging hpasmdcli processes');
}
}
}
sub check_hpasm_client {
my $self = shift;
my $hpasmcli = shift;
if (open HPASMCLI, "$hpasmcli -s help 2>&1 |") {
my @output = <HPASMCLI>;
close HPASMCLI;
if (grep /Could not communicate with hpasmd/, @output) {
$self->add_message(CRITICAL, 'hpasmd needs to be restarted');
} elsif (grep /(asswor[dt]:)|(You must be root)/, @output) {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpasmcli);
} elsif (grep /must have a tty/, @output) {
$self->add_message(CRITICAL,
'sudo must be configured with requiretty=no (man sudo)');
} elsif (! grep /CLEAR/, @output) {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpasmcli);
}
} else {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpasmcli);
}
}
sub check_hpacu_client {
my $self = shift;
my $hpacucli = shift;
if (open HPACUCLI, "$hpacucli help 2>&1 |") {
my @output = <HPACUCLI>;
close HPACUCLI;
if (grep /Another instance of hpacucli is running/, @output) {
$self->add_message(UNKNOWN, 'another hpacucli is running');
} elsif (grep /You need to have administrator rights/, @output) {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpacucli);
} elsif (grep /(asswor[dt]:)|(You must be root)/, @output) {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpacucli);
} elsif (! grep /CLI Syntax/, @output) {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpacucli);
}
} else {
$self->add_message(UNKNOWN,
sprintf "insufficient rights to call %s", $hpacucli);
}
}
sub set_serial {
my $self = shift;
foreach (grep(/^server/, split(/\n/, $self->{rawdata}))) {
if (/System\s+:\s+(.*[^\s])/) {
$self->{product} = lc $1;
} elsif (/Serial No\.\s+:\s+(\w+)/) {
$self->{serial} = $1;
} elsif (/ROM version\s+:\s+(.*[^\s])/) {
$self->{romversion} = $1;
}
}
$self->{serial} = $self->{serial};
$self->{product} = lc $self->{product};
$self->{romversion} = $self->{romversion};
}
package HP::Proliant::SNMP;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
our @ISA = qw(HP::Proliant);
sub collect {
my $self = shift;
if ($self->{runtime}->{plugin}->opts->snmpwalk) {
my $cpqSeMibCondition = '1.3.6.1.4.1.232.1.1.3.0'; # 2=ok
my $cpqHeMibCondition = '1.3.6.1.4.1.232.6.1.3.0'; # hat nicht jeder
if ($self->{productname} =~ /4LEE/) {
# rindsarsch!
$self->{rawdata}->{$cpqHeMibCondition} = 0;
}
if (! exists $self->{rawdata}->{$cpqHeMibCondition} &&
! exists $self->{rawdata}->{$cpqSeMibCondition}) { # vlt. geht doch was
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqhlth-mib)');
}
} else {
my $net_snmp_version = Net::SNMP->VERSION(); # 5.002000 or 6.000000
#$params{'-translate'} = [
# -all => 0x0
#];
my ($session, $error) =
Net::SNMP->session(%{$self->{runtime}->{snmpparams}});
if (! defined $session) {
$self->{plugin}->add_message(CRITICAL, 'cannot create session object');
$self->trace(1, Data::Dumper::Dumper($self->{runtime}->{snmpparams}));
} else {
# revMajor is often used for discovery of hp devices
my $cpqHeMibRev = '1.3.6.1.4.1.232.6.1';
my $cpqHeMibRevMajor = '1.3.6.1.4.1.232.6.1.1.0';
my $cpqHeMibCondition = '1.3.6.1.4.1.232.6.1.3.0';
my $result = $session->get_request(
-varbindlist => [$cpqHeMibCondition]
);
if ($self->{productname} =~ /4LEE/) {
# rindsarsch!
$result->{$cpqHeMibCondition} = 0;
}
if (!defined($result) ||
$result->{$cpqHeMibCondition} eq 'noSuchInstance' ||
$result->{$cpqHeMibCondition} eq 'noSuchObject' ||
$result->{$cpqHeMibCondition} eq 'endOfMibView') {
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqhlth-mib)');
$session->close;
} else {
# this is not reliable. many agents return 4=failed
#if ($result->{$cpqHeMibCondition} != 2) {
# $obstacle = "cmapeerstart";
#}
}
}
if (! $self->{runtime}->{plugin}->check_messages()) {
# snmp peer is alive
$self->trace(2, sprintf "Protocol is %s",
$self->{runtime}->{snmpparams}->{'-version'});
my $cpqStdEquipment = "1.3.6.1.4.1.232";
my $cpqSeProcessor = "1.3.6.1.4.1.232.1.2.2";
my $cpqSeRom = "1.3.6.1.4.1.232.1.2.6";
my $cpqHeComponent = "1.3.6.1.4.1.232.6.2";
my $cpqHePWSComponent = "1.3.6.1.4.1.232.6.2.9";
my $cpqHeThermal = "1.3.6.1.4.1.232.6.2.6";
#my $cpqHeFComponent = "1.3.6.1.4.1.232.6.2.6.7";
#my $cpqHeTComponent = "1.3.6.1.4.1.232.6.2.6.8";
my $cpqHeMComponent = "1.3.6.1.4.1.232.6.2.14";
my $cpqDaComponent = "1.3.6.1.4.1.232.3.2";
my $cpqSasComponent = "1.3.6.1.4.1.232.5";
my $cpqIdeComponent = "1.3.6.1.4.1.232.14";
my $cpqFcaComponent = "1.3.6.1.4.1.232.16.2";
my $cpqSiComponent = "1.3.6.1.4.1.232.2.2";
$session->translate;
my $response = {}; #break the walk up in smaller pieces
my $tic = time; my $tac = $tic;
my $response1 = $session->get_table(
-baseoid => $cpqSeProcessor);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSeProcessor (%d oids)",
$tac - $tic, scalar(keys %{$response1}));
# Walk for PowerSupply
$tic = time;
my $response2p = $session->get_table(
-maxrepetitions => 1,
-baseoid => $cpqHePWSComponent);
if (scalar (keys %{$response2p}) == 0) {
$self->trace(2, sprintf "maxrepetitions failed. fallback");
$response2p = $session->get_table(
-baseoid => $cpqHePWSComponent);
}
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqHePWSComponent (%d oids)",
$tac - $tic, scalar(keys %{$response2p}));
# Walk for Fans/Temp/Overall
$tic = time;
my $response2f = $session->get_table(
-maxrepetitions => 1,
-baseoid => $cpqHeThermal);
if (scalar (keys %{$response2f}) == 0) {
$self->trace(2, sprintf "maxrepetitions failed. fallback");
$response2f = $session->get_table(
-baseoid => $cpqHeThermal);
}
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqHeThermal (%d oids)",
$tac - $tic, scalar(keys %{$response2f}));
# # Walk for Fans
# $tic = time;
# my $response2f = $session->get_table(
# -maxrepetitions => 1,
# -baseoid => $cpqHeFComponent);
# if (scalar (keys %{$response2f}) == 0) {
# $self->trace(2, sprintf "maxrepetitions failed. fallback");
# $response2f = $session->get_table(
# -baseoid => $cpqHeFComponent);
# }
# $tac = time;
# $self->trace(2, sprintf "%03d seconds for walk cpqHeFComponent (%d oids)",
# $tac - $tic, scalar(keys %{$response2f}));
# # Walk for Temp
# $tic = time;
# my $response2t = $session->get_table(
# -maxrepetitions => 1,
# -baseoid => $cpqHeTComponent);
# if (scalar (keys %{$response2t}) == 0) {
# $self->trace(2, sprintf "maxrepetitions failed. fallback");
# $response2t = $session->get_table(
# -baseoid => $cpqHeTComponent);
# }
# $tac = time;
# $self->trace(2, sprintf "%03d seconds for walk cpqHeTComponent (%d oids)",
# $tac - $tic, scalar(keys %{$response2t}));
# Walk for Mem
$tic = time;
my $response2m = $session->get_table(
-maxrepetitions => 1,
-baseoid => $cpqHeMComponent);
if (scalar (keys %{$response2m}) == 0) {
$self->trace(2, sprintf "maxrepetitions failed. fallback");
$response2m = $session->get_table(
-baseoid => $cpqHeMComponent);
}
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqHeMComponent (%d oids)",
$tac - $tic, scalar(keys %{$response2m}));
#
$tic = time;
my $response3 = $session->get_table(
-baseoid => $cpqDaComponent);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqDaComponent (%d oids)",
$tac - $tic, scalar(keys %{$response3}));
$tic = time;
my $response4 = $session->get_table(
-baseoid => $cpqSiComponent);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSiComponent (%d oids)",
$tac - $tic, scalar(keys %{$response4}));
$tic = time;
my $response5 = $session->get_table(
-baseoid => $cpqSeRom);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSeRom (%d oids)",
$tac - $tic, scalar(keys %{$response5}));
$tic = time;
my $response6 = $session->get_table(
-baseoid => $cpqSasComponent);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSasComponent (%d oids)",
$tac - $tic, scalar(keys %{$response6}));
$tic = time;
my $response7 = $session->get_table(
-baseoid => $cpqIdeComponent);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqIdeComponent (%d oids)",
$tac - $tic, scalar(keys %{$response7}));
$tic = time;
my $response8 = $session->get_table(
-baseoid => $cpqFcaComponent);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqFcaComponent (%d oids)",
$tac - $tic, scalar(keys %{$response8}));
$tic = time;
$session->close;
map { $response->{$_} = $response1->{$_} } keys %{$response1};
map { $response->{$_} = $response2p->{$_} } keys %{$response2p};
map { $response->{$_} = $response2f->{$_} } keys %{$response2f};
# map { $response->{$_} = $response2t->{$_} } keys %{$response2t};
map { $response->{$_} = $response2m->{$_} } keys %{$response2m};
map { $response->{$_} = $response3->{$_} } keys %{$response3};
map { $response->{$_} = $response4->{$_} } keys %{$response4};
map { $response->{$_} = $response5->{$_} } keys %{$response5};
map { $response->{$_} = $response6->{$_} } keys %{$response6};
map { $response->{$_} = $response7->{$_} } keys %{$response7};
map { $response->{$_} = $response8->{$_} } keys %{$response8};
map { $response->{$_} =~ s/^\s+//; $response->{$_} =~ s/\s+$//; }
keys %$response;
$self->{rawdata} = $response;
}
}
return $self->{runtime}->{plugin}->check_messages();
}
sub set_serial {
my $self = shift;
my $cpqSiSysSerialNum = "1.3.6.1.4.1.232.2.2.2.1.0";
my $cpqSiProductName = "1.3.6.1.4.1.232.2.2.4.2.0";
my $cpqSeSysRomVer = "1.3.6.1.4.1.232.1.2.6.1.0";
$self->{serial} =
SNMP::Utils::get_object($self->{rawdata}, $cpqSiSysSerialNum);
$self->{product} =
SNMP::Utils::get_object($self->{rawdata}, $cpqSiProductName);
$self->{romversion} =
SNMP::Utils::get_object($self->{rawdata}, $cpqSeSysRomVer);
if ($self->{romversion} && $self->{romversion} =~
#/(\d{2}\/\d{2}\/\d{4}).*?([ADP]{1}\d{2}).*/) {
/(\d{2}\/\d{2}\/\d{4}).*?Family.*?([A-Z]{1})(\d+).*/) {
$self->{romversion} = sprintf("%s%02d %s", $2, $3, $1);
} elsif ($self->{romversion} && $self->{romversion} =~
/([ADP]{1}\d{2})\-(\d{2}\/\d{2}\/\d{4})/) {
$self->{romversion} = sprintf("%s %s", $1, $2);
}
if (!$self->{serial} && $self->{romversion}) {
# this probably is a very, very old server.
$self->{serial} = "METHUSALEM";
$self->{runtime}->{scrapiron} = 1;
}
$self->{serial} = $self->{serial};
$self->{product} = lc $self->{product};
$self->{romversion} = $self->{romversion};
$self->{runtime}->{product} = $self->{product};
}
package HP::BladeSystem::Component::CommonEnclosureSubsystem;
our @ISA = qw(HP::BladeSystem::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
common_enclosures => [],
common_enclosure_temps => [],
common_enclosure_fan_subsys => undef,
common_enclosure_fuses => [],
common_enclosure_frus => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
# jeweils ein block fuer
# enclosures, temps, fans, fuses
# loop ueber oids und entspr. new
my $oids = {
cpqRackCommonEnclosureEntry => '1.3.6.1.4.1.232.22.2.3.1.1.1',
cpqRackCommonEnclosureRack => '1.3.6.1.4.1.232.22.2.3.1.1.1.1',
cpqRackCommonEnclosureIndex => '1.3.6.1.4.1.232.22.2.3.1.1.1.2',
cpqRackCommonEnclosureModel => '1.3.6.1.4.1.232.22.2.3.1.1.1.3',
cpqRackCommonEnclosureSerialNum => '1.3.6.1.4.1.232.22.2.3.1.1.1.7',
cpqRackCommonEnclosureFWRev => '1.3.6.1.4.1.232.22.2.3.1.1.1.8',
cpqRackCommonEnclosureName => '1.3.6.1.4.1.232.22.2.3.1.1.1.9',
cpqRackCommonEnclosureCondition => '1.3.6.1.4.1.232.22.2.3.1.1.1.16',
cpqRackCommonEnclosureHasServerBlades => '1.3.6.1.4.1.232.22.2.3.1.1.1.17',
cpqRackCommonEnclosureHasPowerBlades => '1.3.6.1.4.1.232.22.2.3.1.1.1.18',
cpqRackCommonEnclosureHasNetConnectors => '1.3.6.1.4.1.232.22.2.3.1.1.1.19',
cpqRackCommonEnclosureHasTempSensors => '1.3.6.1.4.1.232.22.2.3.1.1.1.20',
cpqRackCommonEnclosureHasFans => '1.3.6.1.4.1.232.22.2.3.1.1.1.21',
cpqRackCommonEnclosureHasFuses => '1.3.6.1.4.1.232.22.2.3.1.1.1.22',
cpqRackCommonEnclosureConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
cpqRackCommonEnclosureHasServerBladesValue => {
1 => 'false',
2 => 'true',
},
};
$oids->{cpqRackCommonEnclosureHasPowerBladesValue} =
$oids->{cpqRackCommonEnclosureHasServerBladesValue};
$oids->{cpqRackCommonEnclosureHasNetConnectorsValue} =
$oids->{cpqRackCommonEnclosureHasServerBladesValue};
$oids->{cpqRackCommonEnclosureHasTempSensorsValue} =
$oids->{cpqRackCommonEnclosureHasServerBladesValue};
$oids->{cpqRackCommonEnclosureHasFansValue} =
$oids->{cpqRackCommonEnclosureHasServerBladesValue};
$oids->{cpqRackCommonEnclosureHasServerBladesValue} =
$oids->{cpqRackCommonEnclosureHasServerBladesValue};
# INDEX { cpqRackCommonEnclosureRack cpqRackCommonEnclosureIndex }
foreach ($self->get_entries($oids, 'cpqRackCommonEnclosureEntry')) {
push(@{$self->{common_enclosures}},
HP::BladeSystem::Component::CommonEnclosureSubsystem::CommonEnclosure->new(%{$_}));
}
$self->{common_enclosure_fan_subsys} = HP::BladeSystem::Component::CommonEnclosureSubsystem::FanSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub check {
my $self = shift;
foreach (@{$self->{common_enclosures}}) {
$_->check();
}
$self->{common_enclosure_fan_subsys}->check();
}
sub dump {
my $self = shift;
foreach (@{$self->{common_enclosures}}) {
$_->dump();
}
$self->{common_enclosure_fan_subsys}->dump();
}
package HP::BladeSystem::Component::CommonEnclosureSubsystem::CommonEnclosure;
our @ISA = qw(HP::BladeSystem::Component::CommonEnclosureSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
cpqRackCommonEnclosureRack => $params{cpqRackCommonEnclosureRack},
cpqRackCommonEnclosureIndex => $params{cpqRackCommonEnclosureIndex},
cpqRackCommonEnclosureModel => $params{cpqRackCommonEnclosureModel},
cpqRackCommonEnclosureSerialNum => $params{cpqRackCommonEnclosureSerialNum},
cpqRackCommonEnclosureFWRev => $params{cpqRackCommonEnclosureFWRev},
cpqRackCommonEnclosureName => $params{cpqRackCommonEnclosureName},
cpqRackCommonEnclosureCondition => $params{cpqRackCommonEnclosureCondition},
cpqRackCommonEnclosureHasServerBlades => $params{cpqRackCommonEnclosureHasServerBlades},
cpqRackCommonEnclosureHasPowerBlades => $params{cpqRackCommonEnclosureHasPowerBlades},
cpqRackCommonEnclosureHasNetConnectors => $params{cpqRackCommonEnclosureHasNetConnectors},
cpqRackCommonEnclosureHasTempSensors => $params{cpqRackCommonEnclosureHasTempSensors},
cpqRackCommonEnclosureHasFans => $params{cpqRackCommonEnclosureHasFans},
cpqRackCommonEnclosureHasFuses => $params{cpqRackCommonEnclosureHasFuses},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
$self->{name} = $self->{cpqRackCommonEnclosureRack}.':'.$self->{cpqRackCommonEnclosureIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('ce', $self->{cpqRackCommonEnclosureName});
my $info = sprintf 'common enclosure %s condition is %s',
$self->{cpqRackCommonEnclosureName}, $self->{cpqRackCommonEnclosureCondition};
$self->add_info($info);
if ($self->{cpqRackCommonEnclosureCondition} eq 'failed') {
$self->add_message(CRITICAL, $info);
} elsif ($self->{cpqRackCommonEnclosureCondition} eq 'degraded') {
$self->add_message(WARNING, $info);
}
}
sub dump {
my $self = shift;
printf "[COMMON_ENCLOSURE_%s]\n", $self->{cpqRackCommonEnclosureName};
foreach (qw(cpqRackCommonEnclosureRack cpqRackCommonEnclosureIndex cpqRackCommonEnclosureModel
cpqRackCommonEnclosureSerialNum cpqRackCommonEnclosureFWRev cpqRackCommonEnclosureName
cpqRackCommonEnclosureCondition cpqRackCommonEnclosureHasServerBlades
cpqRackCommonEnclosureHasPowerBlades cpqRackCommonEnclosureHasNetConnectors
cpqRackCommonEnclosureHasTempSensors cpqRackCommonEnclosureHasFans cpqRackCommonEnclosureHasFuses)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::BladeSystem::Component::CommonEnclosureSubsystem::FanSubsystem;
our @ISA = qw(HP::BladeSystem::Component::CommonEnclosureSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
blacklisted => 0,
fans => [],
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
my $oids = {
cpqRackCommonEnclosureFanEntry => '1.3.6.1.4.1.232.22.2.3.1.3.1',
cpqRackCommonEnclosureFanRack => '1.3.6.1.4.1.232.22.2.3.1.3.1.1',
cpqRackCommonEnclosureFanChassis => '1.3.6.1.4.1.232.22.2.3.1.3.1.2',
cpqRackCommonEnclosureFanIndex => '1.3.6.1.4.1.232.22.2.3.1.3.1.3',
cpqRackCommonEnclosureFanEnclosureName => '1.3.6.1.4.1.232.22.2.3.1.3.1.4',
cpqRackCommonEnclosureFanLocation => '1.3.6.1.4.1.232.22.2.3.1.3.1.5',
cpqRackCommonEnclosureFanPartNumber => '1.3.6.1.4.1.232.22.2.3.1.3.1.6',
cpqRackCommonEnclosureFanSparePartNumber => '1.3.6.1.4.1.232.22.2.3.1.3.1.7',
cpqRackCommonEnclosureFanPresent => '1.3.6.1.4.1.232.22.2.3.1.3.1.8',
cpqRackCommonEnclosureFanRedundant => '1.3.6.1.4.1.232.22.2.3.1.3.1.9',
cpqRackCommonEnclosureFanRedundantGroupId => '1.3.6.1.4.1.232.22.2.3.1.3.1.10',
cpqRackCommonEnclosureFanCondition => '1.3.6.1.4.1.232.22.2.3.1.3.1.11',
cpqRackCommonEnclosureFanEnclosureSerialNum => '1.3.6.1.4.1.232.22.2.3.1.3.1.12',
cpqRackCommonEnclosureFanPresentValue => {
1 => 'other',
2 => 'absent',
3 => 'present',
},
cpqRackCommonEnclosureFanRedundantValue => {
0 => 'other', # meiner phantasie entsprungen, da sich hp nicht aeussert
1 => 'other',
2 => 'notRedundant',
3 => 'redundant',
},
cpqRackCommonEnclosureFanConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
}
};
# INDEX { cpqRackCommonEnclosureFanRack, cpqRackCommonEnclosureFanChassis, cpqRackCommonEnclosureFanIndex }
foreach ($self->get_entries($oids, 'cpqRackCommonEnclosureFanEntry')) {
push(@{$self->{fans}},
HP::BladeSystem::Component::CommonEnclosureSubsystem::FanSubsystem::Fan->new(%{$_}));
}
}
sub check {
my $self = shift;
foreach (@{$self->{fans}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{fans}}) {
$_->dump();
}
}
package HP::BladeSystem::Component::CommonEnclosureSubsystem::FanSubsystem::Fan;
our @ISA = qw(HP::BladeSystem::Component::CommonEnclosureSubsystem::FanSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},cklisted => 0,
#params
cpqRackCommonEnclosureFanEntry => $params{cpqRackCommonEnclosureFanEntry},
cpqRackCommonEnclosureFanRack => $params{cpqRackCommonEnclosureFanRack},
cpqRackCommonEnclosureFanChassis => $params{cpqRackCommonEnclosureFanChassis},
cpqRackCommonEnclosureFanIndex => $params{cpqRackCommonEnclosureFanIndex},
cpqRackCommonEnclosureFanEnclosureName => $params{cpqRackCommonEnclosureFanEnclosureName},
cpqRackCommonEnclosureFanLocation => $params{cpqRackCommonEnclosureFanLocation},
cpqRackCommonEnclosureFanPartNumber => $params{cpqRackCommonEnclosureFanPartNumber},
cpqRackCommonEnclosureFanSparePartNumber => $params{cpqRackCommonEnclosureFanSparePartNumber},
cpqRackCommonEnclosureFanPresent => $params{cpqRackCommonEnclosureFanPresent},
cpqRackCommonEnclosureFanRedundant => $params{cpqRackCommonEnclosureFanRedundant},
cpqRackCommonEnclosureFanRedundantGroupId => $params{cpqRackCommonEnclosureFanRedundantGroupId},
cpqRackCommonEnclosureFanCondition => $params{cpqRackCommonEnclosureFanCondition},
cpqRackCommonEnclosureFanEnclosureSerialNum => $params{cpqRackCommonEnclosureFanEnclosureSerialNum},
info => undef,
extendedinfo => undef,
};
$self->{name} = $self->{cpqRackCommonEnclosureFanRack}.':'.$self->{cpqRackCommonEnclosureFanChassis}.':'.$self->{cpqRackCommonEnclosureFanIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('f', $self->{name});
$self->add_info(sprintf 'fan %s is %s, location is %s, redundance is %s',
$self->{name}, $self->{cpqRackCommonEnclosureFanPresent},
$self->{cpqRackCommonEnclosureFanLocation},
$self->{cpqRackCommonEnclosureFanRedundant});
if ($self->{cpqRackCommonEnclosureFanCondition} eq 'degraded') {
$self->add_message(WARNING, 'fan %s is %s', $self->{name},
$self->{cpqRackCommonEnclosureFanCondition});
} elsif ($self->{cpqRackCommonEnclosureFanCondition} eq 'failed') {
$self->add_message(CRITICAL, 'fan %s is %s', $self->{name},
$self->{cpqRackCommonEnclosureFanCondition});
}
}
sub dump {
my $self = shift;
printf "[FAN_%s]\n", $self->{name};
foreach (qw(cpqRackCommonEnclosureFanRack cpqRackCommonEnclosureFanChassis
cpqRackCommonEnclosureFanIndex cpqRackCommonEnclosureFanEnclosureName
cpqRackCommonEnclosureFanLocation cpqRackCommonEnclosureFanPartNumber
cpqRackCommonEnclosureFanSparePartNumber cpqRackCommonEnclosureFanPresent
cpqRackCommonEnclosureFanRedundant cpqRackCommonEnclosureFanRedundantGroupId
cpqRackCommonEnclosureFanCondition cpqRackCommonEnclosureFanEnclosureSerialNum)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "info: %s\n", $self->{info};
printf "\n";
}
package HP::BladeSystem::Component::PowerEnclosureSubsystem;
our @ISA = qw(HP::BladeSystem::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
power_enclosures => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
# cpqRackPowerEnclosureTable
my $oids = {
cpqRackPowerEnclosureEntry => '1.3.6.1.4.1.232.22.2.3.3.1.1',
cpqRackPowerEnclosureRack => '1.3.6.1.4.1.232.22.2.3.3.1.1.1',
cpqRackPowerEnclosureIndex => '1.3.6.1.4.1.232.22.2.3.3.1.1.2',
cpqRackPowerEnclosureName => '1.3.6.1.4.1.232.22.2.3.3.1.1.3',
cpqRackPowerEnclosureMgmgtBoardSerialNum => '1.3.6.1.4.1.232.22.2.3.3.1.1.4',
cpqRackPowerEnclosureRedundant => '1.3.6.1.4.1.232.22.2.3.3.1.1.5',
cpqRackPowerEnclosureLoadBalanced => '1.3.6.1.4.1.232.22.2.3.3.1.1.6',
cpqRackPowerEnclosureInputPwrType => '1.3.6.1.4.1.232.22.2.3.3.1.1.7',
cpqRackPowerEnclosurePwrFeedMax => '1.3.6.1.4.1.232.22.2.3.3.1.1.8',
cpqRackPowerEnclosureCondition => '1.3.6.1.4.1.232.22.2.3.3.1.1.9',
cpqRackPowerEnclosureRedundantValue => {
1 => 'other',
2 => 'notRedundant',
3 => 'redundant',
},
cpqRackPowerEnclosureLoadBalancedValue => {
0 => 'aechz',
1 => 'other',
2 => 'notLoadBalanced',
3 => 'loadBalanced',
},
cpqRackPowerEnclosureInputPwrTypeValue => {
1 => 'other',
2 => 'singlePhase',
3 => 'threePhase',
4 => 'directCurrent',
},
cpqRackPowerEnclosureConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
},
};
# INDEX { cpqRackPowerEnclosureRack, cpqRackPowerEnclosureIndex }
# dreckada dreck, dreckada
foreach ($self->get_entries($oids, 'cpqRackPowerEnclosureEntry')) {
push(@{$self->{power_enclosures}},
HP::BladeSystem::Component::PowerEnclosureSubsystem::PowerEnclosure->new(%{$_}));
}
}
sub check {
my $self = shift;
foreach (@{$self->{power_enclosures}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{power_enclosures}}) {
$_->dump();
}
}
package HP::BladeSystem::Component::PowerEnclosureSubsystem::PowerEnclosure;
our @ISA = qw(HP::BladeSystem::Component::PowerEnclosureSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
cpqRackPowerEnclosureEntry => $params{cpqRackPowerEnclosureEntry},
cpqRackPowerEnclosureRack => $params{cpqRackPowerEnclosureRack},
cpqRackPowerEnclosureIndex => $params{cpqRackPowerEnclosureIndex},
cpqRackPowerEnclosureName => $params{cpqRackPowerEnclosureName},
cpqRackPowerEnclosureMgmgtBoardSerialNum => $params{cpqRackPowerEnclosureMgmgtBoardSerialNum},
cpqRackPowerEnclosureRedundant => $params{cpqRackPowerEnclosureRedundant},
cpqRackPowerEnclosureLoadBalanced => $params{cpqRackPowerEnclosureLoadBalanced},
cpqRackPowerEnclosureInputPwrType => $params{cpqRackPowerEnclosureInputPwrType},
cpqRackPowerEnclosurePwrFeedMax => $params{cpqRackPowerEnclosurePwrFeedMax},
cpqRackPowerEnclosureCondition => $params{cpqRackPowerEnclosureCondition},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub check {
my $self = shift;
$self->blacklist('pe', $self->{cpqRackPowerEnclosureName});
my $info = sprintf 'power enclosure %s condition is %s',
$self->{cpqRackPowerEnclosureName}, $self->{cpqRackPowerEnclosureCondition};
$self->add_info($info);
if ($self->{cpqRackPowerEnclosureCondition} eq 'degraded') {
$self->add_message(WARNING, $info);
}
}
sub dump {
my $self = shift;
printf "[POWER_ENCLOSURE_%s]\n", $self->{cpqRackPowerEnclosureName};
foreach (qw(cpqRackPowerEnclosureRack cpqRackPowerEnclosureIndex
cpqRackPowerEnclosureName cpqRackPowerEnclosureMgmgtBoardSerialNum
cpqRackPowerEnclosureRedundant cpqRackPowerEnclosureLoadBalanced
cpqRackPowerEnclosureInputPwrType cpqRackPowerEnclosurePwrFeedMax
cpqRackPowerEnclosureCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::BladeSystem::Component::PowerSupplySubsystem;
our @ISA = qw(HP::BladeSystem::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
power_supplies => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
my $oids = {
cpqRackPowerSupplyEntry => '1.3.6.1.4.1.232.22.2.5.1.1.1',
cpqRackPowerSupplyRack => '1.3.6.1.4.1.232.22.2.5.1.1.1.1',
cpqRackPowerSupplyChassis => '1.3.6.1.4.1.232.22.2.5.1.1.1.2',
cpqRackPowerSupplyIndex => '1.3.6.1.4.1.232.22.2.5.1.1.1.3',
cpqRackPowerSupplyEnclosureName => '1.3.6.1.4.1.232.22.2.5.1.1.1.4',
cpqRackPowerSupplyMaxPwrOutput => '1.3.6.1.4.1.232.22.2.5.1.1.1.9',
cpqRackPowerSupplyCurPwrOutput => '1.3.6.1.4.1.232.22.2.5.1.1.1.10',
cpqRackPowerSupplyIntakeTemp => '1.3.6.1.4.1.232.22.2.5.1.1.1.12',
cpqRackPowerSupplyExhaustTemp => '1.3.6.1.4.1.232.22.2.5.1.1.1.13',
cpqRackPowerSupplyStatus => '1.3.6.1.4.1.232.22.2.5.1.1.1.14',
cpqRackPowerSupplySupplyInputLineStatus => '1.3.6.1.4.1.232.22.2.5.1.1.1.15',
cpqRackPowerSupplyPresent => '1.3.6.1.4.1.232.22.2.5.1.1.1.16',
cpqRackPowerSupplyCondition => '1.3.6.1.4.1.232.22.2.5.1.1.1.17',
cpqRackPowerSupplySupplyInputLineStatusValue => {
1 => 'noError',
2 => 'lineOverVoltage',
3 => 'lineUnderVoltage',
4 => 'lineHit',
5 => 'brownOut',
6 => 'linePowerLoss',
},
cpqRackPowerSupplyStatusValue => {
1 => 'noError',
2 => 'generalFailure',
3 => 'bistFailure',
4 => 'fanFailure',
5 => 'tempFailure',
6 => 'interlockOpen',
7 => 'epromFailed',
8 => 'vrefFailed',
9 => 'dacFailed',
10 => 'ramTestFailed',
11 => 'voltageChannelFailed',
12 => 'orringdiodeFailed',
13 => 'brownOut',
14 => 'giveupOnStartup',
15 => 'nvramInvalid',
16 => 'calibrationTableInvalid',
},
cpqRackPowerSupplyPresentValue => {
1 => 'other',
2 => 'absent',
3 => 'present',
},
cpqRackPowerSupplyConditionValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
};
# INDEX { cpqRackPowerSupplyRack, cpqRackPowerSupplyChassis, cpqRackPowerSupplyIndex }
# dreckada dreck, dreckada
foreach ($self->get_entries($oids, 'cpqRackPowerSupplyEntry')) {
push(@{$self->{power_supplies}},
HP::BladeSystem::Component::PowerSupplySubsystem::PowerSupply->new(%{$_}));
}
my $total_power_output = 0;
foreach my $powersupply (@{$self->{power_supplies}}) {
$total_power_output += $powersupply->{cpqRackPowerSupplyCurPwrOutput};
}
$self->{runtime}->{plugin}->add_perfdata(
label => 'total_power_output',
value => $total_power_output,
uom => 'Watts',
);
}
sub check {
my $self = shift;
foreach (@{$self->{power_supplies}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{power_supplies}}) {
$_->dump();
}
}
package HP::BladeSystem::Component::PowerSupplySubsystem::PowerSupply;
our @ISA = qw(HP::BladeSystem::Component::PowerSupplySubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
cpqRackPowerSupplyRack => $params{cpqRackPowerSupplyRack},
cpqRackPowerSupplyChassis => $params{cpqRackPowerSupplyChassis},
cpqRackPowerSupplyIndex => $params{cpqRackPowerSupplyIndex},
cpqRackPowerSupplyEnclosureName => $params{cpqRackPowerSupplyEnclosureName},
cpqRackPowerSupplyMaxPwrOutput => $params{cpqRackPowerSupplyMaxPwrOutput},
cpqRackPowerSupplyCurPwrOutput => $params{cpqRackPowerSupplyCurPwrOutput},
cpqRackPowerSupplyIntakeTemp => $params{cpqRackPowerSupplyIntakeTemp},
cpqRackPowerSupplyExhaustTemp => $params{cpqRackPowerSupplyExhaustTemp},
cpqRackPowerSupplyStatus => $params{cpqRackPowerSupplyStatus},
cpqRackPowerSupplySupplyInputLineStatus => $params{cpqRackPowerSupplySupplyInputLineStatus},
cpqRackPowerSupplyPresent => $params{cpqRackPowerSupplyPresent},
cpqRackPowerSupplyCondition => $params{cpqRackPowerSupplyCondition},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
$self->{name} = $params{cpqRackPowerSupplyRack}.':'.$params{cpqRackPowerSupplyChassis}.':'.$params{cpqRackPowerSupplyIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('ps', $self->{name});
my $info = sprintf 'power supply %s is %s, condition is %s',
$self->{name}, $self->{cpqRackPowerSupplyPresent},
$self->{cpqRackPowerSupplyCondition};
$self->add_info($info) if $self->{cpqRackPowerSupplyPresent} eq 'present' ||
$self->{runtime}->{options}->{verbose} >= 3;
if ($self->{cpqRackPowerSupplyPresent} eq 'present') {
$self->{runtime}->{plugin}->add_perfdata(
label => sprintf('power_supply_output_%s', $self->{name}),
value => $self->{cpqRackPowerSupplyCurPwrOutput},
uom => 'Watts',
);
if ($self->{cpqRackPowerSupplyCondition} eq 'degraded') {
$self->add_message(WARNING, $info);
$self->add_info(sprintf 'power supply %s status is %s, inp.line status is %s',
$self->{cpqRackPowerSupplyName}, $self->{cpqRackPowerSupplyStatus},
$self->{cpqRackPowerSupplySupplyInputLineStatus});
} elsif ($self->{cpqRackPowerSupplyCondition} eq 'failed') {
$self->add_message(CRITICAL, $info);
$self->add_info(sprintf 'power supply %s status is %s, inp.line status is %s',
$self->{cpqRackPowerSupplyName}, $self->{cpqRackPowerSupplyStatus},
$self->{cpqRackPowerSupplySupplyInputLineStatus});
}
}
}
sub dump {
my $self = shift;
printf "[POWER_SUPPLY%s]\n", $self->{name};
foreach (qw(cpqRackPowerSupplyRack cpqRackPowerSupplyChassis cpqRackPowerSupplyIndex cpqRackPowerSupplyEnclosureName cpqRackPowerSupplyMaxPwrOutput cpqRackPowerSupplyCurPwrOutput cpqRackPowerSupplyIntakeTemp cpqRackPowerSupplyExhaustTemp cpqRackPowerSupplyStatus cpqRackPowerSupplySupplyInputLineStatus cpqRackPowerSupplyPresent cpqRackPowerSupplyCondition)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::BladeSystem::Component::NetConnectorSubsystem;
our @ISA = qw(HP::BladeSystem::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
net_connectors => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
my $oids = {
cpqRackNetConnectorEntry => '1.3.6.1.4.1.232.22.2.6.1.1.1',
cpqRackNetConnectorRack => '1.3.6.1.4.1.232.22.2.6.1.1.1.1',
cpqRackNetConnectorChassis => '1.3.6.1.4.1.232.22.2.6.1.1.1.2',
cpqRackNetConnectorIndex => '1.3.6.1.4.1.232.22.2.6.1.1.1.3',
cpqRackNetConnectorEnclosureName => '1.3.6.1.4.1.232.22.2.6.1.1.1.4',
cpqRackNetConnectorName => '1.3.6.1.4.1.232.22.2.6.1.1.1.5',
cpqRackNetConnectorModel => '1.3.6.1.4.1.232.22.2.6.1.1.1.6',
cpqRackNetConnectorSerialNum => '1.3.6.1.4.1.232.22.2.6.1.1.1.7',
cpqRackNetConnectorPartNumber => '1.3.6.1.4.1.232.22.2.6.1.1.1.8',
cpqRackNetConnectorSparePartNumber => '1.3.6.1.4.1.232.22.2.6.1.1.1.9',
cpqRackNetConnectorFWRev => '1.3.6.1.4.1.232.22.2.6.1.1.1.10',
cpqRackNetConnectorType => '1.3.6.1.4.1.232.22.2.6.1.1.1.11',
cpqRackNetConnectorLocation => '1.3.6.1.4.1.232.22.2.6.1.1.1.12',
cpqRackNetConnectorPresent => '1.3.6.1.4.1.232.22.2.6.1.1.1.13',
cpqRackNetConnectorHasFuses => '1.3.6.1.4.1.232.22.2.6.1.1.1.14',
cpqRackNetConnectorEnclosureSerialNum => '1.3.6.1.4.1.232.22.2.6.1.1.1.15',
cpqRackNetConnectorTypeValue => {
0 => 'other', # undefined
1 => 'other',
2 => 'active',
3 => 'passive',
},
cpqRackNetConnectorPresentValue => {
1 => 'other',
2 => 'absent',
3 => 'present',
},
cpqRackNetConnectorHasFusesValue => {
-1 => 'false', # wird geliefert, also vermute ich false
1 => 'false',
2 => 'true',
},
};
# INDEX { cpqRackNetConnectorRack, cpqRackNetConnectorChassis, cpqRackNetConnectorIndex }
# dreckada dreck, dreckada
foreach ($self->get_entries($oids, 'cpqRackNetConnectorEntry')) {
push(@{$self->{net_connectors}},
HP::BladeSystem::Component::NetConnectorSubsystem::NetConnector->new(%{$_}));
}
}
sub check {
my $self = shift;
foreach (@{$self->{net_connectors}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{net_connectors}}) {
$_->dump();
}
}
package HP::BladeSystem::Component::NetConnectorSubsystem::NetConnector;
our @ISA = qw(HP::BladeSystem::Component::NetConnectorSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
cpqRackNetConnectorRack => $params{cpqRackNetConnectorRack},
cpqRackNetConnectorChassis => $params{cpqRackNetConnectorChassis},
cpqRackNetConnectorIndex => $params{cpqRackNetConnectorIndex},
cpqRackNetConnectorEnclosureName => $params{cpqRackNetConnectorEnclosureName},
cpqRackNetConnectorName => $params{cpqRackNetConnectorName},
cpqRackNetConnectorModel => $params{cpqRackNetConnectorModel},
cpqRackNetConnectorSerialNum => $params{cpqRackNetConnectorSerialNum},
cpqRackNetConnectorPartNumber => $params{cpqRackNetConnectorPartNumber},
cpqRackNetConnectorSparePartNumber => $params{cpqRackNetConnectorSparePartNumber},
cpqRackNetConnectorFWRev => $params{cpqRackNetConnectorFWRev},
cpqRackNetConnectorType => $params{cpqRackNetConnectorType},
cpqRackNetConnectorLocation => $params{cpqRackNetConnectorLocation},
cpqRackNetConnectorPresent => $params{cpqRackNetConnectorPresent},
cpqRackNetConnectorHasFuses => $params{cpqRackNetConnectorHasFuses},
cpqRackNetConnectorEnclosureSerialNum => $params{cpqRackNetConnectorEnclosureSerialNum},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
$self->{name} = $params{cpqRackNetConnectorRack}.':'.$params{cpqRackNetConnectorChassis}.':'.$params{cpqRackNetConnectorIndex};
bless $self, $class;
return $self;
}
sub check {
my $self = shift;
$self->blacklist('nc', $self->{name});
my $info = sprintf 'net connector %s is %s, model is %s',
$self->{name}.($self->{cpqRackNetConnectorName} ? '('.$self->{cpqRackNetConnectorName}.')' : ''),
$self->{cpqRackNetConnectorPresent},
$self->{cpqRackNetConnectorModel};
$self->add_info($info) if $self->{cpqRackNetConnectorPresent} eq 'present' ||
$self->{runtime}->{options}->{verbose} >= 3;
# hat weder status noch condition
}
sub dump {
my $self = shift;
printf "[NET_CONNECTOR_%s]\n", $self->{cpqRackNetConnectorName};
foreach (qw(cpqRackNetConnectorRack cpqRackNetConnectorChassis cpqRackNetConnectorIndex cpqRackNetConnectorEnclosureName cpqRackNetConnectorName cpqRackNetConnectorModel cpqRackNetConnectorSerialNum cpqRackNetConnectorPartNumber cpqRackNetConnectorSparePartNumber cpqRackNetConnectorFWRev cpqRackNetConnectorType cpqRackNetConnectorLocation cpqRackNetConnectorPresent cpqRackNetConnectorHasFuses cpqRackNetConnectorEnclosureSerialNum)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::BladeSystem::Component::ServerBladeSubsystem;
our @ISA = qw(HP::BladeSystem::Component);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
server_blades => [],
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub init {
my $self = shift;
my $oids = {
cpqRackServerBladeEntry => '1.3.6.1.4.1.232.22.2.4.1.1.1',
cpqRackServerBladeRack => '1.3.6.1.4.1.232.22.2.4.1.1.1.1',
cpqRackServerBladeChassis => '1.3.6.1.4.1.232.22.2.4.1.1.1.2',
cpqRackServerBladeIndex => '1.3.6.1.4.1.232.22.2.4.1.1.1.3',
cpqRackServerBladeName => '1.3.6.1.4.1.232.22.2.4.1.1.1.4',
cpqRackServerBladeEnclosureName => '1.3.6.1.4.1.232.22.2.4.1.1.1.5',
cpqRackServerBladePartNumber => '1.3.6.1.4.1.232.22.2.4.1.1.1.6',
cpqRackServerBladeSparePartNumber => '1.3.6.1.4.1.232.22.2.4.1.1.1.7',
cpqRackServerBladePosition => '1.3.6.1.4.1.232.22.2.4.1.1.1.8',
cpqRackServerBladeHeight => '1.3.6.1.4.1.232.22.2.4.1.1.1.9',
cpqRackServerBladeWidth => '1.3.6.1.4.1.232.22.2.4.1.1.1.10',
cpqRackServerBladeDepth => '1.3.6.1.4.1.232.22.2.4.1.1.1.11',
cpqRackServerBladePresent => '1.3.6.1.4.1.232.22.2.4.1.1.1.12',
cpqRackServerBladeHasFuses => '1.3.6.1.4.1.232.22.2.4.1.1.1.13',
cpqRackServerBladeEnclosureSerialNum => '1.3.6.1.4.1.232.22.2.4.1.1.1.14',
cpqRackServerBladeSlotsUsed => '1.3.6.1.4.1.232.22.2.4.1.1.1.15',
cpqRackServerBladeStatus => '1.3.6.1.4.1.232.22.2.4.1.1.1.21',
cpqRackServerBladeDiagnosticString => '1.3.6.1.4.1.232.22.2.4.1.1.1.24',
cpqRackServerBladePowered => '1.3.6.1.4.1.232.22.2.4.1.1.1.25',
cpqRackServerBladePOSTStatus => '1.3.6.1.4.1.232.22.2.4.1.1.1.35',
cpqRackServerBladePresentValue => {
1 => 'other',
2 => 'absent',
3 => 'present',
},
cpqRackServerBladeStatusValue => {
1 => 'other',
2 => 'ok',
3 => 'degraded',
4 => 'failed',
},
cpqRackServerBladePoweredValue => {
0 => 'aechz',
1 => 'other',
2 => 'on',
3 => 'off',
4 => 'powerStagedOff',
5 => 'reboot',
},
cpqRackServerBladePOSTStatusValue => {
1 => 'other',
2 => 'started',
3 => 'completed',
4 => 'failed',
},
};
# INDEX { cpqRackServerBladeRack, cpqRackServerBladeChassis, cpqRackServerBladeIndex }
# dreckada dreck, dreckada
foreach ($self->get_entries($oids, 'cpqRackServerBladeEntry')) {
push(@{$self->{server_blades}},
HP::BladeSystem::Component::ServerBladeSubsystem::ServerBlade->new(%{$_}));
}
}
sub check {
my $self = shift;
foreach (@{$self->{server_blades}}) {
$_->check();
}
}
sub dump {
my $self = shift;
foreach (@{$self->{server_blades}}) {
$_->dump();
}
}
package HP::BladeSystem::Component::ServerBladeSubsystem::ServerBlade;
our @ISA = qw(HP::BladeSystem::Component::ServerBladeSubsystem);
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
rawdata => $params{rawdata},
method => $params{method},
cpqRackServerBladeRack => $params{cpqRackServerBladeRack},
cpqRackServerBladeChassis => $params{cpqRackServerBladeChassis},
cpqRackServerBladeIndex => $params{cpqRackServerBladeIndex},
cpqRackServerBladeName => $params{cpqRackServerBladeName},
cpqRackServerBladeEnclosureName => $params{cpqRackServerBladeEnclosureName},
cpqRackServerBladePartNumber => $params{cpqRackServerBladePartNumber},
cpqRackServerBladeSparePartNumber => $params{cpqRackServerBladeSparePartNumber},
cpqRackServerBladePosition => $params{cpqRackServerBladePosition},
cpqRackServerBladeHeight => $params{cpqRackServerBladeHeight},
cpqRackServerBladeWidth => $params{cpqRackServerBladeWidth},
cpqRackServerBladeDepth => $params{cpqRackServerBladeDepth},
cpqRackServerBladePresent => $params{cpqRackServerBladePresent},
cpqRackServerBladeHasFuses => $params{cpqRackServerBladeHasFuses},
cpqRackServerBladeEnclosureSerialNum => $params{cpqRackServerBladeEnclosureSerialNum},
cpqRackServerBladeSlotsUsed => $params{cpqRackServerBladeSlotsUsed},
cpqRackServerBladeStatus => $params{cpqRackServerBladeStatus},
cpqRackServerBladeDiagnosticString => $params{cpqRackServerBladeDiagnosticString} || '',
cpqRackServerBladePowered => $params{cpqRackServerBladePowered},
cpqRackServerBladePOSTStatus => $params{cpqRackServerBladePOSTStatus},
blacklisted => 0,
info => undef,
extendedinfo => undef,
};
bless $self, $class;
$self->init();
return $self;
}
sub check {
my $self = shift;
$self->blacklist('sb', $self->{cpqRackServerBladeName});
my $info = sprintf 'serverblade %s is %s; partnumber=%s; name=%s',
$self->{cpqRackServerBladeIndex}, $self->{cpqRackServerBladePresent},
$self->{cpqRackServerBladePartNumber}, $self->{cpqRackServerBladeName};
$self->add_info($info) if $self->{cpqRackServerBladePresent} eq 'present' ||
$self->{runtime}->{options}->{verbose} >= 3; # absent blades nur bei -vvv
if ($self->{cpqRackServerBladePowered} eq 'on') {
if ($self->{cpqRackServerBladeCondition} eq 'degraded') {
$self->add_message(WARNING, sprintf 'server blade %s diag is \'%s\', post status is %s',
$self->{cpqRackServerBladeName}, $self->{cpqRackServerBladeDiagnosticString},
$self->{cpqRackServerBladePOSTStatus});
} elsif ($self->{cpqRackServerBladeCondition} eq 'failed') {
$self->add_message(CRITICAL, sprintf 'server blade %s diag is \'%s\', post status is %s',
$self->{cpqRackServerBladeName}, $self->{cpqRackServerBladeDiagnosticString},
$self->{cpqRackServerBladePOSTStatus});
}
}
}
sub dump {
my $self = shift;
printf "[SERVER_BLADE_%s]\n", $self->{cpqRackServerBladeName};
foreach (qw(cpqRackServerBladeRack cpqRackServerBladeChassis cpqRackServerBladeIndex cpqRackServerBladeName cpqRackServerBladeEnclosureName cpqRackServerBladePartNumber cpqRackServerBladeSparePartNumber cpqRackServerBladePosition cpqRackServerBladeHeight cpqRackServerBladeWidth cpqRackServerBladeDepth cpqRackServerBladePresent cpqRackServerBladeHasFuses cpqRackServerBladeEnclosureSerialNum cpqRackServerBladeSlotsUsed cpqRackServerBladeStatus cpqRackServerBladeDiagnosticString cpqRackServerBladePowered cpqRackServerBladePOSTStatus)) {
printf "%s: %s\n", $_, $self->{$_};
}
printf "\n";
}
package HP::BladeSystem::Component;
use strict;
our @ISA = qw(HP::BladeSystem);
package HP::BladeSystem;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
use Data::Dumper;
our @ISA = qw(HP::Server HP::Proliant::Component::SNMP);
sub init {
my $self = shift;
$self->{components} = {
common_enclosure_subsystem => undef,
power_enclosure_subsystem => undef,
power_supply_subsystem => undef,
net_connector_subsystem => undef,
server_blade_subsystem => undef,
};
$self->{serial} = 'unknown';
$self->{product} = 'unknown';
$self->{romversion} = 'unknown';
$self->trace(3, 'BladeSystem identified');
$self->collect();
if (! $self->{runtime}->{plugin}->check_messages()) {
$self->set_serial();
$self->analyze_common_enclosures();
$self->analyze_power_enclosures();
$self->analyze_power_supplies();
$self->analyze_net_connectors();
$self->analyze_server_blades();
$self->check_common_enclosures();
$self->check_power_enclosures();
$self->check_power_supplies();
$self->check_net_connectors();
$self->check_server_blades();
}
}
sub identify {
my $self = shift;
return sprintf "System: '%s', S/N: '%s'",
$self->{product}, $self->{serial};
}
sub dump {
my $self = shift;
printf STDERR "serial %s\n", $self->{serial};
printf STDERR "product %s\n", $self->{product};
printf STDERR "romversion %s\n", $self->{romversion};
printf STDERR "%s\n", Data::Dumper::Dumper($self->{enclosures});
}
sub analyze_common_enclosures {
my $self = shift;
$self->{components}->{common_enclosure_subsystem} =
HP::BladeSystem::Component::CommonEnclosureSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_power_enclosures {
my $self = shift;
$self->{components}->{power_enclosure_subsystem} =
HP::BladeSystem::Component::PowerEnclosureSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_power_supplies {
my $self = shift;
$self->{components}->{power_supply_subsystem} =
HP::BladeSystem::Component::PowerSupplySubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_net_connectors {
my $self = shift;
$self->{components}->{net_connector_subsystem} =
HP::BladeSystem::Component::NetConnectorSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_server_blades {
my $self = shift;
$self->{components}->{server_blade_subsystem} =
HP::BladeSystem::Component::ServerBladeSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub check_common_enclosures {
my $self = shift;
$self->{components}->{common_enclosure_subsystem}->check();
$self->{components}->{common_enclosure_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_power_enclosures {
my $self = shift;
$self->{components}->{power_enclosure_subsystem}->check();
$self->{components}->{power_enclosure_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_power_supplies {
my $self = shift;
$self->{components}->{power_supply_subsystem}->check();
$self->{components}->{power_supply_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_net_connectors {
my $self = shift;
$self->{components}->{net_connector_subsystem}->check();
$self->{components}->{net_connector_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_server_blades {
my $self = shift;
$self->{components}->{server_blade_subsystem}->check();
$self->{components}->{server_blade_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub collect {
my $self = shift;
if ($self->{runtime}->{plugin}->opts->snmpwalk) {
my $cpqRackMibCondition = '1.3.6.1.4.1.232.22.1.3.0';
$self->trace(3, 'getting cpqRackMibCondition');
if (! exists $self->{rawdata}->{$cpqRackMibCondition}) {
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqrack-mib)');
}
} else {
my $net_snmp_version = Net::SNMP->VERSION(); # 5.002000 or 6.000000
#$params{'-translate'} = [
# -all => 0x0
#];
my ($session, $error) =
Net::SNMP->session(%{$self->{runtime}->{snmpparams}});
if (! defined $session) {
$self->{plugin}->add_message(CRITICAL, 'cannot create session object');
$self->trace(1, Data::Dumper::Dumper($self->{runtime}->{snmpparams}));
} else {
# revMajor is often used for discovery of hp devices
my $cpqSeMibRev = '1.3.6.1.4.1.232.22.1';
my $cpqSeMibRevMajor = '1.3.6.1.4.1.232.22.1.1.0';
my $cpqRackMibCondition = '1.3.6.1.4.1.232.22.1.3.0';
$self->trace(3, 'getting cpqRackMibCondition');
my $result = $session->get_request(
-varbindlist => [$cpqRackMibCondition]
);
if (!defined($result) ||
$result->{$cpqRackMibCondition} eq 'noSuchInstance' ||
$result->{$cpqRackMibCondition} eq 'noSuchObject' ||
$result->{$cpqRackMibCondition} eq 'endOfMibView') {
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqrack-mib)');
$session->close;
} else {
$self->trace(3, 'getting cpqRackMibCondition done');
}
}
if (! $self->{runtime}->{plugin}->check_messages()) {
# snmp peer is alive
$self->trace(2, sprintf "Protocol is %s",
$self->{runtime}->{snmpparams}->{'-version'});
my $cpqSiComponent = "1.3.6.1.4.1.232.2.2";
my $cpqSiAsset = "1.3.6.1.4.1.232.2.2.2";
my $cpqRackInfo = "1.3.6.1.4.1.232.22";
$session->translate;
my $response = {}; #break the walk up in smaller pieces
my $tic = time; my $tac = $tic;
# Walk for Asset
$tic = time;
my $response0 = $session->get_table(
-maxrepetitions => 1,
-baseoid => $cpqSiComponent);
if (scalar (keys %{$response0}) == 0) {
$self->trace(2, sprintf "maxrepetitions failed. fallback");
$response0 = $session->get_table(
-baseoid => $cpqSiComponent);
}
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSiComponent (%d oids)",
$tac - $tic, scalar(keys %{$response0}));
$tic = time;
my $response1 = $session->get_table(
-maxrepetitions => 1,
-baseoid => $cpqRackInfo);
if (scalar (keys %{$response1}) == 0) {
$self->trace(2, sprintf "maxrepetitions failed. fallback");
$response1 = $session->get_table(
-baseoid => $cpqRackInfo);
}
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqRackInfo (%d oids)",
$tac - $tic, scalar(keys %{$response1}));
$session->close;
map { $response->{$_} = $response0->{$_} } keys %{$response0};
map { $response->{$_} = $response1->{$_} } keys %{$response1};
map { $response->{$_} =~ s/^\s+//; $response->{$_} =~ s/\s+$//; }
keys %$response;
$self->{rawdata} = $response;
}
}
return $self->{runtime}->{plugin}->check_messages();
}
sub set_serial {
my $self = shift;
my $cpqSiSysSerialNum = "1.3.6.1.4.1.232.2.2.2.1.0";
my $cpqSiProductName = "1.3.6.1.4.1.232.2.2.4.2.0";
$self->{serial} =
SNMP::Utils::get_object($self->{rawdata}, $cpqSiSysSerialNum);
$self->{product} =
SNMP::Utils::get_object($self->{rawdata}, $cpqSiProductName);
$self->{serial} = $self->{serial};
$self->{product} = lc $self->{product};
$self->{romversion} = 'unknown';
#####################################################################
$self->{runtime}->{product} = $self->{product};
}
package HP::Storage;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
use Data::Dumper;
our @ISA = qw(HP::Server);
sub init {
my $self = shift;
$self->{components} = {
powersupply_subsystem => undef,
fan_subsystem => undef,
temperature_subsystem => undef,
cpu_subsystem => undef,
memory_subsystem => undef,
disk_subsystem => undef,
};
$self->{serial} = 'unknown';
$self->{product} = 'unknown';
$self->{romversion} = 'unknown';
$self->collect();
if (! $self->{runtime}->{plugin}->check_messages()) {
$self->set_serial();
# $self->check_for_buggy_firmware();
# $self->analyze_cpus();
# $self->analyze_powersupplies();
# $self->analyze_fan_subsystem();
# $self->analyze_temperatures();
# $self->analyze_memory_subsystem();
$self->analyze_disk_subsystem();
# $self->check_cpus();
# $self->check_powersupplies();
# $self->check_fan_subsystem();
# $self->check_temperatures();
# $self->check_memory_subsystem();
$self->check_disk_subsystem();
}
}
sub identify {
my $self = shift;
return sprintf "System: '%s', S/N: '%s', ROM: '%s'",
$self->{product}, $self->{serial}, $self->{romversion};
}
sub check_for_buggy_firmware {
my $self = shift;
my @buggyfirmwares = (
"P24 12/11/2001",
"P24 11/15/2002",
"D13 06/03/2003",
"D13 09/15/2004",
"P20 12/17/2002"
);
$self->{runtime}->{options}->{buggy_firmware} =
grep /^$self->{romversion}/, @buggyfirmwares;
}
sub dump {
my $self = shift;
printf STDERR "serial %s\n", $self->{serial};
printf STDERR "product %s\n", $self->{product};
printf STDERR "romversion %s\n", $self->{romversion};
printf STDERR "%s\n", Data::Dumper::Dumper($self->{components});
}
sub analyze_powersupplies {
my $self = shift;
$self->{components}->{powersupply_subsystem} =
HP::Storage::Component::PowersupplySubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_fan_subsystem {
my $self = shift;
$self->{components}->{fan_subsystem} =
HP::Storage::Component::FanSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_temperatures {
my $self = shift;
$self->{components}->{temperature_subsystem} =
HP::Storage::Component::TemperatureSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_cpus {
my $self = shift;
$self->{components}->{cpu_subsystem} =
HP::Storage::Component::CpuSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_memory_subsystem {
my $self = shift;
$self->{components}->{memory_subsystem} =
HP::Storage::Component::MemorySubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub analyze_disk_subsystem {
my $self = shift;
$self->{components}->{disk_subsystem} =
HP::Proliant::Component::DiskSubsystem->new(
rawdata => $self->{rawdata},
method => $self->{method},
runtime => $self->{runtime},
);
}
sub check_cpus {
my $self = shift;
$self->{components}->{cpu_subsystem}->check();
$self->{components}->{cpu_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_powersupplies {
my $self = shift;
$self->{components}->{powersupply_subsystem}->check();
$self->{components}->{powersupply_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_fan_subsystem {
my $self = shift;
$self->{components}->{fan_subsystem}->check();
$self->{components}->{fan_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_temperatures {
my $self = shift;
$self->{components}->{temperature_subsystem}->check();
$self->{components}->{temperature_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_memory_subsystem {
my $self = shift;
$self->{components}->{memory_subsystem}->check();
$self->{components}->{memory_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 2;
}
sub check_disk_subsystem {
my $self = shift;
$self->{components}->{disk_subsystem}->check();
$self->{components}->{disk_subsystem}->dump()
if $self->{runtime}->{options}->{verbose} >= 1;
}
sub collect {
my $self = shift;
if ($self->{runtime}->{plugin}->opts->snmpwalk) {
my $cpqSeMibCondition = '1.3.6.1.4.1.232.6.1.3.0';
# rindsarsch!
$self->{rawdata}->{$cpqSeMibCondition} = 0;
if (! exists $self->{rawdata}->{$cpqSeMibCondition}) {
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqhlth-mib)');
}
} else {
my $net_snmp_version = Net::SNMP->VERSION(); # 5.002000 or 6.000000
#$params{'-translate'} = [
# -all => 0x0
#];
my ($session, $error) =
Net::SNMP->session(%{$self->{runtime}->{snmpparams}});
if (! defined $session) {
$self->{plugin}->add_message(CRITICAL, 'cannot create session object');
$self->trace(1, Data::Dumper::Dumper($self->{runtime}->{snmpparams}));
} else {
# revMajor is often used for discovery of hp devices
my $cpqSeMibRev = '1.3.6.1.4.1.232.6.1';
my $cpqSeMibRevMajor = '1.3.6.1.4.1.232.6.1.1.0';
my $cpqSeMibCondition = '1.3.6.1.4.1.232.6.1.3.0';
my $result = $session->get_request(
-varbindlist => [$cpqSeMibCondition]
);
# rindsarsch!
$result->{$cpqSeMibCondition} = 0;
if (!defined($result) ||
$result->{$cpqSeMibCondition} eq 'noSuchInstance' ||
$result->{$cpqSeMibCondition} eq 'noSuchObject' ||
$result->{$cpqSeMibCondition} eq 'endOfMibView') {
$self->add_message(CRITICAL,
'snmpwalk returns no health data (cpqhlth-mib)');
$session->close;
} else {
# this is not reliable. many agents return 4=failed
#if ($result->{$cpqSeMibCondition} != 2) {
# $obstacle = "cmapeerstart";
#}
}
}
if (! $self->{runtime}->{plugin}->check_messages()) {
# snmp peer is alive
$self->trace(2, sprintf "Protocol is %s",
$self->{runtime}->{snmpparams}->{'-version'});
my $cpqSsSys = "1.3.6.1.4.1.232.8";
$session->translate;
my $response = {}; #break the walk up in smaller pieces
my $tic = time; my $tac = $tic;
my $response1 = $session->get_table(
-baseoid => $cpqSsSys);
$tac = time;
$self->trace(2, sprintf "%03d seconds for walk cpqSsSys (%d oids)",
$tac - $tic, scalar(keys %{$response1}));
$session->close;
map { $response->{$_} = $response1->{$_} } keys %{$response1};
map { $response->{$_} =~ s/^\s+//; $response->{$_} =~ s/\s+$//; }
keys %$response;
$self->{rawdata} = $response;
}
}
return $self->{runtime}->{plugin}->check_messages();
}
sub set_serial {
my $self = shift;
my $snmpwalk = $self->{rawdata};
my @serials = ();
my @models = ();
my @fws = ();
my $cpqSsBackplaneEntry = '1.3.6.1.4.1.232.8.2.2.6.1';
my $cpqSsBackplaneFWRev = '1.3.6.1.4.1.232.8.2.2.6.1.3';
my $cpqSsBackplaneModel = '1.3.6.1.4.1.232.8.2.2.6.1.9';
my $cpqSsBackplaneSerialNumber = '1.3.6.1.4.1.232.8.2.2.6.1.13';
# INDEX { cpqSsBackplaneChassisIndex, cpqSsBackplaneIndex }
my @indexes = SNMP::Utils::get_indices($snmpwalk,
$cpqSsBackplaneEntry);
foreach (@indexes) {
my($idx1, $idx2) = ($_->[0], $_->[1]);
my $fw = SNMP::Utils::get_object($snmpwalk,
$cpqSsBackplaneFWRev, $idx1, $idx2);
my $model = SNMP::Utils::get_object($snmpwalk,
$cpqSsBackplaneModel, $idx1, $idx2);
my $serial = SNMP::Utils::get_object($snmpwalk,
$cpqSsBackplaneSerialNumber, $idx1, $idx2);
push(@serials, $serial);
push(@models, $model);
push(@fws, $fw);
}
$self->{serial} = join('/', @serials);
$self->{product} = join('/', @models);
$self->{romversion} = join('/', @fws);
$self->{runtime}->{product} = $self->{product};
}
package HP::Server;
use strict;
use constant { OK => 0, WARNING => 1, CRITICAL => 2, UNKNOWN => 3 };
sub new {
my $class = shift;
my %params = @_;
my $self = {
runtime => $params{runtime},
productname => 'unknown',
};
bless $self, $class;
if (! ($self->{runtime}->{plugin}->opts->hostname ||
$self->{runtime}->{plugin}->opts->snmpwalk)) {
bless $self, 'HP::Proliant::CLI';
$self->{method} = 'cli';
} else {
$self->check_snmp_and_model();
if ($self->{runtime}->{options}->{servertype}) {
$self->{productname} = 'ProLiant' if
$self->{runtime}->{options}->{servertype} eq 'proliant';
$self->{productname} = 'BladeSystem' if
$self->{runtime}->{options}->{servertype} eq 'bladesystem';
$self->{productname} = 'Storage' if
$self->{runtime}->{options}->{servertype} eq 'storage';
}
if (! $self->{runtime}->{plugin}->check_messages()) {
if ($self->{productname} =~ /ProLiant/) {
bless $self, 'HP::Proliant::SNMP';
$self->trace(3, 'using HP::Proliant::SNMP');
} elsif ($self->{productname} =~ /OpenView .* appliance/) {
bless $self, 'HP::Proliant::SNMP';
$self->trace(3, 'using HP::Proliant::SNMP');
} elsif ($self->{productname} =~ /BladeSystem/) {
bless $self, 'HP::BladeSystem';
$self->trace(3, 'using HP::BladeSystem');
} elsif ($self->{productname} =~ /PROLIANT 4LEE/) {
bless $self, 'HP::Storage';
$self->trace(3, 'using HP::Storage');
} elsif ($self->{productname} =~ /Storage/) { # fake
bless $self, 'HP::Storage';
$self->trace(3, 'using HP::Storage');
} else {
$self->add_message(CRITICAL,
sprintf('unknown device%s', $self->{productname} eq 'unknown' ?
'' : '('.$self->{productname}.')'));
}
$self->{method} = 'snmp';
}
}
if ($self->{runtime}->{options}->{blacklist} &&
-f $self->{runtime}->{options}->{blacklist}) {
$self->{runtime}->{options}->{blacklist} = do {
local (@ARGV, $/) = $self->{runtime}->{options}->{blacklist}; <> };
}
return $self;
}
sub check_snmp_and_model {
# uptime pruefen
# dann whoami
my $self = shift;
if ($self->{runtime}->{plugin}->opts->snmpwalk) {
my $response = {};
if (! -f $self->{runtime}->{plugin}->opts->snmpwalk) {
$self->{runtime}->{plugin}->add_message(CRITICAL,
sprintf 'file %s not found',
$self->{runtime}->{plugin}->opts->snmpwalk);
} elsif (-x $self->{runtime}->{plugin}->opts->snmpwalk) {
my $cmd = sprintf "%s -On -v%s -c%s %s 1.3.6.1.4.1.232 2>&1",
$self->{runtime}->{plugin}->opts->snmpwalk,
$self->{runtime}->{plugin}->opts->protocol,
$self->{runtime}->{plugin}->opts->community,
$self->{runtime}->{plugin}->opts->hostname;
open(WALK, "$cmd |");
while (<WALK>) {
if (/^.*?\.(232\.[\d\.]+) = .*?: (\-*\d+)/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
} elsif (/^.*?\.(232\.[\d\.]+) = .*?: "(.*?)"/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
$response->{'1.3.6.1.4.1.'.$1} =~ s/\s+$//;
}
}
close WALK;
} else {
open(MESS, $self->{runtime}->{plugin}->opts->snmpwalk);
while(<MESS>) {
# SNMPv2-SMI::enterprises.232.6.2.6.7.1.3.1.4 = INTEGER: 6
if (/^.*?\.(232\.[\d\.]+) = .*?: (\-*\d+)/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
} elsif (/^.*?\.(232\.[\d\.]+) = .*?: "(.*?)"/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
$response->{'1.3.6.1.4.1.'.$1} =~ s/\s+$//;
} if (/^.*?\.(232\.[\d\.]+) = (\-*\d+)/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
} elsif (/^.*?\.(232\.[\d\.]+) = "(.*?)"/) {
$response->{'1.3.6.1.4.1.'.$1} = $2;
$response->{'1.3.6.1.4.1.'.$1} =~ s/\s+$//;
}
}
close MESS;
}
map { $response->{$_} =~ s/^\s+//; $response->{$_} =~ s/\s+$//; }
keys %$response;
$self->{rawdata} = $response;
$self->whoami();
} else {
if (eval "require Net::SNMP") {
my %params = ();
my $net_snmp_version = Net::SNMP->VERSION(); # 5.002000 or 6.000000
#$params{'-translate'} = [
# -all => 0x0
#];
$params{'-hostname'} = $self->{runtime}->{plugin}->opts->hostname;
$params{'-version'} = $self->{runtime}->{plugin}->opts->protocol;
if ($self->{runtime}->{plugin}->opts->port) {
$params{'-port'} = $self->{runtime}->{plugin}->opts->port;
}
if ($self->{runtime}->{plugin}->opts->protocol eq '3') {
$params{'-username'} = $self->{runtime}->{plugin}->opts->username;
if ($self->{runtime}->{plugin}->opts->authpassword) {
$params{'-authpassword'} = $self->{runtime}->{plugin}->opts->authpassword;
}
if ($self->{runtime}->{plugin}->opts->authprotocol) {
$params{'-authprotocol'} = $self->{runtime}->{plugin}->opts->authprotocol;
}
if ($self->{runtime}->{plugin}->opts->privpassword) {
$params{'-privpassword'} = $self->{runtime}->{plugin}->opts->privpassword;
}
if ($self->{runtime}->{plugin}->opts->privprotocol) {
$params{'-privprotocol'} = $self->{runtime}->{plugin}->opts->privprotocol;
}
} else {
$params{'-community'} = $self->{runtime}->{plugin}->opts->community;
}
$self->{runtime}->{snmpparams} = \%params;
my ($session, $error) = Net::SNMP->session(%params);
$self->{session} = $session;
if (! defined $session) {
$self->{plugin}->add_message(CRITICAL, 'cannot create session object');
$self->trace(1, Data::Dumper::Dumper(\%params));
} else {
my $sysUpTime = '1.3.6.1.2.1.1.3.0';
my $result = $session->get_request(
-varbindlist => [$sysUpTime]
);
if (!defined($result)) {
$self->add_message(CRITICAL,
'could not contact snmp agent');
$session->close;
} else {
$self->trace(3, 'snmp agent answered');
$self->whoami();
}
}
} else {
$self->add_message(CRITICAL,
'could not find Net::SNMP module');
}
}
}
sub whoami {
my $self = shift;
my $productname = undef;
if ($self->{runtime}->{plugin}->opts->snmpwalk) {
my $cpqSiProductName = '1.3.6.1.4.1.232.2.2.4.2.0';
my $cpqSsMibRevMajor = '1.3.6.1.4.1.232.8.1.1.0';
my $cpqSsBackplaneModel = '1.3.6.1.4.1.232.8.2.2.6.1.9'.'.1.1';
if ($productname = $self->{rawdata}->{$cpqSiProductName}) {
if (! $productname) {
$self->{productname} = 'ProLiant';
} else {
$self->{productname} = $self->{rawdata}->{$cpqSiProductName};
}
} elsif (exists $self->{rawdata}->{$cpqSsBackplaneModel}) {
$self->{productname} = $self->{rawdata}->{$cpqSsBackplaneModel};
} elsif (exists $self->{rawdata}->{$cpqSsMibRevMajor}) {
# at least there is a CPQSTSYS-MIB
$self->{productname} = 'Storage'
} else {
$self->add_message(CRITICAL,
'snmpwalk returns no product name (cpqsinfo-mib)');
}
} else {
my $cpqSiProductName = '1.3.6.1.4.1.232.2.2.4.2.0';
my $cpqSsMibRevMajor = '1.3.6.1.4.1.232.8.1.1.0';
my $cpqSsBackplaneModel = '1.3.6.1.4.1.232.8.2.2.6.1.9'.'.1.1';
my $dummy = '1.3.6.1.2.1.1.5.0';
if ($productname = $self->valid_response($cpqSiProductName)) {
if ($productname eq '') {
$self->{productname} = 'ProLiant';
} else {
$self->{productname} = $productname;
}
} elsif ($productname = $self->valid_response($cpqSsBackplaneModel)) {
$self->{productname} = $productname;
} elsif ($self->valid_response($cpqSsMibRevMajor)) {
# at least there is a CPQSTSYS-MIB
$self->{productname} = 'Storage'
} else {
$self->add_message(CRITICAL,
'snmpwalk returns no product name (cpqsinfo-mib)');
$self->{session}->close;
}
$self->trace(3, 'whoami: '.$self->{productname});
}
}
sub valid_response {
my $self = shift;
my $oid = shift;
my $result = $self->{session}->get_request(
-varbindlist => [$oid]
);
if (!defined($result) ||
! defined $result->{$oid} ||
$result->{$oid} eq 'noSuchInstance' ||
$result->{$oid} eq 'noSuchObject' ||
$result->{$oid} eq 'endOfMibView') {
return undef;
} else {
return $result->{$oid};
}
}
sub trace {
my $self = shift;
my $level = shift;
my $message = shift;
if ($self->{runtime}->{options}->{verbose} >= $level) {
printf "%s\n", $message;
}
}
sub blacklist {
my $self = shift;
my $type = shift;
my $name = shift;
$self->{blacklisted} = $self->is_blacklisted($type, $name);
}
sub add_blacklist {
my $self = shift;
my $list = shift;
$self->{runtime}->{options}->{blacklist} = join('/',
(split('/', $self->{runtime}->{options}->{blacklist}), $list));
}
sub is_blacklisted {
my $self = shift;
my $type = shift;
my $name = shift;
my $blacklisted = 0;
# $name =~ s/\:/-/g;
foreach my $bl_items (split(/\//, $self->{runtime}->{options}->{blacklist})) {
if ($bl_items =~ /^(\w+):([\:\d\-,]+)$/) {
my $bl_type = $1;
my $bl_names = $2;
foreach my $bl_name (split(/,/, $bl_names)) {
if ($bl_type eq $type && $bl_name eq $name) {
$blacklisted = 1;
}
}
} elsif ($bl_items =~ /^(\w+)$/) {
my $bl_type = $1;
if ($bl_type eq $type) {
$blacklisted = 1;
}
}
}
return $blacklisted;
}
sub add_message {
my $self = shift;
my $level = shift;
my $message = shift;
$self->{runtime}->{plugin}->add_message($level, $message)
unless $self->{blacklisted};
}
sub add_info {
my $self = shift;
my $info = shift;
$info = $self->{blacklisted} ? $info.' (blacklisted)' : $info;
$self->{info} = $info;
if (! exists $self->{runtime}->{plugin}->{info}) {
$self->{runtime}->{plugin}->{info} = [];
}
push(@{$self->{runtime}->{plugin}->{info}}, $info);
}
sub annotate_info {
my $self = shift;
my $annotation = shift;
my $lastinfo = pop(@{$self->{runtime}->{plugin}->{info}});
$lastinfo .= sprintf ' (%s)', $annotation;
push(@{$self->{runtime}->{plugin}->{info}}, $lastinfo);
}
sub add_extendedinfo {
my $self = shift;
my $info = shift;
$self->{extendedinfo} = $info;
return if ! $self->{runtime}->{options}->{extendedinfo};
if (! exists $self->{runtime}->{plugin}->{extendedinfo}) {
$self->{runtime}->{plugin}->{extendedinfo} = [];
}
push(@{$self->{runtime}->{plugin}->{extendedinfo}}, $info);
}
sub get_extendedinfo {
my $self = shift;
if (! exists $self->{runtime}->{plugin}->{extendedinfo}) {
$self->{runtime}->{plugin}->{extendedinfo} = [];
}
return join(' ', @{$self->{runtime}->{plugin}->{extendedinfo}});
}
sub add_summary {
my $self = shift;
my $summary = shift;
if (! exists $self->{runtime}->{plugin}->{summary}) {
$self->{runtime}->{plugin}->{summary} = [];
}
push(@{$self->{runtime}->{plugin}->{summary}}, $summary);
}
sub get_summary {
my $self = shift;
if (! exists $self->{runtime}->{plugin}->{summary}) {
$self->{runtime}->{plugin}->{summary} = [];
}
return join(', ', @{$self->{runtime}->{plugin}->{summary}});
}
sub dumper {
my $self = shift;
my $object = shift;
my $run = $object->{runtime};
delete $object->{runtime};
printf STDERR "%s\n", Data::Dumper::Dumper($object);
$object->{runtime} = $run;
}
package main;
#! /usr/bin/perl
use strict;
my $CELSIUS = 1;
my $PERFDATA = 1;
my $EXTENDEDINFO = 1;
my $HWINFO = 1;
my $HPACUCLI = 0;
my $NOINSTLEVEL = 'unknown';
use constant OK => 0;
use constant WARNING => 1;
use constant CRITICAL => 2;
use constant UNKNOWN => 3;
use constant DEPENDENT => 4;
my $plugin = Nagios::MiniPlugin->new(
shortname => '',
usage => 'Usage: %s [ -v|--verbose ] [ -t <timeout> ] '.
'--hostname <proliant> --community <snmp-community>'.
' ...]',
version => '4.1.2',
blurb => 'This plugin checks the hardware of hp/compaq proliant servers',
url => 'http://labs.consol.de/nagios/check_hpasm',
timeout => 60,
shortname => '',
);
$plugin->add_arg(
spec => 'blacklist|b=s',
help => '--blacklist
Blacklist some (missing/failed) components',
required => 0,
default => '',
);
$plugin->add_arg(
spec => 'ignore-dimms|i',
help => '--ignore-dimms
Ignore "N/A"-DIMM status on misc. servers (e.g. older DL320)',
required => 0,
);
$plugin->add_arg(
spec => 'ignore-fan-redundancy',
help => '--ignore-fan-redundancy
Ignore missing redundancy partners',
required => 0,
);
$plugin->add_arg(
spec => 'customthresholds|c=s',
help => '--customthresholds
Use custom thresholds for certain temperatures',
required => 0,
);
$plugin->add_arg(
spec => 'perfdata=s',
help => '--perfdata=[short]
Output performance data. If your performance data string becomes
too long and is truncated by Nagios, then you can use --perfdata=short
instead. This will output temperature tags without location information',
required => 0,
);
$plugin->add_arg(
spec => 'hostname|H=s',
help => '--hostname
Hostname or IP-address of the server (SNMP mode only)',
required => 0,
);
$plugin->add_arg(
spec => 'port=i',
help => '--port
The SNMP port to use (default: 161)',
required => 0,
default => 161,
);
$plugin->add_arg(
spec => 'protocol|P=s',
help => '--protocol
The SNMP protocol to use (default: 2c, other possibilities: 1,3)',
required => 0,
default => '2c',
);
$plugin->add_arg(
spec => 'community|C=s',
help => '--community
SNMP community of the server (SNMP v1/2 only)',
required => 0,
default => 'public',
);
$plugin->add_arg(
spec => 'username=s',
help => '--username
The securityName for the USM security model (SNMPv3 only)',
required => 0,
);
$plugin->add_arg(
spec => 'authpassword=s',
help => '--authpassword
The authentication password for SNMPv3',
required => 0,
);
$plugin->add_arg(
spec => 'authprotocol=s',
help => '--authprotocol
The authentication protocol for SNMPv3 (md5|sha)',
required => 0,
);
$plugin->add_arg(
spec => 'privpassword=s',
help => '--privpassword
The password for authPriv security level',
required => 0,
);
$plugin->add_arg(
spec => 'privprotocol=s',
help => '--privprotocol
The private protocol for SNMPv3 (des|aes|aes128|3des|3desde)',
required => 0,
);
$plugin->add_arg(
spec => 'snmpwalk=s',
help => '--snmpwalk
A file with the output of snmpwalk 1.3.6.1.4.1.232',
required => 0,
);
$plugin->add_arg(
spec => 'hpasmcli=s',
help => '--hpasmcli
A file with the output of hpasmcli',
required => 0,
);
$plugin->add_arg(
spec => 'servertype=s',
help => '--servertype
The type of the server: proliant (default) or bladecenter',
required => 0,
);
$plugin->getopts();
if (! $PERFDATA && $plugin->opts->get('perfdata')) {
$PERFDATA = 1;
}
if ($PERFDATA && $plugin->opts->get('perfdata') &&
($plugin->opts->get('perfdata') eq 'short')) {
$PERFDATA = 2;
}
$plugin->{messages}->{unknown} = []; # wg. add_message(UNKNOWN,...)
$plugin->{info} = []; # gefrickel
my $server = HP::Server->new( runtime => {
plugin => $plugin,
options => {
servertype => $plugin->opts->get('servertype'),
verbose => $plugin->opts->get('verbose'),
scrapiron => 0,
ignore_fan_redundancy => $plugin->opts->get('ignore-fan-redundancy'),
ignore_dimms => $plugin->opts->get('ignore-dimms'),
customthresholds => $plugin->opts->get('customthresholds'),
blacklist => $plugin->opts->get('blacklist'),
celsius => $CELSIUS,
perfdata => $PERFDATA,
extendedinfo => $EXTENDEDINFO,
hwinfo => $HWINFO,
hpacucli => $HPACUCLI,
noinstlevel => $NOINSTLEVEL,
},
},);
if (! $plugin->check_messages()) {
$server->init();
$plugin->add_message(OK, $server->identify()) if $HWINFO;
if (! $plugin->check_messages()) {
$plugin->add_message(OK, 'hardware working fine');
$plugin->add_message(OK, $server->get_summary())
if $server->get_summary();
$plugin->add_message(OK, $server->get_extendedinfo())
if $server->get_extendedinfo();
}
} else {
$plugin->add_message(CRITICAL, 'wrong device');
}
my ($code, $message) = $plugin->check_messages(join => ', ', join_all => ', ');
$message .= sprintf "\n%s\n", join("\n", @{$plugin->{info}})
if $plugin->opts->get('verbose') >= 1;
#printf "%s\n", Data::Dumper::Dumper($plugin->{info});
$plugin->nagios_exit($code, $message);