passerelle/tests/test_utils_json.py

180 lines
5.9 KiB
Python

# passerelle - uniform access to multiple data sources and services
# Copyright (C) 2018 Entr'ouvert
#
# This program is free software: you can redistribute it and/or modify it
# under the terms of the GNU Affero General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
# passerelle - uniform access to multiple data sources and services
# Copyright (C) 2018 Entr'ouvert
#
# This program is free software: you can redistribute it and/or modify it
# under the terms of the GNU Affero General Public License as published
# by the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
import pytest
import jsonschema
from passerelle.utils.json import flatten, unflatten, flatten_json_schema, FLATTEN_SEPARATOR as SEP
def test_unflatten_base():
assert unflatten('') == ''
assert unflatten('a') == 'a'
assert unflatten([]) == []
assert unflatten([1]) == [1]
assert unflatten({}) == {}
assert unflatten(0) == 0
assert unflatten(1) == 1
assert unflatten(False) is False
assert unflatten(True) is True
def test_unflatten_dict():
assert unflatten({
'a' + SEP + 'b' + SEP + '0': 1,
'a' + SEP + 'c' + SEP + '1': 'a',
'a' + SEP + 'b' + SEP + '1': True,
'a' + SEP + 'c' + SEP + '0': [1],
}) == {
'a': {
'b': [1, True],
'c': [[1], 'a'],
}
}
def test_unflatten_array():
assert unflatten({
'0' + SEP + 'b' + SEP + '0': 1,
'1' + SEP + 'c' + SEP + '1': 'a',
'0' + SEP + 'b' + SEP + '1': True,
'1' + SEP + 'c' + SEP + '0': [1],
}) == [{'b': [1, True]},
{'c': [[1], 'a']}]
def test_unflatten_missing_final_index():
with pytest.raises(ValueError) as exc_info:
unflatten({
'1': 1
})
assert 'incomplete' in exc_info.value.args[0]
def test_unflatten_missing_intermediate_index():
with pytest.raises(ValueError) as exc_info:
unflatten({
'a' + SEP + '1' + SEP + 'b': 1
})
assert 'incomplete' in exc_info.value.args[0]
def test_flatten_array_schema():
schema = {
'type': 'array',
'items': {
'type': 'object',
'properties': {
'a': {
'type': 'string',
},
'b': {
'type': 'integer',
},
'c': {
'type': 'array',
'items': {
'type': 'integer',
}
}
},
'additionalProperties': False,
}
}
flattened_schema = flatten_json_schema(schema)
data = [
{'a': 'a', 'b': 1, 'c': [1, 2, 3]},
{'a': 'a', 'b': 1, 'c': [1, 2, 3]},
{'a': 'a', 'b': 1, 'c': [1, 2, 3]},
]
flattened_data = flatten(data)
jsonschema.validate(schema=schema, instance=data)
assert flattened_schema == {
'type': 'object',
'description': 'flattened schema *never* use for validation',
'properties': {
'0' + SEP + 'a': {'type': 'string'},
'0' + SEP + 'b': {'type': 'integer'},
'0' + SEP + 'c' + SEP + '0': {'type': 'integer'},
'0' + SEP + 'c' + SEP + '1': {'type': 'integer'},
'0' + SEP + 'c' + SEP + '2': {'type': 'integer'},
'1' + SEP + 'a': {'type': 'string'},
'1' + SEP + 'b': {'type': 'integer'},
'1' + SEP + 'c' + SEP + '0': {'type': 'integer'},
'1' + SEP + 'c' + SEP + '1': {'type': 'integer'},
'1' + SEP + 'c' + SEP + '2': {'type': 'integer'},
'2' + SEP + 'a': {'type': 'string'},
'2' + SEP + 'b': {'type': 'integer'},
'2' + SEP + 'c' + SEP + '0': {'type': 'integer'},
'2' + SEP + 'c' + SEP + '1': {'type': 'integer'},
'2' + SEP + 'c' + SEP + '2': {'type': 'integer'},
},
'additionalProperties': False,
}
# This should never be done as we cannot really validate all keys
# containing array indexes, here it works because array have less than 3
# elements.
jsonschema.validate(schema=flattened_schema, instance=flattened_data)
assert data == unflatten(flattened_data)
def test_flatten_dict_schema():
assert flatten_json_schema({
'type': 'object',
'properties': {
'a': {
'type': 'string',
},
'b': {
'type': 'integer',
},
'c': {
'type': 'array',
'items': {
'type': 'integer',
}
}
}
}) == {
'type': 'object',
'description': 'flattened schema *never* use for validation',
'properties': {
'a': {'type': 'string'},
'b': {'type': 'integer'},
'c' + SEP + '0': {'type': 'integer'},
'c' + SEP + '1': {'type': 'integer'},
'c' + SEP + '2': {'type': 'integer'},
},
'additionalProperties': False,
}