Skip to content

Commit

Permalink
Add default_content_type setting
Browse files Browse the repository at this point in the history
Historically, Sinatra::Response defaults to a text/html Content-Type.
However, in practice, Sinatra immediately clears this attribute after
instantiating a new Sinatra::Response object, so this default is some-
what suspect. Let's break this behavior and have Sinatra::Response
be Content-Type-less by default, and update the tests to reflect this.

Next, let's introduce a new default_content_type setting that will be
applied (instead of text/html) if the Content-Type is not set before the
response is finalized. This will be great for e.g. JSON API developers.
Let's also make it nil-able to indicate that a default Content-Type
should never be applied.

This could potentially lead to circumstances where there is content but
no Content-Type, so let's also modify Sinatra::Response to drop the body
of the response if the Content-Type is not set, and fix the errors
uncovered by that change.

Wherever Sinatra is emitting HTML, e.g. in error pages, force the
Content-Type to text/html.

Finally, clean up the error-handling logic to behave as follows:

* Set the X-Cascade header as early as possible.
* If an error block matches and returns a value, stop processing and
  return that value.
* If we have a not found or bad request error, inspect the exception (if
  any) for an error message and present it as the response body if it
  exists, or present a default error message.

The remaining logic is the same otherwise. This should make error
handlers simpler to write and behave more consistently by not polluting
the body with a default message when none may be necessary.
  • Loading branch information
mwpastore committed Jan 23, 2017
1 parent 27d36fd commit 80f460f
Show file tree
Hide file tree
Showing 5 changed files with 68 additions and 21 deletions.
8 changes: 8 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -2240,6 +2240,14 @@ set :protection, :session => true
<tt>localhost</tt> if your `environment` is set to development). Only used
for built-in server.</dd>

<dt>default_content_type</dt>
<dd>
Content-Type to assume if unknown (defaults to <tt>"text/html"</tt>). Set
to <tt>nil</tt> to not set a default Content-Type on every response; when
configured so, you must always set the Content-Type manually when emitting
content, or the body will be dropped.
</dd>

<dt>default_encoding</dt>
<dd>Encoding to assume if unknown (defaults to <tt>"utf-8"</tt>).</dd>

Expand Down
36 changes: 20 additions & 16 deletions lib/sinatra/base.rb
Original file line number Diff line number Diff line change
Expand Up @@ -132,10 +132,6 @@ def method_missing(*args, &block)
# http://rubydoc.info/github/rack/rack/master/Rack/Response/Helpers
class Response < Rack::Response
DROP_BODY_RESPONSES = [204, 205, 304]
def initialize(*)
super
headers['Content-Type'] ||= 'text/html'
end

def body=(value)
value = value.body while Rack::Response === value
Expand All @@ -162,7 +158,7 @@ def finish
if calculate_content_length?
# if some other code has already set Content-Length, don't muck with it
# currently, this would be the static file-handler
headers["Content-Length"] = body.inject(0) { |l, p| l + p.bytesize }.to_s
headers["Content-Length"] = body.map(&:bytesize).reduce(0, :+).to_s
end

[status.to_i, headers, result]
Expand All @@ -179,7 +175,7 @@ def drop_content_info?
end

def drop_body?
DROP_BODY_RESPONSES.include?(status.to_i)
!headers["Content-Type"] || DROP_BODY_RESPONSES.include?(status.to_i)
end
end

Expand Down Expand Up @@ -918,15 +914,14 @@ def call!(env) # :nodoc:
@response = Response.new
template_cache.clear if settings.reload_templates

@response['Content-Type'] = nil
invoke { dispatch! }
invoke { error_block!(response.status) } unless @env['sinatra.error']

unless @response['Content-Type']
if Array === body and body[0].respond_to? :content_type
if Array === body && body[0].respond_to?(:content_type)
content_type body[0].content_type
else
content_type :html
elsif default = settings.default_content_type
content_type default
end
end

Expand Down Expand Up @@ -1138,19 +1133,27 @@ def handle_exception!(boom)

status(500) unless status.between? 400, 599

boom_message = boom.message if boom.message && boom.message != boom.class.name
if server_error?
dump_errors! boom if settings.dump_errors?
raise boom if settings.show_exceptions? and settings.show_exceptions != :after_handler
elsif not_found?
headers['X-Cascade'] = 'pass' if settings.x_cascade?
body boom_message || '<h1>Not Found</h1>'
elsif bad_request?
body boom_message || '<h1>Bad Request</h1>'
end

res = error_block!(boom.class, boom) || error_block!(status, boom)
return res if res or not server_error?
if res = error_block!(boom.class, boom) || error_block!(status, boom)
return res
end

if not_found? || bad_request?
if boom.message && boom.message != boom.class.name
body boom.message
else
content_type 'text/html'
body '<h1>' + (not_found? ? 'Not Found' : 'Bad Request') + '</h1>'
end
end

return unless server_error?
raise boom if settings.raise_errors? or settings.show_exceptions?
error_block! Exception, boom
end
Expand Down Expand Up @@ -1798,6 +1801,7 @@ def self.force_encoding(data, *) data end
set :add_charset, %w[javascript xml xhtml+xml].map { |t| "application/#{t}" }
settings.add_charset << /^text\//
set :mustermann_opts, {}
set :default_content_type, 'text/html'

# explicitly generating a session secret eagerly to play nice with preforking
begin
Expand Down
8 changes: 4 additions & 4 deletions test/helpers_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -62,8 +62,8 @@ def status_app(code, &block)

describe 'informational?' do
it 'is true for 1xx status' do
status_app(100 + rand(100)) { informational? }
assert_body 'true'
status_app(100 + rand(100)) { response['X-Informational'] = informational?.to_s }
assert_equal 'true', response['X-Informational']
end

it 'is false for status > 199' do
Expand All @@ -79,8 +79,8 @@ def status_app(code, &block)
end

it 'is false for status < 200' do
status_app(100 + rand(100)) { success? }
assert_body 'false'
status_app(100 + rand(100)) { response['X-Success'] = success?.to_s }
assert_equal 'false', response['X-Success']
end

it 'is false for status > 299' do
Expand Down
2 changes: 1 addition & 1 deletion test/response_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
require File.expand_path('../helper', __FILE__)

class ResponseTest < Minitest::Test
setup { @response = Sinatra::Response.new }
setup { @response = Sinatra::Response.new([], 200, { 'Content-Type' => 'text/html' }) }

def assert_same_body(a, b)
assert_equal a.to_enum(:each).to_a, b.to_enum(:each).to_a
Expand Down
35 changes: 35 additions & 0 deletions test/settings_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -163,6 +163,41 @@ def foo=(value)
assert_equal :foo, @base.settings.environment
end

describe 'default_content_type' do
it 'defaults to html' do
assert_equal 'text/html', @base.default_content_type
end

it 'can be changed' do
@base.set :default_content_type, 'application/json'
@base.get('/') { '{"a":1}' }
@app = @base
get '/'
assert_equal 200, status
assert_equal 'application/json', response.content_type
end

it 'can be disabled' do
@base.set :default_content_type, nil
@base.error(404) { "" }
@app = @base
get '/'
assert_equal 404, status
assert_nil response.content_type
assert_empty body
end

it 'checks for a content-type before emitting content' do
@base.set :default_content_type, nil
@base.get('/') { raise Sinatra::BadRequest, "This is a drill" }
@app = @base
get '/'
assert_equal 400, status
assert_nil response.content_type
assert_empty body
end
end

describe 'methodoverride' do
it 'is disabled on Base' do
assert ! @base.method_override?
Expand Down

0 comments on commit 80f460f

Please sign in to comment.