如何对“新风格”Django 中间件进行单元测试

ali*_*s51 3 python django django-middleware django-testing django-tests

我正在学习如何对 Django 中间件进行单元测试。

process_request()在“旧式”中间件中,加载用于测试结果的中间件非常容易。例如

def test_session(self):
    request = self.factory.get('/')
    session_middleware = SessionMiddleware()
    session_middleware.process_request(request)
    // Do stuff
Run Code Online (Sandbox Code Playgroud)

然而,在“新风格”中,这并不存在。例如,我将如何测试Django 文档提供的以下示例?

class TimezoneMiddleware:
    def __init__(self, get_response):
        self.get_response = get_response

    def __call__(self, request):
        tzname = request.session.get('django_timezone')
        if tzname:
            timezone.activate(pytz.timezone(tzname))
        else:
            timezone.deactivate()
        return self.get_response(request)
Run Code Online (Sandbox Code Playgroud)

使用TimezoneMiddleware(request)会调用__init__但不会调用__call__?

sam*_*amu 10

正如您所看到的,中间件只是实现了一个__call__神奇的方法。这意味着该类的实例是可调用的

旧式中间件和新式中间件之间的区别在于,新中间件只是一个返回可调用对象的可调用对象 - 首先使用回调调用它get_response,然后使用实际请求调用返回的可调用对象。get_response是由 Django 本身提供/注入的可调用函数,它是一个用于返回视图响应或链中下一个中间件的函数。

因此,为了测试您的SessionMiddleware,您可以执行以下操作:

import mock

def test_middleware(self):
    get_response = mock.MagicMock()
    request = self.factory.get('/')

    middleware = SessionMiddleware(get_response)
    response = middleware(request)

    # ensure get_response has been returned 
    # (or not, if your middleware does something else)
    self.assertEqual(get_response.return_value, response)
    
Run Code Online (Sandbox Code Playgroud)