From 33cc2c9a03dab087688c74addf2ffd140a100dc3 Mon Sep 17 00:00:00 2001 From: garnier-quentin Date: Mon, 24 Aug 2015 14:50:21 +0200 Subject: [PATCH] + add some modes for exchange 2010 --- .../2010/local/mode/activesyncmailbox.pm | 1 - apps/exchange/2010/local/mode/databases.pm | 23 ++- apps/exchange/2010/local/mode/imapmailbox.pm | 1 - apps/exchange/2010/local/mode/mapimailbox.pm | 1 - apps/exchange/2010/local/mode/owamailbox.pm | 1 - apps/exchange/2010/local/mode/queues.pm | 1 - .../2010/local/mode/replicationhealth.pm | 156 ++++++++++++++++++ apps/exchange/2010/local/mode/services.pm | 1 - apps/exchange/2010/local/plugin.pm | 1 + .../powershell/exchange/2010/databases.pm | 48 ++++++ .../exchange/2010/replicationhealth.pm | 111 +++++++++++++ 11 files changed, 336 insertions(+), 9 deletions(-) create mode 100644 apps/exchange/2010/local/mode/replicationhealth.pm create mode 100644 centreon/common/powershell/exchange/2010/replicationhealth.pm diff --git a/apps/exchange/2010/local/mode/activesyncmailbox.pm b/apps/exchange/2010/local/mode/activesyncmailbox.pm index cfa4f1907..a74a6e40f 100644 --- a/apps/exchange/2010/local/mode/activesyncmailbox.pm +++ b/apps/exchange/2010/local/mode/activesyncmailbox.pm @@ -50,7 +50,6 @@ sub new { "password:s" => { name => 'password', }, "no-trust-ssl" => { name => 'no_trust_ssl', }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/mode/databases.pm b/apps/exchange/2010/local/mode/databases.pm index 1e0f98281..5e1f27bfc 100644 --- a/apps/exchange/2010/local/mode/databases.pm +++ b/apps/exchange/2010/local/mode/databases.pm @@ -41,6 +41,7 @@ sub new { "no-ps" => { name => 'no_ps', }, "no-mailflow" => { name => 'no_mailflow', }, "no-mapi" => { name => 'no_mapi', }, + "no-copystatus" => { name => 'no_copystatus', }, "timeout:s" => { name => 'timeout', default => 50 }, "command:s" => { name => 'command', default => 'powershell.exe' }, "command-path:s" => { name => 'command_path' }, @@ -52,15 +53,16 @@ sub new { "critical-mapi:s" => { name => 'critical_mapi', default => '%{mapi_result} !~ /Success/i' }, "warning-mailflow:s" => { name => 'warning_mailflow', }, "critical-mailflow:s" => { name => 'critical_mailflow', default => '%{mailflow_result} !~ /Success/i' }, + "warning-copystatus:s" => { name => 'warning_copystatus', }, + "critical-copystatus:s" => { name => 'critical_copystatus', default => '%{copystatus_indexstate} !~ /Healthy/i' }, }); - $self->{thresholds} = {}; return $self; } sub change_macros { my ($self, %options) = @_; - foreach (('warning_mapi', 'critical_mapi', 'warning_mailflow', 'critical_mailflow')) { + foreach (('warning_mapi', 'critical_mapi', 'warning_mailflow', 'critical_mailflow', 'warning_copystatus', 'critical_copystatus')) { if (defined($self->{option_results}->{$_})) { $self->{option_results}->{$_} =~ s/%\{(.*?)\}/\$self->{data}->{$1}/g; } @@ -83,6 +85,7 @@ sub run { no_mailflow => $self->{option_results}->{no_mailflow}, no_ps => $self->{option_results}->{no_ps}, no_mapi => $self->{option_results}->{no_mapi}, + no_copystatus => $self->{option_results}->{no_copystatus}, filter_database => $self->{option_results}->{ps_database_filter}, filter_database_test => $self->{option_results}->{ps_database_test_filter}); $self->{option_results}->{command_options} .= " " . $ps; @@ -109,7 +112,7 @@ __END__ =head1 MODE -Check: Exchange Databases are Mounted, Mapi/Mailflow Connectivity to all databases are working. +Check: Exchange Databases are Mounted, Mapi/Mailflow Connectivity to all databases are working and CopyStatus. =over 8 @@ -133,6 +136,10 @@ Don't check mailflow connectivity. Don't check mapi connectivity. +=item B<--no-copystatus> + +Don't check copy status. + =item B<--timeout> Set timeout time for command execution (Default: 50 sec) @@ -186,6 +193,16 @@ Can used special variables like: %{mailflow_result}, %{database}, %{server} Set critical threshold (Default: '%{mailflow_result} !~ /Success/i'). Can used special variables like: %{mailflow_result}, %{database}, %{server} +=item B<--warning-copystatus> + +Set warning threshold. +Can used special variables like: %{mailflow_result}, %{database}, %{server} + +=item B<--critical-copystatus> + +Set critical threshold (Default: '%{contentindexstate} !~ /Healthy/i'). +Can used special variables like: %{copystatus_indexstate}, %{database}, %{server} + =back =cut \ No newline at end of file diff --git a/apps/exchange/2010/local/mode/imapmailbox.pm b/apps/exchange/2010/local/mode/imapmailbox.pm index 8657d8ef9..e1e186cd5 100644 --- a/apps/exchange/2010/local/mode/imapmailbox.pm +++ b/apps/exchange/2010/local/mode/imapmailbox.pm @@ -49,7 +49,6 @@ sub new { "mailbox:s" => { name => 'mailbox', }, "password:s" => { name => 'password', }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/mode/mapimailbox.pm b/apps/exchange/2010/local/mode/mapimailbox.pm index b5baddbbd..6b9a4ee2f 100644 --- a/apps/exchange/2010/local/mode/mapimailbox.pm +++ b/apps/exchange/2010/local/mode/mapimailbox.pm @@ -48,7 +48,6 @@ sub new { "critical-mapi:s" => { name => 'critical_mapi', default => '%{result} !~ /Success/i' }, "mailbox:s" => { name => 'mailbox', }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/mode/owamailbox.pm b/apps/exchange/2010/local/mode/owamailbox.pm index 4465d8a7c..98c027094 100644 --- a/apps/exchange/2010/local/mode/owamailbox.pm +++ b/apps/exchange/2010/local/mode/owamailbox.pm @@ -51,7 +51,6 @@ sub new { "password:s" => { name => 'password', }, "no-trust-ssl" => { name => 'no_trust_ssl', }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/mode/queues.pm b/apps/exchange/2010/local/mode/queues.pm index 5c04a266c..8e9c88437 100644 --- a/apps/exchange/2010/local/mode/queues.pm +++ b/apps/exchange/2010/local/mode/queues.pm @@ -47,7 +47,6 @@ sub new { "warning:s" => { name => 'warning', }, "critical:s" => { name => 'critical', default => '%{status} !~ /Ready|Active/i' }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/mode/replicationhealth.pm b/apps/exchange/2010/local/mode/replicationhealth.pm new file mode 100644 index 000000000..c0533ca1b --- /dev/null +++ b/apps/exchange/2010/local/mode/replicationhealth.pm @@ -0,0 +1,156 @@ +# +# Copyright 2015 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::exchange::2010::local::mode::replicationhealth; + +use base qw(centreon::plugins::mode); + +use strict; +use warnings; +use centreon::plugins::misc; +use centreon::common::powershell::exchange::2010::replicationhealth; + +sub new { + my ($class, %options) = @_; + my $self = $class->SUPER::new(package => __PACKAGE__, %options); + bless $self, $class; + + $self->{version} = '1.0'; + $options{options}->add_options(arguments => + { + "remote-host:s" => { name => 'remote_host', }, + "remote-user:s" => { name => 'remote_user', }, + "remote-password:s" => { name => 'remote_password', }, + "no-ps" => { name => 'no_ps', }, + "timeout:s" => { name => 'timeout', default => 50 }, + "command:s" => { name => 'command', default => 'powershell.exe' }, + "command-path:s" => { name => 'command_path' }, + "command-options:s" => { name => 'command_options', default => '-InputFormat none -NoLogo -EncodedCommand' }, + "ps-exec-only" => { name => 'ps_exec_only', }, + "warning:s" => { name => 'warning', }, + "critical:s" => { name => 'critical', default => '%{result} !~ /Passed/i' }, + }); + return $self; +} + +sub change_macros { + my ($self, %options) = @_; + + foreach (('warning', 'critical')) { + if (defined($self->{option_results}->{$_})) { + $self->{option_results}->{$_} =~ s/%\{(.*?)\}/\$self->{data}->{$1}/g; + } + } +} + +sub check_options { + my ($self, %options) = @_; + $self->SUPER::init(%options); + + $self->change_macros(); +} + +sub run { + my ($self, %options) = @_; + + my $ps = centreon::common::powershell::exchange::2010::replicationhealth::get_powershell( + remote_host => $self->{option_results}->{remote_host}, + remote_user => $self->{option_results}->{remote_user}, + remote_password => $self->{option_results}->{remote_password}, + no_ps => $self->{option_results}->{no_ps}, + ); + $self->{option_results}->{command_options} .= " " . $ps; + my ($stdout) = centreon::plugins::misc::windows_execute(output => $self->{output}, + timeout => $self->{option_results}->{timeout}, + command => $self->{option_results}->{command}, + command_path => $self->{option_results}->{command_path}, + command_options => $self->{option_results}->{command_options}); + if (defined($self->{option_results}->{ps_exec_only})) { + $self->{output}->output_add(severity => 'OK', + short_msg => $stdout); + $self->{output}->display(nolabel => 1, force_ignore_perfdata => 1, force_long_output => 1); + $self->{output}->exit(); + } + centreon::common::powershell::exchange::2010::replicationhealth::check($self, stdout => $stdout); + + $self->{output}->display(); + $self->{output}->exit(); +} + +1; + +__END__ + +=head1 MODE + +Check replication health. + +=over 8 + +=item B<--remote-host> + +Open a session to the remote-host (fully qualified host name). --remote-user and --remote-password are optional + +=item B<--remote-user> + +Open a session to the remote-host with authentication. This also needs --remote-host and --remote-password. + +=item B<--remote-password> + +Open a session to the remote-host with authentication. This also needs --remote-user and --remote-host. + +=item B<--timeout> + +Set timeout time for command execution (Default: 50 sec) + +=item B<--no-ps> + +Don't encode powershell. To be used with --command and 'type' command. + +=item B<--command> + +Command to get information (Default: 'powershell.exe'). +Can be changed if you have output in a file. To be used with --no-ps option!!! + +=item B<--command-path> + +Command path (Default: none). + +=item B<--command-options> + +Command options (Default: '-InputFormat none -NoLogo -EncodedCommand'). + +=item B<--ps-exec-only> + +Print powershell output. + +=item B<--warning> + +Set warning threshold. +Can used special variables like: %{result}, %{server}, %{isvalid}, %{check} + +=item B<--critical> + +Set critical threshold (Default: '%{result} !~ /Passed/i'). +Can used special variables like: %{result}, %{server}, %{isvalid}, %{check} + +=back + +=cut \ No newline at end of file diff --git a/apps/exchange/2010/local/mode/services.pm b/apps/exchange/2010/local/mode/services.pm index 13985fb5d..6aa384657 100644 --- a/apps/exchange/2010/local/mode/services.pm +++ b/apps/exchange/2010/local/mode/services.pm @@ -47,7 +47,6 @@ sub new { "warning:s" => { name => 'warning', }, "critical:s" => { name => 'critical', default => '%{requiredservicesrunning} =~ /True/i and %{servicesnotrunning} ne ""' }, }); - $self->{thresholds} = {}; return $self; } diff --git a/apps/exchange/2010/local/plugin.pm b/apps/exchange/2010/local/plugin.pm index a1af23559..da3257b7c 100644 --- a/apps/exchange/2010/local/plugin.pm +++ b/apps/exchange/2010/local/plugin.pm @@ -39,6 +39,7 @@ sub new { 'mapi-mailbox' => 'apps::exchange::2010::local::mode::mapimailbox', 'owa-mailbox' => 'apps::exchange::2010::local::mode::owamailbox', 'queues' => 'apps::exchange::2010::local::mode::queues', + 'replication-health' => 'apps::exchange::2010::local::mode::replicationhealth', 'services' => 'apps::exchange::2010::local::mode::services', ); diff --git a/centreon/common/powershell/exchange/2010/databases.pm b/centreon/common/powershell/exchange/2010/databases.pm index b3b4a5095..cd7b8b5b0 100644 --- a/centreon/common/powershell/exchange/2010/databases.pm +++ b/centreon/common/powershell/exchange/2010/databases.pm @@ -31,6 +31,7 @@ sub get_powershell { my $no_mailflow = (defined($options{no_mailflow})) ? 1 : 0; my $no_ps = (defined($options{no_ps})) ? 1 : 0; my $no_mapi = (defined($options{no_mapi})) ? 1 : 0; + my $no_copystatus = (defined($options{no_copystatus})) ? 1 : 0; return '' if ($no_ps == 1); @@ -90,6 +91,13 @@ Foreach ($DB in $MountedDB) { Write-Host "[mailflow=" $MailflowResult.testmailflowresult "][latency=" $MailflowResult.MessageLatencyTime.TotalMilliseconds "]" -NoNewline '; } + if ($no_copystatus == 0) { + $ps .= ' + # Test CopyStatus + $CopyStatusResult = Get-MailboxDatabaseCopyStatus -Identity $DB.Name + Write-Host "[contentindexstate=" $CopyStatusResult.ContentIndexState "][[contentindexerrormessage=" $CopyStatusResult.ContentIndexErrorMessage "]]" -NoNewline +'; + } $ps .= ' } @@ -187,6 +195,45 @@ sub check_mailflow { } } +sub check_copystatus { + my ($self, %options) = @_; + + if (defined($self->{option_results}->{no_copystatus})) { + $self->{output}->output_add(long_msg => ' Skip copy status test'); + return ; + } + + if ($options{line} !~ /\[contentindexstate=(.*?)\]\[\[contentindexerrormessage=(.*?)\]\]/) { + $self->{output}->output_add(long_msg => ' Skip copystatus test (information not found)'); + return ; + } + + ($self->{data}->{copystatus_indexstate}, $self->{data}->{copystatus_indexerror}) = (centreon::plugins::misc::trim($1), centreon::plugins::misc::trim($2)); + $self->{output}->output_add(long_msg => " Copystatus state : " . $self->{data}->{copystatus_indexstate}); + + my ($status, $message) = ('ok'); + eval { + local $SIG{__WARN__} = sub { $message = $_[0]; }; + local $SIG{__DIE__} = sub { $message = $_[0]; }; + + if (defined($self->{option_results}->{critical_copystatus}) && $self->{option_results}->{critical_copystatus} ne '' && + eval "$self->{option_results}->{critical_copystatus}") { + $status = 'critical'; + } elsif (defined($self->{option_results}->{warning_copystatus}) && $self->{option_results}->{warning_copystatus} ne '' && + eval "$self->{option_results}->{warning_copystatus}") { + $status = 'warning'; + } + }; + if (defined($message)) { + $self->{output}->output_add(long_msg => 'filter status issue: ' . $message); + } + if (!$self->{output}->is_status(value => $status, compare => 'ok', litteral => 1)) { + $self->{output}->output_add(severity => $status, + short_msg => sprintf("Server '%s' Database '%s' copystatus state is %s [error: %s]", + $self->{data}->{server}, $self->{data}->{database}, $self->{data}->{copystatus_indexstate}, $self->{data}->{copystatus_indexerror})); + } +} + sub check { my ($self, %options) = @_; # options: stdout @@ -246,6 +293,7 @@ sub check { check_mapi($self, line => $line); check_mailflow($self, line => $line); + check_copystatus($self, line => $line); } if ($checked == 0) { diff --git a/centreon/common/powershell/exchange/2010/replicationhealth.pm b/centreon/common/powershell/exchange/2010/replicationhealth.pm new file mode 100644 index 000000000..5d16a6195 --- /dev/null +++ b/centreon/common/powershell/exchange/2010/replicationhealth.pm @@ -0,0 +1,111 @@ +# +# Copyright 2015 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 centreon::common::powershell::exchange::2010::replicationhealth; + +use strict; +use warnings; +use centreon::plugins::misc; +use centreon::common::powershell::exchange::2010::powershell; + +sub get_powershell { + my (%options) = @_; + # options: no_ps + my $no_ps = (defined($options{no_ps})) ? 1 : 0; + + return '' if ($no_ps == 1); + + my $ps = centreon::common::powershell::exchange::2010::powershell::powershell_init(%options); + + $ps .= ' +try { + $ErrorActionPreference = "Stop" + $results = Test-ReplicationHealth +} catch { + Write-Host $Error[0].Exception + exit 1 +} + +Foreach ($result in $results) { + Write-Host "[server=" $result.Server "][check=" $result.Check "][result=" $result.Result "][isvalid=" $result.IsValid "][[error=" $result.Error "]]" +} +exit 0 +'; + + return centreon::plugins::misc::powershell_encoded($ps); +} + +sub check { + my ($self, %options) = @_; + # options: stdout + + # Following output: + #[Server= XXXX ][check= ReplayService][result= Passed ][isvalid= Yes][[error=...]] + $self->{output}->output_add(severity => 'OK', + short_msg => "All replication health tests are ok."); + + my $checked = 0; + $self->{output}->output_add(long_msg => $options{stdout}); + while ($options{stdout} =~ /\[server=(.*?)\]\[check=(.*?)\]\[result=(.*?)\]\[isvalid=(.*?)\]\[\[error=(.*?)\]\]/msg) { + $self->{data} = {}; + ($self->{data}->{server}, $self->{data}->{check}, $self->{data}->{result}, $self->{data}->{isvalid}, $self->{data}->{error}) = + ($self->{output}->to_utf8($1), centreon::plugins::misc::trim($2), + centreon::plugins::misc::trim($3), centreon::plugins::misc::trim($4), centreon::plugins::misc::trim($5)); + + $checked++; + + my ($status, $message) = ('ok'); + eval { + local $SIG{__WARN__} = sub { $message = $_[0]; }; + local $SIG{__DIE__} = sub { $message = $_[0]; }; + + if (defined($self->{option_results}->{critical}) && $self->{option_results}->{critical} ne '' && + eval "$self->{option_results}->{critical}") { + $status = 'critical'; + } elsif (defined($self->{option_results}->{warning}) && $self->{option_results}->{warning} ne '' && + eval "$self->{option_results}->{warning}") { + $status = 'warning'; + } + }; + if (defined($message)) { + $self->{output}->output_add(long_msg => 'filter status issue: ' . $message); + } + if (!$self->{output}->is_status(value => $status, compare => 'ok', litteral => 1)) { + $self->{output}->output_add(severity => $status, + short_msg => sprintf("Replicatin test '%s' status on '%s' is '%s' [error: %s]", + $self->{data}->{check}, $self->{data}->{server}, $self->{data}->{result}, $self->{data}->{error})); + } + } + + if ($checked == 0) { + $self->{output}->output_add(severity => 'UNKNOWN', + short_msg => 'Cannot find informations'); + } +} + +1; + +__END__ + +=head1 DESCRIPTION + +Method to check Exchange 2010 queues. + +=cut \ No newline at end of file