| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202 |
- #!/usr/bin/perl -w
- #
- # $Id$
- #
- # Log file regular expression detector for Nagios.
- # Written by Aaron Bostick (abostick@mydoconline.com)
- # Last modified: 05-02-2002
- #
- # Thanks and acknowledgements to Ethan Galstad for Nagios and the check_log
- # plugin this is modeled after.
- #
- # Usage: check_log2 -l <log_file> -s <seek_file> -p <pattern> [-n <negpattern>]
- #
- # Description:
- #
- # This plugin will scan arbitrary text files looking for regular expression
- # matches. The text file to scan is specified with <log_file>.
- # <log_seek_file> is a temporary file used to store the seek byte position
- # of the last scan. This file will be created automatically on the first
- # scan. <pattern> can be any RE pattern that perl's s/// syntax accepte. Be
- # forewarned that a bad pattern will send this script into never never land!
- #
- # Output:
- #
- # This plugin returns OK when a file is successfully scanned and no pattern
- # matches are found. WARNING is returned when 1 or more patterns are found
- # along with the pattern count and the line of the last pattern matched.
- # CRITICAL is returned when an error occurs, such as file not found, etc.
- #
- # Notes (paraphrased from check_log's notes):
- #
- # 1. The "max_attempts" value for the service should be 1, as this
- # will prevent Nagios from retrying the service check (the
- # next time the check is run it will not produce the same results).
- #
- # 2. The "notify_recovery" value for the service should be 0, so that
- # Nagios does not notify you of "recoveries" for the check. Since
- # pattern matches in the log file will only be reported once and not
- # the next time, there will always be "recoveries" for the service, even
- # though recoveries really don't apply to this type of check.
- #
- # 3. You *must* supply a different <log_Seek_file> for each service that
- # you define to use this plugin script - even if the different services
- # check the same <log_file> for pattern matches. This is necessary
- # because of the way the script operates.
- #
- # Examples:
- #
- # Check for error notices in messages
- # check_log2 -l /var/log/messages -s ./check_log2.messages.seek -p 'err'
- #
- BEGIN {
- if ($0 =~ s/^(.*?)[\/\\]([^\/\\]+)$//) {
- $prog_dir = $1;
- $prog_name = $2;
- }
- }
- require 5.004;
- use lib $main::prog_dir;
- use utils qw($TIMEOUT %ERRORS &print_revision &support &usage);
- use Getopt::Long;
- my ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$mtime,$ctime,$blksize,$blocks);
- sub print_usage ();
- sub print_version ();
- sub print_help ();
- # Initialize strings
- $log_file = '';
- $seek_file = '';
- $critical = '';
- $re_pattern = '';
- $neg_re_pattern = '';
- $pattern_count = 0;
- $pattern_line = '';
- $plugin_revision = '$Revision$ ';
- # Grab options from command line
- GetOptions
- ("l|logfile=s" => \$log_file,
- "s|seekfile=s" => \$seek_file,
- "c|critical" => \$critical,
- "p|pattern=s" => \$re_pattern,
- "n|negpattern:s" => \$neg_re_pattern,
- "v|version" => \$version,
- "h|help" => \$help);
- !($version) || print_version ();
- !($help) || print_help ();
- # Make sure log file is specified
- ($log_file) || usage("Log file not specified.\n");
- # Make sure seek file is specified
- ($seek_file) || usage("Seek file not specified.\n");
- # Make sure re pattern is specified
- ($re_pattern) || usage("Regular expression not specified.\n");
- # Open log file
- open (LOG_FILE, $log_file) || die "Unable to open log file $log_file: $!";
- # Try to open log seek file. If open fails, we seek from beginning of
- # file by default.
- if (open(SEEK_FILE, $seek_file)) {
- chomp(@seek_pos = <SEEK_FILE>);
- close(SEEK_FILE);
- # If file is empty, no need to seek...
- if ($seek_pos[0] != 0) {
-
- # Compare seek position to actual file size.
- # If file size is smaller
- # then we just start from beginning i.e. file was rotated, etc.
- ($dev,$ino,$mode,$nlink,$uid,$gid,$rdev,$size,$atime,$mtime,$ctime,$blksize,$blocks) = stat(LOG_FILE);
- if ($seek_pos[0] <= $size) {
- seek(LOG_FILE, $seek_pos[0], 0);
- }
- }
- }
- # Loop through every line of log file and check for pattern matches.
- # Count the number of pattern matches and remember the full line of
- # the most recent match.
- while (<LOG_FILE>) {
- if ($neg_re_pattern) {
- if ((/$re_pattern/) && !(/$neg_re_pattern/)) {
- $pattern_count += 1;
- $pattern_line = $_;
- }
- } elsif (/$re_pattern/) {
- $pattern_count += 1;
- $pattern_line = $_;
- }
- }
- # Overwrite log seek file and print the byte position we have seeked to.
- open(SEEK_FILE, "> $seek_file") || die "Unable to open seek count file $seek_file: $!";
- print SEEK_FILE tell(LOG_FILE);
- # Close seek file.
- close(SEEK_FILE);
- # Close the log file.
- close(LOG_FILE);
- # Print result and return exit code.
- if ($pattern_count) {
- if ($critical) {
- print "CRITICAL: ";
- } else {
- print "WARNING: ";
- }
- print "($pattern_count): $pattern_line";
- if ($critical) {
- exit $ERRORS{'CRITICAL'};
- } else {
- exit $ERRORS{'WARNING'};
- }
- } else {
- print "OK - No matches found.\n";
- exit $ERRORS{'OK'};
- }
- #
- # Subroutines
- #
- sub print_usage () {
- print "Usage: $prog_name -l <log_file> -s <log_seek_file> -p <pattern> [-n <negpattern>] -c | --critical\n";
- print "Usage: $prog_name [ -v | --version ]\n";
- print "Usage: $prog_name [ -h | --help ]\n";
- }
- sub print_version () {
- print_revision($prog_name, $plugin_revision);
- exit $ERRORS{'OK'};
- }
- sub print_help () {
- print_revision($prog_name, $plugin_revision);
- print "\n";
- print "Scan arbitrary log files for regular expression matches.\n";
- print "\n";
- print_usage();
- print "\n";
- print "-l, --logfile=<logfile>\n";
- print " The log file to be scanned\n";
- print "-s, --seekfile=<seekfile>\n";
- print " The temporary file to store the seek position of the last scan\n";
- print "-p, --pattern=<pattern>\n";
- print " The regular expression to scan for in the log file\n";
- print "-n, --negpattern=<negpattern>\n";
- print " The regular expression to skip in the log file\n";
- print "-c, --critical\n";
- print " Return critical instead of warning on error\n";
- print "\n";
- support();
- exit $ERRORS{'OK'};
- }
|