-
Notifications
You must be signed in to change notification settings - Fork 54
/
Copy pathcheck_netstat.pl
executable file
·820 lines (783 loc) · 35.2 KB
/
check_netstat.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
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
#!/usr/bin/perl
#
# ============================== SUMMARY =====================================
#
# Program : check_netstat.pl
# Version : 0.353
# Date : Mar 23, 2013
# Author : William Leibzon - [email protected]
# Summary : This is a nagios plugin that allows to check number of TCP
# connections on or to given set of ports. It is using
# 'netstat' when run locally or 'snmpnetstat' for remote host.
# 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 is a nagios plugin that checks number of TCP connections from or
# to the system. The plugin gets the data either from local computer with
# 'netstast' or from remote system using 'snmpnetstat' (if '-H' and '-C'
# are not specified then its local).
#
# You can set this up to provide warning and critical values if number of
# connections for specific port out or in is too low or too high. Data is
# also made available for nagios 2.0 post-processing based on specified ports.
#
# This program is written and maintained by:
# William Leibzon - william(at)leibzon.org
#
# =============================== SETUP NOTES ================================
#
# 1. Make sure to check and if necessary adjust the the path to utils.pm
# 2. Make sure you have snmpnetstat in /usr/bin or adjust the path below
# 3. Syntax of snmpnetstat is different depending on which NET-SNMP package
# you have installed. Please specify your version of NET-SNMP package
# as '-N 5.0" or '-N 5.4' if you're unsure use '-v' and see what was
# autodetermined.
# 4. Beware that all service port names are ALWAYS retrieved from system
# running plugin and in case of SNMP check this might not be the same
# as what is set on remote system.
#
# TCP Port names are specified with '-p' or '-a' option for checking number
# of connections based on specified warning and critical values and with '-A'
# for those ports which you want in perfomance output. Each TCP port name
# (or number) should be prefixed with either '>' or '<' to specify if you
# want to check incoming connections to the specified port ('<') or outgoing
# connections ('>') to the specified port on remote system. For example
# using '--ports >smtp' means you want to check on number of outgoing SMTP
# (port 25) connections.
#
# The values retrieved are compared to specified warning and critical levels.
# Warning and critical levels are specified with '-w' and '-c' and each one
# must have exact same number of values (separated by ',') as number of
# attribute (tcp port names) specified with '-p'. Any values you dont want
# to compare you specify as ~. There are also number of other one-letter
# modifiers that can be used before actual data value to direct how data is
# to be checked. These are as follows:
# > : issue alert if data is above this value (default)
# < : issue alert if data is below this value
# = : issue alert if data is equal to this value
# ! : issue alert if data is NOT equal to this value
#
# Supported are also 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 than (or greater than) critical values (it is rarely needed).
#
# Additionally if you want performance output then use '-f' option to get all
# the ports specified with '-p' or specify particular list of of ports for
# performance data with '-A' (this list can include names not found in '-p').
# Note that when reporting perfomance data instead of saying ">smtp' or "<smtp"
# the plugin will report it as 'smtp_out=' or 'smtp_in='.
#
# ============================= SETUP EXAMPLES ===============================
#
# The first example is for your server to check SMTP connections - warnings
# would be sent here fore more then 15 incoming connections or more then 10
# outgoing and critical alerts for more than 40 incoming or 20 outgoing
# ----
# define command {
# command_name check_smtp_connections
# command_line $USER1$/check_netstat.pl -L "SMTP Load" -H $HOSTADDRESS$ -C $ARG1$ -N 5.3 -2 -p "<smtp,>smtp" -w $ARG2$ -c $ARG3$ -f -A "@"
# }
#
# define service{
# use std-service
# service_description SMTP Load
# hostgroups mailserv
# check_command check_smtp_connections!public!">15,>10"!">40,>20"
# }
#
# ----
# The second example is for a webserver to check HTTP connections. In
# this case the server is always little loaded (i.e. you have busy website)
# so not only is their upper bound of 30 for warning and 100 for critical
# but also a bound to send alert if there are < 5 connections (and to
# specify this http name is repeated twice at '-p'; the result is a
# little strange though as it will report "http in connections" twice -
# its on my "todo list" to get this taken care of and only report same
# port/direction information once).
#
# define command {
# command_name check_http_connections
# command_line $USER1$/check_netstat.pl -L "HTTP Load" -H $HOSTADDRESS$ -C $ARG1$ -2 -p "http,http" -w $ARG2$ -c $ARG3$ -A "http,@"
# }
#
# define service{
# use std-service
# service_description HTTP Load
# hostgroups webserv
# check_command check_http_connections!public!"<5,>30"!"~,100"
# }
#
# Note: those who want to minimize load on nagios server and number of extra
# lookups should really combine above into one command if your server
# is doing both HTTP and SMTP (but of course then it might not look
# as nice in the services list)
#
# ====================== VERSION CHANGES AND TODO =============================
#
# Released Versions and Features:
# [0.1] Mar 2006, initial code written as check_tcpconnections.pl plugin
# And around May 2006 (still before official public release):
# a. Sigalarm handler now kills snmpnetstat process at timeout (to avoid zombies)
# [0.2] Oct 2006 (some minor updates later), official public release
# a. Plugin renamed "check_netstat.pl", most of the help above is written then
# b. Allow to use just '-w' or just '-c' (the other is than automaticly '~')
# c. Report an error if getservbyname can not find service
# d. Prefix output with "tcp_" when its just a port number
# e. Feature to add label into output for custom results
# f. Added '-F' option to get performance-only data into main output
# [0.3] Nov 2010, released Dec 2010
# a. Patch by Lars Engels to better support SunOS and FreeBSD
# b. Added '-s' option that is more general than -e and allows to specify
# particular state of TCP sessions to count (somebody wanted CLOSE_WAIT)
# [0.32] Feb 2011 (most changes related to fixing SNMP support)
# a. Fixed bug in a patch applied to 0.3 that caused SNMP checks not to work
# b. Added '-N' option to specify version of NET-SNMP and added code to
# autodetermine this and based on it use correct syntax for snmpnetstat
# This is necessary because from NET-SNMP 5.3 syntax is different and not
# compatible which made this plugin not work with OS released 2010 and later
# c. Added '-r' option to specify protocol. Supported are: TCP,TCP6,UDP,UDP6
# Please note that UDP and UDP6 are not session protocols, so any data
# reported is quite inconsistant
# [0.33] Nov 2011 - Bug fixed. Warning was being issued by Critical conditions.
# Thank you for reporting the bug: Giuliano Ippoliti, Steven Back
# [0.34] Dec 2011 - release and update of documentation
# [0.35] Mar 2012 - Bug fix: updated snmpnetstat options from '-CP' to '-Cp'
# Threshold parsing and check code has been rewritten with new code
# from check_mysqld and now supports ranges in similar way to nagios
# plugin specification. Also plugin now adds threshold info to
# performance data i.e. "var=data:warn:crit" in perfdata
# [0.351] Mar 2012 - Bug fixes in new threshold spec code
# [0.352] Jun 2012 - Another bug fix in threshold check
# [0.353] Mar 2013 - Bug fix in parse_threshold function
#
# TODO (mostly written between 0.1 and 0.2 and still valid):
# 0. Support SNMP v3 (login,authpass,privpass)
# 1. Think what other netstat data is good to check on with nagios
# (unix sockets?)
# 2. Add reporting all incoming connections for all services which system
# is listening on and make sure it works as just '<'.
# And I had code that did it but lost it somewhere except for this doc for it:
#
# Special value of '@' (or just '>') allow to check on
# total number connections (they will be reported as port 'all').
# Even more interesting is another special value of '*' (or just '<') which
# expands to mean all ports your system is listening on [Note: this is
# currently in testing, it'll be released with 0.5 version of the plugin].
# Note that when this is used in '-p' then numbers you specify for warning
# and critical would become default for any such port.
#
# 3. Add capability to report ports that are not within certain range
# (to guess list all 'incoming' ports but discard random ports from
# outgoing connections)
# 4. Fix so that port is reported only once even if listed more
# than once (see example above) this is actually bigger rewrite
# than it may seem at first
#
# ========================== START OF PROGRAM CODE ===========================
use strict;
use Getopt::Long;
use Switch;
# Nagios specific
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);
}
my $snmpnetstat='/usr/bin/snmpnetstat'; # you may want to modify this too
my $Version='0.353';
my $o_host= undef; # hostname
my $o_community= undef; # community
my $o_snmpport= undef; # SNMP port (default 161)
my $o_help= undef; # help option
my $o_timeout= 10; # Default 10s Timeout
my $o_verb= undef; # verbose mode
my $o_version= undef; # version info option
my $o_version2= undef; # use snmp v2c
# SNMPv3 specific - not supported right now
my $o_login= undef; # Login for snmpv3
my $o_passwd= undef; # Pass for snmpv3
my $o_perf= undef; # Performance data option
my $o_perfcopy= undef; # Opposite to o_perf to cause perfomance attributes be listed within main output
my $o_attr= undef; # What port(s) to check (specify more than one separated by '.')
my @o_attrL= (); # array for above list
my @o_attrLn= (); # array of port to check in numeric form
my @o_attrLp= (); # array of attribute name modifiers ('>' or '<')
my $o_perfattr= undef; # List of attributes to only provide values in perfomance data but no checking
my @o_perfattrL=(); # array for above list
my @o_perfattrLp=(); # array of modifiers for perfomance attribute modifiers ('>' or '<')
my @o_perfattrLn=(); # array of ports in numeric form
my $o_warn= undef; # warning level option
my @o_warnLv= (); # array from warn options, before further processing
my @o_warnL= (); # array of warn options, each array element is an array of threshold spec
my $o_crit= undef; # Critical level option
my @o_critLv= (); # array of critical options before processing
my @o_critL= (); # array of critical options, each element is an array of threshold spec
my $o_label= ''; # Label used to show what is in plugin output
my $o_established=undef; # only count established TCP sessions
my $o_state= undef; # Only count TCP sessions in this state
my $o_proto= 'tcp'; # Protocol, default is TCP
my $o_netsnmpv= undef; # Net-SNMP package version
my $netstat_pid=undef;
# For verbose output
sub verb { my $t=shift; print $t,"\n" if defined($o_verb) ; }
sub print_version { print "$0: $Version\n" };
sub print_usage {
print "Usage: $0 [-v] [-H <host> -C <snmp_community> [-2] [-P <snmp port>] [-N <netsnmp version>]] [-t <timeout>] [-r TCP|TCP6|UDP|UDP6] [-p <ports to check> -w <warn levels> -c <crit levels> [-f]] [-A <ports for perfdata> [-F]] [-e] [-s <state> [-V]\n";
}
sub autodetect_netsnmpversion {
my $nver='0.0';
verb("Trying to determine version of Net-SNMP by running: $snmpnetstat -V");
open(SHELL_DATA,"$snmpnetstat -V 2>&1 |");
my $fver=<SHELL_DATA>;
close(SHELL_DATA);
verb("Got: $fver");
if ($fver =~ /version: (\d\.\d)/) {
$nver=$1;
verb("Version autodetermined as: $nver");
}
return $nver;
}
sub isnew_netsnmp {
my $arg = shift;
$arg =~ /(\d)\.(\d)/;
if ($1>5 || ($1 eq 5 && $2>2)) {
verb("NET-SNMP Version later than 5.3");
return 1;
}
return 0;
}
# Return true if arg is a number - in this case negative and real numbers are not allowed
sub isnum {
my $num = shift;
if ( $num =~ /^[+]?(\d*)$/ ) { return 1 ;}
return 0;
}
# help function 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 . " (equal to $lv1)" if $mod eq '=' && $data eq $lv1;
return " " . $attrib . " is " . $data . " (not equal to $lv1)" if $mod eq '!' && $data ne $lv1;
return " " . $attrib . " is " . $data . " (more than $lv1)" if $mod eq '>' && $data>$lv1;
return " " . $attrib . " is " . $data . " (more than $lv2)" if $mod eq ':' && $data>$lv2;
return " " . $attrib . " is " . $data . " (more than or equal $lv1)" if $mod eq '>=' && $data>=$lv1;
return " " . $attrib . " is " . $data . " (less than $lv1)" if ($mod eq '<' || $mod eq ':') && $data<$lv1;
return " " . $attrib . " is " . $data . " (less than or equal $lv1)" if $mod eq '<=' && $data<=$lv1;
return " " . $attrib . " is " . $data . " (in range $lv1..$lv2)" if $mod eq '@' && $data>=$lv1 && $data<=$lv2;
return "";
}
# this is a help function 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], $th_array->[1], $th_array->[2], $th_array->[3], $th_array->[4], $th_array->[5]");
$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
}
sub help {
print "\nNetstat (TCP Connections) Monitor Plugin for Nagios version ",$Version,"\n";
print " by William Leibzon - william(at)leibzon.org\n";
print_usage();
print <<EOD;
-v, --verbose
print extra debugging information
-h, --help
print this help message
-L, --label
Plugin output label
-H, --hostname=HOST
name or IP address of host to check with SNMP (using snmpnetstat)
-C, --community=COMMUNITY NAME
community name for SNMP - can only be used if -H is also specified
-2, --v2c
use SNMP v2 (instead of SNMP v1)
-P, --snmpport=PORT
port number for SNMP - can only be used if -H is specified
-N, --netsnmp_version=VERSION
for SNMP, specify version of your NET-SNMP package here, note that version of
snmpnetstat included with NET-SNMP 5.3 is different than prior versions
-r, --protocol=TCP|TCP6|UDP|UDP6
query the specified protocol, default is TCP
combinations of protocols will be supported in the future, but not right now
-p, --ports=STR[,STR[,STR[..]]] or --attributes=STR[,STR[,[STR[...]]]
Which tcp ports (attributes) to check. The value here can be either
numeric or name that would be come from /etc/services.
The value should be prefixed with:
> : check outgoing TCP connections
< : check incoming TCP connections (default)
Special value of '>\@' or just '>' allow to specify that you
want to check on total number of of TCP connections
(this is reported in perfomance data as 'all_out').
-A, --perf_attributes=STR[,STR[,STR[..]]]
Which tcp ports to add as part of performance data output.
These names can be different than the ones listed in '--ports'
to only output these ports in perf data but not check.
-w, --warn=STR[,STR[,STR[..]]]
Warning level(s) - must be a number
Warning values can have the following prefix modifiers:
> : warn if data is above this value (default)
< : warn if data is below this value
= : warn if data is equal to this value
! : warn if data is not equal to this value
~ : do not check this data (must be by itself)
^ : this disables checks that warning is less than 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[..]]]
critical level(s) (if more than one attrib, must have multiple values)
Critical values can have the same prefix modifiers as warning
(see above) except '^'
-t, --timeout=INTEGER
timeout for SNMP in seconds (Default : 5)
-e, --established_sessions
specifies that listing should include only ESTABLISHED state sessions
This is older option kept for compatibility. Equivalent to: -s='ESTABLISHED'
-s, --state=TCP_STATE
Report only sessions in specific state. The most common states used
here are "ESTABLISHED' and 'CLOSE_WAIT'
-V, --version
prints version number
-f, --perfparse
Used only with '-p'. Causes to output data not only in main status line
but also as perfparse output
-F, --perf_copy
Used only with '-A'. Can be used so that that ports listed '-A' also
get reported as normal plugin output
EOD
}
sub check_options {
my $i;
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_snmpport, 'snmpport:i' => \$o_snmpport,
'C:s' => \$o_community, 'community:s' => \$o_community,
'N:s' => \$o_netsnmpv, 'netsnmp_version:s' => \$o_netsnmpv,
't:i' => \$o_timeout, 'timeout:i' => \$o_timeout,
'V' => \$o_version, 'version' => \$o_version,
'2' => \$o_version2, 'v2c' => \$o_version2,
'L:s' => \$o_label, 'label:s' => \$o_label,
'c:s' => \$o_crit, 'critical:s' => \$o_crit,
'w:s' => \$o_warn, 'warn:s' => \$o_warn,
'f' => \$o_perf, 'perfparse' => \$o_perf,
'F' => \$o_perfcopy, 'perf_copy' => \$o_perfcopy,
'a:s' => \$o_attr, 'attributes:s' => \$o_attr,
'p:s' => \$o_attr, 'ports:s' => \$o_attr,
'r:s' => \$o_proto, 'protocol:s' => \$o_proto,
'A:s' => \$o_perfattr, 'perf_attributes:s' => \$o_perfattr,
'e' => \$o_established, 'established_sessions' => \$o_established,
's:s' => \$o_state, 'state:s' => \$o_state
);
if (defined($o_help) ) { help(); exit $ERRORS{"UNKNOWN"}; }
if (defined($o_version)) { print_version(); exit $ERRORS{"UNKNOWN"}; }
if (defined($o_host))
{
if (!defined($o_community) && (!defined($o_login) || !defined($o_passwd)) )
{ print "Specify SNMP community!\n"; print_usage(); exit $ERRORS{"UNKNOWN"}; }
if (!defined($o_netsnmpv)) { $o_netsnmpv=autodetect_netsnmpversion(); }
$o_netsnmpv=isnew_netsnmp($o_netsnmpv);
}
elsif (defined($o_community) || defined($o_version2) || defined($o_snmpport) ||
defined($o_login) || defined($o_passwd)) {
print "Can not use snmp-specific attributes without specifying host!\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
if (defined($o_perfattr)) {
@o_perfattrL=split(/,/ ,$o_perfattr) if defined($o_perfattr);
for ($i=0; $i<scalar(@o_perfattrL); $i++) {
my ($name,$aliases);
$o_perfattrL[$i] =~ s/^([>|<]?)//;
$o_perfattrLp[$i] = $1;
if ($o_perfattrL[$i] eq '@' || !$o_perfattrL[$i]) {
$o_perfattrLn[$i] = 0;
$o_perfattrL[$i]='all';
}
elsif (isnum($o_perfattrL[$i])) {
$o_perfattrLn[$i] = $o_perfattrL[$i];
} else {
($name, $aliases, $o_perfattrLn[$i]) = getservbyname($o_perfattrL[$i],'tcp');
if ($? || !$o_perfattrLn[$i]) {
print "Failed to find port number for service named \"$o_perfattrL[$i]\"\n";
print_usage();
exit $ERRORS{"UNKNWON"};
}
}
$o_perfattrLp[$i] = '<' if !$o_perfattrLp[$i];
}
}
if (defined($o_warn) || defined($o_crit) || defined($o_attr)) {
if (defined($o_attr)) {
@o_attrL=split(/,/, $o_attr);
@o_warnLv=split(/,/ ,$o_warn) if defined($o_warn);
@o_critLv=split(/,/ ,$o_crit) if defined($o_crit);
}
else {
print "Specifying warning and critical levels requires '-p' parameter with port names\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
if (scalar(@o_warnLv)!=scalar(@o_attrL) || scalar(@o_critLv)!=scalar(@o_attrL)) {
if (scalar(@o_warnLv)==0 && scalar(@o_critLv)==scalar(@o_attrL)) {
verb('Only critical value check is specified - setting warning to ~');
for($i=0;$i<scalar(@o_attrL);$i++) { $o_warnLv[$i]='~'; }
}
elsif (scalar(@o_critLv)==0 && scalar(@o_warnLv)==scalar(@o_attrL)) {
verb('Only warning value check is specified - setting critical to ~');
for($i=0;$i<scalar(@o_attrL);$i++) { $o_critLv[$i]='~'; }
}
else {
printf "Number of specified warning levels (%d) and critical levels (%d) must be equal to the number of attributes specified at '-p' (%d). If you need to ignore some attribute specify it as '~'\n", scalar(@o_warnLv), scalar(@o_critLv), scalar(@o_attrL);
print_usage();
exit $ERRORS{"UNKNOWN"};
}
}
for (my $i=0; $i<scalar(@o_warnLv); $i++) {
$o_warnL[$i] = parse_threshold($o_warnLv[$i]);
$o_critL[$i] = parse_threshold($o_critLv[$i]);
if (threshold_specok($o_warnL[$i],$o_critL[$i])) {
print "Problem with warn threshold '".$o_warnL[$i][5]."' and/or critical threshold '".$o_critL[$i][5]."'\n";
print "All warning and critical values must be numeric or ~. Warning must be less then critical\n";
print "or greater then when '<' is used or within or outside of range for : and @ specification\n";
print "Note: to override less than check prefix warning value with ^\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
}
for ($i=0; $i<scalar(@o_attrL); $i++) {
my ($name,$aliases);
$o_attrL[$i] =~ s/^([>|<]?)//;
$o_attrLp[$i] = $1;
if ($o_attrL[$i] eq '@' || !$o_attrL[$i]) {
$o_attrLn[$i] = 0;
$o_attrL[$i] = 'all';
}
elsif (isnum($o_attrL[$i])) {
$o_attrLn[$i] = $o_attrL[$i];
}
else {
($name, $aliases, $o_attrLn[$i]) = getservbyname($o_attrL[$i],'tcp');
if ($? || !$o_attrLn[$i]) {
print "Failed to find port number for service named \"$o_attrL[$i]\"\n";
print_usage();
exit $ERRORS{"UNKNWON"};
}
}
}
}
if (scalar(@o_attrL)==0 && scalar(@o_perfattrL)==0) {
print "You must specify list of ports with either '-p' ('-a') or '-A'\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
$o_proto =~ tr/A-Z/a-z/;
if ($o_proto ne 'tcp' && $o_proto ne 'tcp6' && $o_proto ne 'udp' && $o_proto ne 'udp6') {
print "Supported protocols are: TCP, TCP6, UDP, UDP6\n";
print_usage();
exit $ERRORS{"UNKNOWN"};
}
}
sub parse_netstatline {
my ($line, $type) = @_;
my @ar = split(/\s+/, $line);
my ($loc, $rem, $state);
switch ($type) {
case 1 {
($loc, $rem, $state) = ($ar[3],$ar[4],$ar[5]);
$loc = $1 if $loc =~ /:(\d+)$/;
$rem = $1 if $rem =~ /:(\d+)$/;
}
case 2 {
($loc, $rem, $state) = ($ar[1],$ar[2],$ar[3]);
$loc = $1 if $loc =~ /\.(\d+)$/;
$rem = $1 if $rem =~ /\.(\d+)$/;
}
case 3 {
($loc, $rem, $state) = ($ar[0],$ar[1],$ar[6]);
$loc = $1 if $loc =~ /\.(\d+)$/;
$rem = $1 if $rem =~ /\.(\d+)$/;
}
case 4 {
($loc, $rem, $state) = ($ar[3],$ar[4],$ar[5]);
$loc = $1 if $loc =~ /\.(\d+)$/;
$rem = $1 if $rem =~ /\.(\d+)$/;
}
}
return ($loc, $rem, $state);
}
sub tcpportname {
my ($name, $inout) = @_;
if (isnum($name)) { return $o_proto.$name."_".$inout; }
else { return $name."_".$inout; }
}
# Get the alarm signal (just in case snmp timeout screws up)
$SIG{'ALRM'} = sub {
print ("ERROR: Alarm signal (Nagios time-out)\n");
kill 9, $netstat_pid if defined($netstat_pid);
exit $ERRORS{"UNKNOWN"};
};
########## MAIN ##############
check_options();
# Check global timeout if something goes wrong
if (defined($TIMEOUT)) {
verb("Alarm at ".($TIMEOUT+10));
alarm($TIMEOUT+10);
} else {
verb("no timeout defined : $o_timeout + 10");
alarm ($o_timeout+10);
}
# next part of the code builds list of attributes to be retrieved
my $statuscode = "OK";
my $statusinfo = "";
my $statusdata = "";
my $perfdata = "";
my $chk = "";
my $i;
my $port_local;
my $port_remote;
my $conn_state;
my %locports=(0=>0);
my %remports=(0=>0);
my $shell_command="";
my $shell_command_auth="";
my $netstat_format;
my $nlines=0;
if (defined($o_host) && defined($o_community)) {
$shell_command = $snmpnetstat . " $o_host -t $o_timeout";
$shell_command.=" -n -P $o_proto" if !$o_netsnmpv;
$shell_command.=" -Cn -Cp $o_proto" if $o_netsnmpv;
$shell_command.= " -v 2c" if defined($o_version2);
$shell_command.= " -p $o_snmpport" if defined($o_snmpport);
$shell_command_auth= "-c $o_community ";
$netstat_format = 2;
}
else {
my $os = `uname`;
chomp $os;
switch ($os) {
case "Linux" {
$shell_command = 'netstat -n';
$netstat_format = 1;
}
case "SunOS" {
$shell_command = 'netstat -n -f inet -P'.$o_proto;
$netstat_format = 3;
}
case "FreeBSD" {
$shell_command = 'netstat -n -f inet';
$netstat_format = 4;
}
else {
die "Unsupported OS!"
}
}
$shell_command_auth = '';
}
verb("Executing $shell_command $shell_command_auth 2>&1");
# I would have prefrerred open3 [# if (!open3($cin, $cout, $cerr, $shell_command))]
# but there are problems when using it within nagios embedded perl
# TODO: Note that putting $shell_command_auth in the command line exposes SNMP community
# for anyone who does 'ps', I should rewrite this to be more secure
$netstat_pid=open(SHELL_DATA, "$shell_command $shell_command_auth 2>&1 |");
if (!$netstat_pid) {
print "UNKNOWN ERROR - could not execute $shell_command - $!";
exit $ERRORS{'UNKNOWN'};
}
while (<SHELL_DATA>) {
$nlines++;
verb("got line: $_");
$conn_state='';
$port_local='';
$port_remote='';
if ($netstat_format == 3) {
if (/^\d{1,3}\./) {
($port_local, $port_remote, $conn_state) = parse_netstatline($_, $netstat_format);
}
}
else { # $netstat_format==1 || $netstat_format==2 || $netstat_format==4
if (/^$o_proto\s/) {
($port_local, $port_remote, $conn_state) = parse_netstatline($_, $netstat_format);
}
}
if ($port_local ne '' && $port_remote ne '') {
verb("local_port: $port_local | remote_port: $port_remote | state: $conn_state");
if ((defined($o_established) && $conn_state eq "ESTABLISHED") ||
(defined($o_state) && $conn_state eq $o_state) ||
(!defined($o_established) && !defined($o_state))) {
$locports{0}++;
if (defined($locports{$port_local})) { $locports{$port_local}++; }
else { $locports{$port_local}=1; }
$remports{0}++;
if (defined($remports{$port_remote})) { $remports{$port_remote}++; }
else { $remports{$port_remote}=1; }
}
}
}
if (!close(SHELL_DATA)) {
print "UNKNOWN ERROR - execution of $shell_command resulted in an error $? - $!";
exit $ERRORS{'UNKNOWN'};
}
if ($nlines eq 0) {
print "UNKNOWN ERROR - did not receive any results from $shell_command";
exit $ERRORS{'UNKNOWN'};
}
# loop to check if warning & critical attributes are ok
for ($i=0;$i<scalar(@o_attrL);$i++) {
if ($o_attrLp[$i] eq '<') {
$locports{$o_attrLn[$i]}=0 if !defined($locports{$o_attrLn[$i]});
if ($chk = check_threshold(tcpportname($o_attrL[$i],"in"),$locports{$o_attrLn[$i]},$o_critL[$i])) {
$statuscode = "CRITICAL";
$statusinfo .= $chk;
}
elsif ($chk = check_threshold(tcpportname($o_attrL[$i],"in"),$locports{$o_attrLn[$i]},$o_warnL[$i])) {
$statuscode="WARNING" if $statuscode eq "OK";
$statusinfo .= $chk;
}
else {
$statusdata .= "," if ($statusdata);
$statusdata .= " ". tcpportname($o_attrL[$i],"in") ." is ". $locports{$o_attrLn[$i]};
}
$perfdata .= " ". tcpportname($o_attrL[$i],"in") ."=". $locports{$o_attrLn[$i]} if defined($o_perf);
}
if ($o_attrLp[$i] eq '>') {
$remports{$o_attrLn[$i]}=0 if !defined($remports{$o_attrLn[$i]});
if ($chk = check_threshold(tcpportname($o_attrL[$i],"out"),$remports{$o_attrLn[$i]},$o_critL[$i])) {
$statuscode = "CRITICAL";
$statusinfo .= $chk;
}
elsif ($chk = check_threshold(tcpportname($o_attrL[$i],"out"),$remports{$o_attrLn[$i]},$o_warnL[$i])) {
$statuscode="WARNING" if $statuscode eq "OK";
$statusinfo .= $chk;
}
else {
$statusdata .= "," if ($statusdata);
$statusdata .= " ". tcpportname($o_attrL[$i],"out") ." is ". $remports{$o_attrLn[$i]};
}
$perfdata .= " ". tcpportname($o_attrL[$i], "out") ."=". $remports{$o_attrLn[$i]} if defined($o_perf);
}
if (defined($o_perf) && 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 '';
}
}
# add data for performance-only attributes
for ($i=0;$i<scalar(@o_perfattrL);$i++) {
if ($o_perfattrLp[$i] eq '<') {
$locports{$o_perfattrLn[$i]}=0 if !defined($locports{$o_perfattrLn[$i]});
$perfdata .= " " . tcpportname($o_perfattrL[$i],"in") ."=". $locports{$o_perfattrLn[$i]};
$statusdata.= "," if ($statusdata) && defined($o_perfcopy);
$statusdata.= " ". tcpportname($o_perfattrL[$i],"in") ." is ". $locports{$o_perfattrLn[$i]} if defined($o_perfcopy);
}
if ($o_perfattrLp[$i] eq '>') {
$remports{$o_perfattrLn[$i]}=0 if !defined($remports{$o_perfattrLn[$i]});
$perfdata .= " " . tcpportname($o_perfattrL[$i],"out") ."=". $remports{$o_perfattrLn[$i]};
$statusdata.= "," if ($statusdata) && defined($o_perfcopy);
$statusdata.= " ". tcpportname($o_perfattrL[$i],"out") ." is ". $remports{$o_perfattrLn[$i]} if defined($o_perfcopy);
}
}
$o_label .= " " if $o_label ne '';
print $o_label . $statuscode;
print " -".$statusinfo if $statusinfo;
print " -".$statusdata if $statusdata;
print " |".$perfdata if $perfdata;
print "\n";
exit $ERRORS{$statuscode};