Files
deb-python-falcon/tests/test_headers.py
Kurt Griffiths f28c90a5b4 perf(Request): Don't normalize CONTENT_* headers
Rather than always normalizing these headers up front, this patch
simply uses the names as they already exist in env. It was found
that this makes Request.__init__ take slightly less time to execute.

Issue #261
2014-10-17 13:15:28 -05:00

558 lines
19 KiB
Python

from collections import defaultdict
from datetime import datetime
import six
from testtools.matchers import Contains, Not
import falcon
import falcon.testing as testing
class StatusTestResource:
sample_body = testing.rand_string(0, 128 * 1024)
def __init__(self, status):
self.status = status
def on_get(self, req, resp):
resp.status = self.status
resp.body = self.sample_body
class XmlResource:
def __init__(self, content_type):
self.content_type = content_type
def on_get(self, req, resp):
resp.set_header('content-type', self.content_type)
class DefaultContentTypeResource:
def __init__(self, body=None):
self.body = body
def on_get(self, req, resp):
if self.body is not None:
resp.body = self.body
class HeaderHelpersResource:
def __init__(self, last_modified=None):
if last_modified is not None:
self.last_modified = last_modified
else:
self.last_modified = datetime.utcnow()
def _overwrite_headers(self, req, resp):
resp.content_type = 'x-falcon/peregrine'
resp.cache_control = ['no-store']
def on_get(self, req, resp):
resp.body = "{}"
resp.content_type = 'x-falcon/peregrine'
resp.cache_control = [
'public', 'private', 'no-cache', 'no-store', 'must-revalidate',
'proxy-revalidate', 'max-age=3600', 's-maxage=60', 'no-transform'
]
resp.etag = 'fa0d1a60ef6616bb28038515c8ea4cb2'
resp.last_modified = self.last_modified
resp.retry_after = 3601
# Relative URI's are OK per http://goo.gl/DbVqR
resp.location = '/things/87'
resp.content_location = '/things/78'
# bytes 0-499/10240
resp.content_range = (0, 499, 10 * 1024)
self.resp = resp
def on_head(self, req, resp):
resp.set_header('Content-Type', 'x-swallow/unladen')
resp.set_header('X-Auth-Token', 'setecastronomy')
resp.set_header('X-AUTH-TOKEN', 'toomanysecrets')
resp.location = '/things/87'
del resp.location
self._overwrite_headers(req, resp)
self.resp = resp
def on_post(self, req, resp):
resp.set_headers([
('CONTENT-TYPE', 'x-swallow/unladen'),
('X-Auth-Token', 'setecastronomy'),
('X-AUTH-TOKEN', 'toomanysecrets')
])
self._overwrite_headers(req, resp)
self.resp = resp
def on_put(self, req, resp):
resp.set_headers({
'CONTENT-TYPE': 'x-swallow/unladen',
'X-aUTH-tOKEN': 'toomanysecrets'
})
self._overwrite_headers(req, resp)
self.resp = resp
class LocationHeaderUnicodeResource:
URL1 = u'/\u00e7runchy/bacon'
URL2 = u'ab\u00e7' if six.PY3 else 'ab\xc3\xa7'
def on_get(self, req, resp):
resp.location = self.URL1
resp.content_location = self.URL2
def on_head(self, req, resp):
resp.location = self.URL2
resp.content_location = self.URL1
class VaryHeaderResource:
def __init__(self, vary):
self.vary = vary
def on_get(self, req, resp):
resp.body = '{}'
resp.vary = self.vary
class LinkHeaderResource:
def __init__(self):
self._links = []
def add_link(self, *args, **kwargs):
self._links.append((args, kwargs))
def on_get(self, req, resp):
resp.body = '{}'
for args, kwargs in self._links:
resp.add_link(*args, **kwargs)
class AppendHeaderResource:
def on_get(self, req, resp):
resp.append_header('X-Things', 'thing-1')
resp.append_header('X-THINGS', 'thing-2')
resp.append_header('x-thiNgs', 'thing-3')
def on_head(self, req, resp):
resp.set_header('X-things', 'thing-1')
resp.append_header('X-THINGS', 'thing-2')
resp.append_header('x-thiNgs', 'thing-3')
def on_post(self, req, resp):
resp.append_header('X-Things', 'thing-1')
class TestHeaders(testing.TestBase):
def before(self):
self.resource = testing.TestResource()
self.api.add_route(self.test_route, self.resource)
def test_content_length(self):
self.simulate_request(self.test_route)
headers = self.srmock.headers
# Test Content-Length header set
content_length = str(len(self.resource.sample_body))
content_length_header = ('content-length', content_length)
self.assertThat(headers, Contains(content_length_header))
def test_default_value(self):
self.simulate_request(self.test_route)
value = self.resource.req.get_header('X-Not-Found') or '876'
self.assertEqual(value, '876')
def test_required_header(self):
self.simulate_request(self.test_route)
self.assertRaises(falcon.HTTPBadRequest,
self.resource.req.get_header, 'X-Not-Found',
required=True)
def test_no_body_on_100(self):
self.resource = StatusTestResource(falcon.HTTP_100)
self.api.add_route('/1xx', self.resource)
body = self.simulate_request('/1xx')
self.assertThat(self.srmock.headers_dict,
Not(Contains('Content-Length')))
self.assertEqual(body, [])
def test_no_body_on_101(self):
self.resource = StatusTestResource(falcon.HTTP_101)
self.api.add_route('/1xx', self.resource)
body = self.simulate_request('/1xx')
self.assertThat(self.srmock.headers_dict,
Not(Contains('Content-Length')))
self.assertEqual(body, [])
def test_no_body_on_204(self):
self.resource = StatusTestResource(falcon.HTTP_204)
self.api.add_route('/204', self.resource)
body = self.simulate_request('/204')
self.assertThat(self.srmock.headers_dict,
Not(Contains('Content-Length')))
self.assertEqual(body, [])
def test_no_body_on_304(self):
self.resource = StatusTestResource(falcon.HTTP_304)
self.api.add_route('/304', self.resource)
body = self.simulate_request('/304')
self.assertThat(self.srmock.headers_dict,
Not(Contains('Content-Length')))
self.assertEqual(body, [])
def test_content_header_missing(self):
environ = testing.create_environ()
req = falcon.Request(environ)
for header in ('Content-Type', 'Content-Length'):
self.assertIs(req.get_header(header), None)
def test_passthrough_req_headers(self):
req_headers = {
'X-Auth-Token': 'Setec Astronomy',
'Content-Type': 'text/plain; charset=utf-8'
}
self.simulate_request(self.test_route, headers=req_headers)
for name, expected_value in req_headers.items():
actual_value = self.resource.req.get_header(name)
self.assertEqual(actual_value, expected_value)
self.simulate_request(self.test_route,
headers=self.resource.req.headers)
# Compare the request HTTP headers with the original headers
for name, expected_value in req_headers.items():
actual_value = self.resource.req.get_header(name)
self.assertEqual(actual_value, expected_value)
def test_get_raw_headers(self):
headers = [
('Client-ID', '692ba466-74bb-11e3-bf3f-7567c531c7ca'),
('Accept', 'audio/*; q=0.2, audio/basic')
]
environ = testing.create_environ(headers=headers)
req = falcon.Request(environ)
for name, value in headers:
self.assertIn((name.upper(), value), req.headers.items())
def test_passthrough_resp_headers(self):
self.simulate_request(self.test_route)
resp_headers = self.srmock.headers
for name, value in self.resource.resp_headers.items():
expected = (name.lower(), value)
self.assertThat(resp_headers, Contains(expected))
def test_default_media_type(self):
self.resource = DefaultContentTypeResource('Hello world!')
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
content_type = falcon.DEFAULT_MEDIA_TYPE
self.assertIn(('content-type', content_type), self.srmock.headers)
def test_custom_media_type(self):
self.resource = DefaultContentTypeResource('Hello world!')
self.api = falcon.API(media_type='application/atom+xml')
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
content_type = 'application/atom+xml'
self.assertIn(('content-type', content_type), self.srmock.headers)
def test_response_header_helpers_on_get(self):
last_modified = datetime(2013, 1, 1, 10, 30, 30)
self.resource = HeaderHelpersResource(last_modified)
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
resp = self.resource.resp
content_type = 'x-falcon/peregrine'
self.assertEqual(content_type, resp.content_type)
self.assertIn(('content-type', content_type), self.srmock.headers)
cache_control = ('public, private, no-cache, no-store, '
'must-revalidate, proxy-revalidate, max-age=3600, '
's-maxage=60, no-transform')
self.assertEqual(cache_control, resp.cache_control)
self.assertIn(('cache-control', cache_control), self.srmock.headers)
etag = 'fa0d1a60ef6616bb28038515c8ea4cb2'
self.assertEqual(etag, resp.etag)
self.assertIn(('etag', etag), self.srmock.headers)
last_modified_http_date = 'Tue, 01 Jan 2013 10:30:30 GMT'
self.assertEqual(last_modified_http_date, resp.last_modified)
self.assertIn(('last-modified', last_modified_http_date),
self.srmock.headers)
self.assertEqual('3601', resp.retry_after)
self.assertIn(('retry-after', '3601'), self.srmock.headers)
self.assertEqual('/things/87', resp.location)
self.assertIn(('location', '/things/87'), self.srmock.headers)
self.assertEqual('/things/78', resp.content_location)
self.assertIn(('content-location', '/things/78'), self.srmock.headers)
self.assertEqual('bytes 0-499/10240', resp.content_range)
self.assertIn(('content-range', 'bytes 0-499/10240'),
self.srmock.headers)
# Check for duplicate headers
hist = defaultdict(lambda: 0)
for name, value in self.srmock.headers:
hist[name] += 1
self.assertEqual(1, hist[name])
def test_unicode_location_headers(self):
self.api.add_route(self.test_route, LocationHeaderUnicodeResource())
self.simulate_request(self.test_route)
location = ('location', '/%C3%A7runchy/bacon')
self.assertIn(location, self.srmock.headers)
content_location = ('content-location', 'ab%C3%A7')
self.assertIn(content_location, self.srmock.headers)
# Test with the values swapped
self.simulate_request(self.test_route, method='HEAD')
location = ('location', 'ab%C3%A7')
self.assertIn(location, self.srmock.headers)
content_location = ('content-location', '/%C3%A7runchy/bacon')
self.assertIn(content_location, self.srmock.headers)
def test_response_set_header(self):
self.resource = HeaderHelpersResource()
self.api.add_route(self.test_route, self.resource)
for method in ('HEAD', 'POST', 'PUT'):
self.simulate_request(self.test_route, method=method)
content_type = 'x-falcon/peregrine'
self.assertIn(('content-type', content_type), self.srmock.headers)
self.assertIn(('cache-control', 'no-store'), self.srmock.headers)
self.assertIn(('x-auth-token', 'toomanysecrets'),
self.srmock.headers)
self.assertEqual(None, self.resource.resp.location)
# Check for duplicate headers
hist = defaultdict(lambda: 0)
for name, value in self.srmock.headers:
hist[name] += 1
self.assertEqual(1, hist[name])
def test_response_append_header(self):
self.resource = AppendHeaderResource()
self.api.add_route(self.test_route, self.resource)
for method in ('HEAD', 'GET'):
self.simulate_request(self.test_route, method=method)
value = self.srmock.headers_dict['x-things']
self.assertEqual('thing-1,thing-2,thing-3', value)
self.simulate_request(self.test_route, method='POST')
value = self.srmock.headers_dict['x-things']
self.assertEqual('thing-1', value)
def test_vary_star(self):
self.resource = VaryHeaderResource(['*'])
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
self.assertIn(('vary', '*'), self.srmock.headers)
def test_vary_header(self):
self.resource = VaryHeaderResource(['accept-encoding'])
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
self.assertIn(('vary', 'accept-encoding'), self.srmock.headers)
def test_vary_headers(self):
self.resource = VaryHeaderResource(['accept-encoding', 'x-auth-token'])
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
vary = 'accept-encoding, x-auth-token'
self.assertIn(('vary', vary), self.srmock.headers)
def test_vary_headers_tuple(self):
self.resource = VaryHeaderResource(('accept-encoding', 'x-auth-token'))
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
vary = 'accept-encoding, x-auth-token'
self.assertIn(('vary', vary), self.srmock.headers)
def test_no_content_type(self):
self.resource = DefaultContentTypeResource()
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
self.assertNotIn('content-type', self.srmock.headers_dict)
def test_custom_content_type(self):
content_type = 'application/xml; charset=utf-8'
self.resource = XmlResource(content_type)
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
self.assertIn(('content-type', content_type), self.srmock.headers)
def test_add_link_single(self):
expected_value = '</things/2842>; rel=next'
self.resource = LinkHeaderResource()
self.resource.add_link('/things/2842', 'next')
self._check_link_header(expected_value)
def test_add_link_multiple(self):
expected_value = (
'</things/2842>; rel=next, ' +
'<http://%C3%A7runchy/bacon>; rel=contents, ' +
'<ab%C3%A7>; rel="http://example.com/ext-type", ' +
'<ab%C3%A7>; rel="http://example.com/%C3%A7runchy", ' +
'<ab%C3%A7>; rel="https://example.com/too-%C3%A7runchy", ' +
'</alt-thing>; rel="alternate http://example.com/%C3%A7runchy"')
uri = u'ab\u00e7' if six.PY3 else 'ab\xc3\xa7'
self.resource = LinkHeaderResource()
self.resource.add_link('/things/2842', 'next')
self.resource.add_link(u'http://\u00e7runchy/bacon', 'contents')
self.resource.add_link(uri, 'http://example.com/ext-type')
self.resource.add_link(uri, u'http://example.com/\u00e7runchy')
self.resource.add_link(uri, u'https://example.com/too-\u00e7runchy')
self.resource.add_link('/alt-thing',
u'alternate http://example.com/\u00e7runchy')
self._check_link_header(expected_value)
def test_add_link_with_title(self):
expected_value = ('</related/thing>; rel=item; '
'title="A related thing"')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'item',
title='A related thing')
self._check_link_header(expected_value)
def test_add_link_with_title_star(self):
expected_value = ('</related/thing>; rel=item; '
"title*=UTF-8''A%20related%20thing, "
'</%C3%A7runchy/thing>; rel=item; '
"title*=UTF-8'en'A%20%C3%A7runchy%20thing")
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'item',
title_star=('', 'A related thing'))
self.resource.add_link(u'/\u00e7runchy/thing', 'item',
title_star=('en', u'A \u00e7runchy thing'))
self._check_link_header(expected_value)
def test_add_link_with_anchor(self):
expected_value = ('</related/thing>; rel=item; '
'anchor="/some%20thing/or-other"')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'item',
anchor='/some thing/or-other')
self._check_link_header(expected_value)
def test_add_link_with_hreflang(self):
expected_value = ('</related/thing>; rel=about; '
'hreflang=en')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'about',
hreflang='en')
self._check_link_header(expected_value)
def test_add_link_with_hreflang_multi(self):
expected_value = ('</related/thing>; rel=about; '
'hreflang=en-GB; hreflang=de')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'about',
hreflang=('en-GB', 'de'))
self._check_link_header(expected_value)
def test_add_link_with_type_hint(self):
expected_value = ('</related/thing>; rel=alternate; '
'type="video/mp4; codecs=avc1.640028"')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'alternate',
type_hint='video/mp4; codecs=avc1.640028')
self._check_link_header(expected_value)
def test_add_link_complex(self):
expected_value = ('</related/thing>; rel=alternate; '
'title="A related thing"; '
"title*=UTF-8'en'A%20%C3%A7runchy%20thing; "
'type="application/json"; '
'hreflang=en-GB; hreflang=de')
self.resource = LinkHeaderResource()
self.resource.add_link('/related/thing', 'alternate',
title='A related thing',
hreflang=('en-GB', 'de'),
type_hint='application/json',
title_star=('en', u'A \u00e7runchy thing'))
self._check_link_header(expected_value)
# ----------------------------------------------------------------------
# Helpers
# ----------------------------------------------------------------------
def _check_link_header(self, expected_value):
self.api.add_route(self.test_route, self.resource)
self.simulate_request(self.test_route)
self.assertEqual(expected_value, self.srmock.headers_dict['link'])