Django测试客户端不处理异常?

时间:2015-02-04 16:36:13

标签: python django django-testing

我需要使用django test dummy client为Django中的自定义handler404和handler500编写一些测试。 第一个是容易测试,而我遇到第二个问题。

基本上,问题是Django测试客户端没有捕获异常并且没有路由到正确的处理程序。这是一个问题,因为我们需要测试是否使用了正确的自定义处理程序和模板。

我们有一个简单的中间件类来模拟测试的异常:

  class HTTPStatusCodeMiddleware(object):
    def process_view(self, request, *args):
        if 'cookie_500_error' in request.COOKIES:
            raise Exception('Test exception')

以上代码适用于浏览器中的手动测试。

现在,测试是:

  def test_404_template(self):                                                 
      c = Client()                                                             
      url = '/not_existing.html'                                               
      response = c.get(url)                                                    
      self.assertEqual(response.status_code, 404)  # success                  
      self.assertTemplateUsed(response, 'custom/404.html')  # success   

  def test_500_template(self):                                                 
      c = Client()                                                             
      c.cookies['cookie_500_error'] = '1'                                  
      response = c.get('/')  # here middleware raises exception!!!
      self.assertEqual(response.status_code, 500)
      self.assertTemplateUsed(response, 'custom/500.html')

有什么想法吗? 我没有选择使用硒。 谢谢!

5 个答案:

答案 0 :(得分:3)

Django测试客户端只处理一些异常(参见the docs)。所有其他的都在测试中可见,你可以测试它们;)

答案 1 :(得分:0)

所以django测试客户端类没有按设计捕获异常,这是一件好事。

由于问题特别是测试自定义handler500,它是由自定义中间件在特定设置上设置的,只需将request.urlconf变量替换为自定义urlconf模块,解决方案就是使用RequestFactory来构建请求和测试请求。 urlconf.handler500查看:

在custom / client1 / urls.py

def handler500(request):
    data = {'client', 'client1 is sorry!'}
    ctx = RequestContext(request)
    content = render_to_string('client1/500.html', data, ctx)
    return http.HttpResponseServerError(content)

in tests / views / test_error_pages.py

def test_500_template(self):
    req = RequestFactory().get('/')
    req.user = AnonymousUser()
    req.session = {}

    cust_mw = CustomUrlconfMiddleware()
    cust_mw.process_request(req)  # set the custom request.urlconf
    urlconf = importlib.import_module(req.urlconf)
    response = urlconf.handler500(req)
    self.assertEqual(response.status_code, 500)
    assert_str = '/static/img/custom/client1/'
    self.assertTrue(assert_str in response.content,
                    msg='500 template is not for client1. String {} is not in content'.format(assert_str))

答案 2 :(得分:0)

好吧,我在单元测试中使用PermissionDenied异常遇到了类似的问题。由于Django没有捕获它们,这导致我的测试崩溃而不是失败。我确实找到了一个尚未见过的简单解决方案,就是这样:

示例视图:

from django.core.exceptions import PermissionDenied

class SampleView(object):
    def setup(request):
        if not request_is_authenticated(request):
            raise PermissionDenied

单元测试:

from django.core.exceptions import PermissionDenied

def test_permission(self):
    try:
        SampleView.setup(request)
    except PermissionDenied:
        print('Hallelujah no crashing!')

答案 3 :(得分:0)

Django 3.0现在支持此功能

新的测试curl --doh-url自变量Client可以控制是否也应在测试中引发请求期间引发的异常。为了向后兼容,该值默认为raise_request_exception。如果它是True并且发生了异常,则测试客户端将返回带有属性False的500响应,该元组提供了发生的异常的信息。

exc_info

有关documentation上的信息

答案 4 :(得分:0)

对于Django 2.x,可以通过修补其store_exc_info方法来阻止测试客户端引发异常:

c = Client()                                                             
c.store_exc_info = lambda *args, **kw: None
response = c.get("/error")
assert response.status_code == 500

假设处理/error网址的视图引发异常,这将使我们能够获得500 Server Error响应。

对于Django> = 3.0,应使用raise_request_exception参数(如@CesarCanassa答案中所述)