diff --git a/app-testing/files/protocols/Flex_S_v2_18_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py b/app-testing/files/protocols/Flex_S_v2_18_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py new file mode 100644 index 00000000000..1e4e4c12060 --- /dev/null +++ b/app-testing/files/protocols/Flex_S_v2_18_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py @@ -0,0 +1,610 @@ +############# +# CHANGELOG # +############# + +# ---- +# 2.18 +# ---- + +# - labware.set_offset +# - Runtime Parameters added +# - TrashContainer.top() and Well.top() now return objects of the same type +# - pipette.drop_tip() if location argument not specified the tips will be dropped at different locations in the bin +# - pipette.drop_tip() if location is specified, the tips will be dropped in the same place every time + +# ---- +# 2.17 +# ---- + +# NOTHING NEW +# This protocol is exactly the same as 2.16 Smoke Test V3 +# The only difference is the API version in the metadata +# There were no new positive test cases for 2.17 +# The negative test cases are captured in the 2.17 dispense changes protocol + +# ---- +# 2.16 +# ---- + +# - prepare_to_aspirate added +# - fixed_trash property changed +# - instrument_context.trash_container property changed + +# ---- +# 2.15 +# ---- + +# - move_labware added - Manual Deck State Modification +# - ProtocolContext.load_adapter added +# - OFF_DECK location added + +from opentrons import protocol_api, types + +metadata = { + "protocolName": "Flex Smoke Test - v2.18", + "author": "Derek Maggio ", +} + +requirements = { + "robotType": "OT-3", + "apiLevel": "2.18", +} + +############# +### FLAGS ### +############# + +# prefer to move off deck, instead of waste chute disposal, if possible +PREFER_MOVE_OFF_DECK = True + +################# +### CONSTANTS ### +################# + +HEATER_SHAKER_ADAPTER_NAME = "opentrons_96_pcr_adapter" +HEATER_SHAKER_NAME = "heaterShakerModuleV1" +MAGNETIC_BLOCK_NAME = "magneticBlockV1" +TEMPERATURE_MODULE_ADAPTER_NAME = "opentrons_96_well_aluminum_block" +TEMPERATURE_MODULE_NAME = "temperature module gen2" +THERMOCYCLER_NAME = "thermocycler module gen2" + +TIPRACK_96_ADAPTER_NAME = "opentrons_flex_96_tiprack_adapter" +TIPRACK_96_NAME = "opentrons_flex_96_tiprack_1000ul" + +PIPETTE_96_CHANNEL_NAME = "flex_96channel_1000" + +############################## +# Runtime Parameters Support # +############################## + +# -------------------------- # +# Added in API version: 2.18 # +# -------------------------- # + + +def add_parameters(parameters: protocol_api.Parameters): + reservoir_choices = [ + {"display_name": "Agilent 1 Well 290 mL", "value": "agilent_1_reservoir_290ml"}, + {"display_name": "Nest 1 Well 290 mL", "value": "nest_1_reservoir_290ml"}, + ] + + well_plate_choices = [ + {"display_name": "Nest 96 Well 100 µL", "value": "nest_96_wellplate_100ul_pcr_full_skirt"}, + {"display_name": "Corning 96 Well 360 µL", "value": "corning_96_wellplate_360ul_flat"}, + {"display_name": "Opentrons Tough 96 Well 200 µL", "value": "opentrons_96_wellplate_200ul_pcr_full_skirt"}, + ] + + parameters.add_str( + variable_name="reservoir_name", + display_name="Reservoir Name", + description="Name of the reservoir", + default="nest_1_reservoir_290ml", + choices=reservoir_choices, + ) + + parameters.add_str( + variable_name="well_plate_name", + display_name="Well Plate Name", + description="Name of the well plate", + default="nest_96_wellplate_100ul_pcr_full_skirt", + choices=well_plate_choices, + ) + + parameters.add_bool( + variable_name="use_gripper", + display_name="Use Gripper", + description="Use Gripper for labware movements?", + default=True, + ) + + parameters.add_bool( + variable_name="reset_after_each_move", + display_name="Reset After Each Move", + description="Reset labware after each move?", + default=True, + ) + + parameters.add_float( + variable_name="heater_shaker_temperature", + display_name="Heater Shaker Temperature", + description="Temperature to set the heater shaker to", + default=75.0, + minimum=37.0, + maximum=100.0, + unit="°C", + ) + + parameters.add_int( + variable_name="heater_shaker_speed", + display_name="Heater Shaker Shake Speed", + description="Speed to set the heater shaker to", + default=1000, + minimum=200, + maximum=3000, + unit="seconds", + ) + + +def run(ctx: protocol_api.ProtocolContext) -> None: + + ############################## + # Runtime Parameters Support # + ############################## + + # -------------------------- # + # Added in API version: 2.18 # + # -------------------------- # + + PCR_PLATE_96_NAME = ctx.params.well_plate_name + RESERVOIR_NAME = ctx.params.reservoir_name + USING_GRIPPER = ctx.params.use_gripper + RESET_AFTER_EACH_MOVE = ctx.params.reset_after_each_move + HEATER_SHAKER_TEMPERATURE: float = ctx.params.heater_shaker_temperature + HEATER_SHAKER_SPEED: int = ctx.params.heater_shaker_speed + + ################ + ### FIXTURES ### + ################ + + trash_bin = ctx.load_trash_bin("B3") + waste_chute = ctx.load_waste_chute() + + ############### + ### MODULES ### + ############### + thermocycler = ctx.load_module(THERMOCYCLER_NAME) # A1 & B1 + magnetic_block = ctx.load_module(MAGNETIC_BLOCK_NAME, "C1") + heater_shaker = ctx.load_module(HEATER_SHAKER_NAME, "A3") + temperature_module = ctx.load_module(TEMPERATURE_MODULE_NAME, "D1") + + thermocycler.open_lid() + heater_shaker.open_labware_latch() + + ####################### + ### MODULE ADAPTERS ### + ####################### + + temperature_module_adapter = temperature_module.load_adapter(TEMPERATURE_MODULE_ADAPTER_NAME) + heater_shaker_adapter = heater_shaker.load_adapter(HEATER_SHAKER_ADAPTER_NAME) + + adapters = [temperature_module_adapter, heater_shaker_adapter] + + ############### + ### LABWARE ### + ############### + + source_reservoir = ctx.load_labware(RESERVOIR_NAME, "D2") + dest_pcr_plate = ctx.load_labware(PCR_PLATE_96_NAME, "C2") + + tip_rack_1 = ctx.load_labware(TIPRACK_96_NAME, "A2", adapter=TIPRACK_96_ADAPTER_NAME) + tip_rack_adapter = tip_rack_1.parent + + tip_rack_2 = ctx.load_labware(TIPRACK_96_NAME, "C3") + tip_rack_3 = ctx.load_labware(TIPRACK_96_NAME, "C4") + + tip_racks = [tip_rack_1, tip_rack_2, tip_rack_3] + + ########################## + ### PIPETTE DEFINITION ### + ########################## + + pipette_96_channel = ctx.load_instrument(PIPETTE_96_CHANNEL_NAME, mount="left", tip_racks=tip_racks) + pipette_96_channel.trash_container = trash_bin + + assert isinstance(pipette_96_channel.trash_container, protocol_api.TrashBin) + + ######################## + ### LOAD SOME LIQUID ### + ######################## + + water = ctx.define_liquid(name="water", description="High Quality H₂O", display_color="#42AB2D") + source_reservoir.wells_by_name()["A1"].load_liquid(liquid=water, volume=29000) + + ################################ + ### GRIPPER LABWARE MOVEMENT ### + ################################ + + def get_disposal_preference(): + """ + Get the disposal preference based on the PREFER_MOVE_OFF_DECK flag. + + Returns: + tuple: A tuple containing the disposal preference. The first element is the location preference, + either `protocol_api.OFF_DECK` or `waste_chute`. The second element is a boolean indicating + whether the gripper is being used or not. + """ + return (protocol_api.OFF_DECK, not USING_GRIPPER) if PREFER_MOVE_OFF_DECK else (waste_chute, USING_GRIPPER) + + def run_moves(labware, move_sequences, reset_location, use_gripper): + """ + Perform a series of moves for a given labware using specified move sequences. + + Will perform 2 versions of the moves: + 1. Moves to each location in the sequence, resetting to the reset location after each move. + 2. Moves to each location in the sequence, resetting to the reset location after all moves. + + Args: + labware (str): The labware to be moved. + move_sequences (list): A list of move sequences, where each sequence is a list of locations. + reset_location (str): The location to reset the labware after each move sequence. + use_gripper (bool): Flag indicating whether to use the gripper during the moves. + """ + + def move_to_locations(labware_to_move, move_locations, reset_after_each_move, use_gripper, reset_location): + """ + Move the labware to the specified locations. + + Args: + labware_to_move (str): The labware to be moved. + move_locations (list): A list of locations to move the labware to. + reset_after_each_move (bool): Flag indicating whether to reset the labware after each move. + use_gripper (bool): Flag indicating whether to use the gripper during the moves. + reset_location (str): The location to reset the labware after each move sequence. + """ + + def reset_labware(): + """ + Reset the labware to the reset location. + """ + ctx.move_labware(labware_to_move, reset_location, use_gripper=use_gripper) + + if len(move_locations) == 0: + return + + for location in move_locations: + ctx.move_labware(labware_to_move, location, use_gripper=use_gripper) + + if reset_after_each_move: + reset_labware() + + if not reset_after_each_move: + reset_labware() + + for move_sequence in move_sequences: + move_to_locations(labware, move_sequence, RESET_AFTER_EACH_MOVE, use_gripper, reset_location) + move_to_locations(labware, move_sequence, not RESET_AFTER_EACH_MOVE, use_gripper, reset_location) + + def test_gripper_moves(): + """ + Function to test the movement of the gripper in various locations. + + This function contains several helper functions to perform the movement of labware using a gripper. + Each function performs a sequence of moves, starting with a specific location on the deck. + + Args: + None + + Returns: + None + """ + + def deck_moves(labware, reset_location): + """ + Function to perform the movement of labware, with the inital position being on the deck. + + Args: + pcr_plate (str): The labware to be moved on the deck. + reset_location (str): The reset location on the deck. + + Returns: + None + """ + deck_move_sequence = [ + ["B2"], # Deck Moves + ["C3"], # Staging Area Slot 3 Moves + ["C4", "D4"], # Staging Area Slot 4 Moves + [thermocycler, temperature_module_adapter, heater_shaker_adapter, magnetic_block], # Module Moves + ] + + run_moves(labware, deck_move_sequence, reset_location, USING_GRIPPER) + + def staging_area_slot_3_moves(labware, reset_location): + """ + Function to perform the movement of labware, with the inital position being on staging area slot 3. + + Args: + labware (str): The labware to be moved in staging area slot 3. + reset_location (str): The reset location in staging area slot 3. + + Returns: + None + """ + staging_area_slot_3_move_sequence = [ + ["B2", "C2"], # Deck Moves + [], # Don't have Staging Area Slot 3 open + ["C4", "D4"], # Staging Area Slot 4 Moves + [thermocycler, temperature_module_adapter, heater_shaker_adapter, magnetic_block], # Module Moves + ] + + run_moves(labware, staging_area_slot_3_move_sequence, reset_location, USING_GRIPPER) + + def staging_area_slot_4_moves(labware, reset_location): + """ + Function to perform the movement of labware, with the inital position being on staging area slot 4. + + Args: + labware (str): The labware to be moved in staging area slot 4. + reset_location (str): The reset location in staging area slot 4. + + Returns: + None + """ + staging_area_slot_4_move_sequence = [ + ["C2", "B2"], # Deck Moves + ["C3"], # Staging Area Slot 3 Moves + ["C4"], # Staging Area Slot 4 Moves + [thermocycler, temperature_module_adapter, heater_shaker_adapter, magnetic_block], # Module Moves + ] + + run_moves(labware, staging_area_slot_4_move_sequence, reset_location, USING_GRIPPER) + + def module_moves(labware, module_locations): + """ + Function to perform the movement of labware, with the inital position being on a module. + + Args: + labware (str): The labware to be moved with modules. + module_locations (list): The locations of the modules. + + Returns: + None + """ + module_move_sequence = [ + ["C2", "B2"], # Deck Moves + ["C3"], # Staging Area Slot 3 Moves + ["C4", "D4"], # Staging Area Slot 4 Moves + ] + + for module_starting_location in module_locations: + labware_move_to_locations = module_locations.copy() + labware_move_to_locations.remove(module_starting_location) + all_sequences = module_move_sequence.copy() + all_sequences.append(labware_move_to_locations) + ctx.move_labware(labware, module_starting_location, use_gripper=USING_GRIPPER) + run_moves(labware, all_sequences, module_starting_location, USING_GRIPPER) + + DECK_MOVE_RESET_LOCATION = "C2" + STAGING_AREA_SLOT_3_RESET_LOCATION = "C3" + STAGING_AREA_SLOT_4_RESET_LOCATION = "D4" + + deck_moves(dest_pcr_plate, DECK_MOVE_RESET_LOCATION) + + ctx.move_labware(dest_pcr_plate, STAGING_AREA_SLOT_3_RESET_LOCATION, use_gripper=USING_GRIPPER) + staging_area_slot_3_moves(dest_pcr_plate, STAGING_AREA_SLOT_3_RESET_LOCATION) + + ctx.move_labware(dest_pcr_plate, STAGING_AREA_SLOT_4_RESET_LOCATION, use_gripper=USING_GRIPPER) + staging_area_slot_4_moves(dest_pcr_plate, STAGING_AREA_SLOT_4_RESET_LOCATION) + + module_locations = [thermocycler, magnetic_block] + adapters + module_moves(dest_pcr_plate, module_locations) + + ctx.move_labware(dest_pcr_plate, DECK_MOVE_RESET_LOCATION, use_gripper=USING_GRIPPER) + + def test_manual_moves(): + # In C4 currently + ctx.move_labware(source_reservoir, "D4", use_gripper=not USING_GRIPPER) + + def test_pipetting(): + def test_partial_tip_pickup_usage(): + pipette_96_channel.configure_nozzle_layout(style=protocol_api.COLUMN, start="A12") + + for i in range(1, 13): + + pipette_96_channel.pick_up_tip(tip_rack_2[f"A{i}"]) + + pipette_96_channel.aspirate(5, source_reservoir["A1"]) + pipette_96_channel.touch_tip() + + pipette_96_channel.dispense(5, dest_pcr_plate[f"A{i}"]) + + if i == 1: + ctx.pause( + "Watch the next 6 tips drop in the waste chute. They should drop in the same location of the waste chute each time." + ) + + if i == 7: + ctx.pause( + "Watch the next 6 tips drop in the waste chute. They should drop in different locations of the waste chute each time." + ) + + if i <= 6: + pipette_96_channel.drop_tip(waste_chute) + else: + pipette_96_channel.drop_tip() + + # leave this dropping in waste chute, do not use get_disposal_preference + # want to test partial drop + ctx.move_labware(tip_rack_2, waste_chute, use_gripper=USING_GRIPPER) + + def test_full_tip_rack_usage(): + pipette_96_channel.configure_nozzle_layout(style=protocol_api.ALL, start="A1") + pipette_96_channel.pick_up_tip(tip_rack_1["A1"]) + + pipette_96_channel.aspirate(5, source_reservoir["A1"]) + pipette_96_channel.touch_tip() + + pipette_96_channel.air_gap(height=30) + + pipette_96_channel.blow_out(waste_chute) + + pipette_96_channel.aspirate(5, source_reservoir["A1"]) + pipette_96_channel.touch_tip() + + pipette_96_channel.air_gap(height=30) + pipette_96_channel.blow_out(trash_bin) + + pipette_96_channel.aspirate(10, source_reservoir["A1"]) + pipette_96_channel.touch_tip() + + pipette_96_channel.dispense(10, dest_pcr_plate["A1"]) + pipette_96_channel.mix(repetitions=5, volume=15) + pipette_96_channel.return_tip() + + ctx.move_labware(tip_rack_1, get_disposal_preference()[0], use_gripper=get_disposal_preference()[1]) + ctx.move_labware(tip_rack_3, tip_rack_adapter, use_gripper=USING_GRIPPER) + + pipette_96_channel.pick_up_tip(tip_rack_3["A1"]) + pipette_96_channel.transfer( + volume=10, + source=source_reservoir["A1"], + dest=dest_pcr_plate["A1"], + new_tip="never", + touch_tip=True, + blow_out=True, + blowout_location="trash", + mix_before=(3, 5), + mix_after=(1, 5), + ) + pipette_96_channel.return_tip() + + test_partial_tip_pickup_usage() + test_full_tip_rack_usage() + + def test_module_usage(): + def test_thermocycler(): + thermocycler.close_lid() + + thermocycler.set_block_temperature(75.0, hold_time_seconds=5.0) + thermocycler.set_lid_temperature(80.0) + thermocycler.deactivate() + + def test_heater_shaker(): + heater_shaker.open_labware_latch() + heater_shaker.close_labware_latch() + + heater_shaker.set_target_temperature(HEATER_SHAKER_TEMPERATURE) + heater_shaker.set_and_wait_for_shake_speed(HEATER_SHAKER_SPEED) + heater_shaker.wait_for_temperature() + + heater_shaker.deactivate_heater() + heater_shaker.deactivate_shaker() + + def test_temperature_module(): + temperature_module.set_temperature(80) + temperature_module.set_temperature(10) + temperature_module.deactivate() + + def test_magnetic_block(): + pass + + test_thermocycler() + test_heater_shaker() + test_temperature_module() + test_magnetic_block() + + def test_labware_set_offset(): + """Test the labware.set_offset method.""" + ###################### + # labware.set_offset # + ###################### + + # -------------------------- # + # Added in API version: 2.18 # + # -------------------------- # + + SET_OFFSET_AMOUNT = 10.0 + ctx.move_labware(labware=source_reservoir, new_location=protocol_api.OFF_DECK, use_gripper=False) + pipette_96_channel.pick_up_tip(tip_rack_3["A1"]) + pipette_96_channel.move_to(dest_pcr_plate.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of the PCR Plate, well A1, in slot C2? It should be at the LPC calibrated height.") + + dest_pcr_plate.set_offset( + x=0.0, + y=0.0, + z=SET_OFFSET_AMOUNT, + ) + + pipette_96_channel.move_to(dest_pcr_plate.wells_by_name()["A1"].top()) + ctx.pause( + "Is the pipette tip in the middle of the PCR Plate, well A1, in slot C2? It should be 10mm higher than the LPC calibrated height." + ) + + ctx.move_labware(labware=dest_pcr_plate, new_location="D2", use_gripper=False) + pipette_96_channel.move_to(dest_pcr_plate.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of the PCR Plate, well A1, in slot D2? It should be at the LPC calibrated height.") + + dest_pcr_plate.set_offset( + x=0.0, + y=0.0, + z=SET_OFFSET_AMOUNT, + ) + + pipette_96_channel.move_to(dest_pcr_plate.wells_by_name()["A1"].top()) + ctx.pause( + "Is the pipette tip in the middle of the PCR Plate, well A1, in slot D2? It should be 10mm higher than the LPC calibrated height." + ) + + ctx.move_labware(labware=dest_pcr_plate, new_location="C2", use_gripper=False) + pipette_96_channel.move_to(dest_pcr_plate.wells_by_name()["A1"].top()) + + ctx.pause( + "Is the pipette tip in the middle of the PCR Plate, well A1, in slot C2? It should be 10mm higher than the LPC calibrated height." + ) + + ctx.move_labware(labware=source_reservoir, new_location="D2", use_gripper=False) + pipette_96_channel.move_to(source_reservoir.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of the reservoir , well A1, in slot D2? It should be at the LPC calibrated height.") + + pipette_96_channel.return_tip() + ctx.move_labware(tip_rack_3, get_disposal_preference()[0], use_gripper=get_disposal_preference()[1]) + + ctx.pause("!!!!!!!!!!YOU NEED TO REDO LPC!!!!!!!!!!") + + def test_unique_top_methods(): + """ + Test the unique top() methods for TrashBin and WasteChute. + + Well objects should remain the same + """ + ######################## + # unique top() methods # + ######################## + + # ---------------------------- # + # Changed in API version: 2.18 # + # ---------------------------- # + + assert isinstance(trash_bin.top(), protocol_api.TrashBin) + assert isinstance(waste_chute.top(), protocol_api.WasteChute) + assert isinstance(source_reservoir.wells_by_name()["A1"].top(), types.Location) + + ################################################################################################### + ### THE ORDER OF THESE FUNCTION CALLS MATTER. CHANGING THEM WILL CAUSE THE PROTOCOL NOT TO WORK ### + ################################################################################################### + test_pipetting() + test_gripper_moves() + test_module_usage() + test_manual_moves() + test_labware_set_offset() + test_unique_top_methods() + + ################################################################################################### + ### THE ORDER OF THESE FUNCTION CALLS MATTER. CHANGING THEM WILL CAUSE THE PROTOCOL NOT TO WORK ### + ################################################################################################### + + +# Cannot test in this protocol +# - Waste Chute w/ Lid diff --git a/app-testing/files/protocols/OT2_S_v2_13_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_13_P300M_P20S_HS_TC_TM_SmokeTestV3.py index da7b3ab385b..990081a9354 100644 --- a/app-testing/files/protocols/OT2_S_v2_13_P300M_P20S_HS_TC_TM_SmokeTestV3.py +++ b/app-testing/files/protocols/OT2_S_v2_13_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -12,6 +12,14 @@ requirements = {"robotType": "OT-2", "apiLevel": "2.13"} +############# +# CHANGELOG # +############# + +# 2.13 + +# - Heater-Shaker Module support added + def run(ctx: protocol_api.ProtocolContext) -> None: """This method is run by the protocol engine.""" @@ -58,6 +66,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + # modules https://docs.opentrons.com/v2/new_modules.html#available-modules hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) temperature_module = ctx.load_module("temperature module gen2", temperature_position) diff --git a/app-testing/files/protocols/OT2_S_v2_14_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_14_P300M_P20S_HS_TC_TM_SmokeTestV3.py index 4ff0f563d37..50a0e707f38 100644 --- a/app-testing/files/protocols/OT2_S_v2_14_P300M_P20S_HS_TC_TM_SmokeTestV3.py +++ b/app-testing/files/protocols/OT2_S_v2_14_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -12,6 +12,23 @@ requirements = {"robotType": "OT-2", "apiLevel": "2.14"} +############# +# CHANGELOG # +############# + +# ---- +# 2.14 +# ---- + +# - ProtocolContext.defined_liquid and Well.load_liquid added +# - load_labware without parameters should still find the labware + +# ---- +# 2.13 +# ---- + +# - Heater-Shaker Module support added + def run(ctx: protocol_api.ProtocolContext) -> None: """This method is run by the protocol engine.""" @@ -58,6 +75,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + # modules https://docs.opentrons.com/v2/new_modules.html#available-modules hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) temperature_module = ctx.load_module("temperature module gen2", temperature_position) @@ -71,7 +96,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: hs_plate = hs_module.load_labware("opentrons_96_pcr_adapter_nest_wellplate_100ul_pcr_full_skirt") tc_plate = thermocycler_module.load_labware("nest_96_wellplate_100ul_pcr_full_skirt") - # A 2.14 difference, no params specified, still should find it. + ################################### + # Load Labware with no parameters # + ################################### + + # -------------------------- # + # Fixed in API version: 2.14 # + # -------------------------- # + custom_labware = ctx.load_labware( "cpx_4_tuberack_100ul", custom_lw_position, @@ -102,7 +134,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: logo_destination_plate.wells_by_name()["E5"], ] - # >= 2.14 define_liquid and load_liquid + ####################################### + # define_liquid & load_liquid Support # + ####################################### + + # -------------------------- # + # Added in API version: 2.14 # + # -------------------------- # + water = ctx.define_liquid( name="water", description="H₂O", display_color="#42AB2D" ) # subscript 2 https://www.compart.com/en/unicode/U+2082 diff --git a/app-testing/files/protocols/OT2_S_v2_15_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_15_P300M_P20S_HS_TC_TM_SmokeTestV3.py index aeb151c2267..4e91ce6964f 100644 --- a/app-testing/files/protocols/OT2_S_v2_15_P300M_P20S_HS_TC_TM_SmokeTestV3.py +++ b/app-testing/files/protocols/OT2_S_v2_15_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -12,6 +12,31 @@ requirements = {"robotType": "OT-2", "apiLevel": "2.15"} +############# +# CHANGELOG # +############# + +# ---- +# 2.15 +# ---- + +# - move_labware added - Manual Deck State Modification +# - ProtocolContext.load_adapter added +# - OFF_DECK location added + +# ---- +# 2.14 +# ---- + +# - ProtocolContext.defined_liquid and Well.load_liquid added +# - load_labware without parameters should still find the labware + +# ---- +# 2.13 +# ---- + +# - Heater-Shaker Module support added + def run(ctx: protocol_api.ProtocolContext) -> None: """This method is run by the protocol engine.""" @@ -58,6 +83,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + # modules https://docs.opentrons.com/v2/new_modules.html#available-modules hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) temperature_module = ctx.load_module("temperature module gen2", temperature_position) @@ -72,7 +105,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: hs_plate = hs_module.load_labware(name="nest_96_wellplate_100ul_pcr_full_skirt", adapter="opentrons_96_pcr_adapter") tc_plate = thermocycler_module.load_labware("nest_96_wellplate_100ul_pcr_full_skirt") - # A 2.14 difference, no params specified, still should find it. + ################################### + # Load Labware with no parameters # + ################################### + + # -------------------------- # + # Fixed in API version: 2.14 # + # -------------------------- # + custom_labware = ctx.load_labware( "cpx_4_tuberack_100ul", custom_lw_position, @@ -103,7 +143,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: logo_destination_plate.wells_by_name()["E5"], ] - # >= 2.14 define_liquid and load_liquid + ####################################### + # define_liquid & load_liquid Support # + ####################################### + + # -------------------------- # + # Added in API version: 2.14 # + # -------------------------- # + water = ctx.define_liquid( name="water", description="H₂O", display_color="#42AB2D" ) # subscript 2 https://www.compart.com/en/unicode/U+2082 diff --git a/app-testing/files/protocols/OT2_S_v2_16_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_16_P300M_P20S_HS_TC_TM_SmokeTestV3.py index 4c0ceafb2ba..7b2de5155f0 100644 --- a/app-testing/files/protocols/OT2_S_v2_16_P300M_P20S_HS_TC_TM_SmokeTestV3.py +++ b/app-testing/files/protocols/OT2_S_v2_16_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -12,6 +12,39 @@ requirements = {"robotType": "OT-2", "apiLevel": "2.16"} +############# +# CHANGELOG # +############# + +# ---- +# 2.16 +# ---- + +# - prepare_to_aspirate added +# - fixed_trash property changed +# - instrument_context.trash_container property changed + +# ---- +# 2.15 +# ---- + +# - move_labware added - Manual Deck State Modification +# - ProtocolContext.load_adapter added +# - OFF_DECK location added + +# ---- +# 2.14 +# ---- + +# - ProtocolContext.defined_liquid and Well.load_liquid added +# - load_labware without parameters should still find the labware + +# ---- +# 2.13 +# ---- + +# - Heater-Shaker Module support added + def run(ctx: protocol_api.ProtocolContext) -> None: """This method is run by the protocol engine.""" @@ -58,6 +91,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + # modules https://docs.opentrons.com/v2/new_modules.html#available-modules hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) temperature_module = ctx.load_module("temperature module gen2", temperature_position) @@ -72,7 +113,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: hs_plate = hs_module.load_labware(name="nest_96_wellplate_100ul_pcr_full_skirt", adapter="opentrons_96_pcr_adapter") tc_plate = thermocycler_module.load_labware("nest_96_wellplate_100ul_pcr_full_skirt") - # A 2.14 difference, no params specified, still should find it. + ################################### + # Load Labware with no parameters # + ################################### + + # -------------------------- # + # Fixed in API version: 2.14 # + # -------------------------- # + custom_labware = ctx.load_labware( "cpx_4_tuberack_100ul", custom_lw_position, @@ -103,7 +151,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: logo_destination_plate.wells_by_name()["E5"], ] - # >= 2.14 define_liquid and load_liquid + ####################################### + # define_liquid & load_liquid Support # + ####################################### + + # -------------------------- # + # Added in API version: 2.14 # + # -------------------------- # + water = ctx.define_liquid( name="water", description="H₂O", display_color="#42AB2D" ) # subscript 2 https://www.compart.com/en/unicode/U+2082 diff --git a/app-testing/files/protocols/OT2_S_v2_17_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_17_P300M_P20S_HS_TC_TM_SmokeTestV3.py index 1c0c1f9802d..822e64ab6e1 100644 --- a/app-testing/files/protocols/OT2_S_v2_17_P300M_P20S_HS_TC_TM_SmokeTestV3.py +++ b/app-testing/files/protocols/OT2_S_v2_17_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -12,19 +12,48 @@ requirements = {"robotType": "OT-2", "apiLevel": "2.17"} +############# +# CHANGELOG # +############# -######################### -#### LOOK AT THIS ####### -######################### +# ---- +# 2.17 +# ---- +# NOTHING NEW # This protocol is exactly the same as 2.16 Smoke Test V3 # The only difference is the API version in the metadata # There were no new positive test cases for 2.17 # The negative test cases are captured in the 2.17 dispense changes protcol -######################### -#### LOOK AT THIS ####### -######################### +# ---- +# 2.16 +# ---- + +# - prepare_to_aspirate added +# - fixed_trash property changed +# - instrument_context.trash_container property changed + +# ---- +# 2.15 +# ---- + +# - move_labware added - Manual Deck State Modification +# - ProtocolContext.load_adapter added +# - OFF_DECK location added + +# ---- +# 2.14 +# ---- + +# - ProtocolContext.defined_liquid and Well.load_liquid added +# - load_labware without parameters should still find the labware + +# ---- +# 2.13 +# ---- + +# - Heater-Shaker Module support added def run(ctx: protocol_api.ProtocolContext) -> None: @@ -72,7 +101,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) - # modules https://docs.opentrons.com/v2/new_modules.html#available-modules + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) temperature_module = ctx.load_module("temperature module gen2", temperature_position) thermocycler_module = ctx.load_module("thermocycler module gen2") @@ -86,7 +122,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: hs_plate = hs_module.load_labware(name="nest_96_wellplate_100ul_pcr_full_skirt", adapter="opentrons_96_pcr_adapter") tc_plate = thermocycler_module.load_labware("nest_96_wellplate_100ul_pcr_full_skirt") - # A 2.14 difference, no params specified, still should find it. + ################################### + # Load Labware with no parameters # + ################################### + + # -------------------------- # + # Fixed in API version: 2.14 # + # -------------------------- # + custom_labware = ctx.load_labware( "cpx_4_tuberack_100ul", custom_lw_position, @@ -117,7 +160,14 @@ def run(ctx: protocol_api.ProtocolContext) -> None: logo_destination_plate.wells_by_name()["E5"], ] - # >= 2.14 define_liquid and load_liquid + ####################################### + # define_liquid & load_liquid Support # + ####################################### + + # -------------------------- # + # Added in API version: 2.14 # + # -------------------------- # + water = ctx.define_liquid( name="water", description="H₂O", display_color="#42AB2D" ) # subscript 2 https://www.compart.com/en/unicode/U+2082 @@ -147,7 +197,7 @@ def run(ctx: protocol_api.ProtocolContext) -> None: # Added in API version: 2.15 # # -------------------------- # - # Putting steps for this at beginning of protocol so you can do the manual stuff + # Putting steps for this at beginning of protocol so y # >= 2.14 define_liquid and load_liquidou can do the manual stuff # then walk away to let the rest of the protocol execute # The test flow is as follows: diff --git a/app-testing/files/protocols/OT2_S_v2_18_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/app-testing/files/protocols/OT2_S_v2_18_P300M_P20S_HS_TC_TM_SmokeTestV3.py new file mode 100644 index 00000000000..a724789eb1f --- /dev/null +++ b/app-testing/files/protocols/OT2_S_v2_18_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -0,0 +1,595 @@ +"""Smoke Test v3.0 """ + +# https://opentrons.atlassian.net/projects/RQA?selectedItem=com.atlassian.plugins.atlassian-connect-plugin:com.kanoah.test-manager__main-project-page#!/testCase/QB-T497 + +############# +# CHANGELOG # +############# + +# ---- +# 2.18 +# ---- + +# - labware.set_offset +# - Runtime Parameters added +# - TrashContainer.top() and Well.top() now return objects of the same type +# - pipette.drop_tip() if location argument not specified the tips will be dropped at different locations in the bin +# - pipette.drop_tip() if location is specified, the tips will be dropped in the same place every time + +# ---- +# 2.17 +# ---- + +# NOTHING NEW +# This protocol is exactly the same as 2.16 Smoke Test V3 +# The only difference is the API version in the metadata +# There were no new positive test cases for 2.17 +# The negative test cases are captured in the 2.17 dispense changes protocol + +# ---- +# 2.16 +# ---- + +# - prepare_to_aspirate added +# - fixed_trash property changed +# - instrument_context.trash_container property changed + +# ---- +# 2.15 +# ---- + +# - move_labware added - Manual Deck State Modification +# - ProtocolContext.load_adapter added +# - OFF_DECK location added + +# ---- +# 2.14 +# ---- + +# - ProtocolContext.defined_liquid and Well.load_liquid added +# - load_labware without parameters should still find the labware + +# ---- +# 2.13 +# ---- + +# - Heater-Shaker Module support added + +from opentrons import protocol_api, types + +metadata = { + "protocolName": "🛠️ 2.18 Smoke Test V3 🪄", + "author": "Opentrons Engineering ", + "source": "Software Testing Team", + "description": ("Description of the protocol that is longish \n has \n returns and \n emoji 😊 ⬆️ "), +} + +requirements = {"robotType": "OT-2", "apiLevel": "2.18"} + +############################## +# Runtime Parameters Support # +############################## + +# -------------------------- # +# Added in API version: 2.18 # +# -------------------------- # + + +def add_parameters(parameters: protocol_api.Parameters): + reservoir_choices = [ + {"display_name": "Nest 12 Well 15 mL", "value": "nest_12_reservoir_15ml"}, + {"display_name": "USA Scientific 12 Well 22 mL", "value": "usascientific_12_reservoir_22ml"}, + ] + + well_plate_choices = [ + {"display_name": "Nest 96 Well 100 µL", "value": "nest_96_wellplate_100ul_pcr_full_skirt"}, + {"display_name": "Corning 96 Well 360 µL", "value": "corning_96_wellplate_360ul_flat"}, + {"display_name": "Opentrons Tough 96 Well 200 µL", "value": "opentrons_96_wellplate_200ul_pcr_full_skirt"}, + ] + + parameters.add_str( + variable_name="reservoir_name", + display_name="Reservoir Name", + description="Name of the reservoir", + default="nest_12_reservoir_15ml", + choices=reservoir_choices, + ) + + parameters.add_str( + variable_name="well_plate_name", + display_name="Well Plate Name", + description="Name of the well plate", + default="nest_96_wellplate_100ul_pcr_full_skirt", + choices=well_plate_choices, + ) + + parameters.add_int( + variable_name="delay_time", + display_name="Delay Time", + description="Time to delay in seconds", + default=3, + minimum=1, + maximum=10, + unit="seconds", + ) + + parameters.add_bool( + variable_name="robot_lights", + display_name="Robot Lights", + description="Turn on the robot lights?", + default=True, + ) + + parameters.add_float( + variable_name="heater_shaker_temperature", + display_name="Heater Shaker Temperature", + description="Temperature to set the heater shaker to", + default=38.0, + minimum=37.0, + maximum=100.0, + unit="°C", + ) + + +def run(ctx: protocol_api.ProtocolContext) -> None: + """This method is run by the protocol engine.""" + + ############################## + # Runtime Parameters Support # + ############################## + + # -------------------------- # + # Added in API version: 2.18 # + # -------------------------- # + + RESERVOIR_NAME: str = ctx.params.reservoir_name + WELL_PLATE_NAME: str = ctx.params.well_plate_name + DELAY_TIME: int = ctx.params.delay_time + ROBOT_LIGHTS: bool = ctx.params.robot_lights + HEATER_SHAKER_TEMPERATURE: float = ctx.params.heater_shaker_temperature + + ctx.set_rail_lights(ROBOT_LIGHTS) + ctx.comment(f"Let there be light! {ctx.rail_lights_on} 🌠🌠🌠") + ctx.comment(f"Is the door is closed? {ctx.door_closed} 🚪🚪🚪") + ctx.comment(f"Is this a simulation? {ctx.is_simulating()} 🔮🔮🔮") + ctx.comment(f"Running against API Version: {ctx.api_version}") + + # deck positions + tips_300ul_position = "5" + tips_20ul_position = "4" + dye_source_position = "3" + logo_position = "2" + temperature_position = "9" + custom_lw_position = "6" + hs_position = "1" + + # Thermocycler has a default position that covers Slots 7, 8, 10, and 11. + # This is the only valid location for the Thermocycler on the OT-2 deck. + # This position is a default parameter when declaring the TC so you do not need to specify. + + # 300ul tips + tips_300ul = [ + ctx.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=tips_300ul_position, + label="300ul tips", + ) + ] + + # 20ul tips + tips_20ul = [ + ctx.load_labware( + load_name="opentrons_96_tiprack_20ul", + location=tips_20ul_position, + label="20ul tips", + ) + ] + + # pipettes + pipette_left = ctx.load_instrument(instrument_name="p300_multi_gen2", mount="left", tip_racks=tips_300ul) + + pipette_right = ctx.load_instrument(instrument_name="p20_single_gen2", mount="right", tip_racks=tips_20ul) + + ######################### + # Heater-Shaker Support # + ######################### + + # -------------------------- # + # Added in API version: 2.13 # + # -------------------------- # + + hs_module = ctx.load_module("heaterShakerModuleV1", hs_position) + temperature_module = ctx.load_module("temperature module gen2", temperature_position) + thermocycler_module = ctx.load_module("thermocycler module gen2") + + # module labware + temp_adapter = temperature_module.load_adapter("opentrons_96_well_aluminum_block") + temp_plate = temp_adapter.load_labware( + WELL_PLATE_NAME, + label="Temperature-Controlled plate", + ) + hs_plate = hs_module.load_labware(name=WELL_PLATE_NAME, adapter="opentrons_96_pcr_adapter") + tc_plate = thermocycler_module.load_labware(WELL_PLATE_NAME) + + ################################### + # Load Labware with no parameters # + ################################### + + # -------------------------- # + # Fixed in API version: 2.14 # + # -------------------------- # + + custom_labware = ctx.load_labware( + "cpx_4_tuberack_100ul", + custom_lw_position, + label="4 custom tubes", + ) + + # create plates and pattern list + logo_destination_plate = ctx.load_labware( + load_name=WELL_PLATE_NAME, + location=logo_position, + label="logo destination", + ) + + dye_container = ctx.load_labware( + load_name=RESERVOIR_NAME, + location=dye_source_position, + label="dye container", + ) + + dye_source = dye_container.wells_by_name()["A2"] + + # Well Location set-up + dye_destination_wells = [ + logo_destination_plate.wells_by_name()["C7"], + logo_destination_plate.wells_by_name()["D6"], + logo_destination_plate.wells_by_name()["D7"], + logo_destination_plate.wells_by_name()["D8"], + logo_destination_plate.wells_by_name()["E5"], + ] + + ####################################### + # define_liquid & load_liquid Support # + ####################################### + + # -------------------------- # + # Added in API version: 2.14 # + # -------------------------- # + + water = ctx.define_liquid( + name="water", description="H₂O", display_color="#42AB2D" + ) # subscript 2 https://www.compart.com/en/unicode/U+2082 + + acetone = ctx.define_liquid( + name="acetone", description="C₃H₆O", display_color="#38588a" + ) # subscript 3 https://www.compart.com/en/unicode/U+2083 + # subscript 6 https://www.compart.com/en/unicode/U+2086 + + dye_container.wells_by_name()["A1"].load_liquid(liquid=water, volume=4000) + dye_container.wells_by_name()["A2"].load_liquid(liquid=water, volume=2000) + dye_container.wells_by_name()["A5"].load_liquid(liquid=acetone, volume=555.55555) + + # 2 different liquids in the same well + dye_container.wells_by_name()["A8"].load_liquid(liquid=water, volume=900.00) + dye_container.wells_by_name()["A8"].load_liquid(liquid=acetone, volume=1001.11) + + hs_module.close_labware_latch() + + pipette_right.pick_up_tip() + + ################################## + # Manual Deck State Modification # + ################################## + + # -------------------------- # + # Added in API version: 2.15 # + # -------------------------- # + + # Putting steps for this at beginning of protocol so y # >= 2.14 define_liquid and load_liquidou can do the manual stuff + # then walk away to let the rest of the protocol execute + + # The test flow is as follows: + # 1. Remove the existing PCR plate from slot 2 + # 2. Move the reservoir from slot 3 to slot 2 + # 3. Pickup P20 tip, move pipette to reservoir A1 in slot 2 + # 4. Pause and ask user to validate that the tip is in the middle of reservoir A1 in slot 2 + # 5. Move the reservoir back to slot 3 from slot 2 + # 6. Move pipette to reservoir A1 in slot 3 + # 7. Pause and ask user to validate that the tip is in the middle of reservoir A1 in slot 3 + # 8. Move custom labware from slot 6 to slot 2 + # 9. Move pipette to well A1 in slot 2 + # 10. Pause and ask user to validate that the tip is in the middle of well A1 in slot 2 + # 11. Move the custom labware back to slot 6 from slot 2 + # 12. Move pipette to well A1 in slot 6 + # 13. Pause and ask user to validate that the tip is in the middle of well A1 in slot 6 + # 14. Move the offdeck PCR plate back to slot 2 + # 15. Move pipette to well A1 in slot 2 + # 16. Pause and ask user to validate that the tip is in the middle of well A1 in slot 2 + + # In effect, nothing will actually change to the protocol, + # but we will be able to test that the UI responds appropriately. + + # Note: + # logo_destination_plate is a nest_96_wellplate_100ul_pcr_full_skirt - starting position is slot 2 + # dye_container is aRESERVOIR_NAME- starting position is slot 3 + + # Step 1 + ctx.move_labware( + labware=logo_destination_plate, + new_location=protocol_api.OFF_DECK, + ) + + # Step 2 + ctx.move_labware(labware=dye_container, new_location="2") + + # Step 3 + pipette_right.move_to(location=dye_container.wells_by_name()["A1"].top()) + + # Step 4 + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 2?") + + # Step 5 + ctx.move_labware(labware=dye_container, new_location="3") + + # Step 6 + pipette_right.move_to(location=dye_container.wells_by_name()["A1"].top()) + + # Step 7 + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 3?") + + # Step 8 + ctx.move_labware(labware=custom_labware, new_location="2") + + # Step 9 + pipette_right.move_to(location=custom_labware.wells_by_name()["A1"].top()) + + # Step 10 + ctx.pause("Is the pipette tip in the middle of custom labware A1 in slot 2?") + + # Step 11 + ctx.move_labware(labware=custom_labware, new_location="6") + + # Step 12 + pipette_right.move_to(location=custom_labware.wells_by_name()["A1"].top()) + + # Step 13 + ctx.pause("Is the pipette tip in the middle of custom labware A1 in slot 6?") + + # Step 14 + ctx.move_labware(labware=logo_destination_plate, new_location="2") + + # Step 15 + pipette_right.move_to(location=logo_destination_plate.wells_by_name()["A1"].top()) + + # Step 16 + ctx.pause("Is the pipette tip in the middle of well A1 in slot 2?") + + ####################### + # prepare_to_aspirate # + ####################### + + # -------------------------- # + # Added in API version: 2.16 # + # -------------------------- # + + pipette_right.prepare_to_aspirate() + pipette_right.move_to(dye_container.wells_by_name()["A1"].bottom(z=2)) + ctx.pause( + "Testing prepare_to_aspirate - watch pipette until next pause.\n The pipette should only move up out of the well after it has aspirated." + ) + pipette_right.aspirate(10, dye_container.wells_by_name()["A1"].bottom(z=2)) + ctx.pause("Did the pipette move up out of the well, only once, after aspirating?") + pipette_right.dispense(10, dye_container.wells_by_name()["A1"].bottom(z=2)) + + ######################################### + # protocol_context.fixed_trash property # + ######################################### + + # ---------------------------- # + # Changed in API version: 2.16 # + # ---------------------------- # + + pipette_right.move_to(ctx.fixed_trash) + ctx.pause("Is the pipette over the trash? Pipette will home after this pause.") + ctx.home() + + ############################################### + # instrument_context.trash_container property # + ############################################### + + # ---------------------------- # + # Changed in API version: 2.16 # + # ---------------------------- # + + pipette_right.move_to(pipette_right.trash_container) + ctx.pause("Is the pipette over the trash?") + + # Distribute dye + pipette_right.distribute( + volume=18, + source=dye_source, + dest=dye_destination_wells, + new_tip="never", + ) + pipette_right.drop_tip() + + # transfer + transfer_destinations = [ + logo_destination_plate.wells_by_name()["A11"], + logo_destination_plate.wells_by_name()["B11"], + logo_destination_plate.wells_by_name()["C11"], + ] + pipette_right.pick_up_tip() + pipette_right.transfer( + volume=60, + source=dye_container.wells_by_name()["A2"], + dest=transfer_destinations, + new_tip="never", + touch_tip=True, + blow_out=True, + blowout_location="destination well", + mix_before=(3, 20), + mix_after=(1, 20), + mix_touch_tip=True, + ) + + # consolidate + pipette_right.consolidate( + volume=20, + source=transfer_destinations, + dest=dye_container.wells_by_name()["A5"], + new_tip="never", + touch_tip=False, + blow_out=True, + blowout_location="destination well", + mix_before=(3, 20), + ) + + # well to well + pipette_right.return_tip() + pipette_right.pick_up_tip() + pipette_right.aspirate(volume=5, location=logo_destination_plate.wells_by_name()["A11"]) + pipette_right.air_gap(volume=10) + ctx.delay(seconds=DELAY_TIME) + pipette_right.dispense(volume=5, location=logo_destination_plate.wells_by_name()["H11"]) + + # move to + pipette_right.move_to(logo_destination_plate.wells_by_name()["E12"].top()) + pipette_right.move_to(logo_destination_plate.wells_by_name()["E11"].bottom()) + pipette_right.blow_out() + # touch tip + # pipette ends in the middle of the well as of 6.3.0 in all touch_tip + pipette_right.touch_tip(location=logo_destination_plate.wells_by_name()["H1"]) + ctx.pause("Is the pipette tip in the middle of the well?") + pipette_right.return_tip() + + # Play with the modules + temperature_module.await_temperature(25) + + hs_module.set_and_wait_for_shake_speed(466) + ctx.delay(seconds=DELAY_TIME) + + hs_module.set_and_wait_for_temperature(HEATER_SHAKER_TEMPERATURE) + + thermocycler_module.open_lid() + thermocycler_module.close_lid() + thermocycler_module.set_lid_temperature(38) # 37 is the minimum + thermocycler_module.set_block_temperature(temperature=28, hold_time_seconds=5) + thermocycler_module.deactivate_block() + thermocycler_module.deactivate_lid() + thermocycler_module.open_lid() + + hs_module.deactivate_shaker() + + # dispense to modules + + # to temperature module + pipette_right.pick_up_tip() + pipette_right.aspirate(volume=15, location=dye_source) + pipette_right.dispense(volume=15, location=temp_plate.well(0)) + pipette_right.drop_tip() + + # to heater shaker + pipette_left.pick_up_tip() + pipette_left.aspirate(volume=50, location=dye_source) + pipette_left.dispense(volume=50, location=hs_plate.well(0)) + hs_module.set_and_wait_for_shake_speed(350) + ctx.delay(DELAY_TIME) + hs_module.deactivate_shaker() + + # to custom labware + # This labware does not EXIST!!!! so... + # Use tip rack lid to catch dye on wet run + pipette_right.pick_up_tip() + pipette_right.aspirate(volume=10, location=dye_source, rate=2.0) + pipette_right.dispense(volume=10, location=custom_labware.well(3), rate=1.5) + pipette_right.drop_tip() + + # to thermocycler + pipette_left.aspirate(volume=75, location=dye_source) + pipette_left.dispense(volume=60, location=tc_plate.wells_by_name()["A6"]) + pipette_left.drop_tip() + + ######################## + # unique top() methods # + ######################## + + # ---------------------------- # + # Changed in API version: 2.18 # + # ---------------------------- # + + assert isinstance(ctx.fixed_trash.top(), protocol_api.TrashBin) + assert isinstance(dye_container.wells_by_name()["A1"].top(), types.Location) + + ############################# + # drop_tip location changes # + ############################# + + # ---------------------------- # + # Changed in API version: 2.18 # + # ---------------------------- # + + ctx.pause("Watch the next 5 tips drop in the trash. They should drop in different locations of the trash each time.") + for _ in range(5): + pipette_right.pick_up_tip() + pipette_right.drop_tip() + + ctx.pause("Watch the next 5 tips drop in the trash. They should drop in the same location of the trash each time.") + for _ in range(5): + pipette_right.pick_up_tip() + pipette_right.drop_tip(location=ctx.fixed_trash) + + ###################### + # labware.set_offset # + ###################### + + # -------------------------- # + # Added in API version: 2.18 # + # -------------------------- # + + SET_OFFSET_AMOUNT = 10.0 + + pipette_right.pick_up_tip() + + ctx.move_labware(labware=logo_destination_plate, new_location=protocol_api.OFF_DECK) + pipette_right.move_to(dye_container.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 3? It should be at the LPC calibrated height.") + + dye_container.set_offset( + x=0.0, + y=0.0, + z=SET_OFFSET_AMOUNT, + ) + + pipette_right.move_to(dye_container.wells_by_name()["A1"].top()) + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 3? It should be 10mm higher than the LPC calibrated height.") + + ctx.move_labware(labware=dye_container, new_location="2") + pipette_right.move_to(dye_container.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 2? It should be at the LPC calibrated height.") + + dye_container.set_offset( + x=0.0, + y=0.0, + z=SET_OFFSET_AMOUNT, + ) + + pipette_right.move_to(dye_container.wells_by_name()["A1"].top()) + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 2? It should be 10mm higher than the LPC calibrated height.") + + ctx.move_labware(labware=dye_container, new_location="3") + pipette_right.move_to(dye_container.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of reservoir A1 in slot 3? It should be 10mm higher than the LPC calibrated height.") + + ctx.move_labware(labware=logo_destination_plate, new_location="2") + pipette_right.move_to(logo_destination_plate.wells_by_name()["A1"].top()) + + ctx.pause("Is the pipette tip in the middle of well A1 in slot 2? It should be at the LPC calibrated height.") + + ctx.pause("!!!!!!!!!!YOU NEED TO REDO LPC!!!!!!!!!!") + + pipette_right.return_tip()