Raises exception when respond_to called multiple times in incompatible way

Nesting respond_to calls can lead to unexpected behavior, so it should be
avoided. Currently, the first respond_to format match sets the content-type
for the resulting response. But, if a nested respond_to occurs, it is possible
to match on a different format. For example:

    respond_to do |outer_type|
      outer_type.js do
        respond_to do |inner_type|
          inner_type.html { render body: "HTML" }
        end
      end
    end

Browsers will often include */* in their Accept headers. In the above example,
such a request would result in the outer_type.js match setting the content-
type of the response to text/javascript, while the inner_type.html match will
cause the actual response to return "HTML".

This change tries to minimize potential breakage by only raising an exception
if the nested respond_to calls are in conflict with each other. So, something
like the following example would not raise an exception:

    respond_to do |outer_type|
      outer_type.js do
        respond_to do |inner_type|
          inner_type.js { render body: "JS" }
        end
      end
    end

While the above is nested, it doesn't affect the content-type of the response.
上级 15a72c6c
*. Raises `ActionController::RespondToMismatchError` with confliciting `respond_to` invocations.
`respond_to` can match multiple types and lead to undefined behavior when
multiple invocations are made and the types do not match:
respond_to do |outer_type|
outer_type.js do
respond_to do |inner_type|
inner_type.html { render body: "HTML" }
end
end
end
*Patrick Toomey*
* `ActionDispatch::Http::UploadedFile` now delegates `to_path` to its tempfile.
This allows uploaded file objects to be passed directly to `File.read`
......
......@@ -51,6 +51,24 @@ class UnknownHttpMethod < ActionControllerError #:nodoc:
class UnknownFormat < ActionControllerError #:nodoc:
end
# Raised when a nested respond_to is triggered and the content types of each
# are incompatible. For exampe:
#
# respond_to do |outer_type|
# outer_type.js do
# respond_to do |inner_type|
# inner_type.html { render body: "HTML" }
# end
# end
# end
class RespondToMismatchError < ActionControllerError
DEFAULT_MESSAGE = "respond_to was called multiple times and matched with conflicting formats in this action. Please note that you may only call respond_to and match on a single format per action."
def initialize(message = nil)
super(message || DEFAULT_MESSAGE)
end
end
class MissingExactTemplate < UnknownFormat #:nodoc:
end
end
......@@ -197,6 +197,9 @@ def respond_to(*mimes)
yield collector if block_given?
if format = collector.negotiate_format(request)
if content_type && content_type != format
raise ActionController::RespondToMismatchError
end
_process_format(format)
_set_rendered_content_type format
response = collector.response
......
......@@ -102,6 +102,26 @@ def made_for_content_type
end
end
def using_conflicting_nested_js_then_html
respond_to do |outer_type|
outer_type.js do
respond_to do |inner_type|
inner_type.html { render body: "HTML" }
end
end
end
end
def using_non_conflicting_nested_js_then_js
respond_to do |outer_type|
outer_type.js do
respond_to do |inner_type|
inner_type.js { render body: "JS" }
end
end
end
end
def custom_type_handling
respond_to do |type|
type.html { render body: "HTML" }
......@@ -430,6 +450,20 @@ def test_using_defaults_with_type_list
assert_equal "<p>Hello world!</p>\n", @response.body
end
def test_using_conflicting_nested_js_then_html
@request.accept = "*/*"
assert_raises(ActionController::RespondToMismatchError) do
get :using_conflicting_nested_js_then_html
end
end
def test_using_non_conflicting_nested_js_then_js
@request.accept = "*/*"
get :using_non_conflicting_nested_js_then_js
assert_equal "text/javascript", @response.content_type
assert_equal "JS", @response.body
end
def test_with_atom_content_type
@request.accept = ""
@request.env["CONTENT_TYPE"] = "application/atom+xml"
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册