From d6f02a22b228d40b217f120922c120ca9a50567f Mon Sep 17 00:00:00 2001 From: TamarZanzouri Date: Fri, 17 May 2024 13:29:12 -0400 Subject: [PATCH] chore(api, robot-server): Manage runners via run orchestrator (#15190) --- api/src/opentrons/protocol_runner/__init__.py | 2 + .../protocol_runner/protocol_runner.py | 59 ++++---- .../protocol_runner/run_orchestrator.py | 101 +++++++++++++ .../protocol_runner/test_protocol_runner.py | 5 +- .../test_run_orchestrator_provider.py | 143 ++++++++++++++++++ .../robot_server/runs/dependencies.py | 4 +- .../robot_server/runs/engine_store.py | 75 ++++----- .../robot_server/runs/run_data_manager.py | 6 +- robot-server/tests/runs/test_engine_store.py | 18 +-- 9 files changed, 317 insertions(+), 96 deletions(-) create mode 100644 api/src/opentrons/protocol_runner/run_orchestrator.py create mode 100644 api/tests/opentrons/protocol_runner/test_run_orchestrator_provider.py diff --git a/api/src/opentrons/protocol_runner/__init__.py b/api/src/opentrons/protocol_runner/__init__.py index 35360cc77d9..9c5015e23fb 100644 --- a/api/src/opentrons/protocol_runner/__init__.py +++ b/api/src/opentrons/protocol_runner/__init__.py @@ -13,6 +13,7 @@ AnyRunner, ) from .create_simulating_runner import create_simulating_runner +from .run_orchestrator import RunOrchestrator __all__ = [ "AbstractRunner", @@ -23,4 +24,5 @@ "PythonAndLegacyRunner", "LiveRunner", "AnyRunner", + "RunOrchestrator", ] diff --git a/api/src/opentrons/protocol_runner/protocol_runner.py b/api/src/opentrons/protocol_runner/protocol_runner.py index 5a8b9134772..96c33a156d6 100644 --- a/api/src/opentrons/protocol_runner/protocol_runner.py +++ b/api/src/opentrons/protocol_runner/protocol_runner.py @@ -416,9 +416,9 @@ async def run( # noqa: D102 def create_protocol_runner( - protocol_config: Optional[Union[JsonProtocolConfig, PythonProtocolConfig]], protocol_engine: ProtocolEngine, hardware_api: HardwareControlAPI, + protocol_config: Union[JsonProtocolConfig, PythonProtocolConfig], task_queue: Optional[TaskQueue] = None, json_file_reader: Optional[JsonFileReader] = None, json_translator: Optional[JsonTranslator] = None, @@ -427,39 +427,32 @@ def create_protocol_runner( python_protocol_executor: Optional[PythonProtocolExecutor] = None, post_run_hardware_state: PostRunHardwareState = PostRunHardwareState.HOME_AND_STAY_ENGAGED, drop_tips_after_run: bool = True, -) -> AnyRunner: +) -> Union[JsonRunner, PythonAndLegacyRunner]: """Create a protocol runner.""" - if protocol_config: - if ( - isinstance(protocol_config, JsonProtocolConfig) - and protocol_config.schema_version >= LEGACY_JSON_SCHEMA_VERSION_CUTOFF - ): - return JsonRunner( - protocol_engine=protocol_engine, - hardware_api=hardware_api, - json_file_reader=json_file_reader, - json_translator=json_translator, - task_queue=task_queue, - post_run_hardware_state=post_run_hardware_state, - drop_tips_after_run=drop_tips_after_run, - ) - else: - return PythonAndLegacyRunner( - protocol_engine=protocol_engine, - hardware_api=hardware_api, - task_queue=task_queue, - python_and_legacy_file_reader=python_and_legacy_file_reader, - protocol_context_creator=protocol_context_creator, - python_protocol_executor=python_protocol_executor, - post_run_hardware_state=post_run_hardware_state, - drop_tips_after_run=drop_tips_after_run, - ) - - return LiveRunner( - protocol_engine=protocol_engine, - hardware_api=hardware_api, - task_queue=task_queue, - ) + if ( + isinstance(protocol_config, JsonProtocolConfig) + and protocol_config.schema_version >= LEGACY_JSON_SCHEMA_VERSION_CUTOFF + ): + return JsonRunner( + protocol_engine=protocol_engine, + hardware_api=hardware_api, + json_file_reader=json_file_reader, + json_translator=json_translator, + task_queue=task_queue, + post_run_hardware_state=post_run_hardware_state, + drop_tips_after_run=drop_tips_after_run, + ) + else: + return PythonAndLegacyRunner( + protocol_engine=protocol_engine, + hardware_api=hardware_api, + task_queue=task_queue, + python_and_legacy_file_reader=python_and_legacy_file_reader, + protocol_context_creator=protocol_context_creator, + python_protocol_executor=python_protocol_executor, + post_run_hardware_state=post_run_hardware_state, + drop_tips_after_run=drop_tips_after_run, + ) async def _yield() -> None: diff --git a/api/src/opentrons/protocol_runner/run_orchestrator.py b/api/src/opentrons/protocol_runner/run_orchestrator.py new file mode 100644 index 00000000000..bbd6088b411 --- /dev/null +++ b/api/src/opentrons/protocol_runner/run_orchestrator.py @@ -0,0 +1,101 @@ +"""Engine/Runner provider.""" +from __future__ import annotations +from typing import Optional, Union + +from . import protocol_runner, AnyRunner +from ..hardware_control import HardwareControlAPI +from ..protocol_engine import ProtocolEngine +from ..protocol_engine.types import PostRunHardwareState +from ..protocol_reader import JsonProtocolConfig, PythonProtocolConfig + + +class RunOrchestrator: + """Provider for runners and associated protocol engine. + + Build runners, manage command execution, run state and in-memory protocol engine associated to the runners. + """ + + _protocol_runner: Optional[ + Union[protocol_runner.JsonRunner, protocol_runner.PythonAndLegacyRunner, None] + ] + _setup_runner: protocol_runner.LiveRunner + _fixit_runner: protocol_runner.LiveRunner + _hardware_api: HardwareControlAPI + _protocol_engine: ProtocolEngine + + def __init__( + self, + protocol_engine: ProtocolEngine, + hardware_api: HardwareControlAPI, + fixit_runner: protocol_runner.LiveRunner, + setup_runner: protocol_runner.LiveRunner, + json_or_python_protocol_runner: Optional[ + Union[protocol_runner.PythonAndLegacyRunner, protocol_runner.JsonRunner] + ] = None, + run_id: Optional[str] = None, + ) -> None: + """Initialize a run orchestrator interface. + + Arguments: + protocol_engine: Protocol engine instance. + hardware_api: Hardware control API instance. + fixit_runner: LiveRunner for fixit commands. + setup_runner: LiveRunner for setup commands. + json_or_python_protocol_runner: JsonRunner/PythonAndLegacyRunner for protocol commands. + run_id: run id if any, associated to the runner/engine. + """ + self.run_id = run_id + self._protocol_engine = protocol_engine + self._hardware_api = hardware_api + self._protocol_runner = json_or_python_protocol_runner + self._setup_runner = setup_runner + self._fixit_runner = fixit_runner + + @property + def engine(self) -> ProtocolEngine: + """Get the "current" persisted ProtocolEngine.""" + return self._protocol_engine + + @property + def runner(self) -> AnyRunner: + """Get the "current" persisted ProtocolRunner.""" + return self._protocol_runner or self._setup_runner + + @classmethod + def build_orchestrator( + cls, + protocol_engine: ProtocolEngine, + hardware_api: HardwareControlAPI, + protocol_config: Optional[ + Union[JsonProtocolConfig, PythonProtocolConfig] + ] = None, + post_run_hardware_state: PostRunHardwareState = PostRunHardwareState.HOME_AND_STAY_ENGAGED, + drop_tips_after_run: bool = True, + run_id: Optional[str] = None, + ) -> "RunOrchestrator": + """Build a RunOrchestrator provider.""" + setup_runner = protocol_runner.LiveRunner( + protocol_engine=protocol_engine, + hardware_api=hardware_api, + ) + fixit_runner = protocol_runner.LiveRunner( + protocol_engine=protocol_engine, + hardware_api=hardware_api, + ) + json_or_python_runner = None + if protocol_config: + json_or_python_runner = protocol_runner.create_protocol_runner( + protocol_config=protocol_config, + protocol_engine=protocol_engine, + hardware_api=hardware_api, + post_run_hardware_state=post_run_hardware_state, + drop_tips_after_run=drop_tips_after_run, + ) + return cls( + run_id=run_id, + json_or_python_protocol_runner=json_or_python_runner, + setup_runner=setup_runner, + fixit_runner=fixit_runner, + hardware_api=hardware_api, + protocol_engine=protocol_engine, + ) diff --git a/api/tests/opentrons/protocol_runner/test_protocol_runner.py b/api/tests/opentrons/protocol_runner/test_protocol_runner.py index 6aa790bee98..f8893fac3fa 100644 --- a/api/tests/opentrons/protocol_runner/test_protocol_runner.py +++ b/api/tests/opentrons/protocol_runner/test_protocol_runner.py @@ -5,7 +5,7 @@ from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] from decoy import Decoy, matchers from pathlib import Path -from typing import List, cast, Optional, Union, Type +from typing import List, cast, Union, Type from opentrons_shared_data.labware.labware_definition import LabwareDefinition from opentrons_shared_data.labware.dev_types import ( @@ -174,7 +174,6 @@ def live_runner_subject( (PythonProtocolConfig(api_version=APIVersion(2, 14)), PythonAndLegacyRunner), (JsonProtocolConfig(schema_version=5), PythonAndLegacyRunner), (PythonProtocolConfig(api_version=APIVersion(2, 13)), PythonAndLegacyRunner), - (None, LiveRunner), ], ) def test_create_protocol_runner( @@ -186,7 +185,7 @@ def test_create_protocol_runner( python_and_legacy_file_reader: PythonAndLegacyFileReader, protocol_context_creator: ProtocolContextCreator, python_protocol_executor: PythonProtocolExecutor, - config: Optional[Union[JsonProtocolConfig, PythonProtocolConfig]], + config: Union[JsonProtocolConfig, PythonProtocolConfig], runner_type: Type[AnyRunner], ) -> None: """It should return protocol runner type depending on the config.""" diff --git a/api/tests/opentrons/protocol_runner/test_run_orchestrator_provider.py b/api/tests/opentrons/protocol_runner/test_run_orchestrator_provider.py new file mode 100644 index 00000000000..4c25ae28a4e --- /dev/null +++ b/api/tests/opentrons/protocol_runner/test_run_orchestrator_provider.py @@ -0,0 +1,143 @@ +"""Tests for the RunOrchestrator.""" +import pytest +from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] +from decoy import Decoy +from typing import Union + +from opentrons.protocols.api_support.types import APIVersion +from opentrons.protocol_engine import ProtocolEngine +from opentrons.protocol_engine.types import PostRunHardwareState +from opentrons.hardware_control import API as HardwareAPI +from opentrons.protocol_reader import JsonProtocolConfig, PythonProtocolConfig +from opentrons.protocol_runner.run_orchestrator import RunOrchestrator +from opentrons import protocol_runner +from opentrons.protocol_runner.protocol_runner import ( + JsonRunner, + PythonAndLegacyRunner, + LiveRunner, +) + + +@pytest.fixture +def mock_protocol_python_runner(decoy: Decoy) -> PythonAndLegacyRunner: + """Get a mocked out PythonAndLegacyRunner dependency.""" + return decoy.mock(cls=PythonAndLegacyRunner) + + +@pytest.fixture +def mock_protocol_json_runner(decoy: Decoy) -> JsonRunner: + """Get a mocked out PythonAndLegacyRunner dependency.""" + return decoy.mock(cls=JsonRunner) + + +@pytest.fixture +def mock_setup_runner(decoy: Decoy) -> LiveRunner: + """Get a mocked out LiveRunner dependency.""" + return decoy.mock(cls=LiveRunner) + + +@pytest.fixture +def mock_fixit_runner(decoy: Decoy) -> LiveRunner: + """Get a mocked out LiveRunner dependency.""" + return decoy.mock(cls=LiveRunner) + + +@pytest.fixture +def mock_protocol_engine(decoy: Decoy) -> ProtocolEngine: + """Get a mocked out ProtocolEngine dependency.""" + return decoy.mock(cls=ProtocolEngine) + + +@pytest.fixture +def mock_hardware_api(decoy: Decoy) -> HardwareAPI: + """Get a mocked out HardwareAPI dependency.""" + return decoy.mock(cls=HardwareAPI) + + +@pytest.fixture +def json_protocol_subject( + mock_protocol_engine: ProtocolEngine, + mock_hardware_api: HardwareAPI, + mock_protocol_json_runner: JsonRunner, + mock_fixit_runner: LiveRunner, + mock_setup_runner: LiveRunner, +) -> RunOrchestrator: + """Get a RunOrchestrator subject with a json runner.""" + return RunOrchestrator( + protocol_engine=mock_protocol_engine, + hardware_api=mock_hardware_api, + fixit_runner=mock_fixit_runner, + setup_runner=mock_setup_runner, + json_or_python_protocol_runner=mock_protocol_json_runner, + ) + + +@pytest.fixture +def python_protocol_subject( + mock_protocol_engine: ProtocolEngine, + mock_hardware_api: HardwareAPI, + mock_protocol_python_runner: PythonAndLegacyRunner, + mock_fixit_runner: LiveRunner, + mock_setup_runner: LiveRunner, +) -> RunOrchestrator: + """Get a RunOrchestrator subject with a python runner.""" + return RunOrchestrator( + protocol_engine=mock_protocol_engine, + hardware_api=mock_hardware_api, + fixit_runner=mock_fixit_runner, + setup_runner=mock_setup_runner, + json_or_python_protocol_runner=mock_protocol_python_runner, + ) + + +@pytest.mark.parametrize( + "input_protocol_config, mock_protocol_runner, subject", + [ + ( + JsonProtocolConfig(schema_version=7), + lazy_fixture("mock_protocol_json_runner"), + lazy_fixture("json_protocol_subject"), + ), + ( + PythonProtocolConfig(api_version=APIVersion(2, 14)), + lazy_fixture("mock_protocol_python_runner"), + lazy_fixture("python_protocol_subject"), + ), + ], +) +def test_build_run_orchestrator_provider( + decoy: Decoy, + monkeypatch: pytest.MonkeyPatch, + subject: RunOrchestrator, + mock_protocol_engine: ProtocolEngine, + mock_hardware_api: HardwareAPI, + input_protocol_config: Union[PythonProtocolConfig, JsonProtocolConfig], + mock_setup_runner: LiveRunner, + mock_fixit_runner: LiveRunner, + mock_protocol_runner: Union[PythonAndLegacyRunner, JsonRunner], +) -> None: + """Should get a RunOrchestrator instance.""" + mock_create_runner_func = decoy.mock(func=protocol_runner.create_protocol_runner) + monkeypatch.setattr( + protocol_runner, "create_protocol_runner", mock_create_runner_func + ) + + decoy.when( + mock_create_runner_func( + protocol_config=input_protocol_config, + protocol_engine=mock_protocol_engine, + hardware_api=mock_hardware_api, + post_run_hardware_state=PostRunHardwareState.HOME_AND_STAY_ENGAGED, + drop_tips_after_run=True, + ) + ).then_return(mock_protocol_runner) + + result = subject.build_orchestrator( + protocol_engine=mock_protocol_engine, + hardware_api=mock_hardware_api, + protocol_config=input_protocol_config, + ) + + assert isinstance(result, RunOrchestrator) + assert isinstance(result._setup_runner, LiveRunner) + assert isinstance(result._fixit_runner, LiveRunner) diff --git a/robot-server/robot_server/runs/dependencies.py b/robot-server/robot_server/runs/dependencies.py index f66ec9fdf1c..8ff687464e2 100644 --- a/robot-server/robot_server/runs/dependencies.py +++ b/robot-server/robot_server/runs/dependencies.py @@ -27,7 +27,7 @@ ) from .run_auto_deleter import RunAutoDeleter -from .engine_store import EngineStore, NoRunnerEnginePairError +from .engine_store import EngineStore, NoRunnerEngineError from .run_store import RunStore from .run_data_manager import RunDataManager from robot_server.errors.robot_errors import ( @@ -131,7 +131,7 @@ async def get_is_okay_to_create_maintenance_run( """Whether a maintenance run can be created if a protocol run already exists.""" try: protocol_run_state = engine_store.engine.state_view - except NoRunnerEnginePairError: + except NoRunnerEngineError: return True return ( not protocol_run_state.commands.has_been_played() diff --git a/robot-server/robot_server/runs/engine_store.py b/robot-server/robot_server/runs/engine_store.py index 5b6d57520a7..3e630cef0ec 100644 --- a/robot-server/robot_server/runs/engine_store.py +++ b/robot-server/robot_server/runs/engine_store.py @@ -1,7 +1,7 @@ """In-memory storage of ProtocolEngine instances.""" import asyncio import logging -from typing import List, NamedTuple, Optional, Callable +from typing import List, Optional, Callable from opentrons.protocol_engine.errors.exceptions import EStopActivatedError from opentrons.protocol_engine.types import PostRunHardwareState @@ -23,7 +23,7 @@ JsonRunner, PythonAndLegacyRunner, RunResult, - create_protocol_runner, + RunOrchestrator, ) from opentrons.protocol_engine import ( Config as ProtocolEngineConfig, @@ -52,18 +52,10 @@ class EngineConflictError(RuntimeError): """ -class NoRunnerEnginePairError(RuntimeError): +class NoRunnerEngineError(RuntimeError): """Raised if you try to get the current engine or runner while there is none.""" -class RunnerEnginePair(NamedTuple): - """A stored Runner/ProtocolEngine pair.""" - - run_id: str - runner: AnyRunner - engine: ProtocolEngine - - async def handle_estop_event(engine_store: "EngineStore", event: HardwareEvent) -> None: """Handle an E-stop event from the hardware API. @@ -108,6 +100,8 @@ def run_handler_in_engine_thread_from_hardware_thread( class EngineStore: """Factory and in-memory storage for ProtocolEngine.""" + _run_orchestrator: Optional[RunOrchestrator] = None + def __init__( self, hardware_api: HardwareControlAPI, @@ -126,32 +120,32 @@ def __init__( self._robot_type = robot_type self._deck_type = deck_type self._default_engine: Optional[ProtocolEngine] = None - self._runner_engine_pair: Optional[RunnerEnginePair] = None hardware_api.register_callback(_get_estop_listener(self)) @property def engine(self) -> ProtocolEngine: """Get the "current" persisted ProtocolEngine.""" - if self._runner_engine_pair is None: - raise NoRunnerEnginePairError() - return self._runner_engine_pair.engine + if self._run_orchestrator is None: + raise NoRunnerEngineError() + return self._run_orchestrator.engine @property def runner(self) -> AnyRunner: """Get the "current" persisted ProtocolRunner.""" - if self._runner_engine_pair is None: - raise NoRunnerEnginePairError() - return self._runner_engine_pair.runner + if self._run_orchestrator is None: + raise NoRunnerEngineError() + return self._run_orchestrator.runner @property def current_run_id(self) -> Optional[str]: """Get the run identifier associated with the current engine/runner pair.""" return ( - self._runner_engine_pair.run_id - if self._runner_engine_pair is not None + self._run_orchestrator.run_id + if self._run_orchestrator is not None else None ) + # TODO(tz, 2024-5-14): remove this once its all redirected via orchestrator # TODO(mc, 2022-03-21): this resource locking is insufficient; # come up with something more sophisticated without race condition holes. async def get_default_engine(self) -> ProtocolEngine: @@ -161,14 +155,13 @@ async def get_default_engine(self) -> ProtocolEngine: EngineConflictError: if a run-specific engine is active. """ if ( - self._runner_engine_pair is not None + self._run_orchestrator is not None and self.engine.state_view.commands.has_been_played() and not self.engine.state_view.commands.get_is_stopped() ): raise EngineConflictError("An engine for a run is currently active") engine = self._default_engine - if engine is None: # TODO(mc, 2022-03-21): potential race condition engine = await create_protocol_engine( @@ -180,7 +173,6 @@ async def get_default_engine(self) -> ProtocolEngine: ), ) self._default_engine = engine - return engine async def create( @@ -231,7 +223,11 @@ async def create( post_run_hardware_state = PostRunHardwareState.HOME_AND_STAY_ENGAGED drop_tips_after_run = True - runner = create_protocol_runner( + if self._run_orchestrator is not None: + raise EngineConflictError("Another run is currently active.") + + self._run_orchestrator = RunOrchestrator.build_orchestrator( + run_id=run_id, protocol_engine=engine, hardware_api=self._hardware_api, protocol_config=protocol.source.config if protocol else None, @@ -239,18 +235,15 @@ async def create( drop_tips_after_run=drop_tips_after_run, ) - if self._runner_engine_pair is not None: - raise EngineConflictError("Another run is currently active.") - # FIXME(mm, 2022-12-21): These `await runner.load()`s introduce a # concurrency hazard. If two requests simultaneously call this method, # they will both "succeed" (with undefined results) instead of one # raising EngineConflictError. - if isinstance(runner, PythonAndLegacyRunner): + if isinstance(self.runner, PythonAndLegacyRunner): assert ( protocol is not None ), "A Python protocol should have a protocol source file." - await runner.load( + await self.runner.load( protocol.source, # Conservatively assume that we're re-running a protocol that # was uploaded before we added stricter validation, and that @@ -258,23 +251,17 @@ async def create( python_parse_mode=PythonParseMode.ALLOW_LEGACY_METADATA_AND_REQUIREMENTS, run_time_param_values=run_time_param_values, ) - elif isinstance(runner, JsonRunner): + elif isinstance(self.runner, JsonRunner): assert ( protocol is not None ), "A JSON protocol should have a protocol source file." - await runner.load(protocol.source) + await self.runner.load(protocol.source) else: - runner.prepare() + self.runner.prepare() for offset in labware_offsets: engine.add_labware_offset(offset) - self._runner_engine_pair = RunnerEnginePair( - run_id=run_id, - runner=runner, - engine=engine, - ) - return engine.state_view.get_summary() async def clear(self) -> RunResult: @@ -285,10 +272,8 @@ async def clear(self) -> RunResult: they cannot be cleared. """ engine = self.engine - state_view = engine.state_view runner = self.runner - - if state_view.commands.get_is_okay_to_clear(): + if engine.state_view.commands.get_is_okay_to_clear(): await engine.finish( drop_tips_after_run=False, set_run_status=False, @@ -297,11 +282,11 @@ async def clear(self) -> RunResult: else: raise EngineConflictError("Current run is not idle or stopped.") - run_data = state_view.get_summary() - commands = state_view.commands.get_all() - run_time_parameters = runner.run_time_parameters + run_data = engine.state_view.get_summary() + commands = engine.state_view.commands.get_all() + run_time_parameters = runner.run_time_parameters if runner else [] - self._runner_engine_pair = None + self._run_orchestrator = None return RunResult( state_summary=run_data, commands=commands, parameters=run_time_parameters diff --git a/robot-server/robot_server/runs/run_data_manager.py b/robot-server/robot_server/runs/run_data_manager.py index 311cfb93b40..960f4635134 100644 --- a/robot-server/robot_server/runs/run_data_manager.py +++ b/robot-server/robot_server/runs/run_data_manager.py @@ -335,7 +335,8 @@ async def update(self, run_id: str, current: Optional[bool]) -> Union[Run, BadRu ) else: state_summary = self._engine_store.engine.state_view.get_summary() - parameters = self._engine_store.runner.run_time_parameters + runner = self._engine_store.runner + parameters = runner.run_time_parameters if runner else [] run_resource = self._run_store.get(run_id=run_id) return _build_run( @@ -423,6 +424,7 @@ def _get_good_state_summary(self, run_id: str) -> Optional[StateSummary]: def _get_run_time_parameters(self, run_id: str) -> List[RunTimeParameter]: if run_id == self._engine_store.current_run_id: - return self._engine_store.runner.run_time_parameters + runner = self._engine_store.runner + return runner.run_time_parameters if runner else [] else: return self._run_store.get_run_time_parameters(run_id=run_id) diff --git a/robot-server/tests/runs/test_engine_store.py b/robot-server/tests/runs/test_engine_store.py index 330e974be9c..49c474b2ce9 100644 --- a/robot-server/tests/runs/test_engine_store.py +++ b/robot-server/tests/runs/test_engine_store.py @@ -11,18 +11,14 @@ from opentrons.hardware_control import HardwareControlAPI, API from opentrons.hardware_control.types import EstopStateNotification, EstopState from opentrons.protocol_engine import ProtocolEngine, StateSummary, types as pe_types -from opentrons.protocol_runner import ( - RunResult, - LiveRunner, - JsonRunner, -) +from opentrons.protocol_runner import RunResult, LiveRunner, JsonRunner from opentrons.protocol_reader import ProtocolReader, ProtocolSource from robot_server.protocols.protocol_store import ProtocolResource from robot_server.runs.engine_store import ( EngineStore, EngineConflictError, - NoRunnerEnginePairError, + NoRunnerEngineError, handle_estop_event, ) @@ -53,7 +49,7 @@ async def json_protocol_source() -> ProtocolSource: return await ProtocolReader().read_saved(files=[simple_protocol], directory=None) -async def test_create_engine(subject: EngineStore) -> None: +async def test_create_engine(decoy: Decoy, subject: EngineStore) -> None: """It should create an engine for a run.""" result = await subject.create( run_id="run-id", @@ -186,10 +182,10 @@ async def test_clear_engine(subject: EngineStore) -> None: assert subject.current_run_id is None assert isinstance(result, RunResult) - with pytest.raises(NoRunnerEnginePairError): + with pytest.raises(NoRunnerEngineError): subject.engine - with pytest.raises(NoRunnerEnginePairError): + with pytest.raises(NoRunnerEngineError): subject.runner @@ -225,9 +221,9 @@ async def test_clear_idle_engine(subject: EngineStore) -> None: await subject.clear() # TODO: test engine finish is called - with pytest.raises(NoRunnerEnginePairError): + with pytest.raises(NoRunnerEngineError): subject.engine - with pytest.raises(NoRunnerEnginePairError): + with pytest.raises(NoRunnerEngineError): subject.runner