我一直在使用flask中的sqlalchemy.orm中的@ validate装饰器来验证字段,只要所有字段彼此独立,一切都很顺利,例如:
@validates('field_one')
def validates_field_one(self, key, value):
#field one validation
@validates('field_two')
def validates_field_two(self, key, value):
#field two validation
然而,现在我需要做一些验证,这将需要同时访问field_one和field_two。看起来validates接受了validates装饰器的多个参数,但是,它只会为每个参数运行一次验证函数,如下所示:
@validates('field_one', 'field_two')
def validates_fields(self, keys, values):
#field validation
这将导致先验证field_one,然后验证field_two的工作流。然而,我想同时验证这两者(一个简单的例子是断言field_one的值不是field_two的值,一个例子是禁止图中的自循环,其中field_one和field_two引用节点,并且它在边上执行验证)。怎样做才是最好的方式呢?
发布于 2015-10-09 04:33:59
按照在模型上定义的顺序对字段进行排序。然后检查最后一个字段是否是正在验证的字段。否则,只需返回未选中的值。如果验证器正在验证其中一个较早的字段,则其中一些字段将尚未设置。
@validates('field_one', 'field_two')
def validates_fields(self, key, value):
if key == 'field_two':
assert self.field_one != value
return value
参见this example。
发布于 2021-07-14 02:55:04
在这里添加另一个答案,因为接受的答案不太符合我使用另一个字段来验证和修改关系/集合字段的用例,这些字段与@validates
并不真正兼容。在这种情况下,您可以使用before_flush
事件的事件侦听器来实现您想要的结果:
@event.listens_for(Session, 'before_flush')
def validate_and_modify_relationships(session, flush_context, instances):
"""
Complex validation that cannot be performed with @valdiates
"""
# new records only (for updates only, use session.dirty)
for instance in session.new:
if isinstance(instance, MyData):
if instance.some_other_value:
instance.some_relation = []
更多详细信息请访问:Flask-SQLAlchemy validation: prevent adding to relationship based on other field
https://stackoverflow.com/questions/32555829
复制相似问题