require 'optparse'
require 'timeout'
module Daemoned
class DieTime < StandardError; end
class TimeoutError < StandardError; end
def self.included(base)
base.extend ClassMethods
base.initialize_options
end
class Config
METHODS = [:script_path]
CONFIG = {}
def method_missing(name, *args)
name = name.to_s.upcase.to_sym
if name.to_s =~ /^(.*)=$/
name = $1.to_sym
CONFIG[name] = args.first
else
CONFIG[name]
end
end
end
module ClassMethods
def initialize_options
@@config = Config.new
@@config.script_path = File.expand_path(File.dirname($0))
$0 = script_name
end
def parse_options
opts = OptionParser.new do |opt|
opt.banner = "Usage: #{script_name} [options] [start|stop]"
opt.on_tail('-h', '--help', 'Show this message') do
puts opt
exit(1)
end
opt.on('--loop-every=SECONDS', 'How long to sleep between each loop') do |value|
options[:loop_every] = value
end
opt.on('-t', '--ontop', 'Stay on top (does not daemonize)') do
options[:ontop] = true
end
opt.on('--instances=NUM', 'Allow multiple instances to run simultaneously? 0 for infinite. default: 1') do |value|
self.instances = value.to_i
end
opt.on('--log-file=LOGFILE', 'Logfile to log to') do |value|
options[:log_file] = File.expand_path(value)
end
opt.on('--pid-file=PIDFILE', 'Location of pidfile') do |value|
options[:pid_file] = File.expand_path(value)
end
opt.on('--no-log-prefix', 'Do not prefix PID and date/time in log file.') do
options[:log_prefix] = false
end
end
extra_args.each do |arg|
opts.on(*arg.first) do |value|
arg.last.call(value) if arg.last
end
end
opts.parse!
if ARGV.include?('stop')
stop
elsif ARGV.include?('reload')
kill('HUP')
exit
elsif not ARGV.include?('start') and not ontop?
puts opts.help
end
end
def arg(*args, &block)
self.extra_args << [args, block]
end
def extra_args
@extra_args ||= []
end
def callbacks
@callbacks ||= {}
end
def options
@options ||= {}
end
def options=(options)
@options = options
end
def config
yield @@config
end
def before(&block)
callbacks[:before] = block
end
def after(&block)
callbacks[:after] = block
end
def sig(*signals, &block)
signals.each do |s|
callbacks["sig_#{s}".to_sym] = block
end
end
def die_if(method=nil,&block)
options[:die_if] = method || block
end
def exit_if(method=nil,&block)
options[:exit_if] = method || block
end
def callback!(callback)
callbacks[callback].call if callbacks[callback]
end
# options may include:
#
# :loop_every Fixnum (DEFAULT 0)
# How many seconds to sleep between calls to your block
#
# :timeout Fixnum (DEFAULT 0)
# Timeout in if block does not execute withing passed number of seconds
#
# :kill_timeout Fixnum (DEFAULT 120)
# Wait number of seconds before using kill -9 on daemon
#
# :die_on_timeout BOOL (DEFAULT False)
# Should the daemon continue running if a block times out, or just run the block again
#
# :ontop BOOL (DEFAULT False)
# Do not daemonize. Run in current process
#
# :before BLOCK
# Run this block after daemonizing but before begining the daemonize loop.
# You can also define the before block by putting a before do/end block in your class.
#
# :after BLOCK
# Run this block before program exists.
# You can also define the after block by putting an after do/end block in your class.
#
# :die_if BLOCK
# Run this check after each iteration of the loop. If the block returns true, throw a DieTime exception and exit
# You can also define the after block by putting an die_if do/end block in your class.
#
# :exit_if BLOCK
# Run this check after each iteration of the loop. If the block returns true, exit gracefully
# You can also define the after block by putting an exit_if do/end block in your class.
#
# :log_prefix BOOL (DEFAULT true)
# Prefix log file entries with PID and timestamp
def daemonize(opts={}, &block)
self.options = opts
parse_options
return unless ok_to_start?
puts "Starting #{script_name}..."
puts "Logging to: #{log_file}" unless ontop?
unless ontop?
safefork do
open(pid_file, 'w'){|f| f << Process.pid }
at_exit { remove_pid! }
trap('TERM') { callback!(:sig_term) }
trap('INT') { callback!(:sig_int) ; Process.kill('TERM', $$) }
trap('HUP') { callback!(:sig_hup) }
sess_id = Process.setsid
reopen_filehandes
begin
at_exit { callback!(:after) }
callback!(:before)
run_block(&block)
rescue SystemExit
rescue Exception => e
$stdout.puts "Something bad happened #{e.inspect} #{e.backtrace.join("\n")}"
end
end
else
begin
callback!(:before)
run_block(&block)
rescue SystemExit, Interrupt
callback!(:after)
end
end
end
private
def run_block(&block)
loop do
if options[:timeout]
begin
Timeout::timeout(options[:timeout].to_i) do
block.call if block
end
rescue Timeout::Error => e
if options[:die_on_timeout]
raise TimeoutError.new("#{self} timed out after #{options[:timeout]} seconds while executing block in loop")
else
$stderr.puts "#{self} timed out after #{options[:timeout]} seconds while executing block in loop #{e.backtrace.join("\n")}"
end
end
else
block.call if block
end
if options[:loop_every]
sleep options[:loop_every].to_i
elsif not block
sleep 0.1
end
break if should_exit?
raise DieTime.new('Die if conditions were met!') if should_die?
end
exit(0)
end
def should_die?
die_if = options[:die_if]
if die_if
if die_if.is_a?(Symbol) or die_if.is_a?(String)
self.send(die_if)
elsif die_if.is_a?(Proc)
die_if.call
end
else
false
end
end
def should_exit?
exit_if = options[:exit_if]
if exit_if
if exit_if.is_a?(Symbol) or exit_if.is_a?(String)
self.send(exit_if.to_sym)
elsif exit_if.is_a?(Proc)
exit_if.call
end
else
false
end
end
def ok_to_start?
return true if pid.nil?
if process_alive?
$stderr.puts "#{script_name} is already running"
return false
else
$stderr.puts "Removing stale pid: #{pid}..."
end
true
end
def stop
puts "Stopping #{script_name}..."
kill
exit
end
def kill(signal = 'TERM')
if pid.nil?
$stderr.puts "#{script_name} doesn't appear to be running"
exit(1)
end
$stdout.puts("Sending pid #{pid} signal #{signal}...")
begin
Process.kill(signal, pid)
return if signal == 'HUP'
if pid_running?(options[:kill_timeout] || 120)
$stdout.puts("Using kill -9 #{pid}")
Process.kill(9, pid)
else
$stdout.puts("Process #{pid} stopped")
end
rescue Errno::ESRCH
$stdout.puts("Couldn't #{signal} #{pid} as it wasn't running")
end
end
def pid_running?(time_to_wait = 0)
times_to_check = 1
if time_to_wait > 0.5
times_to_check = (time_to_wait / 0.5).to_i
end
times_to_check.times do
return false unless process_alive?
sleep 0.5
end
true
end
def safefork(&block)
fork_tries ||= 0
fork(&block)
rescue Errno::EWOULDBLOCK
raise if fork_tries >= 20
fork_tries += 1
sleep 5
retry
end
def process_alive?
Process.kill(0, pid)
true
rescue Errno::ESRCH => e
false
end
LOG_FORMAT = '%-6d %-19s %s'
TIME_FORMAT = '%Y/%m/%d %H:%M:%S'
def reopen_filehandes
STDIN.reopen('/dev/null')
STDOUT.reopen(log_file, 'a')
STDOUT.sync = true
STDERR.reopen(STDOUT)
if log_prefix?
def STDOUT.write(string)
if @no_prefix
@no_prefix = false if string[-1, 1] == "\n"
else
string = LOG_FORMAT % [$$, Time.now.strftime(TIME_FORMAT), string]
@no_prefix = true
end
super(string)
end
end
end
def remove_pid!
if File.file?(pid_file) and File.read(pid_file).to_i == $$
File.unlink(pid_file)
end
end
def ontop?
options[:ontop]
end
def log_prefix?
options[:log_prefix] || true
end
LOG_PATHS = ['log/', 'logs/', '../log/', '../logs/', '../../log', '../../logs', '.']
LOG_PATHS.unshift("#{RAILS_ROOT}/log") if defined?(RAILS_ROOT)
def log_dir
options[:log_dir] ||= begin
LOG_PATHS.detect do |path|
File.exists?(File.expand_path(path))
end
end
end
def log_file
options[:log_file] ||= File.expand_path("#{log_dir}/#{script_name}.log")
end
def pid_dir
options[:pid_dir] ||= log_dir
end
def pid_file
options[:pid_file] ||= File.expand_path("#{pid_dir}/#{script_name}.pid")
end
def pid
@pid ||= File.file?(pid_file) ? File.read(pid_file).to_i : nil
end
def script_name
@script_name ||= File.basename($0).gsub('.rb', '')
end
def script_name=(script_name)
@script_name = script_name
end
end
end