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

Split out common AD file plugin logic into core writer class, create ADTiffWriter #606

Merged
merged 45 commits into from
Jan 6, 2025
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
652de13
Starting to work on ad tiff writer
jwlodek Sep 4, 2024
e289ee4
Resolve merge conflicts
jwlodek Oct 1, 2024
f36ec3a
Continue working on tiff writer
jwlodek Oct 8, 2024
83dff62
Further work on tiff writer, existing tests now passing.
jwlodek Oct 8, 2024
1a52a21
Remove functions moved to superclas from hdf writer
jwlodek Oct 8, 2024
489cfd8
Significant re-org and simplification of ad classes
jwlodek Oct 9, 2024
83c6884
Ruff formatting
jwlodek Oct 9, 2024
3b4f45a
Modify ad sim classes to reflect new superclasses
jwlodek Oct 9, 2024
7175b30
Modify vimba and kinetix classes
jwlodek Oct 9, 2024
faf53d6
Modify aravis and pilatus classes
jwlodek Oct 9, 2024
5b9f60f
Update all tests to make sure they still pass with changes
jwlodek Oct 10, 2024
8bbfd0e
Some cleanup
jwlodek Oct 10, 2024
1eab818
Merge with upstream
jwlodek Oct 10, 2024
f6825b4
Changes to standard detector to account for controller/writer types i…
jwlodek Nov 22, 2024
651b80d
Significant changes to base detector, controller, and writer classes
jwlodek Nov 22, 2024
38a61e8
Update detector and controller classes to reflect changes
jwlodek Nov 22, 2024
aecdf04
Make sure panda standard det uses new type hints
jwlodek Nov 22, 2024
e42fa12
Most tests passing
jwlodek Nov 22, 2024
07684a4
Merge with main and resolve conflicts
jwlodek Nov 22, 2024
6dc09f3
Revert change in test that was resolved by pydantic version update
jwlodek Nov 22, 2024
1f7dcd7
Remove debugging prints
jwlodek Nov 22, 2024
35dd1b1
Linter fixes
jwlodek Nov 22, 2024
8112220
Fix linter error
jwlodek Nov 22, 2024
ac1e509
Move creation of writer outside of base AreaDetector class init per r…
jwlodek Nov 26, 2024
8494da4
Make sure we don't wait for capture to be done!
jwlodek Nov 26, 2024
b212432
Merge with upstream
jwlodek Nov 26, 2024
3242d45
Merge with upstream
jwlodek Dec 9, 2024
488d7eb
Allow for specifying whether or not to use fileio signals for dataset…
jwlodek Dec 9, 2024
a76b70f
Revert "Allow for specifying whether or not to use fileio signals for…
jwlodek Dec 10, 2024
7da935e
Fix linter errors, remove unused enum
jwlodek Dec 10, 2024
0cd0ddf
Change from return to await to conform to return type
jwlodek Dec 10, 2024
f1b9a4e
Apply more suggestions from review
jwlodek Dec 18, 2024
3c2b479
Merge with upstream
jwlodek Dec 18, 2024
d84f2b4
Replace instances of DeviceCollector to init_devices
jwlodek Dec 18, 2024
05dd89c
Update src/ophyd_async/epics/adaravis/_aravis_controller.py
jwlodek Dec 20, 2024
3fcd541
Update src/ophyd_async/epics/adaravis/_aravis_controller.py
jwlodek Dec 20, 2024
87d5fdd
Update src/ophyd_async/epics/adcore/_core_writer.py
jwlodek Dec 20, 2024
52d712e
Update src/ophyd_async/epics/adcore/_core_logic.py
jwlodek Dec 20, 2024
e46cbd4
Update src/ophyd_async/epics/adcore/_core_detector.py
jwlodek Dec 20, 2024
fbb895e
Update src/ophyd_async/epics/adcore/_core_detector.py
jwlodek Dec 20, 2024
6cd79a6
Update src/ophyd_async/epics/adcore/_core_writer.py
jwlodek Dec 20, 2024
05e6fbd
Fix all tests aside from 3.12 unawaited coro after applying suggestions
jwlodek Dec 23, 2024
a616fd2
Resolve unawaited coro error on 3.12
jwlodek Dec 23, 2024
dceb534
Remove unused typevar
jwlodek Jan 6, 2025
d065f9c
Merge with upstream
jwlodek Jan 6, 2025
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
2 changes: 2 additions & 0 deletions src/ophyd_async/core/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
from ._detector import (
DetectorController,
DetectorControllerT,
DetectorTrigger,
DetectorWriter,
StandardDetector,
Expand Down Expand Up @@ -100,6 +101,7 @@

__all__ = [
"DetectorController",
"DetectorControllerT",
"DetectorTrigger",
"DetectorWriter",
"StandardDetector",
Expand Down
67 changes: 40 additions & 27 deletions src/ophyd_async/core/_detector.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,10 +5,16 @@
from abc import ABC, abstractmethod
from collections.abc import AsyncGenerator, AsyncIterator, Callable, Iterator, Sequence
from functools import cached_property
from typing import (
Any,
Generic,
TypeVar,
)

from bluesky.protocols import (
Collectable,
Flyable,
Hints,
Preparable,
Reading,
Stageable,
Expand Down Expand Up @@ -87,7 +93,7 @@ def get_deadtime(self, exposure: float | None) -> float:
"""For a given exposure, how long should the time between exposures be"""

@abstractmethod
async def prepare(self, trigger_info: TriggerInfo):
async def prepare(self, trigger_info: TriggerInfo) -> Any:
jwlodek marked this conversation as resolved.
Show resolved Hide resolved
"""
Do all necessary steps to prepare the detector for triggers.

Expand Down Expand Up @@ -157,6 +163,16 @@ def collect_stream_docs(self, indices_written: int) -> AsyncIterator[StreamAsset
async def close(self) -> None:
"""Close writer, blocks until I/O is complete"""

@property
def hints(self) -> Hints:
return {}


# Add type vars for controller/type, so we can define
# StandardDetector[KinetixController, ADTIFFWriter] for example
DetectorControllerT = TypeVar("DetectorControllerT", bound=DetectorController)
DetectorWriterT = TypeVar("DetectorWriterT", bound=DetectorWriter)


class StandardDetector(
Device,
Expand All @@ -168,6 +184,7 @@ class StandardDetector(
Flyable,
Collectable,
WritesStreamAssets,
Generic[DetectorControllerT, DetectorWriterT],
):
"""
Useful detector base class for step and fly scanning detectors.
Expand All @@ -176,8 +193,8 @@ class StandardDetector(

def __init__(
self,
controller: DetectorController,
writer: DetectorWriter,
controller: DetectorControllerT,
writer: DetectorWriterT,
config_sigs: Sequence[SignalR] = (),
name: str = "",
connector: DeviceConnector | None = None,
Expand Down Expand Up @@ -211,19 +228,11 @@ def __init__(
self._initial_frame: int = 0
super().__init__(name, connector=connector)

@property
def controller(self) -> DetectorController:
return self._controller

@property
def writer(self) -> DetectorWriter:
return self._writer

@AsyncStatus.wrap
async def stage(self) -> None:
# Disarm the detector, stop file writing.
await self._check_config_sigs()
await asyncio.gather(self.writer.close(), self.controller.disarm())
await asyncio.gather(self._writer.close(), self._controller.disarm())
self._trigger_info = None

async def _check_config_sigs(self):
Expand All @@ -244,7 +253,7 @@ async def _check_config_sigs(self):
@AsyncStatus.wrap
async def unstage(self) -> None:
# Stop data writing.
await asyncio.gather(self.writer.close(), self.controller.disarm())
await asyncio.gather(self._writer.close(), self._controller.disarm())

async def read_configuration(self) -> dict[str, Reading]:
return await merge_gathered_dicts(sig.read() for sig in self._config_sigs)
Expand Down Expand Up @@ -274,12 +283,12 @@ async def trigger(self) -> None:
assert self._trigger_info
assert self._trigger_info.trigger is DetectorTrigger.INTERNAL
# Arm the detector and wait for it to finish.
indices_written = await self.writer.get_indices_written()
await self.controller.arm()
await self.controller.wait_for_idle()
indices_written = await self._writer.get_indices_written()
await self._controller.arm()
await self._controller.wait_for_idle()
end_observation = indices_written + 1

async for index in self.writer.observe_indices_written(
async for index in self._writer.observe_indices_written(
DEFAULT_TIMEOUT
+ (self._trigger_info.livetime or 0)
+ (self._trigger_info.deadtime or 0)
Expand Down Expand Up @@ -308,9 +317,9 @@ async def prepare(self, value: TriggerInfo) -> None:
value.deadtime
), "Deadtime must be supplied when in externally triggered mode"
if value.deadtime:
required = self.controller.get_deadtime(value.livetime)
required = self._controller.get_deadtime(value.livetime)
assert required <= value.deadtime, (
f"Detector {self.controller} needs at least {required}s deadtime, "
f"Detector {self._controller} needs at least {required}s deadtime, "
f"but trigger logic provides only {value.deadtime}s"
)
self._trigger_info = value
Expand All @@ -319,12 +328,12 @@ async def prepare(self, value: TriggerInfo) -> None:
if isinstance(self._trigger_info.number_of_triggers, list)
else [self._trigger_info.number_of_triggers]
)
self._initial_frame = await self.writer.get_indices_written()
self._initial_frame = await self._writer.get_indices_written()
self._describe, _ = await asyncio.gather(
self.writer.open(value.multiplier), self.controller.prepare(value)
self._writer.open(value.multiplier), self._controller.prepare(value)
)
if value.trigger != DetectorTrigger.INTERNAL:
await self.controller.arm()
await self._controller.arm()
self._fly_start = time.monotonic()

@AsyncStatus.wrap
Expand All @@ -343,7 +352,7 @@ async def kickoff(self):
@WatchableAsyncStatus.wrap
async def complete(self):
assert self._trigger_info
indices_written = self.writer.observe_indices_written(
indices_written = self._writer.observe_indices_written(
self._trigger_info.frame_timeout
or (
DEFAULT_TIMEOUT
Expand Down Expand Up @@ -372,7 +381,7 @@ async def complete(self):
self._completable_frames = 0
self._frames_to_complete = 0
self._number_of_triggers_iter = None
await self.controller.wait_for_idle()
await self._controller.wait_for_idle()

async def describe_collect(self) -> dict[str, DataKey]:
return self._describe
Expand All @@ -384,9 +393,13 @@ async def collect_asset_docs(
# The index is optional, and provided for fly scans, however this needs to be
# retrieved for step scans.
if index is None:
index = await self.writer.get_indices_written()
async for doc in self.writer.collect_stream_docs(index):
index = await self._writer.get_indices_written()
async for doc in self._writer.collect_stream_docs(index):
yield doc

async def get_index(self) -> int:
return await self.writer.get_indices_written()
return await self._writer.get_indices_written()

@property
def hints(self) -> Hints:
return self._writer.hints
4 changes: 2 additions & 2 deletions src/ophyd_async/core/_flyer.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
from abc import ABC, abstractmethod
from typing import Generic
from typing import Any, Generic

from bluesky.protocols import Flyable, Preparable, Stageable

Expand All @@ -10,7 +10,7 @@

class FlyerController(ABC, Generic[T]):
@abstractmethod
async def prepare(self, value: T):
async def prepare(self, value: T) -> Any:
"""Move to the start of the flyscan"""

@abstractmethod
Expand Down
67 changes: 29 additions & 38 deletions src/ophyd_async/epics/adaravis/_aravis.py
jwlodek marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -1,61 +1,52 @@
from typing import get_args
from collections.abc import Sequence

from bluesky.protocols import HasHints, Hints

from ophyd_async.core import PathProvider, StandardDetector
from ophyd_async.core import PathProvider
from ophyd_async.core._signal import SignalR
from ophyd_async.epics import adcore
from ophyd_async.epics.adcore._core_io import ADBaseDatasetDescriber

from ._aravis_controller import AravisController
from ._aravis_io import AravisDriverIO


class AravisDetector(StandardDetector, HasHints):
class AravisDetector(adcore.AreaDetector[AravisController, adcore.ADWriter]):
"""
Ophyd-async implementation of an ADAravis Detector.
The detector may be configured for an external trigger on a GPIO port,
which must be done prior to preparing the detector
"""

_controller: AravisController
_writer: adcore.ADHDFWriter

def __init__(
self,
prefix: str,
path_provider: PathProvider,
drv_suffix="cam1:",
hdf_suffix="HDF1:",
name="",
writer_cls: type[adcore.ADWriter] = adcore.ADHDFWriter,
fileio_suffix: str | None = None,
name: str = "",
gpio_number: AravisController.GPIO_NUMBER = 1,
config_sigs: Sequence[SignalR] = (),
plugins: dict[str, adcore.NDPluginBaseIO] | None = None,
):
self.drv = AravisDriverIO(prefix + drv_suffix)
self.hdf = adcore.NDFileHDFIO(prefix + hdf_suffix)
controller, driver = AravisController.controller_and_drv(
prefix + drv_suffix, gpio_number=gpio_number, name=name
)
writer, fileio = writer_cls.writer_and_io(
prefix,
path_provider,
lambda: name,
jwlodek marked this conversation as resolved.
Show resolved Hide resolved
ADBaseDatasetDescriber(driver),
fileio_suffix=fileio_suffix,
plugins=plugins,
)

super().__init__(
AravisController(self.drv, gpio_number=gpio_number),
adcore.ADHDFWriter(
self.hdf,
path_provider,
lambda: self.name,
adcore.ADBaseDatasetDescriber(self.drv),
),
config_sigs=(self.drv.acquire_time,),
driver=driver,
jwlodek marked this conversation as resolved.
Show resolved Hide resolved
controller=controller,
fileio=fileio,
writer=writer,
plugins=plugins,
name=name,
config_sigs=config_sigs,
)

def get_external_trigger_gpio(self):
return self._controller.gpio_number

def set_external_trigger_gpio(self, gpio_number: AravisController.GPIO_NUMBER):
supported_gpio_numbers = get_args(AravisController.GPIO_NUMBER)
if gpio_number not in supported_gpio_numbers:
raise ValueError(
f"{self.__class__.__name__} only supports the following GPIO "
f"indices: {supported_gpio_numbers} but was asked to "
f"use {gpio_number}"
)
self._controller.gpio_number = gpio_number

@property
def hints(self) -> Hints:
return self._writer.hints
self.drv = driver
self.fileio = fileio
63 changes: 41 additions & 22 deletions src/ophyd_async/epics/adaravis/_aravis_controller.py
Original file line number Diff line number Diff line change
@@ -1,12 +1,9 @@
import asyncio
from typing import Literal
from typing import Literal, TypeVar, get_args

from ophyd_async.core import (
AsyncStatus,
DetectorController,
DetectorTrigger,
TriggerInfo,
set_and_wait_for_value,
)
from ophyd_async.epics import adcore

Expand All @@ -17,14 +14,33 @@
# runtime. See https://github.com/bluesky/ophyd-async/issues/308
_HIGHEST_POSSIBLE_DEADTIME = 1961e-6

AravisControllerT = TypeVar("AravisControllerT", bound="AravisController")
coretl marked this conversation as resolved.
Show resolved Hide resolved

class AravisController(DetectorController):

class AravisController(adcore.ADBaseController[AravisDriverIO]):
GPIO_NUMBER = Literal[1, 2, 3, 4]

def __init__(self, driver: AravisDriverIO, gpio_number: GPIO_NUMBER) -> None:
self._drv = driver
def __init__(
self,
driver: AravisDriverIO,
good_states: frozenset[adcore.DetectorState] = adcore.DEFAULT_GOOD_STATES,
gpio_number: GPIO_NUMBER = 1,
) -> None:
super().__init__(driver, good_states=good_states)
self.gpio_number = gpio_number
self._arm_status: AsyncStatus | None = None

@classmethod
def controller_and_drv(
cls: type[AravisControllerT],
prefix: str,
good_states: frozenset[adcore.DetectorState] = adcore.DEFAULT_GOOD_STATES,
name: str = "",
gpio_number: GPIO_NUMBER = 1,
) -> tuple[AravisControllerT, AravisDriverIO]:
driver_cls = get_args(cls.__orig_bases__[0])[0] # type: ignore
driver = driver_cls(prefix, name=name)
controller = cls(driver, good_states=good_states, gpio_number=gpio_number)
return controller, driver
jwlodek marked this conversation as resolved.
Show resolved Hide resolved

def get_deadtime(self, exposure: float | None) -> float:
return _HIGHEST_POSSIBLE_DEADTIME
Expand All @@ -35,25 +51,18 @@ async def prepare(self, trigger_info: TriggerInfo):
else:
image_mode = adcore.ImageMode.MULTIPLE
if (exposure := trigger_info.livetime) is not None:
await self._drv.acquire_time.set(exposure)
await self._driver.acquire_time.set(exposure)

trigger_mode, trigger_source = self._get_trigger_info(trigger_info.trigger)
# trigger mode must be set first and on it's own!
await self._drv.trigger_mode.set(trigger_mode)
await self._driver.trigger_mode.set(trigger_mode)

await asyncio.gather(
self._drv.trigger_source.set(trigger_source),
self._drv.num_images.set(trigger_info.total_number_of_triggers),
self._drv.image_mode.set(image_mode),
self._driver.trigger_source.set(trigger_source),
self._driver.num_images.set(trigger_info.total_number_of_triggers),
self._driver.image_mode.set(image_mode),
)

async def arm(self):
self._arm_status = await set_and_wait_for_value(self._drv.acquire, True)

async def wait_for_idle(self):
if self._arm_status:
await self._arm_status

def _get_trigger_info(
self, trigger: DetectorTrigger
) -> tuple[AravisTriggerMode, AravisTriggerSource]:
Expand All @@ -73,5 +82,15 @@ def _get_trigger_info(
else:
return (AravisTriggerMode.ON, f"Line{self.gpio_number}") # type: ignore

async def disarm(self):
await adcore.stop_busy_record(self._drv.acquire, False, timeout=1)
def get_external_trigger_gpio(self):
return self.gpio_number

def set_external_trigger_gpio(self, gpio_number: GPIO_NUMBER):
supported_gpio_numbers = get_args(AravisController.GPIO_NUMBER)
if gpio_number not in supported_gpio_numbers:
raise ValueError(
f"{self.__class__.__name__} only supports the following GPIO "
f"indices: {supported_gpio_numbers} but was asked to "
f"use {gpio_number}"
)
self.gpio_number = gpio_number
jwlodek marked this conversation as resolved.
Show resolved Hide resolved
Loading