如何相互验证 wtforms 字段?

Posted

技术标签:

【中文标题】如何相互验证 wtforms 字段?【英文标题】:How do I validate wtforms fields against one another? 【发布时间】:2014-03-15 22:07:30 【问题描述】:

我在一个表单中有三个相同的SelectField 输入,每个输入都有相同的选项集。我不能使用一个多选。

我想确保用户为这三个字段选择三个不同的选项。

在自定义验证中,您似乎一次只能引用一个字段,而不能将该字段的值与其他字段的值进行比较。我怎样才能做到这一点?谢谢!

【问题讨论】:

【参考方案1】:

您可以在 Form 中覆盖 validate...

class MyForm(Form):
    select1 = SelectField('Select 1', ...)
    select2 = SelectField('Select 2', ...)
    select3 = SelectField('Select 3', ...)
    def validate(self):
        if not Form.validate(self):
            return False
        result = True
        seen = set()
        for field in [self.select1, self.select2, self.select3]:
            if field.data in seen:
                field.errors.append('Please select three distinct choices.')
                result = False
            else:
                seen.add(field.data)
        return result

【讨论】:

谢谢 - 为什么是 if not Form.validate(self): 行?当我尝试你的建议并且验证没有运行时,它会一直触发。 这是做默认验证(好像你没有覆盖验证) 按照您的要求,其他步骤是跨多个字段执行验证。 我只是想补充一点,我被这里的覆盖弄糊涂了。参考 O'Reilly Learning Python,我称之为扩展 validate 方法。当您调用 if not Form.validate(self) 时,您仍在调用默认行为,然后自定义验证代码运行。当我读到它时,它为我清除了一切。仍然@FogleBird 很好的答案 如果您使用的是 Python 3,最好写 super().validate() 而不是 Form.validate(self)。如果您使用的是 Python 2,请使用 super(MyForm, self).validate()【参考方案2】:

您可以在验证中使用表单来获取其他字段的值。

例如:

def validate_name(form, field):
    if form.other_variable.data == 'checked' and len(field.data) > 10:
        raise validation_error("say somgthing")

【讨论】:

【参考方案3】:

我编写了一个小型 python 库required 来简化这样的跨字段验证。您可以将验证逻辑以声明方式编码为成对依赖项。所以你的表单可能看起来像:

from required import R, Requires, RequirementError

class MyForm(Form):

    VALIDATION = (
        Requires("select1", R("select1") != R("select2") +
        Requires("select2", R("select2") != R("select3") +
        Requires("select3", R("select3") != R("select1")
    )

    select1 = SelectField('Select 1', ...)
    select2 = SelectField('Select 2', ...)
    select3 = SelectField('Select 3', ...)

    def validate(self):
        data = 
            "select1": self.select1.data,
            "select2": self.select2.data,
            "select3": self.select3.data,
        

        # you can catch the RequirementError
        # and append the error message to 
        # the form errors

        self.VALIDATION.validate(data)
        return result

您可以获取 VALIDATION 对象并附加更多验证规则,甚至可以将其放在单独的模块中并在不同位置导入/重用验证规则。

【讨论】:

【参考方案4】:

像这样使用 FieldList:

def field_level(form, field):
    all_values = form.selects.data
    value = field.data

def list_level(form, field):
    all_values = field.data

class MyForm(Form):
    selects = FieldList(SelectField('label', validators=[field_level]), validators=[list_level])

form = MyForm()
form.append_entry()
form.append_entry()
form.append_entry()

您可以在字段级别和字段列表级别拥有验证器

【讨论】:

以上是关于如何相互验证 wtforms 字段?的主要内容,如果未能解决你的问题,请参考以下文章

flask web 表单验证 WTForms

WTForms-如何预填充文本区域字段?

wtforms-表单生成及验证

Wtforms:如何使用具有动态选择值的选择字段生成空白值

在python wtforms中选择具有动态选择值的字段。

五十三:WTForms表单验证之常用验证器