Skip to content

Commit

Permalink
Sanitize error output for prevent XSS security issues (#2782)
Browse files Browse the repository at this point in the history
* tests for XSS

* sanitize error output for prevent XSS issues

(cherry picked from commit 9c62600)
  • Loading branch information
msaf1980 authored and deniszh committed Feb 19, 2023
1 parent 5e19f3a commit f58586a
Show file tree
Hide file tree
Showing 3 changed files with 64 additions and 1 deletion.
11 changes: 10 additions & 1 deletion webapp/graphite/errors.py
Original file line number Diff line number Diff line change
Expand Up @@ -94,6 +94,15 @@ def __str__(self):
return msg


# Replace special characters "&", "<" and ">" to HTML-safe sequences.
def escape(s):
s = s.replace("&", "&amp;") # Must be done first!
s = s.replace("<", "&lt;")
s = s.replace(">", "&gt;")

return s


# decorator which turns InputParameterExceptions into Django's HttpResponseBadRequest
def handleInputParameterError(f):
def new_f(*args, **kwargs):
Expand All @@ -102,6 +111,6 @@ def new_f(*args, **kwargs):
except InputParameterError as e:
msgStr = str(e)
log.warning('%s', msgStr)
return HttpResponseBadRequest(msgStr)
return HttpResponseBadRequest(escape(msgStr))

return new_f
12 changes: 12 additions & 0 deletions webapp/tests/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,3 +5,15 @@
class TestCase(OriginalTestCase):
def tearDown(self):
stop_pools()

# Assert that a response is unsanitized (for check XSS issues)
def assertXSS(self, response, status_code=200, msg_prefix=''):
if status_code is not None:
self.assertEqual(
response.status_code, status_code,
msg_prefix + "Couldn't retrieve content: Response code was %d"
" (expected %d)" % (response.status_code, status_code)
)

xss = response.content.find(b"<") != -1 or response.content.find(b">") != -1
self.assertFalse(xss, msg=msg_prefix+str(response.content))
42 changes: 42 additions & 0 deletions webapp/tests/test_xss.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import logging
import sys

try:
from django.urls import reverse
except ImportError: # Django < 1.10
from django.core.urlresolvers import reverse

from .base import TestCase

# Silence logging during tests
LOGGER = logging.getLogger()

# logging.NullHandler is a python 2.7ism
if hasattr(logging, "NullHandler"):
LOGGER.addHandler(logging.NullHandler())

if sys.version_info[0] >= 3:
def resp_text(r):
return r.content.decode('utf-8')
else:
def resp_text(r):
return r.content


class RenderXSSTest(TestCase):
def test_render_xss(self):
url = reverse('render')
xssStr = '<noscript><p title="</noscript><img src=x onerror=alert() onmouseover=alert()>">'

# Check for issue #2779 and others
response = self.client.get(url, {'target': 'test', 'format': 'raw', 'cacheTimeout': xssStr, 'from': xssStr})
self.assertXSS(response, status_code=400, msg_prefix='XSS detected: ')


class FindXSSTest(TestCase):
def test_render_xss(self):
url = reverse('metrics_find')
xssStr = '<noscript><p title="</noscript><img src=x onerror=alert() onmouseover=alert()>">'

response = self.client.get(url, {'query': 'test', 'local': xssStr, 'from': xssStr, 'tz': xssStr})
self.assertXSS(response, status_code=400, msg_prefix='XSS detected: ')

0 comments on commit f58586a

Please sign in to comment.