2009-02-23 21:19:43 +00:00
|
|
|
|
|
|
|
import re
|
|
|
|
from twisted.internet import defer
|
2009-02-25 01:33:00 +00:00
|
|
|
from twisted.web import client
|
2009-02-23 21:19:43 +00:00
|
|
|
from nevow.testutil import FakeRequest
|
|
|
|
from nevow import inevow, context
|
|
|
|
|
|
|
|
class WebRenderingMixin:
|
|
|
|
# d=page.renderString() or s=page.renderSynchronously() will exercise
|
|
|
|
# docFactory, render_*/data_* . It won't exercise want_json(), or my
|
|
|
|
# renderHTTP() override which tests want_json(). To exercise args=, we
|
|
|
|
# must build a context. Pages which use a return_to= argument need a
|
|
|
|
# context.
|
|
|
|
|
|
|
|
# d=page.renderHTTP(ctx) will exercise my renderHTTP, want_json, and
|
|
|
|
# docFactory/render_*/data_*, but it requires building a context. Since
|
|
|
|
# we're already building a context, it is easy to exercise args= .
|
|
|
|
|
|
|
|
# so, use at least two d=page.renderHTTP(ctx) per page (one for json, one
|
|
|
|
# for html), then use lots of simple s=page.renderSynchronously() to
|
|
|
|
# exercise the fine details (the ones that don't require args=).
|
|
|
|
|
|
|
|
def make_context(self, req):
|
|
|
|
ctx = context.RequestContext(tag=req)
|
|
|
|
ctx.remember(req, inevow.IRequest)
|
|
|
|
ctx.remember(None, inevow.IData)
|
|
|
|
ctx = context.WovenContext(parent=ctx, precompile=False)
|
|
|
|
return ctx
|
|
|
|
|
|
|
|
def render1(self, page, **kwargs):
|
|
|
|
# use this to exercise an overridden renderHTTP, usually for
|
|
|
|
# output=json or render_GET. It always returns a Deferred.
|
|
|
|
req = FakeRequest(**kwargs)
|
2009-02-25 00:20:50 +00:00
|
|
|
req.fields = None
|
2009-02-23 21:19:43 +00:00
|
|
|
ctx = self.make_context(req)
|
|
|
|
d = defer.maybeDeferred(page.renderHTTP, ctx)
|
|
|
|
def _done(res):
|
|
|
|
if isinstance(res, str):
|
|
|
|
return res + req.v
|
|
|
|
return req.v
|
|
|
|
d.addCallback(_done)
|
|
|
|
return d
|
|
|
|
|
|
|
|
def render2(self, page, **kwargs):
|
|
|
|
# use this to exercise the normal Nevow docFactory rendering. It
|
|
|
|
# returns a string. If one of the render_* methods returns a
|
|
|
|
# Deferred, this will throw an exception. (note that
|
|
|
|
# page.renderString is the Deferred-returning equivalent)
|
|
|
|
req = FakeRequest(**kwargs)
|
2009-02-25 01:04:59 +00:00
|
|
|
req.fields = None
|
2009-02-23 21:19:43 +00:00
|
|
|
ctx = self.make_context(req)
|
|
|
|
return page.renderSynchronously(ctx)
|
|
|
|
|
|
|
|
def failUnlessIn(self, substring, s):
|
|
|
|
self.failUnless(substring in s, s)
|
|
|
|
|
|
|
|
def remove_tags(self, s):
|
|
|
|
s = re.sub(r'<[^>]*>', ' ', s)
|
|
|
|
s = re.sub(r'\s+', ' ', s)
|
|
|
|
return s
|
|
|
|
|
2009-02-25 01:33:00 +00:00
|
|
|
|
|
|
|
class MyGetter(client.HTTPPageGetter):
|
|
|
|
handleStatus_206 = lambda self: self.handleStatus_200()
|
|
|
|
|
|
|
|
class HTTPClientHEADFactory(client.HTTPClientFactory):
|
|
|
|
protocol = MyGetter
|
|
|
|
|
|
|
|
def noPage(self, reason):
|
|
|
|
# Twisted-2.5.0 and earlier had a bug, in which they would raise an
|
|
|
|
# exception when the response to a HEAD request had no body (when in
|
|
|
|
# fact they are defined to never have a body). This was fixed in
|
|
|
|
# Twisted-8.0 . To work around this, we catch the
|
|
|
|
# PartialDownloadError and make it disappear.
|
|
|
|
if (reason.check(client.PartialDownloadError)
|
|
|
|
and self.method.upper() == "HEAD"):
|
|
|
|
self.page("")
|
|
|
|
return
|
|
|
|
return client.HTTPClientFactory.noPage(self, reason)
|
|
|
|
|
|
|
|
class HTTPClientGETFactory(client.HTTPClientFactory):
|
|
|
|
protocol = MyGetter
|