Skip to content

Commit

Permalink
Internal Sinatra errors now extend Sinatra::Error
Browse files Browse the repository at this point in the history
By extending Sinatra::Error, an error class can set the http status
code on the response to a value other than 500. This commit fixes
issues sinatra#1204 and sinatra#1518 where an error raised by a third party library
that responded to http_status could set the status on the response.
Any error outside of Sinatra errors will now always return a 500 status.

This fixes an issue where an exception could leak sensitive data in
the message to the browser. Errors that have http_status code 400 or
404 use the message as the body of the response. This is why it is
imperative that these errors extend Sinatra::Error so that this is
an explicit decision.
  • Loading branch information
jkowens committed Feb 1, 2019
1 parent 26fa5c8 commit 7397162
Show file tree
Hide file tree
Showing 3 changed files with 32 additions and 12 deletions.
23 changes: 14 additions & 9 deletions lib/sinatra/base.rb
Original file line number Diff line number Diff line change
Expand Up @@ -233,11 +233,14 @@ def call(env)
end
end

class BadRequest < TypeError #:nodoc:
class Error < StandardError #:nodoc:
end

class BadRequest < Error #:nodoc:
def http_status; 400 end
end

class NotFound < NameError #:nodoc:
class NotFound < Error #:nodoc:
def http_status; 404 end
end

Expand Down Expand Up @@ -1113,17 +1116,19 @@ def handle_exception!(boom)
end
@env['sinatra.error'] = boom

if boom.respond_to? :http_status
status(boom.http_status)
elsif settings.use_code? and boom.respond_to? :code and boom.code.between? 400, 599
status(boom.code)
else
status(500)
http_status = if boom.kind_of? Sinatra::Error
if boom.respond_to? :http_status
boom.http_status
elsif settings.use_code? && boom.respond_to?(:code)
boom.code
end
end

status(500) unless status.between? 400, 599
http_status = 500 unless http_status && http_status.between?(400, 599)
status(http_status)

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
Expand Down
6 changes: 3 additions & 3 deletions test/mapped_error_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -6,15 +6,15 @@ class FooError < RuntimeError
class FooNotFound < Sinatra::NotFound
end

class FooSpecialError < RuntimeError
class FooSpecialError < Sinatra::Error
def http_status; 501 end
end

class FooStatusOutOfRangeError < RuntimeError
class FooStatusOutOfRangeError < Sinatra::Error
def code; 4000 end
end

class FooWithCode < RuntimeError
class FooWithCode < Sinatra::Error
def code; 419 end
end

Expand Down
15 changes: 15 additions & 0 deletions test/result_test.rb
Original file line number Diff line number Diff line change
@@ -1,5 +1,9 @@
require File.expand_path('../helper', __FILE__)

class ThirdPartyError < RuntimeError
def http_status; 400 end
end

class ResultTest < Minitest::Test
it "sets response.body when result is a String" do
mock_app { get('/') { 'Hello World' } }
Expand Down Expand Up @@ -73,4 +77,15 @@ def res.each ; yield call ; end
assert_equal 205, status
assert_equal '', body
end

it "sets status to 500 when raised error is not Sinatra::Error" do
mock_app do
set :raise_errors, false
get('/') { raise ThirdPartyError }
end

get '/'
assert_equal 500, status
assert_equal '<h1>Internal Server Error</h1>', body
end
end

0 comments on commit 7397162

Please sign in to comment.