我尝试使用pytest和app工厂对应用程序进行单元测试,但我似乎无法在测试中访问客户端会话对象。我确定在某些情况下我没有推动某个地方。我在我的应用程序中推送应用程序上下文夹具。我应该在某处推送请求上下文吗?
以下是MWE。
mwe.py:
from flask import Flask, session
def create_app():
app = Flask(__name__)
app.secret_key = 'top secret'
@app.route('/set')
def session_set():
session['key'] = 'value'
return 'Set'
@app.route('/check')
def session_check():
return str('key' in session)
@app.route('/clear')
def session_clear():
session.pop('key', None)
return 'Cleared'
return app
if __name__ == "__main__":
mwe = create_app()
mwe.run()
conftest.py:
import pytest
from mwe import create_app
@pytest.fixture(scope='session')
def app(request):
app = create_app()
ctx = app.app_context()
ctx.push()
def teardown():
ctx.pop()
request.addfinalizer(teardown)
return app
@pytest.fixture(scope='function')
def client(app):
return app.test_client()
test_session.py:
import pytest
from flask import session
def test_value_set_for_client_request(client): # PASS
client.get('/set')
r = client.get('/check')
assert 'True' in r.data
def test_value_set_in_session(client): # FAIL
client.get('/set')
assert 'key' in session
def test_value_set_in_session_transaction(client): # FAIL
with client.session_transaction() as sess:
client.get('/set')
assert 'key' in sess
请注意,直接运行它可以正常工作,我可以跳转/设置,/检查,/清除它的行为符合预期。类似地,仅使用测试客户端来获取页面的测试按预期工作。但是,直接访问会话似乎并非如此。
答案 0 :(得分:1)
在cookiecutter-flask中查看conftest.py中的以下内容。它可能会给你一些想法。
@pytest.yield_fixture(scope='function')
def app():
"""An application for the tests."""
_app = create_app(TestConfig)
ctx = _app.test_request_context()
ctx.push()
yield _app
ctx.pop()
@pytest.fixture(scope='function')
def testapp(app):
"""A Webtest app."""
return TestApp(app)
答案 1 :(得分:1)
问题在于您使用测试客户端的方式。
首先,您不必创建客户端夹具。如果您使用pytest-flask
,则会提供client
要素。如果您仍想使用自己的客户端(可能因为您不想要pytest-flask
),那么您的客户端夹具应该充当上下文处理器来包装您的请求。
所以你需要以下内容:
def test_value_set_in_session(client):
with client:
client.get('/set')
assert 'key' in session
当天的信息:pytest-flask有一个类似的客户端固定装置。区别在于pytest-flask使用上下文管理器为您提供客户端,每次测试可节省1行
@pytest.yield_fixture
def client(app):
"""A Flask test client. An instance of :class:`flask.testing.TestClient`
by default.
"""
with app.test_client() as client:
yield client
使用pytest-flask
client
def test_value_set_in_session(client):
client.get('/set')
assert 'key' in session