diff --git a/api/src/opentrons/protocol_engine/clients/sync_client.py b/api/src/opentrons/protocol_engine/clients/sync_client.py index f95611c1b4c..2277ce815a4 100644 --- a/api/src/opentrons/protocol_engine/clients/sync_client.py +++ b/api/src/opentrons/protocol_engine/clients/sync_client.py @@ -127,6 +127,19 @@ def load_labware( return cast(commands.LoadLabwareResult, result) + def reload_labware( + self, + labware_id: str, + ) -> commands.ReloadLabwareResult: + """Execute a ReloadLabware command and return the result.""" + request = commands.ReloadLabwareCreate( + params=commands.ReloadLabwareParams( + labwareId=labware_id, + ) + ) + result = self._transport.execute_command(request=request) + return cast(commands.ReloadLabwareResult, result) + # TODO (spp, 2022-12-14): https://opentrons.atlassian.net/browse/RLAB-237 def move_labware( self, diff --git a/api/src/opentrons/protocol_engine/commands/__init__.py b/api/src/opentrons/protocol_engine/commands/__init__.py index 3dfe6eaf51f..b7aacc6b059 100644 --- a/api/src/opentrons/protocol_engine/commands/__init__.py +++ b/api/src/opentrons/protocol_engine/commands/__init__.py @@ -120,6 +120,14 @@ LoadLabwareCommandType, ) +from .reload_labware import ( + ReloadLabware, + ReloadLabwareParams, + ReloadLabwareCreate, + ReloadLabwareResult, + ReloadLabwareCommandType, +) + from .load_liquid import ( LoadLiquid, LoadLiquidParams, @@ -402,6 +410,12 @@ "LoadLabwareParams", "LoadLabwareResult", "LoadLabwareCommandType", + # reload labware command models + "ReloadLabware", + "ReloadLabwareCreate", + "ReloadLabwareParams", + "ReloadLabwareResult", + "ReloadLabwareCommandType", # load module command models "LoadModule", "LoadModuleCreate", diff --git a/api/src/opentrons/protocol_engine/commands/command_unions.py b/api/src/opentrons/protocol_engine/commands/command_unions.py index dc4cc18c35a..7674508cc96 100644 --- a/api/src/opentrons/protocol_engine/commands/command_unions.py +++ b/api/src/opentrons/protocol_engine/commands/command_unions.py @@ -100,6 +100,14 @@ LoadLabwareCommandType, ) +from .reload_labware import ( + ReloadLabware, + ReloadLabwareParams, + ReloadLabwareCreate, + ReloadLabwareResult, + ReloadLabwareCommandType, +) + from .load_liquid import ( LoadLiquid, LoadLiquidParams, @@ -304,6 +312,7 @@ Home, RetractAxis, LoadLabware, + ReloadLabware, LoadLiquid, LoadModule, LoadPipette, @@ -368,6 +377,7 @@ HomeParams, RetractAxisParams, LoadLabwareParams, + ReloadLabwareParams, LoadLiquidParams, LoadModuleParams, LoadPipetteParams, @@ -431,6 +441,7 @@ HomeCommandType, RetractAxisCommandType, LoadLabwareCommandType, + ReloadLabwareCommandType, LoadLiquidCommandType, LoadModuleCommandType, LoadPipetteCommandType, @@ -494,6 +505,7 @@ HomeCreate, RetractAxisCreate, LoadLabwareCreate, + ReloadLabwareCreate, LoadLiquidCreate, LoadModuleCreate, LoadPipetteCreate, @@ -558,6 +570,7 @@ HomeResult, RetractAxisResult, LoadLabwareResult, + ReloadLabwareResult, LoadLiquidResult, LoadModuleResult, LoadPipetteResult, diff --git a/api/src/opentrons/protocol_engine/commands/reload_labware.py b/api/src/opentrons/protocol_engine/commands/reload_labware.py new file mode 100644 index 00000000000..247f717feb9 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/reload_labware.py @@ -0,0 +1,86 @@ +"""Reload labware command request, result, and implementation models.""" +from __future__ import annotations +from pydantic import BaseModel, Field +from typing import TYPE_CHECKING, Optional, Type +from typing_extensions import Literal + +from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate + +if TYPE_CHECKING: + from ..state import StateView + from ..execution import EquipmentHandler + + +ReloadLabwareCommandType = Literal["reloadLabware"] + + +class ReloadLabwareParams(BaseModel): + """Payload required to load a labware into a slot.""" + + labwareId: str = Field( + ..., description="The already-loaded labware instance to update." + ) + + +class ReloadLabwareResult(BaseModel): + """Result data from the execution of a LoadLabware command.""" + + labwareId: str = Field( + ..., + description="An ID to reference this labware in subsequent commands. Same as the one in the parameters.", + ) + offsetId: Optional[str] = Field( + # Default `None` instead of `...` so this field shows up as non-required in + # OpenAPI. The server is allowed to omit it or make it null. + None, + description=( + "An ID referencing the labware offset that will apply" + " to the reloaded labware." + " This offset will be in effect until the labware is moved" + " with a `moveLabware` command." + " Null or undefined means no offset applies," + " so the default of (0, 0, 0) will be used." + ), + ) + + +class ReloadLabwareImplementation( + AbstractCommandImpl[ReloadLabwareParams, ReloadLabwareResult] +): + """Reload labware command implementation.""" + + def __init__( + self, equipment: EquipmentHandler, state_view: StateView, **kwargs: object + ) -> None: + self._equipment = equipment + self._state_view = state_view + + async def execute(self, params: ReloadLabwareParams) -> ReloadLabwareResult: + """Reload the definition and calibration data for a specific labware.""" + reloaded_labware = await self._equipment.reload_labware( + labware_id=params.labwareId, + ) + + return ReloadLabwareResult( + labwareId=params.labwareId, + offsetId=reloaded_labware.offsetId, + ) + + +class ReloadLabware(BaseCommand[ReloadLabwareParams, ReloadLabwareResult]): + """Reload labware command resource model.""" + + commandType: ReloadLabwareCommandType = "reloadLabware" + params: ReloadLabwareParams + result: Optional[ReloadLabwareResult] + + _ImplementationCls: Type[ReloadLabwareImplementation] = ReloadLabwareImplementation + + +class ReloadLabwareCreate(BaseCommandCreate[ReloadLabwareParams]): + """Reload labware command creation request.""" + + commandType: ReloadLabwareCommandType = "reloadLabware" + params: ReloadLabwareParams + + _CommandCls: Type[ReloadLabware] = ReloadLabware diff --git a/api/src/opentrons/protocol_engine/execution/__init__.py b/api/src/opentrons/protocol_engine/execution/__init__.py index 5d2da5e6840..80f2dfd0d99 100644 --- a/api/src/opentrons/protocol_engine/execution/__init__.py +++ b/api/src/opentrons/protocol_engine/execution/__init__.py @@ -8,6 +8,7 @@ LoadedPipetteData, LoadedModuleData, LoadedConfigureForVolumeData, + ReloadedLabwareData, ) from .movement import MovementHandler from .gantry_mover import GantryMover @@ -29,6 +30,7 @@ "create_queue_worker", "EquipmentHandler", "LoadedLabwareData", + "ReloadedLabwareData", "LoadedPipetteData", "LoadedModuleData", "LoadedConfigureForVolumeData", diff --git a/api/src/opentrons/protocol_engine/execution/equipment.py b/api/src/opentrons/protocol_engine/execution/equipment.py index ee04653bda2..7dc2f3bcfaa 100644 --- a/api/src/opentrons/protocol_engine/execution/equipment.py +++ b/api/src/opentrons/protocol_engine/execution/equipment.py @@ -56,6 +56,14 @@ class LoadedLabwareData: offsetId: Optional[str] +@dataclass(frozen=True) +class ReloadedLabwareData: + """The result of a reload labware procedure.""" + + location: LabwareLocation + offsetId: Optional[str] + + @dataclass(frozen=True) class LoadedPipetteData: """The result of a load pipette procedure.""" @@ -171,6 +179,25 @@ async def load_labware( labware_id=labware_id, definition=definition, offsetId=offset_id ) + async def reload_labware(self, labware_id: str) -> ReloadedLabwareData: + """Reload an already-loaded labware. This cannot change the labware location. + + Args: + labware_id: The ID of the already-loaded labware. + + Raises: + LabwareNotLoadedError: If `labware_id` does not reference a loaded labware. + + """ + location = self._state_store.labware.get_location(labware_id) + definition_uri = self._state_store.labware.get_definition_uri(labware_id) + offset_id = self.find_applicable_labware_offset_id( + labware_definition_uri=definition_uri, + labware_location=location, + ) + + return ReloadedLabwareData(location=location, offsetId=offset_id) + async def load_pipette( self, pipette_name: PipetteNameType, diff --git a/api/src/opentrons/protocol_engine/state/labware.py b/api/src/opentrons/protocol_engine/state/labware.py index a11f1a58e4a..e9750a652b4 100644 --- a/api/src/opentrons/protocol_engine/state/labware.py +++ b/api/src/opentrons/protocol_engine/state/labware.py @@ -31,6 +31,7 @@ Command, LoadLabwareResult, MoveLabwareResult, + ReloadLabwareResult, ) from ..types import ( DeckSlotLocation, @@ -187,18 +188,27 @@ def _handle_command(self, command: Command) -> None: ) self._state.definitions_by_uri[definition_uri] = command.result.definition + if isinstance(command.result, LoadLabwareResult): + location = command.params.location + else: + location = self._state.labware_by_id[command.result.labwareId].location self._state.labware_by_id[ command.result.labwareId ] = LoadedLabware.construct( id=command.result.labwareId, - location=command.params.location, + location=location, loadName=command.result.definition.parameters.loadName, definitionUri=definition_uri, offsetId=command.result.offsetId, displayName=command.params.displayName, ) + elif isinstance(command.result, ReloadLabwareResult): + labware_id = command.params.labwareId + new_offset_id = command.result.offsetId + self._state.labware_by_id[labware_id].offsetId = new_offset_id + elif isinstance(command.result, MoveLabwareResult): labware_id = command.params.labwareId new_location = command.params.newLocation diff --git a/api/tests/opentrons/protocol_engine/clients/test_sync_client.py b/api/tests/opentrons/protocol_engine/clients/test_sync_client.py index d5d1f930cca..e4f5d7602ca 100644 --- a/api/tests/opentrons/protocol_engine/clients/test_sync_client.py +++ b/api/tests/opentrons/protocol_engine/clients/test_sync_client.py @@ -161,6 +161,30 @@ def test_load_labware( assert result == expected_result +def test_reload_labware( + decoy: Decoy, + transport: ChildThreadTransport, + subject: SyncClient, +) -> None: + """It should execute a reload labware command.""" + expected_request = commands.ReloadLabwareCreate( + params=commands.ReloadLabwareParams( + labwareId="some-labware-id", + ) + ) + + expected_result = commands.ReloadLabwareResult( + labwareId="some-labware-id", offsetId=None + ) + decoy.when(transport.execute_command(request=expected_request)).then_return( + expected_result + ) + result = subject.reload_labware( + labware_id="some-labware-id", + ) + assert result == expected_result + + def test_load_module( decoy: Decoy, transport: ChildThreadTransport, diff --git a/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py b/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py new file mode 100644 index 00000000000..556d4975786 --- /dev/null +++ b/api/tests/opentrons/protocol_engine/commands/test_reload_labware.py @@ -0,0 +1,85 @@ +"""Test load labware commands.""" +import inspect +import pytest + +from decoy import Decoy + +from opentrons.types import DeckSlotName +from opentrons.protocols.models import LabwareDefinition + +from opentrons.protocol_engine.errors import ( + LabwareNotLoadedError, +) + +from opentrons.protocol_engine.types import ( + DeckSlotLocation, +) +from opentrons.protocol_engine.execution import ReloadedLabwareData, EquipmentHandler +from opentrons.protocol_engine.resources import labware_validation +from opentrons.protocol_engine.state import StateView + +from opentrons.protocol_engine.commands.reload_labware import ( + ReloadLabwareParams, + ReloadLabwareResult, + ReloadLabwareImplementation, +) + + +@pytest.fixture(autouse=True) +def patch_mock_labware_validation( + decoy: Decoy, monkeypatch: pytest.MonkeyPatch +) -> None: + """Mock out move_types.py functions.""" + for name, func in inspect.getmembers(labware_validation, inspect.isfunction): + monkeypatch.setattr(labware_validation, name, decoy.mock(func=func)) + + +async def test_reload_labware_implementation( + decoy: Decoy, + well_plate_def: LabwareDefinition, + equipment: EquipmentHandler, + state_view: StateView, +) -> None: + """A ReloadLabware command should have an execution implementation.""" + subject = ReloadLabwareImplementation(equipment=equipment, state_view=state_view) + + data = ReloadLabwareParams( + labwareId="my-labware-id", + ) + + decoy.when(await equipment.reload_labware(labware_id="my-labware-id",)).then_return( + ReloadedLabwareData( + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_4), + offsetId="labware-offset-id", + ) + ) + + result = await subject.execute(data) + + assert result == ReloadLabwareResult( + labwareId="my-labware-id", + offsetId="labware-offset-id", + ) + + +async def test_reload_labware_raises_labware_does_not_exist( + decoy: Decoy, + well_plate_def: LabwareDefinition, + equipment: EquipmentHandler, + state_view: StateView, +) -> None: + """A ReloadLabware command should raise if the specified labware is not loaded.""" + subject = ReloadLabwareImplementation(equipment=equipment, state_view=state_view) + + data = ReloadLabwareParams( + labwareId="my-labware-id", + ) + + decoy.when( + await equipment.reload_labware( + labware_id="my-labware-id", + ) + ).then_raise(LabwareNotLoadedError("What labware is this!")) + + with pytest.raises(LabwareNotLoadedError): + await subject.execute(data) diff --git a/api/tests/opentrons/protocol_engine/state/command_fixtures.py b/api/tests/opentrons/protocol_engine/state/command_fixtures.py index 191dd49bd48..82d5c801611 100644 --- a/api/tests/opentrons/protocol_engine/state/command_fixtures.py +++ b/api/tests/opentrons/protocol_engine/state/command_fixtures.py @@ -577,3 +577,27 @@ def create_prepare_to_aspirate_command(pipette_id: str) -> cmd.PrepareToAspirate params=params, result=result, ) + + +def create_reload_labware_command( + labware_id: str, + offset_id: Optional[str], +) -> cmd.ReloadLabware: + """Create a completed ReloadLabware command.""" + params = cmd.ReloadLabwareParams( + labwareId=labware_id, + ) + + result = cmd.ReloadLabwareResult( + labwareId=labware_id, + offsetId=offset_id, + ) + + return cmd.ReloadLabware( + id="command-id", + key="command-key", + status=cmd.CommandStatus.SUCCEEDED, + createdAt=datetime.now(), + params=params, + result=result, + ) diff --git a/api/tests/opentrons/protocol_engine/state/test_labware_store.py b/api/tests/opentrons/protocol_engine/state/test_labware_store.py index 9d926583fb0..960ce423194 100644 --- a/api/tests/opentrons/protocol_engine/state/test_labware_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_labware_store.py @@ -28,6 +28,7 @@ from .command_fixtures import ( create_load_labware_command, create_move_labware_command, + create_reload_labware_command, ) @@ -132,6 +133,64 @@ def test_handles_load_labware( assert subject.state.definitions_by_uri[expected_definition_uri] == well_plate_def +def test_handles_reload_labware( + subject: LabwareStore, + well_plate_def: LabwareDefinition, +) -> None: + """It should override labware data in the state.""" + load_labware = create_load_labware_command( + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_A1), + labware_id="test-labware-id", + definition=well_plate_def, + display_name="display-name", + offset_id=None, + ) + + subject.handle_action( + SucceedCommandAction(private_result=None, command=load_labware) + ) + expected_definition_uri = uri_from_details( + load_name=well_plate_def.parameters.loadName, + namespace=well_plate_def.namespace, + version=well_plate_def.version, + ) + assert ( + subject.state.labware_by_id["test-labware-id"].definitionUri + == expected_definition_uri + ) + + offset_request = LabwareOffsetCreate( + definitionUri="offset-definition-uri", + location=LabwareOffsetLocation(slotName=DeckSlotName.SLOT_1), + vector=LabwareOffsetVector(x=1, y=2, z=3), + ) + subject.handle_action( + AddLabwareOffsetAction( + request=offset_request, + labware_offset_id="offset-id", + created_at=datetime(year=2021, month=1, day=2), + ) + ) + reload_labware = create_reload_labware_command( + labware_id="test-labware-id", + offset_id="offset-id", + ) + subject.handle_action( + SucceedCommandAction(private_result=None, command=reload_labware) + ) + + expected_labware_data = LoadedLabware( + id="test-labware-id", + loadName=well_plate_def.parameters.loadName, + definitionUri=expected_definition_uri, + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_A1), + offsetId="offset-id", + displayName="display-name", + ) + assert subject.state.labware_by_id["test-labware-id"] == expected_labware_data + assert subject.state.definitions_by_uri[expected_definition_uri] == well_plate_def + + def test_handles_add_labware_definition( subject: LabwareStore, well_plate_def: LabwareDefinition, diff --git a/shared-data/command/schemas/8.json b/shared-data/command/schemas/8.json index a17be9ee690..3d25fab2e03 100644 --- a/shared-data/command/schemas/8.json +++ b/shared-data/command/schemas/8.json @@ -19,6 +19,7 @@ "home": "#/definitions/HomeCreate", "retractAxis": "#/definitions/RetractAxisCreate", "loadLabware": "#/definitions/LoadLabwareCreate", + "reloadLabware": "#/definitions/ReloadLabwareCreate", "loadLiquid": "#/definitions/LoadLiquidCreate", "loadModule": "#/definitions/LoadModuleCreate", "loadPipette": "#/definitions/LoadPipetteCreate", @@ -112,6 +113,9 @@ { "$ref": "#/definitions/LoadLabwareCreate" }, + { + "$ref": "#/definitions/ReloadLabwareCreate" + }, { "$ref": "#/definitions/LoadLiquidCreate" }, @@ -1407,6 +1411,49 @@ }, "required": ["params"] }, + "ReloadLabwareParams": { + "title": "ReloadLabwareParams", + "description": "Payload required to load a labware into a slot.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "The already-loaded labware instance to update.", + "type": "string" + } + }, + "required": ["labwareId"] + }, + "ReloadLabwareCreate": { + "title": "ReloadLabwareCreate", + "description": "Reload labware command creation request.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "reloadLabware", + "enum": ["reloadLabware"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/ReloadLabwareParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, "LoadLiquidParams": { "title": "LoadLiquidParams", "description": "Payload required to load a liquid into a well.",