美文网首页
Testing Flask Applications(测试Fla

Testing Flask Applications(测试Fla

作者: 黄智勇atTAFinder | 来源:发表于2016-09-02 12:00 被阅读0次

    以官方文档中的第7章为起点.

    # 7.2 The Testing Skeleton.
    import os
    import flaskr
    import unittest
    import tempfile
    
    class FlaskTestCase(unittest.TestCase):
    
        def setUp(self):
            self.db_fd, flaskr.app.config['DATABASE'] = tempfile.mkstemp()
            flaskr.app.config['TESTING'] = True
            self.app = flaskr.app.test_client()
            with flaskr.app.app_context():
                flaskr.init_db()
    
        def tearDown(self):
            os.close(self.db_fd)
            os.unlink(flaskr.app.config['DATABASE'])
    
    if __name__ == '__main__':
        unittest.main()
    

    下面我们看一下self.app = flaskr.app.test_client()这行代码到底是干什么的.

        # app.py
        def test_client(self, use_cookies=True, **kwargs):
            cls = self.test_client_class
            if cls is None:
                from flask.testing import FlaskClient as cls
            return cls(self, self.response_class, use_cookies=use_cookies, **kwargs)
    

    默认的test_client_class为None.所以看FlaskClient. FlaskClient继承自Werkzeug.test中的Client,但是额外附带了一些方法.
    with flaskr.app.app_context(): flaskr.init_db() 中的app_context()方法已经剖析过了:一个AppContext中含有app, url_adapter, g. 既然创建的AppContext含有g变量,那么flaskr.init_db()就可以顺利的通过执行了。


    在7.8节的Keeping the Context Around有小段代码:

    app = flask.Flask(__name__)
    
    with app.test_client() as c:
        rv = c.get('/?tequila=42')
        assert request.args['tequila'] == '42'
    

    我们来看看flask是如何保持request可以正常工作而不是发出'Working outside of request context.'的错误.
    在app.test_request_context():...中,很显然的RequestContext被push了。但是app.test_client()却绕了一下弯,不是那么明显.

        # testing.py
        def __enter__(self):
            if self.preserve_context:
                raise RuntimeError('Cannot nest client invocations')
            self.preserve_context = True
            return self
    

    enter函数中, 将self.preserve_context设置为True. 并返回FlaskClient实例.FlaskClient继承自Werkzeug中的Client.

        # test.py in Werkzeug.
        def get(self, *args, **kw):
            """Like open but method is enforced to GET."""
            kw['method'] = 'GET'
            return self.open(*args, **kw)
    

    当我们使用c.get('/?tequila=42')调用get方法时,实际上是return self.open(*args, **kw).

        def open(self, *args, **kwargs):
            kwargs.setdefault('environ_overrides', {}) \
                ['flask._preserve_context'] = self.preserve_context
    
            as_tuple = kwargs.pop('as_tuple', False)
            buffered = kwargs.pop('buffered', False)
            follow_redirects = kwargs.pop('follow_redirects', False)
            builder = make_test_environ_builder(self.application, *args, **kwargs)
    
            return Client.open(self, builder,
                               as_tuple=as_tuple,
                               buffered=buffered,
                               follow_redirects=follow_redirects)
    

    FlaskClient重写了Client中的open方法.将kwargs中的flask._preserve_context参数设置为self.preserve_context,也就是True. 紧接着kwargs被传入make_test_environ_builder函数.

    def make_test_environ_builder(app, path='/', base_url=None, *args, **kwargs):
        http_host = app.config.get('SERVER_NAME')
        app_root = app.config.get('APPLICATION_ROOT')
        if base_url is None:
            url = url_parse(path)
            base_url = 'http://%s/' % (url.netloc or http_host or 'localhost')
            if app_root:
                base_url += app_root.lstrip('/')
            if url.netloc:
                path = url.path
                if url.query:
                    path += '?' + url.query
        return EnvironBuilder(path, base_url, *args, **kwargs)
    

    下面我们看看传入make_test_environ_builder的参数都有哪些:

    • app = Flask实例.
    • path = '/?tequila=42'
    • base_url = None
    • args = ()
    • kwargs = {'method': 'GET', 'environ_overrides': {'flask._preserve_context': True}}

    make_test_environ_builder返回EnvironBuilder. 传入EnvironBuilder的参数除了base_url变为'http://localhost', 其它都没有变.返回一个EnvironBuilder.示例.
    接下来调用Client.open(...),在Clien.open()函数中,我们主要看builder中的kwargs中的'flask._preserve_context'是怎么传入到wsgi_app中的。

        # test.py in Werkzeug.
        def open(self, *args, **kwargs):
            as_tuple = kwargs.pop('as_tuple', False)
            buffered = kwargs.pop('buffered', False)
            follow_redirects = kwargs.pop('follow_redirects', False)
            environ = None
            if not kwargs and len(args) == 1:
                if isinstance(args[0], EnvironBuilder):
                    # 在此处获得builder中的environ.
                    environ = args[0].get_environ()  #<---
                elif isinstance(args[0], dict):
                    environ = args[0]
            if environ is None:
                builder = EnvironBuilder(*args, **kwargs)
                try:
                    environ = builder.get_environ()
                finally:
                    builder.close()
            # 在此处开始run_wsgi_app 并获得response.
            response = self.run_wsgi_app(environ, buffered=buffered) # <---
    
            # handle redirects
            redirect_chain = []
            while 1:
                status_code = int(response[1].split(None, 1)[0])
                if status_code not in (301, 302, 303, 305, 307) \
                   or not follow_redirects:
                    break
                new_location = response[2]['location']
    
                method = 'GET'
                if status_code == 307:
                    method = environ['REQUEST_METHOD']
    
                new_redirect_entry = (new_location, status_code)
                if new_redirect_entry in redirect_chain:
                    raise ClientRedirectError('loop detected')
                redirect_chain.append(new_redirect_entry)
                environ, response = self.resolve_redirect(response, new_location,
                                                          environ,
                                                          buffered=buffered)
    
            if self.response_wrapper is not None:
                # 对response进行包装.
                response = self.response_wrapper(*response) # <---
            if as_tuple:
                return environ, response
            return response # <---
    

    在self.run_wsgi_app中, ’flask._preserve_context‘就包含在environ中.

    def run_wsgi_app(app, environ, buffered=False):
       
        environ = _get_environ(environ)
        response = []
        buffer = []
    
        ...
        app_rv = app(environ, start_response) # <---
        ...
    

    现在回过头来,我们又进入了熟悉的Flask.wsgi_app(...)的处理流程中.

        def wsgi_app(self, environ, start_response):
      
            ctx = self.request_context(environ)
            ctx.push()
            ...
            finally:
                if self.should_ignore_error(error):
                    error = None
                ctx.auto_pop(error)  # <---
    

    wsgi_app最后会自动pop ctx.下面我们仔细分析一下相关代码:

        def auto_pop(self, exc):
            if self.request.environ.get('flask._preserve_context') or \
               (exc is not None and self.app.preserve_context_on_exception):
                self.preserved = True
                self._preserved_exc = exc
            else:
                self.pop(exc)
    

    于是就真相大白了,原理auto_pop首行代码就是判断request.environ中是否有'flask._preserve_context'变量,另外两个判断条件可能与调试相关,暂时忽略.于是self.preserved = True. 当然, 也就不会执行self.pop(exc)了.不执行的结果就是,RequestContext仍然保存在_request_ctx_stack上面, 当然也就可以正常获取request.args['tequila']的值,而不会出现working outside of context 的报错了. 那什么时候pop ctx呢? 答案在此.

        # testing.py
        def __exit__(self, exc_type, exc_value, tb):
            # 设置preserve-context变量为False.
            self.preserve_context = False
    
            # 获得ctx, 测试两个条件, 立即执行top.pop().
            top = _request_ctx_stack.top
            if top is not None and top.preserved:
                top.pop()
    

    到此,我们也就搞清楚了官方文档7.8节的那段代码的背后原理.

    相关文章

      网友评论

          本文标题:Testing Flask Applications(测试Fla

          本文链接:https://www.haomeiwen.com/subject/ogpqettx.html