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.
Builtin 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" }
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" }
end
end
end
Using respond_with
with a block follows the same syntax as
respond_to
.
- A
- C
- D
- H
- J
- N
- R
- T
DEFAULT_ACTIONS_FOR_VERBS | = | { :post => :new, :patch => :edit, :put => :edit } |
[R] | controller | |
[R] | format | |
[R] | options | |
[R] | request | |
[R] | resource | |
[R] | resources |
Initializes a new responder and invokes the proper format. If the format is not defined, call to_format.
# File actionpack/lib/action_controller/metal/responder.rb, line 129 def initialize(controller, resources, options={}) @controller = controller @request = @controller.request @format = @controller.formats.first @resource = resources.last @resources = resources @options = options @action = options.delete(:action) @default_response = options.delete(:default_response) end
Main entry point for responder responsible to dispatch to the proper format.
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.
This is the common behavior for formats associated with APIs, such as :xml and :json.
# File actionpack/lib/action_controller/metal/responder.rb, line 203 def api_behavior(error) raise error unless resourceful? raise MissingRenderer.new(format) unless has_renderer? if get? display resource elsif post? display resource, :status => :created, :location => api_location else head :no_content end end
By default, render the :edit
action for HTML requests with errors, unless the verb was
POST.
If a response block was given, use it, otherwise call render on controller.
Display is just a shortcut to render a resource with the current format.
display @user, status: :ok
For XML requests it's equivalent to:
render xml: @user, status: :ok
Options sent by the user are also used:
respond_with(@user, status: :created)
display(@user, status: :ok)
Results in:
render xml: @user, status: :created
Check whether the resource has errors.
Check whether the necessary Renderer is available
This is the common behavior for formats associated with browsing, like :html, :iphone and so forth.
Returns the resource location by retrieving it from the options or returning the resources array.
Checks whether the resource responds to the current format or not.