http.rb 2.9 KB
Newer Older
J
Jeremy Kemper 已提交
1 2
require 'action_controller/abstract'

3
module ActionController
4 5 6 7 8 9
  # ActionController::Http provides a way to get a valid Rack application from a controller.
  #
  # In AbstractController, dispatching is triggered directly by calling #process on a new controller.
  # ActionController::Http provides an #action method that returns a valid Rack application for a
  # given action. Other rack builders, such as Rack::Builder, Rack::URLMap, and the Rails router,
  # can dispatch directly to the action returned by FooController.action(:index).
10 11
  class Http < AbstractController::Base
    abstract!
J
Joshua Peek 已提交
12

13
    # :api: public
14
    attr_internal :params, :env
15

16 17 18 19 20 21
    # Returns the last part of the controller's name, underscored, without the ending
    # "Controller". For instance, MyApp::MyPostsController would return "my_posts" for
    # controller_name
    #
    # ==== Returns
    # String
22 23 24 25
    def self.controller_name
      @controller_name ||= controller_path.split("/").last
    end

26
    # Delegates to the class' #controller_name
J
Joshua Peek 已提交
27 28 29
    def controller_name
      self.class.controller_name
    end
30

31 32 33 34 35 36
    # 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
37
    def self.controller_path
38
      @controller_path ||= name && name.sub(/Controller$/, '').underscore
39
    end
J
Joshua Peek 已提交
40

41
    # Delegates to the class' #controller_path
J
Joshua Peek 已提交
42 43 44 45
    def controller_path
      self.class.controller_path
    end

46 47 48 49 50
    # The details below can be overridden to support a specific
    # Request and Response object. The default ActionController::Base
    # implementation includes RackConvenience, which makes a request
    # and response object available. You might wish to control the
    # environment and response manually for performance reasons.
51

52 53 54 55 56 57 58
    attr_internal :status, :headers, :content_type

    def initialize(*)
      @_headers = {}
      super
    end

59
    # Basic implementations for content_type=, location=, and headers are
60 61 62 63 64 65 66 67 68
    # provided to reduce the dependency on the RackConvenience module
    # in Renderer and Redirector.

    def content_type=(type)
      headers["Content-Type"] = type.to_s
    end

    def location=(url)
      headers["Location"] = url
69
    end
J
Joshua Peek 已提交
70

71
    # :api: private
72
    def call(name, env)
73
      @_env = env
74 75 76
      process(name)
      to_rack
    end
J
Joshua Peek 已提交
77

78 79 80 81 82
    # :api: private
    def to_rack
      [status, headers, response_body]
    end

83 84 85 86 87 88 89 90 91
    # Return a rack endpoint for the given action. Memoize the endpoint, so
    # multiple calls into MyController.action will return the same object
    # for the same action.
    #
    # ==== Parameters
    # action<#to_s>:: An action name
    #
    # ==== Returns
    # Proc:: A rack application
92 93
    def self.action(name)
      @actions ||= {}
J
Jeremy Kemper 已提交
94
      @actions[name.to_s] ||= proc do |env|
95 96
        new.call(name, env)
      end
97 98 99
    end
  end
end