我正在尝试将测试添加到我的基本应用中。访问所有内容需要登录。
这是我的测试用例类:
class MyAppTestCase(FlaskTestCaseMixin):
def _create_app(self):
raise NotImplementedError
def _create_fixtures(self):
self.user = EmployeeFactory()
def setUp(self):
super(MyAppTestCase, self).setUp()
self.app = self._create_app()
self.client = self.app.test_client()
self.app_context = self.app.app_context()
self.app_context.push()
db.create_all()
self._create_fixtures()
self._create_csrf_token()
def tearDown(self):
super(MyAppTestCase, self).tearDown()
db.drop_all()
self.app_context.pop()
def _post(self, route, data=None, content_type=None, follow_redirects=True, headers=None):
content_type = content_type or 'application/x-www-form-urlencoded'
return self.client.post(route, data=data, follow_redirects=follow_redirects, content_type=content_type, headers=headers)
def _login(self, email=None, password=None):
email = email or self.user.email
password = password or 'password'
data = {
'email': email,
'password': password,
'remember': 'y'
}
return self._post('/login', data=data)
class MyFrontendTestCase(MyAppTestCase):
def _create_app(self):
return create_app(settings)
def setUp(self):
super(MyFrontendTestCase, self).setUp()
self._login()
我正在运行我的测试,在终端中使用nosetests,如:source my_env/bin/activate && nosetests --exe
这些基本测试失败了:
class CoolTestCase(MyFrontendTestCase):
def test_logged_in(self):
r = self._login()
self.assertIn('MyAppName', r.data)
def test_authenticated_access(self):
r = self.get('/myroute/')
self.assertIn('MyAppName', r.data)
从输出中,我看到r.data
只是登录页面的HTML,没有错误(例如,用户名或密码错误)或警报(“请登录访问此页面”)。
我正在setUp
进程中登录,因此test_authenticated_access
应该让我访问/myroute/
或者将我重定向到闪存的登录页面消息“请登录以访问此页面”。但它没有。
我无法弄清楚出了什么问题。我的测试基于我在Flask文档和this app boilerplate
中找到的测试答案 0 :(得分:2)
经过一周的自杀后,我终于明白了......
有几个问题:
Flask-Security和Flask-SSLify之间存在一些冲突。虽然自动https重定向在实际的Web服务器上运行良好,但在测试中它会阻止完全登录。我通过制作假测试路线并尝试POST
随机数据来解决这个问题。测试客户端中没有POST
工作。要解决此问题,我必须将测试配置DEBUG
更改为True
。请注意,这不是一个很好的解决方法,因为像CSS和JS这样的东西不会被编译而且其他应用程序行为可能会有所不同......
Flask-Security无法与factory_boy一起使用(或者我没有正确使用factory_boy?)。我试图通过factory_boy创建用户和角色,因为我看到了一些示例应用教程用过它。在我解决了上述问题之后,它一直告诉我用户不存在。我最终窃取了Flask-Security自己的测试代码,用于创建具有不同角色的假用户。
有问题的代码:
session = db.create_scoped_session()
class RoleFactory(SQLAlchemyModelFactory):
FACTORY_FOR = Role
FACTORY_SESSION = session
id = Sequence(int)
name = 'admin'
description = 'Administrator'
class EmployeeFactory(SQLAlchemyModelFactory):
FACTORY_FOR = Employee
FACTORY_SESSION = session
id = Sequence(int)
email = Sequence(lambda n: 'user{0}@app.com'.format(n))
password = LazyAttribute(lambda a: encrypt_password('password'))
username = Sequence(lambda n: 'user{0}'.format(n))
#last_login_at = datetime.utcnow()
#current_login_at = datetime.utcnow()
last_login_ip = '127.0.0.1'
current_login_ip = '127.0.0.1'
login_count = 1
roles = LazyAttribute(lambda _: [RoleFactory()])
active = True
这是我的测试用例:
class MyAppTestCase(FlaskTestCaseMixin, MyTestCase):
def _create_app(self):
raise NotImplementedError
def _create_fixtures(self):
#self.user = EmployeeFactory()
populate_data(1)
def setUp(self):
super(MyAppTestCase, self).setUp()
self.app = self._create_app()
self.client = self.app.test_client()
self.app_context = self.app.app_context()
self.app_context.push()
db.create_all()
self._create_fixtures()
self._create_csrf_token()
def tearDown(self):
super(MyAppTestCase, self).tearDown()
db.drop_all()
self.app_context.pop()
def _post(self, route, data=None, content_type=None, follow_redirects=True, headers=None):
content_type = content_type or 'application/x-www-form-urlencoded'
return self.client.post(route, data=data, follow_redirects=follow_redirects, content_type=content_type, headers=headers)
def _login(self, email=None, password=None):
email = email or 'matt@lp.com' #self.user.email
password = password or 'password'
data = {
'email': email,
'password': password,
'remember': 'y'
}
return self._post('/login', data=data)
答案 1 :(得分:1)
我在使用Flask-Security(使用Flask-Login)应用程序进行测试时遇到登录/注销问题。我通过模拟flask_login._get_user
函数并将app.login_manager._login_disabled
设置为True来解决了这个问题。
嘲笑用户的上下文管理器:
class LoginLogoutMixin(object):
@contextmanager
def login(self, user):
mock_get_user = patch('flask_login._get_user', Mock(return_value=user))
self.app.login_manager._login_disabled = True
mock_get_user.start()
yield
mock_get_user.stop()
self.app.login_manager._login_disabled = False
以及它的用法示例:
class ProtectedViewTestCase(LoginLogoutMixin, TestCase):
def setUp(self):
super(ProtectedViewTestCase, self).setUp()
# create user here
def test_protected_view(self):
with self.login(self.user):
rv = self.client.get(url_for('protected_view'))
self.assert200(rv)