1195 lines
35 KiB
Perl
1195 lines
35 KiB
Perl
#!/usr/bin/perl
|
|
# Pandora FMS Plugin for MySQL
|
|
# (c) Artica ST 2015
|
|
# v1.4, 19 Feb 2015
|
|
# ----------------------------------------------------------------------
|
|
|
|
# Default lib dir for RPM and DEB packages
|
|
BEGIN { push @INC, '/usr/lib/perl5'; }
|
|
|
|
use strict;
|
|
use Data::Dumper;
|
|
use POSIX qw(setsid strftime);
|
|
use POSIX;
|
|
use Time::Local;
|
|
|
|
#---------------------------- Global parameters -----------------------#
|
|
|
|
# OS and OS version
|
|
my $OS = $^O;
|
|
|
|
# Store original PATH
|
|
my $ORIGINAL_PATH = $ENV{'PATH'};
|
|
|
|
# FLUSH in each IO
|
|
$| = 1;
|
|
|
|
# Conf file divided line by line
|
|
my @config_file;
|
|
# Conf filename received by command line
|
|
my $archivo_cfg = $ARGV[0];
|
|
# Hash with this plugin setup
|
|
my %plugin_setup;
|
|
# Array with different block checks
|
|
my @checks;
|
|
|
|
# ----------------------------------------------------------------------
|
|
# parse_dosline (line)
|
|
#
|
|
# This cleans DOS-like line and cleans ^M character. VERY Important when
|
|
# you process .conf edited from DOS
|
|
# ----------------------------------------------------------------------
|
|
sub parse_dosline ($){
|
|
my $str = $_[0];
|
|
|
|
$str =~ s/\r//g;
|
|
return $str;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# load_external_setup (config file)
|
|
#
|
|
# Loads a config file
|
|
# ----------------------------------------------------------------------
|
|
sub load_external_setup ($){
|
|
my $archivo_cfg = $_[0];
|
|
my $buffer_line;
|
|
|
|
# Collect items from config file and put in an array
|
|
if (! open (CFG, "< $archivo_cfg")) {
|
|
print "[ERROR] Error opening configuration file $archivo_cfg: $!.\n";
|
|
logger ("[ERROR] Error opening configuration file $archivo_cfg: $!");
|
|
exit 0;
|
|
}
|
|
|
|
while (<CFG>) {
|
|
$buffer_line = parse_dosline ($_);
|
|
# Parse configuration file, this is specially difficult because can contain SQL code, with many things
|
|
if ($buffer_line !~ /^\#/) { # begins with anything except # (for commenting)
|
|
if ($buffer_line =~ m/(.+)\s(.*)/) {
|
|
push @config_file, $buffer_line;
|
|
}
|
|
}
|
|
}
|
|
|
|
close (CFG);
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# logger_begin (message)
|
|
# Beggining of logging to file
|
|
# ----------------------------------------------------------------------
|
|
sub logger_begin ($) {
|
|
my ($message) = @_;
|
|
|
|
my $file = "/tmp/pandora_mysql";
|
|
|
|
if (! open (FILE, "> $file")) {
|
|
print "[ERROR] Could not open logfile '$file' \n";
|
|
logger ("[ERROR] Error opening logfile $file");
|
|
exit 0;
|
|
}
|
|
|
|
# open (FILE, "> $file") or die "[FATAL] Could not open logfile '$file'";
|
|
print FILE strftime ("%Y-%m-%d %H:%M:%S", localtime()) . " >> " . $message . "\n";
|
|
close (FILE);
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# logger (message)
|
|
# Log to file
|
|
# ----------------------------------------------------------------------
|
|
sub logger ($) {
|
|
my ($message) = @_;
|
|
|
|
my $file = "/tmp/pandora_mysql";
|
|
|
|
if (! open (FILE, ">> $file")) {
|
|
print "[ERROR] Could not open logfile '$file' \n";
|
|
logger ("[ERROR] Error opening logfile $file");
|
|
exit 0;
|
|
}
|
|
|
|
print FILE strftime ("%Y-%m-%d %H:%M:%S", localtime()) . " >> " . $message . "\n";
|
|
close (FILE);
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# trim (string)
|
|
#
|
|
# Erase blank spaces before and after the string
|
|
# ----------------------------------------------------------------------
|
|
sub trim ($) {
|
|
my $string = shift;
|
|
$string =~ s/^\s+//;
|
|
$string =~ s/\s+$//;
|
|
|
|
return $string;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# clean_blank (string)
|
|
#
|
|
# This function return a string without blankspaces, given a simple text
|
|
# string
|
|
# ----------------------------------------------------------------------
|
|
sub clean_blank($) {
|
|
my $input = $_[0];
|
|
$input =~ s/[\s\r\n]*//g;
|
|
|
|
return $input;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# print_module (module name, module type, module value, module
|
|
# description, [module severity])
|
|
#
|
|
# This function returns a XML module part for the plugin
|
|
# ----------------------------------------------------------------------
|
|
sub print_module ($$$$;$) {
|
|
my $MODULE_NAME = $_[0];
|
|
my $MODULE_TYPE = $_[1];
|
|
my $MODULE_VALUE = $_[2];
|
|
my $MODULE_DESC = $_[3];
|
|
my $MODULE_SEVERITY = $_[4];
|
|
|
|
# If not a string type, remove all blank spaces and check for not value returned!
|
|
if ($MODULE_TYPE !~ m/string/) {
|
|
$MODULE_VALUE = clean_blank($MODULE_VALUE);
|
|
}
|
|
|
|
print "<module>\n";
|
|
print "<name>$MODULE_NAME</name>\n";
|
|
print "<type>$MODULE_TYPE</type>\n";
|
|
print "<data><![CDATA[$MODULE_VALUE]]></data>\n";
|
|
print "<description><![CDATA[$MODULE_DESC]]></description>\n";
|
|
if (defined($MODULE_SEVERITY)) {
|
|
print "<status>$MODULE_SEVERITY</status>\n";
|
|
} else {
|
|
print "<status>NORMAL</status>\n";
|
|
}
|
|
|
|
print "</module>\n";
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# parse_config
|
|
#
|
|
# This function load configuration tokens and store in a global hash
|
|
# called %plugin_setup accesible on all program.
|
|
# ----------------------------------------------------------------------
|
|
sub parse_config {
|
|
|
|
my $check_block = 0;
|
|
my $parametro;
|
|
|
|
# Some default options
|
|
$plugin_setup{"conf_mysql_homedir"} = "/var/lib/mysql";
|
|
$plugin_setup{"conf_mysql_basedir"} = "/var/lib/mysql";
|
|
$plugin_setup{"conf_temp"} = "/tmp";
|
|
$plugin_setup{"numchecks"} = 0;
|
|
$plugin_setup{"conf_mysql_version"} = "5.5";
|
|
|
|
foreach (@config_file) {
|
|
$parametro = $_;
|
|
|
|
if ($parametro =~ m/^include\s(.*)/i) {
|
|
load_external_setup ($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_version\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_version"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_user\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_user"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_pass\s(.*)/i) {
|
|
my $temp = $1;
|
|
$temp =~ s/\"//g;
|
|
$plugin_setup{"conf_mysql_pass"} = trim($temp);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_host\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_host"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_homedir\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_homedir"} = trim($1);
|
|
}
|
|
if ($parametro =~ m/^conf\_mysql\_basedir\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_basedir"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_mysql\_logfile\s(.*)/i) {
|
|
$plugin_setup{"conf_mysql_logfile"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_temp\s(.*)/i) {
|
|
$plugin_setup{"conf_temp"} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^conf\_logparser\s(.*)/i) {
|
|
$plugin_setup{"conf_logparser"} = trim($1);
|
|
}
|
|
|
|
# Detect begin of check definition
|
|
if ($parametro =~ m/^check\_begin/i) {
|
|
$check_block = 1;
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'unknown';
|
|
}
|
|
|
|
############### Specific check block parsing ###################
|
|
if ($check_block == 1) {
|
|
|
|
if ($parametro =~ m/^check\_end/i) {
|
|
$check_block = 0;
|
|
$plugin_setup{"numchecks"}++;
|
|
}
|
|
|
|
# Try to parse check type (System parameters)
|
|
if ($parametro =~ m/^check\_mysql\_service/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_service';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_mysql\_memory/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_memory';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_mysql\_cpu/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_cpu';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_system\_timewait/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'system_timewait';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_system\_diskusage/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'system_diskusage';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_mysql\_ibdata1/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_ibdata1';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_mysql\_logs/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_logs';
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_mysql\_connect/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'mysql_connection';
|
|
}
|
|
|
|
|
|
# Try to parse check type (Performance parameters)
|
|
if ($parametro =~ m/^mysql\_status\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'status';
|
|
$checks[$plugin_setup{"numchecks"}]{'show'} = trim($1);
|
|
}
|
|
|
|
# Try to parse check type (Open SQL interface)
|
|
if ($parametro =~ m/^check\_sql\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'sql';
|
|
$checks[$plugin_setup{"numchecks"}]{'sql'} = trim($1);
|
|
}
|
|
|
|
# Try to parse check type (Query Innodb status)
|
|
if ($parametro =~ m/^mysql\_innodb\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'type'} = 'innodb';
|
|
$checks[$plugin_setup{"numchecks"}]{'query'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_name\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'check_name'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_type\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'check_type'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^check\_schema\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'check_schema'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^module\_type\s(generic_data|generic_data_inc|generic_data_string|generic_proc|async_string|async_proc|async_data)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'module_type'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^post\_condition\s+(==|!=|<|>)\s+(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'postcondition_op'} = trim($1);
|
|
$checks[$plugin_setup{"numchecks"}]{'postcondition_val'} = trim($2);
|
|
}
|
|
|
|
if ($parametro =~ m/^post\_execution\s(.*)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'postexecution'} = trim($1);
|
|
}
|
|
|
|
if ($parametro =~ m/^data\_absolute/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'return_type'} = 'data_absolute';
|
|
}
|
|
|
|
if ($parametro =~ m/^data\_delta/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'return_type'} = 'data_delta';
|
|
}
|
|
|
|
if ($parametro =~ m/^post\_status\s(WARNING|CRITICAL)/i) {
|
|
$checks[$plugin_setup{"numchecks"}]{'status'} = trim($1);
|
|
}
|
|
|
|
} # Check block
|
|
|
|
}
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_service
|
|
#
|
|
# This function check MySQL service status
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_service {
|
|
my ($mysql_service_result1, $mysql_service_result2, $mysql_service_result3, $mysql_service_result4);
|
|
my $mysql_service_result = 0;
|
|
|
|
# Try different flavours of mysql status
|
|
$mysql_service_result1 = `service mysql status 2> /dev/null`;
|
|
$mysql_service_result2 = `service mysqld status 2> /dev/null`;
|
|
$mysql_service_result3 = `/etc/init.d/mysql status 2> /dev/null`;
|
|
$mysql_service_result4 = `/etc/init.d/mysqld status 2> /dev/null`;
|
|
|
|
|
|
#print "1 >> " . $mysql_service_result1 . "\n";
|
|
#print "2 >> " . $mysql_service_result2 . "\n";
|
|
#print "3 >> " . $mysql_service_result3 . "\n";
|
|
#print "4 >> " . $mysql_service_result4 . "\n";
|
|
|
|
# If any of these ones is up then check result eq OK!
|
|
if (($mysql_service_result1 =~ /start/i) || ($mysql_service_result1 =~ /running/i) ||
|
|
($mysql_service_result2 =~ /start/i) || ($mysql_service_result2 =~ /running/i) ||
|
|
($mysql_service_result3 =~ /start/i) || ($mysql_service_result3 =~ /running/i) ||
|
|
($mysql_service_result4 =~ /start/i) || ($mysql_service_result4 =~ /running/i)) {
|
|
|
|
# OK!
|
|
$mysql_service_result = 1;
|
|
|
|
} else {
|
|
|
|
# Fail
|
|
$mysql_service_result = 0;
|
|
|
|
}
|
|
|
|
logger("[INFO] check_mysql_service with result: " . $mysql_service_result);
|
|
|
|
return $mysql_service_result;
|
|
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_memory
|
|
#
|
|
# This function check MySQL memory usage
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_memory {
|
|
my $mysql_memory_result = 0;
|
|
|
|
# Depends on OS
|
|
if ($OS =~ /hpux/i){
|
|
$mysql_memory_result = `ps -eo comm,pmem | grep -v "grep" | grep "mysqld --basedir" | awk '{print \$2}'`;
|
|
} else {
|
|
$mysql_memory_result = `ps aux | grep -v "grep" | grep "mysqld --basedir" | awk '{print \$4}'`;
|
|
}
|
|
|
|
logger("[INFO] check_mysql_memory with result: " . $mysql_memory_result);
|
|
|
|
return $mysql_memory_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_cpu
|
|
#
|
|
# This function check MySQL cpu usage
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_cpu {
|
|
my $mysql_cpu_result = 0;
|
|
|
|
# Depends on OS
|
|
if ($OS =~ /hpux/i){
|
|
$mysql_cpu_result = `ps -eo comm,pcpu | grep -v "grep" | grep "mysqld --basedir" | awk '{print \$2}'`;
|
|
} else {
|
|
$mysql_cpu_result = `ps aux | grep -v "grep" | grep "mysqld --basedir" | awk '{print \$3}'`;
|
|
}
|
|
|
|
logger("[INFO] check_mysql_cpu with result: " . $mysql_cpu_result);
|
|
|
|
return $mysql_cpu_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_system_timewait
|
|
#
|
|
# This function check system timewait
|
|
# ----------------------------------------------------------------------
|
|
sub check_system_timewait {
|
|
my $system_timewait_result = 0;
|
|
|
|
$system_timewait_result = `netstat -ntu | grep "TIME_WAIT" | wc -l`;
|
|
|
|
logger("[INFO] check_system_timewait with result: " . $system_timewait_result);
|
|
|
|
return $system_timewait_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_system_diskusage
|
|
#
|
|
# This function check system disk usage
|
|
# ----------------------------------------------------------------------
|
|
sub check_system_diskusage {
|
|
my $system_diskusage_result_tmp = 0;
|
|
my $system_diskusage_result = 0;
|
|
|
|
if (!defined($plugin_setup{"conf_mysql_homedir"})) {
|
|
logger("[INFO] system_diskusage_result check needs conf_mysql_homedir token defined in configuration file.");
|
|
return 0;
|
|
}
|
|
|
|
my $system_diskusage_result_tmp = `df -k "$plugin_setup{'conf_mysql_homedir'}" | awk '{print \$5}' | tail -1 | tr -d "%"`;
|
|
|
|
$system_diskusage_result = 100 - $system_diskusage_result_tmp;
|
|
|
|
logger("[INFO] check_system_diskusage with result: " . $system_diskusage_result);
|
|
|
|
return $system_diskusage_result;
|
|
}
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_ibdata1
|
|
#
|
|
# This function check ibdata1 file disk usage
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_ibdata1 {
|
|
my $mysql_ibdata1_result = 0;
|
|
|
|
if (!defined($plugin_setup{"conf_mysql_basedir"})) {
|
|
logger("[INFO] check_mysql_ibdata1 check needs conf_mysql_basedir token defined in configuration file.");
|
|
}
|
|
|
|
$mysql_ibdata1_result = `du -k $plugin_setup{"conf_mysql_basedir"}/ibdata1 | tail -1 | awk '{ print \$1 }'`;
|
|
|
|
logger("[INFO] check_mysql_ibdata1 with result: " . $mysql_ibdata1_result);
|
|
|
|
return $mysql_ibdata1_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_connection
|
|
#
|
|
# This function check connection against MySQL, if goes wrong abort monitoring
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_connection {
|
|
my $mysql_connection_result = 1;
|
|
my $connection_string = 0;
|
|
if (!defined($plugin_setup{"conf_mysql_pass"})){
|
|
$connection_string = "mysql -u" . $plugin_setup{"conf_mysql_user"} .
|
|
" -h" . $plugin_setup{"conf_mysql_host"} ;
|
|
}else{
|
|
# Connection string
|
|
$connection_string = "mysql -u" . $plugin_setup{"conf_mysql_user"} .
|
|
" -p" . $plugin_setup{"conf_mysql_pass"} .
|
|
" -h" . $plugin_setup{"conf_mysql_host"} ;
|
|
}
|
|
my $tmp_file = $plugin_setup{'conf_temp'} . '/mysql_connection.tmp';
|
|
|
|
my $connection_result = `$connection_string -e "SELECT 1 FROM DUAL" 2> $tmp_file`;
|
|
|
|
# Collect info from temp file
|
|
open (TMP, "< $tmp_file");
|
|
|
|
while (<TMP>) {
|
|
my $buffer_line = parse_dosline ($_);
|
|
|
|
if ($buffer_line =~ /error/i) {
|
|
$mysql_connection_result = 0;
|
|
}
|
|
}
|
|
|
|
close (TMP);
|
|
|
|
unlink($tmp_file);
|
|
|
|
logger("[INFO] check_mysql_connection with result: " . $mysql_connection_result);
|
|
|
|
# Abort monitoring
|
|
if ($mysql_connection_result == 0) {
|
|
print_module("MySQL_connection_error", "async_string", "Error", "MySQL plugin cannot connect to Database. Abort execution of plugin", "CRITICAL");
|
|
}
|
|
|
|
return $mysql_connection_result;
|
|
}
|
|
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_logs
|
|
#
|
|
# This function check ibdata1 file disk usage
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_logs(;$) {
|
|
my $module_type = $_[0];
|
|
|
|
my $mysql_logs_result = 0;
|
|
|
|
if (!defined($plugin_setup{"conf_logparser"})) {
|
|
logger("[INFO] check_mysql_logs check needs conf_logparser token defined in configuration file.");
|
|
return 0;
|
|
}
|
|
|
|
if (!defined($plugin_setup{"conf_logparser"})) {
|
|
logger("[INFO] check_mysql_logs check needs conf_mysql_logfile token defined in configuration file.");
|
|
return 0;
|
|
}
|
|
|
|
my $plugin_call = $plugin_setup{"conf_logparser"}. " " . $plugin_setup{"conf_mysql_logfile"} . " MySQL_error_logs error 2> /dev/null";
|
|
|
|
$mysql_logs_result = `$plugin_call`;
|
|
|
|
if ($mysql_logs_result ne "") {
|
|
logger ("[INFO] check_mysql_logs with result:\n$mysql_logs_result");
|
|
print $mysql_logs_result;
|
|
|
|
# Process output of grep_log plugin
|
|
my @temp = split ("\n", $mysql_logs_result);
|
|
my @result;
|
|
|
|
# Try to get return values
|
|
my $i = 0;
|
|
foreach (@temp) {
|
|
# Get return values
|
|
if ($_ =~ /<data><value><!\[CDATA\[(.*)\]\]><\/value><\/data>/){
|
|
$result[$i] = $1;
|
|
$i++;
|
|
}
|
|
}
|
|
|
|
return @result;
|
|
|
|
} else {
|
|
logger ("[INFO] Blank output in check_mysql_logs searching in logfile: " . $plugin_setup{"conf_mysql_logfile"});
|
|
|
|
# if (defined($module_type)) {
|
|
# print_module("MySQL_error_logs", $module_type, "Blank output", "Blank output in check_mysql_logs searching in logfile " . $plugin_setup{"conf_mysql_logfile"});
|
|
# } else {
|
|
# print_module("MySQL_error_logs", "async_string", "Blank output", "Blank output in check_mysql_logs searching in logfile " . $plugin_setup{"conf_mysql_logfile"});
|
|
# }
|
|
}
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# check_mysql_status (SHOW parameter, [SQL statement/ Query for Innodb status, Mysql_schema])
|
|
#
|
|
# This function check status parameter (Performance parameters)
|
|
# ----------------------------------------------------------------------
|
|
sub check_mysql_status($;$$) {
|
|
my $show_parameter = $_[0];
|
|
my $sql_statement = $_[1];
|
|
my $sql_schema = $_[2];
|
|
my $mysql_status_result = 0;
|
|
|
|
if (!defined($show_parameter)){
|
|
logger("[INFO] Empty parameter in check_mysql_status check, please revise configuration file.");
|
|
return 0;
|
|
}
|
|
|
|
# Writes in temp file
|
|
my $temp_file = $plugin_setup{"conf_temp"} . "/mysql_pandora.tmp";
|
|
|
|
open (TEMP1, "> " . $temp_file);
|
|
|
|
# Depends on the request write different statements ('pending_io', 'sql', 'innodb' is different)
|
|
if ($show_parameter =~ /processlist/i) {
|
|
print TEMP1 "SHOW " . $show_parameter;
|
|
} elsif ($show_parameter =~ /total_size/i) {
|
|
print TEMP1 "SELECT \"Total_size\",
|
|
sum( data_length + index_length ) / 1024 / 1024 / 1024 \"Data Base Size in GB\"
|
|
FROM information_schema.TABLES";
|
|
} elsif (($show_parameter !~ /pending_io/i) and ($show_parameter !~ /sql/i) and ($show_parameter ne 'innodb')) {
|
|
print TEMP1 "SHOW GLOBAL STATUS LIKE '" . $show_parameter . "'";
|
|
}
|
|
|
|
close (TEMP1);
|
|
|
|
# Connection string
|
|
my $connection_string = 0;
|
|
if (!defined($plugin_setup{"conf_mysql_pass"} ) ){
|
|
$connection_string = "mysql -u" . $plugin_setup{"conf_mysql_user"} .
|
|
" -h" . $plugin_setup{"conf_mysql_host"} ;
|
|
}else{
|
|
$connection_string = "mysql -u" . $plugin_setup{"conf_mysql_user"} .
|
|
" -p" . $plugin_setup{"conf_mysql_pass"} .
|
|
" -h" . $plugin_setup{"conf_mysql_host"} ;
|
|
}
|
|
#my $connection_string = "mysql -u" . $plugin_setup{"conf_mysql_user"} .
|
|
# " -p" . $plugin_setup{"conf_mysql_pass"} .
|
|
# " -h" . $plugin_setup{"conf_mysql_host"} ;
|
|
|
|
# If its a query, then try to add schema to the connection string
|
|
if ($show_parameter =~ /sql/i) {
|
|
if (defined($sql_schema)) {
|
|
$connection_string .= " $sql_schema";
|
|
}
|
|
}
|
|
|
|
my $post_sql = '';
|
|
# If we want to retrieve 'active sessions' count lines
|
|
if ($show_parameter =~ /processlist/i) {
|
|
$post_sql = ' | wc -l';
|
|
} # Rest of request have this suffix
|
|
else {
|
|
$post_sql = ' | tail -1 | awk \'{print $2}\'';
|
|
}
|
|
|
|
# 'pending_io' and 'sql' and 'innodb' are different
|
|
if (($show_parameter !~ /pending_io/i) and ($show_parameter !~ /sql/i) and ($show_parameter ne 'innodb')) {
|
|
$mysql_status_result = `$connection_string < $temp_file $post_sql`;
|
|
} elsif ($show_parameter =~ /sql/i) {
|
|
$mysql_status_result = `$connection_string -e "$sql_statement" | tail -1`;
|
|
} elsif ($show_parameter =~ /innodb/i) {
|
|
|
|
|
|
if ($plugin_setup{"conf_mysql_version"} == "5.0") {
|
|
$mysql_status_result = `$connection_string -e "SHOW INNODB STATUS\\G"`;
|
|
} else {
|
|
$mysql_status_result = `$connection_string -e "SHOW ENGINE INNODB STATUS\\G"`;
|
|
}
|
|
|
|
# Process output of show innodb status
|
|
my @temp = split ("\n", $mysql_status_result);
|
|
|
|
# If we detect query lines then output
|
|
$mysql_status_result = 0;
|
|
foreach (@temp) {
|
|
if ($_ =~ m/$sql_statement\s+(.*)/) {
|
|
$mysql_status_result = $1;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
if ($plugin_setup{"conf_mysql_version"} == "5.0") {
|
|
$mysql_status_result = `$connection_string -e "SHOW INNODB STATUS\\G"`;
|
|
} else {
|
|
$mysql_status_result = `$connection_string -e "SHOW ENGINE INNODB STATUS\\G"`;
|
|
}
|
|
|
|
# Process output of show innodb status
|
|
my @temp = split ("\n", $mysql_status_result);
|
|
|
|
# If we detect 'i/o waiting' lines then increment counter
|
|
$mysql_status_result = 0;
|
|
foreach (@temp) {
|
|
if ($_ =~ m/(.*)waiting for i\/o request(.*)/){
|
|
$mysql_status_result++;
|
|
}
|
|
}
|
|
}
|
|
|
|
unlink ($temp_file);
|
|
|
|
# If we want to retrieve active session substract 1 due to header
|
|
if ($show_parameter =~ /processlist/i) {
|
|
if ($mysql_status_result > 0){
|
|
$mysql_status_result--;
|
|
}
|
|
}
|
|
|
|
logger("[INFO] check_mysql_status executing: $show_parameter with result: " . $mysql_status_result);
|
|
|
|
return $mysql_status_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# is_numeric (arg)
|
|
#
|
|
# Return TRUE if given argument is numeric
|
|
# ----------------------------------------------------------------------
|
|
sub is_numeric($) {
|
|
my $val = $_[0];
|
|
|
|
if (!defined($val)){
|
|
return 0;
|
|
}
|
|
# Replace "," for "."
|
|
$val =~ s/\,/\./;
|
|
|
|
my $DIGITS = qr{ \d+ (?: [.] \d*)? | [.] \d+ }xms;
|
|
my $SIGN = qr{ [+-] }xms;
|
|
my $NUMBER = qr{ ($SIGN?) ($DIGITS) }xms;
|
|
if ( $val !~ /^${NUMBER}$/ ) {
|
|
return 0; #Non-numeric
|
|
} else {
|
|
return 1; #Numeric
|
|
}
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# eval_postcondition (result, operator, value)
|
|
#
|
|
# This function eval postcondition and return a boolean
|
|
# ----------------------------------------------------------------------
|
|
sub eval_postcondition($$$) {
|
|
my $result = $_[0];
|
|
my $operator = $_[1];
|
|
my $value = $_[2];
|
|
my $eval_postcondition_result = 0;
|
|
|
|
my $result_type = is_numeric($result);
|
|
|
|
logger("[INFO] Evaluating poscondition: $result $operator $value.");
|
|
|
|
# Numeric result
|
|
if ($result_type == 1) {
|
|
# Equal to
|
|
if ($operator =~ /==/) {
|
|
if (int($result) == int($value)) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# Not equal to
|
|
elsif ($operator =~ /!=/) {
|
|
|
|
if (int($result) != int($value)) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# Less than
|
|
elsif ($operator =~ /</) {
|
|
|
|
if (int($result) < int($value)) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# More than
|
|
elsif ($operator =~ />/) {
|
|
|
|
if (int($result) > int($value)) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
} else {
|
|
|
|
logger("[ERROR] Unknown operator in postcondition, please revise your configuration file.");
|
|
$eval_postcondition_result = 0;
|
|
|
|
}
|
|
} # Non numeric result
|
|
else {
|
|
|
|
my $result_value = sprintf("%s", $result);
|
|
my $string_value = sprintf("%s", $value);
|
|
|
|
# Equal to
|
|
if ($operator =~ /==/) {
|
|
|
|
if ($result_value =~ /$string_value/) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# Not equal to
|
|
elsif ($operator =~ /!=/) {
|
|
|
|
if ($result_value !~ /$string_value/) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# Less than
|
|
elsif ($operator =~ /</) {
|
|
|
|
if ($result_value lt $string_value) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
}# More than
|
|
elsif ($operator =~ />/) {
|
|
|
|
if ($result_value gt $string_value) {
|
|
|
|
$eval_postcondition_result = 1;
|
|
|
|
}
|
|
} else {
|
|
|
|
logger("[ERROR] Unknown operator in postcondition, please revise your configuration file.");
|
|
$eval_postcondition_result = 0;
|
|
|
|
}
|
|
}
|
|
|
|
return $eval_postcondition_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# eval_postcondition_array (Array result, operator, value)
|
|
#
|
|
# This function eval postcondition and return a boolean if some of the
|
|
# elements fulfill the condition
|
|
# ----------------------------------------------------------------------
|
|
sub eval_postcondition_array {
|
|
# This works
|
|
my $value = pop;
|
|
my $operator = pop;
|
|
my @result = @_;
|
|
|
|
my $eval_postcondition_array_result = 0;
|
|
my $return_postcondition_element = 0;
|
|
|
|
# Eval all array
|
|
foreach my $single_result (@result) {
|
|
|
|
$return_postcondition_element = 0;
|
|
|
|
$return_postcondition_element = eval_postcondition($single_result, $operator, $value);
|
|
|
|
if ($return_postcondition_element) {
|
|
$eval_postcondition_array_result = 1;
|
|
}
|
|
}
|
|
|
|
return $eval_postcondition_array_result;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# store_result_check(check_type, value)
|
|
#
|
|
# This function stores in a temporary file the last value of the check
|
|
# ----------------------------------------------------------------------
|
|
sub store_result_check($$) {
|
|
my $type = $_[0];
|
|
my $value = $_[1];
|
|
|
|
# Compose the temp filename
|
|
my $tmp_file = $plugin_setup{'conf_temp'} . '/mysql_check_' . $type . '.tmp';
|
|
|
|
# Try to open temp file
|
|
if (! open (TMP, "> $tmp_file")) {
|
|
logger("[ERROR] Error opening temp file: $tmp_file for write last check value: $type, $value");
|
|
return;
|
|
}
|
|
|
|
# Write las check value
|
|
print TMP $value;
|
|
close (TMP);
|
|
|
|
logger("[INFO] Saving last check value: $value of check: $type in file $tmp_file .");
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# process_delta_data(check_type, value)
|
|
#
|
|
# This function calculates delta value for the current check
|
|
# ----------------------------------------------------------------------
|
|
sub process_delta_data($$) {
|
|
my $type = $_[0];
|
|
my $value = $_[1];
|
|
my $buffer_line;
|
|
|
|
# Compose the temp filename
|
|
my $tmp_file = $plugin_setup{'conf_temp'} . '/mysql_check_' . $type . '.tmp';
|
|
|
|
# Collect info from temp file
|
|
# If it's not possible then return token for not print module
|
|
if (! open (TMP, "< $tmp_file")) {
|
|
return '::MYSQL _ NON EXEC';
|
|
}
|
|
|
|
while (<TMP>) {
|
|
$buffer_line = parse_dosline ($_);
|
|
}
|
|
|
|
my $process_delta_result = int($value) - int($buffer_line);
|
|
|
|
close (TMP);
|
|
|
|
# If delta value is negative then reset check last value
|
|
if ($process_delta_result < 0) {
|
|
logger("[INFO] Reset last value of check_" . $type . " due to negative value: " . $process_delta_result);
|
|
return '::MYSQL _ NON EXEC';
|
|
}
|
|
|
|
logger("[INFO] check_" . $type . " postprocessed by delta calculation: " . $process_delta_result);
|
|
|
|
return $process_delta_result;
|
|
}
|
|
|
|
###############################################################################
|
|
###############################################################################
|
|
######################## MAIN PROGRAM CODE ####################################
|
|
###############################################################################
|
|
###############################################################################
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Checks input parameter from command line (Conf file)
|
|
# ----------------------------------------------------------------------
|
|
my $log_init = 0;
|
|
|
|
# Load config file from command line
|
|
if ($#ARGV == -1){
|
|
print "I need at least one parameter: Complete path to external configuration file \n";
|
|
logger_begin("[ERROR] Path to configuration file it's needed");
|
|
|
|
# Logfile is initiated
|
|
$log_init = 1;
|
|
|
|
exit 0;
|
|
}
|
|
|
|
# Check for conf file
|
|
if ( ! -f $archivo_cfg ) {
|
|
printf "\n [ERROR] Cannot open configuration file at $archivo_cfg. \n\n";
|
|
|
|
if ($log_init == 1) {
|
|
logger("[ERROR] Cannot open configuration file at $archivo_cfg, please set permissions correctly.");
|
|
} else {
|
|
logger_begin("[ERROR] Cannot open configuration file at $archivo_cfg, please set permissions correctly.");
|
|
$log_init = 1;
|
|
}
|
|
|
|
print_module("MySQL_plugin_error", "generic_proc", 0, "Cannot open configuration file at $archivo_cfg, please set permissions correctly.");
|
|
exit 0;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Parse external configuration file
|
|
# ----------------------------------------------------------------------
|
|
load_external_setup ($archivo_cfg);
|
|
|
|
if ($log_init == 1){
|
|
|
|
logger("[INFO] Parsing config file $archivo_cfg.");
|
|
|
|
} else {
|
|
|
|
logger_begin("[INFO] Parsing config file $archivo_cfg.");
|
|
$log_init = 1;
|
|
|
|
}
|
|
|
|
parse_config;
|
|
|
|
=COMMENT
|
|
print Dumper(%plugin_setup);
|
|
print Dumper(@checks);
|
|
|
|
exit;
|
|
=cut
|
|
|
|
# ----------------------------------------------------------------------
|
|
# First check connection to MySQL
|
|
# ----------------------------------------------------------------------
|
|
my $result_connection = check_mysql_connection;
|
|
if ($result_connection == 0) {
|
|
logger ("[ERROR] Connection to MySQL error, abort monitoring.");
|
|
exit 0;
|
|
}
|
|
|
|
# ----------------------------------------------------------------------
|
|
# Process each check
|
|
# ----------------------------------------------------------------------
|
|
my $result_check;
|
|
my $module_type;
|
|
my $module_status;
|
|
my @result_check;
|
|
my $performance_parameter;
|
|
foreach (@checks) {
|
|
|
|
# Don't asumme performance parameter
|
|
$performance_parameter = 0;
|
|
my $type = $_->{'type'};
|
|
my $postcondition_op = $_->{'postcondition_op'};
|
|
my $postcondition_val = $_->{'postcondition_val'};
|
|
my $postexecution = $_->{'postexecution'};
|
|
my $check_status = $_->{'status'};
|
|
my $check_show = $_->{'show'};
|
|
my $return_type = $_->{'return_type'};
|
|
my $check_name = $_->{'check_name'};
|
|
my $check_type = $_->{'check_type'};
|
|
|
|
$result_check = 0;
|
|
# Process check (System parameters)
|
|
if ($_->{'type'} eq 'mysql_service') {
|
|
|
|
$result_check = check_mysql_service;
|
|
|
|
} elsif ($_->{'type'} eq 'mysql_memory') {
|
|
|
|
$result_check = check_mysql_memory;
|
|
|
|
} elsif ($_->{'type'} eq 'mysql_cpu') {
|
|
|
|
$result_check = check_mysql_cpu;
|
|
|
|
} elsif ($_->{'type'} eq 'system_timewait') {
|
|
|
|
$result_check = check_system_timewait;
|
|
|
|
} elsif ($_->{'type'} eq 'system_diskusage') {
|
|
|
|
$result_check = check_system_diskusage;
|
|
|
|
} elsif ($_->{'type'} eq 'mysql_ibdata1') {
|
|
|
|
$result_check = check_mysql_ibdata1;
|
|
|
|
} elsif ($_->{'type'} eq 'mysql_connection') {
|
|
|
|
$result_check = check_mysql_connection;
|
|
|
|
} elsif ($_->{'type'} eq 'mysql_logs') {
|
|
|
|
@result_check = check_mysql_logs($_->{'module_type'});
|
|
|
|
}
|
|
# Process check (Perfomance parameters)
|
|
elsif ($_->{'type'} eq 'status') {
|
|
|
|
# This is a performance parameter
|
|
$performance_parameter = 1;
|
|
$result_check = check_mysql_status($_->{"show"});
|
|
|
|
}
|
|
# Process check (Open SQL interface)
|
|
elsif ($_->{'type'} eq 'sql') {
|
|
|
|
$performance_parameter = 1;
|
|
$result_check = check_mysql_status($_->{'type'}, $_->{"sql"}, $_->{'check_schema'});
|
|
|
|
}
|
|
# Process check (Query Innodb status)
|
|
elsif ($_->{'type'} eq 'innodb') {
|
|
|
|
$result_check = check_mysql_status($_->{'type'}, $_->{"query"});
|
|
|
|
}else {
|
|
|
|
logger("[INFO] Check block with invalid type, discart it. Please revise your configuration file.");
|
|
$result_check = 0;
|
|
|
|
}
|
|
|
|
# Use results
|
|
if (($_->{'type'} ne 'unknown')) {
|
|
|
|
# Prints module ('mysql_logs' check prints it's module inside check_mysql_logs function)
|
|
if ($_->{'type'} ne 'mysql_logs') {
|
|
|
|
# Evalue module type
|
|
$module_type = $_->{'module_type'};
|
|
|
|
# 'mysql_service' and 'mysql_connection' has always generic_proc type
|
|
if (($_->{'type'} eq 'mysql_service') or ($type eq 'mysql_connection')) {
|
|
|
|
$module_type = 'generic_proc';
|
|
|
|
}# By default type is generic_data
|
|
else {
|
|
|
|
$module_type = 'generic_data';
|
|
|
|
}
|
|
|
|
my $last_result_check = $result_check;
|
|
# If where are dealing with a performance parameter then look at return_type value
|
|
if ($performance_parameter and defined($_->{'return_type'})) {
|
|
|
|
if ($_->{'return_type'} eq 'data_delta') {
|
|
|
|
$result_check = process_delta_data($_->{'type'} . '_' . $_->{'show'}, $result_check);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
# Store result_check in temp file for delta postprocess
|
|
if ($performance_parameter) {
|
|
|
|
store_result_check($type . '_' . $check_show, $last_result_check);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
# Postcondition ('mysql_logs' type will have an array like result)
|
|
my $exec_postexecution = 0;
|
|
my $result_check_test = sprintf("%s", $result_check);
|
|
if (($type eq 'mysql_logs') and ($result_check_test ne '::MYSQL _ NON EXEC')) {
|
|
|
|
# if (defined($postcondition_op) and defined($postcondition_val)) {
|
|
|
|
# $exec_postexecution = eval_postcondition_array(@result_check, $postcondition_op, $postcondition_val);
|
|
|
|
# }
|
|
|
|
# If data has been returned, exec postcommand
|
|
if (@result_check) {
|
|
$exec_postexecution = 1;
|
|
}
|
|
|
|
}# Postcondition (other check types)
|
|
elsif (defined($postcondition_op) and defined($postcondition_val) and ($result_check_test ne '::MYSQL _ NON EXEC')) {
|
|
|
|
$exec_postexecution = eval_postcondition($result_check, $postcondition_op, $postcondition_val);
|
|
|
|
}
|
|
|
|
# If postcondition is not fulfilled then don't assign status to module
|
|
if (!$exec_postexecution) {
|
|
$check_status = 'NORMAL';
|
|
}
|
|
|
|
# Prints module ('mysql_logs' check prints it's module inside check_mysql_logs function)
|
|
if ($type ne 'mysql_logs') {
|
|
if ($result_check_test ne '::MYSQL _ NON EXEC') {
|
|
# Prints module
|
|
if ($type eq 'status') {
|
|
print_module("MySQL_" . $type . '_' . $check_show, $module_type, $result_check, '', $check_status);
|
|
} else {
|
|
if (defined($check_name)) {
|
|
print_module("MySQL_" . $type . "_" . $check_name, $check_type, $result_check, '', $check_status);
|
|
} else {
|
|
print_module("MySQL_" . $type, $module_type, $result_check, '', $check_status);
|
|
}
|
|
}
|
|
} else {
|
|
logger("[INFO] First execution of delta value for check $type, ignoring.");
|
|
}
|
|
}
|
|
|
|
# Exec command
|
|
if ($exec_postexecution) {
|
|
|
|
if (defined($postexecution)) {
|
|
|
|
logger("[INFO] Executing postexecution command: " . $postexecution);
|
|
if ($postexecution =~ /\_DATA\_/i) {
|
|
my $round_data = ceil ($result_check);
|
|
|
|
logger("[INFO] Detected macro _DATA_ in postexecution command, replacing with: $result_check");
|
|
$postexecution =~ s/\_DATA\_/$round_data/;
|
|
}
|
|
my $command_output = `$postexecution`;
|
|
logger("[INFO] Postexecution command result: " . $command_output);
|
|
|
|
}
|
|
}
|
|
|
|
} # type ne 'unknown'
|
|
}
|
|
|