base.rb 7.2 KB
Newer Older
1
require 'active_support/configurable'
2
require 'active_support/descendants_tracker'
3
require 'active_support/core_ext/module/anonymous'
4

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

J
Joost Baaij 已提交
9 10
  # <tt>AbstractController::Base</tt> is a low-level API. Nobody should be
  # using it directly, and subclasses (like ActionController::Base) are
11 12
  # expected to provide their own +render+ method, since rendering means
  # different things depending on the context.  
13 14
  class Base
    attr_internal :response_body
15
    attr_internal :action_name
16

17
    include ActiveSupport::Configurable
18
    extend ActiveSupport::DescendantsTracker
19

20
    class << self
J
Joshua Peek 已提交
21
      attr_reader :abstract
22
      alias_method :abstract?, :abstract
J
Joshua Peek 已提交
23

24 25
      # Define a controller as abstract. See internal_methods for more
      # details.
26 27 28
      def abstract!
        @abstract = true
      end
J
Joshua Peek 已提交
29

30 31 32 33 34
      # 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
35
      # (ActionController::Metal and ActionController::Base are defined
36
      # as abstract)
37 38 39 40 41
      def internal_methods
        controller = self
        controller = controller.superclass until controller.abstract?
        controller.public_instance_methods(true)
      end
J
Joshua Peek 已提交
42

J
Joost Baaij 已提交
43 44
      # The list of hidden actions to an empty array. Defaults to an
      # empty array. This can be modified by other modules or subclasses
45 46 47
      # to specify particular actions as hidden.
      #
      # ==== Returns
J
Joost Baaij 已提交
48
      # * <tt>array</tt> - An array of method names that should not be considered actions.
49 50 51
      def hidden_actions
        []
      end
J
Joshua Peek 已提交
52

53 54 55 56 57 58 59
      # 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
J
Joost Baaij 已提交
60
      # * <tt>array</tt> - A list of all methods that should be considered actions.
61
      def action_methods
62
        @action_methods ||= begin
63
          # All public instance methods of this class, including ancestors
A
Aaron Patterson 已提交
64
          methods = (public_instance_methods(true) -
65
            # Except for public instance methods of Base and its ancestors
A
Aaron Patterson 已提交
66
            internal_methods +
67
            # Be sure to include shadowed public instance methods of this class
A
Aaron Patterson 已提交
68
            public_instance_methods(false)).map { |x| x.to_s } -
69
            # And always exclude explicitly hidden actions
A
Aaron Patterson 已提交
70
            hidden_actions.to_a
71 72 73 74

          # Clear out AS callback method pollution
          methods.reject { |method| method =~ /_one_time_conditions/ }
        end
75
      end
76

77 78 79 80 81 82 83
      # action_methods are cached and there is sometimes need to refresh
      # them. clear_action_methods! allows you to do that, so next time
      # you run action_methods, they will be recalculated
      def clear_action_methods!
        @action_methods = nil
      end

84 85 86 87 88
      # Returns the full controller name, underscored, without the ending Controller.
      # For instance, MyApp::MyPostsController would return "my_app/my_posts" for
      # controller_name.
      #
      # ==== Returns
J
Joost Baaij 已提交
89
      # * <tt>string</tt>
90
      def controller_path
91
        @controller_path ||= name.sub(/Controller$/, '').underscore unless anonymous?
92
      end
93 94 95 96 97

      def method_added(name)
        super
        clear_action_methods!
      end
98
    end
J
Joshua Peek 已提交
99

100
    abstract!
J
Joshua Peek 已提交
101

102
    # Calls the action going through the entire action dispatch stack.
103
    #
104 105 106 107 108
    # 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
109
    # * <tt>self</tt>
110
    def process(action, *args)
111
      @_action_name = action_name = action.to_s
112

113
      unless action_name = method_for_action(action_name)
114
        raise ActionNotFound, "The action '#{action}' could not be found for #{self.class.name}"
115
      end
116

J
Joshua Peek 已提交
117 118
      @_response_body = nil

119
      process_action(action_name, *args)
120
    end
J
Joshua Peek 已提交
121

122 123 124 125 126
    # Delegates to the class' #controller_path
    def controller_path
      self.class.controller_path
    end

127 128
    def action_methods
      self.class.action_methods
129
    end
J
Joshua Peek 已提交
130

131
    private
132

133 134 135 136 137
      # 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
138
      # * <tt>name</tt> - The name of an action to be tested
139 140
      #
      # ==== Returns
141
      # * <tt>TrueClass</tt>, <tt>FalseClass</tt>
142 143 144
      def action_method?(name)
        self.class.action_methods.include?(name)
      end
145

146 147 148 149 150 151
      # 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
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
      # 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
185
      # * <tt>action_name</tt> - An action name to find a method name for
186 187
      #
      # ==== Returns
J
Joost Baaij 已提交
188
      # * <tt>string</tt> - The name of the method that handles the action
189
      # * <tt>nil</tt>    - No method name could be found. Raise ActionNotFound.
190 191 192 193
      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
194
      end
195
  end
J
Joshua Peek 已提交
196
end