Testing Flask Applications(测试Flask应用)

以官方文档中的第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节的那段代码的背后原理.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容