From 5e9955bc0952c2e97cc65fff44ea83bd6ef85aee Mon Sep 17 00:00:00 2001 From: CaseyBatten Date: Thu, 2 Jan 2025 12:45:41 -0500 Subject: [PATCH] feat(api, shared-data): Expand Labware architecture to accommodate Lids (#17072) Covers EXEC-1000, EXEC-1001, EXEC-1002, EXEC-1004 For API 2.23 remove Lids as a handled labware concept in PAPI and treat them more as an attribute with new load commands with new Engine commands --- .../protocol_api/core/engine/protocol.py | 115 +++++++++++ .../core/legacy/legacy_protocol_core.py | 29 ++- .../opentrons/protocol_api/core/protocol.py | 30 ++- api/src/opentrons/protocol_api/labware.py | 74 +++++++ .../opentrons/protocol_api/module_contexts.py | 14 ++ .../protocol_api/protocol_context.py | 107 ++++++++++ api/src/opentrons/protocol_api/validation.py | 24 +++ .../protocol_engine/clients/sync_client.py | 12 ++ .../protocol_engine/commands/__init__.py | 30 +++ .../commands/command_unions.py | 26 +++ .../protocol_engine/commands/load_labware.py | 14 +- .../protocol_engine/commands/load_lid.py | 146 ++++++++++++++ .../commands/load_lid_stack.py | 189 ++++++++++++++++++ .../protocol_engine/execution/equipment.py | 78 +++++++- .../resources/labware_validation.py | 5 + .../protocol_engine/state/labware.py | 80 ++++++-- .../protocol_engine/state/update_types.py | 71 +++++++ api/src/opentrons/protocol_engine/types.py | 4 + api/tests/opentrons/conftest.py | 1 + .../core/engine/test_protocol_core.py | 148 ++++++++++++++ .../protocol_api/test_protocol_context.py | 109 ++++++++++ .../state/test_labware_view_old.py | 12 +- shared-data/command/schemas/11.json | 161 +++++++++++++++ shared-data/js/labware.ts | 2 +- shared-data/js/types.ts | 1 + .../1.json | 10 +- .../3/protocol_engine_lid_stack_object/1.json | 41 ++++ shared-data/labware/schemas/2.json | 9 +- shared-data/labware/schemas/3.json | 26 ++- .../labware/labware_definition.py | 16 ++ .../opentrons_shared_data/labware/types.py | 5 + 31 files changed, 1555 insertions(+), 34 deletions(-) create mode 100644 api/src/opentrons/protocol_engine/commands/load_lid.py create mode 100644 api/src/opentrons/protocol_engine/commands/load_lid_stack.py rename shared-data/labware/definitions/{2 => 3}/opentrons_tough_pcr_auto_sealing_lid/1.json (89%) create mode 100644 shared-data/labware/definitions/3/protocol_engine_lid_stack_object/1.json diff --git a/api/src/opentrons/protocol_api/core/engine/protocol.py b/api/src/opentrons/protocol_api/core/engine/protocol.py index bfc808c3091..ece431b0d1e 100644 --- a/api/src/opentrons/protocol_api/core/engine/protocol.py +++ b/api/src/opentrons/protocol_api/core/engine/protocol.py @@ -233,6 +233,9 @@ def load_labware( ) # FIXME(jbl, 2023-08-14) validating after loading the object issue validation.ensure_definition_is_labware(load_result.definition) + validation.ensure_definition_is_not_lid_after_api_version( + self.api_version, load_result.definition + ) # FIXME(mm, 2023-02-21): # @@ -322,6 +325,52 @@ def load_adapter( return labware_core + def load_lid( + self, + load_name: str, + location: LabwareCore, + namespace: Optional[str], + version: Optional[int], + ) -> LabwareCore: + """Load an individual lid using its identifying parameters. Must be loaded on an existing Labware.""" + load_location = self._convert_labware_location(location=location) + custom_labware_params = ( + self._engine_client.state.labware.find_custom_labware_load_params() + ) + namespace, version = load_labware_params.resolve( + load_name, namespace, version, custom_labware_params + ) + load_result = self._engine_client.execute_command_without_recovery( + cmd.LoadLidParams( + loadName=load_name, + location=load_location, + namespace=namespace, + version=version, + ) + ) + # FIXME(chb, 2024-12-06) validating after loading the object issue + validation.ensure_definition_is_lid(load_result.definition) + + deck_conflict.check( + engine_state=self._engine_client.state, + new_labware_id=load_result.labwareId, + existing_disposal_locations=self._disposal_locations, + # TODO: We can now fetch these IDs from engine too. + # See comment in self.load_labware(). + # + # Wrapping .keys() in list() is just to make Decoy verification easier. + existing_labware_ids=list(self._labware_cores_by_id.keys()), + existing_module_ids=list(self._module_cores_by_id.keys()), + ) + + labware_core = LabwareCore( + labware_id=load_result.labwareId, + engine_client=self._engine_client, + ) + + self._labware_cores_by_id[labware_core.labware_id] = labware_core + return labware_core + def move_labware( self, labware_core: LabwareCore, @@ -644,6 +693,72 @@ def set_last_location( self._last_location = location self._last_mount = mount + def load_lid_stack( + self, + load_name: str, + location: Union[DeckSlotName, StagingSlotName, LabwareCore], + quantity: int, + namespace: Optional[str], + version: Optional[int], + ) -> LabwareCore: + """Load a Stack of Lids to a given location, creating a Lid Stack.""" + if quantity < 1: + raise ValueError( + "When loading a lid stack quantity cannot be less than one." + ) + if isinstance(location, DeckSlotName) or isinstance(location, StagingSlotName): + load_location = self._convert_labware_location(location=location) + else: + if isinstance(location, LabwareCore): + load_location = self._convert_labware_location(location=location) + else: + raise ValueError( + "Expected type of Labware Location for lid stack must be Labware, not Legacy Labware or Well." + ) + + custom_labware_params = ( + self._engine_client.state.labware.find_custom_labware_load_params() + ) + namespace, version = load_labware_params.resolve( + load_name, namespace, version, custom_labware_params + ) + + load_result = self._engine_client.execute_command_without_recovery( + cmd.LoadLidStackParams( + loadName=load_name, + location=load_location, + namespace=namespace, + version=version, + quantity=quantity, + ) + ) + + # FIXME(CHB, 2024-12-04) just like load labware and load adapter we have a validating after loading the object issue + validation.ensure_definition_is_lid(load_result.definition) + + deck_conflict.check( + engine_state=self._engine_client.state, + new_labware_id=load_result.stackLabwareId, + existing_disposal_locations=self._disposal_locations, + # TODO (spp, 2023-11-27): We've been using IDs from _labware_cores_by_id + # and _module_cores_by_id instead of getting the lists directly from engine + # because of the chance of engine carrying labware IDs from LPC too. + # But with https://github.com/Opentrons/opentrons/pull/13943, + # & LPC in maintenance runs, we can now rely on engine state for these IDs too. + # Wrapping .keys() in list() is just to make Decoy verification easier. + existing_labware_ids=list(self._labware_cores_by_id.keys()), + existing_module_ids=list(self._module_cores_by_id.keys()), + ) + + labware_core = LabwareCore( + labware_id=load_result.stackLabwareId, + engine_client=self._engine_client, + ) + + self._labware_cores_by_id[labware_core.labware_id] = labware_core + + return labware_core + def get_deck_definition(self) -> DeckDefinitionV5: """Get the geometry definition of the robot's deck.""" return self._engine_client.state.labware.get_deck_definition() diff --git a/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py b/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py index d0b95ed82ca..8adadbe1ecf 100644 --- a/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py +++ b/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py @@ -6,7 +6,13 @@ from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.robot.types import RobotType -from opentrons.types import DeckSlotName, StagingSlotName, Location, Mount, Point +from opentrons.types import ( + DeckSlotName, + StagingSlotName, + Location, + Mount, + Point, +) from opentrons.util.broker import Broker from opentrons.hardware_control import SyncHardwareAPI from opentrons.hardware_control.modules import AbstractModule, ModuleModel, ModuleType @@ -267,6 +273,16 @@ def load_adapter( """Load an adapter using its identifying parameters""" raise APIVersionError(api_element="Loading adapter") + def load_lid( + self, + load_name: str, + location: LegacyLabwareCore, + namespace: Optional[str], + version: Optional[int], + ) -> LegacyLabwareCore: + """Load an individual lid labware using its identifying parameters. Must be loaded on a labware.""" + raise APIVersionError(api_element="Loading lid") + def load_robot(self) -> None: # type: ignore """Load an adapter using its identifying parameters""" raise APIVersionError(api_element="Loading robot") @@ -478,6 +494,17 @@ def set_last_location( self._last_location = location self._last_mount = mount + def load_lid_stack( + self, + load_name: str, + location: Union[DeckSlotName, StagingSlotName, LegacyLabwareCore], + quantity: int, + namespace: Optional[str], + version: Optional[int], + ) -> LegacyLabwareCore: + """Load a Stack of Lids to a given location, creating a Lid Stack.""" + raise APIVersionError(api_element="Lid stack") + def get_module_cores(self) -> List[legacy_module_core.LegacyModuleCore]: """Get loaded module cores.""" return self._module_cores diff --git a/api/src/opentrons/protocol_api/core/protocol.py b/api/src/opentrons/protocol_api/core/protocol.py index ba9f9a7d14a..27d41b921b0 100644 --- a/api/src/opentrons/protocol_api/core/protocol.py +++ b/api/src/opentrons/protocol_api/core/protocol.py @@ -10,7 +10,13 @@ from opentrons_shared_data.labware.types import LabwareDefinition from opentrons_shared_data.robot.types import RobotType -from opentrons.types import DeckSlotName, StagingSlotName, Location, Mount, Point +from opentrons.types import ( + DeckSlotName, + StagingSlotName, + Location, + Mount, + Point, +) from opentrons.hardware_control import SyncHardwareAPI from opentrons.hardware_control.modules.types import ModuleModel from opentrons.protocols.api_support.util import AxisMaxSpeeds @@ -94,6 +100,17 @@ def load_adapter( """Load an adapter using its identifying parameters""" ... + @abstractmethod + def load_lid( + self, + load_name: str, + location: LabwareCoreType, + namespace: Optional[str], + version: Optional[int], + ) -> LabwareCoreType: + """Load an individual lid labware using its identifying parameters. Must be loaded on a labware.""" + ... + @abstractmethod def move_labware( self, @@ -191,6 +208,17 @@ def set_last_location( ) -> None: ... + @abstractmethod + def load_lid_stack( + self, + load_name: str, + location: Union[DeckSlotName, StagingSlotName, LabwareCoreType], + quantity: int, + namespace: Optional[str], + version: Optional[int], + ) -> LabwareCoreType: + ... + @abstractmethod def get_deck_definition(self) -> DeckDefinitionV5: """Get the geometry definition of the robot's deck.""" diff --git a/api/src/opentrons/protocol_api/labware.py b/api/src/opentrons/protocol_api/labware.py index 5e919a44f86..bb8a094e4c2 100644 --- a/api/src/opentrons/protocol_api/labware.py +++ b/api/src/opentrons/protocol_api/labware.py @@ -544,6 +544,7 @@ def load_labware( self, name: str, label: Optional[str] = None, + lid: Optional[str] = None, namespace: Optional[str] = None, version: Optional[int] = None, ) -> Labware: @@ -573,6 +574,20 @@ def load_labware( self._core_map.add(labware_core, labware) + if lid is not None: + if self._api_version < validation.LID_STACK_VERSION_GATE: + raise APIVersionError( + api_element="Loading a Lid on a Labware", + until_version="2.23", + current_version=f"{self._api_version}", + ) + self._protocol_core.load_lid( + load_name=lid, + location=labware_core, + namespace=namespace, + version=version, + ) + return labware @requires_version(2, 15) @@ -597,6 +612,65 @@ def load_labware_from_definition( label=label, ) + @requires_version(2, 23) + def load_lid_stack( + self, + load_name: str, + quantity: int, + namespace: Optional[str] = None, + version: Optional[int] = None, + ) -> Labware: + """ + Load a stack of Lids onto a valid Deck Location or Adapter. + + :param str load_name: A string to use for looking up a lid definition. + You can find the ``load_name`` for any standard lid on the Opentrons + `Labware Library `_. + :param int quantity: The quantity of lids to be loaded in the stack. + :param str namespace: The namespace that the lid labware definition belongs to. + If unspecified, the API will automatically search two namespaces: + + - ``"opentrons"``, to load standard Opentrons labware definitions. + - ``"custom_beta"``, to load custom labware definitions created with the + `Custom Labware Creator `__. + + You might need to specify an explicit ``namespace`` if you have a custom + definition whose ``load_name`` is the same as an Opentrons-verified + definition, and you want to explicitly choose one or the other. + + :param version: The version of the labware definition. You should normally + leave this unspecified to let ``load_lid_stack()`` choose a version + automatically. + + :return: The initialized and loaded labware object representing the Lid Stack. + """ + if self._api_version < validation.LID_STACK_VERSION_GATE: + raise APIVersionError( + api_element="Loading a Lid Stack", + until_version="2.23", + current_version=f"{self._api_version}", + ) + + load_location = self._core + + load_name = validation.ensure_lowercase_name(load_name) + + result = self._protocol_core.load_lid_stack( + load_name=load_name, + location=load_location, + quantity=quantity, + namespace=namespace, + version=version, + ) + + labware = Labware( + core=result, + api_version=self._api_version, + protocol_core=self._protocol_core, + core_map=self._core_map, + ) + return labware + def set_calibration(self, delta: Point) -> None: """ An internal, deprecated method used for updating the labware offset. diff --git a/api/src/opentrons/protocol_api/module_contexts.py b/api/src/opentrons/protocol_api/module_contexts.py index 96950b927ef..614bb4f53c7 100644 --- a/api/src/opentrons/protocol_api/module_contexts.py +++ b/api/src/opentrons/protocol_api/module_contexts.py @@ -125,6 +125,7 @@ def load_labware( namespace: Optional[str] = None, version: Optional[int] = None, adapter: Optional[str] = None, + lid: Optional[str] = None, ) -> Labware: """Load a labware onto the module using its load parameters. @@ -180,6 +181,19 @@ def load_labware( version=version, location=load_location, ) + if lid is not None: + if self._api_version < validation.LID_STACK_VERSION_GATE: + raise APIVersionError( + api_element="Loading a lid on a Labware", + until_version="2.23", + current_version=f"{self._api_version}", + ) + self._protocol_core.load_lid( + load_name=lid, + location=labware_core, + namespace=namespace, + version=version, + ) if isinstance(self._core, LegacyModuleCore): labware = self._core.add_labware_core(cast(LegacyLabwareCore, labware_core)) diff --git a/api/src/opentrons/protocol_api/protocol_context.py b/api/src/opentrons/protocol_api/protocol_context.py index 182019674a5..b9f96e4d536 100644 --- a/api/src/opentrons/protocol_api/protocol_context.py +++ b/api/src/opentrons/protocol_api/protocol_context.py @@ -399,6 +399,7 @@ def load_labware( namespace: Optional[str] = None, version: Optional[int] = None, adapter: Optional[str] = None, + lid: Optional[str] = None, ) -> Labware: """Load a labware onto a location. @@ -443,6 +444,10 @@ def load_labware( values as the ``load_name`` parameter of :py:meth:`.load_adapter`. The adapter will use the same namespace as the labware, and the API will choose the adapter's version automatically. + :param lid: A lid to load the on top of the main labware. Accepts the same + values as the ``load_name`` parameter of :py:meth:`.load_lid_stack`. The + lid will use the same namespace as the labware, and the API will + choose the lid's version automatically. .. versionadded:: 2.15 """ @@ -483,6 +488,20 @@ def load_labware( version=version, ) + if lid is not None: + if self._api_version < validation.LID_STACK_VERSION_GATE: + raise APIVersionError( + api_element="Loading a Lid on a Labware", + until_version="2.23", + current_version=f"{self._api_version}", + ) + self._core.load_lid( + load_name=lid, + location=labware_core, + namespace=namespace, + version=version, + ) + labware = Labware( core=labware_core, api_version=self._api_version, @@ -1334,6 +1353,94 @@ def door_closed(self) -> bool: """Returns ``True`` if the front door of the robot is closed.""" return self._core.door_closed() + @requires_version(2, 23) + def load_lid_stack( + self, + load_name: str, + location: Union[DeckLocation, Labware], + quantity: int, + adapter: Optional[str] = None, + namespace: Optional[str] = None, + version: Optional[int] = None, + ) -> Labware: + """ + Load a stack of Lids onto a valid Deck Location or Adapter. + + :param str load_name: A string to use for looking up a lid definition. + You can find the ``load_name`` for any standard lid on the Opentrons + `Labware Library `_. + :param location: Either a :ref:`deck slot `, + like ``1``, ``"1"``, or ``"D1"``, or the a valid Opentrons Adapter. + :param int quantity: The quantity of lids to be loaded in the stack. + :param adapter: An adapter to load the lid stack on top of. Accepts the same + values as the ``load_name`` parameter of :py:meth:`.load_adapter`. The + adapter will use the same namespace as the lid labware, and the API will + choose the adapter's version automatically. + :param str namespace: The namespace that the lid labware definition belongs to. + If unspecified, the API will automatically search two namespaces: + + - ``"opentrons"``, to load standard Opentrons labware definitions. + - ``"custom_beta"``, to load custom labware definitions created with the + `Custom Labware Creator `__. + + You might need to specify an explicit ``namespace`` if you have a custom + definition whose ``load_name`` is the same as an Opentrons-verified + definition, and you want to explicitly choose one or the other. + + :param version: The version of the labware definition. You should normally + leave this unspecified to let ``load_lid_stack()`` choose a version + automatically. + + :return: The initialized and loaded labware object representing the Lid Stack. + """ + if self._api_version < validation.LID_STACK_VERSION_GATE: + raise APIVersionError( + api_element="Loading a Lid Stack", + until_version="2.23", + current_version=f"{self._api_version}", + ) + + load_location: Union[DeckSlotName, StagingSlotName, LabwareCore] + if isinstance(location, Labware): + load_location = location._core + else: + load_location = validation.ensure_and_convert_deck_slot( + location, self._api_version, self._core.robot_type + ) + + if adapter is not None: + if isinstance(load_location, DeckSlotName) or isinstance( + load_location, StagingSlotName + ): + loaded_adapter = self.load_adapter( + load_name=adapter, + location=load_location.value, + namespace=namespace, + ) + load_location = loaded_adapter._core + else: + raise ValueError( + "Location cannot be a Labware or Adapter when the 'adapter' field is not None." + ) + + load_name = validation.ensure_lowercase_name(load_name) + + result = self._core.load_lid_stack( + load_name=load_name, + location=load_location, + quantity=quantity, + namespace=namespace, + version=version, + ) + + labware = Labware( + core=result, + api_version=self._api_version, + protocol_core=self._core, + core_map=self._core_map, + ) + return labware + def _create_module_context( module_core: Union[ModuleCore, NonConnectedModuleCore], diff --git a/api/src/opentrons/protocol_api/validation.py b/api/src/opentrons/protocol_api/validation.py index cd1e5112718..e734a98e818 100644 --- a/api/src/opentrons/protocol_api/validation.py +++ b/api/src/opentrons/protocol_api/validation.py @@ -53,6 +53,9 @@ # The first APIVersion where Python protocols can specify staging deck slots (e.g. "D4") _STAGING_DECK_SLOT_VERSION_GATE = APIVersion(2, 16) +# The first APIVersion where Python protocols can load lids as stacks and treat them as attributes of a parent labware. +LID_STACK_VERSION_GATE = APIVersion(2, 23) + # Mapping of public Python Protocol API pipette load names # to names used by the internal Opentrons system _PIPETTE_NAMES_MAP = { @@ -364,6 +367,27 @@ def ensure_definition_is_labware(definition: LabwareDefinition) -> None: ) +def ensure_definition_is_lid(definition: LabwareDefinition) -> None: + """Ensure that one of the definition's allowed roles is `lid` or that that field is empty.""" + if LabwareRole.lid not in definition.allowedRoles: + raise LabwareDefinitionIsNotLabwareError( + f"Labware {definition.parameters.loadName} is not a lid." + ) + + +def ensure_definition_is_not_lid_after_api_version( + api_version: APIVersion, definition: LabwareDefinition +) -> None: + """Ensure that one of the definition's allowed roles is not `lid` or that the API Version is below the release where lid loading was seperated.""" + if ( + LabwareRole.lid in definition.allowedRoles + and api_version >= LID_STACK_VERSION_GATE + ): + raise APIVersionError( + f"Labware Lids cannot be loaded like standard labware in Protocols written with an API version greater than {LID_STACK_VERSION_GATE}." + ) + + _MODULE_ALIASES: Dict[str, ModuleModel] = { "magdeck": MagneticModuleModel.MAGNETIC_V1, "magnetic module": MagneticModuleModel.MAGNETIC_V1, diff --git a/api/src/opentrons/protocol_engine/clients/sync_client.py b/api/src/opentrons/protocol_engine/clients/sync_client.py index 71837a7a2ca..4d04353b271 100644 --- a/api/src/opentrons/protocol_engine/clients/sync_client.py +++ b/api/src/opentrons/protocol_engine/clients/sync_client.py @@ -77,6 +77,18 @@ def execute_command_without_recovery( ) -> commands.LoadPipetteResult: pass + @overload + def execute_command_without_recovery( + self, params: commands.LoadLidStackParams + ) -> commands.LoadLidStackResult: + pass + + @overload + def execute_command_without_recovery( + self, params: commands.LoadLidParams + ) -> commands.LoadLidResult: + pass + @overload def execute_command_without_recovery( self, params: commands.LiquidProbeParams diff --git a/api/src/opentrons/protocol_engine/commands/__init__.py b/api/src/opentrons/protocol_engine/commands/__init__.py index b5edda52397..4ad91012b11 100644 --- a/api/src/opentrons/protocol_engine/commands/__init__.py +++ b/api/src/opentrons/protocol_engine/commands/__init__.py @@ -174,6 +174,22 @@ LoadPipetteCommandType, ) +from .load_lid_stack import ( + LoadLidStack, + LoadLidStackParams, + LoadLidStackCreate, + LoadLidStackResult, + LoadLidStackCommandType, +) + +from .load_lid import ( + LoadLid, + LoadLidParams, + LoadLidCreate, + LoadLidResult, + LoadLidCommandType, +) + from .move_labware import ( MoveLabware, MoveLabwareParams, @@ -476,6 +492,20 @@ "LoadPipetteResult", "LoadPipetteCommandType", "LoadPipettePrivateResult", + # load lid stack command models + "LoadLidStack", + "LoadLidStackCreate", + "LoadLidStackParams", + "LoadLidStackResult", + "LoadLidStackCommandType", + "LoadLidStackPrivateResult", + # load lid command models + "LoadLid", + "LoadLidCreate", + "LoadLidParams", + "LoadLidResult", + "LoadLidCommandType", + "LoadLidPrivateResult", # move labware command models "MoveLabware", "MoveLabwareCreate", diff --git a/api/src/opentrons/protocol_engine/commands/command_unions.py b/api/src/opentrons/protocol_engine/commands/command_unions.py index 3f5bb09e510..b04b381ae6b 100644 --- a/api/src/opentrons/protocol_engine/commands/command_unions.py +++ b/api/src/opentrons/protocol_engine/commands/command_unions.py @@ -161,6 +161,22 @@ LoadPipetteCommandType, ) +from .load_lid_stack import ( + LoadLidStack, + LoadLidStackParams, + LoadLidStackCreate, + LoadLidStackResult, + LoadLidStackCommandType, +) + +from .load_lid import ( + LoadLid, + LoadLidParams, + LoadLidCreate, + LoadLidResult, + LoadLidCommandType, +) + from .move_labware import ( GripperMovementError, MoveLabware, @@ -367,6 +383,8 @@ LoadLiquidClass, LoadModule, LoadPipette, + LoadLidStack, + LoadLid, MoveLabware, MoveRelative, MoveToCoordinates, @@ -448,6 +466,8 @@ HomeParams, RetractAxisParams, LoadLabwareParams, + LoadLidStackParams, + LoadLidParams, ReloadLabwareParams, LoadLiquidParams, LoadLiquidClassParams, @@ -537,6 +557,8 @@ LoadLiquidClassCommandType, LoadModuleCommandType, LoadPipetteCommandType, + LoadLidStackCommandType, + LoadLidCommandType, MoveLabwareCommandType, MoveRelativeCommandType, MoveToCoordinatesCommandType, @@ -622,6 +644,8 @@ LoadLiquidClassCreate, LoadModuleCreate, LoadPipetteCreate, + LoadLidStackCreate, + LoadLidCreate, MoveLabwareCreate, MoveRelativeCreate, MoveToCoordinatesCreate, @@ -715,6 +739,8 @@ LoadLiquidClassResult, LoadModuleResult, LoadPipetteResult, + LoadLidStackResult, + LoadLidResult, MoveLabwareResult, MoveRelativeResult, MoveToCoordinatesResult, diff --git a/api/src/opentrons/protocol_engine/commands/load_labware.py b/api/src/opentrons/protocol_engine/commands/load_labware.py index 6b65fe239e4..d0e83863616 100644 --- a/api/src/opentrons/protocol_engine/commands/load_labware.py +++ b/api/src/opentrons/protocol_engine/commands/load_labware.py @@ -172,6 +172,19 @@ async def execute( top_labware_definition=loaded_labware.definition, bottom_labware_id=verified_location.labwareId, ) + # Validate load location is valid for lids + if ( + labware_validation.validate_definition_is_lid( + definition=loaded_labware.definition + ) + and loaded_labware.definition.compatibleParentLabware is not None + and self._state_view.labware.get_load_name(verified_location.labwareId) + not in loaded_labware.definition.compatibleParentLabware + ): + raise ValueError( + f"Labware Lid {params.loadName} may not be loaded on parent labware {self._state_view.labware.get_display_name(verified_location.labwareId)}." + ) + # Validate labware for the absorbance reader elif isinstance(params.location, ModuleLocation): module = self._state_view.modules.get(params.location.moduleId) @@ -179,7 +192,6 @@ async def execute( self._state_view.labware.raise_if_labware_incompatible_with_plate_reader( loaded_labware.definition ) - return SuccessData( public=LoadLabwareResult( labwareId=loaded_labware.labware_id, diff --git a/api/src/opentrons/protocol_engine/commands/load_lid.py b/api/src/opentrons/protocol_engine/commands/load_lid.py new file mode 100644 index 00000000000..4f2e49c7447 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/load_lid.py @@ -0,0 +1,146 @@ +"""Load lid 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 opentrons_shared_data.labware.labware_definition import LabwareDefinition + +from ..errors import LabwareCannotBeStackedError, LabwareIsNotAllowedInLocationError +from ..resources import labware_validation +from ..types import ( + LabwareLocation, + OnLabwareLocation, +) + +from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData +from ..errors.error_occurrence import ErrorOccurrence +from ..state.update_types import StateUpdate + +if TYPE_CHECKING: + from ..state.state import StateView + from ..execution import EquipmentHandler + + +LoadLidCommandType = Literal["loadLid"] + + +class LoadLidParams(BaseModel): + """Payload required to load a lid onto a labware.""" + + location: LabwareLocation = Field( + ..., + description="Labware the lid should be loaded onto.", + ) + loadName: str = Field( + ..., + description="Name used to reference a lid labware definition.", + ) + namespace: str = Field( + ..., + description="The namespace the lid labware definition belongs to.", + ) + version: int = Field( + ..., + description="The lid labware definition version.", + ) + + +class LoadLidResult(BaseModel): + """Result data from the execution of a LoadLabware command.""" + + labwareId: str = Field( + ..., + description="An ID to reference this lid labware in subsequent commands.", + ) + definition: LabwareDefinition = Field( + ..., + description="The full definition data for this lid labware.", + ) + + +class LoadLidImplementation( + AbstractCommandImpl[LoadLidParams, SuccessData[LoadLidResult]] +): + """Load lid 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: LoadLidParams) -> SuccessData[LoadLidResult]: + """Load definition and calibration data necessary for a lid.""" + if not isinstance(params.location, OnLabwareLocation): + raise LabwareIsNotAllowedInLocationError( + "Lid Labware is only allowed to be loaded on top of a labware. Try `load_lid_stack(...)` to load lids without parent labware." + ) + + verified_location = self._state_view.geometry.ensure_location_not_occupied( + params.location + ) + loaded_labware = await self._equipment.load_labware( + load_name=params.loadName, + namespace=params.namespace, + version=params.version, + location=verified_location, + labware_id=None, + ) + + # TODO(chb 2024-12-12) these validation checks happen after the labware is loaded, because they rely on + # on the definition. In practice this will not cause any issues since they will raise protocol ending + # exception, but for correctness should be refactored to do this check beforehand. + if not labware_validation.validate_definition_is_lid(loaded_labware.definition): + raise LabwareCannotBeStackedError( + f"Labware {params.loadName} is not a Lid and cannot be loaded onto {self._state_view.labware.get_display_name(params.location.labwareId)}." + ) + + state_update = StateUpdate() + + # In the case of lids being loaded on top of other labware, set the parent labware's lid + state_update.set_lid( + parent_labware_id=params.location.labwareId, + lid_id=loaded_labware.labware_id, + ) + + state_update.set_loaded_labware( + labware_id=loaded_labware.labware_id, + offset_id=loaded_labware.offsetId, + definition=loaded_labware.definition, + location=verified_location, + display_name=None, + ) + + if isinstance(verified_location, OnLabwareLocation): + self._state_view.labware.raise_if_labware_cannot_be_stacked( + top_labware_definition=loaded_labware.definition, + bottom_labware_id=verified_location.labwareId, + ) + + return SuccessData( + public=LoadLidResult( + labwareId=loaded_labware.labware_id, + definition=loaded_labware.definition, + ), + state_update=state_update, + ) + + +class LoadLid(BaseCommand[LoadLidParams, LoadLidResult, ErrorOccurrence]): + """Load lid command resource model.""" + + commandType: LoadLidCommandType = "loadLid" + params: LoadLidParams + result: Optional[LoadLidResult] + + _ImplementationCls: Type[LoadLidImplementation] = LoadLidImplementation + + +class LoadLidCreate(BaseCommandCreate[LoadLidParams]): + """Load lid command creation request.""" + + commandType: LoadLidCommandType = "loadLid" + params: LoadLidParams + + _CommandCls: Type[LoadLid] = LoadLid diff --git a/api/src/opentrons/protocol_engine/commands/load_lid_stack.py b/api/src/opentrons/protocol_engine/commands/load_lid_stack.py new file mode 100644 index 00000000000..7b430dfaf45 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/load_lid_stack.py @@ -0,0 +1,189 @@ +"""Load lid stack command request, result, and implementation models.""" +from __future__ import annotations +from pydantic import BaseModel, Field +from typing import TYPE_CHECKING, Optional, Type, List +from typing_extensions import Literal + +from opentrons_shared_data.labware.labware_definition import LabwareDefinition + +from ..errors import LabwareIsNotAllowedInLocationError, ProtocolEngineError +from ..resources import fixture_validation, labware_validation +from ..types import ( + LabwareLocation, + OnLabwareLocation, + DeckSlotLocation, + AddressableAreaLocation, +) + +from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData +from ..errors.error_occurrence import ErrorOccurrence +from ..state.update_types import StateUpdate + +if TYPE_CHECKING: + from ..state.state import StateView + from ..execution import EquipmentHandler + + +LoadLidStackCommandType = Literal["loadLidStack"] + +_LID_STACK_PE_LABWARE = "protocol_engine_lid_stack_object" +_LID_STACK_PE_NAMESPACE = "opentrons" +_LID_STACK_PE_VERSION = 1 + + +class LoadLidStackParams(BaseModel): + """Payload required to load a lid stack onto a location.""" + + location: LabwareLocation = Field( + ..., + description="Location the lid stack should be loaded into.", + ) + loadName: str = Field( + ..., + description="Name used to reference a lid labware definition.", + ) + namespace: str = Field( + ..., + description="The namespace the lid labware definition belongs to.", + ) + version: int = Field( + ..., + description="The lid labware definition version.", + ) + quantity: int = Field( + ..., + description="The quantity of lids to load.", + ) + + +class LoadLidStackResult(BaseModel): + """Result data from the execution of a LoadLidStack command.""" + + stackLabwareId: str = Field( + ..., + description="An ID to reference the Protocol Engine Labware Lid Stack in subsequent commands.", + ) + labwareIds: List[str] = Field( + ..., + description="A list of lid labware IDs to reference the lids in this stack by. The first ID is the bottom of the stack.", + ) + definition: LabwareDefinition = Field( + ..., + description="The full definition data for this lid labware.", + ) + location: LabwareLocation = Field( + ..., description="The Location that the stack of lid labware has been loaded." + ) + + +class LoadLidStackImplementation( + AbstractCommandImpl[LoadLidStackParams, SuccessData[LoadLidStackResult]] +): + """Load lid stack 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: LoadLidStackParams + ) -> SuccessData[LoadLidStackResult]: + """Load definition and calibration data necessary for a lid stack.""" + if isinstance(params.location, AddressableAreaLocation): + area_name = params.location.addressableAreaName + if not ( + fixture_validation.is_deck_slot(params.location.addressableAreaName) + or fixture_validation.is_abs_reader(params.location.addressableAreaName) + ): + raise LabwareIsNotAllowedInLocationError( + f"Cannot load {params.loadName} onto addressable area {area_name}" + ) + self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration( + area_name + ) + elif isinstance(params.location, DeckSlotLocation): + self._state_view.addressable_areas.raise_if_area_not_in_deck_configuration( + params.location.slotName.id + ) + + verified_location = self._state_view.geometry.ensure_location_not_occupied( + params.location + ) + + lid_stack_object = await self._equipment.load_labware( + load_name=_LID_STACK_PE_LABWARE, + namespace=_LID_STACK_PE_NAMESPACE, + version=_LID_STACK_PE_VERSION, + location=verified_location, + labware_id=None, + ) + if not labware_validation.validate_definition_is_system( + lid_stack_object.definition + ): + raise ProtocolEngineError( + message="Lid Stack Labware Object Labware Definition does not contain required allowed role 'system'." + ) + + loaded_lid_labwares = await self._equipment.load_lids( + load_name=params.loadName, + namespace=params.namespace, + version=params.version, + location=OnLabwareLocation(labwareId=lid_stack_object.labware_id), + quantity=params.quantity, + ) + loaded_lid_locations_by_id = {} + load_location = OnLabwareLocation(labwareId=lid_stack_object.labware_id) + for loaded_lid in loaded_lid_labwares: + loaded_lid_locations_by_id[loaded_lid.labware_id] = load_location + load_location = OnLabwareLocation(labwareId=loaded_lid.labware_id) + + state_update = StateUpdate() + state_update.set_loaded_lid_stack( + stack_id=lid_stack_object.labware_id, + stack_object_definition=lid_stack_object.definition, + stack_location=verified_location, + labware_ids=list(loaded_lid_locations_by_id.keys()), + labware_definition=loaded_lid_labwares[0].definition, + locations=loaded_lid_locations_by_id, + ) + + if isinstance(verified_location, OnLabwareLocation): + self._state_view.labware.raise_if_labware_cannot_be_stacked( + top_labware_definition=loaded_lid_labwares[ + params.quantity - 1 + ].definition, + bottom_labware_id=verified_location.labwareId, + ) + + return SuccessData( + public=LoadLidStackResult( + stackLabwareId=lid_stack_object.labware_id, + labwareIds=list(loaded_lid_locations_by_id.keys()), + definition=loaded_lid_labwares[0].definition, + location=params.location, + ), + state_update=state_update, + ) + + +class LoadLidStack( + BaseCommand[LoadLidStackParams, LoadLidStackResult, ErrorOccurrence] +): + """Load lid stack command resource model.""" + + commandType: LoadLidStackCommandType = "loadLidStack" + params: LoadLidStackParams + result: Optional[LoadLidStackResult] + + _ImplementationCls: Type[LoadLidStackImplementation] = LoadLidStackImplementation + + +class LoadLidStackCreate(BaseCommandCreate[LoadLidStackParams]): + """Load lid stack command creation request.""" + + commandType: LoadLidStackCommandType = "loadLidStack" + params: LoadLidStackParams + + _CommandCls: Type[LoadLidStack] = LoadLidStack diff --git a/api/src/opentrons/protocol_engine/execution/equipment.py b/api/src/opentrons/protocol_engine/execution/equipment.py index 792bd583b88..3d26b355741 100644 --- a/api/src/opentrons/protocol_engine/execution/equipment.py +++ b/api/src/opentrons/protocol_engine/execution/equipment.py @@ -1,6 +1,6 @@ """Equipment command side-effect logic.""" from dataclasses import dataclass -from typing import Optional, overload, Union +from typing import Optional, overload, Union, List from opentrons_shared_data.pipette.types import PipetteNameType @@ -152,10 +152,6 @@ async def load_labware( Returns: A LoadedLabwareData object. """ - labware_id = ( - labware_id if labware_id is not None else self._model_utils.generate_id() - ) - definition_uri = uri_from_details( load_name=load_name, namespace=namespace, @@ -172,6 +168,10 @@ async def load_labware( version=version, ) + labware_id = ( + labware_id if labware_id is not None else self._model_utils.generate_id() + ) + # Allow propagation of ModuleNotLoadedError. offset_id = self.find_applicable_labware_offset_id( labware_definition_uri=definition_uri, @@ -379,6 +379,74 @@ async def load_module( definition=attached_module.definition, ) + async def load_lids( + self, + load_name: str, + namespace: str, + version: int, + location: LabwareLocation, + quantity: int, + ) -> List[LoadedLabwareData]: + """Load one or many lid labware by assigning an identifier and pulling required data. + + Args: + load_name: The lid labware's load name. + namespace: The lid labware's namespace. + version: The lid labware's version. + location: The deck location at which lid(s) will be placed. + labware_ids: An optional list of identifiers to assign the labware. If None, + an identifier will be generated. + + Raises: + ModuleNotLoadedError: If `location` references a module ID + that doesn't point to a valid loaded module. + + Returns: + A list of LoadedLabwareData objects. + """ + definition_uri = uri_from_details( + load_name=load_name, + namespace=namespace, + version=version, + ) + try: + # Try to use existing definition in state. + definition = self._state_store.labware.get_definition_by_uri(definition_uri) + except LabwareDefinitionDoesNotExistError: + definition = await self._labware_data_provider.get_labware_definition( + load_name=load_name, + namespace=namespace, + version=version, + ) + + stack_limit = definition.stackLimit if definition.stackLimit is not None else 1 + if quantity > stack_limit: + raise ValueError( + f"Requested quantity {quantity} is greater than the stack limit of {stack_limit} provided by definition for {load_name}." + ) + + # Allow propagation of ModuleNotLoadedError. + if ( + isinstance(location, DeckSlotLocation) + and definition.parameters.isDeckSlotCompatible is not None + and not definition.parameters.isDeckSlotCompatible + ): + raise ValueError( + f"Lid Labware {load_name} cannot be loaded onto a Deck Slot." + ) + + load_labware_data_list = [] + for i in range(quantity): + load_labware_data_list.append( + LoadedLabwareData( + labware_id=self._model_utils.generate_id(), + definition=definition, + offsetId=None, + ) + ) + + return load_labware_data_list + async def configure_for_volume( self, pipette_id: str, volume: float, tip_overlap_version: Optional[str] ) -> LoadedConfigureForVolumeData: diff --git a/api/src/opentrons/protocol_engine/resources/labware_validation.py b/api/src/opentrons/protocol_engine/resources/labware_validation.py index 090723ffb7e..efe6d6daf65 100644 --- a/api/src/opentrons/protocol_engine/resources/labware_validation.py +++ b/api/src/opentrons/protocol_engine/resources/labware_validation.py @@ -32,6 +32,11 @@ def validate_definition_is_lid(definition: LabwareDefinition) -> bool: return LabwareRole.lid in definition.allowedRoles +def validate_definition_is_system(definition: LabwareDefinition) -> bool: + """Validate that one of the definition's allowed roles is `system`.""" + return LabwareRole.system in definition.allowedRoles + + def validate_labware_can_be_stacked( top_labware_definition: LabwareDefinition, below_labware_load_name: str ) -> bool: diff --git a/api/src/opentrons/protocol_engine/state/labware.py b/api/src/opentrons/protocol_engine/state/labware.py index 3f00ad14de7..70cb43c8403 100644 --- a/api/src/opentrons/protocol_engine/state/labware.py +++ b/api/src/opentrons/protocol_engine/state/labware.py @@ -156,7 +156,9 @@ def handle_action(self, action: Action) -> None: """Modify state in reaction to an action.""" for state_update in get_state_updates(action): self._add_loaded_labware(state_update) + self._add_loaded_lid_stack(state_update) self._set_labware_location(state_update) + self._set_labware_lid(state_update) if isinstance(action, AddLabwareOffsetAction): labware_offset = LabwareOffset.model_construct( @@ -221,6 +223,63 @@ def _add_loaded_labware(self, state_update: update_types.StateUpdate) -> None: displayName=display_name, ) + def _add_loaded_lid_stack(self, state_update: update_types.StateUpdate) -> None: + loaded_lid_stack_update = state_update.loaded_lid_stack + if loaded_lid_stack_update != update_types.NO_CHANGE: + # Add the stack object + stack_definition_uri = uri_from_details( + namespace=loaded_lid_stack_update.stack_object_definition.namespace, + load_name=loaded_lid_stack_update.stack_object_definition.parameters.loadName, + version=loaded_lid_stack_update.stack_object_definition.version, + ) + self.state.definitions_by_uri[ + stack_definition_uri + ] = loaded_lid_stack_update.stack_object_definition + self._state.labware_by_id[ + loaded_lid_stack_update.stack_id + ] = LoadedLabware.construct( + id=loaded_lid_stack_update.stack_id, + location=loaded_lid_stack_update.stack_location, + loadName=loaded_lid_stack_update.stack_object_definition.parameters.loadName, + definitionUri=stack_definition_uri, + offsetId=None, + displayName=None, + ) + + # Add the Lids on top of the stack object + for i in range(len(loaded_lid_stack_update.labware_ids)): + definition_uri = uri_from_details( + namespace=loaded_lid_stack_update.definition.namespace, + load_name=loaded_lid_stack_update.definition.parameters.loadName, + version=loaded_lid_stack_update.definition.version, + ) + + self._state.definitions_by_uri[ + definition_uri + ] = loaded_lid_stack_update.definition + + location = loaded_lid_stack_update.new_locations_by_id[ + loaded_lid_stack_update.labware_ids[i] + ] + + self._state.labware_by_id[ + loaded_lid_stack_update.labware_ids[i] + ] = LoadedLabware.construct( + id=loaded_lid_stack_update.labware_ids[i], + location=location, + loadName=loaded_lid_stack_update.definition.parameters.loadName, + definitionUri=definition_uri, + offsetId=None, + displayName=None, + ) + + def _set_labware_lid(self, state_update: update_types.StateUpdate) -> None: + labware_lid_update = state_update.labware_lid + if labware_lid_update != update_types.NO_CHANGE: + parent_labware_id = labware_lid_update.parent_labware_id + lid_id = labware_lid_update.lid_id + self._state.labware_by_id[parent_labware_id].lid_id = lid_id + def _set_labware_location(self, state_update: update_types.StateUpdate) -> None: labware_location_update = state_update.labware_location if labware_location_update != update_types.NO_CHANGE: @@ -441,21 +500,7 @@ def get_labware_stacking_maximum(self, labware: LabwareDefinition) -> int: If not defined within a labware, defaults to one. """ - stacking_quirks = { - "stackingMaxFive": 5, - "stackingMaxFour": 4, - "stackingMaxThree": 3, - "stackingMaxTwo": 2, - "stackingMaxOne": 1, - "stackingMaxZero": 0, - } - for quirk in stacking_quirks.keys(): - if ( - labware.parameters.quirks is not None - and quirk in labware.parameters.quirks - ): - return stacking_quirks[quirk] - return 1 + return labware.stackLimit if labware.stackLimit is not None else 1 def get_should_center_pipette_on_target_well(self, labware_id: str) -> bool: """True if a pipette moving to a well of this labware should center its body on the target. @@ -815,6 +860,11 @@ def raise_if_labware_inaccessible_by_pipette(self, labware_id: str) -> None: return self.raise_if_labware_inaccessible_by_pipette( labware_location.labwareId ) + elif labware.lid_id is not None: + raise errors.LocationNotAccessibleByPipetteError( + f"Cannot move pipette to {labware.loadName} " + "because labware is currently covered by a lid." + ) elif isinstance(labware_location, AddressableAreaLocation): if fixture_validation.is_staging_slot(labware_location.addressableAreaName): raise errors.LocationNotAccessibleByPipetteError( diff --git a/api/src/opentrons/protocol_engine/state/update_types.py b/api/src/opentrons/protocol_engine/state/update_types.py index 25b7802976c..568519b2784 100644 --- a/api/src/opentrons/protocol_engine/state/update_types.py +++ b/api/src/opentrons/protocol_engine/state/update_types.py @@ -11,6 +11,7 @@ from opentrons.protocol_engine.types import ( DeckPoint, LabwareLocation, + OnLabwareLocation, TipGeometry, AspiratedFluid, LiquidClassRecord, @@ -120,6 +121,40 @@ class LoadedLabwareUpdate: definition: LabwareDefinition +@dataclasses.dataclass +class LoadedLidStackUpdate: + """An update that loads a new lid stack.""" + + stack_id: str + """The unique ID of the Lid Stack Object.""" + + stack_object_definition: LabwareDefinition + "The System-only Labware Definition of the Lid Stack Object" + + stack_location: LabwareLocation + "The initial location of the Lid Stack Object." + + labware_ids: typing.List[str] + """The unique IDs of the new lids.""" + + new_locations_by_id: typing.Dict[str, OnLabwareLocation] + """Each lid's initial location keyed by Labware ID.""" + + definition: LabwareDefinition + "The Labware Definition of the Lid Labware(s) loaded." + + +@dataclasses.dataclass +class LabwareLidUpdate: + """An update that identifies a lid on a given parent labware.""" + + parent_labware_id: str + """The unique ID of the parent labware.""" + + lid_id: str + """The unique IDs of the new lids.""" + + @dataclasses.dataclass class LoadPipetteUpdate: """An update that loads a new pipette. @@ -301,6 +336,10 @@ class StateUpdate: loaded_labware: LoadedLabwareUpdate | NoChangeType = NO_CHANGE + loaded_lid_stack: LoadedLidStackUpdate | NoChangeType = NO_CHANGE + + labware_lid: LabwareLidUpdate | NoChangeType = NO_CHANGE + tips_used: TipsUsedUpdate | NoChangeType = NO_CHANGE liquid_loaded: LiquidLoadedUpdate | NoChangeType = NO_CHANGE @@ -442,6 +481,38 @@ def set_loaded_labware( ) return self + def set_loaded_lid_stack( + self: Self, + stack_id: str, + stack_object_definition: LabwareDefinition, + stack_location: LabwareLocation, + labware_definition: LabwareDefinition, + labware_ids: typing.List[str], + locations: typing.Dict[str, OnLabwareLocation], + ) -> Self: + """Add a new lid stack to state. See `LoadedLidStackUpdate`.""" + self.loaded_lid_stack = LoadedLidStackUpdate( + stack_id=stack_id, + stack_object_definition=stack_object_definition, + stack_location=stack_location, + definition=labware_definition, + labware_ids=labware_ids, + new_locations_by_id=locations, + ) + return self + + def set_lid( + self: Self, + parent_labware_id: str, + lid_id: str, + ) -> Self: + """Update the labware parent of a loaded or moved lid. See `LabwareLidUpdate`.""" + self.labware_lid = LabwareLidUpdate( + parent_labware_id=parent_labware_id, + lid_id=lid_id, + ) + return self + def set_load_pipette( self: Self, pipette_id: str, diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index b1388d58212..9d596adbaa8 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -804,6 +804,10 @@ class LoadedLabware(BaseModel): location: LabwareLocation = Field( ..., description="The labware's current location." ) + lid_id: Optional[str] = Field( + None, + description=("Labware ID of a Lid currently loaded on top of the labware."), + ) offsetId: Optional[str] = Field( None, description=( diff --git a/api/tests/opentrons/conftest.py b/api/tests/opentrons/conftest.py index 7be480cfe0b..20b8d3a4502 100755 --- a/api/tests/opentrons/conftest.py +++ b/api/tests/opentrons/conftest.py @@ -608,6 +608,7 @@ def minimal_labware_def() -> LabwareDefinition: "displayCategory": "other", "displayVolumeUnits": "mL", }, + "allowedRoles": ["labware"], "cornerOffsetFromSlot": {"x": 10, "y": 10, "z": 5}, "parameters": { "isTiprack": False, diff --git a/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py b/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py index 6b5065f98c9..2889a47cea9 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py @@ -756,6 +756,154 @@ def test_load_adapter_on_staging_slot( assert subject.get_slot_item(StagingSlotName.SLOT_B4) is result +def test_load_lid( + decoy: Decoy, + mock_engine_client: EngineClient, + subject: ProtocolCore, +) -> None: + """It should issue a LoadLid command.""" + mock_labware_core = decoy.mock(cls=LabwareCore) + decoy.when(mock_labware_core.labware_id).then_return("labware-id") + decoy.when( + mock_engine_client.state.labware.find_custom_labware_load_params() + ).then_return([EngineLabwareLoadParams("hello", "world", 654)]) + + decoy.when( + load_labware_params.resolve( + "some_labware", + "a_namespace", + 456, + [EngineLabwareLoadParams("hello", "world", 654)], + ) + ).then_return(("some_namespace", 9001)) + + decoy.when( + mock_engine_client.execute_command_without_recovery( + cmd.LoadLidParams( + location=OnLabwareLocation(labwareId="labware-id"), + loadName="some_labware", + namespace="some_namespace", + version=9001, + ) + ) + ).then_return( + commands.LoadLidResult( + labwareId="abc123", + definition=LabwareDefinition.model_construct(ordering=[]), # type: ignore[call-arg] + ) + ) + + decoy.when(mock_engine_client.state.labware.get_definition("abc123")).then_return( + LabwareDefinition.model_construct(ordering=[]) # type: ignore[call-arg] + ) + + result = subject.load_lid( + load_name="some_labware", + location=mock_labware_core, + namespace="a_namespace", + version=456, + ) + + assert isinstance(result, LabwareCore) + assert result.labware_id == "abc123" + assert subject.get_labware_cores() == [result] + + decoy.verify( + deck_conflict.check( + engine_state=mock_engine_client.state, + existing_labware_ids=[], + existing_module_ids=[], + existing_disposal_locations=[], + new_labware_id="abc123", + ) + ) + + decoy.when( + mock_engine_client.state.geometry.get_slot_item( + slot_name=DeckSlotName.SLOT_5, + ) + ).then_return( + LoadedLabware.model_construct(id="abc123") # type: ignore[call-arg] + ) + + assert subject.get_slot_item(DeckSlotName.SLOT_5) is result + + +def test_load_lid_stack( + decoy: Decoy, + mock_engine_client: EngineClient, + subject: ProtocolCore, +) -> None: + """It should issue a LoadLidStack command.""" + decoy.when( + mock_engine_client.state.labware.find_custom_labware_load_params() + ).then_return([EngineLabwareLoadParams("hello", "world", 654)]) + + decoy.when( + load_labware_params.resolve( + "some_labware", + "a_namespace", + 456, + [EngineLabwareLoadParams("hello", "world", 654)], + ) + ).then_return(("some_namespace", 9001)) + + decoy.when( + mock_engine_client.execute_command_without_recovery( + cmd.LoadLidStackParams( + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_5), + loadName="some_labware", + namespace="some_namespace", + version=9001, + quantity=5, + ) + ) + ).then_return( + commands.LoadLidStackResult( + stackLabwareId="abc123", + labwareIds=["1", "2", "3", "4", "5"], + definition=LabwareDefinition.model_construct(), # type: ignore[call-arg] + location=DeckSlotLocation(slotName=DeckSlotName.SLOT_5), + ) + ) + + decoy.when(mock_engine_client.state.labware.get_definition("abc123")).then_return( + LabwareDefinition.model_construct(ordering=[]) # type: ignore[call-arg] + ) + + result = subject.load_lid_stack( + load_name="some_labware", + location=DeckSlotName.SLOT_5, + namespace="a_namespace", + version=456, + quantity=5, + ) + + assert isinstance(result, LabwareCore) + assert result.labware_id == "abc123" + assert subject.get_labware_cores() == [result] + + decoy.verify( + deck_conflict.check( + engine_state=mock_engine_client.state, + existing_labware_ids=[], + existing_module_ids=[], + existing_disposal_locations=[], + new_labware_id="abc123", + ) + ) + + decoy.when( + mock_engine_client.state.geometry.get_slot_item( + slot_name=DeckSlotName.SLOT_5, + ) + ).then_return( + LoadedLabware.model_construct(id="abc123") # type: ignore[call-arg] + ) + + assert subject.get_slot_item(DeckSlotName.SLOT_5) is result + + def test_load_trash_bin( decoy: Decoy, mock_engine_client: EngineClient, diff --git a/api/tests/opentrons/protocol_api/test_protocol_context.py b/api/tests/opentrons/protocol_api/test_protocol_context.py index e804ac9dd11..80728b7820c 100644 --- a/api/tests/opentrons/protocol_api/test_protocol_context.py +++ b/api/tests/opentrons/protocol_api/test_protocol_context.py @@ -742,6 +742,115 @@ def test_load_labware_on_adapter( decoy.verify(mock_core_map.add(mock_labware_core, result), times=1) +@pytest.mark.parametrize("api_version", [APIVersion(2, 23)]) +def test_load_labware_with_lid( + decoy: Decoy, + mock_core: ProtocolCore, + mock_core_map: LoadedCoreMap, + api_version: APIVersion, + subject: ProtocolContext, +) -> None: + """It should create a labware with a lid on it using its execution core.""" + mock_labware_core = decoy.mock(cls=LabwareCore) + mock_lid_core = decoy.mock(cls=LabwareCore) + + decoy.when(mock_validation.ensure_lowercase_name("UPPERCASE_LABWARE")).then_return( + "lowercase_labware" + ) + + decoy.when(mock_validation.ensure_lowercase_name("UPPERCASE_LID")).then_return( + "lowercase_lid" + ) + decoy.when(mock_core.robot_type).then_return("OT-3 Standard") + decoy.when( + mock_validation.ensure_and_convert_deck_slot(42, api_version, "OT-3 Standard") + ).then_return(DeckSlotName.SLOT_C1) + + decoy.when( + mock_core.load_labware( + load_name="lowercase_labware", + location=DeckSlotName.SLOT_C1, + label="some_display_name", + namespace="some_namespace", + version=1337, + ) + ).then_return(mock_labware_core) + decoy.when(mock_lid_core.get_well_columns()).then_return([]) + + decoy.when( + mock_core.load_lid( + load_name="lowercase_lid", + location=mock_labware_core, + namespace="some_namespace", + version=None, + ) + ).then_return(mock_lid_core) + + decoy.when(mock_labware_core.get_name()).then_return("Full Name") + decoy.when(mock_labware_core.get_display_name()).then_return("Display Name") + decoy.when(mock_labware_core.get_well_columns()).then_return([]) + + result = subject.load_labware( + load_name="UPPERCASE_LABWARE", + location=42, + label="some_display_name", + namespace="some_namespace", + version=1337, + lid="UPPERCASE_LID", + ) + + assert isinstance(result, Labware) + assert result.name == "Full Name" + + decoy.verify(mock_core_map.add(mock_labware_core, result), times=1) + + +@pytest.mark.parametrize("api_version", [APIVersion(2, 23)]) +def test_load_lid_stack( + decoy: Decoy, + mock_core: ProtocolCore, + mock_core_map: LoadedCoreMap, + api_version: APIVersion, + subject: ProtocolContext, +) -> None: + """It should create a labware with a lid on it using its execution core.""" + mock_lid_core = decoy.mock(cls=LabwareCore) + + decoy.when(mock_validation.ensure_lowercase_name("UPPERCASE_LID")).then_return( + "lowercase_lid" + ) + + decoy.when(mock_core.robot_type).then_return("OT-3 Standard") + decoy.when( + mock_validation.ensure_and_convert_deck_slot(42, api_version, "OT-3 Standard") + ).then_return(DeckSlotName.SLOT_C1) + + decoy.when( + mock_core.load_lid_stack( + load_name="lowercase_lid", + location=DeckSlotName.SLOT_C1, + quantity=5, + namespace="some_namespace", + version=1337, + ) + ).then_return(mock_lid_core) + + decoy.when(mock_lid_core.get_name()).then_return("STACK_OBJECT") + decoy.when(mock_lid_core.get_display_name()).then_return("") + decoy.when(mock_lid_core.get_well_columns()).then_return([]) + + result = subject.load_lid_stack( + load_name="UPPERCASE_LID", + location=42, + quantity=5, + namespace="some_namespace", + version=1337, + ) + + assert isinstance(result, Labware) + assert result.name == "STACK_OBJECT" + + def test_loaded_labware( decoy: Decoy, mock_core_map: LoadedCoreMap, diff --git a/api/tests/opentrons/protocol_engine/state/test_labware_view_old.py b/api/tests/opentrons/protocol_engine/state/test_labware_view_old.py index 7ace6d767ad..ac92d5e5eaf 100644 --- a/api/tests/opentrons/protocol_engine/state/test_labware_view_old.py +++ b/api/tests/opentrons/protocol_engine/state/test_labware_view_old.py @@ -1414,25 +1414,25 @@ def test_raise_if_labware_cannot_be_stacked_on_labware_on_adapter() -> None: @pytest.mark.parametrize( argnames=[ "allowed_roles", - "stacking_quirks", + "stack_limit", "exception", ], argvalues=[ [ [LabwareRole.labware], - [], + 1, pytest.raises(errors.LabwareCannotBeStackedError), ], [ [LabwareRole.lid], - ["stackingMaxFive"], + 5, does_not_raise(), ], ], ) def test_labware_stacking_height_passes_or_raises( allowed_roles: List[LabwareRole], - stacking_quirks: List[str], + stack_limit: int, exception: ContextManager[None], ) -> None: """It should raise if the labware is stacked too high, and pass if the labware definition allowed this.""" @@ -1468,11 +1468,11 @@ def test_labware_stacking_height_passes_or_raises( allowedRoles=allowed_roles, parameters=Parameters.model_construct( format="irregular", - quirks=stacking_quirks, isTiprack=False, loadName="name", isMagneticModuleCompatible=False, ), + stackLimit=stack_limit, ) }, ) @@ -1482,7 +1482,6 @@ def test_labware_stacking_height_passes_or_raises( top_labware_definition=LabwareDefinition.model_construct( # type: ignore[call-arg] parameters=Parameters.model_construct( format="irregular", - quirks=stacking_quirks, isTiprack=False, loadName="name", isMagneticModuleCompatible=False, @@ -1490,6 +1489,7 @@ def test_labware_stacking_height_passes_or_raises( stackingOffsetWithLabware={ "test": SharedDataOverlapOffset(x=0, y=0, z=0) }, + stackLimit=stack_limit, ), bottom_labware_id="labware-id4", ) diff --git a/shared-data/command/schemas/11.json b/shared-data/command/schemas/11.json index ec0d854e5f9..ce6a1575062 100644 --- a/shared-data/command/schemas/11.json +++ b/shared-data/command/schemas/11.json @@ -2080,6 +2080,159 @@ "title": "LoadLabwareParams", "type": "object" }, + "LoadLidCreate": { + "description": "Load lid command creation request.", + "properties": { + "commandType": { + "const": "loadLid", + "default": "loadLid", + "enum": ["loadLid"], + "title": "Commandtype", + "type": "string" + }, + "intent": { + "$ref": "#/$defs/CommandIntent", + "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.", + "title": "Intent" + }, + "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.", + "title": "Key", + "type": "string" + }, + "params": { + "$ref": "#/$defs/LoadLidParams" + } + }, + "required": ["params"], + "title": "LoadLidCreate", + "type": "object" + }, + "LoadLidParams": { + "description": "Payload required to load a lid onto a labware.", + "properties": { + "loadName": { + "description": "Name used to reference a lid labware definition.", + "title": "Loadname", + "type": "string" + }, + "location": { + "anyOf": [ + { + "$ref": "#/$defs/DeckSlotLocation" + }, + { + "$ref": "#/$defs/ModuleLocation" + }, + { + "$ref": "#/$defs/OnLabwareLocation" + }, + { + "const": "offDeck", + "enum": ["offDeck"], + "type": "string" + }, + { + "$ref": "#/$defs/AddressableAreaLocation" + } + ], + "description": "Labware the lid should be loaded onto.", + "title": "Location" + }, + "namespace": { + "description": "The namespace the lid labware definition belongs to.", + "title": "Namespace", + "type": "string" + }, + "version": { + "description": "The lid labware definition version.", + "title": "Version", + "type": "integer" + } + }, + "required": ["location", "loadName", "namespace", "version"], + "title": "LoadLidParams", + "type": "object" + }, + "LoadLidStackCreate": { + "description": "Load lid stack command creation request.", + "properties": { + "commandType": { + "const": "loadLidStack", + "default": "loadLidStack", + "enum": ["loadLidStack"], + "title": "Commandtype", + "type": "string" + }, + "intent": { + "$ref": "#/$defs/CommandIntent", + "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.", + "title": "Intent" + }, + "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.", + "title": "Key", + "type": "string" + }, + "params": { + "$ref": "#/$defs/LoadLidStackParams" + } + }, + "required": ["params"], + "title": "LoadLidStackCreate", + "type": "object" + }, + "LoadLidStackParams": { + "description": "Payload required to load a lid stack onto a location.", + "properties": { + "loadName": { + "description": "Name used to reference a lid labware definition.", + "title": "Loadname", + "type": "string" + }, + "location": { + "anyOf": [ + { + "$ref": "#/$defs/DeckSlotLocation" + }, + { + "$ref": "#/$defs/ModuleLocation" + }, + { + "$ref": "#/$defs/OnLabwareLocation" + }, + { + "const": "offDeck", + "enum": ["offDeck"], + "type": "string" + }, + { + "$ref": "#/$defs/AddressableAreaLocation" + } + ], + "description": "Location the lid stack should be loaded into.", + "title": "Location" + }, + "namespace": { + "description": "The namespace the lid labware definition belongs to.", + "title": "Namespace", + "type": "string" + }, + "quantity": { + "description": "The quantity of lids to load.", + "title": "Quantity", + "type": "integer" + }, + "version": { + "description": "The lid labware definition version.", + "title": "Version", + "type": "integer" + } + }, + "required": ["location", "loadName", "namespace", "version", "quantity"], + "title": "LoadLidStackParams", + "type": "object" + }, "LoadLiquidClassCreate": { "description": "Load Liquid Class command creation request.", "properties": { @@ -5580,6 +5733,8 @@ "home": "#/$defs/HomeCreate", "liquidProbe": "#/$defs/LiquidProbeCreate", "loadLabware": "#/$defs/LoadLabwareCreate", + "loadLid": "#/$defs/LoadLidCreate", + "loadLidStack": "#/$defs/LoadLidStackCreate", "loadLiquid": "#/$defs/LoadLiquidCreate", "loadLiquidClass": "#/$defs/LoadLiquidClassCreate", "loadModule": "#/$defs/LoadModuleCreate", @@ -5696,6 +5851,12 @@ { "$ref": "#/$defs/LoadPipetteCreate" }, + { + "$ref": "#/$defs/LoadLidStackCreate" + }, + { + "$ref": "#/$defs/LoadLidCreate" + }, { "$ref": "#/$defs/MoveLabwareCreate" }, diff --git a/shared-data/js/labware.ts b/shared-data/js/labware.ts index a085ffac89c..20d41c3a697 100644 --- a/shared-data/js/labware.ts +++ b/shared-data/js/labware.ts @@ -112,7 +112,7 @@ import opentronsFlex96Tiprack50UlV1Uncasted from '../labware/definitions/2/opent import opentronsFlex96TiprackAdapterV1Uncasted from '../labware/definitions/2/opentrons_flex_96_tiprack_adapter/1.json' import opentronsFlexDeckRiserV1Uncasted from '../labware/definitions/2/opentrons_flex_deck_riser/1.json' import opentronsFlexLidAbsorbancePlateReaderModuleV1Uncasted from '../labware/definitions/2/opentrons_flex_lid_absorbance_plate_reader_module/1.json' -import opentronsToughPcrAutoSealingLidV1Uncasted from '../labware/definitions/2/opentrons_tough_pcr_auto_sealing_lid/1.json' +import opentronsToughPcrAutoSealingLidV1Uncasted from '../labware/definitions/3/opentrons_tough_pcr_auto_sealing_lid/1.json' import opentronsUniversalFlatAdapterV1Uncasted from '../labware/definitions/2/opentrons_universal_flat_adapter/1.json' import opentronsUniversalFlatAdapterCorning384Wellplate112UlFlatV1Uncasted from '../labware/definitions/2/opentrons_universal_flat_adapter_corning_384_wellplate_112ul_flat/1.json' import opentrons96DeepWellTempModAdapterV1Uncasted from '../labware/definitions/2/opentrons_96_deep_well_temp_mod_adapter/1.json' diff --git a/shared-data/js/types.ts b/shared-data/js/types.ts index 6abe511bb8f..dab252a54ec 100644 --- a/shared-data/js/types.ts +++ b/shared-data/js/types.ts @@ -238,6 +238,7 @@ export type LabwareRoles = | 'fixture' | 'maintenance' | 'lid' + | 'system' // NOTE: must be synced with shared-data/labware/schemas/2.json export interface LabwareDefinition2 { diff --git a/shared-data/labware/definitions/2/opentrons_tough_pcr_auto_sealing_lid/1.json b/shared-data/labware/definitions/3/opentrons_tough_pcr_auto_sealing_lid/1.json similarity index 89% rename from shared-data/labware/definitions/2/opentrons_tough_pcr_auto_sealing_lid/1.json rename to shared-data/labware/definitions/3/opentrons_tough_pcr_auto_sealing_lid/1.json index e86f24c6015..479fd00b772 100644 --- a/shared-data/labware/definitions/2/opentrons_tough_pcr_auto_sealing_lid/1.json +++ b/shared-data/labware/definitions/3/opentrons_tough_pcr_auto_sealing_lid/1.json @@ -30,14 +30,14 @@ }, "parameters": { "format": "irregular", - "quirks": ["stackingMaxFive"], + "quirks": [], "isTiprack": false, "isMagneticModuleCompatible": false, "loadName": "opentrons_tough_pcr_auto_sealing_lid" }, "namespace": "opentrons", "version": 1, - "schemaVersion": 2, + "schemaVersion": 3, "stackingOffsetWithModule": { "thermocyclerModuleV2": { "x": 0, @@ -77,6 +77,12 @@ "z": 34 } }, + "stackLimit": 5, + "compatibleParentLabware": [ + "armadillo_96_wellplate_200ul_pcr_full_skirt", + "opentrons_96_wellplate_200ul_pcr_full_skirt", + "opentrons_tough_pcr_auto_sealing_lid" + ], "gripForce": 15, "gripHeightFromLabwareBottom": 7.91, "gripperOffsets": { diff --git a/shared-data/labware/definitions/3/protocol_engine_lid_stack_object/1.json b/shared-data/labware/definitions/3/protocol_engine_lid_stack_object/1.json new file mode 100644 index 00000000000..f8aeb02350c --- /dev/null +++ b/shared-data/labware/definitions/3/protocol_engine_lid_stack_object/1.json @@ -0,0 +1,41 @@ +{ + "allowedRoles": ["system"], + "ordering": [], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "metadata": { + "displayName": "Protocol Engine Lid Stack", + "displayCategory": "system", + "displayVolumeUnits": "\u00b5L", + "tags": [] + }, + "dimensions": { + "xDimension": 0, + "yDimension": 0, + "zDimension": 0 + }, + "wells": {}, + "groups": [ + { + "metadata": {}, + "wells": [] + } + ], + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "parameters": { + "format": "irregular", + "quirks": [], + "isTiprack": false, + "isMagneticModuleCompatible": false, + "loadName": "protocol_engine_lid_stack_object" + }, + "namespace": "opentrons", + "version": 1, + "schemaVersion": 3 +} diff --git a/shared-data/labware/schemas/2.json b/shared-data/labware/schemas/2.json index b60a731c242..51ff453829a 100644 --- a/shared-data/labware/schemas/2.json +++ b/shared-data/labware/schemas/2.json @@ -338,7 +338,14 @@ "description": "Allowed behaviors and usage of a labware in a protocol.", "items": { "type": "string", - "enum": ["labware", "adapter", "fixture", "maintenance", "lid"] + "enum": [ + "labware", + "adapter", + "fixture", + "maintenance", + "lid", + "system" + ] } }, "stackingOffsetWithLabware": { diff --git a/shared-data/labware/schemas/3.json b/shared-data/labware/schemas/3.json index ee6a023877a..d386cbfbb8d 100644 --- a/shared-data/labware/schemas/3.json +++ b/shared-data/labware/schemas/3.json @@ -406,6 +406,11 @@ "description": "Flag marking whether a labware is compatible by default with the Magnetic Module", "type": "boolean" }, + "isDeckSlotCompatible": { + "description": "Flag marking whether a labware is compatible by with being placed or loaded in a base deck slot, defaults to true.", + "type": "boolean", + "default": true + }, "magneticModuleEngageHeight": { "description": "Distance to move magnetic module magnets to engage", "$ref": "#/definitions/positiveNumber" @@ -563,7 +568,14 @@ "description": "Allowed behaviors and usage of a labware in a protocol.", "items": { "type": "string", - "enum": ["labware", "adapter", "fixture", "maintenance", "lid"] + "enum": [ + "labware", + "adapter", + "fixture", + "maintenance", + "lid", + "system" + ] } }, "stackingOffsetWithLabware": { @@ -580,6 +592,18 @@ "$ref": "#/definitions/coordinates" } }, + "stackLimit": { + "type": "number", + "description": "The limit representing the maximum stack size for a given labware.", + "additionalProperties": false + }, + "compatibleParentLabware": { + "type": "array", + "description": "Array of parent Labware on which a labware may be loaded, primarily the labware which owns a lid.", + "items": { + "type": "string" + } + }, "gripperOffsets": { "type": "object", "description": "Offsets to add when picking up or dropping another labware stacked atop this one. Do not use this to adjust the position of the gripper paddles relative to this labware or the child labware; use `gripHeightFromLabwareBottom` on this definition or the child's definition for that.", diff --git a/shared-data/python/opentrons_shared_data/labware/labware_definition.py b/shared-data/python/opentrons_shared_data/labware/labware_definition.py index 994d4e743eb..c25f37962c1 100644 --- a/shared-data/python/opentrons_shared_data/labware/labware_definition.py +++ b/shared-data/python/opentrons_shared_data/labware/labware_definition.py @@ -112,6 +112,7 @@ class DisplayCategory(str, Enum): adapter = "adapter" other = "other" lid = "lid" + system = "system" class LabwareRole(str, Enum): @@ -120,6 +121,7 @@ class LabwareRole(str, Enum): adapter = "adapter" maintenance = "maintenance" lid = "lid" + system = "system" class Metadata(BaseModel): @@ -180,6 +182,11 @@ class Parameters(BaseModel): magneticModuleEngageHeight: Optional[_NonNegativeNumber] = Field( None, description="Distance to move magnetic module magnets to engage" ) + isDeckSlotCompatible: Optional[bool] = Field( + None, + description="Flag marking whether a labware is compatible with placement" + " or load into a base deck slot, will be treated as true if unspecified.", + ) class Dimensions(BaseModel): @@ -730,3 +737,12 @@ class LabwareDefinition(BaseModel): None, description="A dictionary holding all unique inner well geometries in a labware.", ) + stackLimit: Optional[int] = Field( + None, + description="The limit representing the maximum stack size for a given labware," + " defaults to 1 when unspecified indicating a single labware with no labware below it.", + ) + compatibleParentLabware: Optional[List[str]] = Field( + None, + description="List of parent Labware on which a labware may be loaded, primarily the labware which owns a lid.", + ) diff --git a/shared-data/python/opentrons_shared_data/labware/types.py b/shared-data/python/opentrons_shared_data/labware/types.py index 59634a26f54..91e9b12fc96 100644 --- a/shared-data/python/opentrons_shared_data/labware/types.py +++ b/shared-data/python/opentrons_shared_data/labware/types.py @@ -23,6 +23,7 @@ Literal["adapter"], Literal["other"], Literal["lid"], + Literal["system"], ] LabwareFormat = Union[ @@ -39,6 +40,7 @@ Literal["adapter"], Literal["maintenance"], Literal["lid"], + Literal["system"], ] @@ -58,6 +60,7 @@ class LabwareParameters(TypedDict, total=False): isTiprack: bool loadName: str isMagneticModuleCompatible: bool + isDeckSlotCompatible: bool quirks: List[str] tipLength: float tipOverlap: float @@ -140,3 +143,5 @@ class LabwareDefinition(TypedDict): gripForce: NotRequired[float] gripHeightFromLabwareBottom: NotRequired[float] innerLabwareGeometry: NotRequired[Dict[str, InnerWellGeometry]] + compatibleParentLabware: NotRequired[List[str]] + stackLimit: NotRequired[int]