yspec is a deadly simple checker for structures. It is especially usefull for validation of different yaml/json files
yspec ./schema.yaml /tmp/data.jsonNOTE: yspec able to take yaml/json for schema and for data
from yspec.checker import check
# Some code that prepares data and rules
check(data, rules)Schema is a dict of rules. Every rule do some check according to 'match' field. Schema must include 'root' rule which is applied on top object in structure.
For example structure (in YAML):
---
- 'string1'
- 'string2'
will be valid for schema (in YAML):
---
root:
match: list
item: string
string:
match: stringmy_awesome_string:
match: stringmy_awesome_bool:
match: boolmy_awesome_int:
match: intmy_awesome_float:
match: floatempty_object:
match: noneany_type:
match: anyThere are two of them: dict and list. Both has the same recursion logic. At first we apply some checks on object itself, and then apply another (or the same) rule to childrens (list elements, or values of dict).
my_list:
match: list
item: some_other_ruleList is a recurent type. Fist it checks is the object a list, then it check every element according to rule in 'item' attr.
my_list:
match: dict
items:
key1: string_rule
key2: integer_rule
default_item: some_other_rule
required_items:
- key2That is a rule that describe a dict that has two keys (key1 and key2). One of keys (key2) is mandatory. Key1 should be checked according to 'string_rule' rule, while any other keys with any other (non key1 or key2) name will be checked according to some_other_rule.
If we has remove default_item
my_list:
match: dict
items:
key1: string_rule
key2: integer_rule
required_items:
- key2A dict could be with two keys (key1, key2) maximum
Schema:
---
boolean:
match: bool
string:
match: string
integer:
match: int
float:
match: float
list:
match: list
item: string
root:
match: dict
items:
key1: boolean
key2: string
key3: integer
key4: float
key5: listData:
---
key1: true
key2: "That is a string"
key3: 1
key4: 1.0
key5:
- "One more string"
- "Another string"constraint_list_item:
match: one_of
variants:
- integer_rule
- some_other_ruleOneOf match success if any of rules from "variants" success.
Suppose you need to check your little list of fruits:
---
- apple
- plumYou can use Set match for it. In variants clause of Set match you list all posible values for this type:
---
root:
match: list
item: fruits
fruits:
match: set
variants:
- apple
- orange
- plumSome times you need to do a separate checks for dict that have some key/value pair.
For example you have the following data:
---
- type: type1
payload:
- 1
- 1
- type: type2
payload:
- "that is a string"
- "that is a string2"In that example you have a dict wich has diffent payload depends of type key. That is possible to describe with following schema:
---
root:
match: list
item: list_item
list_item:
match: dict_key_selection
selector: type
variants:
type1: dict_with_int
type2: dict_with_string
dict_with_int:
match: dict
items:
type: string
payload: list_of_int
required_items:
- type
- payload
dict_with_string:
match: dict
items:
type: string
payload: list_of_string
required_items:
- type
- payload
list_of_int:
match: list
item: int
int:
match: int
list_of_string:
match: list
item: string
string:
match: string