base.rb 6.7 KB
Newer Older
1
require 'active_support/configurable'
2

3
module AbstractController
4 5
  class Error < StandardError; end
  class ActionNotFound < StandardError; end
J
Joshua Peek 已提交
6

7 8
  class Base
    attr_internal :response_body
9
    attr_internal :action_name
10

11 12
    include ActiveSupport::Configurable

13
    class << self
J
Joshua Peek 已提交
14
      attr_reader :abstract
15
      alias_method :abstract?, :abstract
J
Joshua Peek 已提交
16

17 18
      # Define a controller as abstract. See internal_methods for more
      # details.
19 20 21
      def abstract!
        @abstract = true
      end
J
Joshua Peek 已提交
22

23
      def inherited(klass)
24
        ::AbstractController::Base.descendants << klass.to_s
25 26 27
        super
      end

28 29 30 31
      # A list of all descendents of AbstractController::Base. This is
      # useful for initializers which need to add behavior to all controllers.
      def descendants
        @descendants ||= []
32
      end
J
Joshua Peek 已提交
33

34 35 36 37 38
      # A list of all internal methods for a controller. This finds the first
      # abstract superclass of a controller, and gets a list of all public
      # instance methods on that abstract class. Public instance methods of
      # a controller would normally be considered action methods, so we
      # are removing those methods on classes declared as abstract
39
      # (ActionController::Metal and ActionController::Base are defined
40
      # as abstract)
41 42 43 44 45
      def internal_methods
        controller = self
        controller = controller.superclass until controller.abstract?
        controller.public_instance_methods(true)
      end
J
Joshua Peek 已提交
46

47 48 49 50 51 52 53
      # The list of hidden actions to an empty Array. Defaults to an
      # empty Array. This can be modified by other modules or subclasses
      # to specify particular actions as hidden.
      #
      # ==== Returns
      # Array[String]:: An array of method names that should not be
      #                 considered actions.
54 55 56
      def hidden_actions
        []
      end
J
Joshua Peek 已提交
57

58 59 60 61 62 63 64 65 66
      # A list of method names that should be considered actions. This
      # includes all public instance methods on a controller, less
      # any internal methods (see #internal_methods), adding back in
      # any methods that are internal, but still exist on the class
      # itself. Finally, #hidden_actions are removed.
      #
      # ==== Returns
      # Array[String]:: A list of all methods that should be considered
      #                 actions.
67
      def action_methods
68
        @action_methods ||= begin
69
          # All public instance methods of this class, including ancestors
70 71 72 73 74 75 76 77 78 79 80
          methods = public_instance_methods(true).map { |m| m.to_s }.to_set -
            # Except for public instance methods of Base and its ancestors
            internal_methods.map { |m| m.to_s } +
            # Be sure to include shadowed public instance methods of this class
            public_instance_methods(false).map { |m| m.to_s } -
            # And always exclude explicitly hidden actions
            hidden_actions

          # Clear out AS callback method pollution
          methods.reject { |method| method =~ /_one_time_conditions/ }
        end
81
      end
82 83 84 85 86 87 88 89

      # Returns the full controller name, underscored, without the ending Controller.
      # For instance, MyApp::MyPostsController would return "my_app/my_posts" for
      # controller_name.
      #
      # ==== Returns
      # String
      def controller_path
90
        @controller_path ||= name.sub(/Controller$/, '').underscore unless anonymous?
91
      end
92
    end
J
Joshua Peek 已提交
93

94
    abstract!
J
Joshua Peek 已提交
95

96
    # Calls the action going through the entire action dispatch stack.
97
    #
98 99 100 101 102 103
    # The actual method that is called is determined by calling
    # #method_for_action. If no method can handle the action, then an
    # ActionNotFound error is raised.
    #
    # ==== Returns
    # self
104
    def process(action, *args)
105
      @_action_name = action_name = action.to_s
106

107
      unless action_name = method_for_action(action_name)
108
        raise ActionNotFound, "The action '#{action}' could not be found"
109
      end
110

J
Joshua Peek 已提交
111 112
      @_response_body = nil

113
      process_action(action_name, *args)
114
    end
J
Joshua Peek 已提交
115

116 117 118 119 120
    # Delegates to the class' #controller_path
    def controller_path
      self.class.controller_path
    end

121 122
    def action_methods
      self.class.action_methods
123
    end
J
Joshua Peek 已提交
124

125
    private
126

127 128 129 130 131 132 133 134 135 136 137 138
      # Returns true if the name can be considered an action. This can
      # be overridden in subclasses to modify the semantics of what
      # can be considered an action.
      #
      # ==== Parameters
      # name<String>:: The name of an action to be tested
      #
      # ==== Returns
      # TrueClass, FalseClass
      def action_method?(name)
        self.class.action_methods.include?(name)
      end
139

140 141 142 143 144 145
      # Call the action. Override this in a subclass to modify the
      # behavior around processing an action. This, and not #process,
      # is the intended way to override action dispatching.
      def process_action(method_name, *args)
        send_action(method_name, *args)
      end
146

147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187
      # Actually call the method associated with the action. Override
      # this method if you wish to change how action methods are called,
      # not to add additional behavior around it. For example, you would
      # override #send_action if you want to inject arguments into the
      # method.
      alias send_action send

      # If the action name was not found, but a method called "action_missing"
      # was found, #method_for_action will return "_handle_action_missing".
      # This method calls #action_missing with the current action name.
      def _handle_action_missing
        action_missing(@_action_name)
      end

      # Takes an action name and returns the name of the method that will
      # handle the action. In normal cases, this method returns the same
      # name as it receives. By default, if #method_for_action receives
      # a name that is not an action, it will look for an #action_missing
      # method and return "_handle_action_missing" if one is found.
      #
      # Subclasses may override this method to add additional conditions
      # that should be considered an action. For instance, an HTTP controller
      # with a template matching the action name is considered to exist.
      #
      # If you override this method to handle additional cases, you may
      # also provide a method (like _handle_method_missing) to handle
      # the case.
      #
      # If none of these conditions are true, and method_for_action
      # returns nil, an ActionNotFound exception will be raised.
      #
      # ==== Parameters
      # action_name<String>:: An action name to find a method name for
      #
      # ==== Returns
      # String:: The name of the method that handles the action
      # nil::    No method name could be found. Raise ActionNotFound.
      def method_for_action(action_name)
        if action_method?(action_name) then action_name
        elsif respond_to?(:action_missing, true) then "_handle_action_missing"
        end
188
      end
189
  end
J
Joshua Peek 已提交
190
end