Python 烧瓶:类型错误:';元组';对象不可调用

Python 烧瓶:类型错误:';元组';对象不可调用,python,flask,Python,Flask,我正在尝试使用在线教程学习flask,当我尝试使用以下formpayload数据注册用户时 csrf_token=ImJkOTlkYmVlMWQ2N2MwZGVmMGQ5MTRkZmZhODBmY2E0NWFiYTY4Y2Ei.DZIZ_w.UTdQ_hkzHXoIFS8oH8D84AMWLF8&username=vector&email=vcks%40xyz.com&password=1234&password2=1234&submit=Register

我正在尝试使用在线教程学习flask,当我尝试使用以下formpayload数据注册用户时

csrf_token=ImJkOTlkYmVlMWQ2N2MwZGVmMGQ5MTRkZmZhODBmY2E0NWFiYTY4Y2Ei.DZIZ_w.UTdQ_hkzHXoIFS8oH8D84AMWLF8&username=vector&email=vcks%40xyz.com&password=1234&password2=1234&submit=Register
当我点击submit时,我得到了以下回溯

  File "/Users/vaibhavchauhan/microblog/app/routes.py", line 51, in register
    if form.validate_on_submit():
  File "/Users/vaibhavchauhan/.pyenv/versions/3.6.2/envs/flasky/lib/python3.6/site-packages/flask_wtf/form.py", line 101, in validate_on_submit
    return self.is_submitted() and self.validate()
  File "/Users/vaibhavchauhan/.pyenv/versions/3.6.2/envs/flasky/lib/python3.6/site-packages/wtforms/form.py", line 310, in validate
    return super(Form, self).validate(extra)
  File "/Users/vaibhavchauhan/.pyenv/versions/3.6.2/envs/flasky/lib/python3.6/site-packages/wtforms/form.py", line 152, in validate
    if not field.validate(self, extra):
  File "/Users/vaibhavchauhan/.pyenv/versions/3.6.2/envs/flasky/lib/python3.6/site-packages/wtforms/fields/core.py", line 204, in validate
    stop_validation = self._run_validation_chain(form, chain)
  File "/Users/vaibhavchauhan/.pyenv/versions/3.6.2/envs/flasky/lib/python3.6/site-packages/wtforms/fields/core.py", line 224, in _run_validation_chain
    validator(form, self)
TypeError: 'tuple' object is not callable

@app.route('/register', methods=['GET','POST'])
def register():
    if current_user.is_authenticated:
        return redirect(url_for('index'))
    form = RegistrationForm()
    if form.validate_on_submit():
        user = User(username=form.username.data, email=form.email.data)
        user.set_password(form.password.data)
        db.session.add(user)
        db.session.commit()
        flash('Congratulations, you are now a registered user!')
        return redirect(url_for('login'))
    return render_template('register.html', title='Register', form=form)
根据要求,我正在为登记表添加代码

class RegistrationForm(FlaskForm):
    username = StringField('Username', validators=[DataRequired()])
    email = StringField('Email', validators=[DataRequired(), Email()])
    password = PasswordField('Password', validators=[DataRequired()])
    password2 = PasswordField(
        'Repeat Password', validators=[(DataRequired(), EqualTo('password'))])
    submit = SubmitField('Register')

    def validate_username(self, username):
        user = User.query.filter_by(username=username.data).first()
        if user is not None:
            raise ValidationError('please use diffrent username.')

    def validate_email(self, email):
        user = User.query.filter_by(email=email.data).first()
        if user is not None:
            raise ValidationError('please use diffrent email.')
问题在于:

password2 = PasswordField(
    'Repeat Password', validators=[(DataRequired(), EqualTo('password'))])
验证器
应该是验证器列表。但是,您将传递一个包含两个验证器的元组的列表。要解决此问题,请执行以下操作:

password2 = PasswordField(
    'Repeat Password', validators=[DataRequired(), EqualTo('password')])
validate()
内部,它执行类似(简化伪代码)的操作:


因此,
password2
字段的第一个
validator
结果是一个由两个函数组成的元组,而不是一个函数,因此您不能调用元组。

为RegistrationFormindenting添加的代码被修复,因为我没有提供正确的间距。如果您不介意告诉我您是如何计算出validator需要的列表的,我传递一个元组是虫子?非常感谢。@VaibhavChauhan其他的(只有一个验证器)正在传递一个验证器的列表,而不是一个验证器的元组列表。这种不一致似乎很奇怪。另外,从异常描述来看,很明显,代码试图将元组用作函数,而前者如何导致后者是有道理的。所以我快速搜索了一些flask_wtf示例代码……但没有找到任何具有多个验证器的代码。因此,我查看了源代码(更容易找到),这证实了我的怀疑。@VaibhavChauhan通常会更简单,但这种特殊的不一致性在Python中有时是合法的(
'%d:%d%%(1,2)
需要一个元组,但是
'%d:done'%1
不需要元组就可以工作…),这就是为什么我必须仔细检查。在处理元组及其行为时,我还查看了代码文档wtforms/fields/core.py”,第224行,这是最接近理解为什么在解释“”运行验证链后此函数需要iterable的原因,如果任何验证器引发StopValidation,则停止:param form:此字段所指向的表单实例:param validators:验证程序可调用项的序列或可调用项:返回:如果验证已停止,则为True,否则为False。“@VaibhavChauhan是的,不一致性就像它需要一个元组列表(或一个iterable的iterable),为什么只通过一个简单的列表就可以工作呢?至于文档,Flask文档似乎将requests/bs4风格发挥到了极致:有很多例子可以让你用它编写第一个程序变得非常简单,但对实际引用的深度却很小。这就是为什么我直接找到了大部分可读性和组织性都很好的源代码ust在验证程序链上循环并调用对象,没有嵌套循环。
for field in fields:
    for validator in field.validators:
        validator(field)