direct tests for `@render_exception`
This commit is contained in:
parent
b68c08cff9
commit
a9e9efb336
|
@ -0,0 +1,211 @@
|
|||
"""
|
||||
Tests for ``allmydata.web.common``.
|
||||
"""
|
||||
|
||||
from bs4 import (
|
||||
BeautifulSoup,
|
||||
)
|
||||
from hyperlink import (
|
||||
DecodedURL,
|
||||
)
|
||||
|
||||
from testtools.matchers import (
|
||||
Equals,
|
||||
Contains,
|
||||
MatchesPredicate,
|
||||
)
|
||||
from testtools.twistedsupport import (
|
||||
succeeded,
|
||||
has_no_result,
|
||||
)
|
||||
|
||||
from twisted.internet.defer import (
|
||||
fail,
|
||||
)
|
||||
from twisted.web.server import (
|
||||
NOT_DONE_YET,
|
||||
)
|
||||
from twisted.web.resource import (
|
||||
Resource,
|
||||
)
|
||||
|
||||
from ...web.common import (
|
||||
render_exception,
|
||||
)
|
||||
|
||||
from ..common import (
|
||||
SyncTestCase,
|
||||
)
|
||||
from ..common_web import (
|
||||
render,
|
||||
)
|
||||
from .common import (
|
||||
assert_soup_has_tag_with_attributes,
|
||||
)
|
||||
|
||||
class StaticResource(Resource):
|
||||
def __init__(self, response):
|
||||
Resource.__init__(self)
|
||||
self._response = response
|
||||
|
||||
@render_exception
|
||||
def render(self, request):
|
||||
return self._response
|
||||
|
||||
|
||||
class RenderExceptionTests(SyncTestCase):
|
||||
"""
|
||||
Tests for ``render_exception``.
|
||||
"""
|
||||
def test_exception(self):
|
||||
"""
|
||||
If the decorated method raises an exception then the exception is rendered
|
||||
into the response.
|
||||
"""
|
||||
class R(Resource):
|
||||
@render_exception
|
||||
def render(self, request):
|
||||
raise Exception("synthetic exception")
|
||||
|
||||
self.assertThat(
|
||||
render(R(), {}),
|
||||
succeeded(
|
||||
Contains(b"synthetic exception"),
|
||||
),
|
||||
)
|
||||
|
||||
def test_failure(self):
|
||||
"""
|
||||
If the decorated method returns a ``Deferred`` that fires with a
|
||||
``Failure`` then the exception the ``Failure`` wraps is rendered into
|
||||
the response.
|
||||
"""
|
||||
resource = StaticResource(fail(Exception("synthetic exception")))
|
||||
self.assertThat(
|
||||
render(resource, {}),
|
||||
succeeded(
|
||||
Contains(b"synthetic exception"),
|
||||
),
|
||||
)
|
||||
|
||||
def test_resource(self):
|
||||
"""
|
||||
If the decorated method returns an ``IResource`` provider then that
|
||||
resource is used to render the response.
|
||||
"""
|
||||
resource = StaticResource(StaticResource(b"static result"))
|
||||
self.assertThat(
|
||||
render(resource, {}),
|
||||
succeeded(
|
||||
Equals(b"static result"),
|
||||
),
|
||||
)
|
||||
|
||||
def test_unicode(self):
|
||||
"""
|
||||
If the decorated method returns a ``unicode`` string then that string is
|
||||
UTF-8 encoded and rendered into the response.
|
||||
"""
|
||||
text = u"\N{SNOWMAN}"
|
||||
resource = StaticResource(text)
|
||||
self.assertThat(
|
||||
render(resource, {}),
|
||||
succeeded(
|
||||
Equals(text.encode("utf-8")),
|
||||
),
|
||||
)
|
||||
|
||||
def test_bytes(self):
|
||||
"""
|
||||
If the decorated method returns a ``bytes`` string then that string is
|
||||
rendered into the response.
|
||||
"""
|
||||
data = b"hello world"
|
||||
resource = StaticResource(data)
|
||||
self.assertThat(
|
||||
render(resource, {}),
|
||||
succeeded(
|
||||
Equals(data),
|
||||
),
|
||||
)
|
||||
|
||||
def test_decodedurl(self):
|
||||
"""
|
||||
If the decorated method returns a ``DecodedURL`` then a redirect to that
|
||||
location is rendered into the response.
|
||||
"""
|
||||
loc = u"http://example.invalid/foo?bar=baz"
|
||||
resource = StaticResource(DecodedURL.from_text(loc))
|
||||
self.assertThat(
|
||||
render(resource, {}),
|
||||
succeeded(
|
||||
MatchesPredicate(
|
||||
lambda value: assert_soup_has_tag_with_attributes(
|
||||
self,
|
||||
BeautifulSoup(value),
|
||||
"meta",
|
||||
{"http-equiv": "refresh",
|
||||
"content": "0;URL={}".format(loc.encode("ascii")),
|
||||
},
|
||||
)
|
||||
# The assertion will raise if it has a problem, otherwise
|
||||
# return None. Turn the None into something
|
||||
# MatchesPredicate recognizes as success.
|
||||
or True,
|
||||
"did not find meta refresh tag in %r",
|
||||
),
|
||||
),
|
||||
)
|
||||
|
||||
def test_none(self):
|
||||
"""
|
||||
If the decorated method returns ``None`` then the response is finished
|
||||
with no additional content.
|
||||
"""
|
||||
self.assertThat(
|
||||
render(StaticResource(None), {}),
|
||||
succeeded(
|
||||
Equals(b""),
|
||||
),
|
||||
)
|
||||
|
||||
def test_not_done_yet(self):
|
||||
"""
|
||||
If the decorated method returns ``NOT_DONE_YET`` then the resource is
|
||||
responsible for finishing the request itself.
|
||||
"""
|
||||
the_request = []
|
||||
class R(Resource):
|
||||
@render_exception
|
||||
def render(self, request):
|
||||
the_request.append(request)
|
||||
return NOT_DONE_YET
|
||||
|
||||
d = render(R(), {})
|
||||
|
||||
self.assertThat(
|
||||
d,
|
||||
has_no_result(),
|
||||
)
|
||||
|
||||
the_request[0].write(b"some content")
|
||||
the_request[0].finish()
|
||||
|
||||
self.assertThat(
|
||||
d,
|
||||
succeeded(
|
||||
Equals(b"some content"),
|
||||
),
|
||||
)
|
||||
|
||||
def test_unknown(self):
|
||||
"""
|
||||
If the decorated method returns something which is not explicitly
|
||||
supported, an internal server error is rendered into the response.
|
||||
"""
|
||||
self.assertThat(
|
||||
render(StaticResource(object()), {}),
|
||||
succeeded(
|
||||
Equals(b"Internal Server Error"),
|
||||
),
|
||||
)
|
|
@ -605,7 +605,7 @@ def _finish(result, render, request):
|
|||
result,
|
||||
))
|
||||
request.setResponseCode(http.INTERNAL_SERVER_ERROR)
|
||||
_finish(b"", render, request)
|
||||
_finish(b"Internal Server Error", render, request)
|
||||
|
||||
|
||||
def _renderHTTP_exception(request, failure):
|
||||
|
|
Loading…
Reference in New Issue