如何制定必要的规则取决于情况

Dmi*_*nko 5 python validation json cerberus

我有一个很大的json文档,如果其他字段具有确切值,则应在其中输入某些字段。例如

document = {'is_realty_address': False, 'postcode': 111111}
Run Code Online (Sandbox Code Playgroud)

如果is_realty_address == False,则必须提供邮政编码。所有规则(“必需”除外)都应用于文档中存在的字段,因此当我拥有

document = {'is_realty_address': False}
Run Code Online (Sandbox Code Playgroud)

规则对我的情况无济于事,因为我有很多“有条件需要”的字段,这些字段取决于许多不同的字段。因此,规则将使我的架构非常复杂。依赖关系也不起作用。我试过了:

{'postcode': {'dependencies': {'is_realty_address': False}, 'required': True}}
Run Code Online (Sandbox Code Playgroud)

如果邮政编码没有出现在文档中,则返回错误,无论is_realty_address的值是多少

v = Validator()
print(v.validate({'is_realty_address': False}, schema))
print(v.errors)

print(v.validate({'is_realty_address': True}, schema))
print(v.errors)
Run Code Online (Sandbox Code Playgroud)

此代码返回:

False
{'postcode': ['required field']}
False
{'postcode': ['required field']}
Run Code Online (Sandbox Code Playgroud)

我还尝试实现验证方法:

def _validate_conditional_required(self, conditional_required, field, value):
    """
    :param conditional_required:
    :param field:
    :param value:
    :return:
    The rule's arguments are validated against this schema:
    {'type': 'dict'}
    """
    for conditional_field, conditional_value in conditional_required.items():
        if self.document[conditional_field] == conditional_value and field not in self.document:
            self._error(field, errors.REQUIRED_FIELD)
Run Code Online (Sandbox Code Playgroud)

与模式

schema = {
    'is_realty_address': {'required': True, 'type': 'boolean'},
    'postcode': {'conditional_required': {'is_realty_address': False}},
}
Run Code Online (Sandbox Code Playgroud)

但是如果文档中没有“邮政编码”,则此规则不会运行。

有什么方法可以设置“有条件的”规则?我想看这段代码:

schema = {
    'is_realty_address': {'required': True, 'type': 'boolean'},
    'postcode': {'conditional_required': {'is_realty_address': False}},
}
v = Validator()
print(v.validate({'is_realty_address': False}, schema))
print(v.errors)

print(v.validate({'is_realty_address': True}, schema))
print(v.errors)
Run Code Online (Sandbox Code Playgroud)

返回:

True

False
{'postcode': ['required field']}
Run Code Online (Sandbox Code Playgroud)

Dmi*_*nko 10

我发现,我的问题可以通过“排除”和“oneof”规则的组合来解决

schema = {
'is_realty_address': {
    'required': True, 'type': 'boolean',
    'oneof': [{'excludes': 'postcode', 'allowed': [False]}, {'allowed': [True]}]
},
'postcode': {'type': 'integer', 'required': True}}
v = Validator()
print(v.validate({'is_realty_address': True}, schema))
print(v.errors)

print(v.validate({'is_realty_address': False, 'postcode': 111111}, schema))
print(v.errors)
Run Code Online (Sandbox Code Playgroud)

此代码返回:

False
{'postcode': ['required field']}
False
{'is_realty_address': [{'oneof': ['none or more than one rule validate', {'oneof definition 1': ['unallowed value False'], 'oneof definition 0': ["'postcode' must not be present with 'is_realty_address'"]}]}]}
Run Code Online (Sandbox Code Playgroud)

因此,模式不会因为这个变体而变得非常复杂。

  • 您应该更改为: ```schema = { 'is_realty_address': { 'required': True, 'type': 'boolean', 'oneof': [{'excludes': 'postcode', 'allowed': [False ]}, {'依赖项': '邮政编码', '允许': [True]}] }, '邮政编码': {'类型': '整数'}}``` (2认同)