wtforms 上的单元测试失败

Unit test fails on wtforms

当使用 pytest 对将数据发布到 flask wtforms 进行单元测试时,表单的字段数据总是 '',这似乎数据不正确 "posted" 到表单,它因为 form.validate_on_submit() 总是 return False。我在测试时禁用了 WTF_CSRF_ENABLED

我做了一个最小的项目来演示这个问题。我没有在这里包括数据库,因为数据库在我的真实项目中工作正常。

这个最小的项目结构在这里:

.
├── app.py
├── templates
│   └── login.html
└── tests
    └── test_login.py

test_login.py:

import unittest
from flask import current_app
from app import create_app


class BasicsTestCase(unittest.TestCase):
    def setUp(self):
        self.app = create_app()
        self.app.config['WTF_CSRF_ENABLED'] = False
        self.app_context = self.app.app_context()
        self.app_context.push()
        self.client = self.app.test_client()

    def tearDown(self):
        self.app_context.pop()

    def test_app_exists(self):
        self.assertFalse(current_app is None)

    def test_home_page(self):
        response = self.client.get('/')
        self.assertEqual(response.status_code, 200)

    def test_login(self):
        response = self.client.post(
            '/login', data={
                'username': 'a',
                'password': 'a'
            })

app.py:

from flask import Flask, redirect, render_template, url_for, Blueprint
from flask_wtf import FlaskForm
from wtforms import PasswordField, StringField, SubmitField
from wtforms.validators import DataRequired, Length

bp = Blueprint('myapp', __name__)


@bp.route('/')
def home():
    return 'hello'


@bp.route('/login', methods=['GET', 'POST'])
def login():
    form = LoginForm(prefix='form-login-')
    if form.validate_on_submit():
        return redirect(url_for('home'))
    return render_template('login.html', form=form)


class LoginForm(FlaskForm):
    username = StringField('Username', validators=[DataRequired()])
    password = PasswordField('Password', validators=[DataRequired()])
    submit = SubmitField('Log In')


def create_app():
    app = Flask('__name__')
    app.config['SECRET_KEY'] = 'secretkey'
    app.register_blueprint(bp)
    return app

事实证明,当您的 wtforms 具有 prefix 属性 时,您 post 数据的方式应该改变。

如您所见,我使用的表格是form = LoginForm(prefix='form-login-'),我post将数据添加到表格的方式是

response = self.client.post(
            '/login', data={
                'username': 'a',
                'password': 'a'
            })

在将表单中使用的 prefix 添加到数据字典之前,它不会正确传递数据,如下所示:

response = self.client.post(
            '/login', data={
                'form-login-username': 'a',
                'form-login-password': 'a'
            })

现在一切正常!

但是如果要获取表单域数据,还是应该使用form.username.data,不用任何prefix.