Class: ActionController::Responder
- Inherits:
-
Object
- Object
- ActionController::Responder
- Defined in:
- lib/action_controller/responder.rb
Overview
Responsible for exposing a resource to different mime requests, usually depending on the HTTP verb. The responder is triggered when respond_with
is called. The simplest case to study is a GET request:
class PeopleController < ApplicationController
respond_to :html, :xml, :json
def index
@people = Person.all
respond_with(@people)
end
end
When a request comes in, for example for an XML response, three steps happen:
1) the responder searches for a template at people/index.xml;
2) if the template is not available, it will invoke <code>#to_xml</code> on the given resource;
3) if the responder does not <code>respond_to :to_xml</code>, call <code>#to_format</code> on it.
Built-in HTTP verb semantics
The default Rails responder holds semantics for each HTTP verb. Depending on the content type, verb and the resource status, it will behave differently.
Using Rails default responder, a POST request for creating an object could be written as:
def create
@user = User.new(params[:user])
flash[:notice] = 'User was successfully created.' if @user.save
respond_with(@user)
end
Which is exactly the same as:
def create
@user = User.new(params[:user])
respond_to do |format|
if @user.save
flash[:notice] = 'User was successfully created.'
format.html { redirect_to(@user) }
format.xml { render xml: @user, status: :created, location: @user }
else
format.html { render action: "new", status: :unprocessable_entity }
format.xml { render xml: @user.errors, status: :unprocessable_entity }
end
end
end
The same happens for PATCH/PUT and DELETE requests.
Nested resources
You can supply nested resources as you do in form_for
and polymorphic_url
. Consider the project has many tasks example. The create action for TasksController would be like:
def create
@project = Project.find(params[:project_id])
@task = @project.tasks.build(params[:task])
flash[:notice] = 'Task was successfully created.' if @task.save
respond_with(@project, @task)
end
Giving several resources ensures that the responder will redirect to project_task_url
instead of task_url
.
Namespaced and singleton resources require a symbol to be given, as in polymorphic urls. If a project has one manager which has many tasks, it should be invoked as:
respond_with(@project, :manager, @task)
Note that if you give an array, it will be treated as a collection, so the following is not equivalent:
respond_with [@project, :manager, @task]
Custom options
respond_with
also allows you to pass options that are forwarded to the underlying render call. Those options are only applied for success scenarios. For instance, you can do the following in the create method above:
def create
@project = Project.find(params[:project_id])
@task = @project.tasks.build(params[:task])
flash[:notice] = 'Task was successfully created.' if @task.save
respond_with(@project, @task, status: 201)
end
This will return status 201 if the task was saved successfully. If not, it will simply ignore the given options and return status 422 and the resource errors. You can also override the location to redirect to:
respond_with(@project, location: root_path)
To customize the failure scenario, you can pass a block to respond_with
:
def create
@project = Project.find(params[:project_id])
@task = @project.tasks.build(params[:task])
respond_with(@project, @task, status: 201) do |format|
if @task.save
flash[:notice] = 'Task was successfully created.'
else
format.html { render "some_special_template", status: :unprocessable_entity }
end
end
end
Using respond_with
with a block follows the same syntax as respond_to
.
Constant Summary collapse
- DEFAULT_ACTIONS_FOR_VERBS =
{ post: :new, patch: :edit, put: :edit }
Instance Attribute Summary collapse
-
#controller ⇒ Object
readonly
Returns the value of attribute controller.
-
#format ⇒ Object
readonly
Returns the value of attribute format.
-
#options ⇒ Object
readonly
Returns the value of attribute options.
-
#request ⇒ Object
readonly
Returns the value of attribute request.
-
#resource ⇒ Object
readonly
Returns the value of attribute resource.
-
#resources ⇒ Object
readonly
Returns the value of attribute resources.
Class Method Summary collapse
-
.call(*args) ⇒ Object
Initializes a new responder and invokes the proper format.
Instance Method Summary collapse
-
#initialize(controller, resources, options = {}) ⇒ Responder
constructor
A new instance of Responder.
-
#respond ⇒ Object
Main entry point for responder responsible to dispatch to the proper format.
-
#to_format ⇒ Object
All other formats follow the procedure below.
-
#to_html ⇒ Object
HTML format does not render the resource, it always attempt to render a template.
-
#to_js ⇒ Object
to_js simply tries to render a template.
Constructor Details
#initialize(controller, resources, options = {}) ⇒ Responder
Returns a new instance of Responder.
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 |
# File 'lib/action_controller/responder.rb', line 134 def initialize(controller, resources, = {}) @controller = controller @request = @controller.request @format = @controller.formats.first @resource = resources.last @resources = resources @options = @action = .delete(:action) @default_response = .delete(:default_response) if [:location].respond_to?(:call) location = .delete(:location) [:location] = location.call unless has_errors? end end |
Instance Attribute Details
#controller ⇒ Object (readonly)
Returns the value of attribute controller.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def controller @controller end |
#format ⇒ Object (readonly)
Returns the value of attribute format.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def format @format end |
#options ⇒ Object (readonly)
Returns the value of attribute options.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def @options end |
#request ⇒ Object (readonly)
Returns the value of attribute request.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def request @request end |
#resource ⇒ Object (readonly)
Returns the value of attribute resource.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def resource @resource end |
#resources ⇒ Object (readonly)
Returns the value of attribute resources.
126 127 128 |
# File 'lib/action_controller/responder.rb', line 126 def resources @resources end |
Class Method Details
.call(*args) ⇒ Object
Initializes a new responder and invokes the proper format. If the format is not defined, call to_format.
160 161 162 |
# File 'lib/action_controller/responder.rb', line 160 def self.call(*args) new(*args).respond end |
Instance Method Details
#respond ⇒ Object
Main entry point for responder responsible to dispatch to the proper format.
166 167 168 169 |
# File 'lib/action_controller/responder.rb', line 166 def respond method = "to_#{format}" respond_to?(method) ? send(method) : to_format end |
#to_format ⇒ Object
All other formats follow the procedure below. First we try to render a template, if the template is not available, we verify if the resource responds to :to_format and display it.
189 190 191 192 193 194 195 196 197 198 199 |
# File 'lib/action_controller/responder.rb', line 189 def to_format if !get? && has_errors? && !response_overridden? display_errors elsif has_view_rendering? || response_overridden? default_render else api_behavior end rescue ActionView::MissingTemplate api_behavior end |
#to_html ⇒ Object
HTML format does not render the resource, it always attempt to render a template.
174 175 176 177 178 |
# File 'lib/action_controller/responder.rb', line 174 def to_html default_render rescue ActionView::MissingTemplate => e (e) end |
#to_js ⇒ Object
to_js simply tries to render a template. If no template is found, raises the error.
181 182 183 |
# File 'lib/action_controller/responder.rb', line 181 def to_js default_render end |