From 5d4352e4286b666f867d3abedbf8eae6c14c686f Mon Sep 17 00:00:00 2001 From: Derek Maggio Date: Wed, 17 Jul 2024 08:20:51 -0700 Subject: [PATCH] init let's not push code with syntax errors --- ...2_20_P1000_96_PartialTipPickupSmokeTest.py | 440 ++++++++++++++++++ 1 file changed, 440 insertions(+) create mode 100644 analyses-snapshot-testing/files/protocols/Flex_S_2_20_P1000_96_PartialTipPickupSmokeTest.py diff --git a/analyses-snapshot-testing/files/protocols/Flex_S_2_20_P1000_96_PartialTipPickupSmokeTest.py b/analyses-snapshot-testing/files/protocols/Flex_S_2_20_P1000_96_PartialTipPickupSmokeTest.py new file mode 100644 index 000000000000..ace975ffbb01 --- /dev/null +++ b/analyses-snapshot-testing/files/protocols/Flex_S_2_20_P1000_96_PartialTipPickupSmokeTest.py @@ -0,0 +1,440 @@ +import typing +from dataclasses import dataclass +from opentrons import protocol_api +from opentrons.protocol_api import OFF_DECK, SINGLE, ROW, COLUMN, PARTIAL_COLUMN + +# Need to add transfer, consolidate, and distribute +# Tip pickup around thermocycler + +PipetteNames = typing.Literal["flex_8channel_50", "flex_8channel_1000", "flex_96channel_1000"] +TipRackNames = typing.Literal["opentrons_flex_96_tiprack_1000ul", "opentrons_flex_96_tiprack_200ul", "opentrons_flex_96_tiprack_50ul"] +DropLocationLookupFunction = typing.Callable[[protocol_api.Labware, int], str] + +NUM_ROWS_IN_TIPRACK = 8 +NUM_COLUMNS_IN_TIPRACK = 12 +NUM_TIPS_PER_TIPRACK = NUM_ROWS_IN_TIPRACK * NUM_COLUMNS_IN_TIPRACK + +@dataclass +class LiquidTransferSettings: + source_labware_deck_slot: str + destination_labware_deck_slot: str + +@dataclass +class PipetteConfiguration: + num_to_row_lookup: typing.ClassVar[typing.Dict[int, str]] = { + 1: "A", + 2: "B", + 3: "C", + 4: "D", + 5: "E", + 6: "F", + 7: "G", + 8: "H", + } + number_per_pickup_lookup: typing.ClassVar[typing.Dict[int, str]] = { + 2: "G1", + 3: "F1", + 4: "E1", + 5: "D1", + 6: "C1", + 7: "B1", + } + + pickup_mode: typing.Literal[SINGLE, ROW, COLUMN, PARTIAL_COLUMN] + pickup_start_well: str + pickup_end_well: typing.Optional[str] # for PARTIAL_COLUMN only + + def _calculate_number_per_pickup_for_partial_column(self) -> int: + assert self.pickup_mode == PARTIAL_COLUMN + for num_pickups_key, self.pickup_end_well in self.number_per_pickup_lookup.items(): + if well_name == well_name_key: + return num_pickups_key + else: + raise ValueError( + f"Could not find number of pickups for well {well_name}" + ) + + def _calculate_number_pickups_per_column_for_partial_column(self) -> int: + assert self.pickup_mode == PARTIAL_COLUMN + num_per_pickup = self._calculate_number_per_pickup_for_partial_column() + return NUM_COLUMNS_IN_TIPRACK // num_per_pickup + + def _calculate_max_pickups_for_partial_column(self) -> int: + assert self.pickup_mode == PARTIAL_COLUMN + + return NUM_COLUMNS_IN_TIPRACK * self._calculate_number_pickups_per_column_for_partial_column() + + def _calculate_drop_location_list_for_partial_column(self) -> typing.List[str]: + assert self.pickup_mode == PARTIAL_COLUMN + # we should have the same number of drops as number of pickups + num_drops = self._calculate_number_pickups_per_column_for_partial_column() + num_per_pickup = self._calculate_number_per_pickup_for_partial_column() + + drop_location_list = [] + + for col_number in range(1, NUM_COLUMNS_IN_TIPRACK + 1): + for drop_number in range(1, num_drops + 1): + + drop_row = PipetteConfiguration.num_to_row_lookup[(NUM_ROWS_IN_TIPRACK + 1) - (drop_number * num_per_pickup)] + drop_location_list.append(f"{drop_row}{col_number}") + + + @property + def max_number_of_pickups(self) -> int: + match self.pickup_mode: + case SINGLE: + return NUM_TIPS_PER_TIPRACK + case ROW: + return NUM_ROWS_IN_TIPRACK + case COLUMN: + return NUM_COLUMNS_IN_TIPRACK + case PARTIAL_COLUMN: + return self._calculate_max_pickups_for_partial_column() + case _: + raise ValueError( + f"Unknown pickup mode {self.pickup_mode} for configuration" + ) + + @property + def drop_location_lookup_function(self) -> DropLocationLookupFunction: + match self.pickup_mode: + case SINGLE: + return lambda labware, pickup_number: labware.wells()[pickup_number].well_name + case ROW: + return lambda labware, pickup_number: labware.rows()[pickup_number][0].well_name + case COLUMN: + return lambda labware, pickup_number: labware.columns()[pickup_number][0].well_name + case PARTIAL_COLUMN: + return lambda labware, pickup_number: self._calculate_drop_location_list_for_partial_column()[pickup_number] + case _: + raise ValueError( + f"Unknown pickup mode {self.pickup_mode} for configuration" + ) + + @classmethod + def single_top_left(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=SINGLE, + pickup_start_well="A1", + pickup_end_well=None, + ) + + @classmethod + def single_top_right(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=SINGLE, + pickup_start_well="A12", + pickup_end_well=None, + ) + + @classmethod + def single_bottom_right(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=SINGLE, + pickup_start_well="H12", + pickup_end_well=None, + ) + + @classmethod + def single_bottom_left(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=SINGLE, + pickup_start_well="H1", + pickup_end_well=None, + ) + + @classmethod + def row_top(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=ROW, + pickup_start_well="A1", + pickup_end_well=None, + ) + + @classmethod + def row_bottom(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=ROW, + pickup_start_well="H1", + pickup_end_well=None, + ) + + @classmethod + def column_left(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=COLUMN, + pickup_start_well="A1", + pickup_end_well=None, + ) + + @classmethod + def column_right(cls) -> "PipetteConfiguration": + return cls( + pickup_mode=COLUMN, + pickup_start_well="A12", + pickup_end_well=None, + ) + + @classmethod + def partial_column(cls, num_per_pickup: int) -> "PipetteConfiguration": + assert 2 <= num_per_pickup <= 7 + + return cls( + pickup_mode=PARTIAL_COLUMN, + pickup_start_well="A1", + pickup_end_well=PipetteConfiguration.num_pickup_lookup[num_per_pickup], + ) + + +@dataclass +class PartialTipPickupTestCase: + summary: str + pickup_deck_slot: str + drop_deck_slot: str + pipette_configuration: PipetteConfiguration + liquid_transfer_settings: LiquidTransferSettings + + +NINETY_SIX_CH_TEST_CASES = [ + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with top left tip" + "pickup each tip in column from top to bottom -> shift to column right -> repeat" + ), + pickup_deck_slot="C3", + drop_deck_slot="D2", + pipette_configuration=PipetteConfiguration.single_top_left(), + ), + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with top right tip" + "pickup each tip in column from top to bottom -> shift to column left -> repeat" + ), + pipette_configuration=PipetteConfiguration.single_top_right(), + pickup_deck_slot="C1", + drop_deck_slot="D2", + ), + + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with bottom left tip" + "pickup each tip in column from bottom to top -> shift to column right -> repeat" + ), + pipette_configuration=PipetteConfiguration.single_bottom_left(), + pickup_deck_slot="A3", + drop_deck_slot="A1", + ), + + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with bottom right tip" + "pickup each tip in column from bottom to top -> shift to column left -> repeat" + ), + pipette_configuration=PipetteConfiguration.single_bottom_right(), + pickup_deck_slot="A1", + drop_deck_slot="A3", + ), + + PartialTipPickupTestCase( + summary=( + "row tip pickup" + "starting with top row" + "pickup a row of tips -> shift to row below -> repeat" + ), + pipette_configuration=PipetteConfiguration.row_top(), + pickup_deck_slot="A2", + drop_deck_slot="A3", + ), + + PartialTipPickupTestCase( + summary=( + "row tip pickup" + "starting with bottom row" + "pickup a row tips -> shift to row above -> repeat" + ), + pipette_configuration=PipetteConfiguration.row_bottom(), + pickup_deck_slot="D2", + drop_deck_slot="D1", + ), + + PartialTipPickupTestCase( + summary=( + "column tip pickup" + "starting with left column" + "pickup a column of tips -> shift to column right -> repeat" + ), + pipette_configuration=PipetteConfiguration.column_left(), + pickup_deck_slot="A3", + drop_deck_slot="B3", + ), + + PartialTipPickupTestCase( + summary=( + "column tip pickup" + "starting with right column" + "pickup a column of tips -> shift to column left -> repeat" + ), + pipette_configuration=PipetteConfiguration.column_right(), + pickup_deck_slot="A1", + drop_deck_slot="B1", + ), +] + +EIGHT_CH_TEST_CASES = [ + + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with top left tip" + "pickup each tip in column from top to bottom -> shift to column right -> repeat" + ), + pipette_configuration=PipetteConfiguration.single_top_left(), + pickup_deck_slot="D1", + drop_deck_slot="D2", + ), + + + PartialTipPickupTestCase( + summary=( + "single tip pickup" + "starting with bottom left tip" + "pickup each tip in column from bottom to top -> shift to column right -> repeat" + ), + pipette_configuration=PipetteConfiguration.single_bottom_left(), + pickup_deck_slot="A1", + drop_deck_slot="A2", + ), + + + PartialTipPickupTestCase( + summary=( + "partial column tip pickup" + "starting with bottom left tip" + "pickup bottom 4 tips in column -> pickup top 4 tips in column -> shift to column right -> repeat" + ), + pickup_mode=PARTIAL_COLUMN, + pickup_start_well="H1", + pickup_end_well="E1", + pickup_deck_slot="A1", + drop_deck_slot="A2", + ), + + PartialTipPickupTestCase( + summary=( + "partial column tip pickup" + "starting with bottom left tip" + "pickup bottom 5 tips in column -> shift to column right -> repeat" + ), + pickup_mode=PARTIAL_COLUMN, + pickup_start_well="H1", + pickup_end_well="D1", + pickup_deck_slot="A1", + drop_deck_slot="A2", + ), +] + +def add_parameters(parameters: protocol_api.Parameters): + parameters.add_str( + variable_name="pipette_name", + display_name="Pipette Name", + choice=[ + {"display_name": "8-Channel 50μL", "value": "flex_8channel_50"}, + {"display_name": "8-Channel 1000μL", "value": "flex_8channel_1000"}, + {"display_name": "96-Channel 1000μL", "value": "flex_96channel_1000"}, + ], + default="flex_96channel_1000", + ), + parameters.add_str( + variable_name="tip_rack_name", + display_name="Tip Rack Name", + choice=[ + {"display_name": "50μL", "value": "opentrons_96_tiprack_50ul"}, + {"display_name": "200μL", "value": "opentrons_96_tiprack_200ul"}, + {"display_name": "1000μL", "value": "opentrons_96_tiprack_1000ul"}, + ], + default="opentrons_96_tiprack_1000ul", + ) + parameters.add_str( + variable_name="liquid_transfer_labware_name", + display_name="Liquid Transfer Labware Name", + choice=[ + {"display_name": "Nest 96 Well 100μL", "value": "nest_96_wellplate_100ul_pcr_full_skirt"}, + {"display_name": "Bio-Rad 96 Well 200μL", "value": "biorad_96_wellplate_200ul_pcr"}, + {"display_name": "Corning 96 Well 360μL", "value": "corning_96_wellplate_360ul_flat"}, + ], + default="nest_96_wellplate_100ul_pcr_full_skirt", + ) + +requirements = { + "robotType": "Flex", + "apiLevel": "2.20", +} + +def run(protocol_context: protocol_api.ProtocolContext): + + PIPETTE_NAME = protocol_context.params.pipette_name + TIP_RACK_NAME = protocol_context.params.tip_rack_name + + if "50" in PIPETTE_NAME and "50" not in TIP_RACK_NAME: + raise ValueError("50μL pipette requires 50μL tip rack") + + if "8_channel" in PIPETTE_NAME: + PICKUP_CASES = EIGHT_CH_TEST_CASES + else: + PICKUP_CASES = NINETY_SIX_CH_TEST_CASES + + + pipette = protocol_context.load_instrument( + "flex_96channel_1000", mount="left" + ) + + for pickup_info in SINGLE_MODE_PICKUP_INFOS: + pickup_tip_rack = protocol_context.load_labware( + load_name="opentrons_96_tiprack_1000ul", + location=protocol_api.OFF_DECK + ) + + drop_tip_rack = protocol_context.load_labware( + load_name="opentrons_96_tiprack_1000ul", + location=protocol_api.OFF_DECK + ) + + protocol_context.move_labware( + labware=pickup_tip_rack, + new_location=pickup_info.pickup_deck_slot + ) + + protocol_context.pause("Make sure to load an empty tip rack for the next step") + + protocol_context.move_labware( + labware=drop_tip_rack, + new_location=pickup_info.drop_deck_slot + ) + + pipette.configure_nozzle_layout( + style=pickup_info.pickup_mode, + start=pickup_info.pickup_starting_location, + tip_racks=[pickup_tip_rack], + ) + + for i in range(pickup_info.number_of_pickups): + pipette.pick_up_tip() + pipette.drop_tip(pickup_info.drop_location_lookup_function(drop_tip_rack, i)) + + + + # You don't actually need to move these off deck. + # Want to trick the backend into thinking that we have though + protocol_context.move_labware( + labware=pickup_tip_rack, + new_location=OFF_DECK + ) + + protocol_context.move_labware( + labware=drop_tip_rack, + new_location=OFF_DECK + ) \ No newline at end of file