From 21c20876276755eccb7ee47f2fde82a20c13db89 Mon Sep 17 00:00:00 2001 From: Colin Gagnaire Date: Tue, 6 Aug 2019 18:11:53 +0200 Subject: [PATCH 1/3] add zoom restapi plugin --- apps/zoom/restapi/custom/api.pm | 239 ++++++++++++++++++++++++++++ apps/zoom/restapi/mode/listusers.pm | 121 ++++++++++++++ apps/zoom/restapi/plugin.pm | 55 +++++++ 3 files changed, 415 insertions(+) create mode 100644 apps/zoom/restapi/custom/api.pm create mode 100644 apps/zoom/restapi/mode/listusers.pm create mode 100644 apps/zoom/restapi/plugin.pm diff --git a/apps/zoom/restapi/custom/api.pm b/apps/zoom/restapi/custom/api.pm new file mode 100644 index 000000000..f6d813d4f --- /dev/null +++ b/apps/zoom/restapi/custom/api.pm @@ -0,0 +1,239 @@ +# +# Copyright 2019 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 apps::zoom::restapi::custom::api; + +use strict; +use warnings; +use centreon::plugins::http; +use centreon::plugins::statefile; +use DateTime; +use Digest::MD5 qw(md5_hex); +use JSON::XS; +use JSON::WebToken; + +sub new { + my ($class, %options) = @_; + my $self = {}; + bless $self, $class; + + if (!defined($options{output})) { + print "Class Custom: Need to specify 'output' argument.\n"; + exit 3; + } + if (!defined($options{options})) { + $options{output}->add_option_msg(short_msg => "Class Custom: Need to specify 'options' argument."); + $options{output}->option_exit(); + } + + if (!defined($options{noptions})) { + $options{options}->add_options(arguments => { + "hostname:s" => { name => 'hostname' }, + "url-path:s" => { name => 'url_path' }, + "port:s" => { name => 'port' }, + "proto:s" => { name => 'proto' }, + "api-key:s" => { name => 'api_key' }, + "api-secret:s" => { name => 'api_secret' }, + "timeout:s" => { name => 'timeout' }, + }); + } + $options{options}->add_help(package => __PACKAGE__, sections => 'REST API OPTIONS', once => 1); + + $self->{output} = $options{output}; + $self->{mode} = $options{mode}; + $self->{http} = centreon::plugins::http->new(%options); + $self->{cache} = centreon::plugins::statefile->new(%options); + + return $self; +} + +sub set_options { + my ($self, %options) = @_; + + $self->{option_results} = $options{option_results}; +} + +sub set_defaults { + my ($self, %options) = @_; + + foreach (keys %{$options{default}}) { + if ($_ eq $self->{mode}) { + for (my $i = 0; $i < scalar(@{$options{default}->{$_}}); $i++) { + foreach my $opt (keys %{$options{default}->{$_}[$i]}) { + if (!defined($self->{option_results}->{$opt}[$i])) { + $self->{option_results}->{$opt}[$i] = $options{default}->{$_}[$i]->{$opt}; + } + } + } + } + } +} + +sub check_options { + my ($self, %options) = @_; + + $self->{hostname} = (defined($self->{option_results}->{hostname})) ? $self->{option_results}->{hostname} : 'api.zoom.us'; + $self->{port} = (defined($self->{option_results}->{port})) ? $self->{option_results}->{port} : 443; + $self->{proto} = (defined($self->{option_results}->{proto})) ? $self->{option_results}->{proto} : 'https'; + $self->{url_path} = (defined($self->{option_results}->{url_path})) ? $self->{option_results}->{url_path} : '/v2'; + $self->{timeout} = (defined($self->{option_results}->{timeout})) ? $self->{option_results}->{timeout} : 10; + $self->{api_key} = (defined($self->{option_results}->{api_key})) ? $self->{option_results}->{api_key} : ''; + $self->{api_secret} = (defined($self->{option_results}->{api_secret})) ? $self->{option_results}->{api_secret} : ''; + + if (!defined($self->{api_key}) || $self->{api_key} eq '') { + $self->{output}->add_option_msg(short_msg => "Need to specify --api-key option."); + $self->{output}->option_exit(); + } + if (!defined($self->{api_secret}) || $self->{api_secret} eq '') { + $self->{output}->add_option_msg(short_msg => "Need to specify --api-secret option."); + $self->{output}->option_exit(); + } + + $self->{cache}->check_options(option_results => $self->{option_results}); + + return 0; +} + +sub build_options_for_httplib { + my ($self, %options) = @_; + + $self->{option_results}->{hostname} = $self->{hostname}; + $self->{option_results}->{timeout} = $self->{timeout}; + $self->{option_results}->{port} = $self->{port}; + $self->{option_results}->{proto} = $self->{proto}; + $self->{option_results}->{url_path} = $self->{url_path}; + $self->{option_results}->{warning_status} = ''; + $self->{option_results}->{critical_status} = ''; +} + +sub settings { + my ($self, %options) = @_; + + $self->build_options_for_httplib(); + $self->{http}->add_header(key => 'Accept', value => 'application/json'); + if (defined($self->{jwt_token})) { + $self->{http}->add_header(key => 'Authorization', value => 'Bearer ' . $self->{jwt_token}); + } + $self->{http}->set_options(%{$self->{option_results}}); +} + +sub get_jwt_token { + my ($self, %options) = @_; + + my $has_cache_file = $options{statefile}->read(statefile => 'zoom_api_' . md5_hex($self->{api_key})); + my $expires_on = $options{statefile}->get(name => 'expires_on'); + my $jwt_token = $options{statefile}->get(name => 'jwt_token'); + + if ($has_cache_file == 0 || !defined($jwt_token) || (($expires_on - time()) < 10)) { + my $exp = time() + 3600; + + $jwt_token = JSON::WebToken->encode({ + alg => "HS256", + typ => "JWT", + iss => $self->{api_key}, + exp => $exp, + }, $self->{api_secret}, 'HS256'); + + my $datas = { last_timestamp => time(), jwt_token => $jwt_token, expires_on => $exp }; + $options{statefile}->write(data => $datas); + } + + return $jwt_token; +} + +sub request_api { + my ($self, %options) = @_; + + if (!defined($self->{jwt_token})) { + $self->{jwt_token} = $self->get_jwt_token(statefile => $self->{cache}); + } + + $self->settings; + + $self->{output}->output_add(long_msg => "Query URL: '" . $self->{proto} . "://" . $self->{hostname} . + $self->{url_path} . $options{url_path} . "'", debug => 1); + + my $content = $self->{http}->request(url_path => $self->{url_path} . $options{url_path}); + + my $decoded; + eval { + $decoded = JSON::XS->new->utf8->decode($content); + }; + if ($@) { + $self->{output}->add_option_msg(short_msg => "Cannot decode json response: $@"); + $self->{output}->option_exit(); + } + + return $decoded; +} + +1; + +__END__ + +=head1 NAME + +Zoom Rest API + +=head1 SYNOPSIS + +Zoom Rest API custom mode + +=head1 REST API OPTIONS + +Zoom Rest API + +=over 8 + +=item B<--hostname> + +Zoom API hostname (Default: 'api.zoom.us') + +=item B<--port> + +API port (Default: 443) + +=item B<--proto> + +Specify https if needed (Default: 'https') + +=item B<--url-path> + +API URL path (Default: '/v2') + +=item B<--api-key> + +JWT app API key. + +=item B<--api-secret> + +JWT app API secret. + +=item B<--timeout> + +Set HTTP timeout. + +=back + +=head1 DESCRIPTION + +B. + +=cut diff --git a/apps/zoom/restapi/mode/listusers.pm b/apps/zoom/restapi/mode/listusers.pm new file mode 100644 index 000000000..efa246ab3 --- /dev/null +++ b/apps/zoom/restapi/mode/listusers.pm @@ -0,0 +1,121 @@ +# +# Copyright 2019 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 apps::zoom::restapi::mode::listusers; + +use base qw(centreon::plugins::templates::counter); + +use strict; +use warnings; + +sub new { + my ($class, %options) = @_; + my $self = $class->SUPER::new(package => __PACKAGE__, %options); + bless $self, $class; + + $options{options}->add_options(arguments => {}); + + return $self; +} + +sub check_options { + my ($self, %options) = @_; + $self->SUPER::check_options(%options); +} + +sub manage_selection { + my ($self, %options) = @_; + + my @users; + + my $page = 1; + while (my $results = $options{custom}->request_api(url_path => '/users?page_size=4&page_number=' . $page)) { + push @users, @{$results->{users}}; + ($results->{page_number} < $results->{page_count}) ? $page++ : last; + } + + foreach my $user (@users) { + $self->{users}->{$user->{id}} = { + first_name => $user->{first_name}, + last_name => $user->{last_name}, + email => $user->{email}, + status => $user->{status}, + id => $user->{id}, + } + } +} + +sub run { + my ($self, %options) = @_; + + $self->manage_selection(%options); + foreach my $user (sort keys %{$self->{users}}) { + $self->{output}->output_add( + long_msg => sprintf("[id = %s] [first_name = %s] [last_name = %s] [email = %s] [status = %s]", + $self->{users}->{$user}->{id}, + $self->{users}->{$user}->{first_name}, + $self->{users}->{$user}->{last_name}, + $self->{users}->{$user}->{email}, + $self->{users}->{$user}->{status}) + ); + } + + $self->{output}->output_add( + severity => 'OK', + short_msg => 'List users:' + ); + $self->{output}->display(nolabel => 1, force_ignore_perfdata => 1, force_long_output => 1); + $self->{output}->exit(); +} + +sub disco_format { + my ($self, %options) = @_; + + $self->{output}->add_disco_format(elements => ['id', 'first_name', 'last_name', 'email', 'status']); +} + +sub disco_show { + my ($self, %options) = @_; + + $self->manage_selection(%options); + foreach my $user (sort keys %{$self->{users}}) { + $self->{output}->add_disco_entry( + id => $self->{users}->{$user}->{id}, + first_name => $self->{users}->{$user}->{first_name}, + last_name => $self->{users}->{$user}->{last_name}, + email => $self->{users}->{$user}->{email}, + status => $self->{users}->{$user}->{status}, + ); + } +} + +1; + +__END__ + +=head1 MODE + +List users. + +=over 8 + +=back + +=cut diff --git a/apps/zoom/restapi/plugin.pm b/apps/zoom/restapi/plugin.pm new file mode 100644 index 000000000..0ff793adb --- /dev/null +++ b/apps/zoom/restapi/plugin.pm @@ -0,0 +1,55 @@ +# +# Copyright 2019 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 apps::zoom::restapi::plugin; + +use strict; +use warnings; +use base qw(centreon::plugins::script_custom); + +sub new { + my ($class, %options) = @_; + my $self = $class->SUPER::new(package => __PACKAGE__, %options); + bless $self, $class; + + $self->{version} = '0.1'; + %{$self->{modes}} = ( + 'list-users' => 'apps::zoom::restapi::mode::listusers', + ); + + $self->{custom_modes}{api} = 'apps::zoom::restapi::custom::api'; + return $self; +} + +sub init { + my ( $self, %options ) = @_; + + $self->SUPER::init(%options); +} + +1; + +__END__ + +=head1 PLUGIN DESCRIPTION + +Check Zoom. + +=cut From 5ed8b0c9fb1b8652089c3c16491696d32f71fa55 Mon Sep 17 00:00:00 2001 From: Colin Gagnaire Date: Tue, 6 Aug 2019 18:17:09 +0200 Subject: [PATCH 2/3] change page_size to 30 --- apps/zoom/restapi/mode/listusers.pm | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/zoom/restapi/mode/listusers.pm b/apps/zoom/restapi/mode/listusers.pm index efa246ab3..e84af031c 100644 --- a/apps/zoom/restapi/mode/listusers.pm +++ b/apps/zoom/restapi/mode/listusers.pm @@ -46,7 +46,7 @@ sub manage_selection { my @users; my $page = 1; - while (my $results = $options{custom}->request_api(url_path => '/users?page_size=4&page_number=' . $page)) { + while (my $results = $options{custom}->request_api(url_path => '/users?page_size=30&page_number=' . $page)) { push @users, @{$results->{users}}; ($results->{page_number} < $results->{page_count}) ? $page++ : last; } From 9cf6b3000b851024f92e6bcfeec3d3a6c7cd56ae Mon Sep 17 00:00:00 2001 From: Stephane Duret Date: Wed, 7 Aug 2019 09:55:31 +0200 Subject: [PATCH 3/3] fix hp-p2000-xmlapi - undefined values for sensor and enclosure --- storage/hp/p2000/xmlapi/mode/components/enclosure.pm | 2 +- storage/hp/p2000/xmlapi/mode/components/sensor.pm | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/storage/hp/p2000/xmlapi/mode/components/enclosure.pm b/storage/hp/p2000/xmlapi/mode/components/enclosure.pm index 032380ea4..8bf4bfe10 100644 --- a/storage/hp/p2000/xmlapi/mode/components/enclosure.pm +++ b/storage/hp/p2000/xmlapi/mode/components/enclosure.pm @@ -52,7 +52,7 @@ sub check { my $state = $health{$results->{$enc_id}->{'health-numeric'}}; $self->{output}->output_add(long_msg => sprintf("enclosure '%s' status is %s [instance: %s] [reason: %s]", - $enc_id, $state, $enc_id, $health{$results->{$enc_id}->{'health-reason'}}) + $enc_id, $state, $enc_id, $results->{$enc_id}->{'health-reason'}) ); my $exit = $self->get_severity(label => 'default', section => 'enclosure', value => $state); if (!$self->{output}->is_status(value => $exit, compare => 'ok', litteral => 1)) { diff --git a/storage/hp/p2000/xmlapi/mode/components/sensor.pm b/storage/hp/p2000/xmlapi/mode/components/sensor.pm index c5983c119..082f74365 100644 --- a/storage/hp/p2000/xmlapi/mode/components/sensor.pm +++ b/storage/hp/p2000/xmlapi/mode/components/sensor.pm @@ -62,7 +62,7 @@ sub check { # Warning # foreach my $sensor_id (keys %$results) { - my ($value, $unit); + my ($value, $unit) = ('', '');; ($value, $unit) = ($1, $2) if ($results->{$sensor_id}->{value} =~ /\s*([0-9\.,]+)\s*(\S*)\s*/); if (defined($results->{$sensor_id}->{'sensor-type'}) && defined($sensor_type{$results->{$sensor_id}->{'sensor-type'}})) { $unit = $sensor_type{$results->{$sensor_id}->{'sensor-type'}}->{unit};