class ActionController::Responder

Parent:
Object

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.

Constants

DEFAULT_ACTIONS_FOR_VERBS

Attributes

controller[R]
format[R]
options[R]
request[R]
resource[R]
resources[R]

Public Class Methods

call(*args) Show source

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 150
def self.call(*args)
  new(*args).respond
end
new(controller, resources, options={}) Show source
# 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

Public Instance Methods

respond() Show source

Main entry point for responder responsible to dispatch to the proper format.

# File actionpack/lib/action_controller/metal/responder.rb, line 156
def respond
  method = "to_#{format}"
  respond_to?(method) ? send(method) : to_format
end
to_format() Show source

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.

# File actionpack/lib/action_controller/metal/responder.rb, line 179
def to_format
  if get? || !has_errors? || response_overridden?
    default_render
  else
    display_errors
  end
rescue ActionView::MissingTemplate => e
  api_behavior(e)
end
to_html() Show source

HTML format does not render the resource, it always attempt to render a template.

# File actionpack/lib/action_controller/metal/responder.rb, line 164
def to_html
  default_render
rescue ActionView::MissingTemplate => e
  navigation_behavior(e)
end
to_js() Show source

#to_js simply tries to render a template. If no template is found, raises the error.

# File actionpack/lib/action_controller/metal/responder.rb, line 171
def to_js
  default_render
end

Protected Instance Methods

api_behavior(error) Show source

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
api_location()
Alias for: resource_location
default_action() Show source

By default, render the :edit action for HTML requests with errors, unless the verb was POST.

# File actionpack/lib/action_controller/metal/responder.rb, line 281
def default_action
  @action ||= DEFAULT_ACTIONS_FOR_VERBS[request.request_method_symbol]
end
default_render() Show source

If a response block was given, use it, otherwise call render on controller.

# File actionpack/lib/action_controller/metal/responder.rb, line 234
def default_render
  if @default_response
    @default_response.call(options)
  else
    controller.default_render(options)
  end
end
display(resource, given_options={}) Show source

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
# File actionpack/lib/action_controller/metal/responder.rb, line 259
def display(resource, given_options={})
  controller.render given_options.merge!(options).merge!(format => resource)
end
display_errors() Show source
# File actionpack/lib/action_controller/metal/responder.rb, line 263
def display_errors
  controller.render format => resource_errors, :status => :unprocessable_entity
end
has_errors?() Show source

Check whether the resource has errors.

# File actionpack/lib/action_controller/metal/responder.rb, line 269
def has_errors?
  resource.respond_to?(:errors) && !resource.errors.empty?
end
has_renderer?() Show source

Check whether the necessary Renderer is available

# File actionpack/lib/action_controller/metal/responder.rb, line 274
def has_renderer?
  Renderers::RENDERERS.include?(format)
end
json_resource_errors() Show source
# File actionpack/lib/action_controller/metal/responder.rb, line 289
def json_resource_errors
  {:errors => resource.errors}
end
navigation_behavior(error) Show source

This is the common behavior for formats associated with browsing, like :html, :iphone and so forth.

navigation_location()
Alias for: resource_location
resource_errors() Show source
# File actionpack/lib/action_controller/metal/responder.rb, line 285
def resource_errors
  respond_to?("#{format}_resource_errors", true) ? send("#{format}_resource_errors") : resource.errors
end
resource_location() Show source

Returns the resource location by retrieving it from the options or returning the resources array.

# File actionpack/lib/action_controller/metal/responder.rb, line 225
def resource_location
  options[:location] || resources
end
Also aliased as: navigation_location, api_location
resourceful?() Show source

Checks whether the resource responds to the current format or not.

# File actionpack/lib/action_controller/metal/responder.rb, line 218
def resourceful?
  resource.respond_to?("to_#{format}")
end
response_overridden?() Show source
# File actionpack/lib/action_controller/metal/responder.rb, line 293
def response_overridden?
  @default_response.present?
end

© 2004–2016 David Heinemeier Hansson
Licensed under the MIT License.