lib/active_admin/reloader.rb



module ActiveAdmin

  module Reloader

    # Builds the proper Reloader implementation class given the
    # current version of Rails.
    #
    # @param [Rails::Application] rails_app The current rails application
    # @param [ActiveAdmin::Application] active_admin_app The current Active Admin app
    # @param [String] rails_version The version of Rails we're using.
    #
    # @returns A concrete subclass of AbstractReloader
    def self.build(rails_app, active_admin_app, rails_version)
      reloader_class = rails_version.to_f >= 3.2 ? Rails32Reloader : RailsLessThan31Reloader
      reloader_class.new(rails_app, active_admin_app, rails_version)
    end

    class AbstractReloader

      attr_reader :active_admin_app, :rails_app, :rails_version

      def initialize(rails_app, active_admin_app, rails_version)
        @rails_app = rails_app
        @active_admin_app = active_admin_app
        @rails_version = rails_version.to_s
      end

      def attach!
        raise "Please implement #{self.class}#attach!"
      end

      def reload!
        active_admin_app.unload!
        rails_app.reload_routes!
      end

      def major_rails_version
        @rails_version[0..2]
      end

    end

    # Reloads the application when using Rails 3.2
    #
    # 3.2 introduced a to_prepare block that only gets called when
    # files have actually changed. ActiveAdmin had built this functionality
    # in to speed up applications. So in Rails >= 3.2, we can now piggy
    # back off the existing reloader. This simplifies our duties... which is
    # nice.
    class Rails32Reloader < AbstractReloader

      # Attach to Rails and perform the reload on each request.
      def attach!
        active_admin_app.load_paths.each do |path|
          rails_app.config.watchable_dirs[path] = [:rb]
        end

        reloader = self

        ActionDispatch::Reloader.to_prepare do
          reloader.reload!
        end
      end

    end

    # Deals with reloading Active Admin on each request in
    # development and once in production in Rails < 3.2.
    class RailsLessThan31Reloader < AbstractReloader

      class FileUpdateChecker < ::ActiveSupport::FileUpdateChecker
        def paths
          # hack to support both Rails 3.1 and 3.2
          @files || @paths
        end

        # Over-ride the default #updated_at to support the deletion of files
        def updated_at
          paths.map { |path| File.mtime(path) rescue Time.now }.max
        end

        def execute_if_updated
          super
        end
      end

      attr_reader :file_update_checker

      def initialize(rails_app, active_admin_app, rails_version)
        super
        @file_update_checker = FileUpdateChecker.new(watched_paths) do
          reload!
        end
      end

      def reload!
        super
        file_update_checker.paths.clear
        watched_paths.each{|path| file_update_checker.paths << path }
      end


      # Attach to Rails and perform the reload on each request.
      def attach!
        # Bring the checker into local scope for the ruby block
        checker = file_update_checker

        reloader_class.to_prepare do
          checker.execute_if_updated
        end
      end

      def watched_paths
        paths = active_admin_app.load_paths
        active_admin_app.load_paths.each{|path| paths += Dir[File.join(path, "**", "*.rb")]}
        paths
      end

      def reloader_class
        if major_rails_version == "3.1"
          ActionDispatch::Reloader
        else
          ActionDispatch::Callbacks
        end
      end

    end

  end
end