1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00

Add nagivational behavior to respond_with.

This commit is contained in:
José Valim 2009-08-01 15:29:39 +02:00
parent c44f7e39f4
commit f59984cc81
9 changed files with 213 additions and 76 deletions

View file

@ -57,12 +57,13 @@ module ActionController
end
options = args.extract_options!
status = args.shift || options.delete(:status) || :ok
location = options.delete(:location)
options.each do |key, value|
headers[key.to_s.dasherize.split(/-/).map { |v| v.capitalize }.join("-")] = value.to_s
end
render :nothing => true, :status => status
render :nothing => true, :status => status, :location => location
end
# Sets the etag and/or last_modified on the response and checks it against

View file

@ -198,11 +198,11 @@ module ActionController #:nodoc:
end
# respond_with allows you to respond an action with a given resource. It
# requires that you set your class with a :respond_to method with the
# requires that you set your class with a respond_to method with the
# formats allowed:
#
# class PeopleController < ApplicationController
# respond_to :xml, :json
# respond_to :html, :xml, :json
#
# def index
# @people = Person.find(:all)
@ -210,14 +210,43 @@ module ActionController #:nodoc:
# end
# end
#
# When a request comes with format :xml, the respond_with will first search
# for a template as person/index.xml, if the template is not available, it
# will see if the given resource responds to :to_xml.
# When a request comes, for example with format :xml, three steps happen:
#
# If neither are available, it will raise an error.
# 1) respond_with searches for a template at people/index.xml;
#
# respond_with holds semantics for each HTTP verb. The example above cover
# GET requests. Let's check a POST request example:
# 2) if the template is not available, it will check if the given
# resource responds to :to_xml.
#
# 3) if a :location option was provided, redirect to the location with
# redirect status if a string was given, or render an action if a
# symbol was given.
#
# If all steps fail, a missing template error will be raised.
#
# === Supported options
#
# [status]
# Sets the response status.
#
# [head]
# Tell respond_with to set the content type, status and location header,
# but do not render the object, leaving the response body empty. This
# option only has effect if the resource is being rendered. If a
# template was found, it's going to be rendered anyway.
#
# [location]
# Sets the location header with the given value. It accepts a string,
# representing the location header value, or a symbol representing an
# action name.
#
# === Builtin HTTP verb semantics
#
# respond_with holds semantics for each HTTP verb. Depending on the verb
# and the resource status, respond_with will automatically set the options
# above.
#
# Above we saw an example for GET requests, where actually no option is
# configured. A create action for POST requests, could be written as:
#
# def create
# @person = Person.new(params[:person])
@ -225,34 +254,40 @@ module ActionController #:nodoc:
# respond_with(@person)
# end
#
# Since the request is a POST, respond_with will check wether @people
# resource have errors or not. If it has errors, it will render the error
# object with unprocessable entity status (422).
# respond_with will inspect the @person object and check if we have any
# error. If errors are empty, it will add status and location to the options
# hash. Then the create action in case of success, is equivalent to this:
#
# If no error was found, it will render the @people resource, with status
# created (201) and location header set to person_url(@people).
# respond_with(@person, :status => :created, :location => @person)
#
# If you also want to provide html behavior in the method above, you can
# supply a block to customize it:
# From them on, the lookup happens as described above. Let's suppose a :xml
# request and we don't have a people/create.xml template. But since the
# @person object responds to :to_xml, it will render the newly created
# resource and set status and location.
#
# class PeopleController < ApplicationController
# respond_to :html, :xml, :json # Add :html to respond_to definition
# However, if the request is :html, a template is not available and @person
# does not respond to :to_html. But since a :location options was provided,
# it will redirect to it.
#
# In case of failures (when the @person could not be saved and errors are
# not empty), respond_with can be expanded as this:
#
# respond_with(@person.errors, :status => :unprocessable_entity, :location => :new)
#
# In other words, respond_with(@person) for POST requests is expanded
# internally into this:
#
# def create
# @person = Person.new(params[:pe])
# @person = Person.new(params[:person])
#
# respond_with(@person) do |format|
# if @person.save
# flash[:notice] = 'Person was successfully created.'
# format.html { redirect_to @person }
# respond_with(@person, :status => :created, :location => @person)
# else
# format.html { render :action => "new" }
# end
# end
# respond_with(@person.errors, :status => :unprocessable_entity, :location => :new)
# end
# end
#
# It works similarly for PUT requests:
# For an update action for PUT requests, we would have:
#
# def update
# @person = Person.find(params[:id])
@ -260,10 +295,23 @@ module ActionController #:nodoc:
# respond_with(@person)
# end
#
# In case of failures, it works as POST requests, but in success failures
# it just reply status ok (200) to the client.
# Which, in face of success and failure scenarios, can be expanded as:
#
# A DELETE request also works in the same way:
# def update
# @person = Person.find(params[:id])
# @person.update_attributes(params[:person])
#
# if @person.save
# respond_with(@person, :status => :ok, :location => @person, :head => true)
# else
# respond_with(@person.errors, :status => :unprocessable_entity, :location => :edit)
# end
# end
#
# Notice that in case of success, we just need to reply :ok to the client.
# The option :head ensures that the object is not rendered.
#
# Finally, we have the destroy action with DELETE verb:
#
# def destroy
# @person = Person.find(params[:id])
@ -271,21 +319,30 @@ module ActionController #:nodoc:
# respond_with(@person)
# end
#
# It just replies with status ok, indicating the record was successfuly
# destroyed.
# Which is expanded as:
#
# def destroy
# @person = Person.find(params[:id])
# @person.destroy
# respond_with(@person, :status => :ok, :location => @person, :head => true)
# end
#
# In this case, since @person.destroyed? returns true, polymorphic urls will
# redirect to the collection url, instead of the resource url.
#
def respond_with(resource, options={}, &block)
respond_to(&block)
rescue ActionView::MissingTemplate => e
format = self.formats.first
resource = normalize_resource_options_by_verb(resource, options)
action = options.delete(:location) if options[:location].is_a?(Symbol)
if resource.respond_to?(:"to_#{format}")
if options.delete(:no_content)
head options
else
render options.merge(format => resource)
end
options.delete(:head) ? head(options) : render(options.merge(format => resource))
elsif action
render :action => action
elsif options[:location]
redirect_to options[:location]
else
raise e
end
@ -297,15 +354,21 @@ module ActionController #:nodoc:
#
def normalize_resource_options_by_verb(resource_or_array, options)
resource = resource_or_array.is_a?(Array) ? resource_or_array.last : resource_or_array
has_errors = resource.respond_to?(:errors) && !resource.errors.empty?
if has_errors && (request.post? || request.put?)
options.reverse_merge!(:status => :unprocessable_entity)
if resource.respond_to?(:errors) && !resource.errors.empty?
options[:status] ||= :unprocessable_entity
options[:location] ||= :new if request.post?
options[:location] ||= :edit if request.put?
return resource.errors
elsif request.post?
options.reverse_merge!(:status => :created, :location => resource_or_array)
elsif !request.get?
options.reverse_merge!(:status => :ok, :no_content => true)
options[:location] ||= resource_or_array
if request.post?
options[:status] ||= :created
else
options[:status] ||= :ok
options[:head] = true unless options.key?(:head)
end
end
return resource

View file

@ -86,12 +86,11 @@ module ActionController
else [ record_or_hash_or_array ]
end
inflection =
case
when options[:action].to_s == "new"
inflection = if options[:action].to_s == "new"
args.pop
:singular
when record.respond_to?(:new_record?) && record.new_record?
elsif (record.respond_to?(:new_record?) && record.new_record?) ||
(record.respond_to?(:destroyed?) && record.destroyed?)
args.pop
:plural
else

View file

@ -493,6 +493,10 @@ class RespondResource
def errors
[]
end
def destroyed?
false
end
end
class ParentResource
@ -508,7 +512,7 @@ end
class RespondWithController < ActionController::Base
respond_to :html, :json
respond_to :xml, :except => :using_defaults
respond_to :js, :only => :using_defaults
respond_to :js, :only => [ :using_defaults, :using_resource ]
def using_defaults
respond_to do |format|
@ -547,12 +551,16 @@ protected
self.response_body = js.respond_to?(:to_js) ? js.to_js : js
end
def resources_url
request.host + "/resources"
end
def resource_url(resource)
request.host + "/resource/#{resource.to_param}"
request.host + "/resources/#{resource.to_param}"
end
def parent_resource_url(parent, resource)
request.host + "/parent/#{parent.to_param}/resource/#{resource.to_param}"
request.host + "/parents/#{parent.to_param}/resources/#{resource.to_param}"
end
end
@ -617,10 +625,10 @@ class RespondWithControllerTest < ActionController::TestCase
end
def test_using_resource
@request.accept = "text/html"
@request.accept = "text/javascript"
get :using_resource
assert_equal "text/html", @response.content_type
assert_equal "Hello world!", @response.body
assert_equal "text/javascript", @response.content_type
assert_equal '$("body").visualEffect("highlight");', @response.body
@request.accept = "application/xml"
get :using_resource
@ -633,14 +641,30 @@ class RespondWithControllerTest < ActionController::TestCase
end
end
def test_using_resource_for_post
def test_using_resource_for_post_with_html
post :using_resource
assert_equal "text/html", @response.content_type
assert_equal 302, @response.status
assert_equal "www.example.com/resources/13", @response.location
assert @response.redirect?
errors = { :name => :invalid }
RespondResource.any_instance.stubs(:errors).returns(errors)
post :using_resource
assert_equal "text/html", @response.content_type
assert_equal 200, @response.status
assert_equal "New world!\n", @response.body
assert_nil @response.location
end
def test_using_resource_for_post_with_xml
@request.accept = "application/xml"
post :using_resource
assert_equal "application/xml", @response.content_type
assert_equal 201, @response.status
assert_equal "XML", @response.body
assert_equal "www.example.com/resource/13", @response.location
assert_equal "www.example.com/resources/13", @response.location
errors = { :name => :invalid }
RespondResource.any_instance.stubs(:errors).returns(errors)
@ -651,14 +675,30 @@ class RespondWithControllerTest < ActionController::TestCase
assert_nil @response.location
end
def test_using_resource_for_put
def test_using_resource_for_put_with_html
put :using_resource
assert_equal "text/html", @response.content_type
assert_equal 302, @response.status
assert_equal "www.example.com/resources/13", @response.location
assert @response.redirect?
errors = { :name => :invalid }
RespondResource.any_instance.stubs(:errors).returns(errors)
put :using_resource
assert_equal "text/html", @response.content_type
assert_equal 200, @response.status
assert_equal "Edit world!\n", @response.body
assert_nil @response.location
end
def test_using_resource_for_put_with_xml
@request.accept = "application/xml"
put :using_resource
assert_equal "application/xml", @response.content_type
assert_equal 200, @response.status
assert_equal " ", @response.body
assert_nil @response.location
assert_equal "www.example.com/resources/13", @response.location
errors = { :name => :invalid }
RespondResource.any_instance.stubs(:errors).returns(errors)
@ -669,13 +709,22 @@ class RespondWithControllerTest < ActionController::TestCase
assert_nil @response.location
end
def test_using_resource_for_delete
def test_using_resource_for_delete_with_html
RespondResource.any_instance.stubs(:destroyed?).returns(true)
delete :using_resource
assert_equal "text/html", @response.content_type
assert_equal 302, @response.status
assert_equal "www.example.com/resources", @response.location
end
def test_using_resource_for_delete_with_xml
RespondResource.any_instance.stubs(:destroyed?).returns(true)
@request.accept = "application/xml"
delete :using_resource
assert_equal "application/xml", @response.content_type
assert_equal 200, @response.status
assert_equal " ", @response.body
assert_nil @response.location
assert_equal "www.example.com/resources", @response.location
end
def test_using_resource_with_options
@ -692,14 +741,22 @@ class RespondWithControllerTest < ActionController::TestCase
assert_equal "JS", @response.body
end
def test_using_resource_with_parent
def test_using_resource_with_parent_for_get
@request.accept = "application/xml"
get :using_resource_with_parent
assert_equal "application/xml", @response.content_type
assert_equal 200, @response.status
assert_equal "XML", @response.body
end
def test_using_resource_with_parent_for_post
@request.accept = "application/xml"
post :using_resource_with_parent
assert_equal "application/xml", @response.content_type
assert_equal 201, @response.status
assert_equal "XML", @response.body
assert_equal "www.example.com/parent/11/resource/13", @response.location
assert_equal "www.example.com/parents/11/resources/13", @response.location
errors = { :name => :invalid }
RespondResource.any_instance.stubs(:errors).returns(errors)
@ -739,7 +796,7 @@ class RespondWithControllerTest < ActionController::TestCase
assert_equal 406, @response.status
@request.accept = "text/javascript"
get :using_resource
get :default_overwritten
assert_equal 406, @response.status
end
end

View file

@ -458,6 +458,10 @@ class TestController < ActionController::Base
head :location => "/foo"
end
def head_with_location_object
head :location => Customer.new("david")
end
def head_with_symbolic_status
head :status => params[:status].intern
end
@ -618,6 +622,10 @@ class TestController < ActionController::Base
end
private
def customer_url(customer)
request.host + "/customers/1"
end
def determine_layout
case action_name
when "hello_world", "layout_test", "rendering_without_layout",
@ -1084,6 +1092,13 @@ class RenderTest < ActionController::TestCase
assert_response :ok
end
def test_head_with_location_object
get :head_with_location_object
assert @response.body.blank?
assert_equal "www.nextangle.com/customers/1", @response.headers["Location"]
assert_response :ok
end
def test_head_with_custom_header
get :head_with_custom_header
assert @response.body.blank?

View file

@ -0,0 +1 @@
Edit world!

View file

@ -0,0 +1 @@
New world!

View file

@ -1 +0,0 @@
Hello world!

View file

@ -0,0 +1 @@
page[:body].visual_effect :highlight