Skip to content

Commit 6876b71

Browse files
ctennisdblock
authored andcommitted
When returning an HTML error, make sure it's safe (#1763)
* When calling into an API specifying a crafted format that is HTML, the returned error renders the HTML back to the user, causing a potential XSS issue. For example: http://example.com/api/endpoint?format=%3Cscript%3Ealert(document.cookie)%3C/script%3E Renders as html: The requested format '<script>alert(document.cookie)</script>' is not supported. When an error generates html back to the user, make sure it's properly escaped. Fixes issue #1762 * Add changelog entry * Use a method that also works in rails3 * Add spec formatting for older rails/activesupport version
1 parent 9a4b939 commit 6876b71

File tree

4 files changed

+36
-6
lines changed

4 files changed

+36
-6
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,8 @@
66

77
#### Fixes
88

9+
10+
* [#1762](https://github.com/ruby-grape/grape/pull/1763): Fix unsafe HTML rendering on errors - [@ctennis](https://github.com/ctennis).
911
* [#1759](https://github.com/ruby-grape/grape/pull/1759): Update appraisal for rails_edge - [@zvkemp](https://github.com/zvkemp).
1012
* [#1758](https://github.com/ruby-grape/grape/pull/1758): Fix expanding load_path in gemspec - [@2maz](https://github.com/2maz).
1113
* Your contribution here.

lib/grape/middleware/error.rb

+4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
require 'grape/middleware/base'
2+
require 'active_support/core_ext/string/output_safety'
23

34
module Grape
45
module Middleware
@@ -69,6 +70,9 @@ def error_response(error = {})
6970
end
7071

7172
def rack_response(message, status = options[:default_status], headers = { Grape::Http::Headers::CONTENT_TYPE => content_type })
73+
if headers[Grape::Http::Headers::CONTENT_TYPE] == TEXT_HTML
74+
message = ERB::Util.html_escape(message)
75+
end
7276
Rack::Response.new([message], status, headers).finish
7377
end
7478

spec/grape/api_spec.rb

+26-2
Original file line numberDiff line numberDiff line change
@@ -2142,7 +2142,11 @@ def self.call(message, _backtrace, _option, _env, _original_exception)
21422142
end
21432143
get '/excel.json'
21442144
expect(last_response.status).to eq(406)
2145-
expect(last_response.body).to eq("The requested format 'txt' is not supported.")
2145+
if ActiveSupport::VERSION::MAJOR == 3
2146+
expect(last_response.body).to eq('The requested format &#x27;txt&#x27; is not supported.')
2147+
else
2148+
expect(last_response.body).to eq('The requested format &#39;txt&#39; is not supported.')
2149+
end
21462150
end
21472151
end
21482152

@@ -3524,7 +3528,27 @@ def before
35243528
end
35253529
get '/something'
35263530
expect(last_response.status).to eq(406)
3527-
expect(last_response.body).to eq("{\"error\":\"The requested format 'txt' is not supported.\"}")
3531+
if ActiveSupport::VERSION::MAJOR == 3
3532+
expect(last_response.body).to eq('{&quot;error&quot;:&quot;The requested format &#x27;txt&#x27; is not supported.&quot;}')
3533+
else
3534+
expect(last_response.body).to eq('{&quot;error&quot;:&quot;The requested format &#39;txt&#39; is not supported.&quot;}')
3535+
end
3536+
end
3537+
end
3538+
3539+
context 'with unsafe HTML format specified' do
3540+
it 'escapes the HTML' do
3541+
subject.content_type :json, 'application/json'
3542+
subject.get '/something' do
3543+
'foo'
3544+
end
3545+
get '/something?format=<script>blah</script>'
3546+
expect(last_response.status).to eq(406)
3547+
if ActiveSupport::VERSION::MAJOR == 3
3548+
expect(last_response.body).to eq('The requested format &#x27;&lt;script&gt;blah&lt;/script&gt;&#x27; is not supported.')
3549+
else
3550+
expect(last_response.body).to eq('The requested format &#39;&lt;script&gt;blah&lt;/script&gt;&#39; is not supported.')
3551+
end
35283552
end
35293553
end
35303554

spec/grape/middleware/exception_spec.rb

+4-4
Original file line numberDiff line numberDiff line change
@@ -192,7 +192,7 @@ def app
192192
end
193193
it 'is possible to return errors in jsonapi format' do
194194
get '/'
195-
expect(last_response.body).to eq('{"error":"rain!"}')
195+
expect(last_response.body).to eq('{&quot;error&quot;:&quot;rain!&quot;}')
196196
end
197197
end
198198

@@ -207,8 +207,8 @@ def app
207207

208208
it 'is possible to return hash errors in jsonapi format' do
209209
get '/'
210-
expect(['{"error":"rain!","detail":"missing widget"}',
211-
'{"detail":"missing widget","error":"rain!"}']).to include(last_response.body)
210+
expect(['{&quot;error&quot;:&quot;rain!&quot;,&quot;detail&quot;:&quot;missing widget&quot;}',
211+
'{&quot;detail&quot;:&quot;missing widget&quot;,&quot;error&quot;:&quot;rain!&quot;}']).to include(last_response.body)
212212
end
213213
end
214214

@@ -258,7 +258,7 @@ def app
258258
end
259259
it 'is possible to specify a custom formatter' do
260260
get '/'
261-
expect(last_response.body).to eq('{:custom_formatter=>"rain!"}')
261+
expect(last_response.body).to eq('{:custom_formatter=&gt;&quot;rain!&quot;}')
262262
end
263263
end
264264

0 commit comments

Comments
 (0)