#!/usr/bin/env ruby # # lognotify.rb # http://github.com/omp/lognotify # Retrieve new lines from remote log files over SSH. # # Copyright 2010 David Vazgenovich Shakaryan # Distributed under the terms of the GNU General Public License v3. # See http://www.gnu.org/licenses/gpl.txt for the full license text. # # To use, create a configuration file, such as identifier.conf, in the # configuration directory, which defaults to ~/.config/lognotify/. The # following settings are required: # # hostname = ... Hostname of SSH server. # path = ... Path of log file on server. # # Afterwards, simply run the script with the identifier as an arugment: # # lognotify.rb identifier # # During the initial run, a cache file will be created and the entire log file # will be retrieved. On any subsequent runs, only new lines will be retrieved # and appended to the cached log. require 'ftools' # Global settings. CACHE_DIR="~/.cache/lognotify" CONF_DIR="~/.config/lognotify" # Methods for quickly getting file paths. class String def to_cache_path return File.expand_path(CACHE_DIR + '/' + self + '.log') end def to_conf_path return File.expand_path(CONF_DIR + '/' + self + '.conf') end end # Configuration file parser. def parse identifier conf = Hash.new File.open(identifier.to_conf_path) do |file| file.each_line do |line| # Remove whitespace from beginning of line, allowing for indentation. line.lstrip! # Ignore empty lines and comments. unless line.empty? or line[0,1] == '#' key, value = line.split('=', 2) # Raise an error if line does not contain a key/value pair. raise 'Error reading line ' + file.lineno.to_s + '.' if value.nil? conf[key.strip.to_sym] = value.strip end end end return conf end # Count lines in cached log. def count_lines identifier return File.open(identifier.to_cache_path).readlines.length end # Retrieve new lines via SSH. def retrieve_lines conf, lines command = "cat #{conf[:path]}" command << " | sed '1,#{lines}d'" if lines > 0 return %x[ssh #{conf[:hostname]} "#{command}"] end # Append new lines to cached log. def append_lines identifier, lines File.open(identifier.to_cache_path, 'a') do |file| file.print lines end end # Output all messages immediately, as opposed to buffering. STDOUT.sync = true # Create cache directory, if nonexistent. path = File.expand_path(CACHE_DIR) File.makedirs(path) unless File.directory?(path) # Treat each argument as a log identifier. ARGV.each do |identifier| conf = parse(identifier) # Create cache file, if nonexistent. path = identifier.to_cache_path File.open(path, 'w').close unless File.exist?(path) print '* Counting lines in cached log... ' lines = count_lines(identifier) puts lines print '* Retrieving new lines via SSH... ' newlines = retrieve_lines(conf, lines) puts 'Done' puts '* Number of new lines: ' + newlines.lines.count.to_s unless newlines.lines.count.zero? print '* Appending new lines to cached log... ' append_lines(identifier, newlines) puts 'Done' puts newlines end end