aboutsummaryrefslogtreecommitdiff
path: root/diplomacy/utils/tests/test_parsing.py
diff options
context:
space:
mode:
authorPhilip Paquette <pcpaquette@gmail.com>2018-09-26 07:48:55 -0400
committerPhilip Paquette <pcpaquette@gmail.com>2019-04-18 11:14:24 -0400
commit6187faf20384b0c5a4966343b2d4ca47f8b11e45 (patch)
tree151ccd21aea20180432c13fe4b58240d3d9e98b6 /diplomacy/utils/tests/test_parsing.py
parent96b7e2c03ed98705754f13ae8efa808b948ee3a8 (diff)
Release v1.0.0 - Diplomacy Game Engine - AGPL v3+ License
Diffstat (limited to 'diplomacy/utils/tests/test_parsing.py')
-rw-r--r--diplomacy/utils/tests/test_parsing.py307
1 files changed, 307 insertions, 0 deletions
diff --git a/diplomacy/utils/tests/test_parsing.py b/diplomacy/utils/tests/test_parsing.py
new file mode 100644
index 0000000..f64ad26
--- /dev/null
+++ b/diplomacy/utils/tests/test_parsing.py
@@ -0,0 +1,307 @@
+# ==============================================================================
+# Copyright (C) 2019 - Philip Paquette, Steven Bocco
+#
+# 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 <https://www.gnu.org/licenses/>.
+# ==============================================================================
+""" Test module parsing. """
+from diplomacy.utils import exceptions, parsing
+from diplomacy.utils.sorted_dict import SortedDict
+from diplomacy.utils.sorted_set import SortedSet
+from diplomacy.utils.tests.test_common import assert_raises
+from diplomacy.utils.tests.test_jsonable import MyJsonable
+
+class MyStringable():
+ """ Example of Stringable class.
+ As instances of such class may be used as dict keys, class should define a proper __hash__().
+ """
+
+ def __init__(self, value):
+ self.attribute = str(value)
+
+ def __str__(self):
+ return 'MyStringable %s' % self.attribute
+
+ def __hash__(self):
+ return hash(self.attribute)
+
+ def __eq__(self, other):
+ return isinstance(other, MyStringable) and self.attribute == other.attribute
+
+ def __lt__(self, other):
+ return isinstance(other, MyStringable) and self.attribute < other.attribute
+
+ @staticmethod
+ def from_string(str_repr):
+ """ Converts a string representation `str_repr` of MyStringable to an instance of MyStringable. """
+ return MyStringable(str_repr[len('MyStringable '):])
+
+def test_default_value_type():
+ """ Test default value type. """
+
+ for default_value in (True, False, None):
+ checker = parsing.DefaultValueType(bool, default_value)
+ assert_raises(lambda ch=checker: ch.validate(1), exceptions.TypeException)
+ assert_raises(lambda ch=checker: ch.validate(1.1), exceptions.TypeException)
+ assert_raises(lambda ch=checker: ch.validate(''), exceptions.TypeException)
+ for value in (True, False, None):
+ checker.validate(value)
+ if value is None:
+ assert checker.to_type(value) is default_value
+ assert checker.to_json(value) is default_value
+ else:
+ assert checker.to_type(value) is value
+ assert checker.to_json(value) is value
+ assert checker.update(None) is default_value
+
+def test_optional_value_type():
+ """ Test optional value type. """
+
+ checker = parsing.OptionalValueType(bool)
+ assert_raises(lambda ch=checker: ch.validate(1), exceptions.TypeException)
+ assert_raises(lambda ch=checker: ch.validate(1.1), exceptions.TypeException)
+ assert_raises(lambda ch=checker: ch.validate(''), exceptions.TypeException)
+ for value in (True, False, None):
+ checker.validate(value)
+ assert checker.to_type(value) is value
+ assert checker.to_json(value) is value
+ assert checker.update(None) is None
+
+def test_sequence_type():
+ """ Test sequence type. """
+
+ # With default sequence builder.
+ checker = parsing.SequenceType(int)
+ checker.validate((1, 2, 3))
+ checker.validate([1, 2, 3])
+ checker.validate({1, 2, 3})
+ checker.validate(SortedSet(int))
+ checker.validate(SortedSet(int, (1, 2, 3)))
+ assert_raises(lambda: checker.validate((1, 2, 3.0)), exceptions.TypeException)
+ assert_raises(lambda: checker.validate((1.0, 2.0, 3.0)), exceptions.TypeException)
+ assert isinstance(checker.to_type((1, 2, 3)), list)
+ # With SortedSet as sequence builder.
+ checker = parsing.SequenceType(float)
+ checker.validate((1.0, 2.0, 3.0))
+ checker.validate([1.0, 2.0, 3.0])
+ checker.validate({1.0, 2.0, 3.0})
+ assert_raises(lambda: checker.validate((1, 2, 3.0)), exceptions.TypeException)
+ assert_raises(lambda: checker.validate((1.0, 2.0, 3)), exceptions.TypeException)
+ checker = parsing.SequenceType(int, sequence_builder=SortedSet.builder(int))
+ initial_list = (1, 2, 7, 7, 1)
+ checker.validate(initial_list)
+ updated_list = checker.update(initial_list)
+ assert isinstance(updated_list, SortedSet) and updated_list.element_type is int
+ assert updated_list == SortedSet(int, (1, 2, 7))
+ assert checker.to_json(updated_list) == [1, 2, 7]
+ assert checker.to_type([7, 2, 1, 1, 7, 1, 7]) == updated_list
+
+def test_jsonable_class_type():
+ """ Test parser for Jsonable sub-classes. """
+
+ checker = parsing.JsonableClassType(MyJsonable)
+ my_jsonable = MyJsonable(field_a=False, field_b='test', field_e={1}, field_f=[6.5])
+ my_jsonable_dict = {
+ 'field_a': False,
+ 'field_b': 'test',
+ 'field_e': (1, 2),
+ 'field_f': (1.0, 2.0),
+ }
+ checker.validate(my_jsonable)
+ assert_raises(lambda: checker.validate(None), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(my_jsonable_dict), exceptions.TypeException)
+
+def test_stringable_type():
+ """ Test stringable type. """
+
+ checker = parsing.StringableType(str)
+ checker.validate('0')
+ checker = parsing.StringableType(MyStringable)
+ checker.validate(MyStringable('test'))
+ assert_raises(lambda: checker.validate('test'), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(None), exceptions.TypeException)
+
+def test_dict_type():
+ """ Test dict type. """
+
+ checker = parsing.DictType(str, int)
+ checker.validate({'test': 1})
+ assert_raises(lambda: checker.validate({'test': 1.0}), exceptions.TypeException)
+ checker = parsing.DictType(MyStringable, float)
+ checker.validate({MyStringable('12'): 2.5})
+ assert_raises(lambda: checker.validate({'12': 2.5}), exceptions.TypeException)
+ assert_raises(lambda: checker.validate({MyStringable('12'): 2}), exceptions.TypeException)
+ checker = parsing.DictType(MyStringable, float, dict_builder=SortedDict.builder(MyStringable, float))
+ value = {MyStringable(12): 22.0}
+ checker.validate(value)
+ updated_value = checker.update(value)
+ assert isinstance(updated_value, SortedDict)
+ assert updated_value.key_type is MyStringable
+ assert updated_value.val_type is float
+ json_value = {'MyStringable 12': 22.0}
+ assert checker.to_type(json_value) == SortedDict(MyStringable, float, {MyStringable('12'): 22.0})
+ assert checker.to_json(SortedDict(MyStringable, float, {MyStringable(12): 22.0})) == json_value
+
+def test_sequence_of_values_type():
+ """ Test parser for sequence of allowed values. """
+
+ checker = parsing.EnumerationType({'a', 'b', 'c', 'd'})
+ checker.validate('d')
+ checker.validate('c')
+ checker.validate('b')
+ checker.validate('a')
+ assert_raises(lambda: checker.validate('e'), exceptions.ValueException)
+
+def test_sequence_of_primitives_type():
+ """ Test parser for sequence of primitive types. """
+
+ checker = parsing.SequenceOfPrimitivesType((int, bool))
+ checker.validate(False)
+ checker.validate(True)
+ checker.validate(0)
+ checker.validate(1)
+ assert_raises(lambda: checker.validate(0.0), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(1.0), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(''), exceptions.TypeException)
+ assert_raises(lambda: checker.validate('a non-empty string'), exceptions.TypeException)
+
+def test_primitive_type():
+ """ Test parser for primitive type. """
+
+ checker = parsing.PrimitiveType(bool)
+ checker.validate(True)
+ checker.validate(False)
+ assert_raises(lambda: checker.validate(None), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(0), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(1), exceptions.TypeException)
+ assert_raises(lambda: checker.validate(''), exceptions.TypeException)
+ assert_raises(lambda: checker.validate('a non-empty string'), exceptions.TypeException)
+
+def test_model_parsing():
+ """ Test parsing for a real model. """
+
+ model = {
+ 'name': str,
+ 'language': ('fr', 'en'),
+ 'myjsonable': parsing.JsonableClassType(MyJsonable),
+ 'mydict': parsing.DictType(str, float),
+ 'nothing': (bool, str),
+ 'default_float': parsing.DefaultValueType(float, 33.44),
+ 'optional_float': parsing.OptionalValueType(float)
+ }
+ bad_data_field = {
+ '_name_': 'hello',
+ 'language': 'fr',
+ 'myjsonable': MyJsonable(field_a=False, field_b='test', field_e={1}, field_f=[6.5]),
+ 'mydict': {
+ 'a': 2.5,
+ 'b': -1.6
+ },
+ 'nothing': 'thanks'
+ }
+ bad_data_type = {
+ 'name': 'hello',
+ 'language': 'fr',
+ 'myjsonable': MyJsonable(field_a=False, field_b='test', field_e={1}, field_f=[6.5]),
+ 'mydict': {
+ 'a': 2.5,
+ 'b': -1.6
+ },
+ 'nothing': 2.5
+ }
+ bad_data_value = {
+ 'name': 'hello',
+ 'language': '__',
+ 'myjsonable': MyJsonable(field_a=False, field_b='test', field_e={1}, field_f=[6.5]),
+ 'mydict': {
+ 'a': 2.5,
+ 'b': -1.6
+ },
+ 'nothing': '2.5'
+ }
+ good_data = {
+ 'name': 'hello',
+ 'language': 'fr',
+ 'myjsonable': MyJsonable(field_a=False, field_b='test', field_e={1}, field_f=[6.5]),
+ 'mydict': {
+ 'a': 2.5,
+ 'b': -1.6
+ },
+ 'nothing': '2.5'
+ }
+ assert_raises(lambda: parsing.validate_data(bad_data_field, model), (exceptions.TypeException,))
+ assert_raises(lambda: parsing.validate_data(bad_data_type, model), (exceptions.TypeException,))
+ assert_raises(lambda: parsing.validate_data(bad_data_value, model), (exceptions.ValueException,))
+
+ assert 'default_float' not in good_data
+ assert 'optional_float' not in good_data
+ parsing.validate_data(good_data, model)
+ updated_good_data = parsing.update_data(good_data, model)
+ assert 'default_float' in updated_good_data
+ assert 'optional_float' in updated_good_data
+ assert updated_good_data['default_float'] == 33.44
+ assert updated_good_data['optional_float'] is None
+
+def test_converter_type():
+ """ Test parser converter type. """
+
+ def converter_to_int(val):
+ """ Converts value to integer """
+ try:
+ return int(val)
+ except (ValueError, TypeError):
+ return 0
+
+ checker = parsing.ConverterType(str, converter_function=lambda val: 'String of %s' % val)
+ checker.validate('a string')
+ checker.validate(10)
+ checker.validate(True)
+ checker.validate(None)
+ checker.validate(-2.5)
+ assert checker.update(10) == 'String of 10'
+ assert checker.update(False) == 'String of False'
+ assert checker.update('string') == 'String of string'
+ checker = parsing.ConverterType(int, converter_function=converter_to_int)
+ checker.validate(10)
+ checker.validate(True)
+ checker.validate(None)
+ checker.validate(-2.5)
+ assert checker.update(10) == 10
+ assert checker.update(True) == 1
+ assert checker.update(False) == 0
+ assert checker.update(-2.5) == -2
+ assert checker.update('44') == 44
+ assert checker.update('a') == 0
+
+def test_indexed_sequence():
+ """ Test parser type for dicts stored as sequences. """
+ checker = parsing.IndexedSequenceType(parsing.DictType(str, parsing.JsonableClassType(MyJsonable)), 'field_b')
+ sequence = [
+ MyJsonable(field_a=True, field_b='x1', field_e=[1, 2, 3], field_f=[1., 2., 3.]),
+ MyJsonable(field_a=True, field_b='x3', field_e=[1, 2, 3], field_f=[1., 2., 3.]),
+ MyJsonable(field_a=True, field_b='x2', field_e=[1, 2, 3], field_f=[1., 2., 3.]),
+ MyJsonable(field_a=True, field_b='x5', field_e=[1, 2, 3], field_f=[1., 2., 3.]),
+ MyJsonable(field_a=True, field_b='x4', field_e=[1, 2, 3], field_f=[1., 2., 3.])
+ ]
+ dct = {element.field_b: element for element in sequence}
+ checker.validate(dct)
+ checker.update(dct)
+ jval = checker.to_json(dct)
+ assert isinstance(jval, list), type(jval)
+ from_jval = checker.to_type(jval)
+ assert isinstance(from_jval, dict), type(from_jval)
+ assert len(dct) == 5
+ assert len(from_jval) == 5
+ for key in ('x1', 'x2', 'x3', 'x4', 'x5'):
+ assert key in dct, (key, list(dct.keys()))
+ assert key in from_jval, (key, list(from_jval.keys()))