【问题标题】:How do I validate wtforms fields against one another?如何相互验证 wtforms 字段?
【发布时间】:2023-03-16 09:46:01
【问题描述】:

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

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

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

【问题讨论】:

    标签: python validation flask wtforms flask-wtforms


    【解决方案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()
        

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

        【讨论】:

          猜你喜欢
          • 2020-04-17
          • 2015-01-20
          • 2018-02-24
          • 1970-01-01
          • 2016-01-10
          • 1970-01-01
          • 1970-01-01
          • 2023-02-10
          • 1970-01-01
          相关资源
          最近更新 更多