require 'fileutils'

# Taken unchanged from https://github.com/alexvollmer/daemon-spawn
#
# Large portions of this were liberally stolen from the
# 'simple-daemon' project at http://simple-daemon.rubyforge.org/
module DaemonSpawn
  VERSION = '0.3.0'

  def self.usage(msg=nil) #:nodoc:
    print "#{msg}, " if msg
    puts "usage: #{$0} <command> [options]"
    puts "Where <command> is one of start, stop, restart or status"
    puts "[options] are additional options passed to the underlying process"
  end

  def self.alive?(pid)
    Process.kill 0, pid
  rescue Errno::ESRCH
    false
  end

  def self.start(daemon, args) #:nodoc:
    if !File.writable?(File.dirname(daemon.log_file))
      STDERR.puts "Unable to write log file to #{daemon.log_file}"
      exit 1
    end

    if !File.writable?(File.dirname(daemon.pid_file))
      STDERR.puts "Unable to write PID file to #{daemon.pid_file}"
      exit 1
    end

    if daemon.alive? && daemon.singleton
      STDERR.puts "An instance of #{daemon.app_name} is already " +
        "running (PID #{daemon.pid})"
      exit 0
    end

    fork do
      Process.setsid
      exit if fork
      open(daemon.pid_file, 'w') { |f| f << Process.pid }
      Dir.chdir daemon.working_dir
      old_umask = File.umask 0000
      log = File.new(daemon.log_file, "a")
      File.umask old_umask
      log.sync = daemon.sync_log
      STDIN.reopen "/dev/null"
      STDOUT.reopen log
      STDERR.reopen STDOUT
      trap("TERM") {daemon.stop; exit}
      daemon.start(args)
    end
    puts("#{daemon.app_name} started.") if $stdin.tty?
  end

  def self.stop(daemon) #:nodoc:
    if pid = daemon.pid
      FileUtils.rm(daemon.pid_file)
      Process.kill(daemon.signal, pid)
      begin
        Process.wait(pid)
      rescue Errno::ECHILD
      end
      if ticks = daemon.timeout
        while ticks > 0 and alive?(pid) do
          puts "Process is still alive. #{ticks} seconds until I kill -9 it..."
          sleep 1
          ticks -= 1
        end
        if alive?(pid)
          puts "Process didn't quit after timeout of #{daemon.timeout} seconds. Killing..."
          Process.kill 9, pid
        end
      end
    else
      puts "PID file not found. Is the daemon started?"
    end
  rescue Errno::ESRCH
    puts "PID file found, but process was not running. The daemon may have died."
  end

  def self.status(daemon) #:nodoc:
    puts "#{daemon.app_name} is #{daemon.alive? ? "" : "NOT "}running (PID #{daemon.pid})"
  end

  class Base
    attr_accessor :log_file, :pid_file, :sync_log, :working_dir, :app_name, :singleton, :index, :signal, :timeout

    def initialize(opts = {})
      raise 'You must specify a :working_dir' unless opts[:working_dir]
      self.working_dir = opts[:working_dir]
      self.app_name = opts[:application] || classname
      self.pid_file = opts[:pid_file] || File.join(working_dir, 'tmp', 'pids', app_name + '.pid')
      self.log_file = opts[:log_file] || File.join(working_dir, 'logs', app_name + '.log')
      self.signal = opts[:signal] || 'TERM'
      self.timeout = opts[:timeout]
      self.index = opts[:index] || 0
      if self.index > 0
        self.pid_file += ".#{self.index}"
        self.log_file += ".#{self.index}"
      end
      self.sync_log = opts[:sync_log]
      self.singleton = opts[:singleton] || false
    end

    def classname #:nodoc:
      self.class.to_s.split('::').last
    end

    # Provide your implementation. These are provided as a reminder
    # only and will raise an error if invoked. When started, this
    # method will be invoked with the remaining command-line arguments.
    def start(args)
      raise "You must implement a 'start' method in your class!"
    end

    # Provide your implementation. These are provided as a reminder
    # only and will raise an error if invoked.
    def stop
      raise "You must implement a 'stop' method in your class!"
    end

    def alive? #:nodoc:
      if File.file?(pid_file)
        DaemonSpawn.alive? pid
      else
        false
      end
    end

    def pid #:nodoc:
      IO.read(self.pid_file).to_i rescue nil
    end

    def self.build(options)
      count = options.delete(:processes) || 1
      daemons = []
      count.times do |index|
        daemons << new(options.merge(:index => index))
      end
      daemons
    end

    def self.find(options)
      pid_file = new(options).pid_file
      basename = File.basename(pid_file).split('.').first
      pid_files = Dir.glob(File.join(File.dirname(pid_file), "#{basename}.*pid*"))
      pid_files.map { |f| new(options.merge(:pid_file => f)) }
    end

    # Invoke this method to process command-line args and dispatch
    # appropriately. Valid options include the following _symbols_:
    # - <tt>:working_dir</tt> -- the working directory (required)
    # - <tt>:log_file</tt> -- path to the log file
    # - <tt>:pid_file</tt> -- path to the pid file
    # - <tt>:sync_log</tt> -- indicate whether or not to sync log IO
    # - <tt>:singleton</tt> -- If set to true, only one instance is
    # allowed to start
    # args must begin with 'start', 'stop', 'status', or 'restart'.
    # The first token will be removed and any remaining arguments
    # passed to the daemon's start method.
    def self.spawn!(opts = {}, args = ARGV)
      case args.any? and command = args.shift
      when 'start', 'stop', 'status', 'restart'
        send(command, opts, args)
      when '-h', '--help', 'help'
        DaemonSpawn.usage
        exit
      else
        DaemonSpawn.usage "Invalid command"
        exit 1
      end
    end

    def self.start(opts, args)
      living_daemons = find(opts).select { |d| d.alive? }
      if living_daemons.any?
        puts "Daemons already started! PIDS: #{living_daemons.map {|d| d.pid}.join(', ')}"
        exit 1
      else
        build(opts).map { |d| DaemonSpawn.start(d, args) }
      end
    end

    def self.stop(opts, args)
      daemons = find(opts)
      if daemons.empty?
        puts "No PID files found. Is the daemon started?"
        exit 1
      else
        daemons.each { |d| DaemonSpawn.stop(d) }
      end
    end

    def self.status(opts, args)
      daemons = find(opts)
      if daemons.empty?
        puts 'No PIDs found'
      else
        daemons.each { |d| DaemonSpawn.status(d) }
      end
    end

    def self.restart(opts, args)
      daemons = build(opts)
      daemons.map do |daemon|
        DaemonSpawn.stop(daemon)
        sleep 0.1
        DaemonSpawn.start(daemon, args)
      end
    end
  end
end