diff --git a/database/oracle/mode/dataguard.pm b/database/oracle/mode/dataguard.pm new file mode 100644 index 000000000..b95fb9f6a --- /dev/null +++ b/database/oracle/mode/dataguard.pm @@ -0,0 +1,200 @@ +# +# Copyright 2020 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 database::oracle::mode::dataguard; + +use base qw(centreon::plugins::templates::counter); + +use strict; +use warnings; +use Digest::MD5 qw(md5_hex); +use centreon::plugins::templates::catalog_functions qw(catalog_status_threshold catalog_status_calc); + +sub custom_status_output { + my ($self, %options) = @_; + + return sprintf( + "dataguard role '%s' [open mode: '%s'] - managed recovery process %s status is %s [log transport: %s]", + $self->{result_values}->{role}, + $self->{result_values}->{open_mode}, + $self->{result_values}->{mrp_process}, + $self->{result_values}->{mrp_status}, + $self->{result_values}->{log_transport} + ); +} + +sub custom_status_calc { + my ($self, %options) = @_; + + $self->{result_values}->{roleLast} = $options{old_datas}->{$self->{instance} . '_role'}; + $self->{result_values}->{role} = $options{new_datas}->{$self->{instance} . '_role'}; + $self->{result_values}->{open_mode} = $options{new_datas}->{$self->{instance} . '_open_mode'}; + $self->{result_values}->{mrp_status} = $options{new_datas}->{$self->{instance} . '_mrp_status'}; + $self->{result_values}->{mrp_process} = $options{new_datas}->{$self->{instance} . '_mrp_process'}; + $self->{result_values}->{log_transport} = $options{new_datas}->{$self->{instance} . '_log_transport'}; + if (!defined($options{old_datas}->{$self->{instance} . '_role'})) { + $self->{error_msg} = "buffer creation"; + return -2; + } + + return 0; +} + +sub set_counters { + my ($self, %options) = @_; + + $self->{maps_counters_type} = [ + { name => 'global', type => 0, skipped_code => { -10 => 1 } }, + ]; + + $self->{maps_counters}->{global} = [ + { label => 'status', threshold => 0, set => { + key_values => [ + { name => 'role' }, { name => 'open_mode' }, + { name => 'mrp_status' }, { name => 'mrp_status' }, + { name => 'log_transport' } + ], + closure_custom_calc => $self->can('catalog_status_calc'), + closure_custom_output => $self->can('custom_status_output'), + closure_custom_perfdata => sub { return 0; }, + closure_custom_threshold_check => \&catalog_status_threshold, + } + }, + { label => 'standby-lag', nlabel => 'dataguard.standby.lag.minutes', set => { + key_values => [ { name => 'lag_minutes' } ], + output_template => 'dataguard standby lag %d minutes: %s', + perfdatas => [ + { value => 'lag_minutes_absolute', template => '%s', min => 0, unit => 'm' }, + ], + } + }, + ]; +} + +sub new { + my ($class, %options) = @_; + my $self = $class->SUPER::new(package => __PACKAGE__, %options, statefile => 1, force_new_perfdata => 1); + bless $self, $class; + + $options{options}->add_options(arguments => { + 'unknown-status:s' => { name => 'unknown_status', default => '%{mrp_status} =~ /undefined/ || %{log_transport} =~ /undefined/' }, + 'warning-status:s' => { name => 'warning_status', default => '%{mrp_status} =~ /APPLYING_LOG|WAIT_FOR_LOG/i and %{log_transport} =~ /LGWR/i' }, + 'critical-status:s' => { name => 'critical_status', default => '%{roleLast} ne %{role} || %{mrp_status} !~ /undefined|APPLYING_LOG|WAIT_FOR_LOG/i' }, + }); + + return $self; +} + +sub check_options { + my ($self, %options) = @_; + $self->SUPER::check_options(%options); + + $self->change_macros( + macros => [ + 'unknown_status', 'warning_status', 'critical_status' + ] + ); +} + +sub manage_selection { + my ($self, %options) = @_; + + $options{sql}->connect(); + $options{sql}->query( + query => q{ + SELECT name, open_mode, database_role FROM v$database + } + ); + my @result = $options{sql}->fetchrow_array(); + $self->{global} = { + role => $result[2], + open_mode => $result[1] + }; + + $options{sql}->query( + query => q{ + SELECT process, status FROM v$managed_standby WHERE process LIKE 'MR%' + } + ); + @result = $options{sql}->fetchrow_array(); + $self->{global}->{mrp_process} = defined($result[0]) ? $result[0] : 'undefined'; + $self->{global}->{mrp_status} = defined($result[1]) ? $result[1] : 'undefined'; + + $options{sql}->query( + query => q{ + SELECT DECODE(COUNT(*),0,'ARCH','LGWR') AS log_transport FROM v$managed_standby WHERE client_process = 'LGWR' + } + ); + @result = $options{sql}->fetchrow_array(); + $self->{global}->{log_transport} = defined($result[0]) ? $result[0] : 'undefined'; + + $options{sql}->query( + query => q{ + SELECT + TO_CHAR(MAX(first_time),'YYYYMMDDHH24MISS'), + CEIL((SYSDATE - MAX(first_time)) * 24 * 60) + FROM + v$archived_log + WHERE + applied NOT IN ('NO') AND registrar = 'RFS' + } + ); + @result = $options{sql}->fetchrow_array(); + $self->{global}->{lag_minutes} = defined($result[1]) ? $result[1] : -1; + + $options{sql}->disconnect(); + + $self->{cache_name} = "oracle_" . $self->{mode} . '_' . $options{sql}->get_unique_id4save() . '_' . + (defined($self->{option_results}->{filter_counters}) ? md5_hex($self->{option_results}->{filter_counters}) : md5_hex('all')); +} + +1; + +__END__ + +=head1 MODE + +Check oracle dataguard. + +=over 8 + +=item B<--unknown-status> + +Set unknown threshold for status (Default: '%{mrp_status} =~ /undefined/ || %{log_transport} =~ /undefined/'). +Can used special variables like: %{roleLast}, %{role}, %{open_mode}, %{mrp_status}, %{mrp_process}, %{log_transport} + +=item B<--warning-status> + +Set warning threshold for status (Default: '%{mrp_status} =~ /APPLYING_LOG|WAIT_FOR_LOG/i and %{log_transport} =~ /LGWR/i'). +Can used special variables like: %{roleLast}, %{role}, %{open_mode}, %{mrp_status}, %{mrp_process}, %{log_transport} + +=item B<--critical-status> + +Set critical threshold for status (Default: '%{roleLast} ne %{role} || %{mrp_status} !~ /undefined|APPLYING_LOG|WAIT_FOR_LOG/i'). +Can used special variables like: %{roleLast}, %{role}, %{open_mode}, %{mrp_status}, %{mrp_process}, %{log_transport} + +=item B<--warning-*> B<--critical-*> + +Thresholds. +Can be: 'standby-lag'. + +=back + +=cut diff --git a/database/oracle/mode/dictionarycacheusage.pm b/database/oracle/mode/dictionarycacheusage.pm index 5cae606af..658c084e0 100644 --- a/database/oracle/mode/dictionarycacheusage.pm +++ b/database/oracle/mode/dictionarycacheusage.pm @@ -75,7 +75,7 @@ sub new { } sub manage_selection { - my ($self, %options) = @_; + my ($self, %options) = @_; my $query = q{ SELECT SUM(gets), SUM(gets-getmisses) FROM v$rowcache diff --git a/database/oracle/plugin.pm b/database/oracle/plugin.pm index c1fe54c2e..50c2bb746 100644 --- a/database/oracle/plugin.pm +++ b/database/oracle/plugin.pm @@ -36,6 +36,7 @@ sub new { 'connection-time' => 'centreon::common::protocols::sql::mode::connectiontime', 'connected-users' => 'database::oracle::mode::connectedusers', 'corrupted-blocks' => 'database::oracle::mode::corruptedblocks', + 'dataguard' => 'database::oracle::mode::dataguard', 'data-files-status' => 'database::oracle::mode::datafilesstatus', 'datacache-hitratio' => 'database::oracle::mode::datacachehitratio', 'dictionary-cache-usage' => 'database::oracle::mode::dictionarycacheusage',