Add a new custom module property 'original_id' indicating the equipment_id the modules originally belongs to. This property is only given, when distinct from the equipment_id of the SEC node. It happens when multiple config files are given, for all modules but the ones given in the first file, and for routed modules, when multiple nodes are routed or own modules are given. + fix an issue in router: additional modules were ignore in case of a single node. + small cosmetic changes in config.py reducing IDE complains Change-Id: If846c47a06158629cef807d22b91f69e4f416563 Reviewed-on: https://forge.frm2.tum.de/review/c/secop/frappy/+/35396 Tested-by: Jenkins Automated Tests <pedersen+jenkins@frm2.tum.de> Reviewed-by: Markus Zolliker <markus.zolliker@psi.ch>
977 lines
31 KiB
Python
977 lines
31 KiB
Python
# *****************************************************************************
|
|
#
|
|
# 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>
|
|
# Markus Zolliker <markus.zolliker@psi.ch>
|
|
#
|
|
# *****************************************************************************
|
|
"""test data types."""
|
|
|
|
import sys
|
|
import threading
|
|
import importlib
|
|
from glob import glob
|
|
import pytest
|
|
|
|
from frappy.datatypes import BoolType, FloatRange, StringType, IntRange, ScaledInteger
|
|
from frappy.errors import ProgrammingError, ConfigError, RangeError, HardwareError
|
|
from frappy.modules import Communicator, Drivable, Readable, Module, Writable
|
|
from frappy.params import Command, Parameter, Limit
|
|
from frappy.rwhandler import ReadHandler, WriteHandler, nopoll
|
|
from frappy.lib import generalConfig
|
|
|
|
|
|
class DispatcherStub:
|
|
# the first update from the poller comes a very short time after the
|
|
# initial value from the timestamp. However, in the test below
|
|
# the second update happens after the updates dict is cleared
|
|
# -> we have to inhibit the 'omit unchanged update' feature
|
|
|
|
def __init__(self, updates):
|
|
generalConfig.testinit(omit_unchanged_within=0)
|
|
self.updates = updates
|
|
|
|
def announce_update(self, moduleobj, pobj):
|
|
modulename = moduleobj.name
|
|
self.updates.setdefault(modulename, {})
|
|
if pobj.readerror:
|
|
self.updates[modulename]['error', pobj.name] = str(pobj.readerror)
|
|
else:
|
|
self.updates[modulename][pobj.name] = pobj.value
|
|
|
|
|
|
class LoggerStub:
|
|
def debug(self, fmt, *args):
|
|
print(fmt % args)
|
|
info = warning = exception = error = debug
|
|
handlers = []
|
|
|
|
|
|
logger = LoggerStub()
|
|
|
|
|
|
class ServerStub:
|
|
def __init__(self, updates):
|
|
self.dispatcher = DispatcherStub(updates)
|
|
self.secnode = None
|
|
|
|
|
|
class DummyMultiEvent(threading.Event):
|
|
def get_trigger(self):
|
|
|
|
def trigger(event=self):
|
|
event.set()
|
|
sys.exit()
|
|
return trigger
|
|
|
|
|
|
@pytest.mark.filterwarnings('ignore') # ignore PytestUnhandledThreadExceptionWarning
|
|
def test_Communicator():
|
|
o = Communicator('communicator', LoggerStub(), {'description': ''}, ServerStub({}))
|
|
o.earlyInit()
|
|
o.initModule()
|
|
event = DummyMultiEvent()
|
|
o.initModule()
|
|
o.startModule(event)
|
|
assert event.wait(timeout=0.1)
|
|
|
|
|
|
@pytest.mark.filterwarnings('ignore') # ignore PytestUnhandledThreadExceptionWarning
|
|
def test_ModuleMagic():
|
|
class Newclass1(Drivable):
|
|
param1 = Parameter('param1', datatype=BoolType(), default=False)
|
|
param2 = Parameter('param2', datatype=FloatRange(unit='Ohm'), default=True)
|
|
|
|
@Command(argument=BoolType(), result=BoolType())
|
|
def cmd(self, arg):
|
|
"""stuff"""
|
|
return not arg
|
|
|
|
a1 = Parameter('a1', datatype=BoolType(), default=False)
|
|
a2 = Parameter('a2', datatype=BoolType(), default=True)
|
|
value = Parameter(datatype=StringType(), default='first')
|
|
target = Parameter(datatype=StringType(), default='')
|
|
|
|
@Command(argument=BoolType(), result=BoolType())
|
|
def cmd2(self, arg):
|
|
"""another stuff"""
|
|
return not arg
|
|
|
|
def read_param1(self):
|
|
return True
|
|
|
|
def read_param2(self):
|
|
return False
|
|
|
|
def read_a1(self):
|
|
return True
|
|
|
|
@nopoll
|
|
def read_a2(self):
|
|
return True
|
|
|
|
def read_value(self):
|
|
return 'second'
|
|
|
|
def read_status(self):
|
|
return 'IDLE', 'ok'
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
class Mod1(Module): # pylint: disable=unused-variable
|
|
def do_this(self): # old style command
|
|
pass
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
class Mod2(Module): # pylint: disable=unused-variable
|
|
param = Parameter(), # pylint: disable=trailing-comma-tuple
|
|
|
|
|
|
# first inherited accessibles
|
|
sortcheck1 = ['value', 'status', 'target', 'pollinterval', 'stop',
|
|
'param1', 'param2', 'cmd', 'a1', 'a2', 'cmd2']
|
|
|
|
class Newclass2(Newclass1):
|
|
@Command(description='another stuff')
|
|
def cmd2(self, arg):
|
|
return arg
|
|
|
|
value = Parameter(datatype=FloatRange(unit='deg'))
|
|
target = Parameter(datatype=FloatRange(), default=0)
|
|
a1 = Parameter(datatype=FloatRange(unit='$/s'), readonly=False)
|
|
# remark: it might be a programming error to override the datatype
|
|
# and not overriding the read_* method. This is not checked!
|
|
b2 = Parameter('<b2>', datatype=StringType(), value='empty',
|
|
readonly=False)
|
|
|
|
def write_a1(self, value):
|
|
self._a1_written = value
|
|
return value
|
|
|
|
def write_b2(self, value):
|
|
value = value.upper()
|
|
self._b2_written = value
|
|
return value
|
|
|
|
def read_value(self):
|
|
return 0
|
|
|
|
# first predefined parameters, then in the order of inheritance
|
|
sortcheck2 = ['value', 'status', 'target', 'pollinterval', 'stop',
|
|
'param1', 'param2', 'cmd', 'a1', 'a2', 'cmd2', 'b2']
|
|
|
|
updates = {}
|
|
srv = ServerStub(updates)
|
|
|
|
params_found = set() # set of instance accessibles
|
|
objects = []
|
|
|
|
for newclass, sortcheck in [(Newclass1, sortcheck1), (Newclass2, sortcheck2)]:
|
|
o1 = newclass('o1', logger, {'description':''}, srv)
|
|
o2 = newclass('o2', logger, {'description':''}, srv)
|
|
for obj in [o1, o2]:
|
|
objects.append(obj)
|
|
for o in obj.accessibles.values():
|
|
# check that instance accessibles are unique objects
|
|
assert o not in params_found
|
|
params_found.add(o)
|
|
assert list(obj.accessibles) == sortcheck
|
|
|
|
updates.clear()
|
|
# check for inital updates working properly
|
|
o1 = Newclass1('o1', logger, {'description':''}, srv)
|
|
expectedBeforeStart = {'target': '', 'status': (Drivable.Status.IDLE, ''),
|
|
'param1': False, 'param2': 1.0, 'a1': False, 'a2': True, 'pollinterval': 5.0,
|
|
'value': 'first'}
|
|
for k, v in expectedBeforeStart.items():
|
|
assert getattr(o1, k) == v
|
|
o1.earlyInit()
|
|
event = DummyMultiEvent()
|
|
o1.initModule()
|
|
o1.startModule(event)
|
|
assert event.wait(timeout=0.1)
|
|
# should contain polled values
|
|
expectedAfterStart = {
|
|
'status': (Drivable.Status.IDLE, 'ok'), 'value': 'second',
|
|
'param1': True, 'param2': 0.0, 'a1': True}
|
|
assert updates.pop('o1') == expectedAfterStart
|
|
|
|
# check in addition if parameters are written
|
|
assert not updates
|
|
o2 = Newclass2('o2', logger, {'description':'', 'a1': {'value': 2.7}}, srv)
|
|
expectedBeforeStart.update(a1=2.7, b2='empty', target=0, value=0)
|
|
for k, v in expectedBeforeStart.items():
|
|
assert getattr(o2, k) == v
|
|
o2.earlyInit()
|
|
event = DummyMultiEvent()
|
|
o2.initModule()
|
|
o2.startModule(event)
|
|
assert event.wait(timeout=0.1)
|
|
# value has changed type, b2 and a1 are written
|
|
expectedAfterStart.update(value=0, b2='EMPTY', a1=True)
|
|
# ramerk: a1=True: this behaviour is a Porgamming error
|
|
assert updates.pop('o2') == expectedAfterStart
|
|
assert o2._a1_written == 2.7
|
|
assert o2._b2_written == 'EMPTY'
|
|
|
|
assert not updates
|
|
|
|
o1 = Newclass1('o1', logger, {'description':''}, srv)
|
|
o2 = Newclass2('o2', logger, {'description':''}, srv)
|
|
assert o2.parameters['a1'].datatype.unit == 'deg/s'
|
|
o2 = Newclass2('o2', logger, {'description':'', 'value':{'unit':'mm'},'param2':{'unit':'mm'}}, srv)
|
|
# check datatype is not shared
|
|
assert o1.parameters['param2'].datatype.unit == 'Ohm'
|
|
assert o2.parameters['param2'].datatype.unit == 'mm'
|
|
# check '$' in unit works properly
|
|
assert o2.parameters['a1'].datatype.unit == 'mm/s'
|
|
cfg = Newclass2.configurables
|
|
assert set(cfg.keys()) == {
|
|
'export', 'group', 'description', 'features',
|
|
'meaning', 'visibility', 'implementation', 'interface_classes', 'target', 'stop',
|
|
'status', 'param1', 'param2', 'cmd', 'a2', 'pollinterval', 'slowinterval', 'b2',
|
|
'cmd2', 'value', 'a1', 'omit_unchanged_within', 'original_id'}
|
|
assert set(cfg['value'].keys()) == {
|
|
'group', 'export', 'relative_resolution',
|
|
'visibility', 'unit', 'default', 'value', 'datatype', 'fmtstr',
|
|
'absolute_resolution', 'max', 'min', 'readonly', 'constant',
|
|
'description', 'needscfg', 'update_unchanged', 'influences'}
|
|
|
|
# check on the level of classes
|
|
# this checks Newclass1 too, as it is inherited by Newclass2
|
|
for baseclass in Newclass2.__mro__:
|
|
# every cmd/param has to be collected to accessibles
|
|
acs = getattr(baseclass, 'accessibles', None)
|
|
if issubclass(baseclass, Module):
|
|
assert acs is not None
|
|
else: # do not check object or mixin
|
|
acs = {}
|
|
for o in acs.values():
|
|
# check that class accessibles are not reused as instance accessibles
|
|
assert o not in params_found
|
|
|
|
for o in objects:
|
|
o.earlyInit()
|
|
for o in objects:
|
|
o.initModule()
|
|
|
|
|
|
def test_param_inheritance():
|
|
srv = ServerStub({})
|
|
|
|
class Base(Module):
|
|
param = Parameter()
|
|
|
|
class MissingDatatype(Base):
|
|
param = Parameter('param')
|
|
|
|
class MissingDescription(Base):
|
|
param = Parameter(datatype=FloatRange(), default=0)
|
|
|
|
# missing datatype and/or description of a parameter has to be detected
|
|
# at instantation and only then
|
|
with pytest.raises(ConfigError) as e_info:
|
|
MissingDatatype('o', logger, {'description': ''}, srv)
|
|
assert 'datatype' in repr(e_info.value)
|
|
|
|
with pytest.raises(ConfigError) as e_info:
|
|
MissingDescription('o', logger, {'description': ''}, srv)
|
|
assert 'description' in repr(e_info.value)
|
|
|
|
with pytest.raises(ConfigError) as e_info:
|
|
Base('o', logger, {'description': ''}, srv)
|
|
|
|
|
|
def test_command_inheritance():
|
|
class Base(Module):
|
|
@Command(BoolType(), visibility=2)
|
|
def cmd(self, arg):
|
|
"""base"""
|
|
|
|
class Sub1(Base):
|
|
@Command(group='grp')
|
|
def cmd(self, arg):
|
|
"""first"""
|
|
|
|
class Sub2(Sub1):
|
|
@Command(None, result=BoolType())
|
|
def cmd(self): # pylint: disable=arguments-differ
|
|
"""second"""
|
|
|
|
class Sub3(Base):
|
|
# when either argument or result is given, the other one is assumed to be None
|
|
# i.e. here we override the argument with None
|
|
@Command(result=FloatRange())
|
|
def cmd(self, arg):
|
|
"""third"""
|
|
|
|
assert Sub1.accessibles['cmd'].for_export() == {
|
|
'description': 'first', 'group': 'grp', 'visibility': 2,
|
|
'datainfo': {'type': 'command', 'argument': {'type': 'bool'}}
|
|
}
|
|
|
|
assert Sub2.accessibles['cmd'].for_export() == {
|
|
'description': 'second', 'group': 'grp', 'visibility': 2,
|
|
'datainfo': {'type': 'command', 'result': {'type': 'bool'}}
|
|
}
|
|
|
|
assert Sub3.accessibles['cmd'].for_export() == {
|
|
'description': 'third', 'visibility': 2,
|
|
'datainfo': {'type': 'command', 'result': {'type': 'double'}}
|
|
}
|
|
|
|
for cls in locals().values():
|
|
if hasattr(cls, 'accessibles'):
|
|
for p in cls.accessibles.values():
|
|
assert isinstance(p.ownProperties, dict)
|
|
assert p.copy().ownProperties == {}
|
|
|
|
|
|
def test_command_check():
|
|
srv = ServerStub({})
|
|
|
|
class Good(Module):
|
|
@Command(description='available')
|
|
def with_description(self):
|
|
pass
|
|
@Command()
|
|
def with_docstring(self):
|
|
"""docstring"""
|
|
|
|
Good('o', logger, {'description': ''}, srv)
|
|
|
|
class Bad1(Module):
|
|
@Command
|
|
def without_description(self):
|
|
pass
|
|
|
|
class Bad2(Module):
|
|
@Command()
|
|
def without_description(self):
|
|
pass
|
|
|
|
for cls in Bad1, Bad2:
|
|
with pytest.raises(ConfigError) as e_info:
|
|
cls('o', logger, {'description': ''}, srv)
|
|
assert 'description' in repr(e_info.value)
|
|
|
|
class BadDatatype(Module):
|
|
@Command(FloatRange(0.1, 0.9), result=FloatRange())
|
|
def cmd(self):
|
|
"""valid command"""
|
|
|
|
BadDatatype('o', logger, {'description': ''}, srv)
|
|
|
|
# test for command property checking
|
|
with pytest.raises(ProgrammingError):
|
|
BadDatatype('o', logger, {
|
|
'description': '',
|
|
'cmd': {'argument': {'type': 'double', 'min': 1, 'max': 0}},
|
|
}, srv)
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
BadDatatype('o', logger, {
|
|
'description': '',
|
|
'cmd': {'visibility': 'invalid'},
|
|
}, srv)
|
|
|
|
|
|
def test_mixin():
|
|
class Mixin: # no need to inherit from Module or HasAccessible
|
|
value = Parameter(unit='K') # missing datatype and description acceptable in mixins
|
|
param1 = Parameter('no datatype yet', fmtstr='%.5f')
|
|
param2 = Parameter('no datatype yet', default=1)
|
|
|
|
class MixedReadable(Mixin, Readable):
|
|
pass
|
|
|
|
class MixedDrivable(MixedReadable, Drivable):
|
|
value = Parameter(unit='Ohm', fmtstr='%.3f')
|
|
param1 = Parameter(datatype=FloatRange())
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
class MixedModule(Mixin): # pylint: disable=unused-variable
|
|
param1 = Parameter('', FloatRange(), fmtstr=0) # fmtstr must be a string
|
|
|
|
assert repr(MixedDrivable.status.datatype) == repr(Drivable.status.datatype)
|
|
assert repr(MixedReadable.status.datatype) == repr(Readable.status.datatype)
|
|
assert MixedReadable.value.datatype.unit == 'K'
|
|
assert MixedDrivable.value.datatype.unit == 'Ohm'
|
|
assert MixedDrivable.value.datatype.fmtstr == '%.3f'
|
|
# when datatype is overridden, fmtstr falls back to default:
|
|
assert MixedDrivable.param1.datatype.fmtstr == '%g'
|
|
|
|
srv = ServerStub({})
|
|
|
|
MixedDrivable('o', logger, {
|
|
'description': '',
|
|
'param1': {'value': 0, 'description': 'param1'},
|
|
'param2': {'datatype': {"type": "double"}},
|
|
}, srv)
|
|
|
|
with pytest.raises(ConfigError):
|
|
MixedReadable('o', logger, {
|
|
'description': '',
|
|
'param1': {'value': 0, 'description': 'param1'},
|
|
'param2': {'datatype': {"type": "double"}},
|
|
}, srv)
|
|
|
|
|
|
def test_override():
|
|
class Mod(Drivable):
|
|
value = 5 # overriding the default value
|
|
|
|
def stop(self):
|
|
"""no decorator needed"""
|
|
|
|
assert Mod.value.value == 5
|
|
assert Mod.stop.description == "no decorator needed"
|
|
|
|
class Mod2(Mod):
|
|
def stop(self):
|
|
pass
|
|
|
|
# inherit doc string
|
|
assert Mod2.stop.description == Mod.stop.description
|
|
|
|
|
|
def test_command_config():
|
|
class Mod(Module):
|
|
@Command(IntRange(0, 1), result=IntRange(0, 1))
|
|
def convert(self, value):
|
|
"""dummy conversion"""
|
|
return value
|
|
|
|
srv = ServerStub({})
|
|
mod = Mod('o', logger, {
|
|
'description': '',
|
|
'convert': {'argument': {'type': 'bool'}},
|
|
}, srv)
|
|
assert mod.commands['convert'].datatype.export_datatype() == {
|
|
'type': 'command',
|
|
'argument': {'type': 'bool'},
|
|
'result': {'type': 'int', 'min': 0, 'max': 1},
|
|
}
|
|
|
|
mod = Mod('o', logger, {
|
|
'description': '',
|
|
'convert': {'datatype': {'type': 'command', 'argument': {'type': 'bool'}, 'result': {'type': 'bool'}}},
|
|
}, srv)
|
|
assert mod.commands['convert'].datatype.export_datatype() == {
|
|
'type': 'command',
|
|
'argument': {'type': 'bool'},
|
|
'result': {'type': 'bool'},
|
|
}
|
|
|
|
|
|
def test_command_none():
|
|
srv = ServerStub({})
|
|
|
|
class Mod(Drivable):
|
|
pass
|
|
|
|
class Mod2(Drivable):
|
|
stop = None
|
|
|
|
assert 'stop' in Mod('o', logger, {'description': ''}, srv).accessibles
|
|
assert 'stop' not in Mod2('o', logger, {'description': ''}, srv).accessibles
|
|
|
|
|
|
def test_bad_method():
|
|
class Mod0(Drivable): # pylint: disable=unused-variable
|
|
def write_target(self, value):
|
|
pass
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
class Mod1(Drivable): # pylint: disable=unused-variable
|
|
def write_taget(self, value):
|
|
pass
|
|
|
|
class Mod2(Drivable): # pylint: disable=unused-variable
|
|
def read_value(self, value):
|
|
pass
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
class Mod3(Drivable): # pylint: disable=unused-variable
|
|
def read_valu(self, value):
|
|
pass
|
|
|
|
|
|
def test_generic_access():
|
|
class Mod(Module):
|
|
param = Parameter('handled param', StringType(), readonly=False)
|
|
unhandled = Parameter('unhandled param', StringType(), default='', readonly=False)
|
|
data = {'param': ''}
|
|
|
|
@ReadHandler(['param'])
|
|
def read_handler(self, pname):
|
|
value = self.data[pname]
|
|
setattr(self, pname, value)
|
|
return value
|
|
|
|
@WriteHandler(['param'])
|
|
def write_handler(self, pname, value):
|
|
value = value.lower()
|
|
self.data[pname] = value
|
|
setattr(self, pname, value)
|
|
return value
|
|
|
|
updates = {}
|
|
srv = ServerStub(updates)
|
|
|
|
obj = Mod('obj', logger, {'description': '', 'param': {'value':'initial value'}}, srv)
|
|
assert obj.param == 'initial value'
|
|
assert obj.write_param('Cheese') == 'cheese'
|
|
assert obj.write_unhandled('Cheese') == 'Cheese'
|
|
assert updates == {'obj': {'param': 'cheese', 'unhandled': 'Cheese'}}
|
|
updates.clear()
|
|
assert obj.write_param('Potato') == 'potato'
|
|
assert updates == {'obj': {'param': 'potato'}}
|
|
updates.clear()
|
|
assert obj.read_param() == 'potato'
|
|
assert obj.read_unhandled()
|
|
assert updates == {'obj': {'param': 'potato'}}
|
|
updates.clear()
|
|
assert not updates
|
|
|
|
|
|
def test_duplicate_handler_name():
|
|
with pytest.raises(ProgrammingError):
|
|
class Mod(Module): # pylint: disable=unused-variable
|
|
param = Parameter('handled param', StringType(), readonly=False)
|
|
|
|
@ReadHandler(['param'])
|
|
def handler(self, pname):
|
|
pass
|
|
|
|
@WriteHandler(['param'])
|
|
def handler(self, pname, value): # pylint: disable=function-redefined
|
|
pass
|
|
|
|
|
|
def test_handler_overwrites_method():
|
|
with pytest.raises(RuntimeError):
|
|
class Mod1(Module): # pylint: disable=unused-variable
|
|
param = Parameter('handled param', StringType(), readonly=False)
|
|
|
|
@ReadHandler(['param'])
|
|
def read_handler(self, pname):
|
|
pass
|
|
|
|
def read_param(self):
|
|
pass
|
|
|
|
with pytest.raises(RuntimeError):
|
|
class Mod2(Module): # pylint: disable=unused-variable
|
|
param = Parameter('handled param', StringType(), readonly=False)
|
|
|
|
@WriteHandler(['param'])
|
|
def write_handler(self, pname, value):
|
|
pass
|
|
|
|
def write_param(self, value):
|
|
pass
|
|
|
|
|
|
def test_no_read_write():
|
|
class Mod(Module):
|
|
param = Parameter('test param', StringType(), readonly=False)
|
|
|
|
updates = {}
|
|
srv = ServerStub(updates)
|
|
|
|
obj = Mod('obj', logger, {'description': '', 'param': {'value': 'cheese'}}, srv)
|
|
assert obj.param == 'cheese'
|
|
assert obj.read_param() == 'cheese'
|
|
assert updates == {'obj': {'param': 'cheese'}}
|
|
assert obj.write_param('egg') == 'egg'
|
|
assert obj.param == 'egg'
|
|
assert updates == {'obj': {'param': 'egg'}}
|
|
|
|
|
|
def test_incompatible_value_target():
|
|
class Mod1(Drivable):
|
|
value = Parameter('', FloatRange(0, 10), default=0)
|
|
target = Parameter('', FloatRange(0, 11), default=0)
|
|
|
|
class Mod2(Drivable):
|
|
value = Parameter('', FloatRange(), default=0)
|
|
target = Parameter('', StringType(), default='')
|
|
|
|
class Mod3(Drivable):
|
|
value = Parameter('', FloatRange(), default=0)
|
|
target = Parameter('', ScaledInteger(1, 0, 10), default=0)
|
|
|
|
srv = ServerStub({})
|
|
|
|
with pytest.raises(ConfigError):
|
|
obj = Mod1('obj', logger, {'description': ''}, srv) # pylint: disable=unused-variable
|
|
|
|
with pytest.raises(ProgrammingError):
|
|
obj = Mod2('obj', logger, {'description': ''}, srv)
|
|
|
|
obj = Mod3('obj', logger, {'description': ''}, srv)
|
|
|
|
|
|
def test_problematic_value_range():
|
|
class Mod(Drivable):
|
|
value = Parameter('', FloatRange(0, 10), default=0)
|
|
target = Parameter('', FloatRange(0, 10), default=0)
|
|
|
|
srv = ServerStub({})
|
|
|
|
obj = Mod('obj', logger, {'description': '', 'value':{'max': 10.1}}, srv) # pylint: disable=unused-variable
|
|
|
|
with pytest.raises(ConfigError):
|
|
obj = Mod('obj', logger, {'description': '', 'value.max': 9.9}, srv)
|
|
|
|
class Mod2(Drivable):
|
|
value = Parameter('', FloatRange(), default=0)
|
|
target = Parameter('', FloatRange(), default=0)
|
|
|
|
obj = Mod2('obj', logger, {'description': ''}, srv)
|
|
obj = Mod2('obj', logger, {'description': '', 'target':{'min': 0, 'max': 10}}, srv)
|
|
|
|
obj = Mod('obj', logger, {
|
|
'value': {'min': 0, 'max': 10},
|
|
'target': {'min': 0, 'max': 10}, 'description': ''}, srv)
|
|
|
|
class Mod4(Drivable):
|
|
value = Parameter('', FloatRange(0, 10), default=0)
|
|
target = Parameter('', FloatRange(0, 10), default=0)
|
|
obj = Mod4('obj', logger, {
|
|
'value': {'min': 0, 'max': 10},
|
|
'target': {'min': 0, 'max': 10}, 'description': ''}, srv)
|
|
|
|
|
|
@pytest.mark.parametrize('config, dynamicunit, finalunit, someunit', [
|
|
({}, 'K', 'K', 'K'),
|
|
({'value':{'unit': 'K'}}, 'C', 'C', 'C'),
|
|
({'value':{'unit': 'K'}}, '', 'K', 'K'),
|
|
({'value':{'unit': 'K'}, 'someparam':{'unit': 'A'}}, 'C', 'C', 'A'),
|
|
])
|
|
def test_deferred_main_unit(config, dynamicunit, finalunit, someunit):
|
|
# this pattern is used in frappy_mlz.entangle.AnalogInput
|
|
class Mod(Drivable):
|
|
ramp = Parameter('', datatype=FloatRange(unit='$/min'))
|
|
someparam = Parameter('', datatype=FloatRange(unit='$'))
|
|
__main_unit = None
|
|
|
|
def applyMainUnit(self, mainunit):
|
|
# called from __init__ method
|
|
# replacement of '$' by main unit must be done later
|
|
self.__main_unit = mainunit
|
|
|
|
def startModule(self, start_events):
|
|
super().startModule(start_events)
|
|
if dynamicunit:
|
|
self.accessibles['value'].datatype.setProperty('unit', dynamicunit)
|
|
self.__main_unit = dynamicunit
|
|
if self.__main_unit:
|
|
super().applyMainUnit(self.__main_unit)
|
|
|
|
srv = ServerStub({})
|
|
m = Mod('m', logger, {'description': '', **config}, srv)
|
|
m.startModule(None)
|
|
assert m.parameters['value'].datatype.unit == finalunit
|
|
assert m.parameters['target'].datatype.unit == finalunit
|
|
assert m.parameters['ramp'].datatype.unit == finalunit + '/min'
|
|
# when someparam.unit is configured, this differs from finalunit
|
|
assert m.parameters['someparam'].datatype.unit == someunit
|
|
|
|
|
|
def test_super_call():
|
|
class Base(Readable):
|
|
def read_status(self):
|
|
return Readable.Status.IDLE, 'base'
|
|
|
|
class Mod(Base):
|
|
def read_status(self):
|
|
code, text = super().read_status()
|
|
return code, text + ' (extended)'
|
|
|
|
class DispatcherStub1:
|
|
def __init__(self, updates):
|
|
self.updates = updates
|
|
|
|
def announce_update(self, moduleobj, pobj):
|
|
if pobj.readerror:
|
|
raise pobj.readerror
|
|
self.updates.append((moduleobj.name, pobj.name, pobj.value))
|
|
|
|
class ServerStub1:
|
|
def __init__(self, updates):
|
|
self.dispatcher = DispatcherStub1(updates)
|
|
self.secnode = None
|
|
|
|
updates = []
|
|
srv = ServerStub1(updates)
|
|
b = Base('b', logger, {'description': ''}, srv)
|
|
b.read_status()
|
|
assert updates == [('b', 'status', ('IDLE', 'base'))]
|
|
|
|
updates.clear()
|
|
m = Mod('m', logger, {'description': ''}, srv)
|
|
m.read_status()
|
|
# in the version before change 'allow super calls on read_/write_ methods'
|
|
# updates would contain two items
|
|
assert updates == [('m', 'status', ('IDLE', 'base (extended)'))]
|
|
|
|
assert type(m).__name__ == '_Mod'
|
|
assert type(m).__mro__[1:5] == (Mod, Base, Readable, Module)
|
|
|
|
|
|
def test_write_method_returns_none():
|
|
class Mod(Module):
|
|
a = Parameter('', FloatRange(), readonly=False)
|
|
|
|
def write_a(self, value):
|
|
return None
|
|
|
|
mod = Mod('mod', LoggerStub(), {'description': ''}, ServerStub({}))
|
|
mod.write_a(1.5)
|
|
assert mod.a == 1.5
|
|
|
|
|
|
@pytest.mark.parametrize('arg, value', [
|
|
('always', 0),
|
|
(0, 0),
|
|
('never', 999999999),
|
|
(999999999, 999999999),
|
|
('default', 0.25),
|
|
(1, 1),
|
|
])
|
|
def test_update_unchanged_ok(arg, value):
|
|
srv = ServerStub({})
|
|
generalConfig.testinit(omit_unchanged_within=0.25) # override value from DispatcherStub
|
|
|
|
class Mod1(Module):
|
|
a = Parameter('', FloatRange(), default=0, update_unchanged=arg)
|
|
|
|
mod1 = Mod1('mod1', LoggerStub(), {'description': ''}, srv)
|
|
par = mod1.parameters['a']
|
|
assert par.omit_unchanged_within == value
|
|
assert Mod1.a.omit_unchanged_within == 0
|
|
|
|
class Mod2(Module):
|
|
a = Parameter('', FloatRange(), default=0)
|
|
|
|
mod2 = Mod2('mod2', LoggerStub(), {'description': '', 'a': {'update_unchanged': arg}}, srv)
|
|
par = mod2.parameters['a']
|
|
assert par.omit_unchanged_within == value
|
|
assert Mod2.a.omit_unchanged_within == 0
|
|
|
|
|
|
def test_omit_unchanged_within():
|
|
srv = ServerStub({})
|
|
generalConfig.testinit(omit_unchanged_within=0.25) # override call from DispatcherStub
|
|
|
|
class Mod(Module):
|
|
a = Parameter('', FloatRange())
|
|
|
|
mod1 = Mod('mod1', LoggerStub(), {'description': ''}, srv)
|
|
assert mod1.parameters['a'].omit_unchanged_within == 0.25
|
|
|
|
mod2 = Mod('mod2', LoggerStub(), {'description': '', 'omit_unchanged_within': 0.125}, srv)
|
|
assert mod2.parameters['a'].omit_unchanged_within == 0.125
|
|
|
|
|
|
stdlim = {
|
|
'a_min': -1, 'a_max': 2,
|
|
'b_min': 0,
|
|
'c_max': 10,
|
|
'd_limits': (-1, 1),
|
|
}
|
|
|
|
|
|
class Lim(Module):
|
|
a = Parameter('', FloatRange(-10, 10), readonly=False, default=0)
|
|
a_min = Limit()
|
|
a_max = Limit()
|
|
|
|
b = Parameter('', FloatRange(0, None), readonly=False, default=0)
|
|
b_min = Limit()
|
|
|
|
c = Parameter('', IntRange(None, 100), readonly=False, default=0)
|
|
c_max = Limit()
|
|
|
|
d = Parameter('', FloatRange(-5, 5), readonly=False, default=0)
|
|
d_limits = Limit()
|
|
|
|
e = Parameter('', IntRange(0, 8), readonly=False, default=0)
|
|
|
|
def check_e(self, value):
|
|
if value % 2:
|
|
raise RangeError('e must not be odd')
|
|
|
|
|
|
def test_limit_defaults():
|
|
|
|
srv = ServerStub({})
|
|
|
|
mod = Lim('mod', LoggerStub(), {'description': 'test'}, srv)
|
|
|
|
assert mod.a_min == -10
|
|
assert mod.a_max == 10
|
|
assert isinstance(mod.a_min, float)
|
|
assert isinstance(mod.a_max, float)
|
|
|
|
assert mod.b_min == 0
|
|
assert isinstance(mod.b_min, float)
|
|
|
|
assert mod.c_max == 100
|
|
assert isinstance(mod.c_max, int)
|
|
|
|
assert mod.d_limits == (-5, 5)
|
|
assert isinstance(mod.d_limits[0], float)
|
|
assert isinstance(mod.d_limits[1], float)
|
|
|
|
|
|
@pytest.mark.parametrize('limits, pname, good, bad', [
|
|
(stdlim, 'a', [-1, 2, 0], [-2, 3]),
|
|
(stdlim, 'b', [0, 1e99], [-1, -1e99]),
|
|
(stdlim, 'c', [-999, 0, 10], [11, 999]),
|
|
(stdlim, 'd', [-1, 0.1, 1], [-1.001, 1.001]),
|
|
({'a_min': 0, 'a_max': -1}, 'a', [], [0, -1]),
|
|
(stdlim, 'e', [0, 2, 4, 6, 8], [-1, 1, 7, 9]),
|
|
])
|
|
def test_limits(limits, pname, good, bad):
|
|
|
|
srv = ServerStub({})
|
|
|
|
mod = Lim('mod', LoggerStub(), {'description': 'test'}, srv)
|
|
mod.check_a = 0 # this should not harm. check_a is never called on the instance
|
|
|
|
for k, v in limits.items():
|
|
setattr(mod, k, v)
|
|
|
|
for v in good:
|
|
getattr(mod, 'write_' + pname)(v)
|
|
for v in bad:
|
|
with pytest.raises(RangeError):
|
|
getattr(mod, 'write_' + pname)(v)
|
|
|
|
|
|
def test_limit_inheritance():
|
|
srv = ServerStub({})
|
|
|
|
class Base(Module):
|
|
a = Parameter('', FloatRange(), readonly=False, default=0)
|
|
|
|
def check_a(self, value):
|
|
if int(value * 4) != value * 4:
|
|
raise ValueError('value is not a multiple of 0.25')
|
|
|
|
class Mixin:
|
|
a_min = Limit()
|
|
a_max = Limit()
|
|
|
|
class Mod(Mixin, Base):
|
|
def check_a(self, value):
|
|
if value == 0:
|
|
raise ValueError('value must not be 0')
|
|
|
|
mod = Mod('mod', LoggerStub(), {'description': 'test', 'a_min': {'value': -1}, 'a_max': {'value': 1}}, srv)
|
|
|
|
for good in [-1, -0.75, 0.25, 1]:
|
|
mod.write_a(good)
|
|
|
|
for bad in [-2, -0.1, 0, 0.9, 1.1]:
|
|
with pytest.raises(ValueError):
|
|
mod.write_a(bad)
|
|
|
|
class Mod2(Mixin, Base):
|
|
def check_a(self, value):
|
|
if value == 0:
|
|
raise ValueError('value must not be 0')
|
|
return True # indicates stop checking
|
|
|
|
mod2 = Mod2('mod2', LoggerStub(), {'description': 'test', 'a_min': {'value': -1}, 'a_max': {'value': 1}}, srv)
|
|
|
|
for good in [-2, -1, -0.75, 0.25, 1, 1.1]:
|
|
mod2.write_a(good)
|
|
|
|
with pytest.raises(ValueError):
|
|
mod2.write_a(0)
|
|
|
|
@pytest.mark.parametrize('bases, iface_classes', [
|
|
([Module], ()),
|
|
([Communicator], ('Communicator',)),
|
|
([Readable], ('Readable',)),
|
|
([Writable], ('Writable',)),
|
|
([Drivable], ('Drivable',)),
|
|
])
|
|
def test_interface_classes(bases, iface_classes):
|
|
srv = ServerStub({})
|
|
class Mod(*bases):
|
|
pass
|
|
m = Mod('mod', LoggerStub(), {'description': 'test'}, srv)
|
|
assert m.interface_classes == iface_classes
|
|
|
|
|
|
all_drivables = set()
|
|
for pyfile in glob('frappy_*/*.py'):
|
|
module = pyfile[:-3].replace('/', '.')
|
|
try:
|
|
importlib.import_module(module)
|
|
except Exception as e:
|
|
print(module, e)
|
|
continue
|
|
for obj_ in sys.modules[module].__dict__.values():
|
|
if isinstance(obj_, type) and issubclass(obj_, Drivable):
|
|
all_drivables.add(obj_)
|
|
|
|
|
|
@pytest.mark.parametrize('modcls', all_drivables)
|
|
def test_stop_doc(modcls):
|
|
# make sure that implemented stop methods have a doc string
|
|
if (modcls.stop.description == Drivable.stop.description
|
|
and modcls.stop.func != Drivable.stop.func):
|
|
assert modcls.stop.func.__doc__ # stop method needs a doc string
|
|
|
|
|
|
def test_write_error():
|
|
updates = {}
|
|
srv = ServerStub(updates)
|
|
|
|
class Mod(Module):
|
|
par = Parameter('', FloatRange(), readonly=False, default=0)
|
|
|
|
def read_par(self):
|
|
raise HardwareError('failure')
|
|
|
|
def write_par(self, value):
|
|
if value < 0:
|
|
raise RangeError('outside range')
|
|
|
|
a = Mod('a', LoggerStub(), {'description': 'test'}, srv)
|
|
|
|
# behaviour on read errors:
|
|
with pytest.raises(HardwareError):
|
|
a.read_par()
|
|
assert updates == {'a': {('error', 'par'): 'failure'}}
|
|
updates.clear()
|
|
|
|
# behaviour on normal write
|
|
a.write_par(1)
|
|
assert updates['a']['par'] == 1
|
|
updates.clear()
|
|
|
|
# behaviour when write failed
|
|
with pytest.raises(RangeError):
|
|
a.write_par(-1)
|
|
assert not updates # no error update!
|