centreon-plugins/hardware/devices/cisco/cts/snmp/mode/calls.pm

368 lines
15 KiB
Perl

#
# Copyright 2022 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 hardware::devices::cisco::cts::snmp::mode::calls;
use base qw(centreon::plugins::templates::counter);
use strict;
use warnings;
use Digest::MD5 qw(md5_hex);
sub custom_traffic_calc {
my ($self, %options) = @_;
if (!defined($options{delta_time})) {
$self->{error_msg} = 'Buffer creation';
return -1;
}
my $total_bytes = 0;
foreach (keys %{$options{new_datas}}) {
if (/\Q$self->{instance}\E_.*_bytes/) {
my $new_bytes = $options{new_datas}->{$_};
next if (!defined($options{old_datas}->{$_}));
my $old_bytes = $options{old_datas}->{$_};
my $bytes = $new_bytes - $old_bytes;
$bytes = $new_bytes if ($bytes < 0);
$total_bytes += $bytes;
}
}
$self->{result_values}->{display} = $options{new_datas}->{$self->{instance} . '_display'};
$self->{result_values}->{traffic_per_seconds} = ($total_bytes * 8) / $options{delta_time};
return 0;
}
sub custom_jitter_calc {
my ($self, %options) = @_;
my $max_jitter = 0;
foreach (keys %{$options{new_datas}}) {
if (/\Q$self->{instance}\E_.*_maxjitter/) {
$max_jitter = $options{new_datas}->{$_} if ($options{new_datas}->{$_} > $max_jitter);
}
}
$self->{result_values}->{display} = $options{new_datas}->{$self->{instance} . '_display'};
$self->{result_values}->{max_jitter} = $max_jitter;
return 0;
}
sub custom_loss_calc {
my ($self, %options) = @_;
my ($total_loss, $total_pkts) = (0, 0);
foreach (keys %{$options{new_datas}}) {
if (/\Q$self->{instance}\E_.*_loss_$options{extra_options}->{label_ref}/) {
my $new_loss = $options{new_datas}->{$_};
next if (!defined($options{old_datas}->{$_}));
my $old_loss = $options{old_datas}->{$_};
my $loss = $new_loss - $old_loss;
$loss = $new_loss if ($loss < 0);
$total_loss += $loss;
} elsif (/\Q$self->{instance}\E_.*_packets_$options{extra_options}->{label_ref}/) {
my $new_pkts = $options{new_datas}->{$_};
next if (!defined($options{old_datas}->{$_}));
my $old_pkts = $options{old_datas}->{$_};
my $pkts = $new_pkts - $old_pkts;
$pkts = $new_pkts if ($pkts < 0);
$total_pkts += $pkts;
}
}
$self->{result_values}->{label} = $options{extra_options}->{label_ref};
$self->{result_values}->{display} = $options{new_datas}->{$self->{instance} . '_display'};
$self->{result_values}->{packets_loss} = $total_loss;
$self->{result_values}->{packets_loss_prct} = 0;
$self->{result_values}->{packets} = $total_pkts;
if ($total_pkts > 0) {
$self->{result_values}->{packets_loss_prct} = ($total_loss * 100) / $total_pkts;
}
return 0;
}
sub custom_loss_output {
my ($self, %options) = @_;
return sprintf(
"packets %s loss: %.2f%% (%s on %s)",
$self->{result_values}->{label},
$self->{result_values}->{packets_loss_prct},
$self->{result_values}->{packets_loss},
$self->{result_values}->{packets}
);
}
sub prefix_global_output {
my ($self, %options) = @_;
return 'Calls ';
}
sub set_counters {
my ($self, %options) = @_;
$self->{maps_counters_type} = [
{ name => 'global', type => 0, cb_prefix_output => 'prefix_global_output' },
{ name => 'streams_active', type => 1, cb_prefix_output => 'prefix_stream_active_output', message_multiple => 'All active call streams are ok', skipped_code => { -10 => 1 } }
];
$self->{maps_counters}->{global} = [
{ label => 'dummy', threshold => 0, display_ok => 0, set => {
key_values => [ { name => 'calls_instance_finished' } ],
output_template => 'none',
perfdatas => []
}
},
{ label => 'active', nlabel => 'calls.active.count', set => {
key_values => [ { name => 'active' } ],
output_template => 'active: %s',
perfdatas => [
{ template => '%s', min => 0 }
]
}
},
{ label => 'total', nlabel => 'calls.total.count', display_ok => 0, set => {
key_values => [ { name => 'total' } ],
output_template => 'total: %s',
perfdatas => [
{ template => '%s', min => 0 }
]
}
}
];
my @map = (
['total-unknown', 'unknown: %s', 'unknown'],
['total-other', 'other: %s', 'other'],
['total-internal-error', 'internal error: %s', 'internal.error'],
['total-local-disconnected', 'local disconnected: %s', 'local.disconnected'],
['total-remote-disconnected', 'remote disconnected: %s', 'remote.disconnected'],
['total-network-congestion', 'network congestion: %s', 'network.congestion'],
['total-media-negotiation-failure', 'media negotiation failure: %s', 'media.negotiation.failure'],
['total-security-config-mismatched', 'security config mismatched: %s', 'security.config.mismatched'],
['total-incompatible-remote-endpoint', 'incompatible remote endpoint: %s', 'incompatible.remote.endpoint'],
['total-service-unavailable', 'service unaivalable: %s', 'service.unavailable'],
['total-remote-terminated-error', 'remote terminated with error: %s', 'remote.terminated.error']
);
foreach (@map) {
push @{$self->{maps_counters}->{global}}, { label => $_->[0], nlabel => 'calls.total.' . $_->[2] . '.count', display_ok => 0, set => {
key_values => [ { name => $_->[2] } ],
output_template => $_->[1],
perfdatas => [
{ template => '%s', min => 0 }
]
}
};
}
$self->{maps_counters}->{streams_active} = [
{ label => 'streams-active-maxjitter', nlabel => 'calls.streams.active.maxjitter.milliseconds', set => {
key_values => [],
manual_keys => 1,
closure_custom_calc => $self->can('custom_jitter_calc'),
output_template => 'max jitter: %s ms',
output_use => 'max_jitter', threshold_use => 'max_jitter',
perfdatas => [
{ value => 'max_jitter', template => '%d',
unit => 'ms', min => 0, label_extra_instance => 1, instance_use => 'display' }
]
}
}
];
foreach (('in', 'out')) {
push @{$self->{maps_counters}->{streams_active}},
{ label => 'streams-active-traffic-' . $_, nlabel => 'calls.streams.active.traffic.' . $_ . '.bytes', set => {
key_values => [],
manual_keys => 1,
closure_custom_calc => $self->can('custom_traffic_calc'), closure_custom_calc_extra_options => { label_ref => $_ },
output_template => 'traffic ' . $_ . ': %s %s/s',
output_change_bytes => 1,
output_use => 'traffic_per_seconds', threshold_use => 'traffic_per_seconds',
perfdatas => [
{ value => 'traffic_per_seconds', template => '%d',
unit => 'B/s', min => 0, label_extra_instance => 1, instance_use => 'display' }
]
}
},
{ label => 'streams-active-packetloss-' . $_, nlabel => 'calls.streams.active.packetloss.' . $_ . '.count', set => {
key_values => [],
manual_keys => 1,
closure_custom_calc => $self->can('custom_loss_calc'), closure_custom_calc_extra_options => { label_ref => $_ },
closure_custom_output => $self->can('custom_loss_output'),
threshold_use => 'packets_loss',
perfdatas => [
{ value => 'packets_loss', template => '%d',
min => 0, label_extra_instance => 1, instance_use => 'display' }
]
}
},
{ label => 'streams-active-packetloss-' . $_ . '-prct', nlabel => 'calls.streams.active.packetloss.' . $_ . '.percentage', display_ok => 0, set => {
key_values => [],
manual_keys => 1,
closure_custom_calc => $self->can('custom_loss_calc'), closure_custom_calc_extra_options => { label_ref => $_ },
closure_custom_output => $self->can('custom_loss_output'),
threshold_use => 'packets_loss_prct',
perfdatas => [
{ value => 'packets_loss_prct', template => '%.2f',
unit => '%', min => 0, max => 100, label_extra_instance => 1, instance_use => 'display' }
]
}
};
}
}
sub prefix_stream_active_output {
my ($self, %options) = @_;
return "Stream '" . $options{instance_value}->{display} . "' ";
}
sub new {
my ($class, %options) = @_;
my $self = $class->SUPER::new(package => __PACKAGE__, %options, force_new_perfdata => 1, statefile => 1);
bless $self, $class;
$self->{version} = '1.0';
$options{options}->add_options(arguments => {
});
return $self;
}
my $map_term_reason = {
1 => 'unknown', 2 => 'other', 3 => 'internal.error',
4 => 'local.disconnected', 5 => 'remote.disconnected',
6 => 'network.congestion', 7 => 'media.negotiation.failure',
8 => 'security.config.mismatched', 9 => 'incompatible.remote.endpoint',
10 => 'service.unavailable', 11 => 'remote.terminated.error',
12 => 'incall'
};
my $map_stream_type = {
1 => 'video', 2 => 'audio', 3 => 'content'
};
my $oid_ctpcCallTermReason = '.1.3.6.1.4.1.9.9.644.1.4.8.1.17';
my $mapping = {
traffic_out => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.3' }, # ctpcTxTotalBytes
packets_out => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.4' }, # ctpcTxTotalPackets
packets_lost_out => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.5' }, # ctpcTxLostPackets
traffic_in => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.11' }, # ctpcRxTotalBytes
packets_in => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.12' }, # ctpcRxTotalPackets
packets_lost_in => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.13' }, # ctpcRxLostPackets
maxjitter => { oid => '.1.3.6.1.4.1.9.9.644.1.4.10.1.25' }, # ctpcMaxCallJitter
};
sub manage_selection {
my ($self, %options) = @_;
$self->{cache_name} = 'cisco_cts_' . $options{snmp}->get_hostname() . '_' . $options{snmp}->get_port() . '_' . $self->{mode} . '_' .
(defined($self->{option_results}->{filter_counters}) ? md5_hex($self->{option_results}->{filter_counters}) : md5_hex('all'));
my $calls_instance_finished = $self->read_statefile_key(key => 'global_calls_instance_finished');
$calls_instance_finished = {} if (!defined($calls_instance_finished));
my $calls_instance_finished_new = {};
my $active_calls = {};
$self->{global} = { active => 0, total => 0 };
foreach (values %$map_term_reason) {
$self->{global}->{$_} = 0;
}
my $snmp_result = $options{snmp}->get_table(
oid => $oid_ctpcCallTermReason
);
foreach (keys %$snmp_result) {
/^$oid_ctpcCallTermReason\.(\d+)/;
my $instance = $1;
if ($map_term_reason->{ $snmp_result->{$_} } eq 'incall') {
$active_calls->{$instance} = 1;
$self->{global}->{active}++;
} else {
$calls_instance_finished_new->{$instance} = 1;
next if (defined($calls_instance_finished->{$instance}));
$self->{global}->{total}++;
$self->{global}->{ $map_term_reason->{ $snmp_result->{$_} } }++;
}
}
$snmp_result = $options{snmp}->get_multiple_table(
oids => [
map({ oid => $_->{oid} }, values(%$mapping))
],
return_type => 1
);
$self->{streams_active} = {};
foreach (('audio', 'video', 'content')) {
$self->{streams_active}->{$_} = { display => $_ };
}
foreach my $oid (keys %$snmp_result) {
next if ($oid !~ /^$mapping->{traffic_out}->{oid}\.(\d+)\.(\d+)\.(\d+)/);
my ($index, $type, $source) = ($1, $2, $3);
my $instance = $index . '.' . $type . '.' . $source;
my $result = $options{snmp}->map_instance(mapping => $mapping, results => $snmp_result, instance => $instance);
my $stream_type = $map_stream_type->{$type};
if (defined($active_calls->{$index})) {
$self->{streams_active}->{$stream_type}->{$instance . '_traffic_in'} = defined($result->{traffic_in}) ? $result->{traffic_in} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_traffic_out'} = defined($result->{traffic_out}) ? $result->{traffic_out} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_loss_in'} = defined($result->{packets_lost_in}) ? $result->{packets_lost_in} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_packets_in'} = defined($result->{packets_in}) ? $result->{packets_in} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_loss_out'} = defined($result->{packets_lost_out}) ? $result->{packets_lost_out} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_packets_out'} = defined($result->{packets_out}) ? $result->{packets_out} : 0;
$self->{streams_active}->{$stream_type}->{$instance . '_maxjitter'} = defined($result->{maxjitter}) ? $result->{maxjitter} : 0;
}
}
$self->{global}->{calls_instance_finished} = $calls_instance_finished_new;
}
1;
__END__
=head1 MODE
Check call stream (real-time and history)
=over 8
=item B<--warning-*> B<--critical-*>
Thresholds.
Can be: 'active', 'total', 'total-unknown',
'total-other', 'total-internal-error', 'total-local-disconnected', 'total-remote-disconnected',
'total-network-congestion', 'total-media-negotiation-failure', 'total-security-config-mismatched',
'total-incompatible-remote-endpoint', 'total-service-unavailable', 'total-remote-terminated-error',
'streams-active-maxjitter',
'streams-active-traffic-in', 'streams-active-packetloss-in', 'streams-active-packetloss-in-prct',
'streams-active-traffic-out', 'streams-active-packetloss-out', 'streams-active-packetloss-out-prct'.
=back
=cut