wtforms.BooleanField 的 DataRequired 验证器已损坏

Pal*_*aty 5 python validation json wtforms

我使用 WTForms(与 Flask、flask-wtf、sqlalchemy 一起)来验证 REST API 传入的 JSON。我意识到 WTForms 的目标更多是 HTML 表单渲染和验证,但我选择它是因为它可以从我的 sqlalchemy 模型中自动生成表单(感谢 wtforms.ext.sqlalchemy)。

无论如何,问题就在这里。我的模型之一包括布尔字段,它使用 DataRequired 验证器转换为 wtforms.BooleanField。问题是,即使我传递了正确的数据,验证也会失败,并显示“此字段为必填项”错误消息。我的表格:

class MyForm(Form):
    name = TextField('name', validators=[DataRequired()])
    disabled = BooleanField('disabled', validators=[DataRequired()])
Run Code Online (Sandbox Code Playgroud)

JSON数据是这样的:

'{"name": "John", "disabled": "false"}'
Run Code Online (Sandbox Code Playgroud)

我期待什么:

  1. {"disabled": "false"}-> 验证成功的强制 Python 数据:{'disabled': False}
  2. {"disabled": "true"}-> 验证成功的强制 Python 数据:{'disabled': True}
  3. {"disabled": ""}'{"disabled": "foo"}'-> 验证失败

目前在第一种情况下验证失败{'disabled': [u'This field is required.']}

我知道文档中有一条注释说DataRequired验证器“需要强制数据,而不是输入数据”,但是 1)表单是自动生成的wtforms.ext.sqlalchemy,2)如果我使用验证器,它应该如何表现InputRequired?检查(通过form.validate()某些数据是否存在,然后检查该数据是“真”还是“假”?

总而言之,我的问题是:

  1. 正确的验证方法是什么wtforms.BooleanField
  2. 也许还有其他一些框架可以根据给定的 sqlalchemy 模型验证传入的 JSON?

谢谢。

doo*_*des 0

有多种方法可以解决这个问题。您可以编写自己的转换器来使用具有真/假选择的无线电场,您可以使用数据过滤器,您可以设置默认值,但我认为您想要的行为可以通过以下方式实现:

MyForm = model_form(MyModel, db_session=db, field_args = {
    'disabled' : {
        'false_values': ['false'],
        'validators' : [InputRequired()] }
})
Run Code Online (Sandbox Code Playgroud)

编辑:如果您想要一个更严格的处理程序,您可以执行以下操作:

class BooleanRequired(object):
    field_flags = ('required', )

    def __init__(self, message=None):
        self.message = message

    def __call__(self, form, field):
        if field.data is None:
            if self.message is None:
                message = field.gettext('This field is required.')
            else:
                message = self.message

            field.errors[:] = []
            raise StopValidation(message)


class StrictBooleanField(BooleanField):
    def process_formdata(self, valuelist):
        self.data = None
        if valuelist:
            if valuelist[0] == 'false':
                self.data = False
            elif valuelist[0] == 'true':
                self.data = True


class StrictModelConverter(ModelConverter):
    @converts('Boolean')
    def conv_Boolean(self, field_args, **extra):
        return StrictBooleanField(**field_args)

MyForm = model_form(MyModel, db_session=db, converter=StrictModelConverter(),
           field_args = { 'disabled' : { 'validators': [BooleanRequired()] }
})
Run Code Online (Sandbox Code Playgroud)