enhance cisco meraki
This commit is contained in:
parent
f9696bb834
commit
c3337e71da
|
@ -58,6 +58,7 @@ sub new {
|
||||||
$self->{http} = centreon::plugins::http->new(%options);
|
$self->{http} = centreon::plugins::http->new(%options);
|
||||||
$self->{cache} = centreon::plugins::statefile->new(%options);
|
$self->{cache} = centreon::plugins::statefile->new(%options);
|
||||||
$self->{cache_checked} = 0;
|
$self->{cache_checked} = 0;
|
||||||
|
$self->{cache_uplink_loss_latency} = {};
|
||||||
|
|
||||||
return $self;
|
return $self;
|
||||||
}
|
}
|
||||||
|
@ -100,6 +101,28 @@ sub get_token {
|
||||||
return md5_hex($self->{api_token});
|
return md5_hex($self->{api_token});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub get_shard_hostname {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
my $organization_id = $options{organization_id};
|
||||||
|
my $network_id = $options{network_id};
|
||||||
|
if (defined($options{serial}) && defined($self->{cache_devices}->{ $options{serial} })) {
|
||||||
|
$network_id = $self->{cache_devices}->{ $options{serial} }->{networkId};
|
||||||
|
}
|
||||||
|
if (defined($network_id) && defined($self->{cache_networks}->{$network_id})) {
|
||||||
|
$organization_id = $self->{cache_networks}->{$network_id}->{organizationId};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined($organization_id)) {
|
||||||
|
if (defined($self->{cache_organizations}->{$organization_id})
|
||||||
|
&& $self->{cache_organizations}->{$organization_id}->{url} =~ /^(?:http|https):\/\/(.*?)\//) {
|
||||||
|
return $1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
sub get_cache_organizations {
|
sub get_cache_organizations {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
@ -114,6 +137,13 @@ sub get_cache_networks {
|
||||||
return $self->{cache_networks};
|
return $self->{cache_networks};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sub get_organization_id {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
$self->cache_meraki_entities();
|
||||||
|
return $self->{cache_networks}->{ $options{network_id} }->{organizationId};
|
||||||
|
}
|
||||||
|
|
||||||
sub get_cache_devices {
|
sub get_cache_devices {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
@ -124,7 +154,6 @@ sub get_cache_devices {
|
||||||
sub build_options_for_httplib {
|
sub build_options_for_httplib {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
$self->{option_results}->{hostname} = $self->{hostname};
|
|
||||||
$self->{option_results}->{timeout} = $self->{timeout};
|
$self->{option_results}->{timeout} = $self->{timeout};
|
||||||
$self->{option_results}->{port} = $self->{port};
|
$self->{option_results}->{port} = $self->{port};
|
||||||
$self->{option_results}->{proto} = $self->{proto};
|
$self->{option_results}->{proto} = $self->{proto};
|
||||||
|
@ -143,26 +172,38 @@ sub request_api {
|
||||||
|
|
||||||
$self->settings();
|
$self->settings();
|
||||||
|
|
||||||
|
my $hostname = $self->{hostname};
|
||||||
|
if (defined($options{hostname})) {
|
||||||
|
$hostname = $options{hostname};
|
||||||
|
}
|
||||||
|
|
||||||
#400: Bad Request- You did something wrong, e.g. a malformed request or missing parameter.
|
#400: Bad Request- You did something wrong, e.g. a malformed request or missing parameter.
|
||||||
#403: Forbidden- You don't have permission to do that.
|
#403: Forbidden- You don't have permission to do that.
|
||||||
#404: Not found- No such URL, or you don't have access to the API or organization at all.
|
#404: Not found- No such URL, or you don't have access to the API or organization at all.
|
||||||
#429: Too Many Requests- You submitted more than 5 calls in 1 second to an Organization, triggering rate limiting. This also applies for API calls made across multiple organizations that triggers rate limiting for one of the organizations.
|
#429: Too Many Requests- You submitted more than 5 calls in 1 second to an Organization, triggering rate limiting. This also applies for API calls made across multiple organizations that triggers rate limiting for one of the organizations.
|
||||||
while (1) {
|
while (1) {
|
||||||
my $response = $self->{http}->request(
|
my $response = $self->{http}->request(
|
||||||
|
hostname => $hostname,
|
||||||
url_path => '/api/v0' . $options{endpoint},
|
url_path => '/api/v0' . $options{endpoint},
|
||||||
critical_status => '',
|
critical_status => '',
|
||||||
warning_status => '',
|
warning_status => '',
|
||||||
unknown_status => '(%{http_code} < 200 or %{http_code} >= 300) and %{http_code} != 429'
|
unknown_status => ''
|
||||||
);
|
);
|
||||||
|
|
||||||
my $code = $self->{http}->get_code();
|
my $code = $self->{http}->get_code();
|
||||||
return [] if ($code == 403 && $self->{ignore_permission_errors} == 1);
|
return [] if ($code == 403 && $self->{ignore_permission_errors} == 1);
|
||||||
|
return undef if (defined($options{ignore_codes}) && defined($options{ignore_codes}->{$code}));
|
||||||
|
|
||||||
if ($code == 429) {
|
if ($code == 429) {
|
||||||
sleep(1);
|
sleep(1);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($code < 200 || $code >= 300) {
|
||||||
|
$self->{output}->add_option_msg(short_msg => $code . ' ' . $self->{http}->get_message());
|
||||||
|
$self->{output}->option_exit();
|
||||||
|
}
|
||||||
|
|
||||||
my $content;
|
my $content;
|
||||||
eval {
|
eval {
|
||||||
$content = JSON::XS->new->allow_nonref(1)->decode($response);
|
$content = JSON::XS->new->allow_nonref(1)->decode($response);
|
||||||
|
@ -232,7 +273,8 @@ sub get_networks {
|
||||||
my $results = {};
|
my $results = {};
|
||||||
foreach my $id (keys %{$self->{cache_organizations}}) {
|
foreach my $id (keys %{$self->{cache_organizations}}) {
|
||||||
my $datas = $self->request_api(
|
my $datas = $self->request_api(
|
||||||
endpoint => '/organizations/' . $id . '/networks'
|
endpoint => '/organizations/' . $id . '/networks',
|
||||||
|
hostname => $self->get_shard_hostname(organization_id => $id)
|
||||||
);
|
);
|
||||||
$results->{$_->{id}} = $_ foreach (@$datas);
|
$results->{$_->{id}} = $_ foreach (@$datas);
|
||||||
}
|
}
|
||||||
|
@ -249,7 +291,8 @@ sub get_devices {
|
||||||
my $results = {};
|
my $results = {};
|
||||||
foreach my $id (keys %{$self->{cache_organizations}}) {
|
foreach my $id (keys %{$self->{cache_organizations}}) {
|
||||||
my $datas = $self->request_api(
|
my $datas = $self->request_api(
|
||||||
endpoint => '/organizations/' . $id . '/devices'
|
endpoint => '/organizations/' . $id . '/devices',
|
||||||
|
hostname => $self->get_shard_hostname(organization_id => $id)
|
||||||
);
|
);
|
||||||
$results->{$_->{serial}} = $_ foreach (@$datas);
|
$results->{$_->{serial}} = $_ foreach (@$datas);
|
||||||
}
|
}
|
||||||
|
@ -269,11 +312,6 @@ sub filter_networks {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (scalar(@$network_ids) > 5) {
|
|
||||||
$self->{output}->add_option_msg(short_msg => 'cannot check than 5 networks at once (api rate limit)');
|
|
||||||
$self->{output}->option_exit();
|
|
||||||
}
|
|
||||||
|
|
||||||
return $network_ids;
|
return $network_ids;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -302,7 +340,11 @@ sub get_networks_connection_stats {
|
||||||
$timespan = 1 if ($timespan <= 0);
|
$timespan = 1 if ($timespan <= 0);
|
||||||
my $results = {};
|
my $results = {};
|
||||||
foreach my $id (@$network_ids) {
|
foreach my $id (@$network_ids) {
|
||||||
my $datas = $self->request_api(endpoint => '/networks/' . $id . '/connectionStats?timespan=' . $options{timespan});
|
my $datas = $self->request_api(
|
||||||
|
endpoint => '/networks/' . $id . '/connectionStats?timespan=' . $options{timespan},
|
||||||
|
hostname => $self->get_shard_hostname(network_id => $id),
|
||||||
|
ignore_codes => { 400 => 1 }
|
||||||
|
);
|
||||||
$results->{$id} = $datas;
|
$results->{$id} = $datas;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -321,6 +363,8 @@ sub get_networks_clients {
|
||||||
foreach my $id (@$network_ids) {
|
foreach my $id (@$network_ids) {
|
||||||
my $datas = $self->request_api(
|
my $datas = $self->request_api(
|
||||||
endpoint => '/networks/' . $id . '/clients?timespan=' . $options{timespan},
|
endpoint => '/networks/' . $id . '/clients?timespan=' . $options{timespan},
|
||||||
|
hostname => $self->get_shard_hostname(network_id => $id),
|
||||||
|
ignore_codes => { 400 => 1 }
|
||||||
);
|
);
|
||||||
$results->{$id} = $datas;
|
$results->{$id} = $datas;
|
||||||
}
|
}
|
||||||
|
@ -336,7 +380,8 @@ sub get_organization_device_statuses {
|
||||||
my $results = {};
|
my $results = {};
|
||||||
foreach my $id (@$organization_ids) {
|
foreach my $id (@$organization_ids) {
|
||||||
my $datas = $self->request_api(
|
my $datas = $self->request_api(
|
||||||
endpoint => '/organizations/' . $id . '/deviceStatuses'
|
endpoint => '/organizations/' . $id . '/deviceStatuses',
|
||||||
|
hostname => $self->get_shard_hostname(organization_id => $id)
|
||||||
);
|
);
|
||||||
foreach (@$datas) {
|
foreach (@$datas) {
|
||||||
$results->{$_->{serial}} = $_;
|
$results->{$_->{serial}} = $_;
|
||||||
|
@ -358,7 +403,8 @@ sub get_organization_api_requests_overview {
|
||||||
my $results = {};
|
my $results = {};
|
||||||
foreach my $id (@$organization_ids) {
|
foreach my $id (@$organization_ids) {
|
||||||
$results->{$id} = $self->request_api(
|
$results->{$id} = $self->request_api(
|
||||||
endpoint => '/organizations/' . $id . '/apiRequests/overview?timespan=' . $options{timespan}
|
endpoint => '/organizations/' . $id . '/apiRequests/overview?timespan=' . $options{timespan},
|
||||||
|
hostname => $self->get_shard_hostname(organization_id => $id)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -368,68 +414,77 @@ sub get_organization_api_requests_overview {
|
||||||
sub get_network_device_connection_stats {
|
sub get_network_device_connection_stats {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
if (scalar(keys %{$options{devices}}) > 5) {
|
|
||||||
$self->{output}->add_option_msg(short_msg => 'cannot check more than 5 devices at once (api rate limit)');
|
|
||||||
$self->{output}->option_exit();
|
|
||||||
}
|
|
||||||
|
|
||||||
$self->cache_meraki_entities();
|
$self->cache_meraki_entities();
|
||||||
my $timespan = defined($options{timespan}) ? $options{timespan} : 300;
|
my $timespan = defined($options{timespan}) ? $options{timespan} : 300;
|
||||||
$timespan = 1 if ($timespan <= 0);
|
$timespan = 1 if ($timespan <= 0);
|
||||||
|
|
||||||
my $results = {};
|
return $self->request_api(
|
||||||
foreach (keys %{$options{devices}}) {
|
endpoint => '/networks/' . $options{network_id} . '/devices/' . $options{serial} . '/connectionStats?timespan=' . $options{timespan},
|
||||||
my $data = $self->request_api(
|
hostname => $self->get_shard_hostname(network_id => $options{network_id})
|
||||||
endpoint => '/networks/' . $options{devices}->{$_} . '/devices/' . $_ . '/connectionStats?timespan=' . $options{timespan}
|
|
||||||
);
|
);
|
||||||
$results->{$_} = $data;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $results;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_network_device_uplink {
|
sub get_network_device_uplink {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
if (scalar(keys %{$options{devices}}) > 5) {
|
|
||||||
$self->{output}->add_option_msg(short_msg => 'cannot check more than 5 devices at once (api rate limit)');
|
|
||||||
$self->{output}->option_exit();
|
|
||||||
}
|
|
||||||
|
|
||||||
$self->cache_meraki_entities();
|
$self->cache_meraki_entities();
|
||||||
|
|
||||||
my $results = {};
|
return $self->request_api(
|
||||||
foreach (keys %{$options{devices}}) {
|
endpoint => '/networks/' . $options{network_id} . '/devices/' . $options{serial} . '/uplink',
|
||||||
my $data = $self->request_api(
|
hostname => $self->get_shard_hostname(network_id => $options{network_id})
|
||||||
endpoint => '/networks/' . $options{devices}->{$_} . '/devices/' . $_ . '/uplink'
|
|
||||||
);
|
);
|
||||||
$results->{$_} = $data;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $results;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub get_device_clients {
|
sub get_device_clients {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
if (scalar(keys %{$options{devices}}) > 5) {
|
$self->cache_meraki_entities();
|
||||||
$self->{output}->add_option_msg(short_msg => 'cannot check more than 5 devices at once (api rate limit)');
|
my $timespan = defined($options{timespan}) ? $options{timespan} : 300;
|
||||||
$self->{output}->option_exit();
|
$timespan = 1 if ($timespan <= 0);
|
||||||
}
|
|
||||||
|
return $self->request_api(
|
||||||
|
endpoint => '/devices/' . $options{serial} . '/clients?timespan=' . $options{timespan},
|
||||||
|
hostname => $self->get_shard_hostname(serial => $options{serial})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub get_network_device_performance {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
$self->cache_meraki_entities();
|
||||||
|
|
||||||
|
# 400 = feature not supported. 204 = no content
|
||||||
|
return $self->request_api(
|
||||||
|
endpoint => '/networks/' . $options{network_id} . '/devices/' . $options{serial} . '/performance',
|
||||||
|
hostname => $self->get_shard_hostname(network_id => $options{network_id}),
|
||||||
|
ignore_codes => { 400 => 1, 204 => 1 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub get_organization_uplink_loss_and_latency {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
$self->cache_meraki_entities();
|
$self->cache_meraki_entities();
|
||||||
my $timespan = defined($options{timespan}) ? $options{timespan} : 300;
|
my $timespan = defined($options{timespan}) ? $options{timespan} : 300;
|
||||||
$timespan = 1 if ($timespan <= 0);
|
$timespan = 1 if ($timespan <= 0);
|
||||||
|
|
||||||
my $results = {};
|
if (!defined($self->{cache_uplink_loss_latency}->{ $options{organization_id} })) {
|
||||||
foreach (keys %{$options{devices}}) {
|
$self->{cache_uplink_loss_latency}->{ $options{organization_id} } = $self->request_api(
|
||||||
my $data = $self->request_api(
|
endpoint => '/organizations/' . $options{organization_id} . '/uplinksLossAndLatency?timespan=' . $options{timespan},
|
||||||
endpoint => '/devices/' . $_ . '/clients?timespan=' . $options{timespan}
|
hostname => $self->get_shard_hostname(organization_id => $options{organization_id})
|
||||||
);
|
);
|
||||||
$results->{$_} = $data;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $results;
|
my $result = {};
|
||||||
|
if (defined($self->{cache_uplink_loss_latency}->{ $options{organization_id} })) {
|
||||||
|
foreach (@{$self->{cache_uplink_loss_latency}->{ $options{organization_id} }}) {
|
||||||
|
if ($options{serial} eq $_->{serial}) {
|
||||||
|
$result->{ $_->{uplink} } = $_;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
1;
|
1;
|
||||||
|
|
|
@ -24,7 +24,7 @@ use base qw(centreon::plugins::templates::counter);
|
||||||
|
|
||||||
use strict;
|
use strict;
|
||||||
use warnings;
|
use warnings;
|
||||||
use centreon::plugins::templates::catalog_functions qw(catalog_status_threshold catalog_status_calc);
|
use centreon::plugins::templates::catalog_functions qw(catalog_status_threshold_ng);
|
||||||
use Digest::MD5 qw(md5_hex);
|
use Digest::MD5 qw(md5_hex);
|
||||||
|
|
||||||
sub custom_status_output {
|
sub custom_status_output {
|
||||||
|
@ -47,6 +47,7 @@ sub set_counters {
|
||||||
{ name => 'devices', type => 3, cb_prefix_output => 'prefix_device_output', cb_long_output => 'device_long_output', indent_long_output => ' ', message_multiple => 'All devices are ok',
|
{ name => 'devices', type => 3, cb_prefix_output => 'prefix_device_output', cb_long_output => 'device_long_output', indent_long_output => ' ', message_multiple => 'All devices are ok',
|
||||||
group => [
|
group => [
|
||||||
{ name => 'device_status', type => 0, skipped_code => { -10 => 1 } },
|
{ name => 'device_status', type => 0, skipped_code => { -10 => 1 } },
|
||||||
|
{ name => 'device_performance', type => 0, skipped_code => { -10 => 1 } },
|
||||||
{ name => 'device_connections', type => 0, cb_prefix_output => 'prefix_connection_output', skipped_code => { -10 => 1 } },
|
{ name => 'device_connections', type => 0, cb_prefix_output => 'prefix_connection_output', skipped_code => { -10 => 1 } },
|
||||||
{ name => 'device_traffic', type => 0, cb_prefix_output => 'prefix_traffic_output', skipped_code => { -10 => 1, -11 => 1 } },
|
{ name => 'device_traffic', type => 0, cb_prefix_output => 'prefix_traffic_output', skipped_code => { -10 => 1, -11 => 1 } },
|
||||||
{ name => 'device_links', display_long => 1, cb_prefix_output => 'prefix_link_output', message_multiple => 'All links are ok', type => 1, skipped_code => { -10 => 1 } },
|
{ name => 'device_links', display_long => 1, cb_prefix_output => 'prefix_link_output', message_multiple => 'All links are ok', type => 1, skipped_code => { -10 => 1 } },
|
||||||
|
@ -82,54 +83,64 @@ sub set_counters {
|
||||||
];
|
];
|
||||||
|
|
||||||
$self->{maps_counters}->{device_status} = [
|
$self->{maps_counters}->{device_status} = [
|
||||||
{ label => 'status', threshold => 0, set => {
|
{ label => 'status', type => 2, critical_default => '%{status} =~ /alerting/i', set => {
|
||||||
key_values => [ { name => 'status' }, { name => 'display' } ],
|
key_values => [ { name => 'status' }, { name => 'display' } ],
|
||||||
closure_custom_calc => \&catalog_status_calc,
|
|
||||||
closure_custom_output => $self->can('custom_status_output'),
|
closure_custom_output => $self->can('custom_status_output'),
|
||||||
closure_custom_perfdata => sub { return 0; },
|
closure_custom_perfdata => sub { return 0; },
|
||||||
closure_custom_threshold_check => \&catalog_status_threshold
|
closure_custom_threshold_check => \&catalog_status_threshold_ng
|
||||||
|
}
|
||||||
|
}
|
||||||
|
];
|
||||||
|
|
||||||
|
$self->{maps_counters}->{device_performance} = [
|
||||||
|
{ label => 'load', nlabel => 'device.load.count', set => {
|
||||||
|
key_values => [ { name => 'perfscore' } ],
|
||||||
|
output_template => 'load: %s',
|
||||||
|
perfdatas => [
|
||||||
|
{ template => '%d', min => 0, max => 100, label_extra_instance => 1 }
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
$self->{maps_counters}->{device_connections} = [
|
$self->{maps_counters}->{device_connections} = [
|
||||||
{ label => 'connections-success', nlabel => 'device.connections.success.count', set => {
|
{ label => 'connections-success', nlabel => 'device.connections.success.count', set => {
|
||||||
key_values => [ { name => 'assoc' }, { name => 'display' } ],
|
key_values => [ { name => 'assoc' } ],
|
||||||
output_template => 'success: %s',
|
output_template => 'success: %s',
|
||||||
perfdatas => [
|
perfdatas => [
|
||||||
{ template => '%d', min => 0, label_extra_instance => 1, instance_use => 'display' }
|
{ template => '%d', min => 0, label_extra_instance => 1 }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{ label => 'connections-auth', nlabel => 'device.connections.auth.count', display_ok => 0, set => {
|
{ label => 'connections-auth', nlabel => 'device.connections.auth.count', display_ok => 0, set => {
|
||||||
key_values => [ { name => 'auth' }, { name => 'display' } ],
|
key_values => [ { name => 'auth' } ],
|
||||||
output_template => 'auth: %s',
|
output_template => 'auth: %s',
|
||||||
perfdatas => [
|
perfdatas => [
|
||||||
{ template => '%d', min => 0, label_extra_instance => 1, instance_use => 'display' }
|
{ template => '%d', min => 0, label_extra_instance => 1 }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{ label => 'connections-assoc', nlabel => 'device.connections.assoc.count', display_ok => 0, set => {
|
{ label => 'connections-assoc', nlabel => 'device.connections.assoc.count', display_ok => 0, set => {
|
||||||
key_values => [ { name => 'assoc' }, { name => 'display' } ],
|
key_values => [ { name => 'assoc' } ],
|
||||||
output_template => 'assoc: %s',
|
output_template => 'assoc: %s',
|
||||||
perfdatas => [
|
perfdatas => [
|
||||||
{ template => '%d', min => 0, label_extra_instance => 1, instance_use => 'display' }
|
{ template => '%d', min => 0, label_extra_instance => 1 }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{ label => 'connections-dhcp', nlabel => 'device.connections.dhcp.count', display_ok => 0, set => {
|
{ label => 'connections-dhcp', nlabel => 'device.connections.dhcp.count', display_ok => 0, set => {
|
||||||
key_values => [ { name => 'dhcp' }, { name => 'display' } ],
|
key_values => [ { name => 'dhcp' } ],
|
||||||
output_template => 'dhcp: %s',
|
output_template => 'dhcp: %s',
|
||||||
perfdatas => [
|
perfdatas => [
|
||||||
{ template => '%d', min => 0, label_extra_instance => 1, instance_use => 'display' }
|
{ template => '%d', min => 0, label_extra_instance => 1 }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{ label => 'connections-dns', nlabel => 'device.connections.dns.count', display_ok => 0, set => {
|
{ label => 'connections-dns', nlabel => 'device.connections.dns.count', display_ok => 0, set => {
|
||||||
key_values => [ { name => 'dns' }, { name => 'display' } ],
|
key_values => [ { name => 'dns' } ],
|
||||||
output_template => 'dns: %s',
|
output_template => 'dns: %s',
|
||||||
perfdatas => [
|
perfdatas => [
|
||||||
{ template => '%d', min => 0, label_extra_instance => 1, instance_use => 'display' }
|
{ template => '%d', min => 0, label_extra_instance => 1 }
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -157,12 +168,27 @@ sub set_counters {
|
||||||
];
|
];
|
||||||
|
|
||||||
$self->{maps_counters}->{device_links} = [
|
$self->{maps_counters}->{device_links} = [
|
||||||
{ label => 'link-status', threshold => 0, set => {
|
{ label => 'link-status', type => 2, critical_default => '%{link_status} =~ /failed/i', set => {
|
||||||
key_values => [ { name => 'link_status' }, { name => 'display' } ],
|
key_values => [ { name => 'link_status' }, { name => 'display' } ],
|
||||||
closure_custom_calc => \&catalog_status_calc,
|
|
||||||
closure_custom_output => $self->can('custom_link_status_output'),
|
closure_custom_output => $self->can('custom_link_status_output'),
|
||||||
closure_custom_perfdata => sub { return 0; },
|
closure_custom_perfdata => sub { return 0; },
|
||||||
closure_custom_threshold_check => \&catalog_status_threshold
|
closure_custom_threshold_check => \&catalog_status_threshold_ng
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ label => 'link-latency', nlabel => 'device.link.latency.milliseconds', set => {
|
||||||
|
key_values => [ { name => 'latency_ms' } ],
|
||||||
|
output_template => 'latency: %.2f ms',
|
||||||
|
perfdatas => [
|
||||||
|
{ template => '%.2f', min => 0, unit => 'ms', label_extra_instance => 1 }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ label => 'link-loss', nlabel => 'device.link.loss.percentage', set => {
|
||||||
|
key_values => [ { name => 'loss_percent' } ],
|
||||||
|
output_template => 'loss: %.2f %%',
|
||||||
|
perfdatas => [
|
||||||
|
{ template => '%.2f', min => 0, max => 100, unit => '%', label_extra_instance => 1 }
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
@ -211,25 +237,118 @@ sub new {
|
||||||
|
|
||||||
$options{options}->add_options(arguments => {
|
$options{options}->add_options(arguments => {
|
||||||
'filter-device-name:s' => { name => 'filter_device_name' },
|
'filter-device-name:s' => { name => 'filter_device_name' },
|
||||||
'unknown-status:s' => { name => 'unknown_status', default => '' },
|
'filter-network-id:s' => { name => 'filter_network_id' }
|
||||||
'warning-status:s' => { name => 'warning_status', default => '' },
|
|
||||||
'critical-status:s' => { name => 'critical_status', default => '%{status} =~ /alerting/i' },
|
|
||||||
'unknown-link-status:s' => { name => 'unknown_link_status', default => '' },
|
|
||||||
'warning-link-status:s' => { name => 'warning_link_status', default => '' },
|
|
||||||
'critical-link-status:s' => { name => 'critical_link_status', default => '%{link_status} =~ /failed/i' }
|
|
||||||
});
|
});
|
||||||
|
|
||||||
return $self;
|
return $self;
|
||||||
}
|
}
|
||||||
|
|
||||||
sub check_options {
|
sub add_connection_stats {
|
||||||
my ($self, %options) = @_;
|
my ($self, %options) = @_;
|
||||||
$self->SUPER::check_options(%options);
|
|
||||||
|
|
||||||
$self->change_macros(macros => [
|
my $connections = $options{custom}->get_network_device_connection_stats(
|
||||||
'unknown_status', 'warning_status', 'critical_status',
|
timespan => $options{timespan},
|
||||||
'unknown_link_status', 'warning_link_status', 'critical_link_status'
|
serial => $options{serial},
|
||||||
]);
|
network_id => $options{network_id}
|
||||||
|
);
|
||||||
|
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_connections} = {
|
||||||
|
assoc => defined($connections->{assoc}) ? $connections->{assoc} : 0,
|
||||||
|
auth => defined($connections->{auth}) ? $connections->{auth} : 0,
|
||||||
|
dhcp => defined($connections->{dhcp}) ? $connections->{dhcp} : 0,
|
||||||
|
dns => defined($connections->{dns}) ? $connections->{dns} : 0,
|
||||||
|
success => defined($connections->{assoc}) ? $connections->{success} : 0
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
sub add_clients {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
my $clients = $options{custom}->get_device_clients(
|
||||||
|
timespan => $options{timespan},
|
||||||
|
serial => $options{serial}
|
||||||
|
);
|
||||||
|
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_traffic} = {
|
||||||
|
display => $options{name},
|
||||||
|
traffic_in => 0,
|
||||||
|
traffic_out => 0
|
||||||
|
};
|
||||||
|
|
||||||
|
if (defined($clients)) {
|
||||||
|
foreach (@$clients) {
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_traffic}->{traffic_in} += $_->{usage}->{recv} * 8;
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_traffic}->{traffic_out} += $_->{usage}->{sent} * 8;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub add_uplink {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
my $links = $options{custom}->get_network_device_uplink(
|
||||||
|
serial => $options{serial},
|
||||||
|
network_id => $options{network_id}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (defined($links)) {
|
||||||
|
foreach (@$links) {
|
||||||
|
my $interface = lc($_->{interface});
|
||||||
|
$interface =~ s/\s+//g;
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_links}->{$interface} = {
|
||||||
|
display => $interface,
|
||||||
|
link_status => lc($_->{status})
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub add_uplink_loss_latency {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
# 5 minutes max timespan
|
||||||
|
my $links = $options{custom}->get_organization_uplink_loss_and_latency(
|
||||||
|
timespan => 300,
|
||||||
|
serial => $options{serial},
|
||||||
|
organization_id => $options{custom}->get_organization_id(network_id => $options{network_id})
|
||||||
|
);
|
||||||
|
|
||||||
|
return if (!defined($links));
|
||||||
|
|
||||||
|
foreach (values %$links) {
|
||||||
|
my $interface = lc($_->{uplink});
|
||||||
|
$interface =~ s/\s+//g;
|
||||||
|
next if (!defined($self->{devices}->{ $options{serial} }->{device_links}->{$interface}));
|
||||||
|
|
||||||
|
my ($latency, $loss) = (0, 0);
|
||||||
|
foreach my $ts (@{$_->{timeSeries}}) {
|
||||||
|
$latency += $ts->{latencyMs};
|
||||||
|
$loss += $ts->{lossPercent};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scalar(@{$_->{timeSeries}}) > 0) {
|
||||||
|
$latency /= scalar(@{$_->{timeSeries}});
|
||||||
|
$loss /= scalar(@{$_->{timeSeries}});
|
||||||
|
}
|
||||||
|
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_links}->{$interface}->{loss_percent} = $loss;
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_links}->{$interface}->{latency_ms} = $latency;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub add_performance {
|
||||||
|
my ($self, %options) = @_;
|
||||||
|
|
||||||
|
my $perf = $options{custom}->get_network_device_performance(
|
||||||
|
serial => $options{serial},
|
||||||
|
network_id => $options{network_id}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (defined($perf) && defined($perf->{perfScore})) {
|
||||||
|
$self->{devices}->{ $options{serial} }->{device_performance} = {
|
||||||
|
perfscore => $perf->{perfScore}
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sub manage_selection {
|
sub manage_selection {
|
||||||
|
@ -250,14 +369,24 @@ sub manage_selection {
|
||||||
$self->{output}->output_add(long_msg => "skipping device '" . $_->{name} . "': no matching filter.", debug => 1);
|
$self->{output}->output_add(long_msg => "skipping device '" . $_->{name} . "': no matching filter.", debug => 1);
|
||||||
next;
|
next;
|
||||||
}
|
}
|
||||||
|
if (defined($self->{option_results}->{filter_network_id}) && $self->{option_results}->{filter_network_id} ne '' &&
|
||||||
|
$_->{networkId} !~ /$self->{option_results}->{filter_network_id}/) {
|
||||||
|
$self->{output}->output_add(long_msg => "skipping device '" . $_->{name} . "': no matching filter.", debug => 1);
|
||||||
|
next;
|
||||||
|
}
|
||||||
|
|
||||||
$devices->{$_->{serial}} = $_->{networkId};
|
$devices->{ $_->{serial} } = $_->{model};
|
||||||
}
|
}
|
||||||
|
|
||||||
my $device_statuses = $options{custom}->get_organization_device_statuses();
|
my $device_statuses = $options{custom}->get_organization_device_statuses();
|
||||||
my $connections = $options{custom}->get_network_device_connection_stats(timespan => $timespan, devices => $devices);
|
|
||||||
my $clients = $options{custom}->get_device_clients(timespan => $timespan, devices => $devices);
|
# | /clients | /connectionStats | /performance | /uplink | /uplinksLossAndLatency
|
||||||
my $links = $options{custom}->get_network_device_uplink(devices => $devices);
|
#-------------------|----------|---------------------------------|---------|-------------------------
|
||||||
|
# MV [camera] | | | | X |
|
||||||
|
# MS [switch] | X | | | X |
|
||||||
|
# MG [cellullar gw] | X | X | | X |
|
||||||
|
# MX [appliance] | X | | X | X | X
|
||||||
|
# MR [wireless] | X | X | | X |
|
||||||
|
|
||||||
$self->{global} = { total => 0, online => 0, offline => 0, alerting => 0 };
|
$self->{global} = { total => 0, online => 0, offline => 0, alerting => 0 };
|
||||||
$self->{devices} = {};
|
$self->{devices} = {};
|
||||||
|
@ -268,36 +397,47 @@ sub manage_selection {
|
||||||
display => $cache_devices->{$serial}->{name},
|
display => $cache_devices->{$serial}->{name},
|
||||||
status => $device_statuses->{$serial}->{status}
|
status => $device_statuses->{$serial}->{status}
|
||||||
},
|
},
|
||||||
device_connections => {
|
|
||||||
display => $cache_devices->{$serial}->{name},
|
|
||||||
assoc => defined($connections->{$serial}->{assoc}) ? $connections->{$serial}->{assoc} : 0,
|
|
||||||
auth => defined($connections->{$serial}->{auth}) ? $connections->{$serial}->{auth} : 0,
|
|
||||||
dhcp => defined($connections->{$serial}->{dhcp}) ? $connections->{$serial}->{dhcp} : 0,
|
|
||||||
dns => defined($connections->{$serial}->{dns}) ? $connections->{$serial}->{dns} : 0,
|
|
||||||
success => defined($connections->{$serial}->{assoc}) ? $connections->{$serial}->{success} : 0,
|
|
||||||
},
|
|
||||||
device_traffic => {
|
|
||||||
display => $cache_devices->{$serial}->{name},
|
|
||||||
traffic_in => 0,
|
|
||||||
traffic_out => 0
|
|
||||||
},
|
|
||||||
device_links => {}
|
device_links => {}
|
||||||
};
|
};
|
||||||
|
|
||||||
if (defined($clients->{$serial})) {
|
if ($devices->{$serial} =~ /^(?:MG|MR)/) {
|
||||||
foreach (@{$clients->{$serial}}) {
|
$self->add_connection_stats(
|
||||||
$self->{devices}->{$serial}->{device_traffic}->{traffic_in} += $_->{usage}->{recv} * 8;
|
custom => $options{custom},
|
||||||
$self->{devices}->{$serial}->{device_traffic}->{traffic_out} += $_->{usage}->{sent} * 8;
|
timespan => $timespan,
|
||||||
|
serial => $serial,
|
||||||
|
name => $cache_devices->{$serial}->{name},
|
||||||
|
network_id => $cache_devices->{$serial}->{networkId}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
if ($devices->{$serial} =~ /^(?:MS|MG|MR|MX)/) {
|
||||||
|
$self->add_clients(
|
||||||
|
custom => $options{custom},
|
||||||
|
timespan => $timespan,
|
||||||
|
serial => $serial,
|
||||||
|
name => $cache_devices->{$serial}->{name}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
if ($devices->{$serial} =~ /^(?:MV|MS|MG|MR|MX)/) {
|
||||||
if (defined($links->{$serial})) {
|
$self->add_uplink(
|
||||||
foreach (@{$links->{$serial}}) {
|
custom => $options{custom},
|
||||||
$self->{devices}->{$serial}->{device_links}->{$_->{interface}} = {
|
serial => $serial,
|
||||||
display => $_->{interface},
|
name => $cache_devices->{$serial}->{name},
|
||||||
link_status => lc($_->{status})
|
network_id => $cache_devices->{$serial}->{networkId}
|
||||||
};
|
);
|
||||||
}
|
}
|
||||||
|
if ($devices->{$serial} =~ /^MX/) {
|
||||||
|
$self->add_performance(
|
||||||
|
custom => $options{custom},
|
||||||
|
serial => $serial,
|
||||||
|
name => $cache_devices->{$serial}->{name},
|
||||||
|
network_id => $cache_devices->{$serial}->{networkId}
|
||||||
|
);
|
||||||
|
$self->add_uplink_loss_latency(
|
||||||
|
custom => $options{custom},
|
||||||
|
timespan => $timespan,
|
||||||
|
serial => $serial,
|
||||||
|
network_id => $cache_devices->{$serial}->{networkId}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
$self->{global}->{total}++;
|
$self->{global}->{total}++;
|
||||||
|
@ -324,6 +464,10 @@ Check devices.
|
||||||
|
|
||||||
Filter device name (Can be a regexp).
|
Filter device name (Can be a regexp).
|
||||||
|
|
||||||
|
=item B<--filter-network-id>
|
||||||
|
|
||||||
|
Filter network id (Can be a regexp).
|
||||||
|
|
||||||
=item B<--unknown-status>
|
=item B<--unknown-status>
|
||||||
|
|
||||||
Set unknown threshold for status.
|
Set unknown threshold for status.
|
||||||
|
|
Loading…
Reference in New Issue