1
- # !/usr/bin/perl -w
1
+ # !/usr/bin/env perl
2
2
#
3
3
#
4
4
# This program implements a SNMP agent for MySQL servers
19
19
# GNU General Public License for more details.
20
20
#
21
21
# You should have received a copy of the GNU General Public License
22
- # along with this program. If not, see <http://www.gnu.org/licenses/>.
23
- #
22
+ # along with this program. If not, see <http://www.gnu.org/licenses/>.
23
+
24
+ my $VERSION = " 0.7" ;
25
+ $VERSION = eval $VERSION ;
24
26
25
27
use strict;
28
+ use warnings;
29
+ use Carp;
26
30
use Data::Dumper;
27
31
use Unix::Syslog qw( :subs :macros) ;
28
- use Getopt::Long;
32
+ use Getopt::Long qw( :config auto_help auto_version no_ignore_case ) ;
29
33
use POSIX qw( setsid ) ;
30
34
use NetSNMP::OID (' :all' );
31
35
use NetSNMP::agent(' :all' );
37
41
use DBD::mysql;
38
42
use Pod::Usage;
39
43
40
- sub my_snmp_handler ($$$$);
41
44
netsnmp_ds_set_boolean( NETSNMP_DS_APPLICATION_ID,
42
45
NETSNMP_DS_AGENT_NO_ROOT_ACCESS, 1 );
43
46
my $agent = new NetSNMP::agent( ' Name' => ' mysql' , ' AgentX' => 1 );
44
47
45
- my $VERSION = " 0.6" ;
46
-
47
48
my %opt = (
48
- daemon_pid => ' /var/run/mysql-agent.pid' ,
49
- host => ' localhost ' ,
50
- oid => ' 1.3.6.1.4.1.20267.200.1 ' ,
51
- pass => ' ' ,
52
- port => ' 3306 ' ,
53
- refresh => ' 300 ' ,
54
- user => ' monitor ' ,
49
+ daemon_pid => ' /var/run/mysql-agent.pid' ,
50
+ oid => ' 1.3.6.1.4.1.20267.200.1 ' ,
51
+ port => 3306 ,
52
+ refresh => 300 ,
53
+ master => 1 ,
54
+ slave => 0 ,
55
+ innodb => 1 ,
55
56
);
56
57
57
- Getopt::Long::Configure(' no_ignore_case' );
58
58
GetOptions(
59
59
\%opt ,
60
+ ' host|h=s' ,
61
+ ' port|P=i' ,
62
+ ' user|u=s' ,
63
+ ' password|p=s' ,
64
+ ' config|c=s' ,
65
+ ' master|m!' ,
66
+ ' slave|s!' ,
67
+ ' innodb|i!' ,
68
+ ' oid|o' ,
69
+ ' refresh|r=i' ,
60
70
' daemon_pid|daemon-pid=s' ,
61
- ' help|?' ,
62
- ' host=s' ,
63
- ' man' ,
64
- ' master|m' ,
65
71
' no-daemon' ,
66
- ' oid' ,
67
- ' password=s' ,
68
- ' port|P=i' ,
69
- ' refresh|i=i' ,
70
- ' slave' ,
71
- ' user=s' ,
72
+ ' man' ,
72
73
' usage' ,
73
- ' verbose|v+' ,
74
- ' version|V ' ,
75
- ) or pod2usage(-verbose => 0);
74
+ ' verbose|v+' ,
75
+ " version" => sub { VersionMessage() } ,
76
+ ) or pod2usage( -verbose => 0 );
76
77
77
- pod2usage(-verbose => 0) if $opt {usage };
78
- pod2usage(-verbose => 1) if $opt {help };
79
- pod2usage(-verbose => 2) if $opt {man };
78
+ pod2usage( -verbose => 0 ) if $opt {usage };
79
+ pod2usage( -verbose => 1 ) if $opt {help };
80
+ pod2usage( -verbose => 2 ) if $opt {man };
80
81
81
- if ( $opt {version } ) {
82
- print " mysql-agent.pl $VERSION by brice.figureau\@ daysofwonder.com\n " ;
83
- exit ;
82
+ sub VersionMessage {
83
+ print " mysql-agent $VERSION by brice.figureau\@ daysofwonder.com\n " ;
84
84
}
85
85
86
- my $debugging = $opt {verbose };
87
- my $subagent = 0;
88
- my $chk_innodb = 1; # Do you want to check InnoDB statistics?
89
- my $chk_master = 1; # Do you want to check binary logging?
90
- my $chk_slave = 0; # Do you want to check slave status?
86
+ my $debugging = $opt {verbose };
87
+ my $subagent = 0;
88
+
89
+ my $dsn = ' DBI:mysql:' ;
90
+ if ($opt {config }) {
91
+ $dsn .= " mysql_read_default_file=$opt {config}" ;
92
+ }
93
+ else {
94
+ $dsn .= join (' ;' , " host=$opt {host}" , " port=$opt {port}" );
95
+ }
91
96
92
- my $dsn = " DBI:mysql:host=$opt {host};port=$opt {port}" ;
93
97
my $running = 0;
94
98
my $error = 0;
95
99
96
- # prototypes
97
- sub daemonize ();
98
- sub dolog ($$);
99
-
100
100
openlog( " mysql-agent" , LOG_PID | LOG_PERROR, LOG_DAEMON );
101
101
102
102
daemonize() if !$opt {' no-daemon' };
105
105
my $global_last_refresh = 0;
106
106
107
107
# enterprises.20267.200.1
108
- my $regOID = new NetSNMP::OID($opt {oid });
108
+ my $regOID = new NetSNMP::OID( $opt {oid } );
109
109
$agent -> register( " mysql" , $regOID , \&my_snmp_handler );
110
110
111
111
# various types & definitions
278
278
}
279
279
280
280
# takes only numbers from a string
281
- sub tonum ($) {
281
+ sub tonum {
282
282
my $str = shift ;
283
283
return 0 if !$str ;
284
284
return $1 if $str =~ m / (\d +)/ ;
285
285
return 0;
286
286
}
287
287
288
288
# return a string to build a 64 bit number
289
- sub make_bigint_sql ($$) {
290
- my $hi = shift ;
291
- my $lo = shift ;
289
+ sub make_bigint_sql {
290
+ my ($hi , $lo ) = @_ ;
292
291
return " (($hi << 32) + $lo )" ;
293
292
}
294
293
295
- sub max ($$) {
296
- my $a = shift ;
297
- my $b = shift ;
294
+ sub max {
295
+ my ($a , $b ) = @_ ;
298
296
return $a if $a > $b ;
299
297
return $b ;
300
298
}
301
299
302
300
# daemonize the program
303
- sub daemonize () {
301
+ sub daemonize {
304
302
open STDIN , ' /dev/null' or die " mysql-agent: can't read /dev/null: $! " ;
305
303
open STDOUT , ' >/dev/null'
306
304
or die " mysql-agent: can't write to /dev/null: $! " ;
307
305
defined ( my $pid = fork ) or die " mysql-agent: can't fork: $! " ;
308
306
if ($pid ) {
309
307
310
308
# parent
311
- open PIDFILE , ' >' , $opt {daemon_pidfile }
312
- or die " $0 : can 't write to $opt {daemon_pidfile} : $! \n " ;
313
- print PIDFILE " $pid \n " ;
314
- close (PIDFILE) ;
309
+ open my $pidfile , ' >' , $opt {daemon_pid }
310
+ or croak " Couldn 't open $opt {daemon_pid} for writing : $! " ;
311
+ print { $pidfile } " $pid \n " or croak " Couldn't write pid to $opt {daemon_pid}: $! " ;
312
+ close $pidfile or croak " Couldn't close $opt {daemon_pid}: $! " ;
315
313
exit ;
316
314
}
317
315
@@ -326,10 +324,7 @@ sub fetch_mysql_data {
326
324
my ( $datasource , $dbuser , $dbpass ) = @_ ;
327
325
my %output ;
328
326
eval {
329
- my $dbh
330
- = DBI-> connect ( $datasource , $dbuser , $dbpass ,
331
- { RaiseError => 1, AutoCommit => 1 } );
332
-
327
+ my $dbh = DBI-> connect ( $datasource , $dbuser , $dbpass , { RaiseError => 1, AutoCommit => 1 } );
333
328
if ( !$dbh ) {
334
329
dolog( LOG_CRIT, " Can't connect to database: $datasource , $@ " );
335
330
return ;
@@ -360,7 +355,7 @@ sub fetch_mysql_data {
360
355
$status { $row -> [0] } = $row -> [1];
361
356
}
362
357
363
- if ($chk_slave ) {
358
+ if ($opt { slave } ) {
364
359
$result = $dbh -> selectall_arrayref(" SHOW SLAVE STATUS" );
365
360
foreach my $row (@$result ) {
366
361
@@ -393,7 +388,7 @@ sub fetch_mysql_data {
393
388
394
389
# Get info on master logs.
395
390
my @binlogs = (0);
396
- if ( $chk_master && $status {' log_bin' } eq ' ON' ) { # See issue #8
391
+ if ( $opt { master } && $status {' log_bin' } eq ' ON' ) { # See issue #8
397
392
$result = $dbh -> selectall_arrayref( " SHOW MASTER LOGS" ,
398
393
{ Slice => {} } );
399
394
foreach my $row (@$result ) {
@@ -419,7 +414,7 @@ sub fetch_mysql_data {
419
414
my @spin_rounds ;
420
415
my @os_waits ;
421
416
422
- if ( $chk_innodb && $status {' have_innodb' } eq ' YES' ) {
417
+ if ( $opt { innodb } && $status {' have_innodb' } eq ' YES' ) {
423
418
my $innodb_array
424
419
= $dbh -> selectall_arrayref(
425
420
" SHOW /*!50000 ENGINE*/ INNODB STATUS" ,
@@ -628,18 +623,19 @@ sub refresh_status {
628
623
my $startOID = shift ;
629
624
my $now = time ();
630
625
631
- # Check if we have been called quicker than once every $refresh_interval
632
- if ( ( $now - $global_last_refresh ) < $opt {refresh_interval } ) {
626
+ # Check if we have been called quicker than once every $refresh
627
+ if ( ( $now - $global_last_refresh ) < $opt {refresh } ) {
633
628
634
629
# if yes, do not do anything
635
630
dolog( LOG_DEBUG,
636
631
" not refreshing: "
637
632
. ( $now - $global_last_refresh )
638
- . " < $opt {refresh_interval }" )
633
+ . " < $opt {refresh }" )
639
634
if ($debugging );
640
635
return ;
641
636
}
642
- my ( $oid , $types , $status ) = fetch_mysql_data( $dsn , $opt {user }, $opt {pass } );
637
+ my ( $oid , $types , $status )
638
+ = fetch_mysql_data( $dsn , $opt {user }, $opt {pass } );
643
639
if ($oid ) {
644
640
dolog( LOG_DEBUG, " Setting error to 0" ) if ($debugging );
645
641
$error = 0;
@@ -710,7 +706,7 @@ sub set_value {
710
706
}
711
707
}
712
708
713
- sub my_snmp_handler ($$$$) {
709
+ sub my_snmp_handler {
714
710
my ( $handler , $registration_info , $request_info , $requests ) = @_ ;
715
711
my ($request );
716
712
@@ -770,7 +766,7 @@ ($$$$)
770
766
dolog( LOG_DEBUG, " finished processing" ) if ($debugging );
771
767
}
772
768
773
- sub dolog ($$) {
769
+ sub dolog {
774
770
my ( $level , $msg ) = @_ ;
775
771
syslog( $level , $msg );
776
772
print STDERR $msg . " \n " if ($debugging );
@@ -783,35 +779,40 @@ ($$)
783
779
$SIG {' TERM' } = \&shut_it_down;
784
780
$running = 1;
785
781
while ($running ) {
786
- refresh_status($opt {oid });
782
+ refresh_status( $opt {oid } );
787
783
$agent -> agent_check_and_process(1); # 1 = block
788
784
}
789
785
$agent -> shutdown ();
790
786
791
787
dolog( LOG_INFO, " agent shutdown" );
792
788
793
789
__END__
790
+
794
791
=head1 NAME
795
792
796
793
mysql-agent - report mysql statistics via SNMP
797
794
798
795
=head1 SYNOPSIS
799
796
800
- mysql-agent.pl [options]
797
+ mysql-agent [options]
801
798
802
799
-h HOST, --host=HOST connect to MySQL DB on HOST
800
+ -P PORT, --port=PORT port to connect (default 3306)
803
801
-u USER, --user=USER use USER as user to connect to mysql
804
802
-p PASS, --password=PASS use PASS as password to connect to mysql
805
- -P PORT, --port=PORT port to connect (default 3306)
806
- --daemon-pid=FILE write PID to FILE instead of $default{pid}
807
- -n, --no-daemon do not detach and become a daemon
803
+ -c FILE, --config=FILE read mysql connection details from FILE
808
804
-m, --master check master
809
805
-s, --slave check slave
806
+ -i, --innodb read innodb settings
810
807
-o OID, --oid=OID registering OID
811
- -i INT, --refresh=INT set refresh interval to INT (seconds)
808
+ -r INT, --refresh=INT set refresh interval to INT (seconds)
809
+ --daemon-pid=FILE write PID to FILE instead of $default{pid}
810
+ -n, --no-daemon do not detach and become a daemon
811
+ -v, --verbose be verbose about what you do
812
+
812
813
-?, --help display this help and exit
814
+ --usage display detailed usage information
813
815
--man display program man page
814
- -v, --verbose be verbose about what you do
815
816
-V, --version output version information and exit
816
817
817
818
=head1 OPTIONS
@@ -822,6 +823,10 @@ =head1 OPTIONS
822
823
823
824
connect to MySQL DB on HOST
824
825
826
+ =item B<-P PORT, --port=PORT >
827
+
828
+ port to connect (default 3306)
829
+
825
830
=item B<-u USER, --user=USER >
826
831
827
832
use USER as user to connect to mysql
@@ -830,17 +835,17 @@ =head1 OPTIONS
830
835
831
836
use PASS as password to connect to mysql
832
837
833
- =item B<-P PORT, --port=PORT >
834
-
835
- port to connect (default 3306)
838
+ =item B<-c FILE, --config=FILE >
836
839
837
- =item B< --daemon-pid= FILE>
840
+ read mysql connection details from file FILE.
838
841
839
- write PID to FILE instead of $default{pid}
842
+ These should be stored in a section named [client]. Eg:
840
843
841
- =item B<-n, --no-daemon >
842
-
843
- do not detach and become a daemon
844
+ [client]
845
+ host=dbserver
846
+ port=3306
847
+ user=monitor
848
+ password=secret
844
849
845
850
=item B<-m, --master >
846
851
@@ -850,26 +855,42 @@ =head1 OPTIONS
850
855
851
856
check slave
852
857
858
+ =item B<-i, --innodb >
859
+
860
+ check innodb details
861
+
853
862
=item B<-o OID, --oid=OID >
854
863
855
864
registering OID
856
865
857
- =item B<-i INT, --refresh=INT >
866
+ =item B<-r INT, --refresh=INT >
858
867
859
868
refresh interval in seconds
860
869
861
- =item B<-?, --help >
870
+ =item B<--daemon-pid=FILE >
862
871
863
- Print a brief help message and exits.
872
+ write PID to FILE instead of $default{pid}
864
873
865
- =item B<--man >
874
+ =item B<-n, --no-daemon >
866
875
867
- Prints the manual page and exits.
876
+ do not detach and become a daemon
868
877
869
878
=item B<-v, --verbose >
870
879
871
880
be verbose about what you do
872
881
882
+ =item B<--man >
883
+
884
+ Prints the manual page and exits.
885
+
886
+ =item B<--usage >
887
+
888
+ Prints detailed usage information and exits.
889
+
890
+ =item B<-?, --help >
891
+
892
+ Print a brief help message and exits.
893
+
873
894
=item B<-V, --version >
874
895
875
896
output version information and exit
@@ -881,4 +902,4 @@ =head1 DESCRIPTION
881
902
B<mysql-agent > is a small daemon that connects to a local snmpd daemon
882
903
to report statistics on a local or remote MySQL server.
883
904
884
- =cut
905
+ =cut
0 commit comments