diff --git a/dev-requirements.txt b/dev-requirements.txt index bd5dc79d6..9a0037529 100644 --- a/dev-requirements.txt +++ b/dev-requirements.txt @@ -1,16 +1,16 @@ accessible-pygments==0.0.5 aioca==1.8.1 aiofiles==24.1.0 -aiohappyeyeballs==2.4.3 -aiohttp==3.11.0 +aiohappyeyeballs==2.4.4 +aiohttp==3.11.10 aiosignal==1.3.1 alabaster==1.0.0 annotated-types==0.7.0 -anyio==4.6.2.post1 +anyio==4.7.0 appdirs==1.4.4 asciitree==0.3.3 asgiref==3.8.1 -asttokens==2.4.1 +asttokens==3.0.0 attrs==24.2.0 babel==2.16.0 beautifulsoup4==4.12.3 @@ -32,38 +32,38 @@ colorama==0.4.6 colorlog==6.9.0 comm==0.2.2 compress-pickle==2.1.0 -confluent-kafka==2.6.0 +confluent-kafka==2.6.1 contourpy==1.3.1 copier==9.4.1 -coverage==7.6.4 -cryptography==43.0.3 +coverage==7.6.9 +cryptography==44.0.0 cycler==0.12.1 -dask==2024.11.2 +dask==2024.12.0 databroker==1.2.5 dataclasses-json==0.6.7 decorator==5.1.1 deepdiff==8.0.1 deepmerge==2.0 -Deprecated==1.2.14 +Deprecated==1.2.15 distlib==0.3.9 -dls-dodal==1.36.0 +dls-dodal==1.36.2 dnspython==2.7.0 docopt==0.6.2 doct==1.1.0 docutils==0.21.2 -dunamai==1.22.0 +dunamai==1.23.0 email_validator==2.2.0 entrypoints==0.4 epicscorelibs==7.0.7.99.1.1 event-model==1.22.1 executing==2.1.0 -fastapi==0.115.5 -fastapi-cli==0.0.5 +fastapi==0.115.6 +fastapi-cli==0.0.6 fasteners==0.19 filelock==3.16.1 flexcache==0.3 flexparser==0.4 -fonttools==4.54.1 +fonttools==4.55.2 frozenlist==1.5.0 fsspec==2024.10.0 funcy==2.0 @@ -71,18 +71,18 @@ gitdb==4.0.11 GitPython==3.1.43 googleapis-common-protos==1.66.0 graypy==2.1.0 -grpcio==1.67.1 +grpcio==1.68.1 h11==0.14.0 h5py==3.12.1 HeapDict==1.0.1 historydict==1.2.6 -httpcore==1.0.6 +httpcore==1.0.7 httptools==0.6.4 -httpx==0.27.2 +httpx==0.28.1 humanize==4.11.0 -identify==2.6.2 +identify==2.6.3 idna==3.10 -imageio==2.36.0 +imageio==2.36.1 imagesize==1.4.1 importlib_metadata==8.5.0 importlib_resources==6.4.5 @@ -105,7 +105,7 @@ lz4==4.3.3 markdown-it-py==3.0.0 MarkupSafe==3.0.2 marshmallow==3.23.1 -matplotlib==3.9.2 +matplotlib==3.9.3 matplotlib-inline==0.1.7 mdit-py-plugins==0.4.2 mdurl==0.1.2 @@ -115,35 +115,34 @@ mongoquery==1.4.2 msgpack==1.1.0 msgpack-numpy==0.4.8 multidict==6.1.0 -mypy==1.13.0 mypy-extensions==1.0.0 myst-parser==4.0.0 networkx==3.4.2 nodeenv==1.9.1 nose2==0.15.1 nslsii==0.10.7 -numcodecs==0.14.0 +numcodecs==0.14.1 numpy==1.26.4 observability-utils==0.1.4 opencv-python==4.10.0.84 opencv-python-headless==4.10.0.84 -opentelemetry-api==1.28.1 -opentelemetry-distro==0.49b1 -opentelemetry-exporter-otlp==1.28.1 -opentelemetry-exporter-otlp-proto-common==1.28.1 -opentelemetry-exporter-otlp-proto-grpc==1.28.1 -opentelemetry-exporter-otlp-proto-http==1.28.1 -opentelemetry-instrumentation==0.49b1 -opentelemetry-instrumentation-asgi==0.49b1 -opentelemetry-instrumentation-fastapi==0.49b1 -opentelemetry-proto==1.28.1 -opentelemetry-sdk==1.28.1 -opentelemetry-semantic-conventions==0.49b1 -opentelemetry-util-http==0.49b1 -ophyd==1.9.0 -ophyd-async==0.8.0a4 +opentelemetry-api==1.28.2 +opentelemetry-distro==0.49b2 +opentelemetry-exporter-otlp==1.28.2 +opentelemetry-exporter-otlp-proto-common==1.28.2 +opentelemetry-exporter-otlp-proto-grpc==1.28.2 +opentelemetry-exporter-otlp-proto-http==1.28.2 +opentelemetry-instrumentation==0.49b2 +opentelemetry-instrumentation-asgi==0.49b2 +opentelemetry-instrumentation-fastapi==0.49b2 +opentelemetry-proto==1.28.2 +opentelemetry-sdk==1.28.2 +opentelemetry-semantic-conventions==0.49b2 +opentelemetry-util-http==0.49b2 +ophyd==1.10.0 +ophyd-async==0.9.0a1 orderly-set==5.2.2 -orjson==3.10.11 +orjson==3.10.12 p4p==4.2.0 packaging==24.2 pandas==2.2.3 @@ -157,7 +156,7 @@ pika==1.3.2 pillow==11.0.0 PIMS==0.7 Pint==0.24.4 -pipdeptree==2.23.4 +pipdeptree==2.24.0 platformdirs==4.3.6 pluggy==1.5.0 plumbum==1.9.0 @@ -165,8 +164,8 @@ ply==3.11 pre_commit==4.0.1 prettytable==3.12.0 prompt-toolkit==3.0.36 -propcache==0.2.0 -protobuf==5.28.3 +propcache==0.2.1 +protobuf==5.29.1 psutil==6.1.0 ptyprocess==0.7.0 pure_eval==0.2.3 @@ -175,44 +174,45 @@ py==1.11.0 pyasn1==0.6.1 pycparser==2.22 pycryptodome==3.21.0 -pydantic==2.9.2 -pydantic-extra-types==2.10.0 +pydantic==2.10.3 +pydantic-extra-types==2.10.1 pydantic-settings==2.6.1 -pydantic_core==2.23.4 +pydantic_core==2.27.1 pydantic_numpy==5.0.2 pydata-sphinx-theme==0.16.0 pyepics==3.5.7 Pygments==2.18.0 -PyJWT==2.9.0 +PyJWT==2.10.1 pymongo==4.10.1 pyOlog==4.5.0 pyparsing==3.2.0 -pytest==8.3.3 +pyright==1.1.390 +pytest==8.3.4 pytest-asyncio==0.24.0 pytest-cov==6.0.0 python-dateutil==2.9.0.post0 python-dotenv==1.0.1 -python-multipart==0.0.17 +python-multipart==0.0.19 pytz==2024.2 PyYAML==6.0.2 questionary==2.0.1 -redis==5.2.0 +redis==5.2.1 redis-json-dict==0.2.1 referencing==0.35.1 requests==2.32.3 responses==0.25.3 rich==13.9.4 -rpds-py==0.21.0 +rich-toolkit==0.12.0 +rpds-py==0.22.3 ruamel.yaml==0.18.6 ruamel.yaml.clib==0.2.12 -ruff==0.7.3 +ruff==0.8.2 scanspec==0.7.6 semver==3.0.2 -setuptools==75.5.0 setuptools-dso==2.11 shellingham==1.5.4 shortuuid==1.0.13 -six==1.16.0 +six==1.17.0 slicerator==1.1.0 smmap==5.0.1 sniffio==1.3.1 @@ -233,7 +233,7 @@ sphinxcontrib-openapi==0.8.4 sphinxcontrib-qthelp==2.0.0 sphinxcontrib-serializinghtml==2.0.0 stack-data==0.6.3 -starlette==0.41.2 +starlette==0.41.3 stomp.py==8.2.0 suitcase-mongo==0.6.0 suitcase-msgpack==0.3.0 @@ -243,9 +243,9 @@ tifffile==2024.9.20 toolz==1.0.0 tox==3.28.0 tox-direct==0.4 -tqdm==4.67.0 +tqdm==4.67.1 traitlets==5.14.3 -typer==0.13.0 +typer==0.15.1 types-mock==5.1.0.20240425 types-PyYAML==6.0.12.20240917 types-requests==2.32.0.20241016 @@ -256,19 +256,19 @@ tzdata==2024.2 tzlocal==5.2 ujson==5.10.0 urllib3==2.2.3 -uvicorn==0.32.0 +uvicorn==0.32.1 uvloop==0.21.0 -virtualenv==20.27.1 -watchfiles==0.24.0 +virtualenv==20.28.0 +watchfiles==1.0.0 wcwidth==0.2.13 websocket-client==1.8.0 websockets==14.1 widgetsnbextension==4.0.13 -workflows==2.28 -wrapt==1.16.0 -xarray==2024.10.0 -yarl==1.17.1 +workflows==3.1 +wrapt==1.17.0 +xarray==2024.11.0 +yarl==1.18.3 zarr==2.18.3 zict==2.2.0 zipp==3.21.0 -zocalo==1.1.1 +zocalo==1.2.0 diff --git a/docs/explanations/decisions/0005-connect-devices.md b/docs/explanations/decisions/0005-connect-devices.md new file mode 100644 index 000000000..5a16527a4 --- /dev/null +++ b/docs/explanations/decisions/0005-connect-devices.md @@ -0,0 +1,20 @@ +# 5. Connect all dodal devices during startup + +Date: 10-12-2024 + +## Status + +Accepted + +## Context + +Currently, Dodal devices are not automatically connected when they are created. + +## Decision + +All Dodal devices will be configured to automatically connect upon creation, ensuring they are ready for use immediately after initialization. + +## Consequences + +- Devices that fail to connect during startup will remain disconnected. +- Additional measures will need to be implemented to ensure such devices are identified and reconnected promptly. diff --git a/pyproject.toml b/pyproject.toml index 2311c5914..dde5b0816 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -28,7 +28,7 @@ dependencies = [ "fastapi>=0.112.0", "uvicorn", "requests", - "dls-dodal>=1.36.0", + "dls-dodal>=1.36.2", "super-state-machine", # https://github.com/DiamondLightSource/blueapi/issues/553 "GitPython", "bluesky-stomp>=0.1.2", diff --git a/src/blueapi/core/context.py b/src/blueapi/core/context.py index 6dc4057f1..fcc63227d 100644 --- a/src/blueapi/core/context.py +++ b/src/blueapi/core/context.py @@ -4,15 +4,7 @@ from importlib import import_module from inspect import Parameter, signature from types import ModuleType, UnionType -from typing import ( - Any, - Generic, - TypeVar, - Union, - get_args, - get_origin, - get_type_hints, -) +from typing import Any, Generic, TypeVar, Union, get_args, get_origin, get_type_hints from bluesky.run_engine import RunEngine from dodal.utils import make_all_devices @@ -22,6 +14,7 @@ from pydantic.json_schema import JsonSchemaValue from pydantic_core import CoreSchema, core_schema +from blueapi import utils from blueapi.config import EnvironmentConfig, SourceKind from blueapi.utils import BlueapiPlanModelConfig, load_module_all @@ -115,6 +108,8 @@ def with_device_module(self, module: ModuleType) -> None: def with_dodal_module(self, module: ModuleType, **kwargs) -> None: devices, exceptions = make_all_devices(module, **kwargs) + utils.connect_devices(self.run_engine, module, devices, **kwargs) + for device in devices.values(): self.register_device(device) diff --git a/src/blueapi/utils/__init__.py b/src/blueapi/utils/__init__.py index c81762542..cf45f6936 100644 --- a/src/blueapi/utils/__init__.py +++ b/src/blueapi/utils/__init__.py @@ -1,4 +1,5 @@ from .base_model import BlueapiBaseModel, BlueapiModelConfig, BlueapiPlanModelConfig +from .connect_devices import connect_devices from .invalid_config_error import InvalidConfigError from .modules import load_module_all from .serialization import serialize @@ -12,4 +13,5 @@ "BlueapiModelConfig", "BlueapiPlanModelConfig", "InvalidConfigError", + "connect_devices", ] diff --git a/src/blueapi/utils/connect_devices.py b/src/blueapi/utils/connect_devices.py new file mode 100644 index 000000000..368d856ab --- /dev/null +++ b/src/blueapi/utils/connect_devices.py @@ -0,0 +1,91 @@ +import logging +from collections.abc import Mapping +from types import ModuleType + +from bluesky.run_engine import RunEngine +from dodal.utils import ( + AnyDevice, + DeviceInitializationController, + collect_factories, + filter_ophyd_devices, +) +from ophyd_async.core import NotConnected +from ophyd_async.plan_stubs import ensure_connected + +LOGGER = logging.getLogger(__name__) + + +def _report_successful_devices( + devices: Mapping[str, AnyDevice], + sim_backend: bool, +) -> None: + sim_statement = " (sim mode)" if sim_backend else "" + connected_devices = "\n".join( + sorted([f"\t{device_name}" for device_name in devices.keys()]) + ) + + LOGGER.info(f"{len(devices)} devices connected{sim_statement}:") + LOGGER.info(connected_devices) + + +def _establish_device_connections( + RE: RunEngine, + devices: Mapping[str, AnyDevice], + sim_backend: bool, +) -> tuple[Mapping[str, AnyDevice], Mapping[str, Exception]]: + ophyd_devices, ophyd_async_devices = filter_ophyd_devices(devices) + exceptions = {} + + # Connect ophyd devices + for name, device in ophyd_devices.items(): + try: + device.wait_for_connection() + except Exception as ex: + exceptions[name] = ex + + # Connect ophyd-async devices + try: + RE(ensure_connected(*ophyd_async_devices.values(), mock=sim_backend)) + except NotConnected as ex: + exceptions = {**exceptions, **ex.sub_errors} + + # Only return the subset of devices that haven't raised an exception + successful_devices = { + name: device for name, device in devices.items() if name not in exceptions + } + return successful_devices, exceptions + + +def connect_devices( + run_engine: RunEngine, module: ModuleType, devices: dict[str, AnyDevice], **kwargs +): + factories = collect_factories(module, include_skipped=False) + + def is_simulated_device(name, factory, **kwargs): + device = devices.get(name, None) + mock_flag = kwargs.get("mock", kwargs.get("fake_with_ophyd_sim", False)) + return device is not None and ( + isinstance(factory, DeviceInitializationController) + and (factory._mock or mock_flag) # noqa: SLF001 + and isinstance(device, AnyDevice) + ) + + sim_devices = { + name: devices.get(name) + for name, factory in factories.items() + if is_simulated_device(name, factory, **kwargs) + } + real_devices = { + name: device + for name, device in devices.items() + if sim_devices.get(name, None) is None and (isinstance(device, AnyDevice)) + } + + if len(real_devices) > 0: + real_devices, exceptions = _establish_device_connections( + run_engine, real_devices, False + ) + _report_successful_devices(real_devices, False) + if len(sim_devices) > 0: + sim_devices, _ = _establish_device_connections(run_engine, sim_devices, True) # type: ignore + _report_successful_devices(sim_devices, True) diff --git a/tests/unit_tests/core/fake_device_module.py b/tests/unit_tests/core/fake_device_module.py index d27af8480..0cd5e0b9f 100644 --- a/tests/unit_tests/core/fake_device_module.py +++ b/tests/unit_tests/core/fake_device_module.py @@ -1,6 +1,10 @@ from unittest.mock import MagicMock, NonCallableMock +from dodal.common.beamlines.beamline_utils import device_factory +from dodal.utils import OphydV1Device, OphydV2Device from ophyd import EpicsMotor +from ophyd_async.core import DEFAULT_TIMEOUT, LazyMock, StandardReadable +from ophyd_async.epics.motor import Motor def fake_motor_bundle_b( @@ -14,6 +18,45 @@ def fake_motor_x() -> EpicsMotor: return _mock_with_name("motor_x") +class DeviceA(StandardReadable): + def __init__(self, name: str = "") -> None: + with self.add_children_as_readables(): + self.motor = Motor("X:SIZE") + super().__init__(name) + + +@device_factory(mock=True) +def device_a() -> DeviceA: + return DeviceA() + + +class UnconnectableOphydDevice(OphydV1Device): + def wait_for_connection( + self, + all_signals: bool = False, + timeout: float = 2.0, + ) -> None: + raise RuntimeError(f"{self.name}: fake connection error for tests") + + +def ophyd_device() -> UnconnectableOphydDevice: + return UnconnectableOphydDevice(name="ophyd_device") + + +class UnconnectableOphydAsyncDevice(OphydV2Device): + async def connect( + self, + mock: bool | LazyMock = False, + timeout: float = DEFAULT_TIMEOUT, + force_reconnect: bool = False, + ) -> None: + raise RuntimeError(f"{self.name}: fake connection error for tests") + + +def ophyd_async_device() -> UnconnectableOphydAsyncDevice: + return UnconnectableOphydAsyncDevice(name="ophyd_async_device") + + def fake_motor_y() -> EpicsMotor: return _mock_with_name("motor_y") diff --git a/tests/unit_tests/core/test_context.py b/tests/unit_tests/core/test_context.py index 908fb142b..213abb8a1 100644 --- a/tests/unit_tests/core/test_context.py +++ b/tests/unit_tests/core/test_context.py @@ -190,6 +190,9 @@ def test_add_devices_from_module(empty_context: BlueskyContext) -> None: "motor_y", "motor_bundle_a", "motor_bundle_b", + "device_a", + "ophyd_device", + "ophyd_async_device", } == empty_context.devices.keys() @@ -280,6 +283,9 @@ def test_add_devices_and_plans_from_modules_with_config( "motor_y", "motor_bundle_a", "motor_bundle_b", + "device_a", + "ophyd_device", + "ophyd_async_device", } == empty_context.devices.keys() assert {"spec_scan"} == empty_context.plans.keys() @@ -371,7 +377,7 @@ def test_str_default( spec = empty_context._type_spec_for_function(has_default_reference) assert spec["m"][0] is movable_ref - assert (df := spec["m"][1].default_factory) and df() == SIM_MOTOR_NAME + assert (df := spec["m"][1].default_factory) and df() == SIM_MOTOR_NAME # type: ignore assert has_default_reference.__name__ in empty_context.plans model = empty_context.plans[has_default_reference.__name__].model @@ -390,7 +396,7 @@ def test_nested_str_default( spec = empty_context._type_spec_for_function(has_default_nested_reference) assert spec["m"][0] == list[movable_ref] # type: ignore - assert (df := spec["m"][1].default_factory) and df() == [SIM_MOTOR_NAME] + assert (df := spec["m"][1].default_factory) and df() == [SIM_MOTOR_NAME] # type: ignore assert has_default_nested_reference.__name__ in empty_context.plans model = empty_context.plans[has_default_nested_reference.__name__].model