使用login_required对flask瓶测试进行单元测试

时间:2018-05-14 11:13:10

标签: python unit-testing flask nose

我的烧瓶应用程序设置有以下测试用例

class AboutViewTest(BaseTestCase):
    def setUp(self):
        self.url = url_for('pages_app.about',_external=True)
        self.url_login_redirect = url_for('accounts_app.login')

    def test_render(self):
        # without login
        resp = self.client.get(self.url)
        self.assertRedirects(resp, self.url_login_redirect)

        # after login
        with self.app.test_client() as c:
            login_successful = self.login(username='user1', password="123456", client=c)
            self.assertTrue(login_successful)
            resp = c.get(self.url)
            self.assertStatus(resp, 200) # this fails

使用@login_required

测试视图
@pages_app.route('/about/')
@login_required
def about():
    return render_template('pages/about.html')


def login_required(f):
    @wraps(f)
    def decorated_function(*args, **kwargs):
        if g.user is None:
            return redirect(url_for('accounts_app.login', next=request.url))
        return f(*args, **kwargs)
    return decorated_function

第二个测试用例失败(需要登录),因为它被重定向到登录302 != 200

如何测试此视图以传递两种情况(使用和不使用登录)?这是最好的做法吗?

1 个答案:

答案 0 :(得分:0)

最佳方法可能是按Flask docs中的建议填充g.user变量:

def get_user():
    user = getattr(g, 'user', None)
    if user is None:
        user = fetch_current_user_from_database()
        g.user = user
    return user

显然,您必须定义fetch_current_user_from_database(),您将在其中实例化一个用户(我假设您有一个User类)并将其返回。然后只需调用get_user()即可立即让用户在需要时登录。

作为替代方案,您可以在设置中添加一个标记以禁用登录,并在测试时在login_required方法中使用它。

例如:

from flask import current_app

def login_required(f):
    @wraps(f)
    def decorated_function(*args, **kwargs):
        if not current_app.config.get('LOGIN_DISABLED', False) and g.user is None:
            return redirect(url_for('accounts_app.login', next=request.url))
        return f(*args, **kwargs)
    return decorated_function

然后在你的测试中:

def test_render(self):
    # without login
    resp = self.client.get(self.url)
    self.assertRedirects(resp, self.url_login_redirect)

    self.app.config['LOGIN_DISABLED'] = True

    resp = c.get(self.url)
    self.assertStatus(resp, 200)