0

我尝试验证一个 jsonschema,它定义了一个带有 aradius或 adiameter或两者都没有的圆,然后只设置默认半径。这是我的架构:

{
  "properties": {
    "position": {},
    "radius": {
      { "type": "number" }
    },
    "diameter": {
      { "type": "number" }
    }
  },
  "oneOf": [
    {
      "required": ["radius"]
    },
    {
      "required": ["diameter"]
    },
    {
      "properties": {
        "radius": {
          "default": 16
        }
      }
    }
  ],
  "additionalProperties": false
}

这是验证器,它设置默认值(如JSON 模式常见问题解答中所述):

from jsonschema import Draft7Validator, validators

def extend_with_default(validator_class):
  validate_properties = validator_class.VALIDATORS['properties']

  def set_defaults(validator, properties, instance, schema):
    for property, subschema in properties.items():
      if 'default' in subschema:
        instance.setdefault(property, subschema['default'])

    for error in validate_properties(
      validator, properties, instance, schema,
    ):
      yield error

  return validators.extend(
    validator_class, {'properties' : set_defaults},
  )
Validator = extend_with_default(Draft7Validator)

这个验证器会在我验证架构之前设置默认值,所以我只能设置半径或都不设置,但设置直径总是会引发错误。如果我将其更改为首先验证并稍后设置默认值(我宁愿不这样做,但可以),那么它会设置默认半径,尽管所需的直径已经存在。

有没有办法通过在python中设置默认半径来实现这一点而无需硬编码?

4

1 回答 1

0

验证器不需要更改。这是一个可能的解决方案:

{
  "properties": {
    "position": {},
    "radius": {
      { "type": "number" }
    },
    "diameter": {
      { "type": "number" }
    }
  },
  "if": {
    "not": {
      "anyOf": [
        {
          "required": ["radius"]
        },
        {
          "required": ["diameter"]
        }
      ]
    }
  },
  "then": {
    "properties": {
      "radius": {
        "default": 16
      }
    }
  },
  "oneOf": [
    {
      "required": ["radius"]
    },
    {
      "required": ["diameter"]
    }
  ],
  "additionalProperties": false
}

if仅当未设置半径或直径时才为真。只有这样才会设置默认半径。之后oneOf检查是否同时只设置了一个参数。

于 2021-02-03T13:17:04.670 回答