Skip to content

Commit

Permalink
Considers defaults defined in allow_unknown when normalizing
Browse files Browse the repository at this point in the history
Closes pyeve#310.
  • Loading branch information
funkyfuture committed Jun 18, 2018
1 parent ce0206d commit b62d041
Show file tree
Hide file tree
Showing 4 changed files with 66 additions and 23 deletions.
2 changes: 2 additions & 0 deletions CHANGES.rst
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@ In Development
- Remove ``utils.is_class``
- Remove ``Registry`` from top level namespace. Closes :issue:`354`.
- Fix: Normalization rules defined within the ``items`` rule are applied. (Evgeny Odegov)
- Fix: Defaults are applied to undefined fields from an ``allow_unknown``
definition. Closes :issue:`310`. (Frank Sachsenheim)
- Update homepage URL in package metadata. Closes :issue:`382`.
- Docs: add feature freeze note to CONTRIBUTING and note on Python support in README (Frank Sachsenheim).
- Docs: add the intent of a ``dataclasses`` module to ROADMAP.md
Expand Down
25 changes: 25 additions & 0 deletions cerberus/tests/test_normalization.py
Original file line number Diff line number Diff line change
Expand Up @@ -517,3 +517,28 @@ def test_allow_unknown_with_purge_unknown_subdocument():
document = {'foo': {'bar': 'baz', 'corge': False}, 'thud': 'xyzzy'}
expected = {'foo': {'bar': 'baz', 'corge': False}}
assert_normalized(document, expected, schema, validator)


def test_defaults_in_allow_unknown_schema():
schema = {
'meta': {'type': 'dict'},
'version': {'type': 'string'},
}
allow_unknown = {
'type': 'dict',
'schema': {
'cfg_path': {'type': 'string', 'default': 'cfg.yaml'},
'package': {'type': 'string'},
}
}
validator = Validator(schema=schema, allow_unknown=allow_unknown)

document = {
'version': '1.2.3',
'plugin_foo': {'package': 'foo'}
}
expected = {
'version': '1.2.3',
'plugin_foo': {'package': 'foo', 'cfg_path': 'cfg.yaml'}
}
assert_normalized(document, expected, validator=validator)
2 changes: 1 addition & 1 deletion cerberus/tests/test_schema.py
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,7 @@ def test_validated_schema_cache():
v = Validator({'foozifix': {'coerce': int}})
assert len(v._valid_schemas) == cache_size

max_cache_size = 152
max_cache_size = 157
assert cache_size <= max_cache_size, \
"There's an unexpected high amount (%s) of cached valid " \
"definition schemas. Unless you added further tests, " \
Expand Down
60 changes: 38 additions & 22 deletions cerberus/validator.py
Original file line number Diff line number Diff line change
Expand Up @@ -667,29 +667,31 @@ def __normalize_coerce(self, processor, field, value, nullable, error):

def __normalize_containers(self, mapping, schema):
for field in mapping:
if field not in schema:
continue
rules = set(schema.get(field, ()))

# TODO: This check conflates validation and normalization
if isinstance(mapping[field], Mapping):
if 'keyschema' in schema[field]:
if 'keyschema' in rules:
self.__normalize_mapping_per_keyschema(
field, mapping, schema[field]['keyschema'])
if 'valueschema' in schema[field]:
if 'valueschema' in rules:
self.__normalize_mapping_per_valueschema(
field, mapping, schema[field]['valueschema'])
if set(schema[field]) & set(('allow_unknown', 'purge_unknown',
'schema')):
if rules & set(('allow_unknown', 'purge_unknown', 'schema')) \
or isinstance(self.allow_unknown, Mapping):
try:
self.__normalize_mapping_per_schema(
field, mapping, schema)
except _SchemaRuleTypeError:
pass

elif isinstance(mapping[field], _str_type):
continue

elif isinstance(mapping[field], Sequence):
if 'schema' in schema[field]:
if 'schema' in rules:
self.__normalize_sequence_per_schema(field, mapping, schema)
elif 'items' in schema[field]:
elif 'items' in rules:
self.__normalize_sequence_per_items(field, mapping, schema)

def __normalize_mapping_per_keyschema(self, field, mapping, property_rules):
Expand Down Expand Up @@ -727,11 +729,14 @@ def __normalize_mapping_per_valueschema(self, field, mapping, value_rules):
self._error(validator._errors)

def __normalize_mapping_per_schema(self, field, mapping, schema):
rules = schema.get(field, {})
if not rules and isinstance(self.allow_unknown, Mapping):
rules = self.allow_unknown
validator = self._get_child_validator(
document_crumb=field, schema_crumb=(field, 'schema'),
schema=schema[field].get('schema', {}),
allow_unknown=schema[field].get('allow_unknown', self.allow_unknown), # noqa: E501
purge_unknown=schema[field].get('purge_unknown', self.purge_unknown)) # noqa: E501
schema=rules.get('schema', {}),
allow_unknown=rules.get('allow_unknown', self.allow_unknown), # noqa: E501
purge_unknown=rules.get('purge_unknown', self.purge_unknown)) # noqa: E501
value_type = type(mapping[field])
result_value = validator.normalized(mapping[field],
always_return_document=True)
Expand Down Expand Up @@ -818,34 +823,45 @@ def __validate_readonly_fields(self, mapping, schema):
mapping[field])

def __normalize_default_fields(self, mapping, schema):
fields = [x for x in schema if x not in mapping or
mapping[x] is None and not schema[x].get('nullable', False)]
empty_fields = [
x for x in schema
if x not in mapping
or (mapping[x] is None # noqa: W503
and not schema[x].get('nullable', False)) # noqa: W503
]

try:
fields_with_default = [x for x in fields if 'default' in schema[x]]
fields_with_default = [
x for x in empty_fields
if 'default' in schema[x]
]
except TypeError:
raise _SchemaRuleTypeError
for field in fields_with_default:
self._normalize_default(mapping, schema, field)

known_fields_states = set()
fields = [x for x in fields if 'default_setter' in schema[x]]
while fields:
field = fields.pop(0)
fields_with_default_setter = [
x for x in empty_fields
if 'default_setter' in schema[x]
]
while fields_with_default_setter:
field = fields_with_default_setter.pop(0)
try:
self._normalize_default_setter(mapping, schema, field)
except KeyError:
fields.append(field)
fields_with_default_setter.append(field)
except Exception as e:
self._error(field, errors.SETTING_DEFAULT_FAILED, str(e))

fields_state = tuple(fields)
if fields_state in known_fields_states:
for field in fields:
fields_processing_state = hash(tuple(fields_with_default_setter))
if fields_processing_state in known_fields_states:
for field in fields_with_default_setter:
self._error(field, errors.SETTING_DEFAULT_FAILED,
'Circular dependencies of default setters.')
break
else:
known_fields_states.add(fields_state)
known_fields_states.add(fields_processing_state)

def _normalize_default(self, mapping, schema, field):
""" {'nullable': True} """
Expand Down

0 comments on commit b62d041

Please sign in to comment.