From aef10761ed512e63d50ad6f22fd4b4e58c41846d Mon Sep 17 00:00:00 2001 From: Laura Cox Date: Thu, 13 May 2021 14:59:23 -0400 Subject: [PATCH 1/2] feat(api): allow labware calibration on non liquid handling events closes #7800 --- api/src/opentrons/commands/commands.py | 15 +++++++++++ api/src/opentrons/commands/paired_commands.py | 15 +++++++++++ api/src/opentrons/commands/types.py | 27 ++++++++++++++++--- .../protocol_api/instrument_context.py | 4 +++ .../protocol_api/paired_instrument_context.py | 8 ++++++ api/tests/opentrons/api/test_session.py | 25 +++++++++++++++-- api/tests/opentrons/conftest.py | 3 --- 7 files changed, 89 insertions(+), 8 deletions(-) diff --git a/api/src/opentrons/commands/commands.py b/api/src/opentrons/commands/commands.py index 08f367951c0..24324aad09d 100755 --- a/api/src/opentrons/commands/commands.py +++ b/api/src/opentrons/commands/commands.py @@ -255,3 +255,18 @@ def drop_tip( 'text': text } } + + +def move_to( + instrument: InstrumentContext, + location: Union[Location, Well]) -> command_types.MoveToCommand: + location_text = stringify_location(location) + text = 'Moving to {location}'.format(location=location_text) + return { + 'name': command_types.MOVE_TO, + 'payload': { + 'instrument': instrument, + 'location': location, + 'text': text + } + } diff --git a/api/src/opentrons/commands/paired_commands.py b/api/src/opentrons/commands/paired_commands.py index 74477610926..5e9654193b8 100644 --- a/api/src/opentrons/commands/paired_commands.py +++ b/api/src/opentrons/commands/paired_commands.py @@ -178,3 +178,18 @@ def paired_drop_tip( 'text': text } } + + +def paired_move_to( + instruments: Apiv2Instruments, + locations: Apiv2Locations, pub_type: str) -> command_types.MoveToCommand: + location_text = combine_locations(locations) + text = f'{pub_type}: Moving to {location_text}' + return { + 'name': command_types.MOVE_TO, + 'payload': { + 'instruments': instruments, + 'locations': locations, + 'text': text + } + } diff --git a/api/src/opentrons/commands/types.py b/api/src/opentrons/commands/types.py index e747c11c9d5..827b6243979 100755 --- a/api/src/opentrons/commands/types.py +++ b/api/src/opentrons/commands/types.py @@ -36,6 +36,7 @@ AIR_GAP: Final = 'command.AIR_GAP' TOUCH_TIP: Final = 'command.TOUCH_TIP' RETURN_TIP: Final = 'command.RETURN_TIP' +MOVE_TO: Final = 'command.MOVE_TO' # Modules # @@ -450,6 +451,21 @@ class DropTipCommand(TypedDict): payload: Union[DropTipCommandPayload, PairedDropTipCommandPayload] +class MoveToCommand(TypedDict): + name: Literal['command.MOVE_TO'] + payload: Union[MoveToCommandPayload, PairedMoveToCommandPayload] + + +class MoveToCommandPayload( + TextOnlyPayload, SingleLocationPayload, SingleInstrumentPayload): + pass + + +class PairedMoveToCommandPayload( + TextOnlyPayload, MultiLocationPayload, MultiInstrumentPayload): + pass + + Command = Union[ DropTipCommand, PickUpTipCommand, ReturnTipCommand, AirGapCommand, TouchTipCommand, BlowOutCommand, MixCommand, TransferCommand, @@ -463,7 +479,7 @@ class DropTipCommand(TypedDict): TempdeckDeactivateCommand, TempdeckAwaitTempCommand, TempdeckSetTempCommand, MagdeckCalibrateCommand, MagdeckDisengageCommand, MagdeckEngageCommand, ResumeCommand, PauseCommand, DelayCommand, - CommentCommand] + CommentCommand, MoveToCommand] CommandPayload = Union[ @@ -492,7 +508,8 @@ class DropTipCommand(TypedDict): ThermocyclerSetBlockTempCommandPayload, TempdeckAwaitTempCommandPayload, TempdeckSetTempCommandPayload, - PauseCommandPayload, DelayCommandPayload + PauseCommandPayload, DelayCommandPayload, + MoveToCommandPayload, PairedMoveToCommandPayload ] @@ -510,6 +527,10 @@ class CommandMessageFields(CommandMessageMeta, CommandMessageSequence): pass +class MoveToMessage(CommandMessageFields, MoveToCommand): + pass + + class DropTipMessage(CommandMessageFields, DropTipCommand): pass @@ -673,4 +694,4 @@ class CommentMessage(CommandMessageFields, CommentCommand): ThermocyclerExecuteProfileMessage, ThermocyclerSetBlockTempMessage, ThermocyclerOpenMessage, TempdeckSetTempMessage, TempdeckDeactivateMessage, MagdeckEngageMessage, MagdeckDisengageMessage, MagdeckCalibrateMessage, - CommentMessage, DelayMessage, PauseMessage, ResumeMessage] + CommentMessage, DelayMessage, PauseMessage, ResumeMessage, MoveToMessage] diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 8432457223b..9f4f75986e7 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -1110,12 +1110,16 @@ def move_to(self, if isinstance(mod, ThermocyclerContext): mod.flag_unsafe_move(to_loc=location, from_loc=from_loc) + do_publish(self.broker, cmds.move_to, self.move_to, 'before', + None, None, self, location or self._ctx.location_cache) self._implementation.move_to( location=location, force_direct=force_direct, minimum_z_height=minimum_z_height, speed=speed ) + do_publish(self.broker, cmds.move_to, self.move_to, 'after', + None, None, self, location or self._ctx.location_cache) return self @property # type: ignore diff --git a/api/src/opentrons/protocol_api/paired_instrument_context.py b/api/src/opentrons/protocol_api/paired_instrument_context.py index ffe4f1bb532..c7790fb3e4a 100644 --- a/api/src/opentrons/protocol_api/paired_instrument_context.py +++ b/api/src/opentrons/protocol_api/paired_instrument_context.py @@ -808,8 +808,16 @@ def move_to(self, location: types.Location, force_direct: bool = False, to limit individual axis speeds, you can use :py:attr:`.ProtocolContext.max_speeds`. """ + instruments = list(self._instruments.values()) + locations: Optional[List] = None + if location: + locations = self._get_locations(location) + publish_paired(self.broker, cmds.paired_move_to, + 'before', None, instruments, locations) self.paired_instrument_obj.move_to( location, force_direct, minimum_z_height, speed) + publish_paired(self.broker, cmds.paired_move_to, + 'after', None, instruments, locations) return self def _next_available_tip(self) -> Tuple[Labware, Well]: diff --git a/api/tests/opentrons/api/test_session.py b/api/tests/opentrons/api/test_session.py index 7e0539dec63..d89b7b8d569 100755 --- a/api/tests/opentrons/api/test_session.py +++ b/api/tests/opentrons/api/test_session.py @@ -171,7 +171,6 @@ def run(ctx): contents=proto) -@pytest.mark.api2_only async def test_session_extra_labware(main_router, get_labware_fixture, virtual_smoothie_env): proto = ''' @@ -205,7 +204,6 @@ def run(ctx): contents=proto) -@pytest.mark.api2_only async def test_session_bundle(main_router, get_bundle_fixture, virtual_smoothie_env): bundle = get_bundle_fixture('simple_bundle') @@ -257,6 +255,29 @@ def run(ctx): assert 'p10_single_v1' in [pip.name for pip in session2.instruments] +async def test_session_move_to_labware(main_router, + virtual_smoothie_env): + proto = ''' +metadata = {"apiLevel": "2.0"} +def run(ctx): + rack1 = ctx.load_labware('opentrons_96_tiprack_300ul', '1') + rack2 = ctx.load_labware('opentrons_96_tiprack_300ul', '2') + left = ctx.load_instrument('p300_single', 'left', tip_racks=[rack1]) + plate = ctx.load_labware('corning_96_wellplate_360ul_flat', '4') + left.pick_up_tip() + left.move_to(plate['A1'].top()) + left.move_to(plate['A2'].top()) + left.drop_tip() + ''' + session = main_router.session_manager.create('dummy-pipette', + proto) + assert 'p300_single_v1' in [pip.name for pip in session.instruments] + + # Labware that does not have a liquid handling event, but is interacted + # with using a pipette should still show up in the list of labware. + assert 'corning_96_wellplate_360ul_flat' in [lw.type for lw in session.containers] + + async def test_session_run_concurrently( main_router, get_labware_fixture, diff --git a/api/tests/opentrons/conftest.py b/api/tests/opentrons/conftest.py index 70fcf49a33f..d8b017d5920 100755 --- a/api/tests/opentrons/conftest.py +++ b/api/tests/opentrons/conftest.py @@ -286,9 +286,6 @@ def model(request, hardware, loop): # Use with pytest.mark.parametrize(’labware’, [some-labware-name]) # to have a different labware loaded as .container. If not passed, # defaults to the version-appropriate way to do 96 flat - if request.node.get_closest_marker('api2_only')\ - and request.param != build_v2_model: - pytest.skip('only works with hardware controller') try: lw_name = request.getfixturevalue('labware_name') except Exception: From e0c13a4fa6aa59c4d9b095a9f66731c0f156bcc8 Mon Sep 17 00:00:00 2001 From: Laura Cox Date: Mon, 17 May 2021 14:21:07 -0400 Subject: [PATCH 2/2] Do not publish interal calls to move to, fix some tests --- .../protocol_api/instrument_context.py | 38 +++++++++++-------- .../protocol_api/paired_instrument_context.py | 1 + .../opentrons/protocol_api/test_context.py | 2 +- api/tests/opentrons/test_execute.py | 5 +++ api/tests/opentrons/test_simulate.py | 1 + 5 files changed, 30 insertions(+), 17 deletions(-) diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 9f4f75986e7..7b954596b65 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -191,7 +191,8 @@ def aspirate(self, if self.api_version < APIVersion(2, 3) or \ not self._implementation.is_ready_to_aspirate(): if dest.labware.is_well: - self.move_to(dest.labware.as_well().top()) + self.move_to(dest.labware.as_well().top(), + publish=False) else: # TODO(seth,2019/7/29): This should be a warning exposed # via rpc to the runapp @@ -202,9 +203,9 @@ def aspirate(self, "cause over aspiration if the previous command is a " "blow_out.") self._implementation.prepare_for_aspirate() - self.move_to(dest) + self.move_to(dest, publish=False) elif dest != self._ctx.location_cache: - self.move_to(dest) + self.move_to(dest, publish=False) c_vol = self._implementation.get_available_volume() \ if not volume else volume @@ -271,10 +272,10 @@ def dispense(self, else: loc = location.bottom().move( types.Point(0, 0, self.well_bottom_clearance.dispense)) - self.move_to(loc) + self.move_to(loc, publish=False) elif isinstance(location, types.Location): loc = location - self.move_to(location) + self.move_to(location, publish=False) elif location is not None: raise TypeError( f'location should be a Well or Location, but it is {location}') @@ -393,10 +394,10 @@ def blow_out(self, self._log.warning('Blow_out being performed on a tiprack. ' 'Please re-check your code') loc = location.top() - self.move_to(loc) + self.move_to(loc, publish=False) elif isinstance(location, types.Location): loc = location - self.move_to(loc) + self.move_to(loc, publish=False) elif location is not None: raise TypeError( 'location should be a Well or Location, but it is {}' @@ -496,7 +497,7 @@ def touch_tip(self, move_with_z_offset =\ well.as_well().top().point + types.Point(0, 0, v_offset) to_loc = types.Location(move_with_z_offset, well) - self.move_to(to_loc) + self.move_to(to_loc, publish=False) else: raise TypeError( 'location should be a Well, but it is {}'.format(location)) @@ -554,7 +555,7 @@ def air_gap(self, if not loc or not loc.labware.is_well: raise RuntimeError('No previous Well cached to perform air gap') target = loc.labware.as_well().top(height) - self.move_to(target) + self.move_to(target, publish=False) self.aspirate(volume) return self @@ -660,7 +661,7 @@ def pick_up_tip( do_publish(self.broker, cmds.pick_up_tip, self.pick_up_tip, 'before', None, None, self, location=target) - self.move_to(target.top()) + self.move_to(target.top(), publish=False) self._implementation.pick_up_tip( well=target._impl, @@ -780,7 +781,7 @@ def drop_tip( " However, it is a {}".format(location)) do_publish(self.broker, cmds.drop_tip, self.drop_tip, 'before', None, None, self, location=target) - self.move_to(target) + self.move_to(target, publish=False) self._implementation.drop_tip(home_after=home_after) do_publish(self.broker, cmds.drop_tip, self.drop_tip, @@ -1086,7 +1087,8 @@ def move_to(self, location: types.Location, force_direct: bool = False, minimum_z_height: Optional[float] = None, - speed: Optional[float] = None + speed: Optional[float] = None, + publish: bool = True ) -> InstrumentContext: """ Move the instrument. @@ -1101,6 +1103,8 @@ def move_to(self, the straight linear speed of the motion; to limit individual axis speeds, you can use :py:attr:`.ProtocolContext.max_speeds`. + :param publish: Whether a call to this function should publish to the + runlog or not. """ from_loc = self._ctx.location_cache if not from_loc: @@ -1110,16 +1114,18 @@ def move_to(self, if isinstance(mod, ThermocyclerContext): mod.flag_unsafe_move(to_loc=location, from_loc=from_loc) - do_publish(self.broker, cmds.move_to, self.move_to, 'before', - None, None, self, location or self._ctx.location_cache) + if publish: + do_publish(self.broker, cmds.move_to, self.move_to, 'before', + None, None, self, location or self._ctx.location_cache) self._implementation.move_to( location=location, force_direct=force_direct, minimum_z_height=minimum_z_height, speed=speed ) - do_publish(self.broker, cmds.move_to, self.move_to, 'after', - None, None, self, location or self._ctx.location_cache) + if publish: + do_publish(self.broker, cmds.move_to, self.move_to, 'after', + None, None, self, location or self._ctx.location_cache) return self @property # type: ignore diff --git a/api/src/opentrons/protocol_api/paired_instrument_context.py b/api/src/opentrons/protocol_api/paired_instrument_context.py index c7790fb3e4a..70f00e6ea0e 100644 --- a/api/src/opentrons/protocol_api/paired_instrument_context.py +++ b/api/src/opentrons/protocol_api/paired_instrument_context.py @@ -812,6 +812,7 @@ def move_to(self, location: types.Location, force_direct: bool = False, locations: Optional[List] = None if location: locations = self._get_locations(location) + publish_paired(self.broker, cmds.paired_move_to, 'before', None, instruments, locations) self.paired_instrument_obj.move_to( diff --git a/api/tests/opentrons/protocol_api/test_context.py b/api/tests/opentrons/protocol_api/test_context.py index 88540d39ac5..2576a1212ed 100644 --- a/api/tests/opentrons/protocol_api/test_context.py +++ b/api/tests/opentrons/protocol_api/test_context.py @@ -703,7 +703,7 @@ def test_blow_out(ctx, monkeypatch): instr.pick_up_tip() instr.aspirate(10, lw.wells()[0]) - def fake_move(loc): + def fake_move(loc, publish): nonlocal move_location move_location = loc diff --git a/api/tests/opentrons/test_execute.py b/api/tests/opentrons/test_execute.py index 6151bee0152..59a24ea2f69 100644 --- a/api/tests/opentrons/test_execute.py +++ b/api/tests/opentrons/test_execute.py @@ -80,6 +80,7 @@ def emit_runlog(entry): 'Dispensing 4.5 uL into B1 of Dest Plate on 3 at 2.5 uL/sec', 'Touching tip', 'Blowing out at B1 of Dest Plate on 3', + 'Moving to 5', 'Dropping tip into A1 of Trash on 12' ] @@ -106,6 +107,7 @@ def emit_runlog(entry): 'Dispensing 4.5 uL into B1 of Dest Plate on 3 at 2.5 uL/sec', 'Touching tip', 'Blowing out at B1 of Dest Plate on 3', + 'Moving to 5', 'Dropping tip into A1 of Trash on 12' ] @@ -132,6 +134,9 @@ def emit_runlog(entry): 'Dispensing 4.5 uL into B1 of Dest Plate on 3 at 2.5 uL/sec', 'Touching tip', 'Blowing out at B1 of Dest Plate on 3', + 'Moving to 5', + 'Moving to B2 of Dest Plate on 3', + 'Moving to B2 of Dest Plate on 3', 'Dropping tip into A1 of Trash on 12' ] diff --git a/api/tests/opentrons/test_simulate.py b/api/tests/opentrons/test_simulate.py index e0e028a9798..ad5ea10c765 100644 --- a/api/tests/opentrons/test_simulate.py +++ b/api/tests/opentrons/test_simulate.py @@ -41,6 +41,7 @@ def test_simulate_function_json_apiv2(get_json_protocol_fixture): 'Dispensing 4.5 uL into B1 of Dest Plate on 3 at 2.5 uL/sec', 'Touching tip', 'Blowing out at B1 of Dest Plate on 3', + 'Moving to 5', 'Dropping tip into A1 of Trash on 12' ]