
unfortunately IntEnum can't be bent like we would need it (extensible). So we had to write our own.... The members of the Enum still behave like ints, but also have .name and .value attributes, should they be needed. needed adoptions to correctly use (and test) the EnumType are included. Change-Id: Ie019d2f449a244c4fab00554b6c6daaac8948b59 Reviewed-on: https://forge.frm2.tum.de/review/17843 Tested-by: JenkinsCodeReview <bjoern_pedersen@frm2.tum.de> Reviewed-by: Enrico Faulhaber <enrico.faulhaber@frm2.tum.de>
382 lines
12 KiB
Python
382 lines
12 KiB
Python
# -*- coding: utf-8 -*-
|
|
# *****************************************************************************
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify it under
|
|
# the terms of the GNU General Public License as published by the Free Software
|
|
# Foundation; either version 2 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 General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License along with
|
|
# this program; if not, write to the Free Software Foundation, Inc.,
|
|
# 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
# Module authors:
|
|
# Enrico Faulhaber <enrico.faulhaber@frm2.tum.de>
|
|
#
|
|
# *****************************************************************************
|
|
"""test data types."""
|
|
|
|
import sys
|
|
sys.path.insert(0, sys.path[0] + '/..')
|
|
|
|
# no fixtures needed
|
|
import pytest
|
|
|
|
|
|
from secop.datatypes import DataType, FloatRange, IntRange, \
|
|
EnumType, BLOBType, StringType, BoolType, ArrayOf, TupleOf, StructOf, \
|
|
get_datatype, ProgrammingError
|
|
|
|
|
|
def test_DataType():
|
|
dt = DataType()
|
|
assert dt.as_json == ['undefined']
|
|
|
|
with pytest.raises(NotImplementedError):
|
|
dt = DataType()
|
|
dt.validate('')
|
|
dt.export_value('')
|
|
dt.import_value('')
|
|
|
|
|
|
def test_FloatRange():
|
|
dt = FloatRange(-3.14, 3.14)
|
|
assert dt.as_json == ['double', -3.14, 3.14]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate(-9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('XX')
|
|
with pytest.raises(ValueError):
|
|
dt.validate([19, 'X'])
|
|
dt.validate(1)
|
|
dt.validate(0)
|
|
assert dt.export_value(-2.718) == -2.718
|
|
assert dt.import_value(-2.718) == -2.718
|
|
with pytest.raises(ValueError):
|
|
FloatRange('x', 'Y')
|
|
|
|
dt = FloatRange()
|
|
assert dt.as_json == ['double']
|
|
|
|
|
|
def test_IntRange():
|
|
dt = IntRange(-3, 3)
|
|
assert dt.as_json == ['int', -3, 3]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate(-9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('XX')
|
|
with pytest.raises(ValueError):
|
|
dt.validate([19, 'X'])
|
|
dt.validate(1)
|
|
dt.validate(0)
|
|
with pytest.raises(ValueError):
|
|
IntRange('xc', 'Yx')
|
|
|
|
dt = IntRange()
|
|
assert dt.as_json == ['int']
|
|
|
|
|
|
def test_EnumType():
|
|
# test constructor catching illegal arguments
|
|
with pytest.raises(TypeError):
|
|
EnumType(1)
|
|
with pytest.raises(TypeError):
|
|
EnumType(['b', 0])
|
|
|
|
dt = EnumType('dt', a=3, c=7, stuff=1)
|
|
assert dt.as_json == ['enum', dict(a=3, c=7, stuff=1)]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate(-9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('XX')
|
|
with pytest.raises(TypeError):
|
|
dt.validate([19, 'X'])
|
|
|
|
assert dt.validate('a') == 3
|
|
assert dt.validate('stuff') == 1
|
|
assert dt.validate(1) == 1
|
|
with pytest.raises(ValueError):
|
|
dt.validate(2)
|
|
|
|
assert dt.export_value('c') == 7
|
|
assert dt.export_value('stuff') == 1
|
|
assert dt.export_value(1) == 1
|
|
assert dt.import_value('c') == 7
|
|
assert dt.import_value('a') == 3
|
|
assert dt.import_value('stuff') == 1
|
|
with pytest.raises(ValueError):
|
|
dt.export_value(2)
|
|
with pytest.raises(ValueError):
|
|
dt.import_value('A')
|
|
|
|
|
|
def test_BLOBType():
|
|
# test constructor catching illegal arguments
|
|
with pytest.raises(ValueError):
|
|
dt = BLOBType()
|
|
dt = BLOBType(10)
|
|
assert dt.as_json == ['blob', 10]
|
|
|
|
dt = BLOBType(3, 10)
|
|
assert dt.as_json == ['blob', 10, 3]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('av')
|
|
with pytest.raises(ValueError):
|
|
dt.validate('abcdefghijklmno')
|
|
assert dt.validate('abcd') == b'abcd'
|
|
assert dt.validate(b'abcd') == b'abcd'
|
|
assert dt.validate(u'abcd') == b'abcd'
|
|
|
|
assert dt.export_value('abcd') == 'YWJjZA=='
|
|
assert dt.export_value(b'abcd') == 'YWJjZA=='
|
|
assert dt.export_value(u'abcd') == 'YWJjZA=='
|
|
assert dt.import_value('YWJjZA==') == 'abcd'
|
|
|
|
|
|
def test_StringType():
|
|
# test constructor catching illegal arguments
|
|
dt = StringType()
|
|
dt = StringType(12)
|
|
assert dt.as_json == ['string', 12]
|
|
|
|
dt = StringType(4, 11)
|
|
assert dt.as_json == ['string', 11, 4]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('av')
|
|
with pytest.raises(ValueError):
|
|
dt.validate('abcdefghijklmno')
|
|
with pytest.raises(ValueError):
|
|
dt.validate('abcdefg\0')
|
|
assert dt.validate('abcd') == b'abcd'
|
|
assert dt.validate(b'abcd') == b'abcd'
|
|
assert dt.validate(u'abcd') == b'abcd'
|
|
|
|
assert dt.export_value('abcd') == b'abcd'
|
|
assert dt.export_value(b'abcd') == b'abcd'
|
|
assert dt.export_value(u'abcd') == b'abcd'
|
|
assert dt.import_value(u'abcd') == 'abcd'
|
|
|
|
|
|
def test_BoolType():
|
|
# test constructor catching illegal arguments
|
|
dt = BoolType()
|
|
assert dt.as_json == ['bool']
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('av')
|
|
|
|
assert dt.validate('true') is True
|
|
assert dt.validate('off') is False
|
|
assert dt.validate(1) is True
|
|
|
|
assert dt.export_value('false') is False
|
|
assert dt.export_value(0) is False
|
|
assert dt.export_value('on') is True
|
|
|
|
assert dt.import_value(False) is False
|
|
assert dt.import_value(True) is True
|
|
with pytest.raises(ValueError):
|
|
dt.import_value('av')
|
|
|
|
|
|
def test_ArrayOf():
|
|
# test constructor catching illegal arguments
|
|
with pytest.raises(ValueError):
|
|
ArrayOf(int)
|
|
with pytest.raises(ValueError):
|
|
ArrayOf(IntRange(-10,10))
|
|
dt = ArrayOf(IntRange(-10, 10), 5)
|
|
assert dt.as_json == ['array', ['int', -10, 10], 5]
|
|
|
|
dt = ArrayOf(IntRange(-10, 10), 1, 3)
|
|
assert dt.as_json == ['array', ['int', -10, 10], 3, 1]
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate('av')
|
|
|
|
assert dt.validate([1, 2, 3]) == [1, 2, 3]
|
|
|
|
assert dt.export_value([1, 2, 3]) == [1, 2, 3]
|
|
assert dt.import_value([1, 2, 3]) == [1, 2, 3]
|
|
|
|
|
|
def test_TupleOf():
|
|
# test constructor catching illegal arguments
|
|
with pytest.raises(ValueError):
|
|
TupleOf(2)
|
|
|
|
dt = TupleOf(IntRange(-10, 10), BoolType())
|
|
assert dt.as_json == ['tuple', [['int', -10, 10], ['bool']]]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate([99, 'X'])
|
|
|
|
assert dt.validate([1, True]) == [1, True]
|
|
|
|
assert dt.export_value([1, True]) == [1, True]
|
|
assert dt.import_value([1, True]) == [1, True]
|
|
|
|
|
|
def test_StructOf():
|
|
# test constructor catching illegal arguments
|
|
with pytest.raises(TypeError):
|
|
StructOf(IntRange) # pylint: disable=E1121
|
|
with pytest.raises(ProgrammingError):
|
|
StructOf(IntRange=1)
|
|
|
|
dt = StructOf(a_string=StringType(55), an_int=IntRange(0, 999))
|
|
assert dt.as_json == ['struct', {'a_string': ['string', 55],
|
|
'an_int': ['int', 0, 999],
|
|
}]
|
|
|
|
with pytest.raises(ValueError):
|
|
dt.validate(9)
|
|
with pytest.raises(ValueError):
|
|
dt.validate([99, 'X'])
|
|
with pytest.raises(ValueError):
|
|
dt.validate(dict(a_string='XXX', an_int=1811))
|
|
|
|
assert dt.validate(dict(a_string='XXX', an_int=8)) == {'a_string': 'XXX',
|
|
'an_int': 8}
|
|
assert dt.export_value({'an_int': 13, 'a_string': 'WFEC'}) == {'a_string': 'WFEC', 'an_int': 13}
|
|
assert dt.import_value({'an_int': 13, 'a_string': 'WFEC'}) == {'a_string': 'WFEC', 'an_int': 13}
|
|
|
|
|
|
def test_get_datatype():
|
|
with pytest.raises(ValueError):
|
|
get_datatype(1)
|
|
with pytest.raises(ValueError):
|
|
get_datatype(True)
|
|
with pytest.raises(ValueError):
|
|
get_datatype(str)
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['undefined'])
|
|
|
|
assert isinstance(get_datatype(['bool']), BoolType)
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['bool', 3])
|
|
|
|
assert isinstance(get_datatype(['int']), IntRange)
|
|
assert isinstance(get_datatype(['int', -10]), IntRange)
|
|
assert isinstance(get_datatype(['int', None, 10]), IntRange)
|
|
assert isinstance(get_datatype(['int', -10, 10]), IntRange)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['int', 10, -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['int', 1, 2, 3])
|
|
|
|
assert isinstance(get_datatype(['double']), FloatRange)
|
|
assert isinstance(get_datatype(['double', -2.718]), FloatRange)
|
|
assert isinstance(get_datatype(['double', None, 3.14]), FloatRange)
|
|
assert isinstance(get_datatype(['double', -9.9, 11.1]), FloatRange)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['double', 10, -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['double', 1, 2, 3])
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['enum'])
|
|
assert isinstance(get_datatype(['enum', dict(a=-2)]), EnumType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['enum', 10, -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['enum', [1, 2, 3]])
|
|
|
|
assert isinstance(get_datatype(['blob', 1]), BLOBType)
|
|
assert isinstance(get_datatype(['blob', 10, 1]), BLOBType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['blob', 10, -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['blob', 10, -10, 1])
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['string'])
|
|
assert isinstance(get_datatype(['string', 1]), StringType)
|
|
assert isinstance(get_datatype(['string', 10, 1]), StringType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['string', 10, -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['string', 10, -10, 1])
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['array'])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['array', 1])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['array', [1], 2, 3])
|
|
assert isinstance(get_datatype(['array', ['blob', 1], 1]), ArrayOf)
|
|
assert isinstance(get_datatype(['array', ['blob', 1], 1]).subtype, BLOBType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['array', ['blob', 1], -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['array', ['blob', 1], 10, -10])
|
|
|
|
assert isinstance(get_datatype(['array', ['blob', 1], 10, 1]), ArrayOf)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['tuple'])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['tuple', 1])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['tuple', [1], 2, 3])
|
|
assert isinstance(get_datatype(['tuple', [['blob', 1]]]), TupleOf)
|
|
assert isinstance(get_datatype(
|
|
['tuple', [['blob', 1]]]).subtypes[0], BLOBType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['tuple', [['blob', 1]], -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['tuple', [['blob', 1]], 10, -10])
|
|
|
|
assert isinstance(get_datatype(['tuple', [['blob', 1], ['int']]]), TupleOf)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['struct'])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['struct', 1])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['struct', [1], 2, 3])
|
|
assert isinstance(get_datatype(['struct', {'blob': ['blob', 1]}]), StructOf)
|
|
assert isinstance(get_datatype(
|
|
['struct', {'blob': ['blob', 1]}]).named_subtypes['blob'], BLOBType)
|
|
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['struct', [['blob', 1]], -10])
|
|
with pytest.raises(ValueError):
|
|
get_datatype(['struct', [['blob', 1]], 10, -10])
|
|
|
|
assert isinstance(get_datatype(
|
|
['struct', {'blob': ['blob', 1], 'int':['int']}]), StructOf)
|