From a143a91357b7229be9719260389e05ee4a96b9fc Mon Sep 17 00:00:00 2001 From: Sanniti Date: Tue, 10 Dec 2024 20:31:48 -0500 Subject: [PATCH 01/11] added aspirate executor --- .../core/engine/complex_commands_executor.py | 320 ++++++++++++++++++ .../protocol_api/core/engine/instrument.py | 65 +++- .../opentrons/protocol_api/core/instrument.py | 3 +- .../protocol_api/instrument_context.py | 13 +- api/tests/opentrons/conftest.py | 148 +++++++- .../engine/test_complex_commands_executor.py | 123 +++++++ .../protocol_api/test_instrument_context.py | 17 +- 7 files changed, 668 insertions(+), 21 deletions(-) create mode 100644 api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py create mode 100644 api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py diff --git a/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py b/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py new file mode 100644 index 00000000000..215086945b4 --- /dev/null +++ b/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py @@ -0,0 +1,320 @@ +"""Executor for liquid class based complex commands.""" +from __future__ import annotations +from dataclasses import dataclass +from typing import Union, TYPE_CHECKING + +from opentrons_shared_data.liquid_classes.liquid_class_definition import ( + PositionReference, + Coordinate, +) + +from opentrons.protocol_api._liquid_properties import ( + SingleDispenseProperties, + MultiDispenseProperties, + Submerge, + RetractAspirate, + RetractDispense, + DelayProperties, + TransferProperties, +) +from opentrons.types import Location, Point + +if TYPE_CHECKING: + from .well import WellCore + from .instrument import InstrumentCore + + +@dataclass +class _TargetPosition: + position_reference: PositionReference + offset: Coordinate + + +@dataclass +class _TargetAsLocationAndWell: + location: Location + well: WellCore + + +@dataclass +class _MixData: + volume: float + repetitions: int + aspirate_flow_rate: float + aspirate_delay: DelayProperties + dispense_flow_rate: float + dispense_delay: DelayProperties + + +class LiquidClassTransferExecutor: + def __init__( + self, + instrument_core: InstrumentCore, + transfer_properties: TransferProperties, + ) -> None: + self._instrument = instrument_core + self._transfer_properties = transfer_properties + + def aspirate( + self, + volume: float, + source: WellCore, + ) -> None: + """Execute aspiration steps. + + 1. Submerge + 2. Mix + 3. pre-wet + - 1 combo of aspirate + dispense at the same flow rate as specified in asp & disp and the delays in asp & disp + - Use the target volume/ volume we will be aspirating + - No push out + - Not pre-wet for consolidation + 4. Aspirate + - Aspirate with provided flow rate + 5. Delay- wait inside the liquid + 6. Aspirate retract + """ + aspirate_properties = self._transfer_properties.aspirate + dispense_properties = self._transfer_properties.dispense + aspirate_location_and_well = _TargetAsLocationAndWell( + location=location_from_position_reference_and_offset( + well=source, + position_reference=aspirate_properties.position_reference, + offset=aspirate_properties.offset, + ), + well=source, + ) + + self._submerge( + target_location_and_well=aspirate_location_and_well, + submerge_props=aspirate_properties.submerge, + ) + mix_props = aspirate_properties.mix + if mix_props.enabled: + # This would have been validated in the liquid class. Assertion only for mypy purposes + assert mix_props.repetitions is not None and mix_props.volume is not None + self._mix( + target_location_and_well=aspirate_location_and_well, + mix_data=_MixData( + repetitions=mix_props.repetitions, + volume=mix_props.volume, + aspirate_flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume( + mix_props.volume + ), + aspirate_delay=aspirate_properties.delay, + dispense_flow_rate=dispense_properties.flow_rate_by_volume.get_for_volume( + mix_props.volume + ), + dispense_delay=dispense_properties.delay, + ), + ) + if aspirate_properties.pre_wet: + self._mix( + target_location_and_well=aspirate_location_and_well, + mix_data=_MixData( + repetitions=1, + volume=volume, + aspirate_flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume( + volume + ), + aspirate_delay=aspirate_properties.delay, + dispense_flow_rate=dispense_properties.flow_rate_by_volume.get_for_volume( + volume + ), + dispense_delay=dispense_properties.delay, + ), + ) + + self._instrument.aspirate( + location=aspirate_location_and_well.location, + well_core=source, + volume=volume, + rate=1, + flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume(volume), + in_place=True, + ) + if aspirate_properties.delay.enabled: + asp_delay_duration = aspirate_properties.delay.duration + # This would have been validated in the liquid class. + # Assertion only for mypy purposes + assert asp_delay_duration is not None + self._instrument.delay(asp_delay_duration) + self._retract(retract_props=aspirate_properties.retract) + + def single_dispense( + self, + volume: float, + source: WellCore, + dest: WellCore, + dispense_properties: SingleDispenseProperties, + ) -> None: + """Execute single-dispense steps. + + 1. Move pipette to the ‘submerge’ position with normal speed. + - The pipette will move in an arc- move to max z height of labware (if asp & disp are in same labware) or max z height of all labware (if asp & disp are in separate labware) + 2. Air gap removal: + - If dispense location is above the meniscus, DO NOT remove air gap (it will be dispensed along with rest of the liquid later). All other scenarios, remove the air gap by doing a dispense + - Flow rate = min(dispenseFlowRate, (airGapByVolume)/sec) + 3. Use the post-dispense delay + 4. Move to the dispense position at the specified ‘submerge’ speed (even if we might not be moving into the liquid) + 5. Do a delay + 6. Dispense: + - Dispense at the specified flow rate. + - Do a push out as specified ONLY IF there is no mix following the dispense AND the tip is empty. + Volume for push out is the volume being dispensed. So if we are dispensing 50uL, use pushOutByVolume[50] as push out volume. + 7. Delay + 8. Mix using the same flow rate and delays as specified for asp+disp, with the volume and the number of repetitions specified. Use the delays in asp & disp. + - If the dispense position is outside the liquid, then raise error if mix is enabled. + - If the user wants to perform a mix then they should specify a dispense position that’s inside the liquid OR do mix() on the wells after transfer. + - Do push out at the last dispense. + + """ + + def multi_dispense(self, dispense_properties: MultiDispenseProperties) -> None: + """Execute multi-dispense steps.""" + + def _submerge( + self, + target_location_and_well: _TargetAsLocationAndWell, + submerge_props: Submerge, + ) -> None: + """Execute submerge steps. + + 1. move to position shown by positionReference + offset (should practically be a point outside/above the liquid). + Should raise an error if this point is inside the liquid? + For liquid meniscus this is easy to tell. Can’t be below meniscus + For reference pos of anything else, do not allow submerge position to be below aspirate position + 2. move to aspirate position at desired speed + 3. delay + """ + # TODO: compare submerge start position and aspirate position and raise error if incompatible + self._instrument.move_to( + location=location_from_position_reference_and_offset( + target_location_and_well.well, + submerge_props.position_reference, + submerge_props.offset, + ), + well_core=target_location_and_well.well, + force_direct=False, + minimum_z_height=None, + speed=None, + ) + self._instrument.move_to( + location=target_location_and_well.location, + well_core=target_location_and_well.well, + force_direct=True, + minimum_z_height=None, + speed=submerge_props.speed, + ) + if submerge_props.delay.enabled: + self._instrument.delay(submerge_props.delay.duration) + + def _retract(self, retract_props: Union[RetractAspirate, RetractDispense]) -> None: + """Execute retraction steps. + + Retract aspirate: + 1. Move TO the position reference+offset AT the specified speed + Raise error if retract is below aspirate position or below the meniscus + 2. Delay + 3. Touch tip + - Move to the Z offset position + - Touch tip to the sides at the specified speed (tip moves back to the center as part of touch tip) + - Return back to the retract position + 4. Air gap + - Air gap volume depends on the amount of liquid in the pipette + So if total aspirated volume is 20, use the value for airGapByVolume[20] + Flow rate = min(aspirateFlowRate, (airGapByVolume)/sec) + 5. Use post-aspirate delay + + Retract dispense: + 1. Position ref+offset is the ending position. Move to this position using specified speed + 2. If blowout is enabled and “destination” + - Do blow-out (at the retract position) + - Leave plunger down + 3. Touch-tip + 4. If not ready-to-aspirate + - Prepare-to-aspirate (at the retract position) + 5. Air-gap (at the retract position) + - This air gap is for preventing any stray droplets from falling while moving the pipette. + It will be performed out of caution even if we just did a blow_out and should *hypothetically* + have no liquid left in the tip. + - This air gap will be removed at the next aspirate. + If this is the last step of the transfer, and we aren't dropping the tip off, + then the air gap will be left as is(?). + 6. If blowout is “source” or “trash” + - Move to location (top of Well) + - Do blow-out (top of well) + - Do touch-tip (?????) (only if it’s in a non-trash location) + - Prepare-to-aspirate (top of well) + - Do air-gap (top of well) + 7. If drop tip, move to drop tip location, drop tip + """ + + def _mix( + self, + target_location_and_well: _TargetAsLocationAndWell, + mix_data: _MixData, + ) -> None: + """Execute mix steps. + + 1. Use same flow rates and delays as aspirate and dispense + 2. Do [(aspirate + dispense) x repetitions] at the same position + 3. Do NOT push out at the end of dispense + 4. USE the delay property from aspirate & dispense during mix as well (flow rate and delay are coordinated with each other) + 5. Do not mix during consolidation + NOTE: For most of our built-in definitions, we will keep _mix_ off because it is a very application specific thing. + We should mention in our docs that users should adjust this property according to their application. + """ + for n in range(mix_data.repetitions): + # TODO: figure out what to set is_meniscus as + self._instrument.aspirate( + location=target_location_and_well.location, + well_core=target_location_and_well.well, + volume=mix_data.volume, + rate=1, + flow_rate=mix_data.aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ) + if mix_data.aspirate_delay.enabled: + asp_delay_duration = mix_data.aspirate_delay.duration + # This would have been validated in the liquid class. + # Assertion only for mypy purposes + assert asp_delay_duration is not None + self._instrument.delay(asp_delay_duration) + self._instrument.dispense( + location=target_location_and_well.location, + well_core=target_location_and_well.well, + volume=mix_data.volume, + rate=1, + flow_rate=mix_data.dispense_flow_rate, + in_place=True, + push_out=None, # TODO: check if this should be 0 instead + is_meniscus=None, + ) + if mix_data.dispense_delay.enabled: + disp_delay_duration = mix_data.dispense_delay.duration + assert disp_delay_duration is not None + self._instrument.delay(disp_delay_duration) + + +def location_from_position_reference_and_offset( + well: WellCore, + position_reference: PositionReference, + offset: Coordinate, +) -> Location: + """Get position in `Location` type, given the well, the position reference and offset.""" + match position_reference: + case PositionReference.WELL_TOP: + reference_point = well.get_top(0) + case PositionReference.WELL_BOTTOM: + reference_point = well.get_bottom(0) + case PositionReference.WELL_CENTER: + reference_point = well.get_center() + case PositionReference.LIQUID_MENISCUS: + raise NotImplementedError( + "Liquid transfer using liquid-meniscus relative positioning is not yet implemented" + ) + case _: + raise ValueError(f"Unknown position reference {position_reference}") + return Location(reference_point + Point(offset.x, offset.y, offset.z), labware=None) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index 010f3110fdb..d4bffd5ed76 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -8,7 +8,11 @@ from opentrons.hardware_control.dev_types import PipetteDict from opentrons.protocols.api_support.util import FlowRates, find_value_for_api_version from opentrons.protocols.api_support.types import APIVersion -from opentrons.protocols.advanced_control.transfers.common import TransferTipPolicyV2 +from opentrons.protocols.advanced_control.transfers.common import ( + TransferTipPolicyV2, + check_valid_volume_parameters, + expand_for_volume_constraints, +) from opentrons.protocol_engine import commands as cmd from opentrons.protocol_engine import ( DeckPoint, @@ -40,12 +44,14 @@ from . import overlap_versions, pipette_movement_conflict from .well import WellCore +from .complex_commands_executor import LiquidClassTransferExecutor from ..instrument import AbstractInstrument from ...disposal_locations import TrashBin, WasteChute if TYPE_CHECKING: from .protocol import ProtocolCore from opentrons.protocol_api._liquid import LiquidClass + from opentrons.protocol_api._liquid_properties import TransferProperties _DISPENSE_VOLUME_VALIDATION_ADDED_IN = APIVersion(2, 17) @@ -892,16 +898,67 @@ def load_liquid_class( ) return result.liquidClassId + def get_next_tip(self) -> NextTipInfo: + """Get the next tip to pick up.""" + def transfer_liquid( self, - liquid_class_id: str, + liquid_class: LiquidClass, volume: float, source: List[WellCore], dest: List[WellCore], new_tip: TransferTipPolicyV2, + tiprack_uri: str, trash_location: Union[WellCore, Location, TrashBin, WasteChute], ) -> None: """Execute transfer using liquid class properties.""" + liquid_class_id = self.load_liquid_class( + liquid_class=liquid_class, + pipette_load_name=self.name, + tiprack_uri=tiprack_uri, + ) + transfer_props = liquid_class.get_for( + pipette=pipette_load_name, tiprack=tiprack_uri + ) + aspirate_props = transfer_props.aspirate + dispense_props = transfer_props.dispense + + check_valid_volume_parameters( + disposal_volume=0, # No disposal volume for 1-to-1 transfer + air_gap=aspirate_props.retract.air_gap_by_volume.get_for_volume(volume), + max_volume=self.get_max_volume(), + ) + source_dest_per_volume_step = expand_for_volume_constraints( + volumes=[volume for _ in range(len(source))], + targets=zip(source, dest), + max_volume=self.get_max_volume(), + ) + transfer_executer = LiquidClassTransferExecutor(instrument_core=self) + if new_tip == TransferTipPolicyV2.ONCE: + # TODO: update this once getNextTip is implemented + next_tip = self.get_next_tip() + self.pick_up_tip(next_tip) + for step_volume, (src, dest) in source_dest_per_volume_step: + if new_tip == TransferTipPolicyV2.ALWAYS: + # TODO: update this once getNextTip is implemented + next_tip = self.get_next_tip() + self.pick_up_tip(next_tip) + + transfer_executer.aspirate(aspirate_props) + if new_tip == TransferTipPolicyV2.ALWAYS: + if isinstance(trash_location, (TrashBin, WasteChute)): + self.drop_tip_in_disposal_location( + disposal_location=trash_location, + home_after=False, + alternate_tip_drop=True, + ) + elif isinstance(trash_location, Location): + self.drop_tip( + location=trash_location, + well_core=trash_location.labware.as_well()._core, + home_after=False, + alternate_drop_location=True, + ) def retract(self) -> None: """Retract this instrument to the top of the gantry.""" @@ -994,3 +1051,7 @@ def nozzle_configuration_valid_for_lld(self) -> bool: return self._engine_client.state.pipettes.get_nozzle_configuration_supports_lld( self.pipette_id ) + + def delay(self, seconds: float) -> None: + """Call a protocol delay.""" + self._protocol_core.delay(seconds=seconds, msg=None) diff --git a/api/src/opentrons/protocol_api/core/instrument.py b/api/src/opentrons/protocol_api/core/instrument.py index bc1ec3669df..aac6decb784 100644 --- a/api/src/opentrons/protocol_api/core/instrument.py +++ b/api/src/opentrons/protocol_api/core/instrument.py @@ -326,11 +326,12 @@ def load_liquid_class( @abstractmethod def transfer_liquid( self, - liquid_class_id: str, + liquid_class: LiquidClass, volume: float, source: List[WellCoreType], dest: List[WellCoreType], new_tip: TransferTipPolicyV2, + tiprack_uri: str, trash_location: Union[WellCoreType, types.Location, TrashBin, WasteChute], ) -> None: """Transfer a liquid from source to dest according to liquid class properties.""" diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 8cc3e0bd14e..191f592a3a2 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -1523,7 +1523,7 @@ def transfer_liquid( new_tip: TransferTipPolicyV2Type = "once", tip_drop_location: Optional[ Union[types.Location, labware.Well, TrashBin, WasteChute] - ] = None, # Maybe call this 'tip_drop_location' which is similar to PD + ] = None, ) -> InstrumentContext: """Transfer liquid from source to dest using the specified liquid class properties. @@ -1564,6 +1564,7 @@ def transfer_liquid( else: tiprack = self._last_tip_picked_up_from.parent else: + # TODO: update this with getNextTip result from engine tiprack, well = labware.next_available_tip( starting_tip=self.starting_tip, tip_racks=self.tip_racks, @@ -1592,23 +1593,17 @@ def transfer_liquid( tip_drop_location=_trash_location ) ) - liquid_class_id = self._core.load_liquid_class( - liquid_class=liquid_class, - pipette_load_name=self.name, - tiprack_uri=tiprack.uri, - ) - self._core.transfer_liquid( - liquid_class_id=liquid_class_id, + liquid_class=liquid_class, volume=volume, source=[well._core for well in flat_sources_list], dest=[well._core for well in flat_dests_list], new_tip=valid_new_tip, + tiprack_uri=tiprack.uri, trash_location=checked_trash_location._core if isinstance(checked_trash_location, labware.Well) else checked_trash_location, ) - return self @requires_version(2, 0) diff --git a/api/tests/opentrons/conftest.py b/api/tests/opentrons/conftest.py index 7be480cfe0b..e4f980e36a6 100755 --- a/api/tests/opentrons/conftest.py +++ b/api/tests/opentrons/conftest.py @@ -54,6 +54,11 @@ MixProperties, TouchTipProperties, BlowoutProperties, + MixParams, + LiquidClassTouchTipParams, + MultiDispenseProperties, + BlowoutParams, + BlowoutLocation, ) from opentrons_shared_data.deck.types import ( RobotModel, @@ -810,7 +815,7 @@ def minimal_liquid_class_def2() -> LiquidClassSchemaV1: tiprack="opentrons_flex_96_tiprack_50ul", aspirate=AspirateProperties( submerge=Submerge( - positionReference=PositionReference.LIQUID_MENISCUS, + positionReference=PositionReference.WELL_TOP, offset=Coordinate(x=0, y=0, z=-5), speed=100, delay=DelayProperties( @@ -865,3 +870,144 @@ def minimal_liquid_class_def2() -> LiquidClassSchemaV1: ) ], ) + + +@pytest.fixture +def maximal_liquid_class_def() -> LiquidClassSchemaV1: + """Return a liquid class def with all properties enabled.""" + return LiquidClassSchemaV1( + liquidClassName="test_water", + displayName="Test Water", + schemaVersion=1, + namespace="opentrons", + byPipette=[ + ByPipetteSetting( + pipetteModel="flex_1channel_50", + byTipType=[ + ByTipTypeSetting( + tiprack="opentrons_flex_96_tiprack_50ul", + aspirate=AspirateProperties( + submerge=Submerge( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=1, y=2, z=3), + speed=100, + delay=DelayProperties( + enable=True, params=DelayParams(duration=10.0) + ), + ), + retract=RetractAspirate( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=3, y=2, z=1), + speed=50, + airGapByVolume=[(1.0, 0.1), (49.9, 0.1), (50.0, 0.0)], + touchTip=TouchTipProperties( + enable=True, + params=LiquidClassTouchTipParams( + zOffset=-1, mmToEdge=0.5, speed=30 + ), + ), + delay=DelayProperties( + enable=True, params=DelayParams(duration=20) + ), + ), + positionReference=PositionReference.WELL_BOTTOM, + offset=Coordinate(x=10, y=20, z=30), + flowRateByVolume=[(1.0, 35.0), (10.0, 24.0), (50.0, 35.0)], + correctionByVolume=[(0.0, 0.0)], + preWet=True, + mix=MixProperties( + enable=True, params=MixParams(repetitions=1, volume=50) + ), + delay=DelayProperties( + enable=True, params=DelayParams(duration=0.2) + ), + ), + singleDispense=SingleDispenseProperties( + submerge=Submerge( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=30, y=20, z=10), + speed=100, + delay=DelayProperties( + enable=True, params=DelayParams(duration=0.0) + ), + ), + retract=RetractDispense( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=11, y=22, z=33), + speed=50, + airGapByVolume=[(1.0, 0.1), (49.9, 0.1), (50.0, 0.0)], + blowout=BlowoutProperties( + enable=True, + params=BlowoutParams( + location=BlowoutLocation.SOURCE, + flowRate=100, + ), + ), + touchTip=TouchTipProperties( + enable=True, + params=LiquidClassTouchTipParams( + zOffset=-1, mmToEdge=0.5, speed=30 + ), + ), + delay=DelayProperties( + enable=False, params=DelayParams(duration=0) + ), + ), + positionReference=PositionReference.WELL_BOTTOM, + offset=Coordinate(x=33, y=22, z=11), + flowRateByVolume=[(1.0, 50.0)], + correctionByVolume=[(0.0, 0.0)], + mix=MixProperties( + enable=True, params=MixParams(repetitions=1, volume=50) + ), + pushOutByVolume=[ + (1.0, 7.0), + (4.999, 7.0), + (5.0, 2.0), + (10.0, 2.0), + (50.0, 2.0), + ], + delay=DelayProperties( + enable=True, params=DelayParams(duration=0.2) + ), + ), + multiDispense=MultiDispenseProperties( + submerge=Submerge( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=0, y=0, z=2), + speed=100, + delay=DelayProperties( + enable=False, params=DelayParams(duration=0.0) + ), + ), + retract=RetractDispense( + positionReference=PositionReference.WELL_TOP, + offset=Coordinate(x=2, y=3, z=1), + speed=50, + airGapByVolume=[(1.0, 0.1), (49.9, 0.1), (50.0, 0.0)], + blowout=BlowoutProperties(enable=False, params=None), + touchTip=TouchTipProperties( + enable=False, + params=LiquidClassTouchTipParams( + zOffset=-1, mmToEdge=0.5, speed=30 + ), + ), + delay=DelayProperties( + enable=False, params=DelayParams(duration=0) + ), + ), + positionReference=PositionReference.WELL_BOTTOM, + offset=Coordinate(x=1, y=3, z=2), + flowRateByVolume=[(50.0, 50.0)], + correctionByVolume=[(0.0, 0.0)], + conditioningByVolume=[(1.0, 5.0), (45.0, 5.0), (50.0, 0.0)], + disposalByVolume=[(1.0, 5.0), (45.0, 5.0), (50.0, 0.0)], + delay=DelayProperties( + enable=True, params=DelayParams(duration=0.2) + ), + ), + ) + ], + ) + ], + ) diff --git a/api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py new file mode 100644 index 00000000000..6d115dfe074 --- /dev/null +++ b/api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py @@ -0,0 +1,123 @@ +"""Tests for complex commands executor.""" +import pytest +from decoy import Decoy +from opentrons_shared_data.liquid_classes import LiquidClassSchemaV1 + +from opentrons.protocol_api._liquid import LiquidClass +from opentrons.protocol_api._liquid_properties import TransferProperties +from opentrons.protocol_api.core.engine.well import WellCore +from opentrons.protocol_api.core.engine.instrument import InstrumentCore +from opentrons.protocol_api.core.engine.complex_commands_executor import ( + LiquidClassTransferExecutor, +) +from opentrons.types import Location, Point + + +@pytest.fixture +def mock_instrument_core(decoy: Decoy) -> InstrumentCore: + """Return a mocked out instrument core.""" + return decoy.mock(cls=InstrumentCore) + + +@pytest.fixture +def sample_transfer_props( + maximal_liquid_class_def: LiquidClassSchemaV1, +) -> TransferProperties: + """Return a mocked out liquid class fixture.""" + return LiquidClass.create(maximal_liquid_class_def).get_for( + pipette="flex_1channel_50", tiprack="opentrons_flex_96_tiprack_50ul" + ) + + +@pytest.fixture +def subject( + mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties +) -> LiquidClassTransferExecutor: + """Return a LiquidClassTransferExecutor test subject.""" + return LiquidClassTransferExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + ) + + +""" Test aspirate properties: +"submerge": { + "positionReference": "well-top", + "offset": {"x": 1, "y": 2, "z": 3}, + "speed": 100, + "delay": {"enable": true, "params": {"duration": 10.0}}}, +"retract": { + "positionReference": "well-top", + "offset": {"x": 3, "y": 2, "z": 1}, + "speed": 50, + "airGapByVolume": [[1.0, 0.1], [49.9, 0.1], [50.0, 0.0]], + "touchTip": {"enable": false, "params": {"zOffset": -1, "mmToEdge": 0.5, "speed": 30}}, + "delay": {"enable": false, "params": {"duration": 0}}}, +"positionReference": "well-bottom", +"offset": {"x": 10, "y": 20, "z": 30}, +"flowRateByVolume": [[1.0, 35.0], [10.0, 24.0], [50.0, 35.0]], +"correctionByVolume": [[0.0, 0.0]], +"preWet": true, +"mix": {"enable": true, "params": {"repetitions": 1, "volume": 50}}, +"delay": {"enable": true, "params": {"duration": 0.2}} +""" + + +def test_transfer_aspirate( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, + subject: LiquidClassTransferExecutor, +) -> None: + """Should perform the expected aspiration steps.""" + source_well = decoy.mock(cls=WellCore) + well_top_point = Point(1, 2, 3) + well_bottom_point = Point(4, 5, 6) + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(50) + ) + dispense_flow_rate = ( + sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(50) + ) + + decoy.when(source_well.get_bottom(0)).then_return(well_bottom_point) + decoy.when(source_well.get_top(0)).then_return(well_top_point) + subject.aspirate(volume=20, source=source_well) + + decoy.verify( + mock_instrument_core.move_to( + location=Location(Point(x=2, y=4, z=6), labware=None), + well_core=source_well, + force_direct=False, + minimum_z_height=None, + speed=None, + ), + mock_instrument_core.move_to( + location=Location(Point(14, 25, 36), labware=None), + well_core=source_well, + force_direct=True, + minimum_z_height=None, + speed=100, + ), + mock_instrument_core.delay(10), + mock_instrument_core.aspirate( + location=Location(Point(14, 25, 36), labware=None), + well_core=source_well, + volume=50, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + mock_instrument_core.delay(0.2), + mock_instrument_core.dispense( + location=Location(Point(14, 25, 36), labware=None), + well_core=source_well, + volume=50, + rate=1, + flow_rate=dispense_flow_rate, + in_place=True, + push_out=None, + is_meniscus=None, + ), + ) diff --git a/api/tests/opentrons/protocol_api/test_instrument_context.py b/api/tests/opentrons/protocol_api/test_instrument_context.py index 3f639aff922..7b2054d1067 100644 --- a/api/tests/opentrons/protocol_api/test_instrument_context.py +++ b/api/tests/opentrons/protocol_api/test_instrument_context.py @@ -1978,13 +1978,13 @@ def test_transfer_liquid_delegates_to_engine_core( ).then_return(trash_location.move(Point(1, 2, 3))) decoy.when(next_tiprack.uri).then_return("tiprack-uri") decoy.when(mock_instrument_core.get_pipette_name()).then_return("pipette-name") - decoy.when( - mock_instrument_core.load_liquid_class( - liquid_class=test_liq_class, - pipette_load_name="pipette-name", - tiprack_uri="tiprack-uri", - ) - ).then_return("liq-class-id") + # decoy.when( + # mock_instrument_core.load_liquid_class( + # liquid_class=test_liq_class, + # pipette_load_name="pipette-name", + # tiprack_uri="tiprack-uri", + # ) + # ).then_return("liq-class-id") subject.transfer_liquid( liquid_class=test_liq_class, @@ -1996,11 +1996,12 @@ def test_transfer_liquid_delegates_to_engine_core( ) decoy.verify( mock_instrument_core.transfer_liquid( - liquid_class_id="liq-class-id", + liquid_class=test_liq_class, volume=10, source=[mock_well._core], dest=[mock_well._core], new_tip=TransferTipPolicyV2.ONCE, + tiprack_uri="tiprack-uri", trash_location=trash_location.move(Point(1, 2, 3)), ) ) From 27f114fe785791f6e145ffe1f5bade35a2bcd6d5 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Wed, 11 Dec 2024 17:38:02 -0500 Subject: [PATCH 02/11] refactor- aspirate orchestration moved to instrument core, TransferComponentsExecutor executes individual components delegated by aspirate/ dispense --- .../core/engine/complex_commands_executor.py | 320 ----------------- .../protocol_api/core/engine/instrument.py | 108 +++++- .../engine/transfer_components_executor.py | 337 ++++++++++++++++++ ...y => test_transfer_components_executor.py} | 12 +- 4 files changed, 449 insertions(+), 328 deletions(-) delete mode 100644 api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py create mode 100644 api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py rename api/tests/opentrons/protocol_api/core/engine/{test_complex_commands_executor.py => test_transfer_components_executor.py} (93%) diff --git a/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py b/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py deleted file mode 100644 index 215086945b4..00000000000 --- a/api/src/opentrons/protocol_api/core/engine/complex_commands_executor.py +++ /dev/null @@ -1,320 +0,0 @@ -"""Executor for liquid class based complex commands.""" -from __future__ import annotations -from dataclasses import dataclass -from typing import Union, TYPE_CHECKING - -from opentrons_shared_data.liquid_classes.liquid_class_definition import ( - PositionReference, - Coordinate, -) - -from opentrons.protocol_api._liquid_properties import ( - SingleDispenseProperties, - MultiDispenseProperties, - Submerge, - RetractAspirate, - RetractDispense, - DelayProperties, - TransferProperties, -) -from opentrons.types import Location, Point - -if TYPE_CHECKING: - from .well import WellCore - from .instrument import InstrumentCore - - -@dataclass -class _TargetPosition: - position_reference: PositionReference - offset: Coordinate - - -@dataclass -class _TargetAsLocationAndWell: - location: Location - well: WellCore - - -@dataclass -class _MixData: - volume: float - repetitions: int - aspirate_flow_rate: float - aspirate_delay: DelayProperties - dispense_flow_rate: float - dispense_delay: DelayProperties - - -class LiquidClassTransferExecutor: - def __init__( - self, - instrument_core: InstrumentCore, - transfer_properties: TransferProperties, - ) -> None: - self._instrument = instrument_core - self._transfer_properties = transfer_properties - - def aspirate( - self, - volume: float, - source: WellCore, - ) -> None: - """Execute aspiration steps. - - 1. Submerge - 2. Mix - 3. pre-wet - - 1 combo of aspirate + dispense at the same flow rate as specified in asp & disp and the delays in asp & disp - - Use the target volume/ volume we will be aspirating - - No push out - - Not pre-wet for consolidation - 4. Aspirate - - Aspirate with provided flow rate - 5. Delay- wait inside the liquid - 6. Aspirate retract - """ - aspirate_properties = self._transfer_properties.aspirate - dispense_properties = self._transfer_properties.dispense - aspirate_location_and_well = _TargetAsLocationAndWell( - location=location_from_position_reference_and_offset( - well=source, - position_reference=aspirate_properties.position_reference, - offset=aspirate_properties.offset, - ), - well=source, - ) - - self._submerge( - target_location_and_well=aspirate_location_and_well, - submerge_props=aspirate_properties.submerge, - ) - mix_props = aspirate_properties.mix - if mix_props.enabled: - # This would have been validated in the liquid class. Assertion only for mypy purposes - assert mix_props.repetitions is not None and mix_props.volume is not None - self._mix( - target_location_and_well=aspirate_location_and_well, - mix_data=_MixData( - repetitions=mix_props.repetitions, - volume=mix_props.volume, - aspirate_flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume( - mix_props.volume - ), - aspirate_delay=aspirate_properties.delay, - dispense_flow_rate=dispense_properties.flow_rate_by_volume.get_for_volume( - mix_props.volume - ), - dispense_delay=dispense_properties.delay, - ), - ) - if aspirate_properties.pre_wet: - self._mix( - target_location_and_well=aspirate_location_and_well, - mix_data=_MixData( - repetitions=1, - volume=volume, - aspirate_flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume( - volume - ), - aspirate_delay=aspirate_properties.delay, - dispense_flow_rate=dispense_properties.flow_rate_by_volume.get_for_volume( - volume - ), - dispense_delay=dispense_properties.delay, - ), - ) - - self._instrument.aspirate( - location=aspirate_location_and_well.location, - well_core=source, - volume=volume, - rate=1, - flow_rate=aspirate_properties.flow_rate_by_volume.get_for_volume(volume), - in_place=True, - ) - if aspirate_properties.delay.enabled: - asp_delay_duration = aspirate_properties.delay.duration - # This would have been validated in the liquid class. - # Assertion only for mypy purposes - assert asp_delay_duration is not None - self._instrument.delay(asp_delay_duration) - self._retract(retract_props=aspirate_properties.retract) - - def single_dispense( - self, - volume: float, - source: WellCore, - dest: WellCore, - dispense_properties: SingleDispenseProperties, - ) -> None: - """Execute single-dispense steps. - - 1. Move pipette to the ‘submerge’ position with normal speed. - - The pipette will move in an arc- move to max z height of labware (if asp & disp are in same labware) or max z height of all labware (if asp & disp are in separate labware) - 2. Air gap removal: - - If dispense location is above the meniscus, DO NOT remove air gap (it will be dispensed along with rest of the liquid later). All other scenarios, remove the air gap by doing a dispense - - Flow rate = min(dispenseFlowRate, (airGapByVolume)/sec) - 3. Use the post-dispense delay - 4. Move to the dispense position at the specified ‘submerge’ speed (even if we might not be moving into the liquid) - 5. Do a delay - 6. Dispense: - - Dispense at the specified flow rate. - - Do a push out as specified ONLY IF there is no mix following the dispense AND the tip is empty. - Volume for push out is the volume being dispensed. So if we are dispensing 50uL, use pushOutByVolume[50] as push out volume. - 7. Delay - 8. Mix using the same flow rate and delays as specified for asp+disp, with the volume and the number of repetitions specified. Use the delays in asp & disp. - - If the dispense position is outside the liquid, then raise error if mix is enabled. - - If the user wants to perform a mix then they should specify a dispense position that’s inside the liquid OR do mix() on the wells after transfer. - - Do push out at the last dispense. - - """ - - def multi_dispense(self, dispense_properties: MultiDispenseProperties) -> None: - """Execute multi-dispense steps.""" - - def _submerge( - self, - target_location_and_well: _TargetAsLocationAndWell, - submerge_props: Submerge, - ) -> None: - """Execute submerge steps. - - 1. move to position shown by positionReference + offset (should practically be a point outside/above the liquid). - Should raise an error if this point is inside the liquid? - For liquid meniscus this is easy to tell. Can’t be below meniscus - For reference pos of anything else, do not allow submerge position to be below aspirate position - 2. move to aspirate position at desired speed - 3. delay - """ - # TODO: compare submerge start position and aspirate position and raise error if incompatible - self._instrument.move_to( - location=location_from_position_reference_and_offset( - target_location_and_well.well, - submerge_props.position_reference, - submerge_props.offset, - ), - well_core=target_location_and_well.well, - force_direct=False, - minimum_z_height=None, - speed=None, - ) - self._instrument.move_to( - location=target_location_and_well.location, - well_core=target_location_and_well.well, - force_direct=True, - minimum_z_height=None, - speed=submerge_props.speed, - ) - if submerge_props.delay.enabled: - self._instrument.delay(submerge_props.delay.duration) - - def _retract(self, retract_props: Union[RetractAspirate, RetractDispense]) -> None: - """Execute retraction steps. - - Retract aspirate: - 1. Move TO the position reference+offset AT the specified speed - Raise error if retract is below aspirate position or below the meniscus - 2. Delay - 3. Touch tip - - Move to the Z offset position - - Touch tip to the sides at the specified speed (tip moves back to the center as part of touch tip) - - Return back to the retract position - 4. Air gap - - Air gap volume depends on the amount of liquid in the pipette - So if total aspirated volume is 20, use the value for airGapByVolume[20] - Flow rate = min(aspirateFlowRate, (airGapByVolume)/sec) - 5. Use post-aspirate delay - - Retract dispense: - 1. Position ref+offset is the ending position. Move to this position using specified speed - 2. If blowout is enabled and “destination” - - Do blow-out (at the retract position) - - Leave plunger down - 3. Touch-tip - 4. If not ready-to-aspirate - - Prepare-to-aspirate (at the retract position) - 5. Air-gap (at the retract position) - - This air gap is for preventing any stray droplets from falling while moving the pipette. - It will be performed out of caution even if we just did a blow_out and should *hypothetically* - have no liquid left in the tip. - - This air gap will be removed at the next aspirate. - If this is the last step of the transfer, and we aren't dropping the tip off, - then the air gap will be left as is(?). - 6. If blowout is “source” or “trash” - - Move to location (top of Well) - - Do blow-out (top of well) - - Do touch-tip (?????) (only if it’s in a non-trash location) - - Prepare-to-aspirate (top of well) - - Do air-gap (top of well) - 7. If drop tip, move to drop tip location, drop tip - """ - - def _mix( - self, - target_location_and_well: _TargetAsLocationAndWell, - mix_data: _MixData, - ) -> None: - """Execute mix steps. - - 1. Use same flow rates and delays as aspirate and dispense - 2. Do [(aspirate + dispense) x repetitions] at the same position - 3. Do NOT push out at the end of dispense - 4. USE the delay property from aspirate & dispense during mix as well (flow rate and delay are coordinated with each other) - 5. Do not mix during consolidation - NOTE: For most of our built-in definitions, we will keep _mix_ off because it is a very application specific thing. - We should mention in our docs that users should adjust this property according to their application. - """ - for n in range(mix_data.repetitions): - # TODO: figure out what to set is_meniscus as - self._instrument.aspirate( - location=target_location_and_well.location, - well_core=target_location_and_well.well, - volume=mix_data.volume, - rate=1, - flow_rate=mix_data.aspirate_flow_rate, - in_place=True, - is_meniscus=None, - ) - if mix_data.aspirate_delay.enabled: - asp_delay_duration = mix_data.aspirate_delay.duration - # This would have been validated in the liquid class. - # Assertion only for mypy purposes - assert asp_delay_duration is not None - self._instrument.delay(asp_delay_duration) - self._instrument.dispense( - location=target_location_and_well.location, - well_core=target_location_and_well.well, - volume=mix_data.volume, - rate=1, - flow_rate=mix_data.dispense_flow_rate, - in_place=True, - push_out=None, # TODO: check if this should be 0 instead - is_meniscus=None, - ) - if mix_data.dispense_delay.enabled: - disp_delay_duration = mix_data.dispense_delay.duration - assert disp_delay_duration is not None - self._instrument.delay(disp_delay_duration) - - -def location_from_position_reference_and_offset( - well: WellCore, - position_reference: PositionReference, - offset: Coordinate, -) -> Location: - """Get position in `Location` type, given the well, the position reference and offset.""" - match position_reference: - case PositionReference.WELL_TOP: - reference_point = well.get_top(0) - case PositionReference.WELL_BOTTOM: - reference_point = well.get_bottom(0) - case PositionReference.WELL_CENTER: - reference_point = well.get_center() - case PositionReference.LIQUID_MENISCUS: - raise NotImplementedError( - "Liquid transfer using liquid-meniscus relative positioning is not yet implemented" - ) - case _: - raise ValueError(f"Unknown position reference {position_reference}") - return Location(reference_point + Point(offset.x, offset.y, offset.z), labware=None) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index d4bffd5ed76..ca5cbbf9fc9 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -44,7 +44,11 @@ from . import overlap_versions, pipette_movement_conflict from .well import WellCore -from .complex_commands_executor import LiquidClassTransferExecutor +from .transfer_components_executor import ( + TransferComponentsExecutor, + MixData, + MixAspDispData, +) from ..instrument import AbstractInstrument from ...disposal_locations import TrashBin, WasteChute @@ -905,6 +909,7 @@ def transfer_liquid( self, liquid_class: LiquidClass, volume: float, + # TODO: update source and dest types to list of tuples of Location and WellCore source: List[WellCore], dest: List[WellCore], new_tip: TransferTipPolicyV2, @@ -933,7 +938,7 @@ def transfer_liquid( targets=zip(source, dest), max_volume=self.get_max_volume(), ) - transfer_executer = LiquidClassTransferExecutor(instrument_core=self) + transfer_executer = TransferComponentsExecutor(instrument_core=self) if new_tip == TransferTipPolicyV2.ONCE: # TODO: update this once getNextTip is implemented next_tip = self.get_next_tip() @@ -960,6 +965,105 @@ def transfer_liquid( alternate_drop_location=True, ) + def aspirate_liquid_class( + self, + volume: float, + source: WellCore, + transfer_properties: TransferProperties, + ) -> None: + """Execute aspiration steps. + + 1. Submerge + 2. Mix + 3. pre-wet + 4. Aspirate + - Aspirate with provided flow rate + 5. Delay- wait inside the liquid + 6. Aspirate retract + """ + aspirate_props = transfer_properties.aspirate + dispense_props = transfer_properties.dispense + aspirate_location_and_well = _TargetAsLocationAndWell( + location=location_from_position_reference_and_offset( + well=source, + position_reference=aspirate_props.position_reference, + offset=aspirate_props.offset, + ), + well=source, + ) + transfer_executer = TransferComponentsExecutor( + instrument_core=self, + transfer_properties=transfer_properties, + target_location_and_well=aspirate_location_and_well, + ) + transfer_executer.submerge(submerge_properties=aspirate_props.submerge) + transfer_executer.mix( + mix_properties=aspirate_props.mix, + aspirate_data=MixAspDispData( + flow_rate=aspirate_props.flow_rate_by_volume.get_for_volume( + aspirate_props.mix.volume + ), + delay=aspirate_props.delay, + ), + dispense_data=MixAspDispData( + flow_rate=dispense_props.flow_rate_by_volume.get_for_volume( + aspirate_props.mix.volume + ), + delay=dispense_props.delay, + ), + ) + transfer_executer.pre_wet( + enabled=aspirate_props.pre_wet, + volume=volume, + aspirate_data=MixAspDispData( + flow_rate=aspirate_props.flow_rate_by_volume.get_for_volume(volume), + delay=aspirate_props.delay, + ), + dispense_data=MixAspDispData( + flow_rate=dispense_props.flow_rate_by_volume.get_for_volume(volume), + delay=dispense_props.delay, + ), + ) + transfer_executer.aspirate_and_wait(volume=volume) + transfer_executer.retract_after_aspiration( + air_gap_volume=aspirate_props.retract.air_gap_by_volume.get_for_volume( + volume + ) + ) + + def dispense_liquid_class( + self, + volume: float, + dest: WellCore, + dispense_properties: SingleDispenseProperties, + ) -> None: + """Execute single-dispense steps. + + 1. Move pipette to the ‘submerge’ position with normal speed. + - The pipette will move in an arc- move to max z height of labware + (if asp & disp are in same labware) + or max z height of all labware (if asp & disp are in separate labware) + 2. Air gap removal: + - If dispense location is above the meniscus, DO NOT remove air gap + (it will be dispensed along with rest of the liquid later). + All other scenarios, remove the air gap by doing a dispense + - Flow rate = min(dispenseFlowRate, (airGapByVolume)/sec) + - Use the post-dispense delay + 4. Move to the dispense position at the specified ‘submerge’ speed + (even if we might not be moving into the liquid) + - Do a delay (submerge delay) + 6. Dispense: + - Dispense at the specified flow rate. + - Do a push out as specified ONLY IF there is no mix following the dispense AND the tip is empty. + Volume for push out is the volume being dispensed. So if we are dispensing 50uL, use pushOutByVolume[50] as push out volume. + 7. Delay + 8. Mix using the same flow rate and delays as specified for asp+disp, with the volume and the number of repetitions specified. Use the delays in asp & disp. + - If the dispense position is outside the liquid, then raise error if mix is enabled. + - If the user wants to perform a mix then they should specify a dispense position that’s inside the liquid OR do mix() on the wells after transfer. + - Do push out at the last dispense. + + """ + def retract(self) -> None: """Retract this instrument to the top of the gantry.""" z_axis = self._engine_client.state.pipettes.get_z_axis(self._pipette_id) diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py new file mode 100644 index 00000000000..c9deaba7ef5 --- /dev/null +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -0,0 +1,337 @@ +"""Executor for liquid class based complex commands.""" +from __future__ import annotations +from dataclasses import dataclass +from typing import Union, TYPE_CHECKING, Optional + +from opentrons_shared_data.liquid_classes.liquid_class_definition import ( + PositionReference, + Coordinate, +) + +from opentrons.protocol_api._liquid_properties import ( + SingleDispenseProperties, + MultiDispenseProperties, + Submerge, + RetractAspirate, + RetractDispense, + DelayProperties, + TransferProperties, + MixProperties, + AspirateProperties, +) +from opentrons.types import Location, Point + +if TYPE_CHECKING: + from .well import WellCore + from .instrument import InstrumentCore + + +@dataclass +class _TargetPosition: + position_reference: PositionReference + offset: Coordinate + + +@dataclass +class _TargetAsLocationAndWell: + location: Location + well: WellCore + + +@dataclass +class MixAspDispData: + flow_rate: float + delay: DelayProperties + + +@dataclass +class MixData: + mix_properties: MixProperties + aspirate_data: MixAspDispData + dispense_data: MixAspDispData + + +class TransferComponentsExecutor: + def __init__( + self, + instrument_core: InstrumentCore, + transfer_properties: TransferProperties, + target_location_and_well: _TargetAsLocationAndWell, + ) -> None: + self._instrument = instrument_core + self._transfer_properties = transfer_properties + self._target_location_and_well = target_location_and_well + + def submerge( + self, + submerge_properties: Submerge, + air_gap_volume: float, + ) -> None: + """Execute submerge steps. + + 1. move to position shown by positionReference + offset (should practically be a point outside/above the liquid). + Should raise an error if this point is inside the liquid? + For liquid meniscus this is easy to tell. Can’t be below meniscus + For reference pos of anything else, do not allow submerge position to be below aspirate position + 2. move to aspirate position at desired speed + 3. delay + """ + # TODO: compare submerge start position and aspirate position and raise error if incompatible + submerge_start_location = location_from_position_reference_and_offset( + self._target_location_and_well.well, + submerge_properties.position_reference, + submerge_properties.offset, + ) + self._instrument.move_to( + location=submerge_start_location, + well_core=self._target_location_and_well.well, + force_direct=False, + minimum_z_height=None, + speed=None, + ) + self._remove_air_gap( + location=submerge_start_location, + volume=air_gap_volume, + ) + self._instrument.move_to( + location=self._target_location_and_well.location, + well_core=self._target_location_and_well.well, + force_direct=True, + minimum_z_height=None, + speed=submerge_properties.speed, + ) + if submerge_properties.delay.enabled: + assert submerge_properties.delay.duration is not None + self._instrument.delay(submerge_properties.delay.duration) + + def aspirate_and_wait(self, volume: float) -> None: + """Aspirate according to aspirate properties and wait if enabled.""" + aspirate_props = self._transfer_properties.aspirate + self._instrument.aspirate( + location=self._target_location_and_well.location, + well_core=None, + volume=volume, + rate=1, + flow_rate=aspirate_props.flow_rate_by_volume.get_for_volume(volume), + in_place=True, + is_meniscus=None, # TODO: update this once meniscus is implemented + ) + delay_props = aspirate_props.delay + if delay_props.enabled: + # This would have been validated in the liquid class. + # Assertion only for mypy purposes + assert delay_props.duration is not None + self._instrument.delay(delay_props.duration) + + def dispense_and_wait(self, volume: float, push_out: Optional[float]) -> None: + """Dispense according to dispense properties and wait if enabled.""" + dispense_props = self._transfer_properties.dispense + self._instrument.dispense( + location=self._target_location_and_well.location, + well_core=None, + volume=volume, + rate=1, + flow_rate=dispense_props.flow_rate_by_volume.get_for_volume(volume), + in_place=True, + push_out=push_out, # TODO: check if this should be 0 instead + is_meniscus=None, + ) + dispense_delay = dispense_props.delay + if dispense_delay.enabled: + assert dispense_delay.duration is not None + self._instrument.delay(dispense_delay.duration) + + def mix( + self, + mix_properties: MixProperties, + aspirate_data: MixAspDispData, + dispense_data: MixAspDispData, + ) -> None: + """Execute mix steps. + + 1. Use same flow rates and delays as aspirate and dispense + 2. Do [(aspirate + dispense) x repetitions] at the same position + 3. Do NOT push out at the end of dispense + 4. USE the delay property from aspirate & dispense during mix as well (flow rate and delay are coordinated with each other) + 5. Do not mix during consolidation + NOTE: For most of our built-in definitions, we will keep _mix_ off because it is a very application specific thing. + We should mention in our docs that users should adjust this property according to their application. + """ + if not mix_properties.enabled: + return + # This would have been validated in the liquid class. + # Assertion only for mypy purposes + assert ( + mix_properties.repetitions is not None and mix_properties.volume is not None + ) + for n in range(mix_properties.repetitions): + self.aspirate_and_wait(volume=mix_properties.volume) + # TODO: Update to doing a push out at the end of mix for a post-dispense mix + self.dispense_and_wait(volume=mix_properties.volume, push_out=0) + + def pre_wet( + self, + enabled: bool, + volume: float, + aspirate_data: MixAspDispData, + dispense_data: MixAspDispData, + ) -> None: + """Do a pre-wet. + + - 1 combo of aspirate + dispense at the same flow rate as specified in asp & disp and the delays in asp & disp + - Use the target volume/ volume we will be aspirating + - No push out + - Not pre-wet for consolidation + """ + if not enabled: + return + mix_props = MixProperties(_enabled=True, _repetitions=1, _volume=volume) + self.mix( + mix_properties=mix_props, + aspirate_data=aspirate_data, + dispense_data=dispense_data, + ) + + def retract_after_aspiration(self, air_gap_volume: float) -> None: + """Execute post-aspiration retraction steps. + + 1. Move TO the position reference+offset AT the specified speed + Raise error if retract is below aspirate position or below the meniscus + 2. Delay + 3. Touch tip + - Move to the Z offset position + - Touch tip to the sides at the specified speed (tip moves back to the center as part of touch tip) + - Return back to the retract position + 4. Air gap + - Air gap volume depends on the amount of liquid in the pipette + So if total aspirated volume is 20, use the value for airGapByVolume[20] + Flow rate = min(aspirateFlowRate, (airGapByVolume)/sec) + - Use post-aspirate delay + """ + # TODO: Raise error if retract is below the meniscus + retract_props = self._transfer_properties.aspirate.retract + retract_location = location_from_position_reference_and_offset( + self._target_location_and_well.well, + retract_props.position_reference, + retract_props.offset, + ) + self._instrument.move_to( + location=retract_location, + well_core=self._target_location_and_well.well, + force_direct=True, + minimum_z_height=None, + speed=retract_props.speed, + ) + retract_delay = retract_props.delay + if retract_delay.enabled: + assert retract_delay.duration is not None + self._instrument.delay(retract_delay.duration) + touch_tip_props = retract_props.touch_tip + if touch_tip_props.enabled: + assert ( + touch_tip_props.speed is not None + and touch_tip_props.z_offset is not None + and touch_tip_props.mm_to_edge is not None + ) + # TODO: update this once touch tip has mmToEdge + self._instrument.touch_tip( + location=self._target_location_and_well.location, + well_core=self._target_location_and_well.well, + radius=0, + z_offset=touch_tip_props.z_offset, + speed=touch_tip_props.speed, + ) + self._instrument.move_to( + location=retract_location, + well_core=self._target_location_and_well.well, + force_direct=True, + minimum_z_height=None, + # Full speed because assumption is that the tip will already be out of the liquid + speed=None, + ) + self._add_air_gap(volume=air_gap_volume) + + def _add_air_gap(self, volume: float) -> None: + """Add an air gap.""" + aspirate_props = self._transfer_properties.aspirate + # To err on the side of caution, the maximum flow rate should be air_gap_volume per second + flow_rate = min( + aspirate_props.flow_rate_by_volume.get_for_volume(volume), volume + ) + self._instrument.air_gap_in_place(volume=volume, flow_rate=flow_rate) + delay_props = aspirate_props.delay + if delay_props.enabled: + # This would have been validated in the liquid class. + # Assertion only for mypy purposes + assert delay_props.duration is not None + self._instrument.delay(delay_props.duration) + + def _remove_air_gap(self, location: Location, volume: float) -> None: + """Remove a previously added air gap.""" + dispense_props = self._transfer_properties.dispense + # To err on the side of caution, the maximum flow rate should be air_gap_volume per second + flow_rate = min( + dispense_props.flow_rate_by_volume.get_for_volume(volume), volume + ) + self._instrument.dispense( + location=location, + well_core=None, + volume=volume, + rate=1, + flow_rate=flow_rate, + in_place=True, + is_meniscus=None, + push_out=0, + ) + dispense_delay = dispense_props.delay + if dispense_delay.enabled: + assert dispense_delay.duration is not None + self._instrument.delay(dispense_delay.duration) + + def retract_after_dispensing(self) -> None: + """Execute post-dispense retraction steps. + + 1. Position ref+offset is the ending position. Move to this position using specified speed + 2. If blowout is enabled and “destination” + - Do blow-out (at the retract position) + - Leave plunger down + 3. Touch-tip + 4. If not ready-to-aspirate + - Prepare-to-aspirate (at the retract position) + 5. Air-gap (at the retract position) + - This air gap is for preventing any stray droplets from falling while moving the pipette. + It will be performed out of caution even if we just did a blow_out and should *hypothetically* + have no liquid left in the tip. + - This air gap will be removed at the next aspirate. + If this is the last step of the transfer, and we aren't dropping the tip off, + then the air gap will be left as is(?). + 6. If blowout is “source” or “trash” + - Move to location (top of Well) + - Do blow-out (top of well) + - Do touch-tip (?????) (only if it’s in a non-trash location) + - Prepare-to-aspirate (top of well) + - Do air-gap (top of well) + 7. If drop tip, move to drop tip location, drop tip + """ + + +def location_from_position_reference_and_offset( + well: WellCore, + position_reference: PositionReference, + offset: Coordinate, +) -> Location: + """Get position in `Location` type, given the well, the position reference and offset.""" + match position_reference: + case PositionReference.WELL_TOP: + reference_point = well.get_top(0) + case PositionReference.WELL_BOTTOM: + reference_point = well.get_bottom(0) + case PositionReference.WELL_CENTER: + reference_point = well.get_center() + case PositionReference.LIQUID_MENISCUS: + raise NotImplementedError( + "Liquid transfer using liquid-meniscus relative positioning is not yet implemented" + ) + case _: + raise ValueError(f"Unknown position reference {position_reference}") + return Location(reference_point + Point(offset.x, offset.y, offset.z), labware=None) diff --git a/api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py similarity index 93% rename from api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py rename to api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py index 6d115dfe074..0c3247fc0f8 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_complex_commands_executor.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py @@ -7,8 +7,8 @@ from opentrons.protocol_api._liquid_properties import TransferProperties from opentrons.protocol_api.core.engine.well import WellCore from opentrons.protocol_api.core.engine.instrument import InstrumentCore -from opentrons.protocol_api.core.engine.complex_commands_executor import ( - LiquidClassTransferExecutor, +from opentrons.protocol_api.core.engine.transfer_components_executor import ( + TransferComponentsExecutor, ) from opentrons.types import Location, Point @@ -32,9 +32,9 @@ def sample_transfer_props( @pytest.fixture def subject( mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties -) -> LiquidClassTransferExecutor: - """Return a LiquidClassTransferExecutor test subject.""" - return LiquidClassTransferExecutor( +) -> TransferComponentsExecutor: + """Return a TransferComponentsExecutor test subject.""" + return TransferComponentsExecutor( instrument_core=mock_instrument_core, transfer_properties=sample_transfer_props, ) @@ -67,7 +67,7 @@ def test_transfer_aspirate( decoy: Decoy, mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, - subject: LiquidClassTransferExecutor, + subject: TransferComponentsExecutor, ) -> None: """Should perform the expected aspiration steps.""" source_well = decoy.mock(cls=WellCore) From 07c7b101a85204b27d3515ed546f3eb0b99f4a3d Mon Sep 17 00:00:00 2001 From: Sanniti Date: Wed, 11 Dec 2024 19:12:53 -0500 Subject: [PATCH 03/11] use tuples of location and well core --- .../opentrons/protocol_api/core/engine/instrument.py | 7 +++---- api/src/opentrons/protocol_api/core/instrument.py | 6 +++--- .../core/legacy/legacy_instrument_core.py | 6 +++--- .../core/legacy_simulator/legacy_instrument_core.py | 6 +++--- api/src/opentrons/protocol_api/instrument_context.py | 10 ++++++++-- .../protocol_api/test_instrument_context.py | 12 ++---------- 6 files changed, 22 insertions(+), 25 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index ca5cbbf9fc9..c8b65147dd6 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -2,7 +2,7 @@ from __future__ import annotations -from typing import Optional, TYPE_CHECKING, cast, Union, List +from typing import Optional, TYPE_CHECKING, cast, Union, List, Tuple from opentrons.types import Location, Mount, NozzleConfigurationType, NozzleMapInterface from opentrons.hardware_control import SyncHardwareAPI from opentrons.hardware_control.dev_types import PipetteDict @@ -909,9 +909,8 @@ def transfer_liquid( self, liquid_class: LiquidClass, volume: float, - # TODO: update source and dest types to list of tuples of Location and WellCore - source: List[WellCore], - dest: List[WellCore], + source: List[Tuple[types.Location, WellCore]], + dest: List[Tuple[types.Location, WellCore]], new_tip: TransferTipPolicyV2, tiprack_uri: str, trash_location: Union[WellCore, Location, TrashBin, WasteChute], diff --git a/api/src/opentrons/protocol_api/core/instrument.py b/api/src/opentrons/protocol_api/core/instrument.py index aac6decb784..86d4afc7fac 100644 --- a/api/src/opentrons/protocol_api/core/instrument.py +++ b/api/src/opentrons/protocol_api/core/instrument.py @@ -3,7 +3,7 @@ from __future__ import annotations from abc import abstractmethod, ABC -from typing import Any, Generic, Optional, TypeVar, Union, List +from typing import Any, Generic, Optional, TypeVar, Union, List, Tuple from opentrons import types from opentrons.hardware_control.dev_types import PipetteDict @@ -328,8 +328,8 @@ def transfer_liquid( self, liquid_class: LiquidClass, volume: float, - source: List[WellCoreType], - dest: List[WellCoreType], + source: List[Tuple[types.Location, WellCoreType]], + dest: List[Tuple[types.Location, WellCoreType]], new_tip: TransferTipPolicyV2, tiprack_uri: str, trash_location: Union[WellCoreType, types.Location, TrashBin, WasteChute], diff --git a/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py b/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py index d2d25051d49..6efc936ea21 100644 --- a/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py +++ b/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py @@ -1,7 +1,7 @@ from __future__ import annotations import logging -from typing import TYPE_CHECKING, Optional, Union, List +from typing import TYPE_CHECKING, Optional, Union, List, Tuple from opentrons import types from opentrons.hardware_control import CriticalPoint @@ -570,8 +570,8 @@ def transfer_liquid( self, liquid_class_id: str, volume: float, - source: List[LegacyWellCore], - dest: List[LegacyWellCore], + source: List[Tuple[types.Location, LegacyWellCore]], + dest: List[Tuple[types.Location, LegacyWellCore]], new_tip: TransferTipPolicyV2, trash_location: Union[LegacyWellCore, types.Location, TrashBin, WasteChute], ) -> None: diff --git a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py index ec194874528..47c505470c4 100644 --- a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py +++ b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py @@ -1,7 +1,7 @@ from __future__ import annotations import logging -from typing import TYPE_CHECKING, Optional, Union, List +from typing import TYPE_CHECKING, Optional, Union, List, Tuple from opentrons import types from opentrons.hardware_control.dev_types import PipetteDict @@ -488,8 +488,8 @@ def transfer_liquid( self, liquid_class_id: str, volume: float, - source: List[LegacyWellCore], - dest: List[LegacyWellCore], + source: List[Tuple[types.Location, LegacyWellCore]], + dest: List[Tuple[types.Location, LegacyWellCore]], new_tip: TransferTipPolicyV2, trash_location: Union[LegacyWellCore, types.Location, TrashBin, WasteChute], ) -> None: diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 191f592a3a2..aae1494165f 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -1596,8 +1596,14 @@ def transfer_liquid( self._core.transfer_liquid( liquid_class=liquid_class, volume=volume, - source=[well._core for well in flat_sources_list], - dest=[well._core for well in flat_dests_list], + source=[ + (types.Location(types.Point(), labware=well), well._core) + for well in flat_sources_list + ], + dest=[ + (types.Location(types.Point(), labware=well), well._core) + for well in flat_dests_list + ], new_tip=valid_new_tip, tiprack_uri=tiprack.uri, trash_location=checked_trash_location._core diff --git a/api/tests/opentrons/protocol_api/test_instrument_context.py b/api/tests/opentrons/protocol_api/test_instrument_context.py index 7b2054d1067..965ad406739 100644 --- a/api/tests/opentrons/protocol_api/test_instrument_context.py +++ b/api/tests/opentrons/protocol_api/test_instrument_context.py @@ -1978,14 +1978,6 @@ def test_transfer_liquid_delegates_to_engine_core( ).then_return(trash_location.move(Point(1, 2, 3))) decoy.when(next_tiprack.uri).then_return("tiprack-uri") decoy.when(mock_instrument_core.get_pipette_name()).then_return("pipette-name") - # decoy.when( - # mock_instrument_core.load_liquid_class( - # liquid_class=test_liq_class, - # pipette_load_name="pipette-name", - # tiprack_uri="tiprack-uri", - # ) - # ).then_return("liq-class-id") - subject.transfer_liquid( liquid_class=test_liq_class, volume=10, @@ -1998,8 +1990,8 @@ def test_transfer_liquid_delegates_to_engine_core( mock_instrument_core.transfer_liquid( liquid_class=test_liq_class, volume=10, - source=[mock_well._core], - dest=[mock_well._core], + source=[(Location(Point(), labware=mock_well), mock_well._core)], + dest=[(Location(Point(), labware=mock_well), mock_well._core)], new_tip=TransferTipPolicyV2.ONCE, tiprack_uri="tiprack-uri", trash_location=trash_location.move(Point(1, 2, 3)), From 6e1fbed08609f13dbb348f3321d5bfc84e1707b7 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Wed, 11 Dec 2024 23:42:43 -0500 Subject: [PATCH 04/11] fixes after last change --- .../protocol_api/core/engine/instrument.py | 42 +++++++++++++------ .../engine/transfer_components_executor.py | 28 +++++++------ 2 files changed, 45 insertions(+), 25 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index c8b65147dd6..e4f09be3c7f 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -48,6 +48,7 @@ TransferComponentsExecutor, MixData, MixAspDispData, + absolute_point_from_position_reference_and_offset, ) from ..instrument import AbstractInstrument from ...disposal_locations import TrashBin, WasteChute @@ -913,9 +914,24 @@ def transfer_liquid( dest: List[Tuple[types.Location, WellCore]], new_tip: TransferTipPolicyV2, tiprack_uri: str, - trash_location: Union[WellCore, Location, TrashBin, WasteChute], + tip_drop_location: Union[WellCore, Location, TrashBin, WasteChute], ) -> None: - """Execute transfer using liquid class properties.""" + """Execute transfer using liquid class properties. + + Args: + liquid_class: The liquid class to use for transfer properties. + volume: Volume to transfer per well. + source: List of source wells, with each well represented as a tuple of + types.Location and WellCore. + types.Location is only necessary for saving the last accessed location. + dest: List of destination wells, with each well represented as a tuple of + types.Location and WellCore. + types.Location is only necessary for saving the last accessed location. + new_tip: Whether the transfer should use a new tip 'once', 'never', 'always', + or 'per source'. + tiprack_uri: The URI of the tiprack that the transfer settings are for. + tip_drop_location: Location where the tip will be dropped (if appropriate). + """ liquid_class_id = self.load_liquid_class( liquid_class=liquid_class, pipette_load_name=self.name, @@ -950,16 +966,16 @@ def transfer_liquid( transfer_executer.aspirate(aspirate_props) if new_tip == TransferTipPolicyV2.ALWAYS: - if isinstance(trash_location, (TrashBin, WasteChute)): + if isinstance(tip_drop_location, (TrashBin, WasteChute)): self.drop_tip_in_disposal_location( - disposal_location=trash_location, + disposal_location=tip_drop_location, home_after=False, alternate_tip_drop=True, ) - elif isinstance(trash_location, Location): + elif isinstance(tip_drop_location, Location): self.drop_tip( - location=trash_location, - well_core=trash_location.labware.as_well()._core, + location=tip_drop_location, + well_core=tip_drop_location.labware.as_well()._core, home_after=False, alternate_drop_location=True, ) @@ -967,7 +983,7 @@ def transfer_liquid( def aspirate_liquid_class( self, volume: float, - source: WellCore, + source: Tuple(Location, WellCore), transfer_properties: TransferProperties, ) -> None: """Execute aspiration steps. @@ -982,12 +998,12 @@ def aspirate_liquid_class( """ aspirate_props = transfer_properties.aspirate dispense_props = transfer_properties.dispense + + aspirate_location_point = absolute_point_from_position_reference_and_offset( + well=source, position_reference=aspirate_props.position_reference, + offset=aspirate_props.offset) aspirate_location_and_well = _TargetAsLocationAndWell( - location=location_from_position_reference_and_offset( - well=source, - position_reference=aspirate_props.position_reference, - offset=aspirate_props.offset, - ), + location=Location(aspirate_location_point, labware=source[0].labware), well=source, ) transfer_executer = TransferComponentsExecutor( diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index c9deaba7ef5..5afabdc0ded 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -77,10 +77,14 @@ def submerge( 3. delay """ # TODO: compare submerge start position and aspirate position and raise error if incompatible - submerge_start_location = location_from_position_reference_and_offset( + submerge_start_point = absolute_point_from_position_reference_and_offset( self._target_location_and_well.well, submerge_properties.position_reference, - submerge_properties.offset, + submerge_properties.offset + ) + submerge_start_location = Location( + point=submerge_start_point, + labware=self._target_location_and_well.location.labware ) self._instrument.move_to( location=submerge_start_location, @@ -210,11 +214,10 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: """ # TODO: Raise error if retract is below the meniscus retract_props = self._transfer_properties.aspirate.retract - retract_location = location_from_position_reference_and_offset( - self._target_location_and_well.well, - retract_props.position_reference, - retract_props.offset, - ) + retract_point = absolute_point_from_position_reference_and_offset( + self._target_location_and_well.well, retract_props.position_reference, + retract_props.offset) + retract_location = Location(retract_point, labware=self._target_location_and_well.location.labware) self._instrument.move_to( location=retract_location, well_core=self._target_location_and_well.well, @@ -315,12 +318,12 @@ def retract_after_dispensing(self) -> None: """ -def location_from_position_reference_and_offset( +def absolute_point_from_position_reference_and_offset( well: WellCore, position_reference: PositionReference, offset: Coordinate, -) -> Location: - """Get position in `Location` type, given the well, the position reference and offset.""" +) -> Point: + """Get the absolute point, given the well, the position reference and offset.""" match position_reference: case PositionReference.WELL_TOP: reference_point = well.get_top(0) @@ -330,8 +333,9 @@ def location_from_position_reference_and_offset( reference_point = well.get_center() case PositionReference.LIQUID_MENISCUS: raise NotImplementedError( - "Liquid transfer using liquid-meniscus relative positioning is not yet implemented" + "Liquid transfer using liquid-meniscus relative positioning" + " is not yet implemented." ) case _: raise ValueError(f"Unknown position reference {position_reference}") - return Location(reference_point + Point(offset.x, offset.y, offset.z), labware=None) + return reference_point + Point(offset.x, offset.y, offset.z) From dba4f40eb9795ccd14bca11fe8550366455fbf31 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 10:06:52 -0500 Subject: [PATCH 05/11] linter and misc fixes --- .../protocol_api/core/engine/instrument.py | 106 +++++++++-------- .../engine/transfer_components_executor.py | 107 ++++++------------ .../opentrons/protocol_api/core/instrument.py | 13 --- .../core/legacy/legacy_instrument_core.py | 13 +-- .../legacy_instrument_core.py | 13 +-- .../test_transfer_components_executor.py | 70 +++++------- .../transfers/test_common_functions.py | 24 ++-- 7 files changed, 130 insertions(+), 216 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index e4f09be3c7f..44292d4bc9f 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -46,8 +46,6 @@ from .well import WellCore from .transfer_components_executor import ( TransferComponentsExecutor, - MixData, - MixAspDispData, absolute_point_from_position_reference_and_offset, ) from ..instrument import AbstractInstrument @@ -903,15 +901,16 @@ def load_liquid_class( ) return result.liquidClassId - def get_next_tip(self) -> NextTipInfo: + # TODO: update with getNextTip implementation + def get_next_tip(self) -> None: """Get the next tip to pick up.""" def transfer_liquid( self, liquid_class: LiquidClass, volume: float, - source: List[Tuple[types.Location, WellCore]], - dest: List[Tuple[types.Location, WellCore]], + source: List[Tuple[Location, WellCore]], + dest: List[Tuple[Location, WellCore]], new_tip: TransferTipPolicyV2, tiprack_uri: str, tip_drop_location: Union[WellCore, Location, TrashBin, WasteChute], @@ -932,16 +931,18 @@ def transfer_liquid( tiprack_uri: The URI of the tiprack that the transfer settings are for. tip_drop_location: Location where the tip will be dropped (if appropriate). """ - liquid_class_id = self.load_liquid_class( + # TODO: use the ID returned by load_liquid_class in command annotations + self.load_liquid_class( liquid_class=liquid_class, - pipette_load_name=self.name, + pipette_load_name=self.get_pipette_name(), # TODO: update this to use load name instead tiprack_uri=tiprack_uri, ) transfer_props = liquid_class.get_for( - pipette=pipette_load_name, tiprack=tiprack_uri + # update this to fetch load name instead + pipette=self.get_pipette_name(), + tiprack=tiprack_uri, ) aspirate_props = transfer_props.aspirate - dispense_props = transfer_props.dispense check_valid_volume_parameters( disposal_volume=0, # No disposal volume for 1-to-1 transfer @@ -953,18 +954,16 @@ def transfer_liquid( targets=zip(source, dest), max_volume=self.get_max_volume(), ) - transfer_executer = TransferComponentsExecutor(instrument_core=self) if new_tip == TransferTipPolicyV2.ONCE: # TODO: update this once getNextTip is implemented - next_tip = self.get_next_tip() - self.pick_up_tip(next_tip) - for step_volume, (src, dest) in source_dest_per_volume_step: + self.get_next_tip() + for step_volume, (src, dest) in source_dest_per_volume_step: # type: ignore[assignment] if new_tip == TransferTipPolicyV2.ALWAYS: # TODO: update this once getNextTip is implemented - next_tip = self.get_next_tip() - self.pick_up_tip(next_tip) + self.get_next_tip() + + # TODO: add aspirate and dispense - transfer_executer.aspirate(aspirate_props) if new_tip == TransferTipPolicyV2.ALWAYS: if isinstance(tip_drop_location, (TrashBin, WasteChute)): self.drop_tip_in_disposal_location( @@ -975,15 +974,29 @@ def transfer_liquid( elif isinstance(tip_drop_location, Location): self.drop_tip( location=tip_drop_location, - well_core=tip_drop_location.labware.as_well()._core, + well_core=tip_drop_location.labware.as_well()._core, # type: ignore[arg-type] home_after=False, alternate_drop_location=True, ) + def _get_transfer_components_executor( + self, + transfer_properties: TransferProperties, + target_location: Location, + target_well: WellCore, + ) -> TransferComponentsExecutor: + """Get a TransferComponentsExecutor.""" + return TransferComponentsExecutor( + instrument_core=self, + transfer_properties=transfer_properties, + target_location=target_location, + target_well=target_well, + ) + def aspirate_liquid_class( self, volume: float, - source: Tuple(Location, WellCore), + source: Tuple[Location, WellCore], transfer_properties: TransferProperties, ) -> None: """Execute aspiration steps. @@ -992,55 +1005,38 @@ def aspirate_liquid_class( 2. Mix 3. pre-wet 4. Aspirate - - Aspirate with provided flow rate 5. Delay- wait inside the liquid 6. Aspirate retract """ aspirate_props = transfer_properties.aspirate dispense_props = transfer_properties.dispense - aspirate_location_point = absolute_point_from_position_reference_and_offset( - well=source, position_reference=aspirate_props.position_reference, - offset=aspirate_props.offset) - aspirate_location_and_well = _TargetAsLocationAndWell( - location=Location(aspirate_location_point, labware=source[0].labware), - well=source, + source_loc, source_well = source + + aspirate_point = absolute_point_from_position_reference_and_offset( + well=source_well, + position_reference=aspirate_props.position_reference, + offset=aspirate_props.offset, ) - transfer_executer = TransferComponentsExecutor( - instrument_core=self, + aspirate_location = Location(aspirate_point, labware=source_loc.labware) + + components_executer = self._get_transfer_components_executor( transfer_properties=transfer_properties, - target_location_and_well=aspirate_location_and_well, + target_location=aspirate_location, + target_well=source_well, ) - transfer_executer.submerge(submerge_properties=aspirate_props.submerge) - transfer_executer.mix( - mix_properties=aspirate_props.mix, - aspirate_data=MixAspDispData( - flow_rate=aspirate_props.flow_rate_by_volume.get_for_volume( - aspirate_props.mix.volume - ), - delay=aspirate_props.delay, - ), - dispense_data=MixAspDispData( - flow_rate=dispense_props.flow_rate_by_volume.get_for_volume( - aspirate_props.mix.volume - ), - delay=dispense_props.delay, - ), + components_executer.submerge( + submerge_properties=aspirate_props.submerge, + # Assuming aspirate is not called with liquid already in the tip + air_gap_volume=self.get_current_volume(), ) - transfer_executer.pre_wet( + components_executer.mix(mix_properties=aspirate_props.mix) + components_executer.pre_wet( enabled=aspirate_props.pre_wet, volume=volume, - aspirate_data=MixAspDispData( - flow_rate=aspirate_props.flow_rate_by_volume.get_for_volume(volume), - delay=aspirate_props.delay, - ), - dispense_data=MixAspDispData( - flow_rate=dispense_props.flow_rate_by_volume.get_for_volume(volume), - delay=dispense_props.delay, - ), ) - transfer_executer.aspirate_and_wait(volume=volume) - transfer_executer.retract_after_aspiration( + components_executer.aspirate_and_wait(volume=volume) + components_executer.retract_after_aspiration( air_gap_volume=aspirate_props.retract.air_gap_by_volume.get_for_volume( volume ) @@ -1050,7 +1046,7 @@ def dispense_liquid_class( self, volume: float, dest: WellCore, - dispense_properties: SingleDispenseProperties, + transfer_properties: TransferProperties, ) -> None: """Execute single-dispense steps. diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 5afabdc0ded..33eff091e2f 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -1,7 +1,6 @@ """Executor for liquid class based complex commands.""" from __future__ import annotations -from dataclasses import dataclass -from typing import Union, TYPE_CHECKING, Optional +from typing import TYPE_CHECKING, Optional from opentrons_shared_data.liquid_classes.liquid_class_definition import ( PositionReference, @@ -9,15 +8,9 @@ ) from opentrons.protocol_api._liquid_properties import ( - SingleDispenseProperties, - MultiDispenseProperties, Submerge, - RetractAspirate, - RetractDispense, - DelayProperties, TransferProperties, MixProperties, - AspirateProperties, ) from opentrons.types import Location, Point @@ -26,41 +19,18 @@ from .instrument import InstrumentCore -@dataclass -class _TargetPosition: - position_reference: PositionReference - offset: Coordinate - - -@dataclass -class _TargetAsLocationAndWell: - location: Location - well: WellCore - - -@dataclass -class MixAspDispData: - flow_rate: float - delay: DelayProperties - - -@dataclass -class MixData: - mix_properties: MixProperties - aspirate_data: MixAspDispData - dispense_data: MixAspDispData - - class TransferComponentsExecutor: def __init__( self, instrument_core: InstrumentCore, transfer_properties: TransferProperties, - target_location_and_well: _TargetAsLocationAndWell, + target_location: Location, + target_well: WellCore, ) -> None: self._instrument = instrument_core self._transfer_properties = transfer_properties - self._target_location_and_well = target_location_and_well + self._target_location = target_location + self._target_well = target_well def submerge( self, @@ -78,17 +48,16 @@ def submerge( """ # TODO: compare submerge start position and aspirate position and raise error if incompatible submerge_start_point = absolute_point_from_position_reference_and_offset( - self._target_location_and_well.well, - submerge_properties.position_reference, - submerge_properties.offset + well=self._target_well, + position_reference=submerge_properties.position_reference, + offset=submerge_properties.offset, ) submerge_start_location = Location( - point=submerge_start_point, - labware=self._target_location_and_well.location.labware + point=submerge_start_point, labware=self._target_location.labware ) self._instrument.move_to( location=submerge_start_location, - well_core=self._target_location_and_well.well, + well_core=self._target_well, force_direct=False, minimum_z_height=None, speed=None, @@ -98,8 +67,8 @@ def submerge( volume=air_gap_volume, ) self._instrument.move_to( - location=self._target_location_and_well.location, - well_core=self._target_location_and_well.well, + location=self._target_location, + well_core=self._target_well, force_direct=True, minimum_z_height=None, speed=submerge_properties.speed, @@ -110,9 +79,10 @@ def submerge( def aspirate_and_wait(self, volume: float) -> None: """Aspirate according to aspirate properties and wait if enabled.""" + # TODO: handle volume correction aspirate_props = self._transfer_properties.aspirate self._instrument.aspirate( - location=self._target_location_and_well.location, + location=self._target_location, well_core=None, volume=volume, rate=1, @@ -122,22 +92,22 @@ def aspirate_and_wait(self, volume: float) -> None: ) delay_props = aspirate_props.delay if delay_props.enabled: - # This would have been validated in the liquid class. # Assertion only for mypy purposes assert delay_props.duration is not None self._instrument.delay(delay_props.duration) def dispense_and_wait(self, volume: float, push_out: Optional[float]) -> None: """Dispense according to dispense properties and wait if enabled.""" + # TODO: handle volume correction dispense_props = self._transfer_properties.dispense self._instrument.dispense( - location=self._target_location_and_well.location, + location=self._target_location, well_core=None, volume=volume, rate=1, flow_rate=dispense_props.flow_rate_by_volume.get_for_volume(volume), in_place=True, - push_out=push_out, # TODO: check if this should be 0 instead + push_out=push_out, is_meniscus=None, ) dispense_delay = dispense_props.delay @@ -145,12 +115,7 @@ def dispense_and_wait(self, volume: float, push_out: Optional[float]) -> None: assert dispense_delay.duration is not None self._instrument.delay(dispense_delay.duration) - def mix( - self, - mix_properties: MixProperties, - aspirate_data: MixAspDispData, - dispense_data: MixAspDispData, - ) -> None: + def mix(self, mix_properties: MixProperties) -> None: """Execute mix steps. 1. Use same flow rates and delays as aspirate and dispense @@ -163,7 +128,6 @@ def mix( """ if not mix_properties.enabled: return - # This would have been validated in the liquid class. # Assertion only for mypy purposes assert ( mix_properties.repetitions is not None and mix_properties.volume is not None @@ -177,24 +141,18 @@ def pre_wet( self, enabled: bool, volume: float, - aspirate_data: MixAspDispData, - dispense_data: MixAspDispData, ) -> None: """Do a pre-wet. - 1 combo of aspirate + dispense at the same flow rate as specified in asp & disp and the delays in asp & disp - Use the target volume/ volume we will be aspirating - No push out - - Not pre-wet for consolidation + - No pre-wet for consolidation """ if not enabled: return mix_props = MixProperties(_enabled=True, _repetitions=1, _volume=volume) - self.mix( - mix_properties=mix_props, - aspirate_data=aspirate_data, - dispense_data=dispense_data, - ) + self.mix(mix_properties=mix_props) def retract_after_aspiration(self, air_gap_volume: float) -> None: """Execute post-aspiration retraction steps. @@ -215,12 +173,16 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: # TODO: Raise error if retract is below the meniscus retract_props = self._transfer_properties.aspirate.retract retract_point = absolute_point_from_position_reference_and_offset( - self._target_location_and_well.well, retract_props.position_reference, - retract_props.offset) - retract_location = Location(retract_point, labware=self._target_location_and_well.location.labware) + well=self._target_well, + position_reference=retract_props.position_reference, + offset=retract_props.offset, + ) + retract_location = Location( + retract_point, labware=self._target_location.labware + ) self._instrument.move_to( location=retract_location, - well_core=self._target_location_and_well.well, + well_core=self._target_well, force_direct=True, minimum_z_height=None, speed=retract_props.speed, @@ -238,18 +200,18 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: ) # TODO: update this once touch tip has mmToEdge self._instrument.touch_tip( - location=self._target_location_and_well.location, - well_core=self._target_location_and_well.well, + location=self._target_location, + well_core=self._target_well, radius=0, z_offset=touch_tip_props.z_offset, speed=touch_tip_props.speed, ) self._instrument.move_to( location=retract_location, - well_core=self._target_location_and_well.well, + well_core=self._target_well, force_direct=True, minimum_z_height=None, - # Full speed because assumption is that the tip will already be out of the liquid + # Full speed because the tip will already be out of the liquid speed=None, ) self._add_air_gap(volume=air_gap_volume) @@ -257,14 +219,13 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: def _add_air_gap(self, volume: float) -> None: """Add an air gap.""" aspirate_props = self._transfer_properties.aspirate - # To err on the side of caution, the maximum flow rate should be air_gap_volume per second + # The maximum flow rate should be air_gap_volume per second flow_rate = min( aspirate_props.flow_rate_by_volume.get_for_volume(volume), volume ) self._instrument.air_gap_in_place(volume=volume, flow_rate=flow_rate) delay_props = aspirate_props.delay if delay_props.enabled: - # This would have been validated in the liquid class. # Assertion only for mypy purposes assert delay_props.duration is not None self._instrument.delay(delay_props.duration) @@ -323,7 +284,7 @@ def absolute_point_from_position_reference_and_offset( position_reference: PositionReference, offset: Coordinate, ) -> Point: - """Get the absolute point, given the well, the position reference and offset.""" + """Return the absolute point, given the well, the position reference and offset.""" match position_reference: case PositionReference.WELL_TOP: reference_point = well.get_top(0) diff --git a/api/src/opentrons/protocol_api/core/instrument.py b/api/src/opentrons/protocol_api/core/instrument.py index 86d4afc7fac..08a0611e306 100644 --- a/api/src/opentrons/protocol_api/core/instrument.py +++ b/api/src/opentrons/protocol_api/core/instrument.py @@ -310,19 +310,6 @@ def configure_nozzle_layout( """ ... - @abstractmethod - def load_liquid_class( - self, - liquid_class: LiquidClass, - pipette_load_name: str, - tiprack_uri: str, - ) -> str: - """Load the liquid class properties of given pipette and tiprack into the engine. - - Returns: ID of the liquid class record - """ - ... - @abstractmethod def transfer_liquid( self, diff --git a/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py b/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py index 6efc936ea21..ec725934d90 100644 --- a/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py +++ b/api/src/opentrons/protocol_api/core/legacy/legacy_instrument_core.py @@ -556,23 +556,14 @@ def configure_nozzle_layout( """This will never be called because it was added in API 2.16.""" pass - def load_liquid_class( - self, - liquid_class: LiquidClass, - pipette_load_name: str, - tiprack_uri: str, - ) -> str: - """This will never be called because it was added in ..""" - # TODO(spp, 2024-11-20): update the docstring and error to include API version - assert False, "load_liquid_class is not supported in legacy context" - def transfer_liquid( self, - liquid_class_id: str, + liquid_class: LiquidClass, volume: float, source: List[Tuple[types.Location, LegacyWellCore]], dest: List[Tuple[types.Location, LegacyWellCore]], new_tip: TransferTipPolicyV2, + tiprack_uri: str, trash_location: Union[LegacyWellCore, types.Location, TrashBin, WasteChute], ) -> None: """This will never be called because it was added in ..""" diff --git a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py index 47c505470c4..aacac9392a7 100644 --- a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py +++ b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_instrument_core.py @@ -474,23 +474,14 @@ def configure_nozzle_layout( """This will never be called because it was added in API 2.15.""" pass - def load_liquid_class( - self, - liquid_class: LiquidClass, - pipette_load_name: str, - tiprack_uri: str, - ) -> str: - """This will never be called because it was added in ..""" - # TODO(spp, 2024-11-20): update the docstring and error to include API version - assert False, "load_liquid_class is not supported in legacy context" - def transfer_liquid( self, - liquid_class_id: str, + liquid_class: LiquidClass, volume: float, source: List[Tuple[types.Location, LegacyWellCore]], dest: List[Tuple[types.Location, LegacyWellCore]], new_tip: TransferTipPolicyV2, + tiprack_uri: str, trash_location: Union[LegacyWellCore, types.Location, TrashBin, WasteChute], ) -> None: """Transfer a liquid from source to dest according to liquid class properties.""" diff --git a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py index 0c3247fc0f8..73f1953464c 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py @@ -1,7 +1,9 @@ """Tests for complex commands executor.""" import pytest from decoy import Decoy -from opentrons_shared_data.liquid_classes import LiquidClassSchemaV1 +from opentrons_shared_data.liquid_classes.liquid_class_definition import ( + LiquidClassSchemaV1, +) from opentrons.protocol_api._liquid import LiquidClass from opentrons.protocol_api._liquid_properties import TransferProperties @@ -29,17 +31,6 @@ def sample_transfer_props( ) -@pytest.fixture -def subject( - mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties -) -> TransferComponentsExecutor: - """Return a TransferComponentsExecutor test subject.""" - return TransferComponentsExecutor( - instrument_core=mock_instrument_core, - transfer_properties=sample_transfer_props, - ) - - """ Test aspirate properties: "submerge": { "positionReference": "well-top", @@ -63,26 +54,32 @@ def subject( """ -def test_transfer_aspirate( +def test_submerge( decoy: Decoy, mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, - subject: TransferComponentsExecutor, ) -> None: """Should perform the expected aspiration steps.""" source_well = decoy.mock(cls=WellCore) well_top_point = Point(1, 2, 3) well_bottom_point = Point(4, 5, 6) - aspirate_flow_rate = ( - sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(50) - ) - dispense_flow_rate = ( - sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(50) + air_gap_removal_flow_rate = ( + sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(123) ) + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(), labware=None), + target_well=source_well, + ) decoy.when(source_well.get_bottom(0)).then_return(well_bottom_point) decoy.when(source_well.get_top(0)).then_return(well_top_point) - subject.aspirate(volume=20, source=source_well) + + subject.submerge( + submerge_properties=sample_transfer_props.aspirate.submerge, + air_gap_volume=123, + ) decoy.verify( mock_instrument_core.move_to( @@ -92,32 +89,23 @@ def test_transfer_aspirate( minimum_z_height=None, speed=None, ), - mock_instrument_core.move_to( - location=Location(Point(14, 25, 36), labware=None), - well_core=source_well, - force_direct=True, - minimum_z_height=None, - speed=100, - ), - mock_instrument_core.delay(10), - mock_instrument_core.aspirate( - location=Location(Point(14, 25, 36), labware=None), - well_core=source_well, - volume=50, + mock_instrument_core.dispense( + location=Location(Point(x=2, y=4, z=6), labware=None), + well_core=None, + volume=123, rate=1, - flow_rate=aspirate_flow_rate, + flow_rate=air_gap_removal_flow_rate, in_place=True, is_meniscus=None, + push_out=0, ), mock_instrument_core.delay(0.2), - mock_instrument_core.dispense( - location=Location(Point(14, 25, 36), labware=None), + mock_instrument_core.move_to( + location=Location(Point(), labware=None), well_core=source_well, - volume=50, - rate=1, - flow_rate=dispense_flow_rate, - in_place=True, - push_out=None, - is_meniscus=None, + force_direct=True, + minimum_z_height=None, + speed=100, ), + mock_instrument_core.delay(10), ) diff --git a/api/tests/opentrons/protocols/advanced_control/transfers/test_common_functions.py b/api/tests/opentrons/protocols/advanced_control/transfers/test_common_functions.py index 644c2b7094f..b80667a302b 100644 --- a/api/tests/opentrons/protocols/advanced_control/transfers/test_common_functions.py +++ b/api/tests/opentrons/protocols/advanced_control/transfers/test_common_functions.py @@ -44,20 +44,20 @@ def test_check_valid_volume_parameters( argvalues=[ ( [60, 70, 75], - [("a", "b"), ("c", "d"), ("e", "f")], + [(("a", "b"), (1, 2)), (("c", "d"), (3, 4)), (("e", "f"), (5, 6))], 20, [ - (20, ("a", "b")), - (20, ("a", "b")), - (20, ("a", "b")), - (20, ("c", "d")), - (20, ("c", "d")), - (15, ("c", "d")), - (15, ("c", "d")), - (20, ("e", "f")), - (20, ("e", "f")), - (17.5, ("e", "f")), - (17.5, ("e", "f")), + (20, (("a", "b"), (1, 2))), + (20, (("a", "b"), (1, 2))), + (20, (("a", "b"), (1, 2))), + (20, (("c", "d"), (3, 4))), + (20, (("c", "d"), (3, 4))), + (15, (("c", "d"), (3, 4))), + (15, (("c", "d"), (3, 4))), + (20, (("e", "f"), (5, 6))), + (20, (("e", "f"), (5, 6))), + (17.5, (("e", "f"), (5, 6))), + (17.5, (("e", "f"), (5, 6))), ], ), ], From c74e3180457db8df94f79117583986748f21168e Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 10:58:20 -0500 Subject: [PATCH 06/11] updated and added tests --- api/tests/opentrons/conftest.py | 2 +- .../test_transfer_components_executor.py | 113 +++++++++++++++++- 2 files changed, 113 insertions(+), 2 deletions(-) diff --git a/api/tests/opentrons/conftest.py b/api/tests/opentrons/conftest.py index e4f980e36a6..13c1c88b47d 100755 --- a/api/tests/opentrons/conftest.py +++ b/api/tests/opentrons/conftest.py @@ -968,7 +968,7 @@ def maximal_liquid_class_def() -> LiquidClassSchemaV1: (50.0, 2.0), ], delay=DelayProperties( - enable=True, params=DelayParams(duration=0.2) + enable=True, params=DelayParams(duration=0.5) ), ), multiDispense=MultiDispenseProperties( diff --git a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py index 73f1953464c..172114b553c 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py @@ -99,7 +99,7 @@ def test_submerge( is_meniscus=None, push_out=0, ), - mock_instrument_core.delay(0.2), + mock_instrument_core.delay(0.5), mock_instrument_core.move_to( location=Location(Point(), labware=None), well_core=source_well, @@ -109,3 +109,114 @@ def test_submerge( ), mock_instrument_core.delay(10), ) + + +def test_aspirate_and_wait( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute an aspirate and a delay according to properties.""" + source_well = decoy.mock(cls=WellCore) + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(10) + ) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.aspirate_and_wait(volume=10) + decoy.verify( + mock_instrument_core.aspirate( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=10, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + mock_instrument_core.delay(0.2), + ) + + +def test_dispense_and_wait( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute an aspirate and a delay according to properties.""" + source_well = decoy.mock(cls=WellCore) + dispense_flow_rate = ( + sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(10) + ) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.dispense_and_wait(volume=10, push_out=123) + decoy.verify( + mock_instrument_core.dispense( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=10, + rate=1, + flow_rate=dispense_flow_rate, + in_place=True, + push_out=123, + is_meniscus=None, + ), + mock_instrument_core.delay(0.5), + ) + + +def test_mix( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute mix steps.""" + source_well = decoy.mock(cls=WellCore) + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(50) + ) + dispense_flow_rate = ( + sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(50) + ) + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.mix(mix_properties=sample_transfer_props.aspirate.mix) + + decoy.verify( + mock_instrument_core.aspirate( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=50, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + mock_instrument_core.delay(0.2), + mock_instrument_core.dispense( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=50, + rate=1, + flow_rate=dispense_flow_rate, + in_place=True, + push_out=0, + is_meniscus=None, + ), + mock_instrument_core.delay(0.5), + ) From dabff5a80afd358f144c41624833d37e50ef0019 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 14:11:06 -0500 Subject: [PATCH 07/11] added more tests --- .../protocol_api/core/engine/instrument.py | 15 +- .../engine/transfer_components_executor.py | 20 +- .../test_transfer_components_executor.py | 324 +++++++++++++++++- 3 files changed, 338 insertions(+), 21 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index 44292d4bc9f..de510781526 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -931,6 +931,7 @@ def transfer_liquid( tiprack_uri: The URI of the tiprack that the transfer settings are for. tip_drop_location: Location where the tip will be dropped (if appropriate). """ + # This function is WIP # TODO: use the ID returned by load_liquid_class in command annotations self.load_liquid_class( liquid_class=liquid_class, @@ -1009,10 +1010,7 @@ def aspirate_liquid_class( 6. Aspirate retract """ aspirate_props = transfer_properties.aspirate - dispense_props = transfer_properties.dispense - source_loc, source_well = source - aspirate_point = absolute_point_from_position_reference_and_offset( well=source_well, position_reference=aspirate_props.position_reference, @@ -1027,20 +1025,17 @@ def aspirate_liquid_class( ) components_executer.submerge( submerge_properties=aspirate_props.submerge, - # Assuming aspirate is not called with liquid already in the tip + # Assuming aspirate is not called with *liquid*git swta in the tip air_gap_volume=self.get_current_volume(), ) + # TODO: when aspirating for consolidation, do not perform mix components_executer.mix(mix_properties=aspirate_props.mix) + # TODO: when aspirating for consolidation, do not preform pre-wet components_executer.pre_wet( - enabled=aspirate_props.pre_wet, volume=volume, ) components_executer.aspirate_and_wait(volume=volume) - components_executer.retract_after_aspiration( - air_gap_volume=aspirate_props.retract.air_gap_by_volume.get_for_volume( - volume - ) - ) + components_executer.retract_after_aspiration(volume=volume) def dispense_liquid_class( self, diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 33eff091e2f..65f306b4641 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -139,7 +139,6 @@ def mix(self, mix_properties: MixProperties) -> None: def pre_wet( self, - enabled: bool, volume: float, ) -> None: """Do a pre-wet. @@ -149,12 +148,12 @@ def pre_wet( - No push out - No pre-wet for consolidation """ - if not enabled: + if not self._transfer_properties.aspirate.pre_wet: return mix_props = MixProperties(_enabled=True, _repetitions=1, _volume=volume) self.mix(mix_properties=mix_props) - def retract_after_aspiration(self, air_gap_volume: float) -> None: + def retract_after_aspiration(self, volume: float) -> None: """Execute post-aspiration retraction steps. 1. Move TO the position reference+offset AT the specified speed @@ -200,7 +199,7 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: ) # TODO: update this once touch tip has mmToEdge self._instrument.touch_tip( - location=self._target_location, + location=retract_location, well_core=self._target_well, radius=0, z_offset=touch_tip_props.z_offset, @@ -214,16 +213,21 @@ def retract_after_aspiration(self, air_gap_volume: float) -> None: # Full speed because the tip will already be out of the liquid speed=None, ) - self._add_air_gap(volume=air_gap_volume) + self._add_air_gap( + air_gap_volume=self._transfer_properties.aspirate.retract.air_gap_by_volume.get_for_volume( + volume + ) + ) - def _add_air_gap(self, volume: float) -> None: + def _add_air_gap(self, air_gap_volume: float) -> None: """Add an air gap.""" aspirate_props = self._transfer_properties.aspirate # The maximum flow rate should be air_gap_volume per second flow_rate = min( - aspirate_props.flow_rate_by_volume.get_for_volume(volume), volume + aspirate_props.flow_rate_by_volume.get_for_volume(air_gap_volume), + air_gap_volume, ) - self._instrument.air_gap_in_place(volume=volume, flow_rate=flow_rate) + self._instrument.air_gap_in_place(volume=air_gap_volume, flow_rate=flow_rate) delay_props = aspirate_props.delay if delay_props.enabled: # Assertion only for mypy purposes diff --git a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py index 172114b553c..3c57adb30da 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py @@ -3,6 +3,8 @@ from decoy import Decoy from opentrons_shared_data.liquid_classes.liquid_class_definition import ( LiquidClassSchemaV1, + PositionReference, + Coordinate, ) from opentrons.protocol_api._liquid import LiquidClass @@ -11,6 +13,7 @@ from opentrons.protocol_api.core.engine.instrument import InstrumentCore from opentrons.protocol_api.core.engine.transfer_components_executor import ( TransferComponentsExecutor, + absolute_point_from_position_reference_and_offset, ) from opentrons.types import Location, Point @@ -42,8 +45,8 @@ def sample_transfer_props( "offset": {"x": 3, "y": 2, "z": 1}, "speed": 50, "airGapByVolume": [[1.0, 0.1], [49.9, 0.1], [50.0, 0.0]], - "touchTip": {"enable": false, "params": {"zOffset": -1, "mmToEdge": 0.5, "speed": 30}}, - "delay": {"enable": false, "params": {"duration": 0}}}, + "touchTip": {"enable": true, "params": {"zOffset": -1, "mmToEdge": 0.5, "speed": 30}}, + "delay": {"enable": true, "params": {"duration": 20}}}, "positionReference": "well-bottom", "offset": {"x": 10, "y": 20, "z": 30}, "flowRateByVolume": [[1.0, 35.0], [10.0, 24.0], [50.0, 35.0]], @@ -143,12 +146,34 @@ def test_aspirate_and_wait( ) -def test_dispense_and_wait( +def test_aspirate_and_wait_skips_delay( decoy: Decoy, mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, ) -> None: """It should execute an aspirate and a delay according to properties.""" + sample_transfer_props.aspirate.delay.enabled = False + source_well = decoy.mock(cls=WellCore) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.aspirate_and_wait(volume=10) + decoy.verify( + mock_instrument_core.delay(0.2), + times=0, + ) + + +def test_dispense_and_wait( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute a dispense and a delay according to properties.""" source_well = decoy.mock(cls=WellCore) dispense_flow_rate = ( sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(10) @@ -176,6 +201,28 @@ def test_dispense_and_wait( ) +def test_dispense_and_wait_skips_delay( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute an aspirate and a delay according to properties.""" + sample_transfer_props.dispense.delay.enabled = False + source_well = decoy.mock(cls=WellCore) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.dispense_and_wait(volume=10, push_out=123) + decoy.verify( + mock_instrument_core.delay(0.2), + times=0, + ) + + def test_mix( decoy: Decoy, mock_instrument_core: InstrumentCore, @@ -220,3 +267,274 @@ def test_mix( ), mock_instrument_core.delay(0.5), ) + + +def test_mix_disabled( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should not perform a mix when it is disabled.""" + sample_transfer_props.aspirate.mix.enabled = False + source_well = decoy.mock(cls=WellCore) + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(50) + ) + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.mix(mix_properties=sample_transfer_props.aspirate.mix) + decoy.verify( + mock_instrument_core.aspirate( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=50, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + times=0, + ) + + +def test_pre_wet( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute pre-wet steps.""" + source_well = decoy.mock(cls=WellCore) + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(40) + ) + dispense_flow_rate = ( + sample_transfer_props.dispense.flow_rate_by_volume.get_for_volume(40) + ) + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.pre_wet(volume=40) + + decoy.verify( + mock_instrument_core.aspirate( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=40, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + mock_instrument_core.delay(0.2), + mock_instrument_core.dispense( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=40, + rate=1, + flow_rate=dispense_flow_rate, + in_place=True, + push_out=0, + is_meniscus=None, + ), + mock_instrument_core.delay(0.5), + ) + + +def test_pre_wet_disabled( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should NOT execute pre-wet steps.""" + source_well = decoy.mock(cls=WellCore) + sample_transfer_props.aspirate.pre_wet = False + aspirate_flow_rate = ( + sample_transfer_props.aspirate.flow_rate_by_volume.get_for_volume(40) + ) + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + subject.pre_wet(volume=40) + + decoy.verify( + mock_instrument_core.aspirate( + location=Location(Point(1, 2, 3), labware=None), + well_core=None, + volume=40, + rate=1, + flow_rate=aspirate_flow_rate, + in_place=True, + is_meniscus=None, + ), + times=0, + ) + + +def test_retract_after_aspiration( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute steps to retract from well after an aspiration.""" + source_well = decoy.mock(cls=WellCore) + well_top_point = Point(1, 2, 3) + well_bottom_point = Point(4, 5, 6) + + air_gap_volume = ( + sample_transfer_props.aspirate.retract.air_gap_by_volume.get_for_volume(40) + ) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 1, 1), labware=None), + target_well=source_well, + ) + decoy.when(source_well.get_bottom(0)).then_return(well_bottom_point) + decoy.when(source_well.get_top(0)).then_return(well_top_point) + + subject.retract_after_aspiration(volume=40) + + decoy.verify( + mock_instrument_core.move_to( + location=Location(Point(x=4, y=4, z=4), labware=None), + well_core=source_well, + force_direct=True, + minimum_z_height=None, + speed=50, + ), + mock_instrument_core.delay(20), + mock_instrument_core.touch_tip( + location=Location(Point(x=4, y=4, z=4), labware=None), + well_core=source_well, + radius=0, # Update this to use mmToEdge once implemented + z_offset=-1, + speed=30, + ), + mock_instrument_core.move_to( + location=Location(Point(x=4, y=4, z=4), labware=None), + well_core=source_well, + force_direct=True, + minimum_z_height=None, + speed=None, + ), + mock_instrument_core.air_gap_in_place( + volume=air_gap_volume, + flow_rate=air_gap_volume, + ), + mock_instrument_core.delay(0.2), + ) + + +def test_retract_after_aspiration_without_touch_tip_and_delay( + decoy: Decoy, + mock_instrument_core: InstrumentCore, + sample_transfer_props: TransferProperties, +) -> None: + """It should execute steps to retract from well after an aspiration without a touch tip or delay.""" + source_well = decoy.mock(cls=WellCore) + well_top_point = Point(1, 2, 3) + well_bottom_point = Point(4, 5, 6) + + sample_transfer_props.aspirate.retract.touch_tip.enabled = False + sample_transfer_props.aspirate.retract.delay.enabled = False + + air_gap_volume = ( + sample_transfer_props.aspirate.retract.air_gap_by_volume.get_for_volume(40) + ) + + subject = TransferComponentsExecutor( + instrument_core=mock_instrument_core, + transfer_properties=sample_transfer_props, + target_location=Location(Point(1, 1, 1), labware=None), + target_well=source_well, + ) + decoy.when(source_well.get_bottom(0)).then_return(well_bottom_point) + decoy.when(source_well.get_top(0)).then_return(well_top_point) + + subject.retract_after_aspiration(volume=40) + + decoy.verify( + mock_instrument_core.move_to( + location=Location(Point(x=4, y=4, z=4), labware=None), + well_core=source_well, + force_direct=True, + minimum_z_height=None, + speed=50, + ), + mock_instrument_core.air_gap_in_place( + volume=air_gap_volume, + flow_rate=air_gap_volume, + ), + mock_instrument_core.delay(0.2), + ) + + +@pytest.mark.parametrize( + argnames=["position_reference", "offset", "expected_result"], + argvalues=[ + (PositionReference.WELL_TOP, Coordinate(x=11, y=12, z=13), Point(12, 14, 16)), + ( + PositionReference.WELL_BOTTOM, + Coordinate(x=21, y=22, z=23), + Point(25, 27, 29), + ), + ( + PositionReference.WELL_CENTER, + Coordinate(x=31, y=32, z=33), + Point(38, 40, 42), + ), + ], +) +def test_absolute_point_from_position_reference_and_offset( + decoy: Decoy, + position_reference: PositionReference, + offset: Coordinate, + expected_result: Point, +) -> None: + """It should return the correct absolute point based on well, position reference and offset.""" + well = decoy.mock(cls=WellCore) + + well_top_point = Point(1, 2, 3) + well_bottom_point = Point(4, 5, 6) + well_center_point = Point(7, 8, 9) + decoy.when(well.get_bottom(0)).then_return(well_bottom_point) + decoy.when(well.get_top(0)).then_return(well_top_point) + decoy.when(well.get_center()).then_return(well_center_point) + + assert ( + absolute_point_from_position_reference_and_offset( + well=well, position_reference=position_reference, offset=offset + ) + == expected_result + ) + + +def test_absolute_point_from_position_reference_and_offset_raises_errors( + decoy: Decoy, +) -> None: + """It should raise errors for invalid input.""" + well = decoy.mock(cls=WellCore) + with pytest.raises(NotImplementedError): + absolute_point_from_position_reference_and_offset( + well=well, + position_reference=PositionReference.LIQUID_MENISCUS, + offset=Coordinate(x=0, y=0, z=0), + ) + + with pytest.raises(ValueError, match="Unknown position reference"): + absolute_point_from_position_reference_and_offset( + well=well, + position_reference="PositionReference", # type: ignore[arg-type] + offset=Coordinate(x=0, y=0, z=0), + ) From aa189fa1fb85c4af5989f258df4af5beb3d31b6d Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 14:14:20 -0500 Subject: [PATCH 08/11] remove dispense stuff --- .../protocol_api/core/engine/instrument.py | 33 ------------------- .../engine/transfer_components_executor.py | 26 --------------- 2 files changed, 59 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index de510781526..936b704c426 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -1037,39 +1037,6 @@ def aspirate_liquid_class( components_executer.aspirate_and_wait(volume=volume) components_executer.retract_after_aspiration(volume=volume) - def dispense_liquid_class( - self, - volume: float, - dest: WellCore, - transfer_properties: TransferProperties, - ) -> None: - """Execute single-dispense steps. - - 1. Move pipette to the ‘submerge’ position with normal speed. - - The pipette will move in an arc- move to max z height of labware - (if asp & disp are in same labware) - or max z height of all labware (if asp & disp are in separate labware) - 2. Air gap removal: - - If dispense location is above the meniscus, DO NOT remove air gap - (it will be dispensed along with rest of the liquid later). - All other scenarios, remove the air gap by doing a dispense - - Flow rate = min(dispenseFlowRate, (airGapByVolume)/sec) - - Use the post-dispense delay - 4. Move to the dispense position at the specified ‘submerge’ speed - (even if we might not be moving into the liquid) - - Do a delay (submerge delay) - 6. Dispense: - - Dispense at the specified flow rate. - - Do a push out as specified ONLY IF there is no mix following the dispense AND the tip is empty. - Volume for push out is the volume being dispensed. So if we are dispensing 50uL, use pushOutByVolume[50] as push out volume. - 7. Delay - 8. Mix using the same flow rate and delays as specified for asp+disp, with the volume and the number of repetitions specified. Use the delays in asp & disp. - - If the dispense position is outside the liquid, then raise error if mix is enabled. - - If the user wants to perform a mix then they should specify a dispense position that’s inside the liquid OR do mix() on the wells after transfer. - - Do push out at the last dispense. - - """ - def retract(self) -> None: """Retract this instrument to the top of the gantry.""" z_axis = self._engine_client.state.pipettes.get_z_axis(self._pipette_id) diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 65f306b4641..107a330a9df 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -256,32 +256,6 @@ def _remove_air_gap(self, location: Location, volume: float) -> None: assert dispense_delay.duration is not None self._instrument.delay(dispense_delay.duration) - def retract_after_dispensing(self) -> None: - """Execute post-dispense retraction steps. - - 1. Position ref+offset is the ending position. Move to this position using specified speed - 2. If blowout is enabled and “destination” - - Do blow-out (at the retract position) - - Leave plunger down - 3. Touch-tip - 4. If not ready-to-aspirate - - Prepare-to-aspirate (at the retract position) - 5. Air-gap (at the retract position) - - This air gap is for preventing any stray droplets from falling while moving the pipette. - It will be performed out of caution even if we just did a blow_out and should *hypothetically* - have no liquid left in the tip. - - This air gap will be removed at the next aspirate. - If this is the last step of the transfer, and we aren't dropping the tip off, - then the air gap will be left as is(?). - 6. If blowout is “source” or “trash” - - Move to location (top of Well) - - Do blow-out (top of well) - - Do touch-tip (?????) (only if it’s in a non-trash location) - - Prepare-to-aspirate (top of well) - - Do air-gap (top of well) - 7. If drop tip, move to drop tip location, drop tip - """ - def absolute_point_from_position_reference_and_offset( well: WellCore, From 1851696e7aeb5762facf4cdd15778ca7c80b3a36 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 16:09:47 -0500 Subject: [PATCH 09/11] add aspirate test --- .../protocol_api/core/engine/instrument.py | 35 +++----- .../engine/transfer_components_executor.py | 15 ++++ .../core/engine/test_instrument_core.py | 85 +++++++++++++++++++ 3 files changed, 111 insertions(+), 24 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index 936b704c426..acd4d9f68ed 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -42,12 +42,9 @@ ) from opentrons.protocol_api._nozzle_layout import NozzleLayout from . import overlap_versions, pipette_movement_conflict +from . import transfer_components_executor as tx_comps_executor from .well import WellCore -from .transfer_components_executor import ( - TransferComponentsExecutor, - absolute_point_from_position_reference_and_offset, -) from ..instrument import AbstractInstrument from ...disposal_locations import TrashBin, WasteChute @@ -980,20 +977,6 @@ def transfer_liquid( alternate_drop_location=True, ) - def _get_transfer_components_executor( - self, - transfer_properties: TransferProperties, - target_location: Location, - target_well: WellCore, - ) -> TransferComponentsExecutor: - """Get a TransferComponentsExecutor.""" - return TransferComponentsExecutor( - instrument_core=self, - transfer_properties=transfer_properties, - target_location=target_location, - target_well=target_well, - ) - def aspirate_liquid_class( self, volume: float, @@ -1011,21 +994,25 @@ def aspirate_liquid_class( """ aspirate_props = transfer_properties.aspirate source_loc, source_well = source - aspirate_point = absolute_point_from_position_reference_and_offset( - well=source_well, - position_reference=aspirate_props.position_reference, - offset=aspirate_props.offset, + aspirate_point = ( + tx_comps_executor.absolute_point_from_position_reference_and_offset( + well=source_well, + position_reference=aspirate_props.position_reference, + offset=aspirate_props.offset, + ) ) aspirate_location = Location(aspirate_point, labware=source_loc.labware) - components_executer = self._get_transfer_components_executor( + components_executer = tx_comps_executor.get_transfer_components_executor( + instrument_core=self, transfer_properties=transfer_properties, target_location=aspirate_location, target_well=source_well, ) components_executer.submerge( submerge_properties=aspirate_props.submerge, - # Assuming aspirate is not called with *liquid*git swta in the tip + # Assuming aspirate is not called with *liquid* in the tip + # TODO: evaluate if using the current volume to find air gap is not a good idea. air_gap_volume=self.get_current_volume(), ) # TODO: when aspirating for consolidation, do not perform mix diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 107a330a9df..3663501961c 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -257,6 +257,21 @@ def _remove_air_gap(self, location: Location, volume: float) -> None: self._instrument.delay(dispense_delay.duration) +def get_transfer_components_executor( + instrument_core: InstrumentCore, + transfer_properties: TransferProperties, + target_location: Location, + target_well: WellCore, +) -> TransferComponentsExecutor: + """Get a TransferComponentsExecutor.""" + return TransferComponentsExecutor( + instrument_core=instrument_core, + transfer_properties=transfer_properties, + target_location=target_location, + target_well=target_well, + ) + + def absolute_point_from_position_reference_and_offset( well: WellCore, position_reference: PositionReference, diff --git a/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py b/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py index 352dcb35c58..b8600416de3 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py @@ -7,6 +7,8 @@ from decoy import errors from opentrons_shared_data.liquid_classes.liquid_class_definition import ( LiquidClassSchemaV1, + PositionReference, + Coordinate, ) from opentrons_shared_data.pipette.types import PipetteNameType @@ -14,6 +16,10 @@ from opentrons.hardware_control import SyncHardwareAPI from opentrons.hardware_control.dev_types import PipetteDict from opentrons.protocol_api._liquid_properties import TransferProperties +from opentrons.protocol_api.core.engine import transfer_components_executor +from opentrons.protocol_api.core.engine.transfer_components_executor import ( + TransferComponentsExecutor, +) from opentrons.protocol_engine import ( DeckPoint, LoadedPipette, @@ -90,6 +96,34 @@ def patch_mock_pipette_movement_safety_check( ) +@pytest.fixture +def mock_transfer_components_executor( + decoy: Decoy, +) -> TransferComponentsExecutor: + """Get a mocked out TransferComponentsExecutor.""" + return decoy.mock(cls=TransferComponentsExecutor) + + +@pytest.fixture(autouse=True) +def patch_mock_transfer_components_executor( + decoy: Decoy, + monkeypatch: pytest.MonkeyPatch, +) -> None: + """Replace transfer_components_executor functions with mocks.""" + monkeypatch.setattr( + transfer_components_executor, + "get_transfer_components_executor", + decoy.mock(func=transfer_components_executor.get_transfer_components_executor), + ) + monkeypatch.setattr( + transfer_components_executor, + "absolute_point_from_position_reference_and_offset", + decoy.mock( + func=transfer_components_executor.absolute_point_from_position_reference_and_offset + ), + ) + + @pytest.fixture def subject( decoy: Decoy, @@ -1556,3 +1590,54 @@ def test_load_liquid_class( tiprack_uri="opentrons_flex_96_tiprack_50ul", ) assert result == "liquid-class-id" + + +def test_aspirate_liquid_class( + decoy: Decoy, + mock_engine_client: EngineClient, + subject: InstrumentCore, + minimal_liquid_class_def2: LiquidClassSchemaV1, + mock_transfer_components_executor: TransferComponentsExecutor, +) -> None: + """It should call aspirate sub-steps execution based on liquid class.""" + source_well = decoy.mock(cls=WellCore) + source_location = Location(Point(1, 2, 3), labware=None) + test_liquid_class = LiquidClass.create(minimal_liquid_class_def2) + test_transfer_properties = test_liquid_class.get_for( + "flex_1channel_50", "opentrons_flex_96_tiprack_50ul" + ) + decoy.when( + transfer_components_executor.absolute_point_from_position_reference_and_offset( + well=source_well, + position_reference=PositionReference.WELL_BOTTOM, + offset=Coordinate(x=0, y=0, z=-5), + ) + ).then_return(Point(1, 2, 3)) + decoy.when( + transfer_components_executor.get_transfer_components_executor( + instrument_core=subject, + transfer_properties=test_transfer_properties, + target_location=Location(Point(1, 2, 3), labware=None), + target_well=source_well, + ) + ).then_return(mock_transfer_components_executor) + decoy.when( + mock_engine_client.state.pipettes.get_aspirated_volume("abc123") + ).then_return(111) + subject.aspirate_liquid_class( + volume=123, + source=(source_location, source_well), + transfer_properties=test_transfer_properties, + ) + decoy.verify( + mock_transfer_components_executor.submerge( + submerge_properties=test_transfer_properties.aspirate.submerge, + air_gap_volume=111, + ), + mock_transfer_components_executor.mix( + mix_properties=test_transfer_properties.aspirate.mix + ), + mock_transfer_components_executor.pre_wet(volume=123), + mock_transfer_components_executor.aspirate_and_wait(volume=123), + mock_transfer_components_executor.retract_after_aspiration(volume=123), + ) From fdb1b0be8f9e2a38e7c21c63ad79045443de44d9 Mon Sep 17 00:00:00 2001 From: Sanniti Date: Thu, 12 Dec 2024 16:40:59 -0500 Subject: [PATCH 10/11] docstrings and comments cleanup --- .../core/engine/transfer_components_executor.py | 2 +- .../core/engine/test_transfer_components_executor.py | 6 +++--- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 3663501961c..2e4dae08fab 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -237,7 +237,7 @@ def _add_air_gap(self, air_gap_volume: float) -> None: def _remove_air_gap(self, location: Location, volume: float) -> None: """Remove a previously added air gap.""" dispense_props = self._transfer_properties.dispense - # To err on the side of caution, the maximum flow rate should be air_gap_volume per second + # The maximum flow rate should be air_gap_volume per second flow_rate = min( dispense_props.flow_rate_by_volume.get_for_volume(volume), volume ) diff --git a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py index 3c57adb30da..7be520b173b 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_transfer_components_executor.py @@ -62,7 +62,7 @@ def test_submerge( mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, ) -> None: - """Should perform the expected aspiration steps.""" + """Should perform the expected submerge steps.""" source_well = decoy.mock(cls=WellCore) well_top_point = Point(1, 2, 3) well_bottom_point = Point(4, 5, 6) @@ -151,7 +151,7 @@ def test_aspirate_and_wait_skips_delay( mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, ) -> None: - """It should execute an aspirate and a delay according to properties.""" + """It should skip the wait after aspirate.""" sample_transfer_props.aspirate.delay.enabled = False source_well = decoy.mock(cls=WellCore) @@ -206,7 +206,7 @@ def test_dispense_and_wait_skips_delay( mock_instrument_core: InstrumentCore, sample_transfer_props: TransferProperties, ) -> None: - """It should execute an aspirate and a delay according to properties.""" + """It should skip the wait after dispense.""" sample_transfer_props.dispense.delay.enabled = False source_well = decoy.mock(cls=WellCore) From 14ffa1985e51effcbd9d08fb60445e7345df3d0d Mon Sep 17 00:00:00 2001 From: Sanniti Date: Fri, 20 Dec 2024 01:12:32 -0500 Subject: [PATCH 11/11] monketpatch the class instead of getter --- .../protocol_api/core/engine/instrument.py | 2 +- .../core/engine/transfer_components_executor.py | 15 --------------- .../core/engine/test_instrument_core.py | 7 ++++--- 3 files changed, 5 insertions(+), 19 deletions(-) diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index acd4d9f68ed..4da069335c8 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -1003,7 +1003,7 @@ def aspirate_liquid_class( ) aspirate_location = Location(aspirate_point, labware=source_loc.labware) - components_executer = tx_comps_executor.get_transfer_components_executor( + components_executer = tx_comps_executor.TransferComponentsExecutor( instrument_core=self, transfer_properties=transfer_properties, target_location=aspirate_location, diff --git a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py index 2e4dae08fab..92d28f2cf7a 100644 --- a/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py +++ b/api/src/opentrons/protocol_api/core/engine/transfer_components_executor.py @@ -257,21 +257,6 @@ def _remove_air_gap(self, location: Location, volume: float) -> None: self._instrument.delay(dispense_delay.duration) -def get_transfer_components_executor( - instrument_core: InstrumentCore, - transfer_properties: TransferProperties, - target_location: Location, - target_well: WellCore, -) -> TransferComponentsExecutor: - """Get a TransferComponentsExecutor.""" - return TransferComponentsExecutor( - instrument_core=instrument_core, - transfer_properties=transfer_properties, - target_location=target_location, - target_well=target_well, - ) - - def absolute_point_from_position_reference_and_offset( well: WellCore, position_reference: PositionReference, diff --git a/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py b/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py index b8600416de3..b507740cbc3 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_instrument_core.py @@ -108,12 +108,13 @@ def mock_transfer_components_executor( def patch_mock_transfer_components_executor( decoy: Decoy, monkeypatch: pytest.MonkeyPatch, + mock_transfer_components_executor: TransferComponentsExecutor, ) -> None: """Replace transfer_components_executor functions with mocks.""" monkeypatch.setattr( transfer_components_executor, - "get_transfer_components_executor", - decoy.mock(func=transfer_components_executor.get_transfer_components_executor), + "TransferComponentsExecutor", + mock_transfer_components_executor, ) monkeypatch.setattr( transfer_components_executor, @@ -1614,7 +1615,7 @@ def test_aspirate_liquid_class( ) ).then_return(Point(1, 2, 3)) decoy.when( - transfer_components_executor.get_transfer_components_executor( + transfer_components_executor.TransferComponentsExecutor( instrument_core=subject, transfer_properties=test_transfer_properties, target_location=Location(Point(1, 2, 3), labware=None),