Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

TernaryDevice and ArrayDevice #1119

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
61 changes: 61 additions & 0 deletions ophyd/array.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
from functools import reduce
from ophyd import Device, DeviceStatus


def ArrayDevice(devices, *args, **kwargs):
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Using a function here makes it so that it can't be subclassed. This should be updated to fix that.

"""
A function, that behaves like a class init, that dynamically creates an
ArrayDevice class. This is needed to set class attributes before the init.
Adding devices in the init can subvert important ophyd code that
manages sub devices.

Parameters
----------
devices: interable
An iterable of devices with the same type.

Example
-------
array_device = ArrayDevice([ExampleTernary(i) for i in range(10)], name='array_device')
"""

class _ArrayDeviceBase(Device):
"""
An ophyd.Device that is an array of devices.

The set method takes a list of values.
the get method returns a list of values.
Parameters
----------
devices: iterable
The array of ophyd devices.
"""
def set(self, values):
if len(values) != len(self.devices):
raise ValueError(
f"The number of values ({len(values)}) must match "
f"the number of devices ({len(self.devices)})"
)

# If the device already has the requested state, return a finished status.
diff = [self.devices[i].get() != value for i, value in enumerate(values)]
if not any(diff):
return DeviceStatus(self)._finished()

# Set the value of each device and return a union of the statuses.
statuses = [self.devices[i].set(value) for i, value in enumerate(values)]
st = reduce(lambda a, b: a & b, statuses)
return st

def reset(self):
self.set([0 for i in range(len(self.devices))])

def get(self):
return [device.get() for device in self.devices]

types = {type(device) for device in devices}
if len(types) != 1:
raise TypeError("All devices must have the same type")

_ArrayDevice = type('ArrayDevice', (_ArrayDeviceBase,), {'devices': devices})
return _ArrayDevice(*args, **kwargs)
86 changes: 86 additions & 0 deletions ophyd/ternary.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
from ophyd import Device, DeviceStatus, EpicsSignal, EpicsSignalRO, FormattedComponent


class TernaryDevice(Device):
"""
A general purpose ophyd device with set and reset signals, and a state signal
with 3 posible signals.

Example
-------
class StateEnum(Enum):
In = True
Out = False
Unknown = None

class ExampleTernary(TernaryDevice):
def __init__(self, index, *args, **kwargs):
super().__init__(
*args,
name=f"Filter{index}",
set_name=f"TernaryArray:device{index}_set",
reset_name=f"TernaryArray:device{index}_reset",
state_name=f"TernaryArray:device{index}_rbv",
state_enum=StateEnum,
**kwargs,
)
ternary1 = ExampleTernary(1)
"""

set_cmd = FormattedComponent(EpicsSignal, "{self._set_name}")
reset_cmd = FormattedComponent(EpicsSignal, "{self._reset_name}")
state_rbv = FormattedComponent(EpicsSignalRO, "{self._state_name}", string=True)

def __init__(
self, *args, set_name, reset_name, state_name, state_enum, **kwargs
) -> None:
self._state_enum = state_enum
self._set_name = set_name
self._reset_name = reset_name
self._state_name = state_name
self._state = None
super().__init__(*args, **kwargs)

def set(self, value=True):
if value not in {True, False, 0, 1}:
raise ValueError("value must be one of the following: True, False, 0, 1")

target_value = bool(value)

st = DeviceStatus(self)

# If the device already has the requested state, return a finished status.
if self._state == bool(value):
st._finished()
return st
self._set_st = st

def state_cb(value, timestamp, **kwargs):
"""
Updates self._state and checks if the status should be marked as finished.
"""
try:
self._state = self._state_enum[value].value
except KeyError:
raise ValueError(f"self._state_enum does not contain value: {value}")
if self._state == target_value:
self._set_st = None
self.state_rbv.clear_sub(state_cb)
st._finished()

# Subscribe the callback to the readback signal.
# The callback will be called each time the PV value changes.
self.state_rbv.subscribe(state_cb)

# Write to the signal.
if value:
self.set_cmd.set(1)
else:
self.reset_cmd.set(1)
return st

def reset(self):
self.set(False)

def get(self):
return self._state
Binary file added ophyd/tests/.coverage
Binary file not shown.
175 changes: 175 additions & 0 deletions ophyd/tests/test_array.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,175 @@
import asyncio
import subprocess
import sys
import time

from enum import Enum
from functools import partial, reduce
from collections import OrderedDict
from caproto.server import PVGroup, ioc_arg_parser, pvproperty, run
from ophyd import Device, DeviceStatus, EpicsSignal, EpicsSignalRO, FormattedComponent
from ophyd.array import ArrayDevice
from ophyd.ternary import TernaryDevice


class StateEnum(Enum):
In = True
Out = False
Unknown = None


class TernaryDeviceSim:
"""
A device with three states.

Parameters
----------
delay: float, optional
The time it takes for the device to change from state-0 to state-1.
"""

def __init__(self, delay=0.5):
self._delay = delay
self._state = False

async def set(self):
if not self._state:
self._state = None
await asyncio.sleep(self._delay)
self._state = True

async def reset(self):
if self._state or self._state is None:
self._state = None
await asyncio.sleep(self._delay)
self._state = False

@property
def state(self):
return self._state


class TernaryArrayIOC(PVGroup):
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I plan to update this to use type to create the class.

"""
Example IOC that has an array of TernaryDevices.

Parameters
----------
count: integer
The number of devices in the array.
"""

def __init__(self, count=10, *args, **kwargs):
self._devices = [TernaryDeviceSim() for i in range(count)]

# Dynamically setup the pvs.
for i in range(count):
# Create the set pv.
setattr(
self,
f"device{i}_set",
pvproperty(value=0, dtype=int, name=f"device{i}_set"),
)

# Create the set putter.
partial_set = partial(self.set_putter, i)
partial_set.__name__ = f"set_putter{i}"
getattr(self, f"device{i}_set").putter(partial_set)

# Create the reset pv.
setattr(
self,
f"device{i}_reset",
pvproperty(value=0, dtype=int, name=f"device{i}_reset"),
)

# Create the reset putter.
partial_reset = partial(self.reset_putter, i)
partial_reset.__name__ = f"reset_putter{i}"
getattr(self, f"device{i}_reset").putter(partial_reset)

# Create the readback pv.
setattr(
self,
f"device{i}_rbv",
pvproperty(value="Unknown", dtype=str, name=f"device{i}_rbv"),
)

# Create the readback scan.
partial_scan = partial(self.general_scan, i)
partial_scan.__name__ = f"scan{i}"
getattr(self, f"device{i}_rbv").scan(period=0.1)(partial_scan)

# Unfortunate hack to register the late pvs.
self.__dict__["_pvs_"] = OrderedDict(PVGroup.find_pvproperties(self.__dict__))
super().__init__(*args, **kwargs)

async def set_putter(self, index, group, instance, value):
if value:
await self._devices[index].set()

async def reset_putter(self, index, group, instance, value):
if value:
await self._devices[index].reset()

async def general_scan(self, index, group, instance, async_lib):
# A hacky way to write to the pv.
await self.pvdb[f"{self.prefix}device{index}_rbv"].write(
StateEnum(self._devices[index].state).name
)
# This is the normal way to do this, but it doesn't work correctly for this example.
# await getattr(self, f'device{index}_rbv').write(StateEnum(self._devices[index].state).name)


class ExampleTernary(TernaryDevice):
"""
This class is an example about how to create a TernaryDevice specialization
for a specific implementation.
"""

def __init__(self, index, *args, **kwargs):
super().__init__(
*args,
name=f"Filter{index}",
set_name=f"TernaryArray:device{index}_set",
reset_name=f"TernaryArray:device{index}_reset",
state_name=f"TernaryArray:device{index}_rbv",
state_enum=StateEnum,
**kwargs,
)


array_device = ArrayDevice([ExampleTernary(i) for i in range(10)], name='array_device')


def start_test_ioc():
ioc = TernaryArrayIOC(prefix='TernaryArray:')
print("Prefix =", "TernaryArray:")
print("PVs:", list(ioc.pvdb))
run(ioc.pvdb)


def ioc(f):
"""
Decorator that starts a test ioc using subproccess,
calls your function and then cleans up the process.
"""
def wrap():
try:
ps = subprocess.Popen([sys.executable, '-c', 'from ophyd.tests.test_array import start_test_ioc; start_test_ioc()'])
time.sleep(5)
f()
finally:
ps.kill()
return wrap


@ioc
def test_arraydevice():
arraydevice = ArrayDevice([ExampleTernary(i) for i in range(10)],
name='arraydevice')
values = [1,1,1,0,0,0,1,1,1,0]
arraydevice.set(values)
time.sleep(1)
print(arraydevice.get())
assert arraydevice.get() == values