-
Notifications
You must be signed in to change notification settings - Fork 54
/
check_mysqld.pl
executable file
·756 lines (712 loc) · 34.5 KB
/
check_mysqld.pl
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
#!/usr/bin/perl -w
#
# ============================== SUMMARY =====================================
#
# Program : check_mysqld.pl
# Version : 0.94
# Date : Mar 31, 2012
# Author : William Leibzon - [email protected]
# Licence : GPL - summary below, full text at http://www.fsf.org/licenses/gpl.txt
#
# =========================== PROGRAM LICENSE =================================
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
#
# ===================== INFORMATION ABOUT THIS PLUGIN =========================
#
# This MYSQL check plugin issues "SHOW STATUS" commmand on mysql server
# and can issue alerts if selected parameters are above/below given number
# It also returns perfomance data for further nagios 2.0 post-processing
# and you can find graph templates NagiosGrapher & PNP4Nagios at
# http://william.leibzon.org/nagios/
#
# This program is written and maintained by:
# William Leibzon - william(at)leibzon.org
#
# ============================= SETUP NOTES ====================================
#
# Make sure to install perl DBI and mysql DBD modules, if necessary specify
# path where these can be found and optionally path to utils.pm
#
# There are several ways to specify mysql database access information
# (hostname, port, user,password) for this plugin:
# 1. Specify these as command-line options when calling plugin, this is by fa
# most common way. You'd either use same user/pass and specify it in nagios
# command.cfg or parameters in service definition or custom MYSQL macros
# to specify username and password inside nagios host or service config file.
# NOTE HOWEVER, that this has serious security problems as anyone who has
# login access to nagios server can see user & pass in process list when
# this plugin is being executed
# 2. Modify this file and specify MYSQL user & pass inside this plugin file
# (you can find it after documentation and before rest of the script code)
# and make sure all mysql servers you check have this user. This is good
# way to do it, but sometimes not all mysql servers can be setup with same
# user credentials for nagios access. You also have to remember to update
# this plugin if new version comes out and remember to not have read-all
# unix permissions which is default.
# 3. Specify login credentials in [client] section of my.cnf file and use
# -F option. You can use common user/pass in each file and specify
# host with -H or may even decide to use different files for each host
# being checked specifying this command like this:
# define command{
# command_name check_mysqld
# command_line $USER1$/check_mysqld.pl -F /path/to/configs/my_$HOSTADDRESS$.cfg -T -a uptime,threads_connected,questions,slow_queries,open_tables -w ",,,," -c ",,,,"
# }
#
# The attributes/parameters checked are mysql internal variables retuned from
# "SHOW STATUS" or "SHOW GLOBAL STATUS" (which one depends on mysql version).
# For each attribute to be checked which you specify in '-a' you must
# also specify warning and critical threshold value in the same order
# in the '-w' and '-c' options (see below for more details).
#
# Using '-f' option causes all attribute values to be available for Nagios
# performance processing. You may also directly specify which variables to be
# return as performance data and with '-A' option. Special value of '*' as in
# '-A *' allows to get all variables from 'SHOW STATUS' as performance data.
#
# Warning and critical thresholds are specified with '-w' and '-c' and each
# one must have exact same number of values to be checked (separated by ',')
# as number of variables specified with '-a'. Any values you dont want to
# compare you specify as ~ (or just not specify a value, i.e. ',,').
#
# There are also number of other one-letter modifiers that can be used
# as prefix before actual data value to direct how data is to be checked.
# These prefixes are as follows:
# > : issue alert if data is above this value (default for numeric value)
# < : issue alert if data is below this value (must be followed by number)
# = : issue alert if data is equal to this value (default for non-numeric)
# ! : issue alert if data is NOT equal to this value
#
# Additionally supported are two specifications of range formats:
# number1:number2 issue alert if data is OUTSIDE of range [number1..number2]
# i.e. alert if data<$number1 or data>$number2
# @number1:number2 issue alert if data is WITHIN range [number1..number2]
# i.e. alert if data>=$number and $data<=$number2
#
# A special modifier '^' can also be used to disable checking that warn values
# are less then (or greater then) critical values (it is rarely needed).
# A quick example of specialized use is '--warn=^<100 --crit=>200' which means
# warning alert if value is < 100 and critical alert if its greater then 200.
#
# You can specify more then one type of threshold for the same variable -
# simply repeat the variable more then once in the list in '-a'
#
# --------------------------------------------------------------------------
#
# If you're using version 5.02 or newer of mysql server then this plugin
# will do "SHOW GLOBAL STATUS" rather than "SHOW STATUS". For more information
# on differences see:
# http://dev.mysql.com/doc/refman/5.0/en/server-status-variables.html
#
# Note that it maybe the case that you do actually want "SHOW STATUS" even with
# mysqld 5.0.2, then specify query line with '-q' option like
# ./check_mysqld.pl -p foo -f -u nagios -A uptime,threads_connected,slow_queries,open_tables -H nagios -q 'SHOW STATUS'
#
# Sample command and service definitions:
#
# define command{
# command_name check_mysqld
# command_line $USER1$/check_mysqld.pl -H $HOSTADDRESS$ -u $ARG1$ -p $ARG2$ -f -T -a uptime,threads_connected,questions,slow_queries,open_tables -w ",,,," -c ",,,,"
# }
#
# define service {
# use service-critical
# hostgroup_name mysql
# service_description MYSQLD
# check_command check_mysqld!foo!apples
# }
#
# Examples of command-line use:
# /usr/lib/nagios/plugins/check_mysqld.pl -v
# /usr/lib/nagios/plugins/check_mysqld.pl -p foo -f -u nagios -T -a uptime,threads_connected,questions,slow_queries,open_tables -A threads_running,innodb_row_lock_time_avg -w ",,,," -c ",,,,>25" -H nagios -v
#
# ======================= VERSION HISTORY and TODO ================================
#
# This version history is incomplete as I only started keeping track of it as
# a separate section recently. This is one of my earliest plugins started in 2004.
#
# It was originally based on previous plugins by:
# Matthew Kent (matt at bravenet.com) - added perfomance data from mysqladmin
# Mitch Wright - original check_mysql.pl plugin using mysqladmin command
# Code from their plugins was modified to use DBI rather than mysqladmin and
# substantially rewritten, none of their code probably survives in this version.
#
# [0.7 - May 2006] First public release. Very limited threshold specification.
# [0.8 - Dec 2007] Update to my latest format & code for parsing
# warning & critical parameters so they can be of the form
# "<value" (alert if value is less then specified),
# ">value" (alert if its greatr then specified, default)
# as well as "1,~,1" or '1,,1' (ignore 2nd)
# Note: for backward compatibility specifying '0' for threshold is
# more or less same as '~', if you really want 0 then prefix
# it with '<' or '>' or '=' to force the check
# Number of other code cleanup and fixes. This includes allowing
# more then one threshold for same variable (repeat the variable in
# the list of variables with 'a') but making sure that in performance
# variables its listed only once and in proper order.
# [0.85 - Dec 2007] Thanks to the suggestion by Mike Lykov plugin can now do
# 'SHOW GLOBAL STATUS' since mysqld newer then 5.0.2
# will only report one session data in 'SHOW STATUS'. Also in
# part to allow override default behavior (choosing which SHOW
# to do) and to extend plugin functionality a new option '-q'
# is added that allows to specify which query to execute. This
# will let plugin be used with others types of mysql 'SHOW' commands
# [0.851 - Dec 2007] Fixed bug: -t (timeout) parameter was ignored before
# [0.9 - Jan 2008] Threshold parsing and check code has been rewritten and now
# supports ranges in similar way to nagios plugin specification.
# [0.901 - Jan 2008] Added forced closing of db connection at alarm
# [0.902 - Jan 2008] Bug fixed in comparison of non-numeric data
# [0.91 - Dec 2011] Bug fixes. If unable to connect check if TCP port is even open
# [0.92 - Feb 2012] Patch by Diego Salvi to add -F option that allows to use
# mysql configuration file for login credentials
# [0.93 - Mar 2012] Modified threshold checking to assume bare number is 0:number threshold
# and report WARN/CRIT if data value is <0 as per nagios plugin spec
# Added threshold info to performance data i.e. "var=data:warn:crit"
# [0.94 - Mar 2012] Added -T option that measures connection time (in floating seconds)
# and allows to set thresholds on it. Bug fixes for new threshold code.
#
# TODO or consider for future:
# 1. Add support for storing values in a file or reusing old performance
# values passed as command-line parameters (as I did with check_megaraid
# check_megaraid and check_snmp_netint) and allow to check on rate of
# change rather then actual value.
#
# ============================ START OF PROGRAM CODE =============================
use strict;
use IO::Socket;
use Time::HiRes;
use Getopt::Long qw(:config no_ignore_case);
# default mysql hostname, port, database, user and password, see NOTES above
my $HOSTNAME= 'localhost';
my $PORT= 3306;
my $DATABASE= '';
my $USERNAME= 'mysql';
my $PASSWORD= '';
# default mysql configuration file
my $MY_CNF= '';
# Add path to additional libraries if necessary
use lib '/usr/local/mysql/modules';
use lib '/usr/lib/nagios/plugins';
our $TIMEOUT;
our %ERRORS;
eval 'use utils qw(%ERRORS $TIMEOUT)';
if ($@) {
$TIMEOUT = 20;
%ERRORS = ('OK'=>0,'WARNING'=>1,'CRITICAL'=>2,'UNKNOWN'=>3,'DEPENDENT'=>4);
}
# ============= MAIN PROGRAM CODE - DO NOT MODIFY BELOW THIS LINE ==============
use DBI;
my $o_host= undef; # hostname
my $o_port= undef; # port
my $o_dbname= undef; # database
my $o_login= undef; # Database user
my $o_passwd= undef; # Password
my $o_help= undef; # help option
my $o_verb= undef; # verbose mode
my $o_version= undef; # version info option
my $o_variables=undef; # list of variables for warn and critical
my @o_varsL= (); # array from above list
my $o_perfvars= undef; # list of variables to include in perfomance data
my @o_perfvarsL=(); # array from above list
my $o_warn= undef; # warning level option
my @o_warnL= (); # array of warn data processing threshold
my $o_crit= undef; # Critical level option
my @o_critL= (); # array of critical data processing thresholds
my $o_perf= undef; # Performance data option
my $o_timeout= undef; # Timeout to use - note that normally timeout is take from nagios anyway
my $o_replication=undef; # Normal replication status value, if something else then you'll see CRITICAL error
my $o_slave= undef; # Normal slave status, if something else then you'll see CRITICAL error
my $o_query= undef; # Query to execute instead of default SHOW STATUS
my $o_my_cnf= undef; # mysql default file
my $o_timecheck=undef; # threshold spec for connection time
my $Version='0.94';
my $dbh= undef; # DB connection object
sub p_version { print "check_mysqld version : $Version\n"; }
sub print_usage {
print "Usage: $0 [-v] [-H <host> [-P <port>]] [-u <username>] [-q <query>] [-p <password>] [-F <mysql configuration file>] [-a <mysql variables> -w <variables warning thresholdz> -c <variables critical thresholds>] [-A <performance output variables>] [-s <expected slave status>] [-r <expected replication status>] [-T [conntime_warn,conntime_crit]] [-f] [-t <timeout>] [-V]\n";
}
sub help {
print "\nMySQL Database Monitor for Nagios version ",$Version,"\n";
print " by William Leibzon - william(at)leibzon.org\n\n";
print "This monitoring script connects to database, does 'SELECT VERSION' and then\n";
print "'SHOW STATUS' or 'SHOW GLOBAL STATUS' (for version 5.0.2 or newer of mysql)\n";
print "or other query you specify with '-q'. It then allows to select variables\n";
print "from 'SHOW STATUS' and check them against critical and warning thresholds(s)\n";
print "Status data can also be used for performance output.\n\n";
print_usage();
print <<EOT;
-v, --verbose
print extra debugging information
-h, --help
Print this detailed help screen
-H, --hostname=ADDRESS
Hostname or IP Address to check
-P, --port=INTEGER
MySQL port number (default: 3306)
-D, --database=STRING
Database name to login to
(Default is none and you almost never need to change this)
-u, --username=STRING
Connect using the indicated username (Default is 'mysql')
-p, --password=STRING
Use the indicated password to authenticate the connection
(Default is empty password)
==> IMPORTANT: THIS FORM OF AUTHENTICATION IS NOT SECURE!!! <==
Your clear-text password will be visible as a process table entry
-t, --timeout=NUMBER
Allows to set timeout for execution of this plugin. This overrides nagios default.
-F, --mysql-default-file=STRING
MySQL my.cnf like file, data in [client] section will be read to find mysql port
and used as credentials to connect to database. Poviding a configuration file will
override values specified in the script header (default 'localhost', 'mysql' user).
You can still specify parameters through script options to override my.cnf data.
-q, --query=STRING
Specify query to do instead of 'SHOW STATUS'. This is useful to check
some other specialized mysql commands and tables. In order to work
returned data should be similar to one from "SHOW STATUS", i.e.
table with two columns.
-a, --variables=STRING,[STRING,[STRING...]]
List of variables as found in 'SHOW STATUS' which should be monitored.
The list can be arbitrarily long and the default (if option is not used)
is not to monitor any variable. You can repeat same variable if you need
it checked for both below and above thresholds.
-w, --warn=STR[,STR[,STR[..]]]
This option can only be used if '--variables' (or '-a') option above
is used and number of values listed here must exactly match number
of variables specified with '-a'. The values specify warning threshold
for when Nagios should send WARNING alert. These values are usually
numbers and can have the following prefix modifiers:
> - warn if data is above this value (default for numeric values)
< - warn if data is below this value (must be followed by number)
= - warn if data is equal to this value (default for non-numeric values)
! - warn if data is not equal to this value
~ - do not check this data (must not be followed by number or ':')
^ - for numeric values this disables check that warning < critical
Threshold values can also be specified as range in two forms:
num1:num2 - warn if data is outside range i.e. if data<num1 or data>num2
\@num1:num2 - warn if data is in range i.e. data>=num1 && data<=num2
-c, --crit=STR[,STR[,STR[..]]]
This option can only be used if '--variables' (or '-a') option above
is used and number of values listed here must exactly match number of
variables specified with '-a'. The values specify critical threshold
for when Nagios should send CRITICAL alert. The format is exactly same
as with -w option except no '^' prefix.
-s, --slave=status
If slave status (normally it is 'OFF') is anything other then what is
specified with, then CRITICAL alert would be sent. This can also be done
with '=' option so seperate option is kept for backward compatibility
-r, --replication=status
If replication status (normally it is NULL) is anything other then what
is specified with this option, then CRITICAL alert would be sent.
-T, --connection_time=[WARN,CRIT]
When used as just -T the plugin will measure and output connection response
time in seconds. With -f this would also be provided on perf variables.
You can also specify values for this parameter, these are interprted as
WARNING and CRITICAL thresholds (separated by ','). The format for WARN
and CRIT is same as what you would use in -w and -c.
-f, --perfparse
This should only be used with '-a', '-s' or '-r' and causes to output
variable data not only as part of main status line but also as
perfparse compatible output (for graphing, etc).
-A, --perfvars=STRING,[STRING,[STRING...]]
This allows to list variables which values will go only into perfparse
output (and not for threshold checking). A special value of '*' allows
to output all variables from 'SHOW STATUS' or 'SHOW GLOBAL STATUS'.
-V, --version
Prints version number
EOT
print "\nThere are no required arguments. By default, the local 'mysql' database";
print "from a server listening on MySQL standard port 3306 will be checked\n\n";
}
# For verbose output
sub verb { my $t=shift; print $t,"\n" if defined($o_verb) ; }
# Return true if arg is a number
sub isnum {
my $num = shift;
if (defined($num) && $num =~ /^[-|+]?((\d+\.?\d*)|(^\.\d+))$/ ) { return 1 ;}
return 0;
}
# this function is used when checking data against critical and warn values
sub check_threshold {
my ($attrib, $data, $th_array) = @_;
my $mod = $th_array->[0];
my $lv1 = $th_array->[1];
my $lv2 = $th_array->[2];
# verb("debug check_threshold: $mod : ".(defined($lv1)?$lv1:'')." : ".(defined($lv2)?$lv2:''));
return "" if !defined($lv1) || ($mod eq '' && $lv1 eq '');
return " " . $attrib . " is " . $data . " = " . $lv1 if $mod eq '=' && $data eq $lv1;
return " " . $attrib . " is " . $data . " != " . $lv1 if $mod eq '!' && $data ne $lv1;
return " " . $attrib . " is " . $data . " > " . $lv1 if $mod eq '>' && $data>$lv1;
return " " . $attrib . " is " . $data . " > " . $lv2 if $mod eq ':' && $data>$lv2;
return " " . $attrib . " is " . $data . " >= " . $lv1 if $mod eq '>=' && $data>=$lv1;
return " " . $attrib . " is " . $data . " < " . $lv1 if ($mod eq '<' || $mod eq ':') && $data<$lv1;
return " " . $attrib . " is " . $data . " <= " . $lv1 if $mod eq '<=' && $data<=$lv1;
return " " . $attrib . " is " . $data . " in range $lv1..$lv2" if $mod eq '@' && $data>=$lv1 && $data<=$lv2;
return "";
}
# this function is called when parsing threshold options data
sub parse_threshold {
my $thin = shift;
# link to an array that holds processed threshold data
# array: 1st is type of check, 2nd is value2, 3rd is value2, 4th is option, 5th is nagios spec string representation for perf out
my $th_array = [ '', undef, undef, '', '' ];
my $th = $thin;
my $at = '';
$at = $1 if $th =~ s/^(\^?[@|>|<|=|!]?~?)//; # check mostly for my own threshold format
$th_array->[3]='^' if $at =~ s/\^//; # deal with ^ option
$at =~ s/~//; # ignore ~ if it was entered
if ($th =~ /^\:([-|+]?\d+\.?\d*)/) { # :number format per nagios spec
$th_array->[1]=$1;
$th_array->[0]=($at !~ /@/)?'>':'<=';
$th_array->[5]=($at !~ /@/)?('~:'.$th_array->[1]):($th_array->[1].':');
}
elsif ($th =~ /([-|+]?\d+\.?\d*)\:$/) { # number: format per nagios spec
$th_array->[1]=$1;
$th_array->[0]=($at !~ /@/)?'<':'>=';
$th_array->[5]=($at !~ /@/)?'':'@';
$th_array->[5].=$th_array->[1].':';
}
elsif ($th =~ /([-|+]?\d+\.?\d*)\:([-|+]?\d+\.?\d*)/) { # nagios range format
$th_array->[1]=$1;
$th_array->[2]=$2;
if ($th_array->[1] > $th_array->[2]) {
print "Incorrect format in '$thin' - in range specification first number must be smaller then 2nd\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
$th_array->[0]=($at !~ /@/)?':':'@';
$th_array->[5]=($at !~ /@/)?'':'@';
$th_array->[5].=$th_array->[1].':'.$th_array->[2];
}
if (!defined($th_array->[1])) {
$th_array->[0] = ($at eq '@')?'<=':$at;
$th_array->[1] = $th;
$th_array->[5] = '~:'.$th_array->[1] if ($th_array->[0] eq '>' || $th_array->[0] eq '>=');
$th_array->[5] = $th_array->[1].':' if ($th_array->[0] eq '<' || $th_array->[0] eq '<=');
$th_array->[5] = '@'.$th_array->[1].':'.$th_array->[1] if $th_array->[0] eq '=';
$th_array->[5] = $th_array->[1].':'.$th_array->[1] if $th_array->[0] eq '!';
}
if ($th_array->[0] =~ /[>|<]/ && !isnum($th_array->[1])) {
print "Numeric value required when '>' or '<' are used !\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
# verb("debug parse_threshold: $th_array->[0] and $th_array->[1]");
$th_array->[0] = '=' if !$th_array->[0] && !isnum($th_array->[1]) && $th_array->[1] ne '';
if (!$th_array->[0] && isnum($th_array->[1])) { # this is just the number by itself, becomes 0:number check per nagios guidelines
$th_array->[2]=$th_array->[1];
$th_array->[1]=0;
$th_array->[0]=':';
$th_array->[5]=$th_array->[2];
}
return $th_array;
}
# this function checks that for numeric data warn threshold is within range of critical threshold
# where within range depends on actual threshold spec and normally just means less
sub threshold_specok {
my ($warn_thar,$crit_thar) = @_;
return 0 if (defined($warn_thar->[1]) && !isnum($warn_thar->[1])) || (defined($crit_thar->[1]) && !isnum($crit_thar->[1]));
return 1 if defined($warn_thar) && defined($warn_thar->[1]) &&
defined($crit_thar) && defined($crit_thar->[1]) &&
isnum($warn_thar->[1]) && isnum($crit_thar->[1]) &&
$warn_thar->[0] eq $crit_thar->[0] &&
(!defined($warn_thar->[3]) || $warn_thar->[3] !~ /\^/) &&
(!defined($crit_thar->[3]) || $crit_thar->[3] !~ /\^/) &&
(($warn_thar->[1]>$crit_thar->[1] && ($warn_thar->[0] =~ />/ || $warn_thar->[0] eq '@')) ||
($warn_thar->[1]<$crit_thar->[1] && ($warn_thar->[0] =~ /</ || $warn_thar->[0] eq ':')) ||
($warn_thar->[0] eq ':' && $warn_thar->[2]>=$crit_thar->[2]) ||
($warn_thar->[0] eq '@' && $warn_thar->[2]<=$crit_thar->[2]));
return 0; # return with 0 means specs check out and are ok
}
# parse command line options
sub check_options {
Getopt::Long::Configure ("bundling");
GetOptions(
'v' => \$o_verb, 'verbose' => \$o_verb,
'h' => \$o_help, 'help' => \$o_help,
'H:s' => \$o_host, 'hostname:s' => \$o_host,
'P:i' => \$o_port, 'port:i' => \$o_port,
'D:s' => \$o_dbname, 'database:s' => \$o_dbname,
'u:s' => \$o_login, 'username:s' => \$o_login,
'p:s' => \$o_passwd, 'password:s' => \$o_passwd,
't:i' => \$o_timeout, 'timeout:i' => \$o_timeout,
'V' => \$o_version, 'version' => \$o_version,
'r:s' => \$o_replication, 'replication:s' => \$o_replication,
'T:s' => \$o_timecheck, 'reponse_time:s' => \$o_timecheck,
's:s' => \$o_slave, 'slave:s' => \$o_slave,
'a:s' => \$o_variables, 'variables:s' => \$o_variables,
'c:s' => \$o_crit, 'critical:s' => \$o_crit,
'w:s' => \$o_warn, 'warn:s' => \$o_warn,
'f:s' => \$o_perf, 'perfparse:s' => \$o_perf,
'A:s' => \$o_perfvars, 'perfvars:s' => \$o_perfvars,
'q:s' => \$o_query, 'query:s' => \$o_query,
'F:s' => \$o_my_cnf, 'mysql-default-file:s' => \$o_my_cnf,
);
if (defined($o_help)) { help(); exit $ERRORS{"UNKNOWN"} };
if (defined($o_version)) { p_version(); exit $ERRORS{"UNKNOWN"} };
# below code is common for number of my plugins, including check_snmp_?, netstat, etc
# it is mostly compliant with nagios threshold specification (except use of '~')
# and adds number of additional format options using '>','<','!','=' prefixes
my (@ar_warnLv,@ar_critLv);
@o_perfvarsL=split( /,/ , lc $o_perfvars ) if defined($o_perfvars) && $o_perfvars ne '*';
if (defined($o_warn) || defined($o_crit) || defined($o_variables) || (defined($o_timecheck)&& $o_timecheck ne '')) {
if (defined($o_variables)) {
@o_varsL=split( /,/ , lc $o_variables );
if (defined($o_warn)) {
$o_warn.="~" if $o_warn =~ /,$/;
@ar_warnLv=split( /,/ , lc $o_warn );
}
if (defined($o_crit)) {
$o_crit.="~" if $o_crit =~ /,$/;
@ar_critLv=split( /,/ , lc $o_crit );
}
}
elsif (!defined($o_timecheck)) {
print "Specifying warning and critical levels requires '-a' parameter with list of STATUS variables\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
if (defined($o_timecheck) && $o_timecheck ne '') {
my @o_timeth=split(/,/, lc $o_timecheck);
verb("Processing timecheck thresholds: $o_timecheck");
if (scalar(@o_timeth)!=2) {
printf "Incorrect value '%s' for Connection Time Thresholds. Connection time threshold must include both warning and critical thresholds separated by ','\n", $o_timecheck;
print_usage();
exit $ERRORS{"UNKNOWN"};
}
unshift(@o_varsL,"connection_time");
unshift(@ar_warnLv,$o_timeth[0]);
unshift(@ar_critLv,$o_timeth[1]);
}
if (scalar(@ar_warnLv)!=scalar(@o_varsL) || scalar(@ar_critLv)!=scalar(@o_varsL)) {
printf "Number of specified warning levels (%d) and critical levels (%d) must be equal to the number of attributes specified at '-a' (%d). If you need to ignore some attribute do it as ',,'\n", scalar(@ar_warnLv), scalar(@ar_critLv), scalar(@o_varsL);
verb("Warning Levels: ".join(",",@ar_warnLv));
verb("Critical Levels: ".join(",",@ar_critLv));
print_usage();
exit $ERRORS{"UNKNOWN"};
}
for (my $i=0; $i<scalar(@o_varsL); $i++) {
$o_warnL[$i] = parse_threshold($ar_warnLv[$i]);
$o_critL[$i] = parse_threshold($ar_critLv[$i]);
if (threshold_specok($o_warnL[$i],$o_critL[$i])) {
print "All numeric warning values must be less then critical (or greater then when '<' is used)\n";
print "Note: to override this check prefix warning value with ^\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
}
}
if (defined($o_my_cnf) && ! -e $o_my_cnf) {
print "Mysql config file $o_my_cnf specified with -F does not exist\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
# if (scalar(@o_varsL)==0 && scalar(@o_perfvarsL)==0) {
# print "You must specify list of attributes with either '-a' or '-A'\n";
# print_usage();
# exit $ERRORS{"UNKNOWN"};
# }
$HOSTNAME = $o_host if defined($o_host);
$PORT = $o_port if defined($o_port);
$DATABASE = $o_dbname if defined($o_dbname);
$USERNAME = $o_login if defined($o_login);
$PASSWORD = $o_passwd if defined($o_passwd);
$TIMEOUT = $o_timeout if defined($o_timeout);
$MY_CNF = $o_my_cnf if defined($o_my_cnf);
}
# Get the alarm signal (just in case nagios screws up)
$SIG{'ALRM'} = sub {
$dbh->disconnect() if defined($dbh);
print ("ERROR: Alarm signal (Nagios time-out)\n");
exit $ERRORS{"UNKNOWN"};
};
########## MAIN #######
check_options();
# Check global timeout if plugin screws up
if (defined($TIMEOUT)) {
verb("Alarm at $TIMEOUT");
alarm($TIMEOUT);
}
else {
verb("no timeout defined : $o_timeout + 10");
alarm ($o_timeout+10);
}
my $start_time;
my $dsn = "DBI:mysql:";
if ($MY_CNF) {
# Why this? Because if mysql_read_default_file has been provided
# We mustn't overwrite parameters eventually provided from conf
# file with default ones.
$HOSTNAME = '' unless defined($o_host);
$PORT = '' unless defined($o_port);
$DATABASE = '' unless defined($o_dbname);
$USERNAME = '' unless defined($o_login);
$PASSWORD = '' unless defined($o_passwd);
}
$dsn.="database=$DATABASE" if $DATABASE;
$dsn.=":";
$dsn.="host=$HOSTNAME" if $HOSTNAME;
$dsn.=":port=$PORT" if $PORT;
$dsn.=";mysql_read_default_file=$MY_CNF" if $MY_CNF;
verb("connecting using credentials specified in $MY_CNF file") if $MY_CNF;
my $vstr = "connecting";
$vstr .= " to database '".$DATABASE."'" if $DATABASE;
$vstr .= " on host '".$HOSTNAME."'" if $HOSTNAME;
$vstr .= " with user '".$USERNAME."'" if $USERNAME;
verb($vstr) if $DATABASE || $HOSTNAME || $USERNAME;
$start_time = [ Time::HiRes::gettimeofday() ] if defined($o_timecheck);
$dbh = DBI->connect($dsn,$USERNAME,$PASSWORD, { PrintError => 0 } );
if (!$dbh) {
if (!$MY_CNF) {
my $sock = new IO::Socket::INET(
PeerAddr => $HOSTNAME,
PeerPort => $PORT,
Proto => 'tcp',
);
if ($sock) {
close($sock);
print "CRITICAL ERROR - Unable to connect to database '$DATABASE' on server '$HOSTNAME' on port $PORT with user '$USERNAME' - $DBI::errstr\n";
}
else {
print "CRITICAL ERROR - Can not connect to '$HOSTNAME' on port $PORT\n";
}
}
else {
print "CRITICAL ERROR - Can not connect with credentials specified in $MY_CNF - $DBI::errstr\n";
}
exit $ERRORS{"CRITICAL"};
}
my $db_command="SELECT VERSION()";
verb ("Mysql Query: $db_command");
my $sth=$dbh->prepare($db_command);
if (!$sth->execute()) {
print "CRITICAL ERROR - Unable to execute '$db_command' on server '$HOSTNAME' connected as user '$USERNAME' - $DBI::errstr\n";
exit $ERRORS{"CRITICAL"};
}
my ($mysql_version)=$sth->fetchrow_array();
if ($sth->err) {
print "CRITICAL ERROR - Error retrieving data for '$db_command' on server '$HOSTNAME' connected as user '$USERNAME' - $sth->err\n";
exit $ERRORS{"CRITICAL"};
}
$sth->finish();
my @mvnum=(0,0,0);
@mvnum=($1,$2,$3) if $mysql_version =~ /(\d+)\.(\d+)\.(\d+)/;
verb("Mysql Data: $mysql_version | Numeric: $mvnum[0].$mvnum[1].$mvnum[2]");
if (defined($o_query)) {
$db_command=$o_query;
}
elsif ($mvnum[0]>5 || ($mvnum[0]==5 && ($mvnum[1]>0 || ($mvnum[1]==0 && $mvnum[2]>1)))) {
$db_command = 'SHOW GLOBAL STATUS';
}
else {
$db_command = 'SHOW STATUS';
}
verb("Mysql Query: $db_command");
$sth=$dbh->prepare($db_command);
if (!$sth->execute()) {
print "CRITICAL ERROR - Unable to execute '$db_command' on server '$HOSTNAME' connected as user '$USERNAME' - $DBI::errstr\n";
exit $ERRORS{"CRITICAL"};
}
my %dataresults;
my $statuscode = "OK";
my $statusinfo = "";
my $statusdata = "";
my $perfdata = "";
my $mysql_vname;
my $mysql_value;
my $chk = "";
my $i;
# load all data from mysql into internal hash array
$dataresults{$_} = [undef, 0, 0] foreach(@o_varsL);
$dataresults{$_} = [undef, 0, 0] foreach(@o_perfvarsL);
while (($mysql_vname,$mysql_value)=$sth->fetchrow_array()) {
$mysql_vname =~ tr/[A-Z]/[a-z]/ ;
$mysql_value='NULL' if !defined($mysql_value);
verb("Mysql Data: $mysql_vname = $mysql_value");
$dataresults{$mysql_vname}[0] = $mysql_value if exists($dataresults{$mysql_vname});
if (defined($o_perfvars) && $o_perfvars eq '*') { # this adds all status variables variables into performance data when -A '*' is used
$dataresults{$mysql_vname} = [$mysql_value, 0, 0];
push @o_perfvarsL, $mysql_vname;
}
if (defined($o_replication) && $mysql_vname eq 'rpl_status') {
$statuscode = 'CRITICAL' if $o_replication && ($mysql_value eq 'NULL' || $mysql_value ne $o_replication);
$statusinfo .= " rpl_status=" . $mysql_value;
}
if (defined($o_slave) && $mysql_vname eq 'slave_running') {
$statuscode = 'CRITICAL' if $o_slave && ($mysql_value eq 'NULL' || $mysql_value ne $o_slave);
$statusinfo .= " slave_running=" . $mysql_value;
}
}
if ($sth->err) {
$statuscode = 'CRITICAL';
$statusinfo = "error $sth->err during value retrival " . $statusinfo;
}
$sth->finish();
$dbh->disconnect();
if (defined($o_timecheck)) {
$dataresults{'connection_time'}=[0,0,0] if !defined('connection_time');
$dataresults{'connection_time'}[0]=Time::HiRes::tv_interval($start_time);;
$statusdata .= sprintf(" connection_time=%.3fs", $dataresults{'connection_time'}[0]);
$dataresults{'connection_time'}[1]++;
if ($o_timecheck eq '' && defined($o_perf)) {
$perfdata .= ' connection_time=' . $dataresults{'connection_time'}[0].'s';
}
}
# main loop to check if warning & critical attributes are ok
for ($i=0;$i<scalar(@o_varsL);$i++) {
if (defined($dataresults{$o_varsL[$i]}[0])) {
if ($chk = check_threshold($o_varsL[$i],lc $dataresults{$o_varsL[$i]}[0],$o_critL[$i])) {
$dataresults{$o_varsL[$i]}[1]++;
$statuscode = "CRITICAL";
$statusinfo .= $chk;
}
elsif ($chk = check_threshold($o_varsL[$i],lc $dataresults{$o_varsL[$i]}[0],$o_warnL[$i])) {
$dataresults{$o_varsL[$i]}[1]++;
$statuscode="WARNING" if $statuscode eq "OK";
$statusinfo .= $chk;
}
if ($dataresults{$o_varsL[$i]}[1]==0) {
$dataresults{$o_varsL[$i]}[1]++;
$statusdata .= " " . $o_varsL[$i] . "=" . $dataresults{$o_varsL[$i]}[0];
}
if (defined($o_perf) && $dataresults{$o_varsL[$i]}[2]==0) {
$dataresults{$o_varsL[$i]}[2]++;
$perfdata .= " " . $o_varsL[$i] . "=" . $dataresults{$o_varsL[$i]}[0];
if (defined($o_warnL[$i][5]) && defined($o_critL[$i][5])) {
$perfdata .= ';' if $o_warnL[$i][5] ne '' || $o_critL[$i][5] ne '';
$perfdata .= $o_warnL[$i][5] if $o_warnL[$i][5] ne '';
$perfdata .= ';'.$o_critL[$i][5] if $o_critL[$i][5] ne '';
}
}
}
else {
$statuscode="CRITICAL";
$statusinfo .= " $o_varsL[$i] data is missing";
}
}
# add data for performance-only attributes
for ($i=0;$i<scalar(@o_perfvarsL);$i++) {
if (defined($dataresults{$o_perfvarsL[$i]}[0]) && $dataresults{$o_perfvarsL[$i]}[2]==0) {
$perfdata .= " " . $o_perfvarsL[$i] . "=" . $dataresults{$o_perfvarsL[$i]}[0];
$dataresults{$o_perfvarsL[$i]}[2]++;
}
}
# now output the results
print "MYSQL " . $mysql_version . " " . $statuscode . $statusinfo;
print " -" . $statusdata if $statusdata;
print " |" . $perfdata if $perfdata;
print "\n";
# end exit
exit $ERRORS{$statuscode};