1
0
Fork 0
mirror of https://github.com/rails/rails.git synced 2022-11-09 12:12:34 -05:00
rails--rails/actionpack/lib/action_controller/metal/head.rb
Jarek Radosz 38ec8db3f7 In head, set response body after headers.
When using ActionController::Live setting content type header after response body resulted in a race condition, throwing either “ActionDispatch::IllegalStateError: header already sent” or “RuntimeError: can't modify frozen Hash”.
This commit restores original order (i.e. from before 75757c5c3b).
2020-03-16 20:56:23 -07:00

63 lines
1.7 KiB
Ruby

# frozen_string_literal: true
module ActionController
module Head
# Returns a response that has no content (merely headers). The options
# argument is interpreted to be a hash of header names and values.
# This allows you to easily return a response that consists only of
# significant headers:
#
# head :created, location: person_path(@person)
#
# head :created, location: @person
#
# It can also be used to return exceptional conditions:
#
# return head(:method_not_allowed) unless request.post?
# return head(:bad_request) unless valid_request?
# render
#
# See Rack::Utils::SYMBOL_TO_STATUS_CODE for a full list of valid +status+ symbols.
def head(status, options = {})
if status.is_a?(Hash)
raise ArgumentError, "#{status.inspect} is not a valid value for `status`."
end
status ||= :ok
location = options.delete(:location)
content_type = options.delete(:content_type)
options.each do |key, value|
headers[key.to_s.split(/[-_]/).each { |v| v[0] = v[0].upcase }.join("-")] = value.to_s
end
self.status = status
self.location = url_for(location) if location
if include_content?(response_code)
unless self.media_type
self.content_type = content_type || (Mime[formats.first] if formats) || Mime[:html]
end
response.charset = false
end
self.response_body = ""
true
end
private
def include_content?(status)
case status
when 100..199
false
when 204, 205, 304
false
else
true
end
end
end
end