From f672ade530608654e59c5cde05e5056426087c08 Mon Sep 17 00:00:00 2001 From: qgarnier Date: Mon, 26 Jun 2017 15:05:51 +0200 Subject: [PATCH] Fix #350 --- network/cisco/asa/{ => snmp}/mode/failover.pm | 4 +- network/cisco/asa/{ => snmp}/plugin.pm | 5 +- snmp_standard/mode/entity.pm | 232 ++++++++++++++++++ 3 files changed, 237 insertions(+), 4 deletions(-) rename network/cisco/asa/{ => snmp}/mode/failover.pm (99%) rename network/cisco/asa/{ => snmp}/plugin.pm (92%) create mode 100644 snmp_standard/mode/entity.pm diff --git a/network/cisco/asa/mode/failover.pm b/network/cisco/asa/snmp/mode/failover.pm similarity index 99% rename from network/cisco/asa/mode/failover.pm rename to network/cisco/asa/snmp/mode/failover.pm index 2fef70077..287458c5f 100644 --- a/network/cisco/asa/mode/failover.pm +++ b/network/cisco/asa/snmp/mode/failover.pm @@ -18,7 +18,7 @@ # limitations under the License. # -package network::cisco::asa::mode::failover; +package network::cisco::asa::snmp::mode::failover; use base qw(centreon::plugins::templates::counter); @@ -211,4 +211,4 @@ Can be: 'active-units'. =back =cut - \ No newline at end of file + diff --git a/network/cisco/asa/plugin.pm b/network/cisco/asa/snmp/plugin.pm similarity index 92% rename from network/cisco/asa/plugin.pm rename to network/cisco/asa/snmp/plugin.pm index e86070999..d6dc6df1a 100644 --- a/network/cisco/asa/plugin.pm +++ b/network/cisco/asa/snmp/plugin.pm @@ -18,7 +18,7 @@ # limitations under the License. # -package network::cisco::asa::plugin; +package network::cisco::asa::snmp::plugin; use strict; use warnings; @@ -32,11 +32,12 @@ sub new { $self->{version} = '1.0'; %{$self->{modes}} = ( 'cpu' => 'centreon::common::cisco::standard::snmp::mode::cpu', - 'failover' => 'network::cisco::asa::mode::failover', + 'failover' => 'network::cisco::asa::snmp::mode::failover', 'interfaces' => 'snmp_standard::mode::interfaces', 'ipsec-tunnel' => 'centreon::common::cisco::standard::snmp::mode::ipsectunnel', 'list-interfaces' => 'snmp_standard::mode::listinterfaces', 'memory' => 'centreon::common::cisco::standard::snmp::mode::memory', + 'sensors' => 'snmp_standard::mode::entity', 'sessions' => 'centreon::common::cisco::standard::snmp::mode::sessions', ); diff --git a/snmp_standard/mode/entity.pm b/snmp_standard/mode/entity.pm new file mode 100644 index 000000000..de889ac4f --- /dev/null +++ b/snmp_standard/mode/entity.pm @@ -0,0 +1,232 @@ +# +# Copyright 2017 Centreon (http://www.centreon.com/) +# +# Centreon is a full-fledged industry-strength solution that meets +# the needs in IT infrastructure and application monitoring for +# service performance. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +package snmp_standard::mode::entity; + +use base qw(centreon::plugins::templates::hardware); + +use strict; +use warnings; + +sub set_system { + my ($self, %options) = @_; + + $self->{regexp_threshold_overload_check_section_option} = '^(sensor\..*)$'; + $self->{regexp_threshold_numeric_check_section_option} = '^(sensor\..*)$'; + + $self->{cb_hook2} = 'snmp_execute'; + + $self->{thresholds} = { + sensor => [ + ['unavailable', 'OK'], + ['ok', 'OK'], + ['nonoperational', 'WARNING'], + ], + }; + + $self->{components_path} = 'snmp_standard::mode::components'; + $self->{components_module} = ['sensor']; +} + +sub new { + my ($class, %options) = @_; + my $self = $class->SUPER::new(package => __PACKAGE__, %options, no_absent => 1, no_load_components => 1); + bless $self, $class; + + $self->{version} = '1.0'; + $options{options}->add_options(arguments => + { + "sensor-scale" => { name => 'sensor_scale' }, + }); + + return $self; +} + +sub snmp_execute { + my ($self, %options) = @_; + + my $oid_entPhysicalName = '.1.3.6.1.2.1.47.1.1.1.1.7'; + $self->{snmp} = $options{snmp}; + push @{$self->{request}}, { oid => $oid_entPhysicalName}; + $self->{results} = $self->{snmp}->get_multiple_table(oids => $self->{request}); +} + +1; + +=head1 MODE + +Check entity sensors (ENTITY-SENSOR-MIB). + +=over 8 + +=item B<--component> + +Which component to check (Default: '.*'). +Can be: 'sensor'. + +=item B<--filter> + +Exclude some parts (comma seperated list) +Can also exclude specific instance: --filter=sensor,celsius.* + +=item B<--no-component> + +Return an error if no compenents are checked. +If total (with skipped) is 0. (Default: 'critical' returns). + +=item B<--threshold-overload> + +Set to overload default threshold values (syntax: section,[instance,]status,regexp) +It used before default thresholds (order stays). +Example: --threshold-overload='sensor.celsius,OK,nonoperational' + +=item B<--warning> + +Set warning threshold (syntax: type,regexp,threshold) +Example: --warning='sensor.celsius,.*,30' + +=item B<--critical> + +Set critical threshold (syntax: type,regexp,threshold) +Example: --critical='sensor.celsius,.*,40' + +=item B<--sensor-scale> + +Apply scaling value (some equipments are buggy. So we don't use scale by default). + +=back + +=cut + +package snmp_standard::mode::components::sensor; + +use strict; +use warnings; + +my %map_sensor_status = (1 => 'ok', 2 => 'unavailable', 3 => 'nonoperational'); +my %map_sensor_type = ( + 1 => 'other', + 2 => 'unknown', + 3 => 'voltsAC', + 4 => 'voltsDC', + 5 => 'amperes', + 6 => 'watts', + 7 => 'hertz', + 8 => 'celsius', + 9 => 'percentRH', + 10 => 'rpm', + 11 => 'cmm', + 12 => 'truthvalue', + 13 => 'specialEnum', + 14 => 'dBm', +); +my %map_sensor_scale = ( + 1 => -24, # yocto, + 2 => -21, # zepto + 3 => -18, # atto + 4 => -15, # femto + 5 => -12, # pico + 6 => -9, # nano + 7 => -6, # micro + 8 => -3, # milli + 9 => 0, #units + 10 => 3, #kilo + 11 => 6, #mega + 12 => 9, #giga + 13 => 12, #tera + 14 => 18, #exa + 15 => 15, #peta + 16 => 21, #zetta + 17 => 24, #yotta +); +my %perfdata_unit = ('other' => '', 'unknown' => '', 'voltsAC' => 'V', + 'voltsDC' => 'V', 'amperes' => 'A', 'watts' => 'W', + 'hertz' => 'Hz', 'celsius' => 'C', 'percentRH' => '%', + 'rpm' => 'rpm', 'cmm' => '', 'truthvalue' => '', + 'specialEnum' => '', 'dBm' => 'dBm', +); + +my $mapping = { + entPhySensorType => { oid => '.1.3.6.1.2.1.99.1.1.1.1', map => \%map_sensor_type }, + entPhySensorScale => { oid => '.1.3.6.1.2.1.99.1.1.1.2', map => \%map_sensor_scale }, + entPhySensorPrecision => { oid => '.1.3.6.1.2.1.99.1.1.1.3' }, + entPhySensorValue => { oid => '.1.3.6.1.2.1.99.1.1.1.4' }, + entPhySensorOperStatus => { oid => '.1.3.6.1.2.1.99.1.1.1.5', map => \%map_sensor_status }, +}; +my $oid_entPhySensorEntry = '.1.3.6.1.2.1.99.1.1.1'; + +sub load { + my ($self) = @_; + + push @{$self->{request}}, { oid => $oid_entPhySensorEntry, end => $mapping->{entPhySensorOperStatus}->{oid} }; +} + +sub check { + my ($self) = @_; + + $self->{output}->output_add(long_msg => "Checking sensors"); + $self->{components}->{sensor} = {name => 'sensors', total => 0, skip => 0}; + return if ($self->check_filter(section => 'sensor')); + + my $oid_entPhysicalName = '.1.3.6.1.2.1.47.1.1.1.1.7'; + foreach my $oid ($self->{snmp}->oid_lex_sort(keys %{$self->{results}->{$oid_entPhysicalName}})) { + next if ($oid !~ /^$oid_entPhysicalName\.(.*)$/); + my $instance = $1; + my $result = $self->{snmp}->map_instance(mapping => $mapping, results => $self->{results}->{$oid_entPhySensorEntry}, instance => $instance); + next if (!defined($result->{entPhySensorOperStatus})); + + next if ($self->check_filter(section => 'sensor', instance => $result->{entPhySensorType} . '.' . $instance)); + + my $name = $self->{results}->{$oid_entPhysicalName}->{$oid}; + # It seems there is no scale + if (!defined($self->{option_results}->{sensor_scale})) { + $result->{entPhySensorValue} = defined($result->{entPhySensorValue}) ? + $result->{entPhySensorValue} * (10 ** -($result->{entPhySensorPrecision})) : undef; + } else { + $result->{entPhySensorValue} = defined($result->{entPhySensorValue}) ? + $result->{entPhySensorValue} * (10 ** ($result->{entPhySensorScale}) * (10 ** -($result->{entPhySensorPrecision}))) : undef; + } + + $self->{components}->{sensor}->{total}++; + $self->{output}->output_add(long_msg => sprintf("sensor '%s' status is '%s' [instance = %s, value = %s]", + $name, $result->{entPhySensorOperStatus}, $result->{entPhySensorType} . '.' . $instance, + defined($result->{entPhySensorValue}) ? $result->{entPhySensorValue} : '-')); + my $exit = $self->get_severity(label => 'sensor', section => 'sensor.' . $result->{entPhySensorType}, value => $result->{entPhySensorOperStatus}); + if (!$self->{output}->is_status(value => $exit, compare => 'ok', litteral => 1)) { + $self->{output}->output_add(severity => $exit, + short_msg => sprintf("Sensor '%s' status is '%s'", $name, $result->{entPhySensorOperStatus})); + } + + next if (!defined($result->{entPhySensorValue}) || $result->{entPhySensorValue} !~ /[0-9]/); + + my $component = 'sensor.' . $result->{entPhySensorType}; + my ($exit2, $warn, $crit, $checked) = $self->get_severity_numeric(section => $component, instance => $instance, value => $result->{entPhySensorValue}); + if (!$self->{output}->is_status(value => $exit2, compare => 'ok', litteral => 1)) { + $self->{output}->output_add(severity => $exit2, + short_msg => sprintf("Sensor '%s/%s' is %s %s", $name, $instance, $result->{entPhySensorValue}, $perfdata_unit{$result->{entPhySensorType}})); + } + $self->{output}->perfdata_add(label => $component . '_' . $name, unit => $perfdata_unit{$result->{entPhySensorType}}, + value => $result->{entPhySensorValue}, + warning => $warn, + critical => $crit); + } +} + +1;