-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge branch 'main' into adsim_async
- Loading branch information
Showing
17 changed files
with
904 additions
and
21 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Empty file.
File renamed without changes.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,150 @@ | ||
import itertools | ||
from collections.abc import Mapping | ||
from typing import Annotated, Any | ||
|
||
import bluesky.plan_stubs as bps | ||
from bluesky.protocols import Movable | ||
from bluesky.utils import MsgGenerator | ||
|
||
""" | ||
Wrappers for Bluesky built-in plan stubs with type hinting | ||
""" | ||
|
||
Group = Annotated[str, "String identifier used by 'wait' or stubs that await"] | ||
|
||
|
||
# After bluesky 1.14, bounds for stubs that move can be narrowed | ||
# https://github.com/bluesky/bluesky/issues/1821 | ||
def set_absolute( | ||
movable: Movable, value: Any, group: Group | None = None, wait: bool = False | ||
) -> MsgGenerator: | ||
""" | ||
Set a device, wrapper for `bp.abs_set`. | ||
Args: | ||
movable (Movable): The device to set | ||
value (T): The new value | ||
group (Group | None, optional): The message group to associate with the | ||
setting, for sequencing. Defaults to None. | ||
wait (bool, optional): The group should wait until all setting is complete | ||
(e.g. a motor has finished moving). Defaults to False. | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
return (yield from bps.abs_set(movable, value, group=group, wait=wait)) | ||
|
||
|
||
def set_relative( | ||
movable: Movable, value: Any, group: Group | None = None, wait: bool = False | ||
) -> MsgGenerator: | ||
""" | ||
Change a device, wrapper for `bp.rel_set`. | ||
Args: | ||
movable (Movable): The device to set | ||
value (T): The new value | ||
group (Group | None, optional): The message group to associate with the | ||
setting, for sequencing. Defaults to None. | ||
wait (bool, optional): The group should wait until all setting is complete | ||
(e.g. a motor has finished moving). Defaults to False. | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
|
||
return (yield from bps.rel_set(movable, value, group=group, wait=wait)) | ||
|
||
|
||
def move(moves: Mapping[Movable, Any], group: Group | None = None) -> MsgGenerator: | ||
""" | ||
Move a device, wrapper for `bp.mv`. | ||
Args: | ||
moves (Mapping[Movable, T]): Mapping of Movables to target positions | ||
group (Group | None, optional): The message group to associate with the | ||
setting, for sequencing. Defaults to None. | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
|
||
return ( | ||
# type ignore until https://github.com/bluesky/bluesky/issues/1809 | ||
yield from bps.mv(*itertools.chain.from_iterable(moves.items()), group=group) # type: ignore | ||
) | ||
|
||
|
||
def move_relative( | ||
moves: Mapping[Movable, Any], group: Group | None = None | ||
) -> MsgGenerator: | ||
""" | ||
Move a device relative to its current position, wrapper for `bp.mvr`. | ||
Args: | ||
moves (Mapping[Movable, T]): Mapping of Movables to target deltas | ||
group (Group | None, optional): The message group to associate with the | ||
setting, for sequencing. Defaults to None. | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
|
||
return ( | ||
# type ignore until https://github.com/bluesky/bluesky/issues/1809 | ||
yield from bps.mvr(*itertools.chain.from_iterable(moves.items()), group=group) # type: ignore | ||
) | ||
|
||
|
||
def sleep(time: float) -> MsgGenerator: | ||
""" | ||
Suspend all action for a given time, wrapper for `bp.sleep` | ||
Args: | ||
time (float): Time to wait in seconds | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
|
||
return (yield from bps.sleep(time)) | ||
|
||
|
||
def wait( | ||
group: Group | None = None, | ||
timeout: float | None = None, | ||
) -> MsgGenerator: | ||
""" | ||
Wait for a group status to complete, wrapper for `bp.wait`. | ||
Does not expose move_on, as when used as a stub will not fail on Timeout. | ||
Args: | ||
group (Group | None, optional): The name of the group to wait for, defaults | ||
to None, in which case waits for all | ||
groups that have not yet been awaited. | ||
timeout (float | None, default=None): a timeout in seconds | ||
Returns: | ||
MsgGenerator: Plan | ||
Yields: | ||
Iterator[MsgGenerator]: Bluesky messages | ||
""" | ||
|
||
return (yield from bps.wait(group, timeout=timeout)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
from .scanspec import spec_scan | ||
from .wrapped import count | ||
|
||
__all__ = ["count", "spec_scan"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,66 @@ | ||
import operator | ||
from functools import reduce | ||
from typing import Annotated, Any | ||
|
||
import bluesky.plans as bp | ||
from bluesky.protocols import Movable, Readable | ||
from cycler import Cycler, cycler | ||
from pydantic import Field, validate_call | ||
from scanspec.specs import Spec | ||
|
||
from dodal.common import MsgGenerator | ||
from dodal.plan_stubs.data_session import attach_data_session_metadata_decorator | ||
|
||
|
||
@attach_data_session_metadata_decorator() | ||
@validate_call(config={"arbitrary_types_allowed": True}) | ||
def spec_scan( | ||
detectors: Annotated[ | ||
set[Readable], | ||
Field( | ||
description="Set of readable devices, will take a reading at each point, \ | ||
in addition to any Movables in the Spec", | ||
), | ||
], | ||
spec: Annotated[ | ||
Spec[Movable], | ||
Field(description="ScanSpec modelling the path of the scan"), | ||
], | ||
metadata: dict[str, Any] | None = None, | ||
) -> MsgGenerator: | ||
"""Generic plan for reading `detectors` at every point of a ScanSpec `Spec`. | ||
A `Spec` is an N-dimensional path. | ||
""" | ||
# TODO: https://github.com/bluesky/scanspec/issues/154 | ||
# support Static.duration: Spec[Literal["DURATION"]] | ||
|
||
_md = { | ||
"plan_args": { | ||
"detectors": {det.name for det in detectors}, | ||
"spec": repr(spec), | ||
}, | ||
"plan_name": "spec_scan", | ||
"shape": spec.shape(), | ||
**(metadata or {}), | ||
} | ||
|
||
yield from bp.scan_nd(tuple(detectors), _as_cycler(spec), md=_md) | ||
|
||
|
||
def _as_cycler(spec: Spec[Movable]) -> Cycler: | ||
""" | ||
Convert a scanspec to a cycler for compatibility with legacy Bluesky plans such as | ||
`bp.scan_nd`. Use the midpoints of the scanspec since cyclers are normally used | ||
for software triggered scans. | ||
Args: | ||
spec: A scanspec | ||
Returns: | ||
Cycler: A new cycler | ||
""" | ||
|
||
midpoints = spec.frames().midpoints | ||
# Need to "add" the cyclers for all the axes together. The code below is | ||
# effectively: cycler(motor1, [...]) + cycler(motor2, [...]) + ... | ||
return reduce(operator.add, (cycler(*args) for args in midpoints.items())) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,57 @@ | ||
from collections.abc import Sequence | ||
from typing import Annotated, Any | ||
|
||
import bluesky.plans as bp | ||
from bluesky.protocols import Readable | ||
from pydantic import Field, NonNegativeFloat, validate_call | ||
|
||
from dodal.common import MsgGenerator | ||
from dodal.plan_stubs.data_session import attach_data_session_metadata_decorator | ||
|
||
"""This module wraps plan(s) from bluesky.plans until required handling for them is | ||
moved into bluesky or better handled in downstream services. | ||
Required decorators are installed on plan import | ||
https://github.com/DiamondLightSource/blueapi/issues/474 | ||
Non-serialisable fields are ignored when they are optional | ||
https://github.com/DiamondLightSource/blueapi/issues/711 | ||
We may also need other adjustments for UI purposes, e.g. | ||
Forcing uniqueness or orderedness of Readables | ||
Limits and metadata (e.g. units) | ||
""" | ||
|
||
|
||
@attach_data_session_metadata_decorator() | ||
@validate_call(config={"arbitrary_types_allowed": True}) | ||
def count( | ||
detectors: Annotated[ | ||
set[Readable], | ||
Field( | ||
description="Set of readable devices, will take a reading at each point", | ||
min_length=1, | ||
), | ||
], | ||
num: Annotated[int, Field(description="Number of frames to collect", ge=1)] = 1, | ||
delay: Annotated[ | ||
NonNegativeFloat | Sequence[NonNegativeFloat], | ||
Field( | ||
description="Delay between readings: if tuple, len(delay) == num - 1 and \ | ||
the delays are between each point, if value or None is the delay for every \ | ||
gap", | ||
json_schema_extra={"units": "s"}, | ||
), | ||
] = 0.0, | ||
metadata: dict[str, Any] | None = None, | ||
) -> MsgGenerator: | ||
"""Reads from a number of devices. | ||
Wraps bluesky.plans.count(det, num, delay, md=metadata) exposing only serializable | ||
parameters and metadata.""" | ||
if isinstance(delay, Sequence): | ||
assert ( | ||
len(delay) == num - 1 | ||
), f"Number of delays given must be {num - 1}: was given {len(delay)}" | ||
metadata = metadata or {} | ||
metadata["shape"] = (num,) | ||
yield from bp.count(tuple(detectors), num, delay=delay, md=metadata) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.