class Daemons::ApplicationGroup

def create_monitor(an_app)

def create_monitor(an_app)
  if @monitor && options[:monitor]
    @monitor.stop
    @monitor = nil
  end
  if options[:monitor]
    opt = {}
    opt[:monitor_interval] = options[:monitor_interval] if options[:monitor_interval]
    @monitor = Monitor.new(an_app, opt)
    @monitor.start(self)
  end
end

def find_applications(dir)

def find_applications(dir)
  if @no_pidfiles
    return find_applications_by_app_name(app_name)
  else
    return find_applications_by_pidfiles(dir)
  end
end

def find_applications_by_app_name(app_name)

TODO: identifiy the monitor process
def find_applications_by_app_name(app_name)
  pids = []
  begin
    x = `ps auxw | grep -v grep | awk '{print $2, $11, $12}' | grep #{app_name}`
    if x && x.chomp!
      processes = x.split(/\n/).compact
      processes = processes.delete_if do |p|
        _pid, name, add = p.split(/\s/)
        # We want to make sure that the first part of the process name matches
        # so that app_name matches app_name_22
        app_name != name[0..(app_name.length - 1)] and not add.include?(app_name)
      end
      pids = processes.map { |p| p.split(/\s/)[0].to_i }
    end
  rescue ::Exception
  end
  pids.map do |f|
    app = Application.new(self, {}, PidMem.existing(f))
    setup_app(app)
    app
  end
end

def find_applications_by_pidfiles(dir)

def find_applications_by_pidfiles(dir)
  @monitor = Monitor.find(dir, app_name + '_monitor')
  reporter = Reporter.new(options)
  pid_files = PidFile.find_files(dir, app_name, ! @keep_pid_files, @pid_delimiter) do |pid, file|
    reporter.deleted_found_pidfile(pid, file)
  end
  pid_files.map do |f|
    app = Application.new(self, {}, PidFile.existing(f))
    setup_app(app)
    app
  end
end

def initialize(app_name, options = {})

def initialize(app_name, options = {})
  @app_name = app_name
  @options = options
  if @options[:script]
    @script = File.expand_path(@options[:script])
  end
  @monitor = nil
  @multiple = @options[:multiple] || false
  @dir_mode = @options[:dir_mode] || :script
  ['dir'].each do |k|
    @options[k] = File.expand_path(@options[k]) if @options.key?(k)
  end
  @dir = @options[:dir] || ''
  @keep_pid_files = @options[:keep_pid_files] || false
  @no_pidfiles = @options[:no_pidfiles] || false
  @pid_delimiter = @options[:pid_delimiter]
  @applications = []
end

def new_application(add_options = {})

def new_application(add_options = {})
  if @applications.size > 0 && !@multiple
    if options[:force]
      @applications.delete_if do |a|
        unless a.running?
          a.zap
          true
        end
      end
    end
    fail RuntimeException.new('there is already one or more instance(s) of the program running') unless @applications.empty?
  end
  app = Application.new(self, add_options)
  setup_app(app)
  @applications << app
  app
end

def pidfile_dir

def pidfile_dir
  PidFile.dir(@dir_mode, @dir, script)
end

def reload_all

def reload_all
  @applications.each { |a| a.reload }
end

def running?

Check whether at least one of the applications in the group is running. If yes, return true.
def running?
  @applications.each { |a| return true if a.running? }
  return false
end

def setup


all running instances of the application and populates the application array.
Currently this functions calls find_applications which finds
Setup the application group.
def setup
  @applications = find_applications(pidfile_dir)
end

def setup_app(app)

def setup_app(app)
  app.controller_argv = @controller_argv
  app.app_argv = @app_argv
  if @options[:show_status_callback]
    app.show_status_callback = @options[:show_status_callback]
  end
end

def show_status

def show_status
  @applications.each { |a| a.show_status }
end

def start_all

def start_all
  @monitor.stop if @monitor
  @monitor = nil
  pids = []
  @applications.each do |a|
    pids << fork do
      a.start
    end
  end
  pids.each { |pid| Process.waitpid(pid) }
end

def stop_all(no_wait = false)

def stop_all(no_wait = false)
  if @monitor
    @monitor.stop
    @monitor = nil
    setup
  end
  threads = []
  @applications.each do |a|
    threads << Thread.new do
      a.stop(no_wait)
    end
  end
  threads.each { |t| t.join }
end

def zap_all

def zap_all
  @monitor.stop if @monitor
  @applications.each { |a| a.zap }
end