diff --git a/.github/PULL_REQUEST_TEMPLATE.md b/.github/PULL_REQUEST_TEMPLATE.md index 4ffac94d4ce..e4690c51590 100644 --- a/.github/PULL_REQUEST_TEMPLATE.md +++ b/.github/PULL_REQUEST_TEMPLATE.md @@ -1,62 +1,48 @@ # Overview -# Test Plan +## Test Plan and Hands on Testing -# Changelog +## Changelog -# Review requests +## Review requests -# Risk assessment +## Risk assessment diff --git a/.github/workflows/analyses-snapshot-test.yaml b/.github/workflows/analyses-snapshot-test.yaml index 86f6ccab31c..26576899188 100644 --- a/.github/workflows/analyses-snapshot-test.yaml +++ b/.github/workflows/analyses-snapshot-test.yaml @@ -23,8 +23,18 @@ on: - 'api/**' - '!api/tests/**' - '!api/docs/**' + - '!api/release-notes-internal.md' + - '!api/release-notes.md' - 'shared-data/**/*' - '!shared-data/js/**' + - '.github/workflows/analyses-snapshot-test.yaml' + - 'analyses-snapshot-testing/**' + + types: + - opened #default + - synchronize #default + - reopened #default + - labeled concurrency: group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} @@ -40,14 +50,27 @@ jobs: # If we're running because of workflow_dispatch, use the user input to decide # whether to open a PR on failure. Otherwise, there is no user input, so always # open a PR on failure. - OPEN_PR_ON_FAILURE: ${{ (github.event_name == 'workflow_dispatch' && github.events.inputs.OPEN_PR_ON_FAILURE) || (github.event_name != 'workflow_dispatch') }} - + OPEN_PR_ON_FAILURE: ${{ (github.event_name == 'workflow_dispatch' && github.events.inputs.OPEN_PR_ON_FAILURE) || ((github.event_name != 'workflow_dispatch') && (contains(github.event.pull_request.labels.*.name, 'gen-analyses-snapshot-pr'))) }} + PR_TARGET_BRANCH: ${{ github.event.pull_request.base.ref || 'not a pr'}} steps: - name: Checkout Repository uses: actions/checkout@v4 with: ref: ${{ env.SNAPSHOT_REF }} + - name: Are the analyses snapshots in my PR branch in sync with the target branch? + if: github.event_name == 'pull_request' + run: | + git fetch origin ${{ env.PR_TARGET_BRANCH }} + DIFF_OUTPUT=$(git diff HEAD origin/${{ env.PR_TARGET_BRANCH }} -- analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test) + if [ -n "$DIFF_OUTPUT" ]; then + echo "Analyses snapshots do NOT match ${{ env.PR_TARGET_BRANCH }} snapshots." + echo "Is this becasue you have not pulled and merged ${{ env.PR_TARGET_BRANCH }}?" + echo "Or is this because you have already updated your snapshots and are all good 😊?" + else + echo "Analyses snapshots match ${{ env.PR_TARGET_BRANCH }} snapshots." + fi + - name: Docker Build working-directory: analyses-snapshot-testing run: make build-opentrons-analysis @@ -77,30 +100,40 @@ jobs: - name: Handle Test Failure id: handle_failure - if: always() && steps.run_test.outcome == 'failure' + if: always() && steps.run_test.outcome == 'failure' && (env.OPEN_PR_ON_FAILURE == 'true' || github.event_name == 'schedule') working-directory: analyses-snapshot-testing run: make snapshot-test-update - name: Create Snapshot update Request id: create_pull_request - if: always() && steps.handle_failure.outcome == 'success' && env.OPEN_PR_ON_FAILURE + if: always() && steps.handle_failure.outcome == 'success' && env.OPEN_PR_ON_FAILURE == 'true' && github.event_name == 'pull_request' uses: peter-evans/create-pull-request@v6 with: - commit-message: 'fix(analyses-snapshot-testing): snapshot failure capture' - title: 'fix(analyses-snapshot-testing): ${{ env.ANALYSIS_REF }} snapshot failure capture' - body: 'This PR is an automated snapshot update request. Please review the changes and merge if they are acceptable or find your bug and fix it.' + commit-message: 'fix(analyses-snapshot-testing): heal analyses snapshots' + title: 'fix(analyses-snapshot-testing): heal ${{ env.ANALYSIS_REF }} snapshots' + body: 'This PR was requested on the PR https://github.com/${{ github.repository }}/pull/${{ github.event.pull_request.number }}' branch: 'analyses-snapshot-testing/${{ env.ANALYSIS_REF }}-from-${{ env.SNAPSHOT_REF}}' base: ${{ env.SNAPSHOT_REF}} - - name: Comment on PR + - name: Comment on feature PR if: always() && steps.create_pull_request.outcome == 'success' && github.event_name == 'pull_request' uses: actions/github-script@v7 with: script: | - const message = 'A PR has been opened to address analyses snapshot changes. Please review the changes here: https://github.com/${{ github.repository }}/pull/${{ steps.create-pull-request.outputs.pull-request-number }}'; + const message = 'A PR has been opened to address analyses snapshot changes. Please review the changes here: https://github.com/${{ github.repository }}/pull/${{ steps.create_pull_request.outputs.pull-request-number }}'; github.rest.issues.createComment({ owner: context.repo.owner, repo: context.repo.repo, issue_number: context.issue.number, body: message }); + + - name: Create Snapshot update Request on edge overnight failure + if: always() && steps.handle_failure.outcome == 'success' && github.event_name == 'schedule' + uses: peter-evans/create-pull-request@v6 + with: # scheduled run uses the default values for ANALYSIS_REF and SNAPSHOT_REF which are edge + commit-message: 'fix(analyses-snapshot-testing): heal ${{ env.ANALYSIS_REF }} snapshots' + title: 'fix(analyses-snapshot-testing): heal ${{ env.ANALYSIS_REF }} snapshots' + body: 'The ${{ env.ANALYSIS_REF }} overnight analyses snapshot test is failing. This PR was opened to alert us to the failure.' + branch: 'analyses-snapshot-testing/${{ env.ANALYSIS_REF }}-from-${{ env.SNAPSHOT_REF}}' + base: ${{ env.SNAPSHOT_REF}} diff --git a/abr-testing/abr_testing/automation/jira_tool.py b/abr-testing/abr_testing/automation/jira_tool.py index df31c5231e8..4bc8ba4686b 100644 --- a/abr-testing/abr_testing/automation/jira_tool.py +++ b/abr-testing/abr_testing/automation/jira_tool.py @@ -23,25 +23,79 @@ def __init__(self, url: str, api_token: str, email: str) -> None: "Content-Type": "application/json", } - def issues_on_board(self, board_id: str) -> List[str]: + def issues_on_board(self, project_key: str) -> List[List[Any]]: """Print Issues on board.""" + params = {"jql": f"project = {project_key}"} response = requests.get( - f"{self.url}/rest/agile/1.0/board/{board_id}/issue", + f"{self.url}/rest/api/3/search", headers=self.headers, + params=params, auth=self.auth, ) + response.raise_for_status() try: board_data = response.json() all_issues = board_data["issues"] except json.JSONDecodeError as e: print("Error decoding json: ", e) + # convert issue id's into array and have one key as + # the issue key and one be summary, return entire array issue_ids = [] for i in all_issues: issue_id = i.get("id") - issue_ids.append(issue_id) + issue_summary = i["fields"].get("summary") + issue_ids.append([issue_id, issue_summary]) return issue_ids + def match_issues(self, issue_ids: List[List[str]], ticket_summary: str) -> List: + """Matches related ticket ID's.""" + to_link = [] + error = ticket_summary.split("_")[3] + robot = ticket_summary.split("_")[0] + # for every issue see if both match, if yes then grab issue ID and add it to a list + for issue in issue_ids: + summary = issue[1] + try: + issue_error = summary.split("_")[3] + issue_robot = summary.split("_")[0] + except IndexError: + continue + issue_id = issue[0] + if robot == issue_robot and error == issue_error: + to_link.append(issue_id) + return to_link + + def link_issues(self, to_link: list, ticket_key: str) -> None: + """Links relevant issues in Jira.""" + for issue in to_link: + link_data = json.dumps( + { + "inwardIssue": {"key": ticket_key}, + "outwardIssue": {"id": issue}, + "type": {"name": "Relates"}, + } + ) + try: + response = requests.post( + f"{self.url}/rest/api/3/issueLink", + headers=self.headers, + auth=self.auth, + data=link_data, + ) + response.raise_for_status() + except requests.exceptions.HTTPError: + print( + f"HTTP error occurred. Ticket ID {issue} was not linked. \ + Check user permissions and authentication credentials" + ) + except requests.exceptions.ConnectionError: + print(f"Connection error occurred. Ticket ID {issue} was not linked.") + except json.JSONDecodeError: + print( + f"JSON decoding error occurred. Ticket ID {issue} was not linked." + ) + def open_issue(self, issue_key: str) -> str: """Open issue on web browser.""" url = f"{self.url}/browse/{issue_key}" diff --git a/abr-testing/abr_testing/data_collection/abr_robot_error.py b/abr-testing/abr_testing/data_collection/abr_robot_error.py index 98af232304d..a35a93f54ae 100644 --- a/abr-testing/abr_testing/data_collection/abr_robot_error.py +++ b/abr-testing/abr_testing/data_collection/abr_robot_error.py @@ -481,7 +481,6 @@ def get_run_error_info_from_robot( reporter_id = args.reporter_id[0] file_paths = read_robot_logs.get_logs(storage_directory, ip) ticket = jira_tool.JiraTicket(url, api_token, email) - ticket.issues_on_board(board_id) users_file_path = ticket.get_jira_users(storage_directory) assignee_id = get_user_id(users_file_path, assignee) run_log_file_path = "" @@ -519,6 +518,9 @@ def get_run_error_info_from_robot( print(robot) parent_key = project_key + "-" + robot.split("ABR")[1] + # Grab all previous issues + all_issues = ticket.issues_on_board(project_key) + # TODO: read board to see if ticket for run id already exists. # CREATE TICKET issue_key, raw_issue_url = ticket.create_ticket( @@ -533,6 +535,11 @@ def get_run_error_info_from_robot( affects_version, parent_key, ) + + # Link Tickets + to_link = ticket.match_issues(all_issues, summary) + ticket.link_issues(to_link, issue_key) + # OPEN TICKET issue_url = ticket.open_issue(issue_key) # MOVE FILES TO ERROR FOLDER. diff --git a/analyses-snapshot-testing/automation/data/protocols.py b/analyses-snapshot-testing/automation/data/protocols.py index e5dc06306b1..580c9183d9d 100644 --- a/analyses-snapshot-testing/automation/data/protocols.py +++ b/analyses-snapshot-testing/automation/data/protocols.py @@ -635,6 +635,12 @@ class Protocols: robot="Flex", ) + Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke: Protocol = Protocol( + file_stem="Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke", + file_extension="py", + robot="Flex", + ) + OT2_X_v2_18_None_None_duplicateRTPVariableName: Protocol = Protocol( file_stem="OT2_X_v2_18_None_None_duplicateRTPVariableName", file_extension="py", @@ -665,6 +671,12 @@ class Protocols: robot="OT2", ) + OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3: Protocol = Protocol( + file_stem="OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3", + file_extension="py", + robot="OT2", + ) + ########################################################################################################## # Begin Protocol Library Protocols ####################################################################### ########################################################################################################## diff --git a/analyses-snapshot-testing/files/protocols/Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py b/analyses-snapshot-testing/files/protocols/Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py new file mode 100644 index 00000000000..0702bca32cb --- /dev/null +++ b/analyses-snapshot-testing/files/protocols/Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py @@ -0,0 +1,654 @@ +############# +# CHANGELOG # +############# + +# ---- +# 2.19 +# ---- + +# NOTHING NEW +# This protocol is exactly the same as 2.16 Smoke Test V3 +# The only difference is the API version in the metadata +# The only change was changing pipette overlap values, which is not anything that can be validated by the smoke test +# Just make sure the protocol runs normally + +# ---- +# 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 +import dataclasses +import typing + +metadata = { + "protocolName": "Flex Smoke Test - v2.19", + "author": "Derek Maggio ", +} + +requirements = { + "robotType": "OT-3", + "apiLevel": "2.19", +} + +DeckSlots = typing.Literal[ + "A1", + "A2", + "A3", + "A4", + "B1", + "B2", + "B3", + "B4", + "C1", + "C2", + "C3", + "C4", + "D1", + "D2", + "D3", + "D4", +] +ValidModuleLocations = typing.List[ + typing.Union[ + protocol_api.ThermocyclerContext, + protocol_api.MagneticBlockContext, + protocol_api.Labware, # H/S Adapter or Temp Module Adapter + ] +] + +TestConfigurationChoices = typing.Literal["qa", "dev"] + + +@dataclasses.dataclass +class MoveSequence: + """A sequence of moves for a given labware.""" + + move_tos: typing.List[DeckSlots | ValidModuleLocations] + starting_location: DeckSlots | ValidModuleLocations + reset_to_start_after_each_move: bool + + def do_moves(self, ctx: protocol_api.ProtocolContext, labware: protocol_api.Labware): + + if labware.parent is not self.starting_location: + ctx.move_labware(labware, self.starting_location, use_gripper=True) + + for location in self.move_tos: + ctx.move_labware(labware, location, use_gripper=True) + + if self.reset_to_start_after_each_move: + ctx.move_labware(labware, self.starting_location, use_gripper=True) + + +@dataclasses.dataclass +class AllMoveSequences: + """All move sequences for the gripper.""" + + moves: typing.List[MoveSequence] + + @classmethod + def abbreviated_moves(cls, all_modules: typing.List[ValidModuleLocations]) -> "AllMoveSequences": + module_to_move_to = all_modules[0] + return cls( + [MoveSequence(move_tos=["B2", module_to_move_to, "D4", "C3"], starting_location="C2", reset_to_start_after_each_move=False)], + ) + + @classmethod + def all_moves(cls, all_modules: typing.List[ValidModuleLocations]) -> "AllMoveSequences": + return cls( + [ + # Covers + # Deck -> Deck + # Deck -> Staging Area Slot 3 + # Deck -> Staging Area Slot 4 + # Deck -> All modules + # Staging Area Slot 3 -> Deck + # Staging Area Slot 4 -> Deck + # All modules -> Deck + MoveSequence(move_tos=["B2", "C3", "D4"] + all_modules, starting_location="C2", reset_to_start_after_each_move=True), + # Covers + # Staging Area Slot 3 -> Staging Area Slot 4 + # Staging Area Slot 3 -> All modules + # Staging Area Slot 4 -> Staging Area Slot 3 + # All modules -> Staging Area Slot 3 + # Note: cannot cover staging area slot 3 -> staging area slot 3. Not enough room on deck + MoveSequence(move_tos=["D4"] + all_modules, starting_location="C3", reset_to_start_after_each_move=True), + # Covers + # Staging Area Slot 4 -> Staging Area Slot 4 + # Staging Area Slot 4 -> All modules + # All modules -> Staging Area Slot 4 + MoveSequence(move_tos=["C4"] + all_modules, starting_location="D4", reset_to_start_after_each_move=True), + ] + + + # Covers + # module -> module + [ + MoveSequence( + move_tos=[module_location for module_location in all_modules if module_location != starting_location], + starting_location=starting_location, + reset_to_start_after_each_move=True, + ) + for starting_location in all_modules + ], + ) + + def do_moves( + self, ctx: protocol_api.ProtocolContext, labware: protocol_api.Labware, original_labware_location: DeckSlots | ValidModuleLocations + ): + for move_sequence in self.moves: + move_sequence.do_moves(ctx, labware) + + if labware.parent is not original_labware_location: + ctx.move_labware(labware, original_labware_location, use_gripper=True) + + +@dataclasses.dataclass +class ModuleTemperatureConfiguration: + thermocycler_block: float + thermocycler_lid: float + heater_shaker: float + temperature_module: float + + @classmethod + def qa_configuration(cls) -> "ModuleTemperatureConfiguration": + return cls( + thermocycler_block=60.0, + thermocycler_lid=80.0, + heater_shaker=50.0, + temperature_module=50.0, + ) + + @classmethod + def dev_configuration(cls) -> "ModuleTemperatureConfiguration": + return cls( + thermocycler_block=50.0, + thermocycler_lid=50.0, + heater_shaker=45.0, + temperature_module=40.0, + ) + + +@dataclasses.dataclass +class TestConfiguration: + # Don't default these, they are set by runtime parameters + configuration_name: TestConfigurationChoices + reservoir_name: str + well_plate_name: str + prefer_gripper_disposal: bool + + test_set_offset: bool + run_abbreviated_pipetting_test: bool + + # Make this greater than or equal to 2, and less than or equal to 12 + partial_tip_pickup_column_count: int + + module_temps: ModuleTemperatureConfiguration + gripper_moves: AllMoveSequences + + @property + def is_qa(self) -> bool: + return self.configuration_name == "qa" + + @property + def is_dev(self) -> bool: + return self.configuration_name == "dev" + + @classmethod + def _get_qa_config( + cls, prefer_gripper_disposal: bool, reservoir_name: str, well_plate_name: str, all_modules: typing.List[ValidModuleLocations] + ) -> "TestConfiguration": + return cls( + configuration_name="qa", + reservoir_name=reservoir_name, + well_plate_name=well_plate_name, + test_set_offset=True, + run_abbreviated_pipetting_test=False, + partial_tip_pickup_column_count=12, + prefer_gripper_disposal=prefer_gripper_disposal, + module_temps=ModuleTemperatureConfiguration.qa_configuration(), + gripper_moves=AllMoveSequences.all_moves(all_modules), + ) + + @classmethod + def _get_dev_config( + cls, prefer_gripper_disposal: bool, reservoir_name: str, well_plate_name: str, all_modules: typing.List[ValidModuleLocations] + ) -> "TestConfiguration": + return cls( + configuration_name="dev", + reservoir_name=reservoir_name, + well_plate_name=well_plate_name, + test_set_offset=False, + run_abbreviated_pipetting_test=True, + partial_tip_pickup_column_count=2, + prefer_gripper_disposal=prefer_gripper_disposal, + module_temps=ModuleTemperatureConfiguration.dev_configuration(), + gripper_moves=AllMoveSequences.abbreviated_moves(all_modules), + ) + + @classmethod + def get_configuration( + cls, + parameters: protocol_api.Parameters, + where_to_put_labware_on_modules: typing.List[ + protocol_api.ThermocyclerContext | protocol_api.MagneticBlockContext | protocol_api.Labware + ], + ) -> "TestConfiguration": + test_configuration = parameters.test_configuration + prefer_gripper_disposal = parameters.prefer_gripper_disposal + reservoir_name = parameters.reservoir_name + well_plate_name = parameters.well_plate_name + + if test_configuration == "qa": + return cls._get_qa_config( + prefer_gripper_disposal=prefer_gripper_disposal, + reservoir_name=reservoir_name, + well_plate_name=well_plate_name, + all_modules=where_to_put_labware_on_modules, + ) + elif test_configuration == "dev": + return cls._get_dev_config( + prefer_gripper_disposal=prefer_gripper_disposal, + reservoir_name=reservoir_name, + well_plate_name=well_plate_name, + all_modules=where_to_put_labware_on_modules, + ) + else: + raise ValueError(f"Invalid test configuration: {test_configuration}") + + +################# +### 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): + + test_configuration_choices = [ + {"display_name": "QA Smoke Test", "value": "qa"}, + {"display_name": "Developer Validation", "value": "dev"}, + ] + + 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="test_configuration", + display_name="Test Configuration", + description="Configuration of QA test to perform", + default="qa", + choices=test_configuration_choices, + ) + + 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="prefer_gripper_disposal", + display_name="I LOVE TO REFILL TIP RACKS", + description="Prefer to use the gripper to dispose of labware, instead of manual moves off deck", + default=False, + ) + + +def run(ctx: protocol_api.ProtocolContext) -> None: + ################ + ### 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] + + ########################## + ### TEST CONFIGURATION ### + ########################## + + test_config: TestConfiguration = TestConfiguration.get_configuration( + ctx.params, [thermocycler, magnetic_block, temperature_module_adapter, heater_shaker_adapter] + ) + + ############### + ### LABWARE ### + ############### + + source_reservoir = ctx.load_labware(test_config.reservoir_name, "D2") + dest_pcr_plate = ctx.load_labware(test_config.well_plate_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 dispose_with_preferred_method(labware: protocol_api.Labware): + """ + 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. + """ + if test_config.prefer_gripper_disposal: + ctx.move_labware(labware, waste_chute, use_gripper=True) + else: + ctx.move_labware(labware, protocol_api.OFF_DECK, use_gripper=False) + + def test_manual_moves(): + # In C4 currently + ctx.move_labware(source_reservoir, "D4", use_gripper=False) + + 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, test_config.partial_tip_pickup_column_count + 1): + + 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 test_config.is_qa: + if i == 1: + ctx.pause( + "Watch this next tip drop in the waste chute. We are going to compare it against the next drop in the waste chute." + ) + + if i == 2: + ctx.pause( + "Watch this next tip drop in the waste chute. It should drop in a different location than the previous drop." + ) + + if i == 1: + pipette_96_channel.drop_tip(waste_chute) + elif i == 2: + pipette_96_channel.drop_tip() + else: + pipette_96_channel.drop_tip(trash_bin) + + dispose_with_preferred_method(tip_rack_2) + + 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(10, source_reservoir["A1"]) + pipette_96_channel.touch_tip() + + pipette_96_channel.air_gap(height=30) + + pipette_96_channel.dispense(10, dest_pcr_plate["A1"]) + + pipette_96_channel.blow_out(waste_chute) + + pipette_96_channel.return_tip() + dispose_with_preferred_method(tip_rack_1) + ctx.move_labware(tip_rack_3, tip_rack_adapter, use_gripper=True) + + if not test_config.run_abbreviated_pipetting_test: + 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=(5, 15), + ) + 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(test_config.module_temps.thermocycler_block, hold_time_seconds=5.0) + thermocycler.set_lid_temperature(test_config.module_temps.thermocycler_lid) + thermocycler.deactivate() + + def test_heater_shaker(): + heater_shaker.open_labware_latch() + heater_shaker.close_labware_latch() + + heater_shaker.set_target_temperature(test_config.module_temps.heater_shaker) + heater_shaker.set_and_wait_for_shake_speed(1000) + heater_shaker.wait_for_temperature() + + heater_shaker.deactivate_heater() + heater_shaker.deactivate_shaker() + + def test_temperature_module(): + temperature_module.set_temperature(test_config.module_temps.temperature_module) + temperature_module.deactivate() + + def test_magnetic_block(): + pass + + test_thermocycler() + test_heater_shaker() + test_temperature_module() + test_magnetic_block() + + def test_labware_waste_chute_disposal_with_gripper(): + ctx.move_labware(source_reservoir, waste_chute, use_gripper=True) + ctx.move_labware(dest_pcr_plate, waste_chute, use_gripper=True) + + 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.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_config.gripper_moves.do_moves(ctx=ctx, labware=dest_pcr_plate, original_labware_location="C2") + test_module_usage() + test_manual_moves() + if test_config.test_set_offset: + test_labware_set_offset() + test_unique_top_methods() + test_labware_waste_chute_disposal_with_gripper() + + ################################################################################################### + ### 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/analyses-snapshot-testing/files/protocols/OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py b/analyses-snapshot-testing/files/protocols/OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py new file mode 100644 index 00000000000..c571d2a593a --- /dev/null +++ b/analyses-snapshot-testing/files/protocols/OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py @@ -0,0 +1,605 @@ +"""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.19 +# ---- + +# NOTHING NEW +# This protocol is exactly the same as 2.16 Smoke Test V3 +# The only difference is the API version in the metadata +# The only change was changing pipette overlap values, which is not anything that can be validated by the smoke test +# Just make sure the protocol runs normally + +# ---- +# 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.19 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.19"} + +############################## +# 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() diff --git a/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[057de2035d][OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3].json b/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[057de2035d][OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3].json new file mode 100644 index 00000000000..a73a19e4c88 --- /dev/null +++ b/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[057de2035d][OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3].json @@ -0,0 +1,9684 @@ +{ + "commands": [ + { + "commandType": "home", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "50c7ae73a4e3f7129874f39dfb514803", + "notes": [], + "params": {}, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "setRailLights", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c0f556802f0eafbdbce20c171f217b13", + "notes": [], + "params": { + "on": true + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "comment", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b0ba40f2987795790258bb3b52aef419", + "notes": [], + "params": { + "message": "Let there be light! True 🌠🌠🌠" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "comment", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f69df71d649b9cbb0f560491504b17bf", + "notes": [], + "params": { + "message": "Is the door is closed? True 🚪🚪🚪" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "comment", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "252352128dcb2c6ac11f78895872a7bb", + "notes": [], + "params": { + "message": "Is this a simulation? True 🔮🔮🔮" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "comment", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "22812ba0d8d8f561d6f94d12dbd191a6", + "notes": [], + "params": { + "message": "Running against API Version: 2.19" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2fef6b1c39bcc1af9b9d88c5ae54d919", + "notes": [], + "params": { + "displayName": "300ul tips", + "loadName": "opentrons_96_tiprack_300ul", + "location": { + "slotName": "5" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "Opentrons", + "brandId": [], + "links": [ + "https://shop.opentrons.com/collections/opentrons-tips/products/opentrons-300ul-tips" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 64.49 + }, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "tipRack", + "displayName": "Opentrons OT-2 96 Tip Rack 300 µL", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": true, + "loadName": "opentrons_96_tiprack_300ul", + "tipLength": 59.3, + "tipOverlap": 7.47 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 74.24, + "z": 5.39 + }, + "A10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 74.24, + "z": 5.39 + }, + "A11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 74.24, + "z": 5.39 + }, + "A12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 74.24, + "z": 5.39 + }, + "A2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 74.24, + "z": 5.39 + }, + "A3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 74.24, + "z": 5.39 + }, + "A4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 74.24, + "z": 5.39 + }, + "A5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 74.24, + "z": 5.39 + }, + "A6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 74.24, + "z": 5.39 + }, + "A7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 74.24, + "z": 5.39 + }, + "A8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 74.24, + "z": 5.39 + }, + "A9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 74.24, + "z": 5.39 + }, + "B1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 65.24, + "z": 5.39 + }, + "B10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 65.24, + "z": 5.39 + }, + "B11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 65.24, + "z": 5.39 + }, + "B12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 65.24, + "z": 5.39 + }, + "B2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 65.24, + "z": 5.39 + }, + "B3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 65.24, + "z": 5.39 + }, + "B4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 65.24, + "z": 5.39 + }, + "B5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 65.24, + "z": 5.39 + }, + "B6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 65.24, + "z": 5.39 + }, + "B7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 65.24, + "z": 5.39 + }, + "B8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 65.24, + "z": 5.39 + }, + "B9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 65.24, + "z": 5.39 + }, + "C1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 56.24, + "z": 5.39 + }, + "C10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 56.24, + "z": 5.39 + }, + "C11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 56.24, + "z": 5.39 + }, + "C12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 56.24, + "z": 5.39 + }, + "C2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 56.24, + "z": 5.39 + }, + "C3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 56.24, + "z": 5.39 + }, + "C4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 56.24, + "z": 5.39 + }, + "C5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 56.24, + "z": 5.39 + }, + "C6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 56.24, + "z": 5.39 + }, + "C7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 56.24, + "z": 5.39 + }, + "C8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 56.24, + "z": 5.39 + }, + "C9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 56.24, + "z": 5.39 + }, + "D1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 47.24, + "z": 5.39 + }, + "D10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 47.24, + "z": 5.39 + }, + "D11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 47.24, + "z": 5.39 + }, + "D12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 47.24, + "z": 5.39 + }, + "D2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 47.24, + "z": 5.39 + }, + "D3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 47.24, + "z": 5.39 + }, + "D4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 47.24, + "z": 5.39 + }, + "D5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 47.24, + "z": 5.39 + }, + "D6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 47.24, + "z": 5.39 + }, + "D7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 47.24, + "z": 5.39 + }, + "D8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 47.24, + "z": 5.39 + }, + "D9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 47.24, + "z": 5.39 + }, + "E1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 38.24, + "z": 5.39 + }, + "E10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 38.24, + "z": 5.39 + }, + "E11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 38.24, + "z": 5.39 + }, + "E12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 38.24, + "z": 5.39 + }, + "E2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 38.24, + "z": 5.39 + }, + "E3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 38.24, + "z": 5.39 + }, + "E4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 38.24, + "z": 5.39 + }, + "E5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 38.24, + "z": 5.39 + }, + "E6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 38.24, + "z": 5.39 + }, + "E7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 38.24, + "z": 5.39 + }, + "E8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 38.24, + "z": 5.39 + }, + "E9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 38.24, + "z": 5.39 + }, + "F1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 29.24, + "z": 5.39 + }, + "F10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 29.24, + "z": 5.39 + }, + "F11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 29.24, + "z": 5.39 + }, + "F12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 29.24, + "z": 5.39 + }, + "F2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 29.24, + "z": 5.39 + }, + "F3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 29.24, + "z": 5.39 + }, + "F4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 29.24, + "z": 5.39 + }, + "F5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 29.24, + "z": 5.39 + }, + "F6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 29.24, + "z": 5.39 + }, + "F7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 29.24, + "z": 5.39 + }, + "F8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 29.24, + "z": 5.39 + }, + "F9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 29.24, + "z": 5.39 + }, + "G1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 20.24, + "z": 5.39 + }, + "G10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 20.24, + "z": 5.39 + }, + "G11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 20.24, + "z": 5.39 + }, + "G12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 20.24, + "z": 5.39 + }, + "G2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 20.24, + "z": 5.39 + }, + "G3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 20.24, + "z": 5.39 + }, + "G4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 20.24, + "z": 5.39 + }, + "G5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 20.24, + "z": 5.39 + }, + "G6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 20.24, + "z": 5.39 + }, + "G7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 20.24, + "z": 5.39 + }, + "G8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 20.24, + "z": 5.39 + }, + "G9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 20.24, + "z": 5.39 + }, + "H1": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 14.38, + "y": 11.24, + "z": 5.39 + }, + "H10": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 95.38, + "y": 11.24, + "z": 5.39 + }, + "H11": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 104.38, + "y": 11.24, + "z": 5.39 + }, + "H12": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 113.38, + "y": 11.24, + "z": 5.39 + }, + "H2": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 23.38, + "y": 11.24, + "z": 5.39 + }, + "H3": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 32.38, + "y": 11.24, + "z": 5.39 + }, + "H4": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 41.38, + "y": 11.24, + "z": 5.39 + }, + "H5": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 50.38, + "y": 11.24, + "z": 5.39 + }, + "H6": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 59.38, + "y": 11.24, + "z": 5.39 + }, + "H7": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 68.38, + "y": 11.24, + "z": 5.39 + }, + "H8": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 77.38, + "y": 11.24, + "z": 5.39 + }, + "H9": { + "depth": 59.3, + "diameter": 5.23, + "shape": "circular", + "totalLiquidVolume": 300, + "x": 86.38, + "y": 11.24, + "z": 5.39 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "641d89a1769e495364a6511b4123aaee", + "notes": [], + "params": { + "displayName": "20ul tips", + "loadName": "opentrons_96_tiprack_20ul", + "location": { + "slotName": "4" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "Opentrons", + "brandId": [], + "links": [ + "https://shop.opentrons.com/collections/opentrons-tips/products/opentrons-10ul-tips" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 64.69 + }, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "tipRack", + "displayName": "Opentrons OT-2 96 Tip Rack 20 µL", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": true, + "loadName": "opentrons_96_tiprack_20ul", + "tipLength": 39.2, + "tipOverlap": 8.25 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 74.24, + "z": 25.49 + }, + "A10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 74.24, + "z": 25.49 + }, + "A11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 74.24, + "z": 25.49 + }, + "A12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 74.24, + "z": 25.49 + }, + "A2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 74.24, + "z": 25.49 + }, + "A3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 74.24, + "z": 25.49 + }, + "A4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 74.24, + "z": 25.49 + }, + "A5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 74.24, + "z": 25.49 + }, + "A6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 74.24, + "z": 25.49 + }, + "A7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 74.24, + "z": 25.49 + }, + "A8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 74.24, + "z": 25.49 + }, + "A9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 74.24, + "z": 25.49 + }, + "B1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 65.24, + "z": 25.49 + }, + "B10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 65.24, + "z": 25.49 + }, + "B11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 65.24, + "z": 25.49 + }, + "B12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 65.24, + "z": 25.49 + }, + "B2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 65.24, + "z": 25.49 + }, + "B3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 65.24, + "z": 25.49 + }, + "B4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 65.24, + "z": 25.49 + }, + "B5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 65.24, + "z": 25.49 + }, + "B6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 65.24, + "z": 25.49 + }, + "B7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 65.24, + "z": 25.49 + }, + "B8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 65.24, + "z": 25.49 + }, + "B9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 65.24, + "z": 25.49 + }, + "C1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 56.24, + "z": 25.49 + }, + "C10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 56.24, + "z": 25.49 + }, + "C11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 56.24, + "z": 25.49 + }, + "C12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 56.24, + "z": 25.49 + }, + "C2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 56.24, + "z": 25.49 + }, + "C3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 56.24, + "z": 25.49 + }, + "C4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 56.24, + "z": 25.49 + }, + "C5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 56.24, + "z": 25.49 + }, + "C6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 56.24, + "z": 25.49 + }, + "C7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 56.24, + "z": 25.49 + }, + "C8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 56.24, + "z": 25.49 + }, + "C9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 56.24, + "z": 25.49 + }, + "D1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 47.24, + "z": 25.49 + }, + "D10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 47.24, + "z": 25.49 + }, + "D11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 47.24, + "z": 25.49 + }, + "D12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 47.24, + "z": 25.49 + }, + "D2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 47.24, + "z": 25.49 + }, + "D3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 47.24, + "z": 25.49 + }, + "D4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 47.24, + "z": 25.49 + }, + "D5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 47.24, + "z": 25.49 + }, + "D6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 47.24, + "z": 25.49 + }, + "D7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 47.24, + "z": 25.49 + }, + "D8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 47.24, + "z": 25.49 + }, + "D9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 47.24, + "z": 25.49 + }, + "E1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 38.24, + "z": 25.49 + }, + "E10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 38.24, + "z": 25.49 + }, + "E11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 38.24, + "z": 25.49 + }, + "E12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 38.24, + "z": 25.49 + }, + "E2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 38.24, + "z": 25.49 + }, + "E3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 38.24, + "z": 25.49 + }, + "E4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 38.24, + "z": 25.49 + }, + "E5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 38.24, + "z": 25.49 + }, + "E6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 38.24, + "z": 25.49 + }, + "E7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 38.24, + "z": 25.49 + }, + "E8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 38.24, + "z": 25.49 + }, + "E9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 38.24, + "z": 25.49 + }, + "F1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 29.24, + "z": 25.49 + }, + "F10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 29.24, + "z": 25.49 + }, + "F11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 29.24, + "z": 25.49 + }, + "F12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 29.24, + "z": 25.49 + }, + "F2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 29.24, + "z": 25.49 + }, + "F3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 29.24, + "z": 25.49 + }, + "F4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 29.24, + "z": 25.49 + }, + "F5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 29.24, + "z": 25.49 + }, + "F6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 29.24, + "z": 25.49 + }, + "F7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 29.24, + "z": 25.49 + }, + "F8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 29.24, + "z": 25.49 + }, + "F9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 29.24, + "z": 25.49 + }, + "G1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 20.24, + "z": 25.49 + }, + "G10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 20.24, + "z": 25.49 + }, + "G11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 20.24, + "z": 25.49 + }, + "G12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 20.24, + "z": 25.49 + }, + "G2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 20.24, + "z": 25.49 + }, + "G3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 20.24, + "z": 25.49 + }, + "G4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 20.24, + "z": 25.49 + }, + "G5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 20.24, + "z": 25.49 + }, + "G6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 20.24, + "z": 25.49 + }, + "G7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 20.24, + "z": 25.49 + }, + "G8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 20.24, + "z": 25.49 + }, + "G9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 20.24, + "z": 25.49 + }, + "H1": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 14.38, + "y": 11.24, + "z": 25.49 + }, + "H10": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 95.38, + "y": 11.24, + "z": 25.49 + }, + "H11": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 104.38, + "y": 11.24, + "z": 25.49 + }, + "H12": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 113.38, + "y": 11.24, + "z": 25.49 + }, + "H2": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 23.38, + "y": 11.24, + "z": 25.49 + }, + "H3": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 32.38, + "y": 11.24, + "z": 25.49 + }, + "H4": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 41.38, + "y": 11.24, + "z": 25.49 + }, + "H5": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 50.38, + "y": 11.24, + "z": 25.49 + }, + "H6": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 59.38, + "y": 11.24, + "z": 25.49 + }, + "H7": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 68.38, + "y": 11.24, + "z": 25.49 + }, + "H8": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 77.38, + "y": 11.24, + "z": 25.49 + }, + "H9": { + "depth": 39.2, + "diameter": 3.27, + "shape": "circular", + "totalLiquidVolume": 20, + "x": 86.38, + "y": 11.24, + "z": 25.49 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadPipette", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2925ebb53a28d20bb95d53e4773e40c6", + "notes": [], + "params": { + "liquidPresenceDetection": false, + "mount": "left", + "pipetteName": "p300_multi_gen2", + "tipOverlapNotAfterVersion": "v1" + }, + "result": { + "pipetteId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadPipette", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3b864d00b2bf4e88edb0dc034c1afeaf", + "notes": [], + "params": { + "liquidPresenceDetection": false, + "mount": "right", + "pipetteName": "p20_single_gen2", + "tipOverlapNotAfterVersion": "v1" + }, + "result": { + "pipetteId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "be987e9fdcf2dbe0234a88c9fa47f418", + "notes": [], + "params": { + "location": { + "slotName": "1" + }, + "model": "heaterShakerModuleV1" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 12.0, + "y": 8.75, + "z": 68.275 + }, + "compatibleWith": [], + "dimensions": { + "bareOverallHeight": 82.0, + "overLabwareHeight": 0.0 + }, + "displayName": "Heater-Shaker Module GEN1", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + } + } + }, + "labwareOffset": { + "x": -0.125, + "y": 1.125, + "z": 68.275 + }, + "model": "heaterShakerModuleV1", + "moduleType": "heaterShakerModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot2_short_trash": { + "3": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot2_standard": { + "3": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot3_standard": { + "A1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "A3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "B1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "B3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "C1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "C3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "D1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "D3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + } + } + } + }, + "model": "heaterShakerModuleV1", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "53acfbfde5c1f9c0e34b2d3ac8d26926", + "notes": [], + "params": { + "location": { + "slotName": "9" + }, + "model": "temperatureModuleV2" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 11.7, + "y": 8.75, + "z": 80.09 + }, + "compatibleWith": [ + "temperatureModuleV1" + ], + "dimensions": { + "bareOverallHeight": 84.0, + "overLabwareHeight": 0.0 + }, + "displayName": "Temperature Module GEN2", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + } + } + }, + "labwareOffset": { + "x": -1.45, + "y": -0.15, + "z": 80.09 + }, + "model": "temperatureModuleV2", + "moduleType": "temperatureModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot2_short_trash": { + "3": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot2_standard": { + "3": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot3_standard": { + "A1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "A3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "B1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "B3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "C1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "C3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "D1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "D3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + } + } + } + }, + "model": "temperatureModuleV2", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "dc701dc10e773175dfe266b65d1ff3e6", + "notes": [], + "params": { + "location": { + "slotName": "7" + }, + "model": "thermocyclerModuleV2" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 14.4, + "y": 64.93, + "z": 97.8 + }, + "compatibleWith": [], + "dimensions": { + "bareOverallHeight": 108.96, + "lidHeight": 61.7, + "overLabwareHeight": 0.0 + }, + "displayName": "Thermocycler Module GEN2", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 5.6 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 4.6 + } + } + }, + "labwareOffset": { + "x": 0.0, + "y": 68.8, + "z": 108.96 + }, + "model": "thermocyclerModuleV2", + "moduleType": "thermocyclerModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot3_standard": { + "B1": { + "cornerOffsetFromSlot": [ + [ + -98, + 0, + 0, + 1 + ], + [ + -20.005, + 0, + 0, + 1 + ], + [ + -0.84, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ], + "labwareOffset": [ + [ + -98, + 0, + 0, + 1 + ], + [ + -20.005, + 0, + 0, + 1 + ], + [ + -0.84, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + } + } + }, + "model": "thermocyclerModuleV2", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "760eaed99cc2a56688adb15cc79ca09b", + "notes": [], + "params": { + "loadName": "opentrons_96_well_aluminum_block", + "location": { + "moduleId": "UUID" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [ + "adapter" + ], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 18.16 + }, + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0, + "y": 0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "aluminumBlock", + "displayName": "Opentrons 96 Well Aluminum Block", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "opentrons_96_well_aluminum_block", + "quirks": [] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 74.24, + "z": 3.38 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 74.24, + "z": 3.38 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 74.24, + "z": 3.38 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 74.24, + "z": 3.38 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 74.24, + "z": 3.38 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 74.24, + "z": 3.38 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 74.24, + "z": 3.38 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 74.24, + "z": 3.38 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 74.24, + "z": 3.38 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 74.24, + "z": 3.38 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 74.24, + "z": 3.38 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 74.24, + "z": 3.38 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 65.24, + "z": 3.38 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 65.24, + "z": 3.38 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 65.24, + "z": 3.38 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 65.24, + "z": 3.38 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 65.24, + "z": 3.38 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 65.24, + "z": 3.38 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 65.24, + "z": 3.38 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 65.24, + "z": 3.38 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 65.24, + "z": 3.38 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 65.24, + "z": 3.38 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 65.24, + "z": 3.38 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 65.24, + "z": 3.38 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 56.24, + "z": 3.38 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 56.24, + "z": 3.38 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 56.24, + "z": 3.38 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 56.24, + "z": 3.38 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 56.24, + "z": 3.38 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 56.24, + "z": 3.38 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 56.24, + "z": 3.38 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 56.24, + "z": 3.38 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 56.24, + "z": 3.38 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 56.24, + "z": 3.38 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 56.24, + "z": 3.38 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 56.24, + "z": 3.38 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 47.24, + "z": 3.38 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 47.24, + "z": 3.38 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 47.24, + "z": 3.38 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 47.24, + "z": 3.38 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 47.24, + "z": 3.38 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 47.24, + "z": 3.38 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 47.24, + "z": 3.38 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 47.24, + "z": 3.38 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 47.24, + "z": 3.38 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 47.24, + "z": 3.38 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 47.24, + "z": 3.38 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 47.24, + "z": 3.38 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 38.24, + "z": 3.38 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 38.24, + "z": 3.38 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 38.24, + "z": 3.38 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 38.24, + "z": 3.38 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 38.24, + "z": 3.38 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 38.24, + "z": 3.38 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 38.24, + "z": 3.38 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 38.24, + "z": 3.38 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 38.24, + "z": 3.38 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 38.24, + "z": 3.38 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 38.24, + "z": 3.38 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 38.24, + "z": 3.38 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 29.24, + "z": 3.38 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 29.24, + "z": 3.38 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 29.24, + "z": 3.38 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 29.24, + "z": 3.38 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 29.24, + "z": 3.38 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 29.24, + "z": 3.38 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 29.24, + "z": 3.38 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 29.24, + "z": 3.38 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 29.24, + "z": 3.38 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 29.24, + "z": 3.38 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 29.24, + "z": 3.38 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 29.24, + "z": 3.38 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 20.24, + "z": 3.38 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 20.24, + "z": 3.38 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 20.24, + "z": 3.38 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 20.24, + "z": 3.38 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 20.24, + "z": 3.38 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 20.24, + "z": 3.38 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 20.24, + "z": 3.38 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 20.24, + "z": 3.38 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 20.24, + "z": 3.38 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 20.24, + "z": 3.38 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 20.24, + "z": 3.38 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 20.24, + "z": 3.38 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 11.24, + "z": 3.38 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 11.24, + "z": 3.38 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 11.24, + "z": 3.38 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 11.24, + "z": 3.38 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 11.24, + "z": 3.38 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 11.24, + "z": 3.38 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 11.24, + "z": 3.38 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 11.24, + "z": 3.38 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 11.24, + "z": 3.38 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 11.24, + "z": 3.38 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 11.24, + "z": 3.38 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 11.24, + "z": 3.38 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "018c33eefca131b3f7a2381ee5dd4451", + "notes": [], + "params": { + "displayName": "Temperature-Controlled plate", + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "labwareId": "UUID" + }, + "namespace": "opentrons", + "version": 2 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "NEST", + "brandId": [ + "402501" + ], + "links": [ + "https://www.nest-biotech.com/pcr-plates/58773587.html" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 15.7 + }, + "gripForce": 15.0, + "gripHeightFromLabwareBottom": 10.65, + "gripperOffsets": {}, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "wellPlate", + "displayName": "NEST 96 Well Plate 100 µL PCR Full Skirt", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": true, + "isTiprack": false, + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "magneticModuleEngageHeight": 20 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_96_pcr_adapter": { + "x": 0, + "y": 0, + "z": 10.2 + }, + "opentrons_96_well_aluminum_block": { + "x": 0, + "y": 0, + "z": 12.66 + } + }, + "stackingOffsetWithModule": { + "thermocyclerModuleV2": { + "x": 0, + "y": 0, + "z": 10.8 + } + }, + "version": 2, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 74.24, + "z": 0.92 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 74.24, + "z": 0.92 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 74.24, + "z": 0.92 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 74.24, + "z": 0.92 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 74.24, + "z": 0.92 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 74.24, + "z": 0.92 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 74.24, + "z": 0.92 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 74.24, + "z": 0.92 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 74.24, + "z": 0.92 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 74.24, + "z": 0.92 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 74.24, + "z": 0.92 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 74.24, + "z": 0.92 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 65.24, + "z": 0.92 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 65.24, + "z": 0.92 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 65.24, + "z": 0.92 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 65.24, + "z": 0.92 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 65.24, + "z": 0.92 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 65.24, + "z": 0.92 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 65.24, + "z": 0.92 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 65.24, + "z": 0.92 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 65.24, + "z": 0.92 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 65.24, + "z": 0.92 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 65.24, + "z": 0.92 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 65.24, + "z": 0.92 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 56.24, + "z": 0.92 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 56.24, + "z": 0.92 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 56.24, + "z": 0.92 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 56.24, + "z": 0.92 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 56.24, + "z": 0.92 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 56.24, + "z": 0.92 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 56.24, + "z": 0.92 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 56.24, + "z": 0.92 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 56.24, + "z": 0.92 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 56.24, + "z": 0.92 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 56.24, + "z": 0.92 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 56.24, + "z": 0.92 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 47.24, + "z": 0.92 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 47.24, + "z": 0.92 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 47.24, + "z": 0.92 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 47.24, + "z": 0.92 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 47.24, + "z": 0.92 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 47.24, + "z": 0.92 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 47.24, + "z": 0.92 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 47.24, + "z": 0.92 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 47.24, + "z": 0.92 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 47.24, + "z": 0.92 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 47.24, + "z": 0.92 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 47.24, + "z": 0.92 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 38.24, + "z": 0.92 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 38.24, + "z": 0.92 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 38.24, + "z": 0.92 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 38.24, + "z": 0.92 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 38.24, + "z": 0.92 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 38.24, + "z": 0.92 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 38.24, + "z": 0.92 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 38.24, + "z": 0.92 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 38.24, + "z": 0.92 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 38.24, + "z": 0.92 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 38.24, + "z": 0.92 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 38.24, + "z": 0.92 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 29.24, + "z": 0.92 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 29.24, + "z": 0.92 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 29.24, + "z": 0.92 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 29.24, + "z": 0.92 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 29.24, + "z": 0.92 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 29.24, + "z": 0.92 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 29.24, + "z": 0.92 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 29.24, + "z": 0.92 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 29.24, + "z": 0.92 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 29.24, + "z": 0.92 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 29.24, + "z": 0.92 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 29.24, + "z": 0.92 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 20.24, + "z": 0.92 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 20.24, + "z": 0.92 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 20.24, + "z": 0.92 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 20.24, + "z": 0.92 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 20.24, + "z": 0.92 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 20.24, + "z": 0.92 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 20.24, + "z": 0.92 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 20.24, + "z": 0.92 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 20.24, + "z": 0.92 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 20.24, + "z": 0.92 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 20.24, + "z": 0.92 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 20.24, + "z": 0.92 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 11.24, + "z": 0.92 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 11.24, + "z": 0.92 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 11.24, + "z": 0.92 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 11.24, + "z": 0.92 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 11.24, + "z": 0.92 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 11.24, + "z": 0.92 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 11.24, + "z": 0.92 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 11.24, + "z": 0.92 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 11.24, + "z": 0.92 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 11.24, + "z": 0.92 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 11.24, + "z": 0.92 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 11.24, + "z": 0.92 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c721864b5eca3153a587771422848a84", + "notes": [], + "params": { + "loadName": "opentrons_96_pcr_adapter", + "location": { + "moduleId": "UUID" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [ + "adapter" + ], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 8.5, + "y": 5.5, + "z": 0 + }, + "dimensions": { + "xDimension": 111, + "yDimension": 75, + "zDimension": 13.85 + }, + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0, + "y": 0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "adapter", + "displayName": "Opentrons 96 PCR Heater-Shaker Adapter", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "opentrons_96_pcr_adapter", + "quirks": [] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 69, + "z": 1.85 + }, + "A10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 69, + "z": 1.85 + }, + "A11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 69, + "z": 1.85 + }, + "A12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 69, + "z": 1.85 + }, + "A2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 69, + "z": 1.85 + }, + "A3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 69, + "z": 1.85 + }, + "A4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 69, + "z": 1.85 + }, + "A5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 69, + "z": 1.85 + }, + "A6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 69, + "z": 1.85 + }, + "A7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 69, + "z": 1.85 + }, + "A8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 69, + "z": 1.85 + }, + "A9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 69, + "z": 1.85 + }, + "B1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 60, + "z": 1.85 + }, + "B10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 60, + "z": 1.85 + }, + "B11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 60, + "z": 1.85 + }, + "B12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 60, + "z": 1.85 + }, + "B2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 60, + "z": 1.85 + }, + "B3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 60, + "z": 1.85 + }, + "B4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 60, + "z": 1.85 + }, + "B5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 60, + "z": 1.85 + }, + "B6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 60, + "z": 1.85 + }, + "B7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 60, + "z": 1.85 + }, + "B8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 60, + "z": 1.85 + }, + "B9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 60, + "z": 1.85 + }, + "C1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 51, + "z": 1.85 + }, + "C10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 51, + "z": 1.85 + }, + "C11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 51, + "z": 1.85 + }, + "C12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 51, + "z": 1.85 + }, + "C2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 51, + "z": 1.85 + }, + "C3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 51, + "z": 1.85 + }, + "C4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 51, + "z": 1.85 + }, + "C5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 51, + "z": 1.85 + }, + "C6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 51, + "z": 1.85 + }, + "C7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 51, + "z": 1.85 + }, + "C8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 51, + "z": 1.85 + }, + "C9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 51, + "z": 1.85 + }, + "D1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 42, + "z": 1.85 + }, + "D10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 42, + "z": 1.85 + }, + "D11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 42, + "z": 1.85 + }, + "D12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 42, + "z": 1.85 + }, + "D2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 42, + "z": 1.85 + }, + "D3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 42, + "z": 1.85 + }, + "D4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 42, + "z": 1.85 + }, + "D5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 42, + "z": 1.85 + }, + "D6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 42, + "z": 1.85 + }, + "D7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 42, + "z": 1.85 + }, + "D8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 42, + "z": 1.85 + }, + "D9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 42, + "z": 1.85 + }, + "E1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 33, + "z": 1.85 + }, + "E10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 33, + "z": 1.85 + }, + "E11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 33, + "z": 1.85 + }, + "E12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 33, + "z": 1.85 + }, + "E2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 33, + "z": 1.85 + }, + "E3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 33, + "z": 1.85 + }, + "E4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 33, + "z": 1.85 + }, + "E5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 33, + "z": 1.85 + }, + "E6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 33, + "z": 1.85 + }, + "E7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 33, + "z": 1.85 + }, + "E8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 33, + "z": 1.85 + }, + "E9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 33, + "z": 1.85 + }, + "F1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 24, + "z": 1.85 + }, + "F10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 24, + "z": 1.85 + }, + "F11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 24, + "z": 1.85 + }, + "F12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 24, + "z": 1.85 + }, + "F2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 24, + "z": 1.85 + }, + "F3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 24, + "z": 1.85 + }, + "F4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 24, + "z": 1.85 + }, + "F5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 24, + "z": 1.85 + }, + "F6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 24, + "z": 1.85 + }, + "F7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 24, + "z": 1.85 + }, + "F8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 24, + "z": 1.85 + }, + "F9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 24, + "z": 1.85 + }, + "G1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 15, + "z": 1.85 + }, + "G10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 15, + "z": 1.85 + }, + "G11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 15, + "z": 1.85 + }, + "G12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 15, + "z": 1.85 + }, + "G2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 15, + "z": 1.85 + }, + "G3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 15, + "z": 1.85 + }, + "G4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 15, + "z": 1.85 + }, + "G5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 15, + "z": 1.85 + }, + "G6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 15, + "z": 1.85 + }, + "G7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 15, + "z": 1.85 + }, + "G8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 15, + "z": 1.85 + }, + "G9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 15, + "z": 1.85 + }, + "H1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 6, + "z": 1.85 + }, + "H10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 6, + "z": 1.85 + }, + "H11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 6, + "z": 1.85 + }, + "H12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 6, + "z": 1.85 + }, + "H2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 6, + "z": 1.85 + }, + "H3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 6, + "z": 1.85 + }, + "H4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 6, + "z": 1.85 + }, + "H5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 6, + "z": 1.85 + }, + "H6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 6, + "z": 1.85 + }, + "H7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 6, + "z": 1.85 + }, + "H8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 6, + "z": 1.85 + }, + "H9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 6, + "z": 1.85 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "0137be81e4cddff6b188c578784942ec", + "notes": [], + "params": { + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "labwareId": "UUID" + }, + "namespace": "opentrons", + "version": 2 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "NEST", + "brandId": [ + "402501" + ], + "links": [ + "https://www.nest-biotech.com/pcr-plates/58773587.html" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 15.7 + }, + "gripForce": 15.0, + "gripHeightFromLabwareBottom": 10.65, + "gripperOffsets": {}, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "wellPlate", + "displayName": "NEST 96 Well Plate 100 µL PCR Full Skirt", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": true, + "isTiprack": false, + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "magneticModuleEngageHeight": 20 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_96_pcr_adapter": { + "x": 0, + "y": 0, + "z": 10.2 + }, + "opentrons_96_well_aluminum_block": { + "x": 0, + "y": 0, + "z": 12.66 + } + }, + "stackingOffsetWithModule": { + "thermocyclerModuleV2": { + "x": 0, + "y": 0, + "z": 10.8 + } + }, + "version": 2, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 74.24, + "z": 0.92 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 74.24, + "z": 0.92 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 74.24, + "z": 0.92 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 74.24, + "z": 0.92 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 74.24, + "z": 0.92 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 74.24, + "z": 0.92 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 74.24, + "z": 0.92 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 74.24, + "z": 0.92 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 74.24, + "z": 0.92 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 74.24, + "z": 0.92 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 74.24, + "z": 0.92 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 74.24, + "z": 0.92 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 65.24, + "z": 0.92 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 65.24, + "z": 0.92 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 65.24, + "z": 0.92 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 65.24, + "z": 0.92 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 65.24, + "z": 0.92 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 65.24, + "z": 0.92 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 65.24, + "z": 0.92 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 65.24, + "z": 0.92 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 65.24, + "z": 0.92 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 65.24, + "z": 0.92 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 65.24, + "z": 0.92 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 65.24, + "z": 0.92 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 56.24, + "z": 0.92 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 56.24, + "z": 0.92 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 56.24, + "z": 0.92 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 56.24, + "z": 0.92 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 56.24, + "z": 0.92 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 56.24, + "z": 0.92 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 56.24, + "z": 0.92 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 56.24, + "z": 0.92 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 56.24, + "z": 0.92 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 56.24, + "z": 0.92 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 56.24, + "z": 0.92 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 56.24, + "z": 0.92 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 47.24, + "z": 0.92 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 47.24, + "z": 0.92 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 47.24, + "z": 0.92 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 47.24, + "z": 0.92 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 47.24, + "z": 0.92 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 47.24, + "z": 0.92 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 47.24, + "z": 0.92 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 47.24, + "z": 0.92 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 47.24, + "z": 0.92 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 47.24, + "z": 0.92 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 47.24, + "z": 0.92 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 47.24, + "z": 0.92 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 38.24, + "z": 0.92 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 38.24, + "z": 0.92 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 38.24, + "z": 0.92 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 38.24, + "z": 0.92 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 38.24, + "z": 0.92 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 38.24, + "z": 0.92 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 38.24, + "z": 0.92 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 38.24, + "z": 0.92 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 38.24, + "z": 0.92 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 38.24, + "z": 0.92 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 38.24, + "z": 0.92 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 38.24, + "z": 0.92 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 29.24, + "z": 0.92 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 29.24, + "z": 0.92 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 29.24, + "z": 0.92 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 29.24, + "z": 0.92 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 29.24, + "z": 0.92 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 29.24, + "z": 0.92 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 29.24, + "z": 0.92 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 29.24, + "z": 0.92 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 29.24, + "z": 0.92 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 29.24, + "z": 0.92 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 29.24, + "z": 0.92 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 29.24, + "z": 0.92 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 20.24, + "z": 0.92 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 20.24, + "z": 0.92 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 20.24, + "z": 0.92 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 20.24, + "z": 0.92 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 20.24, + "z": 0.92 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 20.24, + "z": 0.92 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 20.24, + "z": 0.92 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 20.24, + "z": 0.92 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 20.24, + "z": 0.92 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 20.24, + "z": 0.92 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 20.24, + "z": 0.92 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 20.24, + "z": 0.92 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 11.24, + "z": 0.92 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 11.24, + "z": 0.92 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 11.24, + "z": 0.92 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 11.24, + "z": 0.92 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 11.24, + "z": 0.92 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 11.24, + "z": 0.92 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 11.24, + "z": 0.92 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 11.24, + "z": 0.92 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 11.24, + "z": 0.92 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 11.24, + "z": 0.92 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 11.24, + "z": 0.92 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 11.24, + "z": 0.92 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7feed4cbc55e2030e0692ba1082065f4", + "notes": [], + "params": { + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "moduleId": "UUID" + }, + "namespace": "opentrons", + "version": 2 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "NEST", + "brandId": [ + "402501" + ], + "links": [ + "https://www.nest-biotech.com/pcr-plates/58773587.html" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 15.7 + }, + "gripForce": 15.0, + "gripHeightFromLabwareBottom": 10.65, + "gripperOffsets": {}, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "wellPlate", + "displayName": "NEST 96 Well Plate 100 µL PCR Full Skirt", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": true, + "isTiprack": false, + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "magneticModuleEngageHeight": 20 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_96_pcr_adapter": { + "x": 0, + "y": 0, + "z": 10.2 + }, + "opentrons_96_well_aluminum_block": { + "x": 0, + "y": 0, + "z": 12.66 + } + }, + "stackingOffsetWithModule": { + "thermocyclerModuleV2": { + "x": 0, + "y": 0, + "z": 10.8 + } + }, + "version": 2, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 74.24, + "z": 0.92 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 74.24, + "z": 0.92 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 74.24, + "z": 0.92 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 74.24, + "z": 0.92 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 74.24, + "z": 0.92 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 74.24, + "z": 0.92 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 74.24, + "z": 0.92 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 74.24, + "z": 0.92 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 74.24, + "z": 0.92 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 74.24, + "z": 0.92 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 74.24, + "z": 0.92 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 74.24, + "z": 0.92 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 65.24, + "z": 0.92 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 65.24, + "z": 0.92 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 65.24, + "z": 0.92 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 65.24, + "z": 0.92 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 65.24, + "z": 0.92 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 65.24, + "z": 0.92 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 65.24, + "z": 0.92 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 65.24, + "z": 0.92 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 65.24, + "z": 0.92 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 65.24, + "z": 0.92 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 65.24, + "z": 0.92 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 65.24, + "z": 0.92 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 56.24, + "z": 0.92 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 56.24, + "z": 0.92 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 56.24, + "z": 0.92 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 56.24, + "z": 0.92 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 56.24, + "z": 0.92 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 56.24, + "z": 0.92 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 56.24, + "z": 0.92 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 56.24, + "z": 0.92 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 56.24, + "z": 0.92 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 56.24, + "z": 0.92 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 56.24, + "z": 0.92 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 56.24, + "z": 0.92 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 47.24, + "z": 0.92 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 47.24, + "z": 0.92 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 47.24, + "z": 0.92 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 47.24, + "z": 0.92 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 47.24, + "z": 0.92 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 47.24, + "z": 0.92 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 47.24, + "z": 0.92 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 47.24, + "z": 0.92 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 47.24, + "z": 0.92 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 47.24, + "z": 0.92 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 47.24, + "z": 0.92 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 47.24, + "z": 0.92 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 38.24, + "z": 0.92 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 38.24, + "z": 0.92 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 38.24, + "z": 0.92 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 38.24, + "z": 0.92 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 38.24, + "z": 0.92 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 38.24, + "z": 0.92 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 38.24, + "z": 0.92 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 38.24, + "z": 0.92 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 38.24, + "z": 0.92 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 38.24, + "z": 0.92 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 38.24, + "z": 0.92 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 38.24, + "z": 0.92 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 29.24, + "z": 0.92 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 29.24, + "z": 0.92 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 29.24, + "z": 0.92 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 29.24, + "z": 0.92 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 29.24, + "z": 0.92 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 29.24, + "z": 0.92 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 29.24, + "z": 0.92 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 29.24, + "z": 0.92 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 29.24, + "z": 0.92 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 29.24, + "z": 0.92 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 29.24, + "z": 0.92 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 29.24, + "z": 0.92 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 20.24, + "z": 0.92 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 20.24, + "z": 0.92 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 20.24, + "z": 0.92 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 20.24, + "z": 0.92 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 20.24, + "z": 0.92 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 20.24, + "z": 0.92 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 20.24, + "z": 0.92 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 20.24, + "z": 0.92 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 20.24, + "z": 0.92 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 20.24, + "z": 0.92 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 20.24, + "z": 0.92 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 20.24, + "z": 0.92 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 11.24, + "z": 0.92 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 11.24, + "z": 0.92 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 11.24, + "z": 0.92 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 11.24, + "z": 0.92 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 11.24, + "z": 0.92 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 11.24, + "z": 0.92 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 11.24, + "z": 0.92 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 11.24, + "z": 0.92 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 11.24, + "z": 0.92 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 11.24, + "z": 0.92 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 11.24, + "z": 0.92 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 11.24, + "z": 0.92 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "error": { + "createdAt": "TIMESTAMP", + "detail": "FileNotFoundError: Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".", + "errorCode": "4000", + "errorInfo": { + "args": "('Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".',)", + "class": "FileNotFoundError", + "errno": "None", + "filename": "None", + "filename2": "None", + "strerror": "None", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/command_executor.py\", line N, in execute\n result = await command_impl.execute(\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/commands/load_labware.py\", line N, in execute\n loaded_labware = await self._equipment.load_labware(\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/equipment.py\", line N, in load_labware\n definition = await self._labware_data_provider.get_labware_definition(\n\n File \"/usr/local/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line N, in run_sync_in_worker_thread\n return await future\n\n File \"/usr/local/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line N, in run\n result = context.run(func, *args)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/resources/labware_data_provider.py\", line N, in _get_labware_definition_sync\n get_labware_definition(load_name, namespace, version)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocols/labware.py\", line N, in get_labware_definition\n return _get_standard_labware_definition(load_name, namespace, version)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocols/labware.py\", line N, in _get_standard_labware_definition\n raise FileNotFoundError(\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "opentrons.protocol_engine.errors.exceptions.LabwareDefinitionDoesNotExistError: Error 4000 GENERAL_ERROR (LabwareDefinitionDoesNotExistError): Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.", + "errorCode": "4000", + "errorInfo": { + "args": "('Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.',)", + "class": "LabwareDefinitionDoesNotExistError", + "code": "ErrorCodes.GENERAL_ERROR", + "detail": "{}", + "message": "Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/equipment.py\", line N, in load_labware\n definition = self._state_store.labware.get_definition_by_uri(definition_uri)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n raise errors.LabwareDefinitionDoesNotExistError(\n", + "wrapping": "[]" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "KeyError: 'opentrons/cpx_4_tuberack_100ul/1'", + "errorCode": "4000", + "errorInfo": { + "args": "('opentrons/cpx_4_tuberack_100ul/1',)", + "class": "KeyError", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n return self._state.definitions_by_uri[uri]\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [] + }, + { + "createdAt": "TIMESTAMP", + "detail": "KeyError: 'opentrons/cpx_4_tuberack_100ul/1'", + "errorCode": "4000", + "errorInfo": { + "args": "('opentrons/cpx_4_tuberack_100ul/1',)", + "class": "KeyError", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n return self._state.definitions_by_uri[uri]\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [] + } + ] + } + ] + }, + "id": "UUID", + "key": "08e16a2cac011d4bef561f8b0854d19e", + "notes": [], + "params": { + "displayName": "4 custom tubes", + "loadName": "cpx_4_tuberack_100ul", + "location": { + "slotName": "6" + }, + "namespace": "opentrons", + "version": 1 + }, + "startedAt": "TIMESTAMP", + "status": "failed" + } + ], + "config": { + "apiVersion": [ + 2, + 19 + ], + "protocolType": "python" + }, + "createdAt": "TIMESTAMP", + "errors": [ + { + "createdAt": "TIMESTAMP", + "detail": "ProtocolCommandFailedError [line 232]: Error 4000 GENERAL_ERROR (ProtocolCommandFailedError): PythonException: FileNotFoundError: Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".", + "errorCode": "4000", + "errorInfo": {}, + "errorType": "ExceptionInProtocolError", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "PythonException: FileNotFoundError: Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".", + "errorCode": "4000", + "errorInfo": {}, + "errorType": "ProtocolCommandFailedError", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "FileNotFoundError: Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".", + "errorCode": "4000", + "errorInfo": { + "args": "('Labware \"cpx_4_tuberack_100ul\" not found with version 1 in namespace \"opentrons\".',)", + "class": "FileNotFoundError", + "errno": "None", + "filename": "None", + "filename2": "None", + "strerror": "None", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/command_executor.py\", line N, in execute\n result = await command_impl.execute(\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/commands/load_labware.py\", line N, in execute\n loaded_labware = await self._equipment.load_labware(\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/equipment.py\", line N, in load_labware\n definition = await self._labware_data_provider.get_labware_definition(\n\n File \"/usr/local/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line N, in run_sync_in_worker_thread\n return await future\n\n File \"/usr/local/lib/python3.10/site-packages/anyio/_backends/_asyncio.py\", line N, in run\n result = context.run(func, *args)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/resources/labware_data_provider.py\", line N, in _get_labware_definition_sync\n get_labware_definition(load_name, namespace, version)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocols/labware.py\", line N, in get_labware_definition\n return _get_standard_labware_definition(load_name, namespace, version)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocols/labware.py\", line N, in _get_standard_labware_definition\n raise FileNotFoundError(\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "opentrons.protocol_engine.errors.exceptions.LabwareDefinitionDoesNotExistError: Error 4000 GENERAL_ERROR (LabwareDefinitionDoesNotExistError): Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.", + "errorCode": "4000", + "errorInfo": { + "args": "('Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.',)", + "class": "LabwareDefinitionDoesNotExistError", + "code": "ErrorCodes.GENERAL_ERROR", + "detail": "{}", + "message": "Labware definition for matching opentrons/cpx_4_tuberack_100ul/1 not found.", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/execution/equipment.py\", line N, in load_labware\n definition = self._state_store.labware.get_definition_by_uri(definition_uri)\n\n File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n raise errors.LabwareDefinitionDoesNotExistError(\n", + "wrapping": "[]" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [ + { + "createdAt": "TIMESTAMP", + "detail": "KeyError: 'opentrons/cpx_4_tuberack_100ul/1'", + "errorCode": "4000", + "errorInfo": { + "args": "('opentrons/cpx_4_tuberack_100ul/1',)", + "class": "KeyError", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n return self._state.definitions_by_uri[uri]\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [] + }, + { + "createdAt": "TIMESTAMP", + "detail": "KeyError: 'opentrons/cpx_4_tuberack_100ul/1'", + "errorCode": "4000", + "errorInfo": { + "args": "('opentrons/cpx_4_tuberack_100ul/1',)", + "class": "KeyError", + "traceback": " File \"/usr/local/lib/python3.10/site-packages/opentrons/protocol_engine/state/labware.py\", line N, in get_definition_by_uri\n return self._state.definitions_by_uri[uri]\n" + }, + "errorType": "PythonException", + "id": "UUID", + "isDefined": false, + "wrappedErrors": [] + } + ] + } + ] + } + ] + } + ] + } + ], + "files": [ + { + "name": "OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py", + "role": "main" + } + ], + "labware": [ + { + "definitionUri": "opentrons/opentrons_96_tiprack_300ul/1", + "displayName": "300ul tips", + "id": "UUID", + "loadName": "opentrons_96_tiprack_300ul", + "location": { + "slotName": "5" + } + }, + { + "definitionUri": "opentrons/opentrons_96_tiprack_20ul/1", + "displayName": "20ul tips", + "id": "UUID", + "loadName": "opentrons_96_tiprack_20ul", + "location": { + "slotName": "4" + } + }, + { + "definitionUri": "opentrons/opentrons_96_well_aluminum_block/1", + "id": "UUID", + "loadName": "opentrons_96_well_aluminum_block", + "location": { + "moduleId": "UUID" + } + }, + { + "definitionUri": "opentrons/nest_96_wellplate_100ul_pcr_full_skirt/2", + "displayName": "Temperature-Controlled plate", + "id": "UUID", + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "labwareId": "UUID" + } + }, + { + "definitionUri": "opentrons/opentrons_96_pcr_adapter/1", + "id": "UUID", + "loadName": "opentrons_96_pcr_adapter", + "location": { + "moduleId": "UUID" + } + }, + { + "definitionUri": "opentrons/nest_96_wellplate_100ul_pcr_full_skirt/2", + "id": "UUID", + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "labwareId": "UUID" + } + }, + { + "definitionUri": "opentrons/nest_96_wellplate_100ul_pcr_full_skirt/2", + "id": "UUID", + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "moduleId": "UUID" + } + } + ], + "liquids": [], + "metadata": { + "author": "Opentrons Engineering ", + "description": "Description of the protocol that is longish \n has \n returns and \n emoji 😊 ⬆️ ", + "protocolName": "🛠️ 2.19 Smoke Test V3 🪄", + "source": "Software Testing Team" + }, + "modules": [ + { + "id": "UUID", + "location": { + "slotName": "1" + }, + "model": "heaterShakerModuleV1", + "serialNumber": "UUID" + }, + { + "id": "UUID", + "location": { + "slotName": "9" + }, + "model": "temperatureModuleV2", + "serialNumber": "UUID" + }, + { + "id": "UUID", + "location": { + "slotName": "7" + }, + "model": "thermocyclerModuleV2", + "serialNumber": "UUID" + } + ], + "pipettes": [ + { + "id": "UUID", + "mount": "left", + "pipetteName": "p300_multi_gen2" + }, + { + "id": "UUID", + "mount": "right", + "pipetteName": "p20_single_gen2" + } + ], + "result": "not-ok", + "robotType": "OT-2 Standard", + "runTimeParameters": [ + { + "choices": [ + { + "displayName": "Nest 12 Well 15 mL", + "value": "nest_12_reservoir_15ml" + }, + { + "displayName": "USA Scientific 12 Well 22 mL", + "value": "usascientific_12_reservoir_22ml" + } + ], + "default": "nest_12_reservoir_15ml", + "description": "Name of the reservoir", + "displayName": "Reservoir Name", + "type": "str", + "value": "nest_12_reservoir_15ml", + "variableName": "reservoir_name" + }, + { + "choices": [ + { + "displayName": "Nest 96 Well 100 µL", + "value": "nest_96_wellplate_100ul_pcr_full_skirt" + }, + { + "displayName": "Corning 96 Well 360 µL", + "value": "corning_96_wellplate_360ul_flat" + }, + { + "displayName": "Opentrons Tough 96 Well 200 µL", + "value": "opentrons_96_wellplate_200ul_pcr_full_skirt" + } + ], + "default": "nest_96_wellplate_100ul_pcr_full_skirt", + "description": "Name of the well plate", + "displayName": "Well Plate Name", + "type": "str", + "value": "nest_96_wellplate_100ul_pcr_full_skirt", + "variableName": "well_plate_name" + }, + { + "default": 3.0, + "description": "Time to delay in seconds", + "displayName": "Delay Time", + "max": 10.0, + "min": 1.0, + "suffix": "seconds", + "type": "int", + "value": 3.0, + "variableName": "delay_time" + }, + { + "default": true, + "description": "Turn on the robot lights?", + "displayName": "Robot Lights", + "type": "bool", + "value": true, + "variableName": "robot_lights" + }, + { + "default": 38.0, + "description": "Temperature to set the heater shaker to", + "displayName": "Heater Shaker Temperature", + "max": 100.0, + "min": 37.0, + "suffix": "°C", + "type": "float", + "value": 38.0, + "variableName": "heater_shaker_temperature" + } + ] +} diff --git a/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[dabb7872d8][Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke].json b/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[dabb7872d8][Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke].json new file mode 100644 index 00000000000..0e252fdc93b --- /dev/null +++ b/analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[dabb7872d8][Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke].json @@ -0,0 +1,13236 @@ +{ + "commands": [ + { + "commandType": "home", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "50c7ae73a4e3f7129874f39dfb514803", + "notes": [], + "params": {}, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "8511b05ba5565bf0e6dcccd800e2ee23", + "notes": [], + "params": { + "location": { + "slotName": "B1" + }, + "model": "thermocyclerModuleV2" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 14.4, + "y": 64.93, + "z": 97.8 + }, + "compatibleWith": [], + "dimensions": { + "bareOverallHeight": 108.96, + "lidHeight": 61.7, + "overLabwareHeight": 0.0 + }, + "displayName": "Thermocycler Module GEN2", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 5.6 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 4.6 + } + } + }, + "labwareOffset": { + "x": 0.0, + "y": 68.8, + "z": 108.96 + }, + "model": "thermocyclerModuleV2", + "moduleType": "thermocyclerModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot3_standard": { + "B1": { + "cornerOffsetFromSlot": [ + [ + -98, + 0, + 0, + 1 + ], + [ + -20.005, + 0, + 0, + 1 + ], + [ + -0.84, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ], + "labwareOffset": [ + [ + -98, + 0, + 0, + 1 + ], + [ + -20.005, + 0, + 0, + 1 + ], + [ + -0.84, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + } + } + }, + "model": "thermocyclerModuleV2", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b97e4480a1578eb15e73787ba193bdd5", + "notes": [], + "params": { + "location": { + "slotName": "C1" + }, + "model": "magneticBlockV1" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "compatibleWith": [], + "dimensions": { + "bareOverallHeight": 45.0, + "overLabwareHeight": 0.0 + }, + "displayName": "Magnetic Block GEN1", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + } + } + }, + "labwareOffset": { + "x": 0.0, + "y": 0.0, + "z": 38.0 + }, + "model": "magneticBlockV1", + "moduleType": "magneticBlockType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot2_short_trash": {}, + "ot2_standard": {}, + "ot3_standard": {} + } + }, + "model": "magneticBlockV1", + "moduleId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5f5358348287e62e64a99dc4423561f5", + "notes": [], + "params": { + "location": { + "slotName": "A3" + }, + "model": "heaterShakerModuleV1" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 12.0, + "y": 8.75, + "z": 68.275 + }, + "compatibleWith": [], + "dimensions": { + "bareOverallHeight": 82.0, + "overLabwareHeight": 0.0 + }, + "displayName": "Heater-Shaker Module GEN1", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + } + } + }, + "labwareOffset": { + "x": -0.125, + "y": 1.125, + "z": 68.275 + }, + "model": "heaterShakerModuleV1", + "moduleType": "heaterShakerModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot2_short_trash": { + "3": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot2_standard": { + "3": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + 0, + 0, + 0 + ], + [ + -1, + 0, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot3_standard": { + "A1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "A3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "B1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "B3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "C1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "C3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "D1": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + }, + "D3": { + "labwareOffset": [ + [ + -49.325, + 0, + 0, + 1 + ], + [ + -1.125, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.125, + 1 + ] + ] + } + } + } + }, + "model": "heaterShakerModuleV1", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadModule", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1bdb1d25a2f030a6ee69219303e3b8df", + "notes": [], + "params": { + "location": { + "slotName": "D1" + }, + "model": "temperatureModuleV2" + }, + "result": { + "definition": { + "calibrationPoint": { + "x": 11.7, + "y": 8.75, + "z": 80.09 + }, + "compatibleWith": [ + "temperatureModuleV1" + ], + "dimensions": { + "bareOverallHeight": 84.0, + "overLabwareHeight": 0.0 + }, + "displayName": "Temperature Module GEN2", + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0.0, + "y": 0.0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + } + } + }, + "labwareOffset": { + "x": -1.45, + "y": -0.15, + "z": 80.09 + }, + "model": "temperatureModuleV2", + "moduleType": "temperatureModuleType", + "otSharedSchema": "module/schemas/2", + "quirks": [], + "slotTransforms": { + "ot2_short_trash": { + "3": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + -0.15, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot2_standard": { + "3": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "6": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + }, + "9": { + "labwareOffset": [ + [ + -1, + -0.3, + 0, + 0 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ] + ] + } + }, + "ot3_standard": { + "A1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "A3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "B1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "B3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "C1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "C3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "D1": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + }, + "D3": { + "labwareOffset": [ + [ + -71.09, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0, + 1 + ], + [ + 0, + 0, + 0.15, + 1 + ], + [ + 0, + 0, + 1, + 1.45 + ] + ] + } + } + } + }, + "model": "temperatureModuleV2", + "moduleId": "UUID", + "serialNumber": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/openLid", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6715dfdabd16f6acb7c207dbf23a87d9", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/openLabwareLatch", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "983d3f6fe64a3f60de367ee4ff439714", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": { + "pipetteRetracted": true + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9b3efe722729a38b24e14fabc8fe10cd", + "notes": [], + "params": { + "loadName": "opentrons_96_well_aluminum_block", + "location": { + "moduleId": "UUID" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [ + "adapter" + ], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 18.16 + }, + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0, + "y": 0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "aluminumBlock", + "displayName": "Opentrons 96 Well Aluminum Block", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "opentrons_96_well_aluminum_block", + "quirks": [] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 74.24, + "z": 3.38 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 74.24, + "z": 3.38 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 74.24, + "z": 3.38 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 74.24, + "z": 3.38 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 74.24, + "z": 3.38 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 74.24, + "z": 3.38 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 74.24, + "z": 3.38 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 74.24, + "z": 3.38 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 74.24, + "z": 3.38 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 74.24, + "z": 3.38 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 74.24, + "z": 3.38 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 74.24, + "z": 3.38 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 65.24, + "z": 3.38 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 65.24, + "z": 3.38 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 65.24, + "z": 3.38 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 65.24, + "z": 3.38 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 65.24, + "z": 3.38 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 65.24, + "z": 3.38 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 65.24, + "z": 3.38 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 65.24, + "z": 3.38 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 65.24, + "z": 3.38 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 65.24, + "z": 3.38 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 65.24, + "z": 3.38 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 65.24, + "z": 3.38 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 56.24, + "z": 3.38 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 56.24, + "z": 3.38 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 56.24, + "z": 3.38 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 56.24, + "z": 3.38 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 56.24, + "z": 3.38 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 56.24, + "z": 3.38 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 56.24, + "z": 3.38 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 56.24, + "z": 3.38 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 56.24, + "z": 3.38 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 56.24, + "z": 3.38 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 56.24, + "z": 3.38 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 56.24, + "z": 3.38 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 47.24, + "z": 3.38 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 47.24, + "z": 3.38 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 47.24, + "z": 3.38 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 47.24, + "z": 3.38 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 47.24, + "z": 3.38 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 47.24, + "z": 3.38 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 47.24, + "z": 3.38 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 47.24, + "z": 3.38 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 47.24, + "z": 3.38 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 47.24, + "z": 3.38 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 47.24, + "z": 3.38 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 47.24, + "z": 3.38 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 38.24, + "z": 3.38 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 38.24, + "z": 3.38 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 38.24, + "z": 3.38 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 38.24, + "z": 3.38 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 38.24, + "z": 3.38 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 38.24, + "z": 3.38 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 38.24, + "z": 3.38 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 38.24, + "z": 3.38 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 38.24, + "z": 3.38 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 38.24, + "z": 3.38 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 38.24, + "z": 3.38 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 38.24, + "z": 3.38 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 29.24, + "z": 3.38 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 29.24, + "z": 3.38 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 29.24, + "z": 3.38 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 29.24, + "z": 3.38 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 29.24, + "z": 3.38 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 29.24, + "z": 3.38 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 29.24, + "z": 3.38 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 29.24, + "z": 3.38 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 29.24, + "z": 3.38 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 29.24, + "z": 3.38 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 29.24, + "z": 3.38 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 29.24, + "z": 3.38 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 20.24, + "z": 3.38 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 20.24, + "z": 3.38 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 20.24, + "z": 3.38 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 20.24, + "z": 3.38 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 20.24, + "z": 3.38 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 20.24, + "z": 3.38 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 20.24, + "z": 3.38 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 20.24, + "z": 3.38 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 20.24, + "z": 3.38 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 20.24, + "z": 3.38 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 20.24, + "z": 3.38 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 20.24, + "z": 3.38 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 14.38, + "y": 11.24, + "z": 3.38 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 95.38, + "y": 11.24, + "z": 3.38 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 104.38, + "y": 11.24, + "z": 3.38 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 113.38, + "y": 11.24, + "z": 3.38 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 23.38, + "y": 11.24, + "z": 3.38 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 32.38, + "y": 11.24, + "z": 3.38 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 41.38, + "y": 11.24, + "z": 3.38 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 50.38, + "y": 11.24, + "z": 3.38 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 59.38, + "y": 11.24, + "z": 3.38 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 68.38, + "y": 11.24, + "z": 3.38 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 77.38, + "y": 11.24, + "z": 3.38 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 86.38, + "y": 11.24, + "z": 3.38 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "567e61648ee88976deb8d73faac0e083", + "notes": [], + "params": { + "loadName": "opentrons_96_pcr_adapter", + "location": { + "moduleId": "UUID" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [ + "adapter" + ], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 8.5, + "y": 5.5, + "z": 0 + }, + "dimensions": { + "xDimension": 111, + "yDimension": 75, + "zDimension": 13.85 + }, + "gripperOffsets": { + "default": { + "dropOffset": { + "x": 0, + "y": 0, + "z": 1.0 + }, + "pickUpOffset": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "adapter", + "displayName": "Opentrons 96 PCR Heater-Shaker Adapter", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "opentrons_96_pcr_adapter", + "quirks": [] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 69, + "z": 1.85 + }, + "A10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 69, + "z": 1.85 + }, + "A11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 69, + "z": 1.85 + }, + "A12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 69, + "z": 1.85 + }, + "A2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 69, + "z": 1.85 + }, + "A3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 69, + "z": 1.85 + }, + "A4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 69, + "z": 1.85 + }, + "A5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 69, + "z": 1.85 + }, + "A6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 69, + "z": 1.85 + }, + "A7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 69, + "z": 1.85 + }, + "A8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 69, + "z": 1.85 + }, + "A9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 69, + "z": 1.85 + }, + "B1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 60, + "z": 1.85 + }, + "B10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 60, + "z": 1.85 + }, + "B11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 60, + "z": 1.85 + }, + "B12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 60, + "z": 1.85 + }, + "B2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 60, + "z": 1.85 + }, + "B3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 60, + "z": 1.85 + }, + "B4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 60, + "z": 1.85 + }, + "B5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 60, + "z": 1.85 + }, + "B6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 60, + "z": 1.85 + }, + "B7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 60, + "z": 1.85 + }, + "B8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 60, + "z": 1.85 + }, + "B9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 60, + "z": 1.85 + }, + "C1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 51, + "z": 1.85 + }, + "C10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 51, + "z": 1.85 + }, + "C11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 51, + "z": 1.85 + }, + "C12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 51, + "z": 1.85 + }, + "C2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 51, + "z": 1.85 + }, + "C3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 51, + "z": 1.85 + }, + "C4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 51, + "z": 1.85 + }, + "C5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 51, + "z": 1.85 + }, + "C6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 51, + "z": 1.85 + }, + "C7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 51, + "z": 1.85 + }, + "C8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 51, + "z": 1.85 + }, + "C9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 51, + "z": 1.85 + }, + "D1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 42, + "z": 1.85 + }, + "D10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 42, + "z": 1.85 + }, + "D11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 42, + "z": 1.85 + }, + "D12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 42, + "z": 1.85 + }, + "D2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 42, + "z": 1.85 + }, + "D3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 42, + "z": 1.85 + }, + "D4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 42, + "z": 1.85 + }, + "D5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 42, + "z": 1.85 + }, + "D6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 42, + "z": 1.85 + }, + "D7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 42, + "z": 1.85 + }, + "D8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 42, + "z": 1.85 + }, + "D9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 42, + "z": 1.85 + }, + "E1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 33, + "z": 1.85 + }, + "E10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 33, + "z": 1.85 + }, + "E11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 33, + "z": 1.85 + }, + "E12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 33, + "z": 1.85 + }, + "E2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 33, + "z": 1.85 + }, + "E3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 33, + "z": 1.85 + }, + "E4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 33, + "z": 1.85 + }, + "E5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 33, + "z": 1.85 + }, + "E6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 33, + "z": 1.85 + }, + "E7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 33, + "z": 1.85 + }, + "E8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 33, + "z": 1.85 + }, + "E9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 33, + "z": 1.85 + }, + "F1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 24, + "z": 1.85 + }, + "F10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 24, + "z": 1.85 + }, + "F11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 24, + "z": 1.85 + }, + "F12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 24, + "z": 1.85 + }, + "F2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 24, + "z": 1.85 + }, + "F3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 24, + "z": 1.85 + }, + "F4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 24, + "z": 1.85 + }, + "F5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 24, + "z": 1.85 + }, + "F6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 24, + "z": 1.85 + }, + "F7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 24, + "z": 1.85 + }, + "F8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 24, + "z": 1.85 + }, + "F9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 24, + "z": 1.85 + }, + "G1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 15, + "z": 1.85 + }, + "G10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 15, + "z": 1.85 + }, + "G11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 15, + "z": 1.85 + }, + "G12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 15, + "z": 1.85 + }, + "G2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 15, + "z": 1.85 + }, + "G3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 15, + "z": 1.85 + }, + "G4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 15, + "z": 1.85 + }, + "G5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 15, + "z": 1.85 + }, + "G6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 15, + "z": 1.85 + }, + "G7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 15, + "z": 1.85 + }, + "G8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 15, + "z": 1.85 + }, + "G9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 15, + "z": 1.85 + }, + "H1": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 6, + "y": 6, + "z": 1.85 + }, + "H10": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 87, + "y": 6, + "z": 1.85 + }, + "H11": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 96, + "y": 6, + "z": 1.85 + }, + "H12": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 105, + "y": 6, + "z": 1.85 + }, + "H2": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 15, + "y": 6, + "z": 1.85 + }, + "H3": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 24, + "y": 6, + "z": 1.85 + }, + "H4": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 33, + "y": 6, + "z": 1.85 + }, + "H5": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 42, + "y": 6, + "z": 1.85 + }, + "H6": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 51, + "y": 6, + "z": 1.85 + }, + "H7": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 60, + "y": 6, + "z": 1.85 + }, + "H8": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 69, + "y": 6, + "z": 1.85 + }, + "H9": { + "depth": 12, + "diameter": 5.64, + "shape": "circular", + "totalLiquidVolume": 0, + "x": 78, + "y": 6, + "z": 1.85 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b9260591301549a7e5b967fca505245c", + "notes": [], + "params": { + "loadName": "nest_1_reservoir_290ml", + "location": { + "slotName": "D2" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "NEST", + "brandId": [ + "360206", + "360266" + ], + "links": [ + "https://www.nest-biotech.com/reagent-reserviors" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.47, + "zDimension": 44.4 + }, + "gripperOffsets": {}, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1" + ] + } + ], + "metadata": { + "displayCategory": "reservoir", + "displayName": "NEST 1 Well Reservoir 290 mL", + "displayVolumeUnits": "mL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1" + ] + ], + "parameters": { + "format": "trough", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "nest_1_reservoir_290ml", + "quirks": [ + "centerMultichannelOnWells", + "touchTipDisabled" + ] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 39.55, + "shape": "rectangular", + "totalLiquidVolume": 290000, + "x": 63.88, + "xDimension": 106.8, + "y": 42.74, + "yDimension": 71.2, + "z": 4.85 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ab92268f4b504e3234fad5d7d617170f", + "notes": [], + "params": { + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": { + "slotName": "C2" + }, + "namespace": "opentrons", + "version": 2 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "NEST", + "brandId": [ + "402501" + ], + "links": [ + "https://www.nest-biotech.com/pcr-plates/58773587.html" + ] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 15.7 + }, + "gripForce": 15.0, + "gripHeightFromLabwareBottom": 10.65, + "gripperOffsets": {}, + "groups": [ + { + "metadata": { + "wellBottomShape": "v" + }, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "wellPlate", + "displayName": "NEST 96 Well Plate 100 µL PCR Full Skirt", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": true, + "isTiprack": false, + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "magneticModuleEngageHeight": 20 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_96_pcr_adapter": { + "x": 0, + "y": 0, + "z": 10.2 + }, + "opentrons_96_well_aluminum_block": { + "x": 0, + "y": 0, + "z": 12.66 + } + }, + "stackingOffsetWithModule": { + "thermocyclerModuleV2": { + "x": 0, + "y": 0, + "z": 10.8 + } + }, + "version": 2, + "wells": { + "A1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 74.24, + "z": 0.92 + }, + "A10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 74.24, + "z": 0.92 + }, + "A11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 74.24, + "z": 0.92 + }, + "A12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 74.24, + "z": 0.92 + }, + "A2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 74.24, + "z": 0.92 + }, + "A3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 74.24, + "z": 0.92 + }, + "A4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 74.24, + "z": 0.92 + }, + "A5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 74.24, + "z": 0.92 + }, + "A6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 74.24, + "z": 0.92 + }, + "A7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 74.24, + "z": 0.92 + }, + "A8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 74.24, + "z": 0.92 + }, + "A9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 74.24, + "z": 0.92 + }, + "B1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 65.24, + "z": 0.92 + }, + "B10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 65.24, + "z": 0.92 + }, + "B11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 65.24, + "z": 0.92 + }, + "B12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 65.24, + "z": 0.92 + }, + "B2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 65.24, + "z": 0.92 + }, + "B3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 65.24, + "z": 0.92 + }, + "B4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 65.24, + "z": 0.92 + }, + "B5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 65.24, + "z": 0.92 + }, + "B6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 65.24, + "z": 0.92 + }, + "B7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 65.24, + "z": 0.92 + }, + "B8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 65.24, + "z": 0.92 + }, + "B9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 65.24, + "z": 0.92 + }, + "C1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 56.24, + "z": 0.92 + }, + "C10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 56.24, + "z": 0.92 + }, + "C11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 56.24, + "z": 0.92 + }, + "C12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 56.24, + "z": 0.92 + }, + "C2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 56.24, + "z": 0.92 + }, + "C3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 56.24, + "z": 0.92 + }, + "C4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 56.24, + "z": 0.92 + }, + "C5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 56.24, + "z": 0.92 + }, + "C6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 56.24, + "z": 0.92 + }, + "C7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 56.24, + "z": 0.92 + }, + "C8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 56.24, + "z": 0.92 + }, + "C9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 56.24, + "z": 0.92 + }, + "D1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 47.24, + "z": 0.92 + }, + "D10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 47.24, + "z": 0.92 + }, + "D11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 47.24, + "z": 0.92 + }, + "D12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 47.24, + "z": 0.92 + }, + "D2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 47.24, + "z": 0.92 + }, + "D3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 47.24, + "z": 0.92 + }, + "D4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 47.24, + "z": 0.92 + }, + "D5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 47.24, + "z": 0.92 + }, + "D6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 47.24, + "z": 0.92 + }, + "D7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 47.24, + "z": 0.92 + }, + "D8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 47.24, + "z": 0.92 + }, + "D9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 47.24, + "z": 0.92 + }, + "E1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 38.24, + "z": 0.92 + }, + "E10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 38.24, + "z": 0.92 + }, + "E11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 38.24, + "z": 0.92 + }, + "E12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 38.24, + "z": 0.92 + }, + "E2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 38.24, + "z": 0.92 + }, + "E3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 38.24, + "z": 0.92 + }, + "E4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 38.24, + "z": 0.92 + }, + "E5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 38.24, + "z": 0.92 + }, + "E6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 38.24, + "z": 0.92 + }, + "E7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 38.24, + "z": 0.92 + }, + "E8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 38.24, + "z": 0.92 + }, + "E9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 38.24, + "z": 0.92 + }, + "F1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 29.24, + "z": 0.92 + }, + "F10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 29.24, + "z": 0.92 + }, + "F11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 29.24, + "z": 0.92 + }, + "F12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 29.24, + "z": 0.92 + }, + "F2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 29.24, + "z": 0.92 + }, + "F3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 29.24, + "z": 0.92 + }, + "F4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 29.24, + "z": 0.92 + }, + "F5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 29.24, + "z": 0.92 + }, + "F6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 29.24, + "z": 0.92 + }, + "F7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 29.24, + "z": 0.92 + }, + "F8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 29.24, + "z": 0.92 + }, + "F9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 29.24, + "z": 0.92 + }, + "G1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 20.24, + "z": 0.92 + }, + "G10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 20.24, + "z": 0.92 + }, + "G11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 20.24, + "z": 0.92 + }, + "G12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 20.24, + "z": 0.92 + }, + "G2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 20.24, + "z": 0.92 + }, + "G3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 20.24, + "z": 0.92 + }, + "G4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 20.24, + "z": 0.92 + }, + "G5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 20.24, + "z": 0.92 + }, + "G6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 20.24, + "z": 0.92 + }, + "G7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 20.24, + "z": 0.92 + }, + "G8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 20.24, + "z": 0.92 + }, + "G9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 20.24, + "z": 0.92 + }, + "H1": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 14.38, + "y": 11.24, + "z": 0.92 + }, + "H10": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 95.38, + "y": 11.24, + "z": 0.92 + }, + "H11": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 104.38, + "y": 11.24, + "z": 0.92 + }, + "H12": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 113.38, + "y": 11.24, + "z": 0.92 + }, + "H2": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 23.38, + "y": 11.24, + "z": 0.92 + }, + "H3": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 32.38, + "y": 11.24, + "z": 0.92 + }, + "H4": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 41.38, + "y": 11.24, + "z": 0.92 + }, + "H5": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 50.38, + "y": 11.24, + "z": 0.92 + }, + "H6": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 59.38, + "y": 11.24, + "z": 0.92 + }, + "H7": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 68.38, + "y": 11.24, + "z": 0.92 + }, + "H8": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 77.38, + "y": 11.24, + "z": 0.92 + }, + "H9": { + "depth": 14.78, + "diameter": 5.34, + "shape": "circular", + "totalLiquidVolume": 100, + "x": 86.38, + "y": 11.24, + "z": 0.92 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b78a4996a5952413f60b9207cb19867d", + "notes": [], + "params": { + "loadName": "opentrons_flex_96_tiprack_adapter", + "location": { + "slotName": "A2" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [ + "adapter" + ], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": -14.25, + "y": -3.5, + "z": 0 + }, + "dimensions": { + "xDimension": 156.5, + "yDimension": 93, + "zDimension": 132 + }, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [] + } + ], + "metadata": { + "displayCategory": "adapter", + "displayName": "Opentrons Flex 96 Tip Rack Adapter", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": false, + "loadName": "opentrons_flex_96_tiprack_adapter", + "quirks": [ + "tiprackAdapterFor96Channel" + ] + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": {}, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": {} + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5651f92e0b3fa3005baf4698ab8d0591", + "notes": [], + "params": { + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": { + "labwareId": "UUID" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.75, + "yDimension": 85.75, + "zDimension": 99 + }, + "gripForce": 16.0, + "gripHeightFromLabwareBottom": 23.9, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "tipRack", + "displayName": "Opentrons Flex 96 Tip Rack 1000 µL", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": true, + "loadName": "opentrons_flex_96_tiprack_1000ul", + "quirks": [], + "tipLength": 95.6, + "tipOverlap": 10.5 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_flex_96_tiprack_adapter": { + "x": 0, + "y": 0, + "z": 121 + } + }, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 74.38, + "z": 1.5 + }, + "A10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 74.38, + "z": 1.5 + }, + "A11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 74.38, + "z": 1.5 + }, + "A12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 74.38, + "z": 1.5 + }, + "A2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 74.38, + "z": 1.5 + }, + "A3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 74.38, + "z": 1.5 + }, + "A4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 74.38, + "z": 1.5 + }, + "A5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 74.38, + "z": 1.5 + }, + "A6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 74.38, + "z": 1.5 + }, + "A7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 74.38, + "z": 1.5 + }, + "A8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 74.38, + "z": 1.5 + }, + "A9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 74.38, + "z": 1.5 + }, + "B1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 65.38, + "z": 1.5 + }, + "B10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 65.38, + "z": 1.5 + }, + "B11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 65.38, + "z": 1.5 + }, + "B12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 65.38, + "z": 1.5 + }, + "B2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 65.38, + "z": 1.5 + }, + "B3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 65.38, + "z": 1.5 + }, + "B4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 65.38, + "z": 1.5 + }, + "B5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 65.38, + "z": 1.5 + }, + "B6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 65.38, + "z": 1.5 + }, + "B7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 65.38, + "z": 1.5 + }, + "B8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 65.38, + "z": 1.5 + }, + "B9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 65.38, + "z": 1.5 + }, + "C1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 56.38, + "z": 1.5 + }, + "C10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 56.38, + "z": 1.5 + }, + "C11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 56.38, + "z": 1.5 + }, + "C12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 56.38, + "z": 1.5 + }, + "C2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 56.38, + "z": 1.5 + }, + "C3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 56.38, + "z": 1.5 + }, + "C4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 56.38, + "z": 1.5 + }, + "C5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 56.38, + "z": 1.5 + }, + "C6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 56.38, + "z": 1.5 + }, + "C7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 56.38, + "z": 1.5 + }, + "C8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 56.38, + "z": 1.5 + }, + "C9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 56.38, + "z": 1.5 + }, + "D1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 47.38, + "z": 1.5 + }, + "D10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 47.38, + "z": 1.5 + }, + "D11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 47.38, + "z": 1.5 + }, + "D12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 47.38, + "z": 1.5 + }, + "D2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 47.38, + "z": 1.5 + }, + "D3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 47.38, + "z": 1.5 + }, + "D4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 47.38, + "z": 1.5 + }, + "D5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 47.38, + "z": 1.5 + }, + "D6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 47.38, + "z": 1.5 + }, + "D7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 47.38, + "z": 1.5 + }, + "D8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 47.38, + "z": 1.5 + }, + "D9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 47.38, + "z": 1.5 + }, + "E1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 38.38, + "z": 1.5 + }, + "E10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 38.38, + "z": 1.5 + }, + "E11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 38.38, + "z": 1.5 + }, + "E12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 38.38, + "z": 1.5 + }, + "E2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 38.38, + "z": 1.5 + }, + "E3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 38.38, + "z": 1.5 + }, + "E4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 38.38, + "z": 1.5 + }, + "E5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 38.38, + "z": 1.5 + }, + "E6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 38.38, + "z": 1.5 + }, + "E7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 38.38, + "z": 1.5 + }, + "E8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 38.38, + "z": 1.5 + }, + "E9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 38.38, + "z": 1.5 + }, + "F1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 29.38, + "z": 1.5 + }, + "F10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 29.38, + "z": 1.5 + }, + "F11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 29.38, + "z": 1.5 + }, + "F12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 29.38, + "z": 1.5 + }, + "F2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 29.38, + "z": 1.5 + }, + "F3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 29.38, + "z": 1.5 + }, + "F4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 29.38, + "z": 1.5 + }, + "F5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 29.38, + "z": 1.5 + }, + "F6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 29.38, + "z": 1.5 + }, + "F7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 29.38, + "z": 1.5 + }, + "F8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 29.38, + "z": 1.5 + }, + "F9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 29.38, + "z": 1.5 + }, + "G1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 20.38, + "z": 1.5 + }, + "G10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 20.38, + "z": 1.5 + }, + "G11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 20.38, + "z": 1.5 + }, + "G12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 20.38, + "z": 1.5 + }, + "G2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 20.38, + "z": 1.5 + }, + "G3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 20.38, + "z": 1.5 + }, + "G4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 20.38, + "z": 1.5 + }, + "G5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 20.38, + "z": 1.5 + }, + "G6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 20.38, + "z": 1.5 + }, + "G7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 20.38, + "z": 1.5 + }, + "G8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 20.38, + "z": 1.5 + }, + "G9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 20.38, + "z": 1.5 + }, + "H1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 11.38, + "z": 1.5 + }, + "H10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 11.38, + "z": 1.5 + }, + "H11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 11.38, + "z": 1.5 + }, + "H12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 11.38, + "z": 1.5 + }, + "H2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 11.38, + "z": 1.5 + }, + "H3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 11.38, + "z": 1.5 + }, + "H4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 11.38, + "z": 1.5 + }, + "H5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 11.38, + "z": 1.5 + }, + "H6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 11.38, + "z": 1.5 + }, + "H7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 11.38, + "z": 1.5 + }, + "H8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 11.38, + "z": 1.5 + }, + "H9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 11.38, + "z": 1.5 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a7aa955a3b241ebff1821c35e8eb6bdf", + "notes": [], + "params": { + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": { + "slotName": "C3" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.75, + "yDimension": 85.75, + "zDimension": 99 + }, + "gripForce": 16.0, + "gripHeightFromLabwareBottom": 23.9, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "tipRack", + "displayName": "Opentrons Flex 96 Tip Rack 1000 µL", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": true, + "loadName": "opentrons_flex_96_tiprack_1000ul", + "quirks": [], + "tipLength": 95.6, + "tipOverlap": 10.5 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_flex_96_tiprack_adapter": { + "x": 0, + "y": 0, + "z": 121 + } + }, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 74.38, + "z": 1.5 + }, + "A10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 74.38, + "z": 1.5 + }, + "A11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 74.38, + "z": 1.5 + }, + "A12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 74.38, + "z": 1.5 + }, + "A2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 74.38, + "z": 1.5 + }, + "A3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 74.38, + "z": 1.5 + }, + "A4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 74.38, + "z": 1.5 + }, + "A5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 74.38, + "z": 1.5 + }, + "A6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 74.38, + "z": 1.5 + }, + "A7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 74.38, + "z": 1.5 + }, + "A8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 74.38, + "z": 1.5 + }, + "A9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 74.38, + "z": 1.5 + }, + "B1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 65.38, + "z": 1.5 + }, + "B10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 65.38, + "z": 1.5 + }, + "B11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 65.38, + "z": 1.5 + }, + "B12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 65.38, + "z": 1.5 + }, + "B2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 65.38, + "z": 1.5 + }, + "B3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 65.38, + "z": 1.5 + }, + "B4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 65.38, + "z": 1.5 + }, + "B5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 65.38, + "z": 1.5 + }, + "B6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 65.38, + "z": 1.5 + }, + "B7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 65.38, + "z": 1.5 + }, + "B8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 65.38, + "z": 1.5 + }, + "B9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 65.38, + "z": 1.5 + }, + "C1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 56.38, + "z": 1.5 + }, + "C10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 56.38, + "z": 1.5 + }, + "C11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 56.38, + "z": 1.5 + }, + "C12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 56.38, + "z": 1.5 + }, + "C2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 56.38, + "z": 1.5 + }, + "C3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 56.38, + "z": 1.5 + }, + "C4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 56.38, + "z": 1.5 + }, + "C5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 56.38, + "z": 1.5 + }, + "C6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 56.38, + "z": 1.5 + }, + "C7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 56.38, + "z": 1.5 + }, + "C8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 56.38, + "z": 1.5 + }, + "C9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 56.38, + "z": 1.5 + }, + "D1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 47.38, + "z": 1.5 + }, + "D10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 47.38, + "z": 1.5 + }, + "D11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 47.38, + "z": 1.5 + }, + "D12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 47.38, + "z": 1.5 + }, + "D2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 47.38, + "z": 1.5 + }, + "D3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 47.38, + "z": 1.5 + }, + "D4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 47.38, + "z": 1.5 + }, + "D5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 47.38, + "z": 1.5 + }, + "D6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 47.38, + "z": 1.5 + }, + "D7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 47.38, + "z": 1.5 + }, + "D8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 47.38, + "z": 1.5 + }, + "D9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 47.38, + "z": 1.5 + }, + "E1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 38.38, + "z": 1.5 + }, + "E10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 38.38, + "z": 1.5 + }, + "E11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 38.38, + "z": 1.5 + }, + "E12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 38.38, + "z": 1.5 + }, + "E2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 38.38, + "z": 1.5 + }, + "E3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 38.38, + "z": 1.5 + }, + "E4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 38.38, + "z": 1.5 + }, + "E5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 38.38, + "z": 1.5 + }, + "E6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 38.38, + "z": 1.5 + }, + "E7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 38.38, + "z": 1.5 + }, + "E8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 38.38, + "z": 1.5 + }, + "E9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 38.38, + "z": 1.5 + }, + "F1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 29.38, + "z": 1.5 + }, + "F10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 29.38, + "z": 1.5 + }, + "F11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 29.38, + "z": 1.5 + }, + "F12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 29.38, + "z": 1.5 + }, + "F2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 29.38, + "z": 1.5 + }, + "F3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 29.38, + "z": 1.5 + }, + "F4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 29.38, + "z": 1.5 + }, + "F5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 29.38, + "z": 1.5 + }, + "F6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 29.38, + "z": 1.5 + }, + "F7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 29.38, + "z": 1.5 + }, + "F8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 29.38, + "z": 1.5 + }, + "F9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 29.38, + "z": 1.5 + }, + "G1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 20.38, + "z": 1.5 + }, + "G10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 20.38, + "z": 1.5 + }, + "G11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 20.38, + "z": 1.5 + }, + "G12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 20.38, + "z": 1.5 + }, + "G2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 20.38, + "z": 1.5 + }, + "G3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 20.38, + "z": 1.5 + }, + "G4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 20.38, + "z": 1.5 + }, + "G5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 20.38, + "z": 1.5 + }, + "G6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 20.38, + "z": 1.5 + }, + "G7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 20.38, + "z": 1.5 + }, + "G8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 20.38, + "z": 1.5 + }, + "G9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 20.38, + "z": 1.5 + }, + "H1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 11.38, + "z": 1.5 + }, + "H10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 11.38, + "z": 1.5 + }, + "H11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 11.38, + "z": 1.5 + }, + "H12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 11.38, + "z": 1.5 + }, + "H2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 11.38, + "z": 1.5 + }, + "H3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 11.38, + "z": 1.5 + }, + "H4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 11.38, + "z": 1.5 + }, + "H5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 11.38, + "z": 1.5 + }, + "H6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 11.38, + "z": 1.5 + }, + "H7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 11.38, + "z": 1.5 + }, + "H8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 11.38, + "z": 1.5 + }, + "H9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 11.38, + "z": 1.5 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "393a8041bc4a5079a30a7e2e0b416d0b", + "notes": [], + "params": { + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": { + "addressableAreaName": "C4" + }, + "namespace": "opentrons", + "version": 1 + }, + "result": { + "definition": { + "allowedRoles": [], + "brand": { + "brand": "Opentrons", + "brandId": [] + }, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + }, + "dimensions": { + "xDimension": 127.75, + "yDimension": 85.75, + "zDimension": 99 + }, + "gripForce": 16.0, + "gripHeightFromLabwareBottom": 23.9, + "gripperOffsets": {}, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "A10", + "A11", + "A12", + "A2", + "A3", + "A4", + "A5", + "A6", + "A7", + "A8", + "A9", + "B1", + "B10", + "B11", + "B12", + "B2", + "B3", + "B4", + "B5", + "B6", + "B7", + "B8", + "B9", + "C1", + "C10", + "C11", + "C12", + "C2", + "C3", + "C4", + "C5", + "C6", + "C7", + "C8", + "C9", + "D1", + "D10", + "D11", + "D12", + "D2", + "D3", + "D4", + "D5", + "D6", + "D7", + "D8", + "D9", + "E1", + "E10", + "E11", + "E12", + "E2", + "E3", + "E4", + "E5", + "E6", + "E7", + "E8", + "E9", + "F1", + "F10", + "F11", + "F12", + "F2", + "F3", + "F4", + "F5", + "F6", + "F7", + "F8", + "F9", + "G1", + "G10", + "G11", + "G12", + "G2", + "G3", + "G4", + "G5", + "G6", + "G7", + "G8", + "G9", + "H1", + "H10", + "H11", + "H12", + "H2", + "H3", + "H4", + "H5", + "H6", + "H7", + "H8", + "H9" + ] + } + ], + "metadata": { + "displayCategory": "tipRack", + "displayName": "Opentrons Flex 96 Tip Rack 1000 µL", + "displayVolumeUnits": "µL", + "tags": [] + }, + "namespace": "opentrons", + "ordering": [ + [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1" + ], + [ + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10" + ], + [ + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11" + ], + [ + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ], + [ + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2" + ], + [ + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3" + ], + [ + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4" + ], + [ + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5" + ], + [ + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6" + ], + [ + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7" + ], + [ + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8" + ], + [ + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9" + ] + ], + "parameters": { + "format": "96Standard", + "isMagneticModuleCompatible": false, + "isTiprack": true, + "loadName": "opentrons_flex_96_tiprack_1000ul", + "quirks": [], + "tipLength": 95.6, + "tipOverlap": 10.5 + }, + "schemaVersion": 2, + "stackingOffsetWithLabware": { + "opentrons_flex_96_tiprack_adapter": { + "x": 0, + "y": 0, + "z": 121 + } + }, + "stackingOffsetWithModule": {}, + "version": 1, + "wells": { + "A1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 74.38, + "z": 1.5 + }, + "A10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 74.38, + "z": 1.5 + }, + "A11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 74.38, + "z": 1.5 + }, + "A12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 74.38, + "z": 1.5 + }, + "A2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 74.38, + "z": 1.5 + }, + "A3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 74.38, + "z": 1.5 + }, + "A4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 74.38, + "z": 1.5 + }, + "A5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 74.38, + "z": 1.5 + }, + "A6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 74.38, + "z": 1.5 + }, + "A7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 74.38, + "z": 1.5 + }, + "A8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 74.38, + "z": 1.5 + }, + "A9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 74.38, + "z": 1.5 + }, + "B1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 65.38, + "z": 1.5 + }, + "B10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 65.38, + "z": 1.5 + }, + "B11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 65.38, + "z": 1.5 + }, + "B12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 65.38, + "z": 1.5 + }, + "B2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 65.38, + "z": 1.5 + }, + "B3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 65.38, + "z": 1.5 + }, + "B4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 65.38, + "z": 1.5 + }, + "B5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 65.38, + "z": 1.5 + }, + "B6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 65.38, + "z": 1.5 + }, + "B7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 65.38, + "z": 1.5 + }, + "B8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 65.38, + "z": 1.5 + }, + "B9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 65.38, + "z": 1.5 + }, + "C1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 56.38, + "z": 1.5 + }, + "C10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 56.38, + "z": 1.5 + }, + "C11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 56.38, + "z": 1.5 + }, + "C12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 56.38, + "z": 1.5 + }, + "C2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 56.38, + "z": 1.5 + }, + "C3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 56.38, + "z": 1.5 + }, + "C4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 56.38, + "z": 1.5 + }, + "C5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 56.38, + "z": 1.5 + }, + "C6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 56.38, + "z": 1.5 + }, + "C7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 56.38, + "z": 1.5 + }, + "C8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 56.38, + "z": 1.5 + }, + "C9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 56.38, + "z": 1.5 + }, + "D1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 47.38, + "z": 1.5 + }, + "D10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 47.38, + "z": 1.5 + }, + "D11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 47.38, + "z": 1.5 + }, + "D12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 47.38, + "z": 1.5 + }, + "D2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 47.38, + "z": 1.5 + }, + "D3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 47.38, + "z": 1.5 + }, + "D4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 47.38, + "z": 1.5 + }, + "D5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 47.38, + "z": 1.5 + }, + "D6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 47.38, + "z": 1.5 + }, + "D7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 47.38, + "z": 1.5 + }, + "D8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 47.38, + "z": 1.5 + }, + "D9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 47.38, + "z": 1.5 + }, + "E1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 38.38, + "z": 1.5 + }, + "E10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 38.38, + "z": 1.5 + }, + "E11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 38.38, + "z": 1.5 + }, + "E12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 38.38, + "z": 1.5 + }, + "E2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 38.38, + "z": 1.5 + }, + "E3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 38.38, + "z": 1.5 + }, + "E4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 38.38, + "z": 1.5 + }, + "E5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 38.38, + "z": 1.5 + }, + "E6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 38.38, + "z": 1.5 + }, + "E7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 38.38, + "z": 1.5 + }, + "E8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 38.38, + "z": 1.5 + }, + "E9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 38.38, + "z": 1.5 + }, + "F1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 29.38, + "z": 1.5 + }, + "F10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 29.38, + "z": 1.5 + }, + "F11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 29.38, + "z": 1.5 + }, + "F12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 29.38, + "z": 1.5 + }, + "F2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 29.38, + "z": 1.5 + }, + "F3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 29.38, + "z": 1.5 + }, + "F4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 29.38, + "z": 1.5 + }, + "F5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 29.38, + "z": 1.5 + }, + "F6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 29.38, + "z": 1.5 + }, + "F7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 29.38, + "z": 1.5 + }, + "F8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 29.38, + "z": 1.5 + }, + "F9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 29.38, + "z": 1.5 + }, + "G1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 20.38, + "z": 1.5 + }, + "G10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 20.38, + "z": 1.5 + }, + "G11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 20.38, + "z": 1.5 + }, + "G12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 20.38, + "z": 1.5 + }, + "G2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 20.38, + "z": 1.5 + }, + "G3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 20.38, + "z": 1.5 + }, + "G4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 20.38, + "z": 1.5 + }, + "G5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 20.38, + "z": 1.5 + }, + "G6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 20.38, + "z": 1.5 + }, + "G7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 20.38, + "z": 1.5 + }, + "G8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 20.38, + "z": 1.5 + }, + "G9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 20.38, + "z": 1.5 + }, + "H1": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 14.38, + "y": 11.38, + "z": 1.5 + }, + "H10": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 95.38, + "y": 11.38, + "z": 1.5 + }, + "H11": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 104.38, + "y": 11.38, + "z": 1.5 + }, + "H12": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 113.38, + "y": 11.38, + "z": 1.5 + }, + "H2": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 23.38, + "y": 11.38, + "z": 1.5 + }, + "H3": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 32.38, + "y": 11.38, + "z": 1.5 + }, + "H4": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 41.38, + "y": 11.38, + "z": 1.5 + }, + "H5": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 50.38, + "y": 11.38, + "z": 1.5 + }, + "H6": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 59.38, + "y": 11.38, + "z": 1.5 + }, + "H7": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 68.38, + "y": 11.38, + "z": 1.5 + }, + "H8": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 77.38, + "y": 11.38, + "z": 1.5 + }, + "H9": { + "depth": 97.5, + "diameter": 5.47, + "shape": "circular", + "totalLiquidVolume": 1000, + "x": 86.38, + "y": 11.38, + "z": 1.5 + } + } + }, + "labwareId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadPipette", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d37878e4a76e56d1c9f78661c19fc268", + "notes": [], + "params": { + "liquidPresenceDetection": false, + "mount": "left", + "pipetteName": "p1000_96", + "tipOverlapNotAfterVersion": "v1" + }, + "result": { + "pipetteId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "loadLiquid", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ed3eae81517739fb8732a8c33660a711", + "notes": [], + "params": { + "labwareId": "UUID", + "liquidId": "UUID", + "volumeByWell": { + "A1": 29000.0 + } + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "configureNozzleLayout", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2d531312465be507684de1d74b786e5d", + "notes": [], + "params": { + "configurationParams": { + "primaryNozzle": "A12", + "style": "COLUMN" + }, + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "71b4dacb427e6f6bffccd915d013e8e4", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 342.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7e5f17ae8d0f29f168e5912bde10adc6", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "fde6a3018b3952d6b3e5038eb0da9842", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "737d00e8af88f6106a69291b3774a334", + "notes": [], + "params": { + "message": "Watch this next tip drop in the waste chute. We are going to compare it against the next drop in the waste chute." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableArea", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "136ab01b35f7645925a5a772a2cf1b5e", + "notes": [], + "params": { + "addressableAreaName": "96ChannelWasteChute", + "forceDirect": false, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID", + "stayAtHighestPossibleZ": false + }, + "result": { + "position": { + "x": 391.945, + "y": 10.585, + "z": 114.5 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "140b261e88c4c6dfb7bb22243f05e1c3", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "529bb1ff92101ceebf086866dce08749", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A2" + }, + "result": { + "position": { + "x": 351.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "74a5d3594180c7914c947ce8772dc467", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5fe5c3ef924f24957267740883f451dc", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A2" + }, + "result": { + "position": { + "x": 187.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d78e3169a284434442095668a85c896e", + "notes": [], + "params": { + "message": "Watch this next tip drop in the waste chute. It should drop in a different location than the previous drop." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2153b23b18fc0912f28e60ebd44bd19f", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": true, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 466.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4be843dfe95cbc9c51d3d9a143c230af", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d1d38afd5ca73bcaa38bc5517ae9aeca", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A3" + }, + "result": { + "position": { + "x": 360.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "acbb4946ad49f071c843d6c0ec8d400d", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d80d56bfb4ae6bed1079191e77332bbc", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A3" + }, + "result": { + "position": { + "x": 196.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "31bd451e64123dcbdaaaae7f8cdd203c", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ef24273115ed0d7a3b9082948368e5d9", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "997094b10367e3b4c54342ca6709d511", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A4" + }, + "result": { + "position": { + "x": 369.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7a3f9efb6bf02bd6e268a349a6a8aa2e", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b97a854748fee541d65c3fbaaed79ffc", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A4" + }, + "result": { + "position": { + "x": 205.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "995bbfbdf9462fae9e633f300f75f3f3", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9fb2a736746160f6ddbdccf215e716c1", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "838336a67eab9798fb8f8b4c11f1f498", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A5" + }, + "result": { + "position": { + "x": 378.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "0287a4dd7c406290aef242da06bd93cc", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c819578ece99ddbe3105450a14fe38dc", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A5" + }, + "result": { + "position": { + "x": 214.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4c61b9776d93f178e8817aed0a3d4113", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9adfd7ad1927d998c32d896d3b3a4c0c", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5b407a4561742d53f4764f1ee080355c", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A6" + }, + "result": { + "position": { + "x": 387.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3fe7e3b2cd692cd018963854ad46f636", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "e098701f0d883d66e7f0043291a8a7ea", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A6" + }, + "result": { + "position": { + "x": 223.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4ed1139e9fa176861436700eb549e63c", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7a0706a6752115866aea9c74a3073c03", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "cda957cc63b8bca863d0e6b1b18274ee", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A7" + }, + "result": { + "position": { + "x": 396.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a37858df7638086f7b13397a95924418", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "cb198e468644948976e46996738ee070", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A7" + }, + "result": { + "position": { + "x": 232.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "47cdb4266a0f3992c3be0bce84d91ace", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1901ecc7b664d90a81e9fdd95eabeaf5", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "03ec35758a340692e9910e8d4bbde18e", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A8" + }, + "result": { + "position": { + "x": 405.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "67ed9e8092a71a19e5ad3506b3b82045", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "32d6f6f302080d0737d35146f9d7962e", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A8" + }, + "result": { + "position": { + "x": 241.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a95dd8f3e9ba2ac4089909bf6414ebf5", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ecf1419bcf9906e98e6405262d3debfc", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "36a56828b2f81b4ccfcfebe423a4b3fa", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A9" + }, + "result": { + "position": { + "x": 414.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a12f4438255de6a8902274ad1a62a921", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "582afe7bd1564a6e9f6d6b4607b95104", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A9" + }, + "result": { + "position": { + "x": 250.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7b335fd958008f5947a4dcc1a0bf4f19", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3c48219dcc5f23489c590efb076dd94c", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a64fd253c1e0891dbf266e46a1bcb424", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A10" + }, + "result": { + "position": { + "x": 423.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "97b1b7da3cb8a586c4ab184838e09292", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9eb61c9387bd891c204e6a4061479c0e", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A10" + }, + "result": { + "position": { + "x": 259.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9cff67de46a45cb055d50947afcdecb4", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d201058c88cd48141b21c6d259571857", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "0ac0a4a2aa3d84d895c2f83ff38acdaf", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A11" + }, + "result": { + "position": { + "x": 432.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "29a279105f69a2309d73058e3d4983ef", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "11cb9af5ede3f0642861520e7621dcb5", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A11" + }, + "result": { + "position": { + "x": 268.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "930758db855014596bfa19b1c1529bd1", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1e3120f6ad4fc16b39c06a47c4a1611c", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "e544dd10d11033ab0d4623b8d9f9e512", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A12" + }, + "result": { + "position": { + "x": 441.38, + "y": 181.38, + "z": 99.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "39010129d0b58c303ab66fc296688790", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5bac9325bb5f7bbec4e4404173de46ce", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A12" + }, + "result": { + "position": { + "x": 277.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f0a9749b2ac87cc6af972d9812f4f706", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTipInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f1131c7320aa73be4a3056cc5f490d22", + "notes": [], + "params": { + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a7ff27922283214930ea97630e68a1df", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": "offDeck", + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "configureNozzleLayout", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "fb78431e9807e7ad279d9e3aacc25321", + "notes": [], + "params": { + "configurationParams": { + "style": "ALL" + }, + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3fad7505f312db97b533fc37f1c4fcfc", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 110.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "41ee13ca47528f8a69a38241ad62ee92", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 10.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 10.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f752cea04bc326b8f91d05dbe24cf477", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 29.999999999999993 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 74.39999999999998 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5fe884095744c9b86e6b89273bcdff32", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 990.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 29.999999999999993 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 74.39999999999998 + }, + "volume": 990.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "39eb47551939017fd9c2c61536e8965d", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 10.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 10.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableArea", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6ba70f1e65749b3ce1cfb55a5f52a6af", + "notes": [], + "params": { + "addressableAreaName": "96ChannelWasteChute", + "forceDirect": false, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID", + "stayAtHighestPossibleZ": false + }, + "result": { + "position": { + "x": 391.945, + "y": 10.585, + "z": 114.5 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "blowOutInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5b9de7703b5060c5ad36828e1c2af497", + "notes": [], + "params": { + "flowRate": 80.0, + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2203608fa4f4f82959f4a0b5d4e82be7", + "notes": [], + "params": { + "alternateDropLocation": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0, + "y": 0, + "z": 0 + }, + "origin": "default" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 90.88 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "fbf547207d3091d7e144257ca660187f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": "offDeck", + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b3d15d6b27ee21f5f94c28b6af98e449", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "fe2dcdd1096d8f10bc6f7c1e08afba32", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 110.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6f454d64b916f2622b6eaeef969359c8", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "45771a871fe661d32db5ec1728ec253e", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "42367b3d46709d62f72cc29a5e44c7a2", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "8fe076a253930698bb48c1ae02d5df41", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "086b29b955fc444e72abfbabd0864bba", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "313f01a8a81c4c32484ed2529618c09f", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 5.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 5.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "dfcfd66f05b9efe9c398f8235700f5bf", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 10.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -38.55 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 5.85 + }, + "volume": 10.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "88c96216497a2360391135c7f6df743f", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 10.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 10.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f45d0387f245571d753d4d2698398053", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "297958a056744a17134d3ac02184a893", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "83abe3ff7cd4e66bd1619ffe7d86514e", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1b804e4e3057f049715c4e876fbdab1a", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d583be183d5d0252c49840aa4c5f4e17", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ebf7b91a638db36c212c7816de35d62a", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "074950bd6544c5682e96e6266457e3e7", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ff2d7c53a987593d9e40fd7359029243", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "pushOut": 0.0, + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "aspirate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "0c23c9fc9b148efcad4542b8b1c981cb", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dispense", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6065049d6cb26a02713bab91f45409fb", + "notes": [], + "params": { + "flowRate": 160.0, + "labwareId": "UUID", + "pipetteId": "UUID", + "volume": 15.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -13.78 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 1.92 + }, + "volume": 15.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "touchTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d8c9b0e21562d9e8a2d71bd1a3088120", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "radius": 1.0, + "speed": 60.0, + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": -1.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 14.7 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToAddressableAreaForDropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "216b5144155e6328c29185e03e04d9fc", + "notes": [], + "params": { + "addressableAreaName": "movableTrashB3", + "alternateDropLocation": false, + "forceDirect": false, + "ignoreTipConfiguration": true, + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "pipetteId": "UUID" + }, + "result": { + "position": { + "x": 434.25, + "y": 257.0, + "z": 40.0 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "blowOutInPlace", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "76d8d70471c849fbb67590c55569e0d3", + "notes": [], + "params": { + "flowRate": 80.0, + "pipetteId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d5f11283aab1b275929aa6646e2501c2", + "notes": [], + "params": { + "alternateDropLocation": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0, + "y": 0, + "z": 0 + }, + "origin": "default" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 90.88 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "160a7ab232d0bde9614f5b50222e6c47", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "B2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "0b2e5f3f7e608c2e7809bd44834e49de", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c1b1f44c9384e17cee9f4803cb6677a6", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "10afb0f01a2fb77a5782f457a43d2e6f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c7450aedb2378d40924e4cc7ca21c1be", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "bedc428849a98177a0948e3ba09aa2fb", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "995e4fa6089b0cedb71bd5b09166f328", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "bc07c3745e60ace9bd79e32da0282c78", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "bd7ca9ffc5e07f93ec347f82c91ce0d8", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b2160ce092056873801d0a32dce0a4b8", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3f57afd0650f406adf94d5a92458aeb7", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f90db57f7711f960a031e604bbfd3527", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "73e73a08e679ca0a7cbe7429f8a19b4a", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6692e58ba746f30a1b20687b8e99f4db", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4937b5c0e66d249d5ea96c43fa6d3b2c", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d371cbd83b68cbffb7be6e25c3b68254", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ed9ecc43aab2bd29ad40098a385c93f9", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a9e809dd92546b32573d4ad6bcf3ece7", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c09f686c7dfe6697f6a3650094dbbc84", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "57506d028b0bcce90126749a1fd78c14", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "95a76257deaafc3511b4371fd4fdfe7c", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9170705ba451d685ebfaf3f76ee1f1c3", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "bf641708f9433ec6744081d5573913ab", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "c6a63df9d1acc9e73b5f53f2245450bb", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "85fca02411de028dbfdbbdc6c1449f04", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C3" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b3191029b7084631dbe2a973173fa0fd", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a2149fb398199b39e9e20632f6d86267", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "C4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4b59c64f39d68ad908977acf4f00d679", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "292d13a13702fc64e5a545f66084a9ed", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5f28890d1583622413104e78e49b972f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1a8be3a9a462d031db7b4e83d006043d", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7192937c83a4e042b1de5f4e3ed7a369", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9db1443bddce9820e6b2a0c9ebc06fbc", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d4fb5f99aa44449ee8e8fb97e826adb7", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "b2cb512ab153071d3a68707d72641970", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "413db0f9371d377115869f86146fcfdf", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7b77a6ee758490ba92d9c680f05c2260", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "8241488919ed3ef8f98b798f5fe6dcbe", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "96567f35c139359e0dd78d2e2efa2f4e", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4943cfdb3c86ccd8d7d1a11fb475f735", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "94e6e80c5c1825072c8cd5d2ebb08d58", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "58bfa52776a904da1e9f1b712e921bab", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "064d7ea7cb0199e8940a61bd8ef4a304", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2ae533a493737ec7675837e6d2cf5b12", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "71d14cb299af0a874e39a12e99622ac8", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "83f9d678c823c6871201a8b8b2b25016", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5eae1da80a301e29d1cc96bc9a29e469", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5a8d298ec5e6d6874e893b82db46a522", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3dccee169c309dd15da64f18169cf442", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6727a0e54c2c0f7700a43624095ec428", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "19561ef0794ab922a456e8c45915ff37", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "114a0dcbd0064a8ed77cea50000d9b7c", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "97f2d2f8c37f5c63a2fcb149b2909a23", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "267d50a0e30718468d7c2bb196e5d595", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "109b0d6cc2de4a735a725661c4f53dff", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "85291ee0a8adc54d24f054a67fb02403", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "363dfae708e94787fc96b8bb7aa0603f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7fd72c0a3dcc8a53bcee5a0a8798c362", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "71d26c514a7f9e4f58d5c295865ad52f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "e8be22847742ce9065c4553921286c9f", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "acc5a3b11a76a3b94fd2ea1c50be5524", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "moduleId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "9b181dc059582b482ffa981202edcb04", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "14065f8e3ac83f81fd739499a2da7f4b", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7647627c7c4e481c781ee10a041dbb29", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "labwareId": "UUID" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a6e7f07768306bb5569bcb5e709e8f90", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/closeLid", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "42e95cae75c0eb587706690e34d41856", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/setTargetBlockTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "815b8e29cd92839a9881d53248c2b76d", + "notes": [], + "params": { + "celsius": 60.0, + "holdTimeSeconds": 5.0, + "moduleId": "UUID" + }, + "result": { + "targetBlockTemperature": 60.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/waitForBlockTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "56aaddcfd610c2ef06fce593f5078359", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/setTargetLidTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7ab0747b1e81aba42ba062b7844445da", + "notes": [], + "params": { + "celsius": 80.0, + "moduleId": "UUID" + }, + "result": { + "targetLidTemperature": 80.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/waitForLidTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3b0b16b60e7c54da388623e97b9abfe1", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/deactivateBlock", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4c5a8cdcf1b80f8b52784c617c6a3935", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "thermocycler/deactivateLid", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "89f39e33eeea745fecab5677e73dd375", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/openLabwareLatch", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1c9e89c5ad50b1e55f2e067c61e7a1e5", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": { + "pipetteRetracted": true + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/closeLabwareLatch", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "260b10dd96e32aaa9819280862ef4de1", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/setTargetTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "e9fda4fea2acd77b7dc906ec137b9993", + "notes": [], + "params": { + "celsius": 50.0, + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/setAndWaitForShakeSpeed", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "cf976c11efc8a4838c9b6519eac529be", + "notes": [], + "params": { + "moduleId": "UUID", + "rpm": 1000.0 + }, + "result": { + "pipetteRetracted": true + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/waitForTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "2805479d9b17392b077dceb0104fec1b", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/deactivateHeater", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4f94f50070ec9fc5a620c23719aafd01", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "heaterShaker/deactivateShaker", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "86d319fee8ceae8095389836864b2d43", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "temperatureModule/setTargetTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "ebba384aa94f7264d7c484a31cbb738f", + "notes": [], + "params": { + "celsius": 50.0, + "moduleId": "UUID" + }, + "result": { + "targetTemperature": 50.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "temperatureModule/waitForTemperature", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "d043d5ac8a84b6974db1bbbbcba211e0", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "temperatureModule/deactivate", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "550806039eea80d5525c6001c3a7666d", + "notes": [], + "params": { + "moduleId": "UUID" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "47c406a0d89c37753d88aa54bf7f72b3", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "D4" + }, + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "72676cdca5ae8394dd6e6158b9dab9ee", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": "offDeck", + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "pickUpTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "1a284def350e912e902a648cba064cce", + "notes": [], + "params": { + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 110.0 + }, + "tipDiameter": 5.47, + "tipLength": 85.38999999999999, + "tipVolume": 1000.0 + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "20d79f182152c0f3b5c8bcafc07aa4c8", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 15.7 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "217563003b8a0a95100c4e1153819c38", + "notes": [], + "params": { + "message": "Is the pipette tip in the middle of the PCR Plate, well A1, in slot C2? It should be at the LPC calibrated height." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "reloadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "efc8bf902896ebfa2df28b5309c0ad2b", + "notes": [], + "params": { + "labwareId": "UUID" + }, + "result": { + "labwareId": "UUID", + "offsetId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "953c0d4cf00b4b15bad57b0068917a1a", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 25.700000000000003 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "11fccc4812167d91db9c6794d60377d2", + "notes": [], + "params": { + "message": "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." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "87397b0bc4efe14e59047204629bac0c", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "D2" + }, + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "5504d2c0b6725179832363f618535123", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 74.24, + "z": 15.7 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "6db8184e49b8f0dc341fab21cd407193", + "notes": [], + "params": { + "message": "Is the pipette tip in the middle of the PCR Plate, well A1, in slot D2? It should be at the LPC calibrated height." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "reloadLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "598d9f8200e30e4b18930759933c177e", + "notes": [], + "params": { + "labwareId": "UUID" + }, + "result": { + "labwareId": "UUID", + "offsetId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "3e47d788bbb80170708184edfe5d5855", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 74.24, + "z": 25.700000000000003 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "4d36a2b966013bbc7b01c4b09ff722a5", + "notes": [], + "params": { + "message": "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." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "7105ddfa795ef939e45de1f624ad74e2", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "C2" + }, + "strategy": "manualMoveWithPause" + }, + "result": { + "offsetId": "UUID" + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "96531fb16b3554d5394f56a186180323", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 181.24, + "z": 25.700000000000003 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "8ce8f477d0773532932dc9ce801b5161", + "notes": [], + "params": { + "message": "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." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "a756799f8ef42de5f9dc552e9361a39c", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "slotName": "D2" + }, + "strategy": "manualMoveWithPause" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveToWell", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "8a2981064bc103a5a4f9b865d234d674", + "notes": [], + "params": { + "forceDirect": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "origin": "top" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 227.88, + "y": 42.74, + "z": 44.4 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "77bb267c15b76e8e075a1d8d6e626960", + "notes": [], + "params": { + "message": "Is the pipette tip in the middle of the reservoir , well A1, in slot D2? It should be at the LPC calibrated height." + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "dropTip", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "f98e7bf42290f6ec61132a6d11b082fc", + "notes": [], + "params": { + "alternateDropLocation": false, + "labwareId": "UUID", + "pipetteId": "UUID", + "wellLocation": { + "offset": { + "x": 0, + "y": 0, + "z": 0 + }, + "origin": "default" + }, + "wellName": "A1" + }, + "result": { + "position": { + "x": 178.38, + "y": 395.38, + "z": 90.88 + } + }, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "waitForResume", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "dba1918199fa07e1b9158aeecf0d4794", + "notes": [], + "params": { + "message": "!!!!!!!!!!YOU NEED TO REDO LPC!!!!!!!!!!" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "09d6512b2e7f3d96cac736da70819111", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "gripperWasteChute" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + }, + { + "commandType": "moveLabware", + "completedAt": "TIMESTAMP", + "createdAt": "TIMESTAMP", + "id": "UUID", + "key": "624e7ca7028f54c8442ef1de0638b427", + "notes": [], + "params": { + "labwareId": "UUID", + "newLocation": { + "addressableAreaName": "gripperWasteChute" + }, + "strategy": "usingGripper" + }, + "result": {}, + "startedAt": "TIMESTAMP", + "status": "succeeded" + } + ], + "config": { + "apiVersion": [ + 2, + 19 + ], + "protocolType": "python" + }, + "createdAt": "TIMESTAMP", + "errors": [], + "files": [ + { + "name": "Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py", + "role": "main" + } + ], + "labware": [ + { + "definitionUri": "opentrons/opentrons_96_well_aluminum_block/1", + "id": "UUID", + "loadName": "opentrons_96_well_aluminum_block", + "location": { + "moduleId": "UUID" + } + }, + { + "definitionUri": "opentrons/opentrons_96_pcr_adapter/1", + "id": "UUID", + "loadName": "opentrons_96_pcr_adapter", + "location": { + "moduleId": "UUID" + } + }, + { + "definitionUri": "opentrons/nest_1_reservoir_290ml/1", + "id": "UUID", + "loadName": "nest_1_reservoir_290ml", + "location": "offDeck" + }, + { + "definitionUri": "opentrons/nest_96_wellplate_100ul_pcr_full_skirt/2", + "id": "UUID", + "loadName": "nest_96_wellplate_100ul_pcr_full_skirt", + "location": "offDeck" + }, + { + "definitionUri": "opentrons/opentrons_flex_96_tiprack_adapter/1", + "id": "UUID", + "loadName": "opentrons_flex_96_tiprack_adapter", + "location": { + "slotName": "A2" + } + }, + { + "definitionUri": "opentrons/opentrons_flex_96_tiprack_1000ul/1", + "id": "UUID", + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": "offDeck" + }, + { + "definitionUri": "opentrons/opentrons_flex_96_tiprack_1000ul/1", + "id": "UUID", + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": "offDeck" + }, + { + "definitionUri": "opentrons/opentrons_flex_96_tiprack_1000ul/1", + "id": "UUID", + "loadName": "opentrons_flex_96_tiprack_1000ul", + "location": { + "labwareId": "UUID" + } + } + ], + "liquids": [ + { + "description": "High Quality H₂O", + "displayColor": "#42AB2D", + "displayName": "water", + "id": "UUID" + } + ], + "metadata": { + "author": "Derek Maggio ", + "protocolName": "Flex Smoke Test - v2.19" + }, + "modules": [ + { + "id": "UUID", + "location": { + "slotName": "B1" + }, + "model": "thermocyclerModuleV2", + "serialNumber": "UUID" + }, + { + "id": "UUID", + "location": { + "slotName": "C1" + }, + "model": "magneticBlockV1" + }, + { + "id": "UUID", + "location": { + "slotName": "A3" + }, + "model": "heaterShakerModuleV1", + "serialNumber": "UUID" + }, + { + "id": "UUID", + "location": { + "slotName": "D1" + }, + "model": "temperatureModuleV2", + "serialNumber": "UUID" + } + ], + "pipettes": [ + { + "id": "UUID", + "mount": "left", + "pipetteName": "p1000_96" + } + ], + "result": "ok", + "robotType": "OT-3 Standard", + "runTimeParameters": [ + { + "choices": [ + { + "displayName": "QA Smoke Test", + "value": "qa" + }, + { + "displayName": "Developer Validation", + "value": "dev" + } + ], + "default": "qa", + "description": "Configuration of QA test to perform", + "displayName": "Test Configuration", + "type": "str", + "value": "qa", + "variableName": "test_configuration" + }, + { + "choices": [ + { + "displayName": "Agilent 1 Well 290 mL", + "value": "agilent_1_reservoir_290ml" + }, + { + "displayName": "Nest 1 Well 290 mL", + "value": "nest_1_reservoir_290ml" + } + ], + "default": "nest_1_reservoir_290ml", + "description": "Name of the reservoir", + "displayName": "Reservoir Name", + "type": "str", + "value": "nest_1_reservoir_290ml", + "variableName": "reservoir_name" + }, + { + "choices": [ + { + "displayName": "Nest 96 Well 100 µL", + "value": "nest_96_wellplate_100ul_pcr_full_skirt" + }, + { + "displayName": "Corning 96 Well 360 µL", + "value": "corning_96_wellplate_360ul_flat" + }, + { + "displayName": "Opentrons Tough 96 Well 200 µL", + "value": "opentrons_96_wellplate_200ul_pcr_full_skirt" + } + ], + "default": "nest_96_wellplate_100ul_pcr_full_skirt", + "description": "Name of the well plate", + "displayName": "Well Plate Name", + "type": "str", + "value": "nest_96_wellplate_100ul_pcr_full_skirt", + "variableName": "well_plate_name" + }, + { + "default": false, + "description": "Prefer to use the gripper to dispose of labware, instead of manual moves off deck", + "displayName": "I LOVE TO REFILL TIP RACKS", + "type": "bool", + "value": false, + "variableName": "prefer_gripper_disposal" + } + ] +} diff --git a/api-client/src/client_data/getClientData.ts b/api-client/src/client_data/getClientData.ts new file mode 100644 index 00000000000..34f3c9f5d38 --- /dev/null +++ b/api-client/src/client_data/getClientData.ts @@ -0,0 +1,12 @@ +import { GET, request } from '../request' + +import type { ResponsePromise } from '../request' +import type { HostConfig } from '../types' +import type { ClientDataResponse, DefaultClientData } from './types' + +export function getClientData( + config: HostConfig, + key: string +): ResponsePromise> { + return request>(GET, `/clientData/${key}`, null, config) +} diff --git a/api-client/src/client_data/index.ts b/api-client/src/client_data/index.ts new file mode 100644 index 00000000000..150791479c4 --- /dev/null +++ b/api-client/src/client_data/index.ts @@ -0,0 +1,8 @@ +export { getClientData } from './getClientData' +export { updateClientData } from './updateClientData' + +export type { + ClientDataResponse, + ClientDataRequest, + DefaultClientData, +} from './types' diff --git a/api-client/src/client_data/types.ts b/api-client/src/client_data/types.ts new file mode 100644 index 00000000000..3e523173481 --- /dev/null +++ b/api-client/src/client_data/types.ts @@ -0,0 +1,9 @@ +export type DefaultClientData = Record + +export interface ClientDataResponse { + data: T +} + +export interface ClientDataRequest { + data: T +} diff --git a/api-client/src/client_data/updateClientData.ts b/api-client/src/client_data/updateClientData.ts new file mode 100644 index 00000000000..e0cde9e7b51 --- /dev/null +++ b/api-client/src/client_data/updateClientData.ts @@ -0,0 +1,22 @@ +import { PUT, request } from '../request' + +import type { ResponsePromise } from '../request' +import type { HostConfig } from '../types' +import type { + ClientDataResponse, + ClientDataRequest, + DefaultClientData, +} from './types' + +export function updateClientData( + config: HostConfig, + key: string, + clientData: T +): ResponsePromise> { + return request, ClientDataRequest>( + PUT, + `/clientData/${key}`, + { data: clientData }, + config + ) +} diff --git a/api-client/src/dataFiles/getCsvFileRaw.ts b/api-client/src/dataFiles/getCsvFileRaw.ts new file mode 100644 index 00000000000..a8cdd67f915 --- /dev/null +++ b/api-client/src/dataFiles/getCsvFileRaw.ts @@ -0,0 +1,17 @@ +import { GET, request } from '../request' + +import type { DownloadedCsvFileResponse } from './types' +import type { ResponsePromise } from '../request' +import type { HostConfig } from '../types' + +export function getCsvFileRaw( + config: HostConfig, + fileId: string +): ResponsePromise { + return request( + GET, + `/dataFiles/${fileId}/download`, + null, + config + ) +} diff --git a/api-client/src/dataFiles/index.ts b/api-client/src/dataFiles/index.ts index 03cba1330b9..3496c8acaa0 100644 --- a/api-client/src/dataFiles/index.ts +++ b/api-client/src/dataFiles/index.ts @@ -1,3 +1,4 @@ +export { getCsvFileRaw } from './getCsvFileRaw' export { uploadCsvFile } from './uploadCsvFile' export * from './types' diff --git a/api-client/src/dataFiles/types.ts b/api-client/src/dataFiles/types.ts index 294f10723ec..41029bc4380 100644 --- a/api-client/src/dataFiles/types.ts +++ b/api-client/src/dataFiles/types.ts @@ -18,7 +18,7 @@ export interface UploadedCsvFileResponse { } export interface UploadedCsvFilesResponse { - data: { - files: CsvFileData[] - } + data: CsvFileData[] } + +export type DownloadedCsvFileResponse = string diff --git a/api-client/src/index.ts b/api-client/src/index.ts index 5eb2e960b9b..858772034ab 100644 --- a/api-client/src/index.ts +++ b/api-client/src/index.ts @@ -1,5 +1,6 @@ // api client entry point export * from './calibration' +export * from './client_data' export * from './dataFiles' export * from './deck_configuration' export * from './health' diff --git a/api-client/src/protocols/getCsvFiles.ts b/api-client/src/protocols/getCsvFiles.ts index ebfd7f19a74..5fa122a0951 100644 --- a/api-client/src/protocols/getCsvFiles.ts +++ b/api-client/src/protocols/getCsvFiles.ts @@ -1,43 +1,17 @@ -// import { GET, request } from '../request' +import { GET, request } from '../request' -// import type { ResponsePromise } from '../request' +import type { ResponsePromise } from '../request' import type { HostConfig } from '../types' import type { UploadedCsvFilesResponse } from '../dataFiles/types' -/** export function getCsvFiles( config: HostConfig, protocolId: string -): ResponsePromise { - return request( +): ResponsePromise { + return request( GET, `/protocols/${protocolId}/dataFiles`, null, config ) -} - */ - -// ToDo (kk:06/14/2024) remove when activate the above code -export function getCsvFiles( - config: HostConfig, - protocolId: string -): Promise<{ data: UploadedCsvFilesResponse }> { - const stub = { - data: { - files: [ - { - id: '1', - createdAt: '2024-06-07T19:19:56.268029+00:00', - name: 'rtp_mock_file1.csv', - }, - { - id: '2', - createdAt: '2024-06-17T19:19:56.268029+00:00', - name: 'rtp_mock_file2.csv', - }, - ], - }, - } - return Promise.resolve({ data: stub }) } diff --git a/api-client/src/runs/index.ts b/api-client/src/runs/index.ts index 01653713c81..02bf0c0e036 100644 --- a/api-client/src/runs/index.ts +++ b/api-client/src/runs/index.ts @@ -12,5 +12,7 @@ export { createRunAction } from './createRunAction' export * from './createLabwareOffset' export * from './createLabwareDefinition' export * from './constants' +export * from './updateErrorRecoveryPolicy' + export * from './types' export type { CreateRunData } from './createRun' diff --git a/api-client/src/runs/types.ts b/api-client/src/runs/types.ts index 45e40f2f8b9..1986a34a0b8 100644 --- a/api-client/src/runs/types.ts +++ b/api-client/src/runs/types.ts @@ -146,3 +146,29 @@ export interface CommandData { // Although run errors are semantically different from command errors, // the server currently happens to use the exact same model for both. export type RunError = RunCommandError + +/** + * Error Policy + */ + +export type IfMatchType = 'ignoreAndContinue' | 'failRun' | 'waitForRecovery' + +export interface ErrorRecoveryPolicy { + policyRules: Array<{ + matchCriteria: { + command: { + commandType: RunTimeCommand['commandType'] + error: { + errorType: RunCommandError['errorType'] + } + } + } + ifMatch: IfMatchType + }> +} + +export interface UpdateErrorRecoveryPolicyRequest { + data: ErrorRecoveryPolicy +} + +export type UpdateErrorRecoveryPolicyResponse = Record diff --git a/api-client/src/runs/updateErrorRecoveryPolicy.ts b/api-client/src/runs/updateErrorRecoveryPolicy.ts new file mode 100644 index 00000000000..2efdd974775 --- /dev/null +++ b/api-client/src/runs/updateErrorRecoveryPolicy.ts @@ -0,0 +1,48 @@ +import { PUT, request } from '../request' + +import type { HostConfig } from '../types' +import type { ResponsePromise } from '../request' +import type { + ErrorRecoveryPolicy, + IfMatchType, + UpdateErrorRecoveryPolicyRequest, + UpdateErrorRecoveryPolicyResponse, +} from './types' +import type { RunCommandError, RunTimeCommand } from '@opentrons/shared-data' + +export type RecoveryPolicyRulesParams = Array<{ + commandType: RunTimeCommand['commandType'] + errorType: RunCommandError['errorType'] + ifMatch: IfMatchType +}> + +export function updateErrorRecoveryPolicy( + config: HostConfig, + runId: string, + policyRules: RecoveryPolicyRulesParams +): ResponsePromise { + const policy = buildErrorRecoveryPolicyBody(policyRules) + + return request< + UpdateErrorRecoveryPolicyResponse, + UpdateErrorRecoveryPolicyRequest + >(PUT, `/runs/${runId}/errorRecoveryPolicy`, { data: policy }, config) +} + +function buildErrorRecoveryPolicyBody( + policyRules: RecoveryPolicyRulesParams +): ErrorRecoveryPolicy { + return { + policyRules: policyRules.map(rule => ({ + matchCriteria: { + command: { + commandType: rule.commandType, + error: { + errorType: rule.errorType, + }, + }, + }, + ifMatch: rule.ifMatch, + })), + } +} diff --git a/api/docs/v2/basic_commands/pipette_tips.rst b/api/docs/v2/basic_commands/pipette_tips.rst index f3913445f54..1b0e47f73f2 100644 --- a/api/docs/v2/basic_commands/pipette_tips.rst +++ b/api/docs/v2/basic_commands/pipette_tips.rst @@ -79,7 +79,7 @@ Dropping a Tip To drop a tip in the pipette's trash container, call the :py:meth:`~.InstrumentContext.drop_tip` method with no arguments:: - pipette.pick_up_tip() + pipette.drop_tip() You can specify where to drop the tip by passing in a location. For example, this code drops a tip in the trash bin and returns another tip to to a previously used well in a tip rack:: diff --git a/api/release-notes-internal.md b/api/release-notes-internal.md index 4cffb961116..273056c4670 100644 --- a/api/release-notes-internal.md +++ b/api/release-notes-internal.md @@ -2,15 +2,31 @@ For more details about this release, please see the full [technical change log][ [technical change log]: https://github.com/Opentrons/opentrons/releases +## Internal Release 2.0.0-alpha.3 + +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. + +- [Opentrons changes since the latest stable release](https://github.com/Opentrons/opentrons/compare/v7.3.1...ot3@2.0.0-alpha.3) +- [Opentrons changes since the last internal release](https://github.com/Opentrons/opentrons/compare/ot3@2.0.0-alpha.2...ot3@2.0.0-alpha.3) +- [Flex changes](https://github.com/Opentrons/oe-core/compare/internal@2.0.0-alpha.2...internal@2.0.0-alpha.3) +- [Flex firmware changes](https://github.com/Opentrons/ot3-firmware/compare/internal@v9...internal@v10) +- [OT2 changes](https://github.com/Opentrons/buildroot/compare/v1.17.7...internal@2.0.0-alpha.0) + +## Internal Release 2.0.0-alpha.2 + +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. + + + ## Internal Release 2.0.0-alpha.1 -This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. Usage may require a robot factory reset to restore robot stability. +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. ## Internal Release 2.0.0-alpha.0 -This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. Usage may require a robot factory reset to restore robot stability. +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. diff --git a/api/release-notes.md b/api/release-notes.md index dbfbfc5bad4..d073629a97c 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -6,15 +6,23 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- -## Opentrons Robot Software Changes in 7.4.0 +## Opentrons Robot Software Changes in 7.5.0 -Welcome to the v7.4.0 release of the Opentrons robot software! +Welcome to the v7.5.0 release of the Opentrons robot software! -This release adds support for the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module). +### Hardware Support + +- [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module) +- Latest Flex Gripper model (serial numbers beginning `GRPV13`) ### Bug Fixes - Fixed certain string runtime parameter values being misinterpreted as an incorrect type. + +### Known Issue + +- The HEPA/UV Module's buttons may not respond properly after its safety shutoff is activated. This happens when the module is removed from the top of Flex while its lights are on. Power cycle the module to restore normal behavior. The module is safe to use even if you do not power cycle it. + --- ## Opentrons Robot Software Changes in 7.3.1 diff --git a/api/src/opentrons/calibration_storage/helpers.py b/api/src/opentrons/calibration_storage/helpers.py index b4cc6afe777..1d271add9dd 100644 --- a/api/src/opentrons/calibration_storage/helpers.py +++ b/api/src/opentrons/calibration_storage/helpers.py @@ -14,8 +14,8 @@ from . import types as local_types if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition - from opentrons_shared_data.pipette.dev_types import LabwareUri + from opentrons_shared_data.labware.types import LabwareDefinition + from opentrons_shared_data.pipette.types import LabwareUri def dict_filter_none(data: List[Tuple[str, Any]]) -> Dict[str, Any]: diff --git a/api/src/opentrons/calibration_storage/ot2/models/v1.py b/api/src/opentrons/calibration_storage/ot2/models/v1.py index 585700c84c5..922922415c8 100644 --- a/api/src/opentrons/calibration_storage/ot2/models/v1.py +++ b/api/src/opentrons/calibration_storage/ot2/models/v1.py @@ -4,7 +4,7 @@ from pydantic import BaseModel, Field, validator from datetime import datetime -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons.types import Point from opentrons.calibration_storage import types diff --git a/api/src/opentrons/calibration_storage/ot2/tip_length.py b/api/src/opentrons/calibration_storage/ot2/tip_length.py index 8b5e5369805..a0bcdcabf9d 100644 --- a/api/src/opentrons/calibration_storage/ot2/tip_length.py +++ b/api/src/opentrons/calibration_storage/ot2/tip_length.py @@ -7,7 +7,7 @@ from opentrons import config from .. import file_operators as io, helpers, types as local_types -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons.protocols.api_support.constants import OPENTRONS_NAMESPACE from opentrons.util.helpers import utc_now @@ -16,7 +16,7 @@ from .models import v1 if typing.TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.labware.types import LabwareDefinition log = logging.getLogger(__name__) diff --git a/api/src/opentrons/calibration_storage/ot3/models/v1.py b/api/src/opentrons/calibration_storage/ot3/models/v1.py index 2e621483880..55e028465c7 100644 --- a/api/src/opentrons/calibration_storage/ot3/models/v1.py +++ b/api/src/opentrons/calibration_storage/ot3/models/v1.py @@ -6,7 +6,7 @@ from pydantic import BaseModel, Field, validator from datetime import datetime -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons.types import Point from opentrons.calibration_storage import types diff --git a/api/src/opentrons/cli/analyze.py b/api/src/opentrons/cli/analyze.py index 69a50108762..0daacd711b3 100644 --- a/api/src/opentrons/cli/analyze.py +++ b/api/src/opentrons/cli/analyze.py @@ -51,7 +51,7 @@ ) from opentrons.protocol_engine.protocol_engine import code_in_error_tree -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.errors import ErrorCodes from opentrons_shared_data.errors.exceptions import ( @@ -248,6 +248,7 @@ async def _do_analyze(protocol_source: ProtocolSource) -> RunResult: protocol_source=protocol_source, parse_mode=ParseMode.NORMAL, run_time_param_values=None, + run_time_param_files=None, ) except Exception as error: err_id = "analysis-setup-error" diff --git a/api/src/opentrons/config/advanced_settings.py b/api/src/opentrons/config/advanced_settings.py index 6e3647939cb..812ca73a661 100644 --- a/api/src/opentrons/config/advanced_settings.py +++ b/api/src/opentrons/config/advanced_settings.py @@ -17,7 +17,7 @@ ) from opentrons.config import CONFIG -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum if TYPE_CHECKING: from pathlib import Path diff --git a/api/src/opentrons/config/feature_flags.py b/api/src/opentrons/config/feature_flags.py index 719c0dc43f3..7eb40721511 100644 --- a/api/src/opentrons/config/feature_flags.py +++ b/api/src/opentrons/config/feature_flags.py @@ -1,5 +1,5 @@ from opentrons.config import advanced_settings as advs -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum def short_fixed_trash() -> bool: diff --git a/api/src/opentrons/config/reset.py b/api/src/opentrons/config/reset.py index eac5cf26982..ae69b539607 100644 --- a/api/src/opentrons/config/reset.py +++ b/api/src/opentrons/config/reset.py @@ -5,7 +5,7 @@ from pathlib import Path from typing import NamedTuple, Dict, Set -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum from opentrons.config import IS_ROBOT from opentrons.calibration_storage import ( delete_robot_deck_attitude, diff --git a/api/src/opentrons/execute.py b/api/src/opentrons/execute.py index 91a6c7ba01c..e4109d5d390 100644 --- a/api/src/opentrons/execute.py +++ b/api/src/opentrons/execute.py @@ -24,7 +24,7 @@ ) from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons import protocol_api, __version__, should_use_ot3 @@ -77,7 +77,7 @@ from .util import entrypoint_util if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import ( + from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionDict, ) diff --git a/api/src/opentrons/hardware_control/api.py b/api/src/opentrons/hardware_control/api.py index 0cb7a96be54..ea4c44265c3 100644 --- a/api/src/opentrons/hardware_control/api.py +++ b/api/src/opentrons/hardware_control/api.py @@ -27,8 +27,8 @@ from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, ) -from opentrons_shared_data.pipette.dev_types import PipetteName -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.pipette.types import PipetteName +from opentrons_shared_data.robot.types import RobotType from opentrons import types as top_types from opentrons.config import robot_configs from opentrons.config.types import RobotConfig, OT3Config diff --git a/api/src/opentrons/hardware_control/backends/controller.py b/api/src/opentrons/hardware_control/backends/controller.py index f35d6092134..cdc73c0ea9a 100644 --- a/api/src/opentrons/hardware_control/backends/controller.py +++ b/api/src/opentrons/hardware_control/backends/controller.py @@ -27,7 +27,7 @@ pipette_load_name_conversions as pipette_load_name, mutable_configurations, ) -from opentrons_shared_data.pipette.dev_types import PipetteName +from opentrons_shared_data.pipette.types import PipetteName from opentrons.drivers.smoothie_drivers import SmoothieDriver from opentrons.drivers.rpi_drivers import build_gpio_chardev @@ -40,7 +40,7 @@ from ..util import ot2_axis_to_string if TYPE_CHECKING: - from opentrons_shared_data.pipette.dev_types import PipetteModel + from opentrons_shared_data.pipette.types import PipetteModel from ..dev_types import ( AttachedPipette, AttachedInstruments, diff --git a/api/src/opentrons/hardware_control/backends/flex_protocol.py b/api/src/opentrons/hardware_control/backends/flex_protocol.py index 9e7218099cc..71ce9833251 100644 --- a/api/src/opentrons/hardware_control/backends/flex_protocol.py +++ b/api/src/opentrons/hardware_control/backends/flex_protocol.py @@ -12,7 +12,7 @@ Set, TypeVar, ) -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( PipetteName, ) from opentrons.config.types import GantryLoad, OutputOptions diff --git a/api/src/opentrons/hardware_control/backends/ot3controller.py b/api/src/opentrons/hardware_control/backends/ot3controller.py index cd6aa9e112a..386e6a36159 100644 --- a/api/src/opentrons/hardware_control/backends/ot3controller.py +++ b/api/src/opentrons/hardware_control/backends/ot3controller.py @@ -178,7 +178,7 @@ ) from opentrons_hardware.drivers.gpio import OT3GPIO, RemoteOT3GPIO -from opentrons_shared_data.pipette.dev_types import PipetteName +from opentrons_shared_data.pipette.types import PipetteName from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, load_data as load_pipette_data, diff --git a/api/src/opentrons/hardware_control/backends/ot3simulator.py b/api/src/opentrons/hardware_control/backends/ot3simulator.py index 34c8fe0df68..97d3661e32e 100644 --- a/api/src/opentrons/hardware_control/backends/ot3simulator.py +++ b/api/src/opentrons/hardware_control/backends/ot3simulator.py @@ -47,7 +47,7 @@ HardwareEventUnsubscriber, ) -from opentrons_shared_data.pipette.dev_types import PipetteName, PipetteModel +from opentrons_shared_data.pipette.types import PipetteName, PipetteModel from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, load_data as load_pipette_data, diff --git a/api/src/opentrons/hardware_control/backends/simulator.py b/api/src/opentrons/hardware_control/backends/simulator.py index 9441d478738..da72b2111fe 100644 --- a/api/src/opentrons/hardware_control/backends/simulator.py +++ b/api/src/opentrons/hardware_control/backends/simulator.py @@ -26,7 +26,7 @@ from ..util import ot2_axis_to_string if TYPE_CHECKING: - from opentrons_shared_data.pipette.dev_types import PipetteName, PipetteModel + from opentrons_shared_data.pipette.types import PipetteName, PipetteModel from ..dev_types import ( AttachedPipette, AttachedInstruments, diff --git a/api/src/opentrons/hardware_control/dev_types.py b/api/src/opentrons/hardware_control/dev_types.py index e2b8e542037..a6773cb9184 100644 --- a/api/src/opentrons/hardware_control/dev_types.py +++ b/api/src/opentrons/hardware_control/dev_types.py @@ -10,7 +10,7 @@ from opentrons.hardware_control.instruments.ot3.instrument_calibration import ( GripperCalibrationOffset, ) -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( PipetteModel, PipetteName, ChannelCount, diff --git a/api/src/opentrons/hardware_control/instruments/ot2/instrument_calibration.py b/api/src/opentrons/hardware_control/instruments/ot2/instrument_calibration.py index f2f8a7fc426..e093763dcd1 100644 --- a/api/src/opentrons/hardware_control/instruments/ot2/instrument_calibration.py +++ b/api/src/opentrons/hardware_control/instruments/ot2/instrument_calibration.py @@ -12,8 +12,8 @@ from opentrons_shared_data.labware.labware_definition import LabwareDefinition if typing.TYPE_CHECKING: - from opentrons_shared_data.pipette.dev_types import LabwareUri - from opentrons_shared_data.labware.dev_types import ( + from opentrons_shared_data.pipette.types import LabwareUri + from opentrons_shared_data.labware.types import ( LabwareDefinition as TypeDictLabwareDef, ) diff --git a/api/src/opentrons/hardware_control/instruments/ot2/pipette.py b/api/src/opentrons/hardware_control/instruments/ot2/pipette.py index 5ce653f28a4..7fc15c4c2d3 100644 --- a/api/src/opentrons/hardware_control/instruments/ot2/pipette.py +++ b/api/src/opentrons/hardware_control/instruments/ot2/pipette.py @@ -52,7 +52,7 @@ from opentrons.hardware_control import nozzle_manager -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( UlPerMmAction, PipetteName, PipetteModel, diff --git a/api/src/opentrons/hardware_control/instruments/ot2/pipette_handler.py b/api/src/opentrons/hardware_control/instruments/ot2/pipette_handler.py index 8c333d990fd..99a7a49d41a 100644 --- a/api/src/opentrons/hardware_control/instruments/ot2/pipette_handler.py +++ b/api/src/opentrons/hardware_control/instruments/ot2/pipette_handler.py @@ -21,7 +21,7 @@ UnexpectedTipRemovalError, UnexpectedTipAttachError, ) -from opentrons_shared_data.pipette.dev_types import UlPerMmAction +from opentrons_shared_data.pipette.types import UlPerMmAction from opentrons_shared_data.pipette.types import Quirks from opentrons_shared_data.errors.exceptions import CommandPreconditionViolated diff --git a/api/src/opentrons/hardware_control/instruments/ot3/pipette.py b/api/src/opentrons/hardware_control/instruments/ot3/pipette.py index c9ef68e4e1b..4c079f80b20 100644 --- a/api/src/opentrons/hardware_control/instruments/ot3/pipette.py +++ b/api/src/opentrons/hardware_control/instruments/ot3/pipette.py @@ -36,7 +36,7 @@ load_pipette_offset, PipetteOffsetByPipetteMount, ) -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( UlPerMmAction, PipetteName, PipetteModel, diff --git a/api/src/opentrons/hardware_control/instruments/ot3/pipette_handler.py b/api/src/opentrons/hardware_control/instruments/ot3/pipette_handler.py index 94c5ce8b736..4f24b19c51b 100644 --- a/api/src/opentrons/hardware_control/instruments/ot3/pipette_handler.py +++ b/api/src/opentrons/hardware_control/instruments/ot3/pipette_handler.py @@ -12,7 +12,7 @@ ) from typing_extensions import Final import numpy -from opentrons_shared_data.pipette.dev_types import UlPerMmAction +from opentrons_shared_data.pipette.types import UlPerMmAction from opentrons_shared_data.errors.exceptions import ( CommandPreconditionViolated, diff --git a/api/src/opentrons/hardware_control/modules/types.py b/api/src/opentrons/hardware_control/modules/types.py index 1382961dc21..347d8b0a5f5 100644 --- a/api/src/opentrons/hardware_control/modules/types.py +++ b/api/src/opentrons/hardware_control/modules/types.py @@ -19,7 +19,7 @@ from opentrons.drivers.rpi_drivers.types import USBPort if TYPE_CHECKING: - from opentrons_shared_data.module.dev_types import ( + from opentrons_shared_data.module.types import ( ThermocyclerModuleType, MagneticModuleType, TemperatureModuleType, diff --git a/api/src/opentrons/hardware_control/motion_utilities.py b/api/src/opentrons/hardware_control/motion_utilities.py index ba787d5c767..15604dfd360 100644 --- a/api/src/opentrons/hardware_control/motion_utilities.py +++ b/api/src/opentrons/hardware_control/motion_utilities.py @@ -3,7 +3,7 @@ from typing import Callable, Dict, Union, Optional, cast from collections import OrderedDict -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import Mount, Point from opentrons.calibration_storage.types import AttitudeMatrix diff --git a/api/src/opentrons/hardware_control/ot3api.py b/api/src/opentrons/hardware_control/ot3api.py index cdc95bdd7de..4f0cf262775 100644 --- a/api/src/opentrons/hardware_control/ot3api.py +++ b/api/src/opentrons/hardware_control/ot3api.py @@ -26,13 +26,13 @@ ) -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( PipetteName, ) from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, ) -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons import types as top_types from opentrons.config import robot_configs @@ -761,7 +761,7 @@ async def reset_tip_detectors( @ExecutionManagerProvider.wait_for_running async def _update_position_estimation( - self, axes: Optional[List[Axis]] = None + self, axes: Optional[Sequence[Axis]] = None ) -> None: """ Function to update motor estimation for a set of axes @@ -1141,6 +1141,12 @@ async def gantry_position( z=cur_pos[Axis.by_mount(realmount)], ) + async def update_axis_position_estimations(self, axes: Sequence[Axis]) -> None: + """Update specified axes position estimators from their encoders.""" + await self._update_position_estimation(axes) + await self._cache_current_position() + await self._cache_encoder_position() + async def move_to( self, mount: Union[top_types.Mount, OT3Mount], diff --git a/api/src/opentrons/hardware_control/protocols/__init__.py b/api/src/opentrons/hardware_control/protocols/__init__.py index 41de2b54506..cff17ff1d9a 100644 --- a/api/src/opentrons/hardware_control/protocols/__init__.py +++ b/api/src/opentrons/hardware_control/protocols/__init__.py @@ -1,8 +1,6 @@ """Typing protocols describing a hardware controller.""" from typing_extensions import Protocol, Type -from opentrons.hardware_control.types import Axis - from .module_provider import ModuleProvider from .hardware_manager import HardwareManager from .chassis_accessory_manager import ChassisAccessoryManager @@ -20,6 +18,7 @@ from .gripper_controller import GripperController from .flex_calibratable import FlexCalibratable from .flex_instrument_configurer import FlexInstrumentConfigurer +from .position_estimator import PositionEstimator from .types import ( CalibrationType, @@ -64,6 +63,7 @@ def cache_tip(self, mount: MountArgType, tip_length: float) -> None: class FlexHardwareControlInterface( + PositionEstimator, ModuleProvider, ExecutionControllable, LiquidHandler[CalibrationType, MountArgType, ConfigType], @@ -87,12 +87,6 @@ class FlexHardwareControlInterface( def get_robot_type(self) -> Type[FlexRobotType]: return FlexRobotType - def motor_status_ok(self, axis: Axis) -> bool: - ... - - def encoder_status_ok(self, axis: Axis) -> bool: - ... - def cache_tip(self, mount: MountArgType, tip_length: float) -> None: ... diff --git a/api/src/opentrons/hardware_control/protocols/instrument_configurer.py b/api/src/opentrons/hardware_control/protocols/instrument_configurer.py index ab5b37acc99..11e718a9aff 100644 --- a/api/src/opentrons/hardware_control/protocols/instrument_configurer.py +++ b/api/src/opentrons/hardware_control/protocols/instrument_configurer.py @@ -1,7 +1,7 @@ from typing import Dict, Optional from typing_extensions import Protocol -from opentrons_shared_data.pipette.dev_types import PipetteName +from opentrons_shared_data.pipette.types import PipetteName from opentrons.types import Mount from .types import MountArgType diff --git a/api/src/opentrons/hardware_control/protocols/position_estimator.py b/api/src/opentrons/hardware_control/protocols/position_estimator.py new file mode 100644 index 00000000000..04d551020c3 --- /dev/null +++ b/api/src/opentrons/hardware_control/protocols/position_estimator.py @@ -0,0 +1,43 @@ +from typing import Protocol, Sequence + +from ..types import Axis + + +class PositionEstimator(Protocol): + """Position-control extensions for harwdare with encoders.""" + + async def update_axis_position_estimations(self, axes: Sequence[Axis]) -> None: + """Update the specified axes' position estimators from their encoders. + + This will allow these axes to make a non-home move even if they do not currently have + a position estimation (unless there is no tracked poition from the encoders, as would be + true immediately after boot). + + Axis encoders have less precision than their position estimators. Calling this function will + cause absolute position drift. After this function is called, the axis should be homed before + it is relied upon for accurate motion. + + This function updates only the requested axes. If other axes have bad position estimation, + moves that require those axes or attempts to get the position of those axes will still fail. + """ + ... + + def motor_status_ok(self, axis: Axis) -> bool: + """Return whether an axis' position estimator is healthy. + + The position estimator is healthy if the axis has + 1) been homed + 2) not suffered a loss-of-positioning (from a cancel or stall, for instance) since being homed + + If this function returns false, getting the position of this axis or asking it to move will fail. + """ + ... + + def encoder_status_ok(self, axis: Axis) -> bool: + """Return whether an axis' position encoder tracking is healthy. + + The encoder status is healthy if the axis has been homed since booting up. + + If this function returns false, updating the estimator from the encoder will fail. + """ + ... diff --git a/api/src/opentrons/motion_planning/adjacent_slots_getters.py b/api/src/opentrons/motion_planning/adjacent_slots_getters.py index 9644f40f157..3a6166eb487 100644 --- a/api/src/opentrons/motion_planning/adjacent_slots_getters.py +++ b/api/src/opentrons/motion_planning/adjacent_slots_getters.py @@ -2,7 +2,7 @@ from dataclasses import dataclass from typing import Optional, List, Dict, Union -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName, StagingSlotName diff --git a/api/src/opentrons/motion_planning/deck_conflict.py b/api/src/opentrons/motion_planning/deck_conflict.py index 8b26897dc1b..69453928511 100644 --- a/api/src/opentrons/motion_planning/deck_conflict.py +++ b/api/src/opentrons/motion_planning/deck_conflict.py @@ -5,8 +5,8 @@ from typing import List, Mapping, NamedTuple, Optional, Set, Union from typing_extensions import Final -from opentrons_shared_data.labware.dev_types import LabwareUri -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.labware.types import LabwareUri +from opentrons_shared_data.robot.types import RobotType from opentrons.motion_planning.adjacent_slots_getters import ( get_east_west_slots, get_south_slot, diff --git a/api/src/opentrons/protocol_api/_parameter_context.py b/api/src/opentrons/protocol_api/_parameter_context.py index b81c9250f2f..32528dab8d6 100644 --- a/api/src/opentrons/protocol_api/_parameter_context.py +++ b/api/src/opentrons/protocol_api/_parameter_context.py @@ -12,10 +12,15 @@ ParameterChoice, UserFacingTypes, ) -from opentrons.protocols.parameters.exceptions import ParameterDefinitionError +from opentrons.protocols.parameters.exceptions import ( + ParameterDefinitionError, + ParameterValueError, +) from opentrons.protocol_engine.types import ( RunTimeParameter, - RunTimeParamValuesType, + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, + FileInfo, ) from ._parameters import Parameters @@ -185,7 +190,9 @@ def add_csv_file( ) self._parameters[parameter.variable_name] = parameter - def set_parameters(self, parameter_overrides: RunTimeParamValuesType) -> None: + def set_parameters( + self, parameter_overrides: PrimitiveRunTimeParamValuesType + ) -> None: """Sets parameters to values given by client, validating them as well. :meta private: @@ -200,13 +207,45 @@ def set_parameters(self, parameter_overrides: RunTimeParamValuesType) -> None: f"Parameter {variable_name} is not defined as a parameter for this protocol." ) if isinstance(parameter, csv_parameter_definition.CSVParameterDefinition): - pass + raise ParameterValueError( + f"A primitive param value was provided for the parameter '{variable_name}'," + f" but '{variable_name}' is a CSV parameter that can only accept file IDs." + ) else: validated_value = validation.ensure_value_type( override_value, parameter.parameter_type ) parameter.value = validated_value + def initialize_csv_files( + self, run_time_param_file_overrides: CSVRunTimeParamFilesType + ) -> None: + """Initializes the files for CSV parameters. + + :meta private: + + This is intended for Opentrons internal use only and is not a guaranteed API. + """ + for variable_name, file_id in run_time_param_file_overrides.items(): + try: + parameter = self._parameters[variable_name] + except KeyError: + raise ParameterDefinitionError( + f"Parameter {variable_name} is not defined as a parameter for this protocol." + ) + if not isinstance( + parameter, csv_parameter_definition.CSVParameterDefinition + ): + raise ParameterValueError( + f"File Id was provided for the parameter '{variable_name}'," + f" but '{variable_name}' is not a CSV parameter." + ) + + parameter.file_info = FileInfo(id=file_id, name="") + # TODO (spp, 2024-07-16): set the file name and assign the file as parameter.value. + # Most likely, we will be creating a temporary file copy of the original + # to pass onto the protocol context + def export_parameters_for_analysis(self) -> List[RunTimeParameter]: """Exports all parameters into a protocol engine models for reporting in analysis. diff --git a/api/src/opentrons/protocol_api/core/engine/instrument.py b/api/src/opentrons/protocol_api/core/engine/instrument.py index d89e946dadc..e57575593de 100644 --- a/api/src/opentrons/protocol_api/core/engine/instrument.py +++ b/api/src/opentrons/protocol_api/core/engine/instrument.py @@ -30,7 +30,7 @@ from opentrons.protocol_engine.errors.exceptions import TipNotAttachedError from opentrons.protocol_engine.clients import SyncClient as EngineClient from opentrons.protocols.api_support.definitions import MAX_SUPPORTED_VERSION -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.protocol_api._nozzle_layout import NozzleLayout from opentrons.hardware_control.nozzle_manager import NozzleConfigurationType from opentrons.hardware_control.nozzle_manager import NozzleMap diff --git a/api/src/opentrons/protocol_api/core/engine/labware.py b/api/src/opentrons/protocol_api/core/engine/labware.py index abe6e3c70bd..fc32038d5a1 100644 --- a/api/src/opentrons/protocol_api/core/engine/labware.py +++ b/api/src/opentrons/protocol_api/core/engine/labware.py @@ -1,7 +1,7 @@ """ProtocolEngine-based Labware core implementations.""" from typing import List, Optional, cast -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareParameters as LabwareParametersDict, LabwareDefinition as LabwareDefinitionDict, ) diff --git a/api/src/opentrons/protocol_api/core/engine/protocol.py b/api/src/opentrons/protocol_api/core/engine/protocol.py index 6b040243193..67cb369c306 100644 --- a/api/src/opentrons/protocol_api/core/engine/protocol.py +++ b/api/src/opentrons/protocol_api/core/engine/protocol.py @@ -4,11 +4,11 @@ from opentrons.protocol_engine import commands as cmd from opentrons.protocol_engine.commands import LoadModuleResult -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5, SlotDefV3 +from opentrons_shared_data.deck.types import DeckDefinitionV5, SlotDefV3 from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import ( DeckSlotName, diff --git a/api/src/opentrons/protocol_api/core/labware.py b/api/src/opentrons/protocol_api/core/labware.py index ada1a7ff0ed..67b452cca6d 100644 --- a/api/src/opentrons/protocol_api/core/labware.py +++ b/api/src/opentrons/protocol_api/core/labware.py @@ -4,7 +4,7 @@ from abc import ABC, abstractmethod from typing import Any, Generic, List, NamedTuple, Optional, TypeVar -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareUri, LabwareParameters as LabwareParametersDict, LabwareDefinition as LabwareDefinitionDict, diff --git a/api/src/opentrons/protocol_api/core/legacy/deck.py b/api/src/opentrons/protocol_api/core/legacy/deck.py index 685f0f5d553..1a225404bd7 100644 --- a/api/src/opentrons/protocol_api/core/legacy/deck.py +++ b/api/src/opentrons/protocol_api/core/legacy/deck.py @@ -8,8 +8,8 @@ from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.deck.dev_types import SlotDefV3 -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.deck.types import SlotDefV3 +from opentrons_shared_data.labware.types import LabwareUri from opentrons.hardware_control.modules.types import ModuleType from opentrons.motion_planning import deck_conflict diff --git a/api/src/opentrons/protocol_api/core/legacy/legacy_labware_core.py b/api/src/opentrons/protocol_api/core/legacy/legacy_labware_core.py index ece9be66f19..575fd7a8cc6 100644 --- a/api/src/opentrons/protocol_api/core/legacy/legacy_labware_core.py +++ b/api/src/opentrons/protocol_api/core/legacy/legacy_labware_core.py @@ -6,7 +6,7 @@ from opentrons.types import DeckSlotName, Location, Point from opentrons.hardware_control.nozzle_manager import NozzleMap -from opentrons_shared_data.labware.dev_types import LabwareParameters, LabwareDefinition +from opentrons_shared_data.labware.types import LabwareParameters, LabwareDefinition from ..labware import AbstractLabware, LabwareLoadParams from .legacy_well_core import LegacyWellCore diff --git a/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py b/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py index 1c8181f1afb..d698604ac30 100644 --- a/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py +++ b/api/src/opentrons/protocol_api/core/legacy/legacy_protocol_core.py @@ -1,10 +1,10 @@ import logging from typing import Dict, List, Optional, Set, Union, cast, Tuple -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5, SlotDefV3 -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5, SlotDefV3 +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName, StagingSlotName, Location, Mount, Point from opentrons.util.broker import Broker diff --git a/api/src/opentrons/protocol_api/core/legacy/load_info.py b/api/src/opentrons/protocol_api/core/legacy/load_info.py index 098ce46a971..7ceaf01c393 100644 --- a/api/src/opentrons/protocol_api/core/legacy/load_info.py +++ b/api/src/opentrons/protocol_api/core/legacy/load_info.py @@ -7,7 +7,7 @@ from dataclasses import dataclass from typing import Optional, Union -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from opentrons.hardware_control.dev_types import PipetteDict from opentrons.hardware_control.modules.types import ModuleModel diff --git a/api/src/opentrons/protocol_api/core/legacy/module_geometry.py b/api/src/opentrons/protocol_api/core/legacy/module_geometry.py index 839154a76d1..504b3c639bf 100644 --- a/api/src/opentrons/protocol_api/core/legacy/module_geometry.py +++ b/api/src/opentrons/protocol_api/core/legacy/module_geometry.py @@ -16,7 +16,7 @@ from numpy.typing import NDArray from opentrons_shared_data import module -from opentrons_shared_data.module.dev_types import ModuleDefinitionV3 +from opentrons_shared_data.module.types import ModuleDefinitionV3 from opentrons_shared_data.module import OLD_TC_GEN2_LABWARE_OFFSET from opentrons.types import Location, Point, LocationLabware diff --git a/api/src/opentrons/protocol_api/core/legacy/well_geometry.py b/api/src/opentrons/protocol_api/core/legacy/well_geometry.py index 8855997b304..6083b84f6ab 100644 --- a/api/src/opentrons/protocol_api/core/legacy/well_geometry.py +++ b/api/src/opentrons/protocol_api/core/legacy/well_geometry.py @@ -3,7 +3,7 @@ from typing import Optional, cast, TYPE_CHECKING from opentrons.types import Point -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( WellDefinition, CircularWellDefinition, RectangularWellDefinition, diff --git a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py index 002ca5f6017..d0002763b1c 100644 --- a/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py +++ b/api/src/opentrons/protocol_api/core/legacy_simulator/legacy_protocol_core.py @@ -1,7 +1,7 @@ import logging from typing import Dict, Optional -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.pipette.pipette_load_name_conversions import ( convert_to_pipette_name_type, ) diff --git a/api/src/opentrons/protocol_api/core/protocol.py b/api/src/opentrons/protocol_api/core/protocol.py index 3b7aa87a5bb..a8403cc40da 100644 --- a/api/src/opentrons/protocol_api/core/protocol.py +++ b/api/src/opentrons/protocol_api/core/protocol.py @@ -5,10 +5,10 @@ from abc import abstractmethod, ABC from typing import Generic, List, Optional, Union, Tuple, Dict, TYPE_CHECKING -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5, SlotDefV3 -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5, SlotDefV3 +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName, StagingSlotName, Location, Mount, Point from opentrons.hardware_control import SyncHardwareAPI diff --git a/api/src/opentrons/protocol_api/create_protocol_context.py b/api/src/opentrons/protocol_api/create_protocol_context.py index b01d4bbbbe0..e74b48e23c0 100644 --- a/api/src/opentrons/protocol_api/create_protocol_context.py +++ b/api/src/opentrons/protocol_api/create_protocol_context.py @@ -2,7 +2,7 @@ import asyncio from typing import Any, Dict, Optional, Union, cast -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from opentrons.hardware_control import ( HardwareControlAPI, diff --git a/api/src/opentrons/protocol_api/deck.py b/api/src/opentrons/protocol_api/deck.py index b4ebe8ae766..352bb4912cd 100644 --- a/api/src/opentrons/protocol_api/deck.py +++ b/api/src/opentrons/protocol_api/deck.py @@ -2,13 +2,13 @@ from dataclasses import dataclass from typing import Iterator, List, Mapping, Optional, Tuple, Union -from opentrons_shared_data.deck.dev_types import SlotDefV3 +from opentrons_shared_data.deck.types import SlotDefV3 from opentrons.motion_planning import adjacent_slots_getters from opentrons.protocols.api_support.types import APIVersion from opentrons.protocols.api_support.util import APIVersionError from opentrons.types import DeckLocation, DeckSlotName, StagingSlotName, Location, Point -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from .core.common import ProtocolCore diff --git a/api/src/opentrons/protocol_api/labware.py b/api/src/opentrons/protocol_api/labware.py index 6e6932e0cf6..3ad328258c6 100644 --- a/api/src/opentrons/protocol_api/labware.py +++ b/api/src/opentrons/protocol_api/labware.py @@ -15,7 +15,7 @@ from itertools import dropwhile from typing import TYPE_CHECKING, Any, List, Dict, Optional, Union, Tuple, cast -from opentrons_shared_data.labware.dev_types import LabwareDefinition, LabwareParameters +from opentrons_shared_data.labware.types import LabwareDefinition, LabwareParameters from opentrons.types import Location, Point from opentrons.protocols.api_support.types import APIVersion diff --git a/api/src/opentrons/protocol_api/module_contexts.py b/api/src/opentrons/protocol_api/module_contexts.py index c8947049a96..45f53a95de1 100644 --- a/api/src/opentrons/protocol_api/module_contexts.py +++ b/api/src/opentrons/protocol_api/module_contexts.py @@ -3,8 +3,8 @@ import logging from typing import List, Optional, Union, cast -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.module.dev_types import ModuleModel, ModuleType +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.module.types import ModuleModel, ModuleType from opentrons.legacy_broker import LegacyBroker from opentrons.hardware_control.modules import ThermocyclerStep diff --git a/api/src/opentrons/protocol_api/protocol_context.py b/api/src/opentrons/protocol_api/protocol_context.py index 57a04d664a6..ad1f326b40e 100644 --- a/api/src/opentrons/protocol_api/protocol_context.py +++ b/api/src/opentrons/protocol_api/protocol_context.py @@ -13,8 +13,8 @@ cast, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import Mount, Location, DeckLocation, DeckSlotName, StagingSlotName from opentrons.legacy_broker import LegacyBroker diff --git a/api/src/opentrons/protocol_api/validation.py b/api/src/opentrons/protocol_api/validation.py index 62e5ecc3dc1..207c417cf5e 100644 --- a/api/src/opentrons/protocol_api/validation.py +++ b/api/src/opentrons/protocol_api/validation.py @@ -15,8 +15,8 @@ from typing_extensions import TypeGuard from opentrons_shared_data.labware.labware_definition import LabwareRole -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocols.api_support.types import APIVersion from opentrons.protocols.api_support.util import APIVersionError diff --git a/api/src/opentrons/protocol_engine/actions/actions.py b/api/src/opentrons/protocol_engine/actions/actions.py index c8e6643335f..142870ee40a 100644 --- a/api/src/opentrons/protocol_engine/actions/actions.py +++ b/api/src/opentrons/protocol_engine/actions/actions.py @@ -20,7 +20,7 @@ CommandDefinedErrorData, CommandPrivateResult, ) -from ..error_recovery_policy import ErrorRecoveryType +from ..error_recovery_policy import ErrorRecoveryPolicy, ErrorRecoveryType from ..notes.notes import CommandNote from ..types import ( LabwareOffsetCreate, @@ -266,6 +266,13 @@ class SetPipetteMovementSpeedAction: speed: Optional[float] +@dataclass(frozen=True) +class SetErrorRecoveryPolicyAction: + """See `ProtocolEngine.set_error_recovery_policy()`.""" + + error_recovery_policy: ErrorRecoveryPolicy + + Action = Union[ PlayAction, PauseAction, @@ -286,4 +293,5 @@ class SetPipetteMovementSpeedAction: AddLiquidAction, ResetTipsAction, SetPipetteMovementSpeedAction, + SetErrorRecoveryPolicyAction, ] diff --git a/api/src/opentrons/protocol_engine/clients/sync_client.py b/api/src/opentrons/protocol_engine/clients/sync_client.py index 5750ba72d21..59407e1d1fe 100644 --- a/api/src/opentrons/protocol_engine/clients/sync_client.py +++ b/api/src/opentrons/protocol_engine/clients/sync_client.py @@ -2,7 +2,7 @@ from typing import cast, Any, Optional, overload -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition from .. import commands diff --git a/api/src/opentrons/protocol_engine/clients/transports.py b/api/src/opentrons/protocol_engine/clients/transports.py index 6de08db97ed..348bbc286c2 100644 --- a/api/src/opentrons/protocol_engine/clients/transports.py +++ b/api/src/opentrons/protocol_engine/clients/transports.py @@ -3,7 +3,7 @@ from typing import Any, Final, overload from typing_extensions import Literal -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition @@ -125,11 +125,13 @@ async def run_in_pe_thread() -> Command: ) if command.error is not None: - error_was_recovered_from = ( + error_recovery_type = ( self._engine.state_view.commands.get_error_recovery_type(command.id) - == ErrorRecoveryType.WAIT_FOR_RECOVERY ) - if not error_was_recovered_from: + error_should_fail_run = ( + error_recovery_type == ErrorRecoveryType.FAIL_RUN + ) + if error_should_fail_run: error = command.error # TODO: this needs to have an actual code raise ProtocolCommandFailedError( diff --git a/api/src/opentrons/protocol_engine/commands/__init__.py b/api/src/opentrons/protocol_engine/commands/__init__.py index 75904ab00a3..d0550fce8c5 100644 --- a/api/src/opentrons/protocol_engine/commands/__init__.py +++ b/api/src/opentrons/protocol_engine/commands/__init__.py @@ -19,6 +19,7 @@ from . import temperature_module from . import thermocycler from . import calibration +from . import unsafe from .hash_command_params import hash_protocol_command_params from .generate_command_schema import generate_command_schema @@ -548,6 +549,8 @@ "thermocycler", # calibration command bundle "calibration", + # unsafe command bundle + "unsafe", # configure pipette volume command bundle "ConfigureForVolume", "ConfigureForVolumeCreate", diff --git a/api/src/opentrons/protocol_engine/commands/aspirate.py b/api/src/opentrons/protocol_engine/commands/aspirate.py index 46e1147a559..29daea563bb 100644 --- a/api/src/opentrons/protocol_engine/commands/aspirate.py +++ b/api/src/opentrons/protocol_engine/commands/aspirate.py @@ -138,6 +138,7 @@ async def execute(self, params: AspirateParams) -> _ExecuteReturn: error=e, ) ], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, ), private=OverpressureErrorInternalData( position=DeckPoint.construct( diff --git a/api/src/opentrons/protocol_engine/commands/aspirate_in_place.py b/api/src/opentrons/protocol_engine/commands/aspirate_in_place.py index a70d0cf7f39..23b11598573 100644 --- a/api/src/opentrons/protocol_engine/commands/aspirate_in_place.py +++ b/api/src/opentrons/protocol_engine/commands/aspirate_in_place.py @@ -1,9 +1,11 @@ """Aspirate in place command request, result, and implementation models.""" from __future__ import annotations -from typing import TYPE_CHECKING, Optional, Type +from typing import TYPE_CHECKING, Optional, Type, Union from typing_extensions import Literal +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError + from opentrons.hardware_control import HardwareControlAPI from .pipetting_common import ( @@ -11,13 +13,23 @@ AspirateVolumeMixin, FlowRateMixin, BaseLiquidHandlingResult, + OverpressureError, + OverpressureErrorInternalData, +) +from .command import ( + AbstractCommandImpl, + BaseCommand, + BaseCommandCreate, + SuccessData, + DefinedErrorData, ) -from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence from ..errors.exceptions import PipetteNotReadyToAspirateError +from ..types import DeckPoint if TYPE_CHECKING: - from ..execution import PipettingHandler + from ..execution import PipettingHandler, GantryMover + from ..resources import ModelUtils from ..state import StateView from ..notes import CommandNoteAdder @@ -36,8 +48,14 @@ class AspirateInPlaceResult(BaseLiquidHandlingResult): pass +_ExecuteReturn = Union[ + SuccessData[AspirateInPlaceResult, None], + DefinedErrorData[OverpressureError, OverpressureErrorInternalData], +] + + class AspirateInPlaceImplementation( - AbstractCommandImpl[AspirateInPlaceParams, SuccessData[AspirateInPlaceResult, None]] + AbstractCommandImpl[AspirateInPlaceParams, _ExecuteReturn] ): """AspirateInPlace command implementation.""" @@ -47,16 +65,18 @@ def __init__( hardware_api: HardwareControlAPI, state_view: StateView, command_note_adder: CommandNoteAdder, + model_utils: ModelUtils, + gantry_mover: GantryMover, **kwargs: object, ) -> None: self._pipetting = pipetting self._state_view = state_view self._hardware_api = hardware_api self._command_note_adder = command_note_adder + self._model_utils = model_utils + self._gantry_mover = gantry_mover - async def execute( - self, params: AspirateInPlaceParams - ) -> SuccessData[AspirateInPlaceResult, None]: + async def execute(self, params: AspirateInPlaceParams) -> _ExecuteReturn: """Aspirate without moving the pipette. Raises: @@ -73,14 +93,48 @@ async def execute( " The first aspirate following a blow-out must be from a specific well" " so the plunger can be reset in a known safe position." ) - volume = await self._pipetting.aspirate_in_place( - pipette_id=params.pipetteId, - volume=params.volume, - flow_rate=params.flowRate, - command_note_adder=self._command_note_adder, - ) - - return SuccessData(public=AspirateInPlaceResult(volume=volume), private=None) + try: + volume = await self._pipetting.aspirate_in_place( + pipette_id=params.pipetteId, + volume=params.volume, + flow_rate=params.flowRate, + command_note_adder=self._command_note_adder, + ) + except PipetteOverpressureError as e: + current_position = await self._gantry_mover.get_position(params.pipetteId) + return DefinedErrorData( + public=OverpressureError( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + wrappedErrors=[ + ErrorOccurrence.from_failed( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + error=e, + ) + ], + errorInfo=( + { + "retryLocation": ( + current_position.x, + current_position.y, + current_position.z, + ) + } + ), + ), + private=OverpressureErrorInternalData( + position=DeckPoint( + x=current_position.x, + y=current_position.y, + z=current_position.z, + ), + ), + ) + else: + return SuccessData( + public=AspirateInPlaceResult(volume=volume), private=None + ) class AspirateInPlace( diff --git a/api/src/opentrons/protocol_engine/commands/command_unions.py b/api/src/opentrons/protocol_engine/commands/command_unions.py index d20b64f363b..eeafb1770b6 100644 --- a/api/src/opentrons/protocol_engine/commands/command_unions.py +++ b/api/src/opentrons/protocol_engine/commands/command_unions.py @@ -22,6 +22,7 @@ from . import thermocycler from . import calibration +from . import unsafe from .set_rail_lights import ( SetRailLights, @@ -387,6 +388,9 @@ calibration.CalibratePipette, calibration.CalibrateModule, calibration.MoveToMaintenancePosition, + unsafe.UnsafeBlowOutInPlace, + unsafe.UnsafeDropTipInPlace, + unsafe.UpdatePositionEstimators, ], Field(discriminator="commandType"), ] @@ -456,6 +460,9 @@ calibration.CalibratePipetteParams, calibration.CalibrateModuleParams, calibration.MoveToMaintenancePositionParams, + unsafe.UnsafeBlowOutInPlaceParams, + unsafe.UnsafeDropTipInPlaceParams, + unsafe.UpdatePositionEstimatorsParams, ] CommandType = Union[ @@ -523,6 +530,9 @@ calibration.CalibratePipetteCommandType, calibration.CalibrateModuleCommandType, calibration.MoveToMaintenancePositionCommandType, + unsafe.UnsafeBlowOutInPlaceCommandType, + unsafe.UnsafeDropTipInPlaceCommandType, + unsafe.UpdatePositionEstimatorsCommandType, ] CommandCreate = Annotated[ @@ -591,6 +601,9 @@ calibration.CalibratePipetteCreate, calibration.CalibrateModuleCreate, calibration.MoveToMaintenancePositionCreate, + unsafe.UnsafeBlowOutInPlaceCreate, + unsafe.UnsafeDropTipInPlaceCreate, + unsafe.UpdatePositionEstimatorsCreate, ], Field(discriminator="commandType"), ] @@ -660,6 +673,9 @@ calibration.CalibratePipetteResult, calibration.CalibrateModuleResult, calibration.MoveToMaintenancePositionResult, + unsafe.UnsafeBlowOutInPlaceResult, + unsafe.UnsafeDropTipInPlaceResult, + unsafe.UpdatePositionEstimatorsResult, ] # todo(mm, 2024-06-12): Ideally, command return types would have specific diff --git a/api/src/opentrons/protocol_engine/commands/dispense.py b/api/src/opentrons/protocol_engine/commands/dispense.py index 7ba9fe2ae52..b346fb5845a 100644 --- a/api/src/opentrons/protocol_engine/commands/dispense.py +++ b/api/src/opentrons/protocol_engine/commands/dispense.py @@ -1,8 +1,10 @@ """Dispense command request, result, and implementation models.""" from __future__ import annotations -from typing import TYPE_CHECKING, Optional, Type +from typing import TYPE_CHECKING, Optional, Type, Union from typing_extensions import Literal +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError + from pydantic import Field from ..types import DeckPoint @@ -13,12 +15,21 @@ WellLocationMixin, BaseLiquidHandlingResult, DestinationPositionResult, + OverpressureError, + OverpressureErrorInternalData, +) +from .command import ( + AbstractCommandImpl, + BaseCommand, + BaseCommandCreate, + DefinedErrorData, + SuccessData, ) -from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence if TYPE_CHECKING: from ..execution import MovementHandler, PipettingHandler + from ..resources import ModelUtils DispenseCommandType = Literal["dispense"] @@ -41,20 +52,27 @@ class DispenseResult(BaseLiquidHandlingResult, DestinationPositionResult): pass -class DispenseImplementation( - AbstractCommandImpl[DispenseParams, SuccessData[DispenseResult, None]] -): +_ExecuteReturn = Union[ + SuccessData[DispenseResult, None], + DefinedErrorData[OverpressureError, OverpressureErrorInternalData], +] + + +class DispenseImplementation(AbstractCommandImpl[DispenseParams, _ExecuteReturn]): """Dispense command implementation.""" def __init__( - self, movement: MovementHandler, pipetting: PipettingHandler, **kwargs: object + self, + movement: MovementHandler, + pipetting: PipettingHandler, + model_utils: ModelUtils, + **kwargs: object, ) -> None: self._movement = movement self._pipetting = pipetting + self._model_utils = model_utils - async def execute( - self, params: DispenseParams - ) -> SuccessData[DispenseResult, None]: + async def execute(self, params: DispenseParams) -> _ExecuteReturn: """Move to and dispense to the requested well.""" position = await self._movement.move_to_well( pipette_id=params.pipetteId, @@ -62,20 +80,41 @@ async def execute( well_name=params.wellName, well_location=params.wellLocation, ) - volume = await self._pipetting.dispense_in_place( - pipette_id=params.pipetteId, - volume=params.volume, - flow_rate=params.flowRate, - push_out=params.pushOut, - ) - - return SuccessData( - public=DispenseResult( - volume=volume, - position=DeckPoint(x=position.x, y=position.y, z=position.z), - ), - private=None, - ) + try: + volume = await self._pipetting.dispense_in_place( + pipette_id=params.pipetteId, + volume=params.volume, + flow_rate=params.flowRate, + push_out=params.pushOut, + ) + except PipetteOverpressureError as e: + return DefinedErrorData( + public=OverpressureError( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + wrappedErrors=[ + ErrorOccurrence.from_failed( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + error=e, + ) + ], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint.construct( + x=position.x, y=position.y, z=position.z + ) + ), + ) + else: + return SuccessData( + public=DispenseResult( + volume=volume, + position=DeckPoint(x=position.x, y=position.y, z=position.z), + ), + private=None, + ) class Dispense(BaseCommand[DispenseParams, DispenseResult, ErrorOccurrence]): diff --git a/api/src/opentrons/protocol_engine/commands/dispense_in_place.py b/api/src/opentrons/protocol_engine/commands/dispense_in_place.py index 160345de469..d71f191d1df 100644 --- a/api/src/opentrons/protocol_engine/commands/dispense_in_place.py +++ b/api/src/opentrons/protocol_engine/commands/dispense_in_place.py @@ -1,21 +1,32 @@ """Dispense-in-place command request, result, and implementation models.""" from __future__ import annotations -from typing import TYPE_CHECKING, Optional, Type +from typing import TYPE_CHECKING, Optional, Type, Union from typing_extensions import Literal - from pydantic import Field +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError + from .pipetting_common import ( PipetteIdMixin, DispenseVolumeMixin, FlowRateMixin, BaseLiquidHandlingResult, + OverpressureError, + OverpressureErrorInternalData, +) +from .command import ( + AbstractCommandImpl, + BaseCommand, + BaseCommandCreate, + SuccessData, + DefinedErrorData, ) -from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData from ..errors.error_occurrence import ErrorOccurrence +from ..types import DeckPoint if TYPE_CHECKING: - from ..execution import PipettingHandler + from ..execution import PipettingHandler, GantryMover + from ..resources import ModelUtils DispenseInPlaceCommandType = Literal["dispenseInPlace"] @@ -36,25 +47,72 @@ class DispenseInPlaceResult(BaseLiquidHandlingResult): pass +_ExecuteReturn = Union[ + SuccessData[DispenseInPlaceResult, None], + DefinedErrorData[OverpressureError, OverpressureErrorInternalData], +] + + class DispenseInPlaceImplementation( - AbstractCommandImpl[DispenseInPlaceParams, SuccessData[DispenseInPlaceResult, None]] + AbstractCommandImpl[DispenseInPlaceParams, _ExecuteReturn] ): """DispenseInPlace command implementation.""" - def __init__(self, pipetting: PipettingHandler, **kwargs: object) -> None: + def __init__( + self, + pipetting: PipettingHandler, + gantry_mover: GantryMover, + model_utils: ModelUtils, + **kwargs: object, + ) -> None: self._pipetting = pipetting + self._gantry_mover = gantry_mover + self._model_utils = model_utils - async def execute( - self, params: DispenseInPlaceParams - ) -> SuccessData[DispenseInPlaceResult, None]: + async def execute(self, params: DispenseInPlaceParams) -> _ExecuteReturn: """Dispense without moving the pipette.""" - volume = await self._pipetting.dispense_in_place( - pipette_id=params.pipetteId, - volume=params.volume, - flow_rate=params.flowRate, - push_out=params.pushOut, - ) - return SuccessData(public=DispenseInPlaceResult(volume=volume), private=None) + try: + volume = await self._pipetting.dispense_in_place( + pipette_id=params.pipetteId, + volume=params.volume, + flow_rate=params.flowRate, + push_out=params.pushOut, + ) + except PipetteOverpressureError as e: + current_position = await self._gantry_mover.get_position(params.pipetteId) + return DefinedErrorData( + public=OverpressureError( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + wrappedErrors=[ + ErrorOccurrence.from_failed( + id=self._model_utils.generate_id(), + createdAt=self._model_utils.get_timestamp(), + error=e, + ) + ], + errorInfo=( + { + "retryLocation": ( + current_position.x, + current_position.y, + current_position.z, + ) + } + ), + ), + private=OverpressureErrorInternalData( + position=DeckPoint( + x=current_position.x, + y=current_position.y, + z=current_position.z, + ), + ), + ) + else: + return SuccessData( + public=DispenseInPlaceResult(volume=volume), private=None + ) class DispenseInPlace( diff --git a/api/src/opentrons/protocol_engine/commands/load_pipette.py b/api/src/opentrons/protocol_engine/commands/load_pipette.py index 6e46e00d1d8..ff000a30f0f 100644 --- a/api/src/opentrons/protocol_engine/commands/load_pipette.py +++ b/api/src/opentrons/protocol_engine/commands/load_pipette.py @@ -6,12 +6,12 @@ ) from opentrons_shared_data.pipette.types import PipetteGenerationType from opentrons_shared_data.robot import user_facing_robot_type -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum from pydantic import BaseModel, Field from typing import TYPE_CHECKING, Optional, Type from typing_extensions import Literal -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType from .command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData diff --git a/api/src/opentrons/protocol_engine/commands/pipetting_common.py b/api/src/opentrons/protocol_engine/commands/pipetting_common.py index 408b1c71478..2be1e6f2d54 100644 --- a/api/src/opentrons/protocol_engine/commands/pipetting_common.py +++ b/api/src/opentrons/protocol_engine/commands/pipetting_common.py @@ -2,7 +2,7 @@ from dataclasses import dataclass from opentrons_shared_data.errors import ErrorCodes from pydantic import BaseModel, Field -from typing import Literal, Optional +from typing import Literal, Optional, Tuple, TypedDict from opentrons.protocol_engine.errors.error_occurrence import ErrorOccurrence @@ -123,12 +123,20 @@ class DestinationPositionResult(BaseModel): ) +class ErrorLocationInfo(TypedDict): + """Holds a retry location for in-place error recovery.""" + + retryLocation: Tuple[float, float, float] + + class OverpressureError(ErrorOccurrence): """Returned when sensors detect an overpressure error while moving liquid. - The pipette plunger motion is stopped at the point of the error. The next thing to - move the plunger must be a `home` or `blowout` command; commands like `aspirate` - will return an error. + The pipette plunger motion is stopped at the point of the error. + + The next thing to move the plunger must account for the robot not having a valid + estimate of its position. It should be a `home`, `unsafe/updatePositionEstimators`, + `unsafe/dropTipInPlace`, or `unsafe/blowOutInPlace`. """ isDefined: bool = True @@ -138,6 +146,8 @@ class OverpressureError(ErrorOccurrence): errorCode: str = ErrorCodes.PIPETTE_OVERPRESSURE.value.code detail: str = ErrorCodes.PIPETTE_OVERPRESSURE.value.detail + errorInfo: ErrorLocationInfo + @dataclass(frozen=True) class OverpressureErrorInternalData: diff --git a/api/src/opentrons/protocol_engine/commands/unsafe/__init__.py b/api/src/opentrons/protocol_engine/commands/unsafe/__init__.py new file mode 100644 index 00000000000..2875d38cb8e --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/unsafe/__init__.py @@ -0,0 +1,45 @@ +"""Commands that will cause inaccuracy or incorrect behavior but are still necessary.""" + +from .unsafe_blow_out_in_place import ( + UnsafeBlowOutInPlaceCommandType, + UnsafeBlowOutInPlaceParams, + UnsafeBlowOutInPlaceResult, + UnsafeBlowOutInPlace, + UnsafeBlowOutInPlaceCreate, +) +from .unsafe_drop_tip_in_place import ( + UnsafeDropTipInPlaceCommandType, + UnsafeDropTipInPlaceParams, + UnsafeDropTipInPlaceResult, + UnsafeDropTipInPlace, + UnsafeDropTipInPlaceCreate, +) + +from .update_position_estimators import ( + UpdatePositionEstimatorsCommandType, + UpdatePositionEstimatorsParams, + UpdatePositionEstimatorsResult, + UpdatePositionEstimators, + UpdatePositionEstimatorsCreate, +) + +__all__ = [ + # Unsafe blow-out-in-place command models + "UnsafeBlowOutInPlaceCommandType", + "UnsafeBlowOutInPlaceParams", + "UnsafeBlowOutInPlaceResult", + "UnsafeBlowOutInPlace", + "UnsafeBlowOutInPlaceCreate", + # Unsafe drop-tip command models + "UnsafeDropTipInPlaceCommandType", + "UnsafeDropTipInPlaceParams", + "UnsafeDropTipInPlaceResult", + "UnsafeDropTipInPlace", + "UnsafeDropTipInPlaceCreate", + # Update position estimate command models + "UpdatePositionEstimatorsCommandType", + "UpdatePositionEstimatorsParams", + "UpdatePositionEstimatorsResult", + "UpdatePositionEstimators", + "UpdatePositionEstimatorsCreate", +] diff --git a/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py b/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py new file mode 100644 index 00000000000..cbf17ff1026 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_blow_out_in_place.py @@ -0,0 +1,93 @@ +"""Command models to blow out in place while plunger positions are unknown.""" + +from __future__ import annotations +from typing import TYPE_CHECKING, Optional, Type +from typing_extensions import Literal + +from pydantic import BaseModel + +from ..command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData +from ..pipetting_common import PipetteIdMixin, FlowRateMixin +from ...resources import ensure_ot3_hardware +from ...errors.error_occurrence import ErrorOccurrence + +from opentrons.hardware_control import HardwareControlAPI +from opentrons.hardware_control.types import Axis + +if TYPE_CHECKING: + from ...execution import PipettingHandler + from ...state import StateView + + +UnsafeBlowOutInPlaceCommandType = Literal["unsafe/blowOutInPlace"] + + +class UnsafeBlowOutInPlaceParams(PipetteIdMixin, FlowRateMixin): + """Payload required to blow-out in place while position is unknown.""" + + pass + + +class UnsafeBlowOutInPlaceResult(BaseModel): + """Result data from an UnsafeBlowOutInPlace command.""" + + pass + + +class UnsafeBlowOutInPlaceImplementation( + AbstractCommandImpl[ + UnsafeBlowOutInPlaceParams, SuccessData[UnsafeBlowOutInPlaceResult, None] + ] +): + """UnsafeBlowOutInPlace command implementation.""" + + def __init__( + self, + pipetting: PipettingHandler, + state_view: StateView, + hardware_api: HardwareControlAPI, + **kwargs: object, + ) -> None: + self._pipetting = pipetting + self._state_view = state_view + self._hardware_api = hardware_api + + async def execute( + self, params: UnsafeBlowOutInPlaceParams + ) -> SuccessData[UnsafeBlowOutInPlaceResult, None]: + """Blow-out without moving the pipette even when position is unknown.""" + ot3_hardware_api = ensure_ot3_hardware(self._hardware_api) + pipette_location = self._state_view.motion.get_pipette_location( + params.pipetteId + ) + await ot3_hardware_api.update_axis_position_estimations( + [Axis.of_main_tool_actuator(pipette_location.mount.to_hw_mount())] + ) + await self._pipetting.blow_out_in_place( + pipette_id=params.pipetteId, flow_rate=params.flowRate + ) + + return SuccessData(public=UnsafeBlowOutInPlaceResult(), private=None) + + +class UnsafeBlowOutInPlace( + BaseCommand[UnsafeBlowOutInPlaceParams, UnsafeBlowOutInPlaceResult, ErrorOccurrence] +): + """UnsafeBlowOutInPlace command model.""" + + commandType: UnsafeBlowOutInPlaceCommandType = "unsafe/blowOutInPlace" + params: UnsafeBlowOutInPlaceParams + result: Optional[UnsafeBlowOutInPlaceResult] + + _ImplementationCls: Type[ + UnsafeBlowOutInPlaceImplementation + ] = UnsafeBlowOutInPlaceImplementation + + +class UnsafeBlowOutInPlaceCreate(BaseCommandCreate[UnsafeBlowOutInPlaceParams]): + """UnsafeBlowOutInPlace command request model.""" + + commandType: UnsafeBlowOutInPlaceCommandType = "unsafe/blowOutInPlace" + params: UnsafeBlowOutInPlaceParams + + _CommandCls: Type[UnsafeBlowOutInPlace] = UnsafeBlowOutInPlace diff --git a/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py b/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py new file mode 100644 index 00000000000..2cb3fa78dd8 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/unsafe/unsafe_drop_tip_in_place.py @@ -0,0 +1,98 @@ +"""Command models to drop tip in place while plunger positions are unknown.""" +from __future__ import annotations +from pydantic import Field, BaseModel +from typing import TYPE_CHECKING, Optional, Type +from typing_extensions import Literal + +from opentrons.hardware_control import HardwareControlAPI +from opentrons.hardware_control.types import Axis + +from ..pipetting_common import PipetteIdMixin +from ..command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData +from ...errors.error_occurrence import ErrorOccurrence +from ...resources import ensure_ot3_hardware + +if TYPE_CHECKING: + from ...execution import TipHandler + from ...state import StateView + + +UnsafeDropTipInPlaceCommandType = Literal["unsafe/dropTipInPlace"] + + +class UnsafeDropTipInPlaceParams(PipetteIdMixin): + """Payload required to drop a tip in place even if the plunger position is not known.""" + + homeAfter: Optional[bool] = Field( + None, + description=( + "Whether to home this pipette's plunger after dropping the tip." + " You should normally leave this unspecified to let the robot choose" + " a safe default depending on its hardware." + ), + ) + + +class UnsafeDropTipInPlaceResult(BaseModel): + """Result data from the execution of an UnsafeDropTipInPlace command.""" + + pass + + +class UnsafeDropTipInPlaceImplementation( + AbstractCommandImpl[ + UnsafeDropTipInPlaceParams, SuccessData[UnsafeDropTipInPlaceResult, None] + ] +): + """Unsafe drop tip in place command implementation.""" + + def __init__( + self, + tip_handler: TipHandler, + state_view: StateView, + hardware_api: HardwareControlAPI, + **kwargs: object, + ) -> None: + self._state_view = state_view + self._tip_handler = tip_handler + self._hardware_api = hardware_api + + async def execute( + self, params: UnsafeDropTipInPlaceParams + ) -> SuccessData[UnsafeDropTipInPlaceResult, None]: + """Drop a tip using the requested pipette, even if the plunger position is not known.""" + ot3_hardware_api = ensure_ot3_hardware(self._hardware_api) + pipette_location = self._state_view.motion.get_pipette_location( + params.pipetteId + ) + await ot3_hardware_api.update_axis_position_estimations( + [Axis.of_main_tool_actuator(pipette_location.mount.to_hw_mount())] + ) + await self._tip_handler.drop_tip( + pipette_id=params.pipetteId, home_after=params.homeAfter + ) + + return SuccessData(public=UnsafeDropTipInPlaceResult(), private=None) + + +class UnsafeDropTipInPlace( + BaseCommand[UnsafeDropTipInPlaceParams, UnsafeDropTipInPlaceResult, ErrorOccurrence] +): + """Drop tip in place command model.""" + + commandType: UnsafeDropTipInPlaceCommandType = "unsafe/dropTipInPlace" + params: UnsafeDropTipInPlaceParams + result: Optional[UnsafeDropTipInPlaceResult] + + _ImplementationCls: Type[ + UnsafeDropTipInPlaceImplementation + ] = UnsafeDropTipInPlaceImplementation + + +class UnsafeDropTipInPlaceCreate(BaseCommandCreate[UnsafeDropTipInPlaceParams]): + """Drop tip in place command creation request model.""" + + commandType: UnsafeDropTipInPlaceCommandType = "unsafe/dropTipInPlace" + params: UnsafeDropTipInPlaceParams + + _CommandCls: Type[UnsafeDropTipInPlace] = UnsafeDropTipInPlace diff --git a/api/src/opentrons/protocol_engine/commands/unsafe/update_position_estimators.py b/api/src/opentrons/protocol_engine/commands/unsafe/update_position_estimators.py new file mode 100644 index 00000000000..96be2eb8551 --- /dev/null +++ b/api/src/opentrons/protocol_engine/commands/unsafe/update_position_estimators.py @@ -0,0 +1,87 @@ +"""Update position estimators payload, result, and implementaiton.""" + +from __future__ import annotations +from pydantic import BaseModel, Field +from typing import TYPE_CHECKING, Optional, List, Type +from typing_extensions import Literal + +from ...types import MotorAxis +from ..command import AbstractCommandImpl, BaseCommand, BaseCommandCreate, SuccessData +from ...errors.error_occurrence import ErrorOccurrence +from ...resources import ensure_ot3_hardware + +from opentrons.hardware_control import HardwareControlAPI + +if TYPE_CHECKING: + from ...execution import GantryMover + + +UpdatePositionEstimatorsCommandType = Literal["unsafe/updatePositionEstimators"] + + +class UpdatePositionEstimatorsParams(BaseModel): + """Payload required for an UpdatePositionEstimators command.""" + + axes: List[MotorAxis] = Field( + ..., description="The axes for which to update the position estimators." + ) + + +class UpdatePositionEstimatorsResult(BaseModel): + """Result data from the execution of an UpdatePositionEstimators command.""" + + +class UpdatePositionEstimatorsImplementation( + AbstractCommandImpl[ + UpdatePositionEstimatorsParams, + SuccessData[UpdatePositionEstimatorsResult, None], + ] +): + """Update position estimators command implementation.""" + + def __init__( + self, + hardware_api: HardwareControlAPI, + gantry_mover: GantryMover, + **kwargs: object, + ) -> None: + self._hardware_api = hardware_api + self._gantry_mover = gantry_mover + + async def execute( + self, params: UpdatePositionEstimatorsParams + ) -> SuccessData[UpdatePositionEstimatorsResult, None]: + """Update axis position estimators from their encoders.""" + ot3_hardware_api = ensure_ot3_hardware(self._hardware_api) + await ot3_hardware_api.update_axis_position_estimations( + [ + self._gantry_mover.motor_axis_to_hardware_axis(axis) + for axis in params.axes + ] + ) + return SuccessData(public=UpdatePositionEstimatorsResult(), private=None) + + +class UpdatePositionEstimators( + BaseCommand[ + UpdatePositionEstimatorsParams, UpdatePositionEstimatorsResult, ErrorOccurrence + ] +): + """UpdatePositionEstimators command model.""" + + commandType: UpdatePositionEstimatorsCommandType = "unsafe/updatePositionEstimators" + params: UpdatePositionEstimatorsParams + result: Optional[UpdatePositionEstimatorsResult] + + _ImplementationCls: Type[ + UpdatePositionEstimatorsImplementation + ] = UpdatePositionEstimatorsImplementation + + +class UpdatePositionEstimatorsCreate(BaseCommandCreate[UpdatePositionEstimatorsParams]): + """UpdatePositionEstimators command request model.""" + + commandType: UpdatePositionEstimatorsCommandType = "unsafe/updatePositionEstimators" + params: UpdatePositionEstimatorsParams + + _CommandCls: Type[UpdatePositionEstimators] = UpdatePositionEstimators diff --git a/api/src/opentrons/protocol_engine/create_protocol_engine.py b/api/src/opentrons/protocol_engine/create_protocol_engine.py index 8a6a4355fd7..59f004cc7cd 100644 --- a/api/src/opentrons/protocol_engine/create_protocol_engine.py +++ b/api/src/opentrons/protocol_engine/create_protocol_engine.py @@ -53,6 +53,7 @@ async def create_protocol_engine( deck_fixed_labware=deck_fixed_labware, robot_definition=robot_definition, is_door_open=hardware_api.door_state is DoorState.OPEN, + error_recovery_policy=error_recovery_policy, module_calibration_offsets=module_calibration_offsets, deck_configuration=deck_configuration, notify_publishers=notify_publishers, @@ -61,7 +62,6 @@ async def create_protocol_engine( return ProtocolEngine( state_store=state_store, hardware_api=hardware_api, - error_recovery_policy=error_recovery_policy, ) diff --git a/api/src/opentrons/protocol_engine/error_recovery_policy.py b/api/src/opentrons/protocol_engine/error_recovery_policy.py index f7468961131..f9f39d99f4d 100644 --- a/api/src/opentrons/protocol_engine/error_recovery_policy.py +++ b/api/src/opentrons/protocol_engine/error_recovery_policy.py @@ -28,10 +28,8 @@ class ErrorRecoveryType(enum.Enum): WAIT_FOR_RECOVERY = enum.auto() """Stop and wait for the error to be recovered from manually.""" - # TODO(mm, 2023-03-18): Add something like this for - # https://opentrons.atlassian.net/browse/EXEC-302. - # CONTINUE = enum.auto() - # """Continue with the run, as if the command never failed.""" + IGNORE_AND_CONTINUE = enum.auto() + """Continue with the run, as if the command never failed.""" class ErrorRecoveryPolicy(Protocol): @@ -58,28 +56,6 @@ def __call__( # noqa: D102 ... -# todo(mm, 2024-07-05): This "static" policy will need to somehow become dynamic for -# https://opentrons.atlassian.net/browse/EXEC-589. -def standard_run_policy( - config: Config, - failed_command: Command, - defined_error_data: Optional[CommandDefinedErrorData], -) -> ErrorRecoveryType: - """An error recovery policy suitable for normal protocol runs via robot-server.""" - # Although error recovery can theoretically work on OT-2s, we haven't tested it, - # and it's generally scarier because the OT-2 has much less hardware feedback. - robot_is_flex = config.robot_type == "OT-3 Standard" - # If the error is defined, we're taking that to mean that we should - # WAIT_FOR_RECOVERY. This is not necessarily the right long-term logic--we might - # want to FAIL_RUN on certain defined errors and WAIT_FOR_RECOVERY on certain - # undefined errors--but this is convenient for now. - error_is_defined = defined_error_data is not None - if robot_is_flex and error_is_defined: - return ErrorRecoveryType.WAIT_FOR_RECOVERY - else: - return ErrorRecoveryType.FAIL_RUN - - def never_recover( config: Config, failed_command: Command, diff --git a/api/src/opentrons/protocol_engine/errors/error_occurrence.py b/api/src/opentrons/protocol_engine/errors/error_occurrence.py index d890b121c0f..02bcfb38b62 100644 --- a/api/src/opentrons/protocol_engine/errors/error_occurrence.py +++ b/api/src/opentrons/protocol_engine/errors/error_occurrence.py @@ -3,7 +3,7 @@ from datetime import datetime from textwrap import dedent -from typing import Any, Dict, List, Type, Union, Optional, Sequence +from typing import Any, Dict, Mapping, List, Type, Union, Optional, Sequence from pydantic import BaseModel, Field from opentrons_shared_data.errors.codes import ErrorCodes from .exceptions import ProtocolEngineError @@ -118,7 +118,7 @@ def from_failed( ), ) - errorInfo: Dict[str, str] = Field( + errorInfo: Mapping[str, object] = Field( default={}, description=dedent( """\ diff --git a/api/src/opentrons/protocol_engine/execution/command_executor.py b/api/src/opentrons/protocol_engine/execution/command_executor.py index efcaf847d89..e427b945e0d 100644 --- a/api/src/opentrons/protocol_engine/execution/command_executor.py +++ b/api/src/opentrons/protocol_engine/execution/command_executor.py @@ -12,7 +12,6 @@ ) from opentrons.protocol_engine.commands.command import SuccessData -from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from ..state import StateStore from ..resources import ModelUtils @@ -84,7 +83,6 @@ def __init__( run_control: RunControlHandler, rail_lights: RailLightsHandler, status_bar: StatusBarHandler, - error_recovery_policy: ErrorRecoveryPolicy, model_utils: Optional[ModelUtils] = None, command_note_tracker_provider: Optional[CommandNoteTrackerProvider] = None, ) -> None: @@ -105,7 +103,6 @@ def __init__( self._command_note_tracker_provider = ( command_note_tracker_provider or _NoteTracker ) - self._error_recovery_policy = error_recovery_policy async def execute(self, command_id: str) -> None: """Run a given command's execution procedure. @@ -138,6 +135,7 @@ async def execute(self, command_id: str) -> None: RunCommandAction(command_id=queued_command.id, started_at=started_at) ) running_command = self._state_store.commands.get(queued_command.id) + error_recovery_policy = self._state_store.commands.get_error_recovery_policy() log.debug( f"Executing {running_command.id}, {running_command.commandType}, {running_command.params}" @@ -168,7 +166,7 @@ async def execute(self, command_id: str) -> None: error_id=self._model_utils.generate_id(), failed_at=self._model_utils.get_timestamp(), notes=note_tracker.get_notes(), - type=self._error_recovery_policy( + type=error_recovery_policy( self._state_store.config, running_command, None, @@ -200,7 +198,7 @@ async def execute(self, command_id: str) -> None: error_id=result.public.id, failed_at=result.public.createdAt, notes=note_tracker.get_notes(), - type=self._error_recovery_policy( + type=error_recovery_policy( self._state_store.config, running_command, result, diff --git a/api/src/opentrons/protocol_engine/execution/create_queue_worker.py b/api/src/opentrons/protocol_engine/execution/create_queue_worker.py index c6b00cdf70f..3596ce6d96e 100644 --- a/api/src/opentrons/protocol_engine/execution/create_queue_worker.py +++ b/api/src/opentrons/protocol_engine/execution/create_queue_worker.py @@ -2,7 +2,6 @@ from typing import AsyncGenerator, Callable from opentrons.hardware_control import HardwareControlAPI -from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from opentrons.protocol_engine.execution.rail_lights import RailLightsHandler from ..state import StateStore @@ -23,7 +22,6 @@ def create_queue_worker( hardware_api: HardwareControlAPI, state_store: StateStore, action_dispatcher: ActionDispatcher, - error_recovery_policy: ErrorRecoveryPolicy, command_generator: Callable[[], AsyncGenerator[str, None]], ) -> QueueWorker: """Create a ready-to-use QueueWorker instance. @@ -91,7 +89,6 @@ def create_queue_worker( run_control=run_control_handler, rail_lights=rail_lights_handler, status_bar=status_bar_handler, - error_recovery_policy=error_recovery_policy, ) return QueueWorker( diff --git a/api/src/opentrons/protocol_engine/execution/equipment.py b/api/src/opentrons/protocol_engine/execution/equipment.py index 17e9fbf0ffe..4093c93489c 100644 --- a/api/src/opentrons/protocol_engine/execution/equipment.py +++ b/api/src/opentrons/protocol_engine/execution/equipment.py @@ -2,7 +2,7 @@ from dataclasses import dataclass from typing import Optional, overload, Union -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.calibration_storage.helpers import uri_from_details from opentrons.protocols.models import LabwareDefinition diff --git a/api/src/opentrons/protocol_engine/execution/gantry_mover.py b/api/src/opentrons/protocol_engine/execution/gantry_mover.py index 7e05c8db247..26ab20f69de 100644 --- a/api/src/opentrons/protocol_engine/execution/gantry_mover.py +++ b/api/src/opentrons/protocol_engine/execution/gantry_mover.py @@ -81,6 +81,10 @@ async def prepare_for_mount_movement(self, mount: Mount) -> None: """Retract the 'idle' mount if necessary.""" ... + def motor_axis_to_hardware_axis(self, motor_axis: MotorAxis) -> HardwareAxis: + """Transform an engine motor axis into a hardware axis.""" + ... + class HardwareGantryMover(GantryMover): """Hardware API based gantry movement handler.""" @@ -89,6 +93,10 @@ def __init__(self, hardware_api: HardwareControlAPI, state_view: StateView) -> N self._hardware_api = hardware_api self._state_view = state_view + def motor_axis_to_hardware_axis(self, motor_axis: MotorAxis) -> HardwareAxis: + """Transform an engine motor axis into a hardware axis.""" + return _MOTOR_AXIS_TO_HARDWARE_AXIS[motor_axis] + async def get_position( self, pipette_id: str, @@ -227,6 +235,10 @@ class VirtualGantryMover(GantryMover): def __init__(self, state_view: StateView) -> None: self._state_view = state_view + def motor_axis_to_hardware_axis(self, motor_axis: MotorAxis) -> HardwareAxis: + """Transform an engine motor axis into a hardware axis.""" + return _MOTOR_AXIS_TO_HARDWARE_AXIS[motor_axis] + async def get_position( self, pipette_id: str, diff --git a/api/src/opentrons/protocol_engine/protocol_engine.py b/api/src/opentrons/protocol_engine/protocol_engine.py index d319e00a0b1..2c0b4cc1925 100644 --- a/api/src/opentrons/protocol_engine/protocol_engine.py +++ b/api/src/opentrons/protocol_engine/protocol_engine.py @@ -2,7 +2,10 @@ from contextlib import AsyncExitStack from logging import getLogger from typing import Dict, Optional, Union, AsyncGenerator, Callable -from opentrons.protocol_engine.actions.actions import ResumeFromRecoveryAction +from opentrons.protocol_engine.actions.actions import ( + ResumeFromRecoveryAction, + SetErrorRecoveryPolicyAction, +) from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from opentrons.protocols.models import LabwareDefinition @@ -84,7 +87,6 @@ def __init__( self, hardware_api: HardwareControlAPI, state_store: StateStore, - error_recovery_policy: ErrorRecoveryPolicy, action_dispatcher: Optional[ActionDispatcher] = None, plugin_starter: Optional[PluginStarter] = None, queue_worker: Optional[QueueWorker] = None, @@ -103,7 +105,6 @@ def __init__( self._hardware_api = hardware_api self._state_store = state_store self._model_utils = model_utils or ModelUtils() - self._error_recovery_policy = error_recovery_policy self._action_dispatcher = action_dispatcher or ActionDispatcher( sink=self._state_store ) @@ -605,15 +606,19 @@ def set_and_start_queue_worker( self, command_generator: Callable[[], AsyncGenerator[str, None]] ) -> None: """Set QueueWorker and start it.""" + assert self._queue_worker is None self._queue_worker = create_queue_worker( hardware_api=self._hardware_api, state_store=self._state_store, action_dispatcher=self._action_dispatcher, - error_recovery_policy=self._error_recovery_policy, command_generator=command_generator, ) self._queue_worker.start() + def set_error_recovery_policy(self, policy: ErrorRecoveryPolicy) -> None: + """Replace the run's error recovery policy with a new one.""" + self._action_dispatcher.dispatch(SetErrorRecoveryPolicyAction(policy)) + # TODO(tz, 7-12-23): move this to shared data when we dont relay on ErrorOccurrence def code_in_error_tree( diff --git a/api/src/opentrons/protocol_engine/resources/deck_configuration_provider.py b/api/src/opentrons/protocol_engine/resources/deck_configuration_provider.py index 648bd4f4484..739d56ded00 100644 --- a/api/src/opentrons/protocol_engine/resources/deck_configuration_provider.py +++ b/api/src/opentrons/protocol_engine/resources/deck_configuration_provider.py @@ -1,7 +1,7 @@ """Deck configuration resource provider.""" from typing import List, Set, Tuple -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5, CutoutFixture +from opentrons_shared_data.deck.types import DeckDefinitionV5, CutoutFixture from opentrons.types import DeckSlotName diff --git a/api/src/opentrons/protocol_engine/resources/deck_data_provider.py b/api/src/opentrons/protocol_engine/resources/deck_data_provider.py index 017fc58f552..c373ce766db 100644 --- a/api/src/opentrons/protocol_engine/resources/deck_data_provider.py +++ b/api/src/opentrons/protocol_engine/resources/deck_data_provider.py @@ -9,7 +9,7 @@ load as load_deck, DEFAULT_DECK_DEFINITION_VERSION, ) -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.protocols.models import LabwareDefinition from opentrons.types import DeckSlotName diff --git a/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py b/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py index d453543266d..43b3be16f38 100644 --- a/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py +++ b/api/src/opentrons/protocol_engine/resources/pipette_data_provider.py @@ -3,7 +3,7 @@ from typing import Dict, Optional, Sequence import re -from opentrons_shared_data.pipette.dev_types import PipetteName, PipetteModel +from opentrons_shared_data.pipette.types import PipetteName, PipetteModel from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, load_data as load_pipette_data, diff --git a/api/src/opentrons/protocol_engine/slot_standardization.py b/api/src/opentrons/protocol_engine/slot_standardization.py index c4e733b3ca6..b600258bbf0 100644 --- a/api/src/opentrons/protocol_engine/slot_standardization.py +++ b/api/src/opentrons/protocol_engine/slot_standardization.py @@ -17,7 +17,7 @@ from typing import Any, Callable, Dict, Type -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from . import commands from .types import ( diff --git a/api/src/opentrons/protocol_engine/state/addressable_areas.py b/api/src/opentrons/protocol_engine/state/addressable_areas.py index 7e3a0325ed4..ab9c3d8462d 100644 --- a/api/src/opentrons/protocol_engine/state/addressable_areas.py +++ b/api/src/opentrons/protocol_engine/state/addressable_areas.py @@ -3,8 +3,8 @@ from functools import cached_property from typing import Dict, List, Optional, Set, Union -from opentrons_shared_data.robot.dev_types import RobotType, RobotDefinition -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.robot.types import RobotType, RobotDefinition +from opentrons_shared_data.deck.types import ( DeckDefinitionV5, SlotDefV3, CutoutFixture, diff --git a/api/src/opentrons/protocol_engine/state/commands.py b/api/src/opentrons/protocol_engine/state/commands.py index a558210cbff..9989f9aec01 100644 --- a/api/src/opentrons/protocol_engine/state/commands.py +++ b/api/src/opentrons/protocol_engine/state/commands.py @@ -15,8 +15,12 @@ from opentrons.protocol_engine.actions.actions import ( ResumeFromRecoveryAction, RunCommandAction, + SetErrorRecoveryPolicyAction, +) +from opentrons.protocol_engine.error_recovery_policy import ( + ErrorRecoveryPolicy, + ErrorRecoveryType, ) -from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryType from opentrons.protocol_engine.notes.notes import CommandNote from ..actions import ( @@ -202,6 +206,9 @@ class CommandState: stopped_by_estop: bool """If this is set to True, the engine was stopped by an estop event.""" + error_recovery_policy: ErrorRecoveryPolicy + """See `CommandView.get_error_recovery_policy()`.""" + class CommandStore(HasState[CommandState], HandlesActions): """Command state container for run-level command concerns.""" @@ -213,6 +220,7 @@ def __init__( *, config: Config, is_door_open: bool, + error_recovery_policy: ErrorRecoveryPolicy, ) -> None: """Initialize a CommandStore and its state.""" self._config = config @@ -230,6 +238,7 @@ def __init__( run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=error_recovery_policy, ) def handle_action(self, action: Action) -> None: @@ -257,6 +266,8 @@ def handle_action(self, action: Action) -> None: self._handle_hardware_stopped_action(action) case DoorChangeAction(): self._handle_door_change_action(action) + case SetErrorRecoveryPolicyAction(): + self._handle_set_error_recovery_policy_action(action) case _: pass @@ -337,7 +348,10 @@ def _handle_fail_command_action(self, action: FailCommandAction) -> None: other_command_ids_to_fail = list( self._state.command_history.get_queue_ids() ) - elif action.type == ErrorRecoveryType.WAIT_FOR_RECOVERY: + elif ( + action.type == ErrorRecoveryType.WAIT_FOR_RECOVERY + or action.type == ErrorRecoveryType.IGNORE_AND_CONTINUE + ): other_command_ids_to_fail = [] else: assert_never(action.type) @@ -456,6 +470,11 @@ def _handle_door_change_action(self, action: DoorChangeAction) -> None: elif action.door_state == DoorState.CLOSED: self._state.is_door_blocking = False + def _handle_set_error_recovery_policy_action( + self, action: SetErrorRecoveryPolicyAction + ) -> None: + self._state.error_recovery_policy = action.error_recovery_policy + def _update_to_failed( self, command_id: str, @@ -982,3 +1001,12 @@ def get_status(self) -> EngineStatus: # noqa: C901 def get_latest_protocol_command_hash(self) -> Optional[str]: """Get the command hash of the last queued command, if any.""" return self._state.latest_protocol_command_hash + + def get_error_recovery_policy(self) -> ErrorRecoveryPolicy: + """Return the run's current error recovery policy (see `ErrorRecoveryPolicy`). + + This error recovery policy is not ever evaluated by + `CommandStore`/`CommandView`. It's stored here for convenience, but evaluated by + higher-level code. + """ + return self._state.error_recovery_policy diff --git a/api/src/opentrons/protocol_engine/state/config.py b/api/src/opentrons/protocol_engine/state/config.py index c5ba5fb07db..74f1b038e6e 100644 --- a/api/src/opentrons/protocol_engine/state/config.py +++ b/api/src/opentrons/protocol_engine/state/config.py @@ -1,7 +1,7 @@ """Top-level ProtocolEngine configuration options.""" from dataclasses import dataclass -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocol_engine.types import DeckType diff --git a/api/src/opentrons/protocol_engine/state/geometry.py b/api/src/opentrons/protocol_engine/state/geometry.py index 904e0c470b2..7b02e1242da 100644 --- a/api/src/opentrons/protocol_engine/state/geometry.py +++ b/api/src/opentrons/protocol_engine/state/geometry.py @@ -9,9 +9,9 @@ from opentrons.types import Point, DeckSlotName, StagingSlotName, MountType from opentrons_shared_data.labware.constants import WELL_NAME_PATTERN -from opentrons_shared_data.deck.dev_types import CutoutFixture +from opentrons_shared_data.deck.types import CutoutFixture from opentrons_shared_data.pipette import PIPETTE_X_SPAN -from opentrons_shared_data.pipette.dev_types import ChannelCount +from opentrons_shared_data.pipette.types import ChannelCount from .. import errors from ..errors import ( diff --git a/api/src/opentrons/protocol_engine/state/labware.py b/api/src/opentrons/protocol_engine/state/labware.py index e9750a652b4..96eb1dac23b 100644 --- a/api/src/opentrons/protocol_engine/state/labware.py +++ b/api/src/opentrons/protocol_engine/state/labware.py @@ -15,10 +15,10 @@ Union, ) -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data.gripper.constants import LABWARE_GRIP_FORCE from opentrons_shared_data.labware.labware_definition import LabwareRole -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons.types import DeckSlotName, StagingSlotName, MountType from opentrons.protocols.api_support.constants import OPENTRONS_NAMESPACE diff --git a/api/src/opentrons/protocol_engine/state/pipettes.py b/api/src/opentrons/protocol_engine/state/pipettes.py index 92344dd9600..60720c917ec 100644 --- a/api/src/opentrons/protocol_engine/state/pipettes.py +++ b/api/src/opentrons/protocol_engine/state/pipettes.py @@ -13,6 +13,9 @@ ) from opentrons.protocol_engine.actions.actions import FailCommandAction from opentrons.protocol_engine.commands.aspirate import Aspirate +from opentrons.protocol_engine.commands.dispense import Dispense +from opentrons.protocol_engine.commands.aspirate_in_place import AspirateInPlace +from opentrons.protocol_engine.commands.dispense_in_place import DispenseInPlace from opentrons.protocol_engine.commands.command import DefinedErrorData from opentrons.protocol_engine.commands.pipetting_common import ( OverpressureError, @@ -50,6 +53,7 @@ RetractAxisResult, BlowOutResult, BlowOutInPlaceResult, + unsafe, TouchTipResult, thermocycler, heater_shaker, @@ -275,7 +279,10 @@ def _handle_command( # noqa: C901 default_dispense=tip_configuration.default_dispense_flowrate.values_by_api_level, ) - elif isinstance(command.result, (DropTipResult, DropTipInPlaceResult)): + elif isinstance( + command.result, + (DropTipResult, DropTipInPlaceResult, unsafe.UnsafeDropTipInPlaceResult), + ): pipette_id = command.params.pipetteId self._state.aspirated_volume_by_id[pipette_id] = None self._state.attached_tip_by_id[pipette_id] = None @@ -316,7 +323,7 @@ def _update_current_location( # noqa: C901 ) elif ( isinstance(action, FailCommandAction) - and isinstance(action.running_command, Aspirate) + and isinstance(action.running_command, (Aspirate, Dispense)) and isinstance(action.error, DefinedErrorData) and isinstance(action.error.public, OverpressureError) ): @@ -412,7 +419,10 @@ def _update_deck_point( ) elif ( isinstance(action, FailCommandAction) - and isinstance(action.running_command, Aspirate) + and isinstance( + action.running_command, + (Aspirate, Dispense, AspirateInPlace, DispenseInPlace), + ) and isinstance(action.error, DefinedErrorData) and isinstance(action.error.public, OverpressureError) ): @@ -477,7 +487,8 @@ def _update_volumes( self._state.aspirated_volume_by_id[pipette_id] = next_volume elif isinstance(action, SucceedCommandAction) and isinstance( - action.command.result, (BlowOutResult, BlowOutInPlaceResult) + action.command.result, + (BlowOutResult, BlowOutInPlaceResult, unsafe.UnsafeBlowOutInPlaceResult), ): pipette_id = action.command.params.pipetteId self._state.aspirated_volume_by_id[pipette_id] = None diff --git a/api/src/opentrons/protocol_engine/state/state.py b/api/src/opentrons/protocol_engine/state/state.py index e343a4dfde1..430ca1e5738 100644 --- a/api/src/opentrons/protocol_engine/state/state.py +++ b/api/src/opentrons/protocol_engine/state/state.py @@ -5,9 +5,10 @@ from typing import Callable, Dict, List, Optional, Sequence, TypeVar from typing_extensions import ParamSpec -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 -from opentrons_shared_data.robot.dev_types import RobotDefinition +from opentrons_shared_data.deck.types import DeckDefinitionV5 +from opentrons_shared_data.robot.types import RobotDefinition +from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from opentrons.protocol_engine.types import ModuleOffsetData from opentrons.util.change_notifier import ChangeNotifier @@ -147,6 +148,7 @@ def __init__( deck_fixed_labware: Sequence[DeckFixedLabware], robot_definition: RobotDefinition, is_door_open: bool, + error_recovery_policy: ErrorRecoveryPolicy, change_notifier: Optional[ChangeNotifier] = None, module_calibration_offsets: Optional[Dict[str, ModuleOffsetData]] = None, deck_configuration: Optional[DeckConfigurationType] = None, @@ -161,13 +163,18 @@ def __init__( deck_fixed_labware: Labware definitions from the deck definition to preload into labware state. is_door_open: Whether the robot's door is currently open. + error_recovery_policy: The run's initial error recovery policy. change_notifier: Internal state change notifier. module_calibration_offsets: Module offsets to preload. deck_configuration: The initial deck configuration the addressable area store will be instantiated with. robot_definition: Static information about the robot type being used. notify_publishers: Notifies robot server publishers of internal state change. """ - self._command_store = CommandStore(config=config, is_door_open=is_door_open) + self._command_store = CommandStore( + config=config, + is_door_open=is_door_open, + error_recovery_policy=error_recovery_policy, + ) self._pipette_store = PipetteStore() if deck_configuration is None: deck_configuration = [] diff --git a/api/src/opentrons/protocol_engine/state/tips.py b/api/src/opentrons/protocol_engine/state/tips.py index 85d437888fb..9911b1f85b3 100644 --- a/api/src/opentrons/protocol_engine/state/tips.py +++ b/api/src/opentrons/protocol_engine/state/tips.py @@ -17,6 +17,7 @@ PickUpTipResult, DropTipResult, DropTipInPlaceResult, + unsafe, ) from ..commands.configuring_common import ( PipetteConfigUpdateResultMixin, @@ -126,7 +127,10 @@ def _handle_succeeded_command(self, command: Command) -> None: ) self._state.length_by_pipette_id[pipette_id] = length - elif isinstance(command.result, (DropTipResult, DropTipInPlaceResult)): + elif isinstance( + command.result, + (DropTipResult, DropTipInPlaceResult, unsafe.UnsafeDropTipInPlaceResult), + ): pipette_id = command.params.pipetteId self._state.length_by_pipette_id.pop(pipette_id, None) diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index 17a18a8ae4f..9da73149043 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -13,10 +13,20 @@ StrictStr, validator, ) -from typing import Optional, Union, List, Dict, Any, NamedTuple, Tuple, FrozenSet +from typing import ( + Optional, + Union, + List, + Dict, + Any, + NamedTuple, + Tuple, + FrozenSet, + Mapping, +) from typing_extensions import Literal, TypeGuard -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType, DeckSlotName, StagingSlotName from opentrons.hardware_control.types import ( TipStateType as HwTipStateType, @@ -26,11 +36,11 @@ ModuleType as ModuleType, ) -from opentrons_shared_data.pipette.dev_types import ( # noqa: F401 +from opentrons_shared_data.pipette.types import ( # noqa: F401 # convenience re-export of LabwareUri type LabwareUri as LabwareUri, ) -from opentrons_shared_data.module.dev_types import ModuleType as SharedDataModuleType +from opentrons_shared_data.module.types import ModuleType as SharedDataModuleType # todo(mm, 2024-06-24): This monolithic status field is getting to be a bit much. @@ -347,7 +357,7 @@ class MotorAxis(str, Enum): EXTENSION_JAW = "extensionJaw" -# TODO(mc, 2022-01-18): use opentrons_shared_data.module.dev_types.ModuleModel +# TODO(mc, 2022-01-18): use opentrons_shared_data.module.types.ModuleModel class ModuleModel(str, Enum): """All available modules' models.""" @@ -1029,13 +1039,14 @@ class EnumParameter(RTPBase): ) -class FileId(BaseModel): +class FileInfo(BaseModel): """A file UUID descriptor.""" id: str = Field( ..., description="The UUID identifier of the file stored on the robot.", ) + name: str = Field(..., description="Name of the file, including the extension.") class CSVParameter(RTPBase): @@ -1044,15 +1055,17 @@ class CSVParameter(RTPBase): type: Literal["csv_file"] = Field( default="csv_file", description="String specifying the type of this parameter" ) - file: Optional[FileId] = Field( - ..., - description="The CSV file stored on the robot, to be used as the CSV RTP override value." - " For local analysis this will be empty.", + file: Optional[FileInfo] = Field( + default=None, + description="ID of the CSV file stored on the robot; to be used for fetching the CSV file." + " For local analysis this will most likely be empty.", ) RunTimeParameter = Union[NumberParameter, EnumParameter, BooleanParameter, CSVParameter] -RunTimeParamValuesType = Dict[ +PrimitiveRunTimeParamValuesType = Mapping[ StrictStr, Union[StrictInt, StrictFloat, StrictBool, StrictStr] ] # update value types as more RTP types are added + +CSVRunTimeParamFilesType = Mapping[StrictStr, StrictStr] diff --git a/api/src/opentrons/protocol_reader/file_identifier.py b/api/src/opentrons/protocol_reader/file_identifier.py index 2ef902b7c4c..e5eb971e17e 100644 --- a/api/src/opentrons/protocol_reader/file_identifier.py +++ b/api/src/opentrons/protocol_reader/file_identifier.py @@ -6,7 +6,7 @@ import anyio -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.errors.exceptions import EnumeratedError, PythonException from opentrons.protocols.api_support.definitions import MAX_SUPPORTED_VERSION diff --git a/api/src/opentrons/protocol_reader/protocol_source.py b/api/src/opentrons/protocol_reader/protocol_source.py index ab1aa21e375..8280f9597f4 100644 --- a/api/src/opentrons/protocol_reader/protocol_source.py +++ b/api/src/opentrons/protocol_reader/protocol_source.py @@ -7,7 +7,7 @@ from opentrons.protocols.api_support.types import APIVersion -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType class ProtocolType(str, Enum): diff --git a/api/src/opentrons/protocol_runner/create_simulating_orchestrator.py b/api/src/opentrons/protocol_runner/create_simulating_orchestrator.py index b6e41f020e7..0aa5114b5a5 100644 --- a/api/src/opentrons/protocol_runner/create_simulating_orchestrator.py +++ b/api/src/opentrons/protocol_runner/create_simulating_orchestrator.py @@ -11,7 +11,7 @@ from opentrons.protocol_engine.create_protocol_engine import create_protocol_engine from opentrons.protocol_reader.protocol_source import ProtocolConfig -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from .python_protocol_wrappers import SimulatingContextCreator from .run_orchestrator import RunOrchestrator diff --git a/api/src/opentrons/protocol_runner/json_translator.py b/api/src/opentrons/protocol_runner/json_translator.py index c210d51ec77..65410662e77 100644 --- a/api/src/opentrons/protocol_runner/json_translator.py +++ b/api/src/opentrons/protocol_runner/json_translator.py @@ -2,7 +2,7 @@ from typing import cast, List, Union from pydantic import parse_obj_as -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.protocol.models import ( ProtocolSchemaV6, protocol_schema_v6, diff --git a/api/src/opentrons/protocol_runner/legacy_command_mapper.py b/api/src/opentrons/protocol_runner/legacy_command_mapper.py index 021d03de809..b744c03351c 100644 --- a/api/src/opentrons/protocol_runner/legacy_command_mapper.py +++ b/api/src/opentrons/protocol_runner/legacy_command_mapper.py @@ -11,7 +11,7 @@ ThermocyclerModuleModel, HeaterShakerModuleModel, ) -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType, DeckSlotName, Location from opentrons.legacy_commands import types as legacy_command_types from opentrons.protocol_api import InstrumentContext diff --git a/api/src/opentrons/protocol_runner/protocol_runner.py b/api/src/opentrons/protocol_runner/protocol_runner.py index 54e1c3411ef..bfe959ca0eb 100644 --- a/api/src/opentrons/protocol_runner/protocol_runner.py +++ b/api/src/opentrons/protocol_runner/protocol_runner.py @@ -43,7 +43,8 @@ PostRunHardwareState, DeckConfigurationType, RunTimeParameter, - RunTimeParamValuesType, + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, ) from ..protocols.types import PythonProtocol @@ -130,7 +131,7 @@ async def run( self, deck_configuration: DeckConfigurationType, protocol_source: Optional[ProtocolSource] = None, - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, ) -> RunResult: """Run a given protocol to completion.""" @@ -184,7 +185,8 @@ async def load( self, protocol_source: ProtocolSource, python_parse_mode: PythonParseMode, - run_time_param_values: Optional[RunTimeParamValuesType], + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_files: Optional[CSVRunTimeParamFilesType], ) -> None: """Load a Python or JSONv5(& older) ProtocolSource into managed ProtocolEngine.""" labware_definitions = await protocol_reader.extract_labware_definitions( @@ -207,6 +209,7 @@ async def load( protocol=protocol, parameter_context=self._parameter_context, run_time_param_overrides=run_time_param_values, + run_time_param_file_overrides=run_time_param_files, ) ) else: @@ -250,7 +253,8 @@ async def run( # noqa: D102 self, deck_configuration: DeckConfigurationType, protocol_source: Optional[ProtocolSource] = None, - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, + run_time_param_files: Optional[CSVRunTimeParamFilesType] = None, python_parse_mode: PythonParseMode = PythonParseMode.NORMAL, ) -> RunResult: # TODO(mc, 2022-01-11): move load to runner creation, remove from `run` @@ -260,6 +264,7 @@ async def run( # noqa: D102 protocol_source=protocol_source, python_parse_mode=python_parse_mode, run_time_param_values=run_time_param_values, + run_time_param_files=run_time_param_files, ) self.play(deck_configuration=deck_configuration) @@ -361,7 +366,7 @@ async def run( # noqa: D102 self, deck_configuration: DeckConfigurationType, protocol_source: Optional[ProtocolSource] = None, - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, ) -> RunResult: # TODO(mc, 2022-01-11): move load to runner creation, remove from `run` # currently `protocol_source` arg is only used by tests @@ -386,13 +391,15 @@ async def _add_and_execute_commands(self) -> None: ) ) if executed_command.error is not None: - error_was_recovered_from = ( + error_recovery_type = ( self._protocol_engine.state_view.commands.get_error_recovery_type( executed_command.id ) - == ErrorRecoveryType.WAIT_FOR_RECOVERY ) - if not error_was_recovered_from: + error_should_fail_run = ( + error_recovery_type == ErrorRecoveryType.FAIL_RUN + ) + if error_should_fail_run: raise ProtocolCommandFailedError( original_error=executed_command.error, message=f"{executed_command.error.errorType}: {executed_command.error.detail}", @@ -433,7 +440,7 @@ async def run( # noqa: D102 self, deck_configuration: DeckConfigurationType, protocol_source: Optional[ProtocolSource] = None, - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, ) -> RunResult: assert protocol_source is None await self._hardware_api.home() diff --git a/api/src/opentrons/protocol_runner/python_protocol_wrappers.py b/api/src/opentrons/protocol_runner/python_protocol_wrappers.py index 4a022ca184f..e1090d98fa4 100644 --- a/api/src/opentrons/protocol_runner/python_protocol_wrappers.py +++ b/api/src/opentrons/protocol_runner/python_protocol_wrappers.py @@ -4,7 +4,7 @@ from anyio import to_thread -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionTypedDict, ) from opentrons_shared_data.labware.labware_definition import LabwareDefinition @@ -13,7 +13,10 @@ from opentrons.hardware_control import HardwareControlAPI from opentrons.legacy_broker import LegacyBroker from opentrons.protocol_engine import ProtocolEngine -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, +) from opentrons.protocol_reader import ProtocolSource, ProtocolFileRole from opentrons.util.broker import Broker @@ -161,11 +164,13 @@ async def execute( def extract_run_parameters( protocol: PythonProtocol, parameter_context: ParameterContext, - run_time_param_overrides: Optional[RunTimeParamValuesType], + run_time_param_overrides: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_file_overrides: Optional[CSVRunTimeParamFilesType], ) -> Optional[Parameters]: """Extract the parameters defined in the protocol, overridden with values for the run.""" return exec_add_parameters( protocol=protocol, parameter_context=parameter_context, run_time_param_overrides=run_time_param_overrides, + run_time_param_file_overrides=run_time_param_file_overrides, ) diff --git a/api/src/opentrons/protocol_runner/run_orchestrator.py b/api/src/opentrons/protocol_runner/run_orchestrator.py index 18ad48b6d5b..fe5cf4483f6 100644 --- a/api/src/opentrons/protocol_runner/run_orchestrator.py +++ b/api/src/opentrons/protocol_runner/run_orchestrator.py @@ -6,10 +6,10 @@ from anyio import move_on_after -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition from opentrons_shared_data.errors import GeneralError -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from . import protocol_runner, RunResult, JsonRunner, PythonAndLegacyRunner from ..hardware_control import HardwareControlAPI @@ -31,8 +31,11 @@ LabwareOffset, DeckConfigurationType, RunTimeParameter, - RunTimeParamValuesType, + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, ) +from ..protocol_engine.error_recovery_policy import ErrorRecoveryPolicy + from ..protocol_reader import JsonProtocolConfig, PythonProtocolConfig, ProtocolSource from ..protocols.parse import PythonParseMode @@ -168,7 +171,7 @@ async def run( self, deck_configuration: DeckConfigurationType, protocol_source: Optional[ProtocolSource] = None, - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, ) -> RunResult: """Start the run.""" if self._protocol_runner: @@ -227,7 +230,20 @@ def get_loaded_labware_definitions(self) -> List[LabwareDefinition]: return self._protocol_engine.state_view.labware.get_loaded_labware_definitions() def get_run_time_parameters(self) -> List[RunTimeParameter]: - """Parameter definitions defined by protocol, if any. Will always be empty before execution.""" + """Get the list of run time parameters defined in the protocol, if any. + + This returns a list of all run time parameters with their validated definitions + and client-requested values. Will always be empty before loading the runner. + + If there was an error during RTP definition validation, then this list will + contain the parameter definitions that were validated before the error occurred. + These parameters' values will be default values. + + If all definitions validated successfully but an error occurred while + setting the RTP values with those sent by the client, then only the parameters + whose values were successfully set will have the client-requested values while + the others will contain the default values. + """ return ( [] if self._protocol_runner is None @@ -323,7 +339,8 @@ def get_protocol_runner(self) -> Optional[Union[JsonRunner, PythonAndLegacyRunne async def load( self, protocol_source: ProtocolSource, - run_time_param_values: Optional[RunTimeParamValuesType], + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_files: Optional[CSVRunTimeParamFilesType], parse_mode: ParseMode, ) -> None: """Load a json/python protocol.""" @@ -339,6 +356,7 @@ async def load( # doesn't conform to the new rules. python_parse_mode=python_parse_mode, run_time_param_values=run_time_param_values, + run_time_param_files=run_time_param_files, ) def get_is_okay_to_clear(self) -> bool: @@ -357,6 +375,10 @@ def get_deck_type(self) -> DeckType: """Get engine deck type.""" return self._protocol_engine.state_view.config.deck_type + def set_error_recovery_policy(self, policy: ErrorRecoveryPolicy) -> None: + """Create error recovery policy for the run.""" + self._protocol_engine.set_error_recovery_policy(policy) + async def command_generator(self) -> AsyncGenerator[str, None]: """Yield next command to execute.""" while True: diff --git a/api/src/opentrons/protocols/api_support/deck_type.py b/api/src/opentrons/protocols/api_support/deck_type.py index 4bd70c5fc28..c2c573bf711 100644 --- a/api/src/opentrons/protocols/api_support/deck_type.py +++ b/api/src/opentrons/protocols/api_support/deck_type.py @@ -1,6 +1,6 @@ from typing import Sequence, Dict, Optional, Any -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.errors import ErrorCodes from opentrons_shared_data.errors.exceptions import EnumeratedError diff --git a/api/src/opentrons/protocols/api_support/instrument.py b/api/src/opentrons/protocols/api_support/instrument.py index d6d9613b1cf..0137b43a4c8 100644 --- a/api/src/opentrons/protocols/api_support/instrument.py +++ b/api/src/opentrons/protocols/api_support/instrument.py @@ -1,7 +1,7 @@ import logging from typing import Optional, Any -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionDict, ) @@ -16,7 +16,7 @@ ) from opentrons.protocol_api.labware import Labware from opentrons.protocols.api_support.types import APIVersion -from opentrons_shared_data.protocol.dev_types import ( +from opentrons_shared_data.protocol.types import ( LiquidHandlingCommand, BlowoutLocation, ) diff --git a/api/src/opentrons/protocols/api_support/util.py b/api/src/opentrons/protocols/api_support/util.py index e1eb1195b12..00d66330ae2 100644 --- a/api/src/opentrons/protocols/api_support/util.py +++ b/api/src/opentrons/protocols/api_support/util.py @@ -21,7 +21,7 @@ from opentrons.protocols.api_support.types import APIVersion from opentrons.hardware_control.types import Axis from opentrons.hardware_control.util import ot2_axis_to_string -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.errors.exceptions import ( APIRemoved, IncorrectAPIVersion, diff --git a/api/src/opentrons/protocols/bundle.py b/api/src/opentrons/protocols/bundle.py index 6cba5b2d2ac..859d37a87c8 100644 --- a/api/src/opentrons/protocols/bundle.py +++ b/api/src/opentrons/protocols/bundle.py @@ -12,7 +12,7 @@ from .types import BundleContents if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.labware.types import LabwareDefinition MAIN_PROTOCOL_FILENAME = "protocol.ot2.py" LABWARE_DIR = "labware" diff --git a/api/src/opentrons/protocols/execution/dev_types.py b/api/src/opentrons/protocols/execution/dev_types.py index 241fda93751..7c40339278a 100644 --- a/api/src/opentrons/protocols/execution/dev_types.py +++ b/api/src/opentrons/protocols/execution/dev_types.py @@ -2,7 +2,7 @@ from typing_extensions import Protocol, TypedDict -from opentrons_shared_data.protocol.dev_types import ( +from opentrons_shared_data.protocol.types import ( BlowoutParams, DelayParams, PipetteAccessParams, diff --git a/api/src/opentrons/protocols/execution/execute_json_v3.py b/api/src/opentrons/protocols/execution/execute_json_v3.py index 44084d302f0..e119db5575e 100644 --- a/api/src/opentrons/protocols/execution/execute_json_v3.py +++ b/api/src/opentrons/protocols/execution/execute_json_v3.py @@ -11,7 +11,7 @@ if TYPE_CHECKING: - from opentrons_shared_data.protocol.dev_types import ( + from opentrons_shared_data.protocol.types import ( JsonProtocolV3, JsonProtocol, PipetteAccessParams, diff --git a/api/src/opentrons/protocols/execution/execute_json_v4.py b/api/src/opentrons/protocols/execution/execute_json_v4.py index 97a2b93456c..6060e82cb47 100644 --- a/api/src/opentrons/protocols/execution/execute_json_v4.py +++ b/api/src/opentrons/protocols/execution/execute_json_v4.py @@ -17,7 +17,7 @@ if TYPE_CHECKING: - from opentrons_shared_data.protocol.dev_types import ( + from opentrons_shared_data.protocol.types import ( JsonProtocolV4, JsonProtocolV5, MagneticModuleEngageParams, diff --git a/api/src/opentrons/protocols/execution/execute_json_v5.py b/api/src/opentrons/protocols/execution/execute_json_v5.py index b54e5f1d263..fd8bcaa367f 100644 --- a/api/src/opentrons/protocols/execution/execute_json_v5.py +++ b/api/src/opentrons/protocols/execution/execute_json_v5.py @@ -5,7 +5,7 @@ from .execute_json_v3 import _get_well if TYPE_CHECKING: - from opentrons_shared_data.protocol.dev_types import MoveToWellParams + from opentrons_shared_data.protocol.types import MoveToWellParams MODULE_LOG = logging.getLogger(__name__) diff --git a/api/src/opentrons/protocols/execution/execute_python.py b/api/src/opentrons/protocols/execution/execute_python.py index 5a1e945a5a2..59c9db943df 100644 --- a/api/src/opentrons/protocols/execution/execute_python.py +++ b/api/src/opentrons/protocols/execution/execute_python.py @@ -10,7 +10,10 @@ from opentrons.protocol_api._parameters import Parameters from opentrons.protocols.execution.errors import ExceptionInProtocolError from opentrons.protocols.types import PythonProtocol, MalformedPythonProtocolError -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, +) from opentrons_shared_data.errors.exceptions import ExecutionCancelledError @@ -67,7 +70,8 @@ def _raise_pretty_protocol_error(exception: Exception, filename: str) -> None: def _parse_and_set_parameters( parameter_context: ParameterContext, - run_time_param_overrides: Optional[RunTimeParamValuesType], + run_time_param_overrides: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_file_overrides: Optional[CSVRunTimeParamFilesType], new_globs: Dict[Any, Any], filename: str, ) -> Parameters: @@ -80,6 +84,8 @@ def _parse_and_set_parameters( exec("add_parameters(__param_context)", new_globs) if run_time_param_overrides is not None: parameter_context.set_parameters(run_time_param_overrides) + if run_time_param_file_overrides is not None: + parameter_context.initialize_csv_files(run_time_param_file_overrides) except Exception as e: _raise_pretty_protocol_error(exception=e, filename=filename) return parameter_context.export_parameters_for_protocol() @@ -104,7 +110,8 @@ def _get_filename( def exec_add_parameters( protocol: PythonProtocol, parameter_context: ParameterContext, - run_time_param_overrides: Optional[RunTimeParamValuesType], + run_time_param_overrides: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_file_overrides: Optional[CSVRunTimeParamFilesType], ) -> Optional[Parameters]: """Exec the add_parameters function and get the final run time parameters with overrides.""" new_globs: Dict[Any, Any] = {} @@ -115,6 +122,7 @@ def exec_add_parameters( _parse_and_set_parameters( parameter_context=parameter_context, run_time_param_overrides=run_time_param_overrides, + run_time_param_file_overrides=run_time_param_file_overrides, new_globs=new_globs, filename=filename, ) diff --git a/api/src/opentrons/protocols/geometry/labware_geometry.py b/api/src/opentrons/protocols/geometry/labware_geometry.py index 44e13c14410..c19b66761af 100644 --- a/api/src/opentrons/protocols/geometry/labware_geometry.py +++ b/api/src/opentrons/protocols/geometry/labware_geometry.py @@ -1,7 +1,7 @@ from abc import ABC, abstractmethod from opentrons.types import Location, Point -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition class AbstractLabwareGeometry(ABC): diff --git a/api/src/opentrons/protocols/labware.py b/api/src/opentrons/protocols/labware.py index 6e2a5998090..06c157404b6 100644 --- a/api/src/opentrons/protocols/labware.py +++ b/api/src/opentrons/protocols/labware.py @@ -17,7 +17,7 @@ STANDARD_DEFS_PATH, USER_DEFS_PATH, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition MODULE_LOG = logging.getLogger(__name__) diff --git a/api/src/opentrons/protocols/models/json_protocol.py b/api/src/opentrons/protocols/models/json_protocol.py index 6cd7c32aa2d..979d0192f62 100644 --- a/api/src/opentrons/protocols/models/json_protocol.py +++ b/api/src/opentrons/protocols/models/json_protocol.py @@ -13,61 +13,59 @@ from typing_extensions import Literal from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.protocol import dev_types - -CommandAspirate: dev_types.AspirateCommandId = "aspirate" -CommandDispense: dev_types.DispenseCommandId = "dispense" -CommandAirGap: dev_types.AirGapCommandId = "airGap" -CommandBlowout: dev_types.BlowoutCommandId = "blowout" -CommandTouchTip: dev_types.TouchTipCommandId = "touchTip" -CommandPickUpTip: dev_types.PickUpTipCommandId = "pickUpTip" -CommandDropTip: dev_types.DropTipCommandId = "dropTip" -CommandMoveToSlot: dev_types.MoveToSlotCommandId = "moveToSlot" -CommandMoveToWell: dev_types.MoveToWellCommandId = "moveToWell" -CommandDelay: dev_types.DelayCommandId = "delay" -CommandMagneticModuleEngage: dev_types.MagneticModuleEngageCommandId = ( +from opentrons_shared_data.protocol import types + +CommandAspirate: types.AspirateCommandId = "aspirate" +CommandDispense: types.DispenseCommandId = "dispense" +CommandAirGap: types.AirGapCommandId = "airGap" +CommandBlowout: types.BlowoutCommandId = "blowout" +CommandTouchTip: types.TouchTipCommandId = "touchTip" +CommandPickUpTip: types.PickUpTipCommandId = "pickUpTip" +CommandDropTip: types.DropTipCommandId = "dropTip" +CommandMoveToSlot: types.MoveToSlotCommandId = "moveToSlot" +CommandMoveToWell: types.MoveToWellCommandId = "moveToWell" +CommandDelay: types.DelayCommandId = "delay" +CommandMagneticModuleEngage: types.MagneticModuleEngageCommandId = ( "magneticModule/engageMagnet" ) -CommandMagneticModuleDisengage: dev_types.MagneticModuleDisengageCommandId = ( +CommandMagneticModuleDisengage: types.MagneticModuleDisengageCommandId = ( "magneticModule/disengageMagnet" ) -CommandTemperatureModuleSetTarget: dev_types.TemperatureModuleSetTargetCommandId = ( +CommandTemperatureModuleSetTarget: types.TemperatureModuleSetTargetCommandId = ( "temperatureModule/setTargetTemperature" ) -CommandTemperatureModuleAwait: dev_types.TemperatureModuleAwaitCommandId = ( +CommandTemperatureModuleAwait: types.TemperatureModuleAwaitCommandId = ( "temperatureModule/awaitTemperature" ) -CommandTemperatureModuleDeactivate: dev_types.TemperatureModuleDeactivateCommandId = ( +CommandTemperatureModuleDeactivate: types.TemperatureModuleDeactivateCommandId = ( "temperatureModule/deactivate" ) -CommandThermocyclerSetTargetBlock: dev_types.ThermocyclerSetTargetBlockCommandId = ( +CommandThermocyclerSetTargetBlock: types.ThermocyclerSetTargetBlockCommandId = ( "thermocycler/setTargetBlockTemperature" ) -CommandThermocyclerSetTargetLid: dev_types.ThermocyclerSetTargetLidCommandId = ( +CommandThermocyclerSetTargetLid: types.ThermocyclerSetTargetLidCommandId = ( "thermocycler/setTargetLidTemperature" ) -CommandThermocyclerAwaitLidTemperature: dev_types.ThermocyclerAwaitLidTemperatureCommandId = ( +CommandThermocyclerAwaitLidTemperature: types.ThermocyclerAwaitLidTemperatureCommandId = ( "thermocycler/awaitLidTemperature" ) -CommandThermocyclerAwaitBlockTemperature: dev_types.ThermocyclerAwaitBlockTemperatureCommandId = ( +CommandThermocyclerAwaitBlockTemperature: types.ThermocyclerAwaitBlockTemperatureCommandId = ( "thermocycler/awaitBlockTemperature" ) -CommandThermocyclerDeactivateBlock: dev_types.ThermocyclerDeactivateBlockCommandId = ( +CommandThermocyclerDeactivateBlock: types.ThermocyclerDeactivateBlockCommandId = ( "thermocycler/deactivateBlock" ) -CommandThermocyclerDeactivateLid: dev_types.ThermocyclerDeactivateLidCommandId = ( +CommandThermocyclerDeactivateLid: types.ThermocyclerDeactivateLidCommandId = ( "thermocycler/deactivateLid" ) -CommandThermocyclerOpenLid: dev_types.ThermocyclerOpenLidCommandId = ( - "thermocycler/openLid" -) -CommandThermocyclerCloseLid: dev_types.ThermocyclerCloseLidCommandId = ( +CommandThermocyclerOpenLid: types.ThermocyclerOpenLidCommandId = "thermocycler/openLid" +CommandThermocyclerCloseLid: types.ThermocyclerCloseLidCommandId = ( "thermocycler/closeLid" ) -CommandThermocyclerRunProfile: dev_types.ThermocyclerRunProfileCommandId = ( +CommandThermocyclerRunProfile: types.ThermocyclerRunProfileCommandId = ( "thermocycler/runProfile" ) -CommandThermocyclerAwaitProfile: dev_types.ThermocyclerAwaitProfileCommandId = ( +CommandThermocyclerAwaitProfile: types.ThermocyclerAwaitProfileCommandId = ( "thermocycler/awaitProfileComplete" ) diff --git a/api/src/opentrons/protocols/parameters/csv_parameter_definition.py b/api/src/opentrons/protocols/parameters/csv_parameter_definition.py index 35e0d4f2345..342f4e1f180 100644 --- a/api/src/opentrons/protocols/parameters/csv_parameter_definition.py +++ b/api/src/opentrons/protocols/parameters/csv_parameter_definition.py @@ -3,8 +3,8 @@ from opentrons.protocol_engine.types import ( RunTimeParameter, - FileId, CSVParameter as ProtocolEngineCSVParameter, + FileInfo, ) from . import validation @@ -29,7 +29,7 @@ def __init__( self._variable_name = validation.ensure_variable_name(variable_name) self._description = validation.ensure_description(description) self._value: Optional[TextIO] = None - self._id: Optional[str] = None + self._file_info: Optional[FileInfo] = None @property def variable_name(self) -> str: @@ -46,12 +46,12 @@ def value(self, new_file: TextIO) -> None: self._value = new_file @property - def id(self) -> Optional[str]: - return self._id + def file_info(self) -> Optional[FileInfo]: + return self._file_info - @id.setter - def id(self, uuid: str) -> None: - self._id = uuid + @file_info.setter + def file_info(self, file_info: FileInfo) -> None: + self._file_info = file_info def as_csv_parameter_interface(self) -> CSVParameter: return CSVParameter(csv_file=self._value) @@ -62,7 +62,7 @@ def as_protocol_engine_type(self) -> RunTimeParameter: displayName=self._display_name, variableName=self._variable_name, description=self._description, - file=FileId(id=self._id) if self._id is not None else None, + file=self._file_info, ) diff --git a/api/src/opentrons/protocols/parse.py b/api/src/opentrons/protocols/parse.py index 712b4fe4416..03e5bcfd62f 100644 --- a/api/src/opentrons/protocols/parse.py +++ b/api/src/opentrons/protocols/parse.py @@ -21,7 +21,7 @@ Schema as JSONProtocolSchema, load_schema as load_protocol_schema, ) -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.ordered_set import OrderedSet @@ -41,8 +41,8 @@ from .bundle import extract_bundle if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition - from opentrons_shared_data.protocol.dev_types import JsonProtocol as JsonProtocolDef + from opentrons_shared_data.labware.types import LabwareDefinition + from opentrons_shared_data.protocol.types import JsonProtocol as JsonProtocolDef MODULE_LOG = logging.getLogger(__name__) diff --git a/api/src/opentrons/protocols/types.py b/api/src/opentrons/protocols/types.py index 273a3e877d4..099ceb2a7a2 100644 --- a/api/src/opentrons/protocols/types.py +++ b/api/src/opentrons/protocols/types.py @@ -1,13 +1,13 @@ from typing import Any, Dict, NamedTuple, Optional, Union, TYPE_CHECKING from dataclasses import dataclass -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from .api_support.definitions import MIN_SUPPORTED_VERSION from .api_support.types import APIVersion if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition - from opentrons_shared_data.protocol.dev_types import ( + from opentrons_shared_data.labware.types import LabwareDefinition + from opentrons_shared_data.protocol.types import ( JsonProtocol as JsonProtocolDef, Metadata as JsonProtocolMetadata, ) diff --git a/api/src/opentrons/simulate.py b/api/src/opentrons/simulate.py index b765a01d02d..01a1484c6b5 100644 --- a/api/src/opentrons/simulate.py +++ b/api/src/opentrons/simulate.py @@ -27,7 +27,7 @@ ) from typing_extensions import Literal -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType import opentrons from opentrons import should_use_ot3 @@ -76,7 +76,7 @@ from .util import entrypoint_util if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import ( + from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionDict, ) diff --git a/api/src/opentrons/types.py b/api/src/opentrons/types.py index 44035851b35..49b3476e489 100644 --- a/api/src/opentrons/types.py +++ b/api/src/opentrons/types.py @@ -3,7 +3,7 @@ from math import sqrt, isclose from typing import TYPE_CHECKING, Any, NamedTuple, Iterator, Union, List -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from .protocols.api_support.labware_like import LabwareLike diff --git a/api/src/opentrons/util/entrypoint_util.py b/api/src/opentrons/util/entrypoint_util.py index 63779eda18f..a65a2cc3fea 100644 --- a/api/src/opentrons/util/entrypoint_util.py +++ b/api/src/opentrons/util/entrypoint_util.py @@ -32,7 +32,7 @@ from opentrons.protocols.types import JsonProtocol, Protocol, PythonProtocol if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.labware.types import LabwareDefinition log = logging.getLogger(__name__) diff --git a/api/src/opentrons/util/performance_helpers.py b/api/src/opentrons/util/performance_helpers.py index 021fd8166ed..e14ad20ff51 100644 --- a/api/src/opentrons/util/performance_helpers.py +++ b/api/src/opentrons/util/performance_helpers.py @@ -1,9 +1,9 @@ -"""Performance helpers for tracking robot context.""" +"""Performance helpers for tracking robot activity.""" import functools from pathlib import Path -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum import typing from opentrons.config import ( get_performance_metrics_data_dir, @@ -12,7 +12,7 @@ ) if typing.TYPE_CHECKING: - from performance_metrics import RobotContextState, SupportsTracking + from performance_metrics import RobotActivityState, SupportsTracking _UnderlyingFunctionParameters = typing.ParamSpec("_UnderlyingFunctionParameters") @@ -36,7 +36,7 @@ def __init__(self, storage_location: Path, should_track: bool) -> None: def track( self, - state: "RobotContextState", + state: "RobotActivityState", ) -> typing.Callable[ [_UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn]], _UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn], @@ -72,38 +72,38 @@ def _handle_package_import() -> typing.Type["SupportsTracking"]: If the package is not available, return a stubbed tracker. """ try: - from performance_metrics import RobotContextTracker + from performance_metrics import RobotActivityTracker - return RobotContextTracker + return RobotActivityTracker except ImportError: return _StubbedTracker _package_to_use = _handle_package_import() -_robot_context_tracker: typing.Optional["SupportsTracking"] = None +_robot_activity_tracker: typing.Optional["SupportsTracking"] = None # TODO: derek maggio (06-03-2024): investigate if _should_track should be -# reevaluated each time _get_robot_context_tracker is called. I think this +# reevaluated each time _get_robot_activity_tracker is called. I think this # might get stuck in a state where after the first call, _should_track is # always considered the initial value. It might miss changes to the feature # flag. The easiest way to test this is on a robot when that is working. -def _get_robot_context_tracker() -> "SupportsTracking": - """Singleton for the robot context tracker.""" - global _robot_context_tracker - if _robot_context_tracker is None: - _robot_context_tracker = _package_to_use( +def _get_robot_activity_tracker() -> "SupportsTracking": + """Singleton for the robot activity tracker.""" + global _robot_activity_tracker + if _robot_activity_tracker is None: + _robot_activity_tracker = _package_to_use( get_performance_metrics_data_dir(), _should_track ) - return _robot_context_tracker + return _robot_activity_tracker def _track_a_function( - state_name: "RobotContextState", + state_name: "RobotActivityState", func: _UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn], ) -> typing.Callable[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn]: - """Track a function. + """Wrap a passed function with RobotActivityTracker.track. This function is a decorator that will track the given state for the decorated function. @@ -115,7 +115,7 @@ def _track_a_function( Returns: The decorated function. """ - tracker: SupportsTracking = _get_robot_context_tracker() + tracker: SupportsTracking = _get_robot_activity_tracker() wrapped = tracker.track(state=state_name)(func) @functools.wraps(func) diff --git a/api/tests/opentrons/calibration_storage/test_deck_attitude.py b/api/tests/opentrons/calibration_storage/test_deck_attitude.py index 759e9f97190..bbb832651d1 100644 --- a/api/tests/opentrons/calibration_storage/test_deck_attitude.py +++ b/api/tests/opentrons/calibration_storage/test_deck_attitude.py @@ -22,7 +22,7 @@ ) if TYPE_CHECKING: - from opentrons_shared_data.deck.dev_types import RobotModel + from opentrons_shared_data.deck.types import RobotModel @pytest.fixture(autouse=True) diff --git a/api/tests/opentrons/calibration_storage/test_pipette_offset_ot2.py b/api/tests/opentrons/calibration_storage/test_pipette_offset_ot2.py index 1411c7fada7..1eca823d45b 100644 --- a/api/tests/opentrons/calibration_storage/test_pipette_offset_ot2.py +++ b/api/tests/opentrons/calibration_storage/test_pipette_offset_ot2.py @@ -13,7 +13,7 @@ ) if TYPE_CHECKING: - from opentrons_shared_data.deck.dev_types import RobotModel + from opentrons_shared_data.deck.types import RobotModel @pytest.fixture diff --git a/api/tests/opentrons/calibration_storage/test_pipette_offset_ot3.py b/api/tests/opentrons/calibration_storage/test_pipette_offset_ot3.py index 7ce0603083c..abe8a7edd79 100644 --- a/api/tests/opentrons/calibration_storage/test_pipette_offset_ot3.py +++ b/api/tests/opentrons/calibration_storage/test_pipette_offset_ot3.py @@ -13,7 +13,7 @@ ) if TYPE_CHECKING: - from opentrons_shared_data.deck.dev_types import RobotModel + from opentrons_shared_data.deck.types import RobotModel @pytest.fixture diff --git a/api/tests/opentrons/calibration_storage/test_tip_length_ot2.py b/api/tests/opentrons/calibration_storage/test_tip_length_ot2.py index 2d593bda67e..df503241d75 100644 --- a/api/tests/opentrons/calibration_storage/test_tip_length_ot2.py +++ b/api/tests/opentrons/calibration_storage/test_tip_length_ot2.py @@ -17,10 +17,10 @@ clear_tip_length_calibration, models, ) -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri if TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.labware.types import LabwareDefinition @pytest.fixture diff --git a/api/tests/opentrons/config/test_advanced_settings.py b/api/tests/opentrons/config/test_advanced_settings.py index 17122fca0dd..5cfed20cac7 100644 --- a/api/tests/opentrons/config/test_advanced_settings.py +++ b/api/tests/opentrons/config/test_advanced_settings.py @@ -4,7 +4,7 @@ from unittest.mock import MagicMock, patch from opentrons.config import advanced_settings, CONFIG -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum @pytest.fixture diff --git a/api/tests/opentrons/config/test_reset.py b/api/tests/opentrons/config/test_reset.py index aacea130e1f..f7f8a848671 100644 --- a/api/tests/opentrons/config/test_reset.py +++ b/api/tests/opentrons/config/test_reset.py @@ -4,10 +4,10 @@ from opentrons.config import reset -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum if TYPE_CHECKING: - from opentrons_shared_data.deck.dev_types import RobotModel + from opentrons_shared_data.deck.types import RobotModel @pytest.fixture diff --git a/api/tests/opentrons/conftest.py b/api/tests/opentrons/conftest.py index f25cb781d28..ff41e83b870 100755 --- a/api/tests/opentrons/conftest.py +++ b/api/tests/opentrons/conftest.py @@ -33,11 +33,11 @@ except (OSError, ModuleNotFoundError): aionotify = None -from opentrons_shared_data.robot.dev_types import RobotTypeEnum -from opentrons_shared_data.protocol.dev_types import JsonProtocol -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.module.dev_types import ModuleDefinitionV3 -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.robot.types import RobotTypeEnum +from opentrons_shared_data.protocol.types import JsonProtocol +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.module.types import ModuleDefinitionV3 +from opentrons_shared_data.deck.types import ( RobotModel, DeckDefinitionV3, DeckDefinitionV5, diff --git a/api/tests/opentrons/hardware_control/instruments/test_instrument_calibration.py b/api/tests/opentrons/hardware_control/instruments/test_instrument_calibration.py index d1f705d596f..6f9ad72c460 100644 --- a/api/tests/opentrons/hardware_control/instruments/test_instrument_calibration.py +++ b/api/tests/opentrons/hardware_control/instruments/test_instrument_calibration.py @@ -6,7 +6,7 @@ from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] from decoy import Decoy -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareUri, LabwareDefinition as LabwareDefDict, ) diff --git a/api/tests/opentrons/hardware_control/test_ot3_api.py b/api/tests/opentrons/hardware_control/test_ot3_api.py index 0c1fff849c0..21ab1ad8ef9 100644 --- a/api/tests/opentrons/hardware_control/test_ot3_api.py +++ b/api/tests/opentrons/hardware_control/test_ot3_api.py @@ -87,7 +87,7 @@ from opentrons_shared_data.pipette import ( load_data as load_pipette_data, ) -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel from opentrons.hardware_control.modules import ( Thermocycler, TempDeck, diff --git a/api/tests/opentrons/hardware_control/test_ot3_transforms.py b/api/tests/opentrons/hardware_control/test_ot3_transforms.py index 37328043e84..c75629968ac 100644 --- a/api/tests/opentrons/hardware_control/test_ot3_transforms.py +++ b/api/tests/opentrons/hardware_control/test_ot3_transforms.py @@ -5,7 +5,7 @@ from opentrons.hardware_control import ot3api from opentrons.hardware_control.types import Axis from opentrons_shared_data.pipette import name_for_model -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel @pytest.mark.parametrize( diff --git a/api/tests/opentrons/hardware_control/test_simulator_setup.py b/api/tests/opentrons/hardware_control/test_simulator_setup.py index 63dca593bff..2b53bda67d6 100644 --- a/api/tests/opentrons/hardware_control/test_simulator_setup.py +++ b/api/tests/opentrons/hardware_control/test_simulator_setup.py @@ -7,7 +7,7 @@ from opentrons.hardware_control.modules import MagDeck, Thermocycler, TempDeck from opentrons.hardware_control import simulator_setup, API from opentrons.types import Mount -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.hardware_control.types import OT3Mount if TYPE_CHECKING: diff --git a/api/tests/opentrons/motion_planning/test_adjacent_slots_getters.py b/api/tests/opentrons/motion_planning/test_adjacent_slots_getters.py index 09805e93ca8..2011aecb45a 100644 --- a/api/tests/opentrons/motion_planning/test_adjacent_slots_getters.py +++ b/api/tests/opentrons/motion_planning/test_adjacent_slots_getters.py @@ -2,7 +2,7 @@ import pytest from typing import List, Optional -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName, StagingSlotName from opentrons.motion_planning.adjacent_slots_getters import ( diff --git a/api/tests/opentrons/motion_planning/test_deck_conflict.py b/api/tests/opentrons/motion_planning/test_deck_conflict.py index 553821289fc..05ca12b585c 100644 --- a/api/tests/opentrons/motion_planning/test_deck_conflict.py +++ b/api/tests/opentrons/motion_planning/test_deck_conflict.py @@ -4,8 +4,8 @@ import pytest -from opentrons_shared_data.labware.dev_types import LabwareUri -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.labware.types import LabwareUri +from opentrons_shared_data.robot.types import RobotType from opentrons.motion_planning import deck_conflict diff --git a/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py b/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py index c50ffe4687e..d0171bff798 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_deck_conflict.py @@ -3,8 +3,8 @@ from typing import ContextManager, Any, NamedTuple, List, Tuple from decoy import Decoy from contextlib import nullcontext as does_not_raise -from opentrons_shared_data.labware.dev_types import LabwareUri -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.labware.types import LabwareUri +from opentrons_shared_data.robot.types import RobotType from opentrons.hardware_control.nozzle_manager import NozzleConfigurationType from opentrons.motion_planning import deck_conflict as wrapped_deck_conflict 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 c3adca3f5a8..8854c070ef0 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 @@ -6,7 +6,7 @@ from decoy import Decoy from decoy import errors -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.hardware_control import SyncHardwareAPI from opentrons.hardware_control.dev_types import PipetteDict diff --git a/api/tests/opentrons/protocol_api/core/engine/test_labware_core.py b/api/tests/opentrons/protocol_api/core/engine/test_labware_core.py index 2fb96fc634a..847c80d2125 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_labware_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_labware_core.py @@ -4,7 +4,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefDict, LabwareParameters as LabwareParamsDict, LabwareUri, diff --git a/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py b/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py index 18286397a76..aa575ea1f16 100644 --- a/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py +++ b/api/tests/opentrons/protocol_api/core/engine/test_protocol_core.py @@ -7,17 +7,17 @@ from decoy import Decoy from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.deck.types import ( DeckDefinitionV5, SlotDefV3, ) -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefDict, LabwareUri, ) from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName, StagingSlotName, Mount, MountType, Point from opentrons.protocol_api import OFF_DECK diff --git a/api/tests/opentrons/protocol_api/core/legacy/test_deck.py b/api/tests/opentrons/protocol_api/core/legacy/test_deck.py index 618920650d4..13c6da49193 100644 --- a/api/tests/opentrons/protocol_api/core/legacy/test_deck.py +++ b/api/tests/opentrons/protocol_api/core/legacy/test_deck.py @@ -4,7 +4,7 @@ import pytest from decoy import Decoy, matchers -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons.motion_planning import deck_conflict from opentrons.protocols.api_support.deck_type import ( diff --git a/api/tests/opentrons/protocol_api/core/legacy/test_module_geometry.py b/api/tests/opentrons/protocol_api/core/legacy/test_module_geometry.py index 744235ea03a..f2c62a823c7 100644 --- a/api/tests/opentrons/protocol_api/core/legacy/test_module_geometry.py +++ b/api/tests/opentrons/protocol_api/core/legacy/test_module_geometry.py @@ -18,7 +18,7 @@ ) from opentrons.protocol_api.core.legacy.deck import Deck -from opentrons_shared_data.module.dev_types import ( +from opentrons_shared_data.module.types import ( ModuleDefinitionV3, ModuleDefinitionV1, ) diff --git a/api/tests/opentrons/protocol_api/core/legacy/test_protocol_context_implementation.py b/api/tests/opentrons/protocol_api/core/legacy/test_protocol_context_implementation.py index a2993444d6b..d9fcfa8e29b 100644 --- a/api/tests/opentrons/protocol_api/core/legacy/test_protocol_context_implementation.py +++ b/api/tests/opentrons/protocol_api/core/legacy/test_protocol_context_implementation.py @@ -5,9 +5,9 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.module.dev_types import ModuleDefinitionV3 +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.module.types import ModuleDefinitionV3 from opentrons.types import DeckSlotName, StagingSlotName, Location, Mount, Point from opentrons.util.broker import Broker diff --git a/api/tests/opentrons/protocol_api/test_deck.py b/api/tests/opentrons/protocol_api/test_deck.py index f471cb936e1..74d1a8a4c3b 100644 --- a/api/tests/opentrons/protocol_api/test_deck.py +++ b/api/tests/opentrons/protocol_api/test_deck.py @@ -5,7 +5,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5, SlotDefV3 +from opentrons_shared_data.deck.types import DeckDefinitionV5, SlotDefV3 from opentrons.motion_planning import adjacent_slots_getters as mock_adjacent_slots from opentrons.protocols.api_support.types import APIVersion diff --git a/api/tests/opentrons/protocol_api/test_labware.py b/api/tests/opentrons/protocol_api/test_labware.py index bfbbb7b33a7..4610145162f 100644 --- a/api/tests/opentrons/protocol_api/test_labware.py +++ b/api/tests/opentrons/protocol_api/test_labware.py @@ -5,7 +5,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.protocols.api_support.types import APIVersion from opentrons.protocols.api_support.util import APIVersionError, UnsupportedAPIError diff --git a/api/tests/opentrons/protocol_api/test_module_context.py b/api/tests/opentrons/protocol_api/test_module_context.py index c57f1ff52dc..1fb5132b59c 100644 --- a/api/tests/opentrons/protocol_api/test_module_context.py +++ b/api/tests/opentrons/protocol_api/test_module_context.py @@ -4,7 +4,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.hardware_control.modules.types import ModuleType, HeaterShakerModuleModel from opentrons.legacy_broker import LegacyBroker diff --git a/api/tests/opentrons/protocol_api/test_protocol_context.py b/api/tests/opentrons/protocol_api/test_protocol_context.py index a72ed7f8856..6674e228b2d 100644 --- a/api/tests/opentrons/protocol_api/test_protocol_context.py +++ b/api/tests/opentrons/protocol_api/test_protocol_context.py @@ -5,8 +5,8 @@ import pytest from decoy import Decoy, matchers -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.types import Mount, DeckSlotName, StagingSlotName from opentrons.protocol_api import OFF_DECK diff --git a/api/tests/opentrons/protocol_api/test_validation.py b/api/tests/opentrons/protocol_api/test_validation.py index b06e28e0785..2a2ed6375b0 100644 --- a/api/tests/opentrons/protocol_api/test_validation.py +++ b/api/tests/opentrons/protocol_api/test_validation.py @@ -10,8 +10,8 @@ LabwareRole, Parameters as LabwareDefinitionParameters, ) -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import Mount, DeckSlotName, StagingSlotName, Location, Point from opentrons.hardware_control.modules.types import ( diff --git a/api/tests/opentrons/protocol_api_old/core/simulator/conftest.py b/api/tests/opentrons/protocol_api_old/core/simulator/conftest.py index 723dc568add..2946eb16220 100644 --- a/api/tests/opentrons/protocol_api_old/core/simulator/conftest.py +++ b/api/tests/opentrons/protocol_api_old/core/simulator/conftest.py @@ -21,8 +21,8 @@ LegacyProtocolCoreSimulator, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.pipette.types import PipetteNameType @pytest.fixture diff --git a/api/tests/opentrons/protocol_api_old/core/simulator/test_protocol_context.py b/api/tests/opentrons/protocol_api_old/core/simulator/test_protocol_context.py index 022ce3e5853..caa63806b11 100644 --- a/api/tests/opentrons/protocol_api_old/core/simulator/test_protocol_context.py +++ b/api/tests/opentrons/protocol_api_old/core/simulator/test_protocol_context.py @@ -4,7 +4,7 @@ from _pytest.fixtures import SubRequest from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import Location, Mount from opentrons.protocol_api.core.common import LabwareCore, ProtocolCore diff --git a/api/tests/opentrons/protocol_api_old/test_context.py b/api/tests/opentrons/protocol_api_old/test_context.py index f2406500e6a..e8ec2859d51 100644 --- a/api/tests/opentrons/protocol_api_old/test_context.py +++ b/api/tests/opentrons/protocol_api_old/test_context.py @@ -5,7 +5,7 @@ from typing import Any, Dict from opentrons_shared_data import load_shared_data -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons_shared_data.errors.exceptions import UnexpectedTipRemovalError import opentrons.protocol_api as papi diff --git a/api/tests/opentrons/protocol_api_old/test_labware.py b/api/tests/opentrons/protocol_api_old/test_labware.py index 8f6f1da267b..b98d603b28f 100644 --- a/api/tests/opentrons/protocol_api_old/test_labware.py +++ b/api/tests/opentrons/protocol_api_old/test_labware.py @@ -3,7 +3,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import WellDefinition +from opentrons_shared_data.labware.types import WellDefinition from opentrons.hardware_control.modules.types import ( MagneticModuleModel, diff --git a/api/tests/opentrons/protocol_api_old/test_labware_load.py b/api/tests/opentrons/protocol_api_old/test_labware_load.py index 69a41f7a365..760ca3f9392 100644 --- a/api/tests/opentrons/protocol_api_old/test_labware_load.py +++ b/api/tests/opentrons/protocol_api_old/test_labware_load.py @@ -1,6 +1,6 @@ import pytest from opentrons import protocol_api as papi, types -from opentrons_shared_data.deck.dev_types import DeckDefinitionV3 +from opentrons_shared_data.deck.types import DeckDefinitionV3 labware_name = "corning_96_wellplate_360ul_flat" diff --git a/api/tests/opentrons/protocol_api_old/test_module_context.py b/api/tests/opentrons/protocol_api_old/test_module_context.py index 105e897e1ed..30ec8804721 100644 --- a/api/tests/opentrons/protocol_api_old/test_module_context.py +++ b/api/tests/opentrons/protocol_api_old/test_module_context.py @@ -24,7 +24,7 @@ from opentrons.protocols.api_support.types import APIVersion from opentrons_shared_data.labware import load_definition as load_labware_definition -from opentrons_shared_data.module.dev_types import ModuleDefinitionV3 +from opentrons_shared_data.module.types import ModuleDefinitionV3 @pytest.fixture diff --git a/api/tests/opentrons/protocol_api_old/test_offsets.py b/api/tests/opentrons/protocol_api_old/test_offsets.py index 593abdc8041..b05ecdcc6cc 100644 --- a/api/tests/opentrons/protocol_api_old/test_offsets.py +++ b/api/tests/opentrons/protocol_api_old/test_offsets.py @@ -5,7 +5,7 @@ from opentrons.types import Point, Location if typing.TYPE_CHECKING: - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.labware.types import LabwareDefinition def test_wells_rebuilt_with_offset(minimal_labware_def: "LabwareDefinition") -> None: diff --git a/api/tests/opentrons/protocol_engine/clients/test_child_thread_transport.py b/api/tests/opentrons/protocol_engine/clients/test_child_thread_transport.py index a5026fc4b46..9cbd03c3ec8 100644 --- a/api/tests/opentrons/protocol_engine/clients/test_child_thread_transport.py +++ b/api/tests/opentrons/protocol_engine/clients/test_child_thread_transport.py @@ -8,7 +8,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition from opentrons.protocol_engine import ProtocolEngine, commands, DeckPoint diff --git a/api/tests/opentrons/protocol_engine/clients/test_sync_client.py b/api/tests/opentrons/protocol_engine/clients/test_sync_client.py index 3cfb9abdaa7..03d6912371c 100644 --- a/api/tests/opentrons/protocol_engine/clients/test_sync_client.py +++ b/api/tests/opentrons/protocol_engine/clients/test_sync_client.py @@ -12,7 +12,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition from opentrons.protocol_engine import commands diff --git a/api/tests/opentrons/protocol_engine/commands/conftest.py b/api/tests/opentrons/protocol_engine/commands/conftest.py index 99046f7c84a..8749023c96f 100644 --- a/api/tests/opentrons/protocol_engine/commands/conftest.py +++ b/api/tests/opentrons/protocol_engine/commands/conftest.py @@ -13,6 +13,7 @@ LabwareMovementHandler, StatusBarHandler, TipHandler, + GantryMover, ) from opentrons.protocol_engine.resources.model_utils import ModelUtils from opentrons.protocol_engine.state import StateView @@ -76,3 +77,9 @@ def model_utils(decoy: Decoy) -> ModelUtils: def status_bar(decoy: Decoy) -> StatusBarHandler: """Get a mocked out StatusBarHandler.""" return decoy.mock(cls=StatusBarHandler) + + +@pytest.fixture +def gantry_mover(decoy: Decoy) -> GantryMover: + """Get a mocked out GantryMover.""" + return decoy.mock(cls=GantryMover) diff --git a/api/tests/opentrons/protocol_engine/commands/test_aspirate.py b/api/tests/opentrons/protocol_engine/commands/test_aspirate.py index 1dba452ff45..b1e3c1e52df 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_aspirate.py +++ b/api/tests/opentrons/protocol_engine/commands/test_aspirate.py @@ -263,7 +263,10 @@ async def test_overpressure_error( assert result == DefinedErrorData( public=OverpressureError.construct( - id=error_id, createdAt=error_timestamp, wrappedErrors=[matchers.Anything()] + id=error_id, + createdAt=error_timestamp, + wrappedErrors=[matchers.Anything()], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, ), private=OverpressureErrorInternalData( position=DeckPoint(x=position.x, y=position.y, z=position.z) diff --git a/api/tests/opentrons/protocol_engine/commands/test_aspirate_in_place.py b/api/tests/opentrons/protocol_engine/commands/test_aspirate_in_place.py index c6197f2d26f..26f62231a56 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_aspirate_in_place.py +++ b/api/tests/opentrons/protocol_engine/commands/test_aspirate_in_place.py @@ -1,22 +1,32 @@ """Test aspirate-in-place commands.""" +from datetime import datetime + import pytest -from decoy import Decoy +from decoy import Decoy, matchers + +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError +from opentrons.types import Point from opentrons.hardware_control import API as HardwareAPI -from opentrons.protocol_engine.execution import PipettingHandler +from opentrons.protocol_engine.execution import PipettingHandler, GantryMover from opentrons.protocol_engine.commands.aspirate_in_place import ( AspirateInPlaceParams, AspirateInPlaceResult, AspirateInPlaceImplementation, ) -from opentrons.protocol_engine.commands.command import SuccessData +from opentrons.protocol_engine.commands.command import SuccessData, DefinedErrorData from opentrons.protocol_engine.errors.exceptions import PipetteNotReadyToAspirateError from opentrons.protocol_engine.notes import CommandNoteAdder - +from opentrons.protocol_engine.resources import ModelUtils from opentrons.protocol_engine.state import ( StateStore, ) +from opentrons.protocol_engine.types import DeckPoint +from opentrons.protocol_engine.commands.pipetting_common import ( + OverpressureError, + OverpressureErrorInternalData, +) @pytest.fixture @@ -43,6 +53,8 @@ def subject( state_store: StateStore, hardware_api: HardwareAPI, mock_command_note_adder: CommandNoteAdder, + model_utils: ModelUtils, + gantry_mover: GantryMover, ) -> AspirateInPlaceImplementation: """Get the impelementation subject.""" return AspirateInPlaceImplementation( @@ -50,6 +62,8 @@ def subject( hardware_api=hardware_api, state_view=state_store, command_note_adder=mock_command_note_adder, + model_utils=model_utils, + gantry_mover=gantry_mover, ) @@ -143,3 +157,57 @@ async def test_aspirate_raises_volume_error( with pytest.raises(AssertionError): await subject.execute(data) + + +async def test_overpressure_error( + decoy: Decoy, + gantry_mover: GantryMover, + pipetting: PipettingHandler, + subject: AspirateInPlaceImplementation, + model_utils: ModelUtils, + mock_command_note_adder: CommandNoteAdder, +) -> None: + """It should return an overpressure error if the hardware API indicates that.""" + pipette_id = "pipette-id" + + position = Point(x=1, y=2, z=3) + + error_id = "error-id" + error_timestamp = datetime(year=2020, month=1, day=2) + + data = AspirateInPlaceParams( + pipetteId=pipette_id, + volume=50, + flowRate=1.23, + ) + + decoy.when(pipetting.get_is_ready_to_aspirate(pipette_id=pipette_id)).then_return( + True + ) + + decoy.when( + await pipetting.aspirate_in_place( + pipette_id=pipette_id, + volume=50, + flow_rate=1.23, + command_note_adder=mock_command_note_adder, + ), + ).then_raise(PipetteOverpressureError()) + + decoy.when(model_utils.generate_id()).then_return(error_id) + decoy.when(model_utils.get_timestamp()).then_return(error_timestamp) + decoy.when(await gantry_mover.get_position(pipette_id)).then_return(position) + + result = await subject.execute(data) + + assert result == DefinedErrorData( + public=OverpressureError.construct( + id=error_id, + createdAt=error_timestamp, + wrappedErrors=[matchers.Anything()], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=position.x, y=position.y, z=position.z) + ), + ) diff --git a/api/tests/opentrons/protocol_engine/commands/test_configure_for_volume.py b/api/tests/opentrons/protocol_engine/commands/test_configure_for_volume.py index 218d299ee29..95e1e856bd4 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_configure_for_volume.py +++ b/api/tests/opentrons/protocol_engine/commands/test_configure_for_volume.py @@ -18,7 +18,7 @@ ConfigureForVolumePrivateResult, ConfigureForVolumeImplementation, ) -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from ..pipette_fixtures import get_default_nozzle_map from opentrons.types import Point diff --git a/api/tests/opentrons/protocol_engine/commands/test_dispense.py b/api/tests/opentrons/protocol_engine/commands/test_dispense.py index 4df18a19152..86c4f6ac93b 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_dispense.py +++ b/api/tests/opentrons/protocol_engine/commands/test_dispense.py @@ -1,26 +1,47 @@ """Test dispense commands.""" -from decoy import Decoy +from datetime import datetime + +import pytest +from decoy import Decoy, matchers + +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError from opentrons.protocol_engine import WellLocation, WellOrigin, WellOffset, DeckPoint from opentrons.protocol_engine.execution import MovementHandler, PipettingHandler from opentrons.types import Point -from opentrons.protocol_engine.commands.command import SuccessData +from opentrons.protocol_engine.commands.command import SuccessData, DefinedErrorData from opentrons.protocol_engine.commands.dispense import ( DispenseParams, DispenseResult, DispenseImplementation, ) +from opentrons.protocol_engine.resources.model_utils import ModelUtils +from opentrons.protocol_engine.commands.pipetting_common import ( + OverpressureError, + OverpressureErrorInternalData, +) + + +@pytest.fixture +def subject( + movement: MovementHandler, + pipetting: PipettingHandler, + model_utils: ModelUtils, +) -> DispenseImplementation: + """Get the implementation subject.""" + return DispenseImplementation( + movement=movement, pipetting=pipetting, model_utils=model_utils + ) async def test_dispense_implementation( decoy: Decoy, movement: MovementHandler, pipetting: PipettingHandler, + subject: DispenseImplementation, ) -> None: """It should move to the target location and then dispense.""" - subject = DispenseImplementation(movement=movement, pipetting=pipetting) - well_location = WellLocation( origin=WellOrigin.BOTTOM, offset=WellOffset(x=0, y=0, z=1) ) @@ -55,3 +76,65 @@ async def test_dispense_implementation( public=DispenseResult(volume=42, position=DeckPoint(x=1, y=2, z=3)), private=None, ) + + +async def test_overpressure_error( + decoy: Decoy, + movement: MovementHandler, + pipetting: PipettingHandler, + subject: DispenseImplementation, + model_utils: ModelUtils, +) -> None: + """It should return an overpressure error if the hardware API indicates that.""" + pipette_id = "pipette-id" + labware_id = "labware-id" + well_name = "well-name" + well_location = WellLocation( + origin=WellOrigin.BOTTOM, offset=WellOffset(x=0, y=0, z=1) + ) + + position = Point(x=1, y=2, z=3) + + error_id = "error-id" + error_timestamp = datetime(year=2020, month=1, day=2) + + data = DispenseParams( + pipetteId=pipette_id, + labwareId=labware_id, + wellName=well_name, + wellLocation=well_location, + volume=50, + flowRate=1.23, + ) + + decoy.when( + await movement.move_to_well( + pipette_id=pipette_id, + labware_id=labware_id, + well_name=well_name, + well_location=well_location, + ), + ).then_return(position) + + decoy.when( + await pipetting.dispense_in_place( + pipette_id=pipette_id, volume=50, flow_rate=1.23, push_out=None + ), + ).then_raise(PipetteOverpressureError()) + + decoy.when(model_utils.generate_id()).then_return(error_id) + decoy.when(model_utils.get_timestamp()).then_return(error_timestamp) + + result = await subject.execute(data) + + assert result == DefinedErrorData( + public=OverpressureError.construct( + id=error_id, + createdAt=error_timestamp, + wrappedErrors=[matchers.Anything()], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=position.x, y=position.y, z=position.z) + ), + ) diff --git a/api/tests/opentrons/protocol_engine/commands/test_dispense_in_place.py b/api/tests/opentrons/protocol_engine/commands/test_dispense_in_place.py index e1bb654613c..3b37e1078b7 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_dispense_in_place.py +++ b/api/tests/opentrons/protocol_engine/commands/test_dispense_in_place.py @@ -1,22 +1,37 @@ """Test dispense-in-place commands.""" -from decoy import Decoy +from datetime import datetime -from opentrons.protocol_engine.execution import PipettingHandler +from decoy import Decoy, matchers -from opentrons.protocol_engine.commands.command import SuccessData +from opentrons_shared_data.errors.exceptions import PipetteOverpressureError + +from opentrons.types import Point +from opentrons.protocol_engine.execution import PipettingHandler, GantryMover + +from opentrons.protocol_engine.commands.command import SuccessData, DefinedErrorData from opentrons.protocol_engine.commands.dispense_in_place import ( DispenseInPlaceParams, DispenseInPlaceResult, DispenseInPlaceImplementation, ) +from opentrons.protocol_engine.types import DeckPoint +from opentrons.protocol_engine.commands.pipetting_common import ( + OverpressureError, + OverpressureErrorInternalData, +) +from opentrons.protocol_engine.resources import ModelUtils async def test_dispense_in_place_implementation( decoy: Decoy, pipetting: PipettingHandler, + gantry_mover: GantryMover, + model_utils: ModelUtils, ) -> None: """It should dispense in place.""" - subject = DispenseInPlaceImplementation(pipetting=pipetting) + subject = DispenseInPlaceImplementation( + pipetting=pipetting, gantry_mover=gantry_mover, model_utils=model_utils + ) data = DispenseInPlaceParams( pipetteId="pipette-id-abc", @@ -33,3 +48,56 @@ async def test_dispense_in_place_implementation( result = await subject.execute(data) assert result == SuccessData(public=DispenseInPlaceResult(volume=42), private=None) + + +async def test_overpressure_error( + decoy: Decoy, + gantry_mover: GantryMover, + pipetting: PipettingHandler, + model_utils: ModelUtils, +) -> None: + """It should return an overpressure error if the hardware API indicates that.""" + subject = DispenseInPlaceImplementation( + pipetting=pipetting, gantry_mover=gantry_mover, model_utils=model_utils + ) + + pipette_id = "pipette-id" + + position = Point(x=1, y=2, z=3) + + error_id = "error-id" + error_timestamp = datetime(year=2020, month=1, day=2) + + data = DispenseInPlaceParams( + pipetteId=pipette_id, + volume=50, + flowRate=1.23, + pushOut=10, + ) + + decoy.when( + await pipetting.dispense_in_place( + pipette_id=pipette_id, + volume=50, + flow_rate=1.23, + push_out=10, + ), + ).then_raise(PipetteOverpressureError()) + + decoy.when(model_utils.generate_id()).then_return(error_id) + decoy.when(model_utils.get_timestamp()).then_return(error_timestamp) + decoy.when(await gantry_mover.get_position(pipette_id)).then_return(position) + + result = await subject.execute(data) + + assert result == DefinedErrorData( + public=OverpressureError.construct( + id=error_id, + createdAt=error_timestamp, + wrappedErrors=[matchers.Anything()], + errorInfo={"retryLocation": (position.x, position.y, position.z)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=position.x, y=position.y, z=position.z) + ), + ) diff --git a/api/tests/opentrons/protocol_engine/commands/test_load_module.py b/api/tests/opentrons/protocol_engine/commands/test_load_module.py index 9f5692b53b0..2dbd0e31e97 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_load_module.py +++ b/api/tests/opentrons/protocol_engine/commands/test_load_module.py @@ -5,7 +5,7 @@ from opentrons.protocol_engine.errors import LocationIsOccupiedError from opentrons.protocol_engine.state import StateView -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName from opentrons.protocol_engine.types import ( DeckSlotLocation, @@ -29,7 +29,7 @@ ThermocyclerModuleModel, HeaterShakerModuleModel, ) -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.deck.types import ( DeckDefinitionV5, SlotDefV3, ) diff --git a/api/tests/opentrons/protocol_engine/commands/test_load_pipette.py b/api/tests/opentrons/protocol_engine/commands/test_load_pipette.py index e90e20586f1..72721343478 100644 --- a/api/tests/opentrons/protocol_engine/commands/test_load_pipette.py +++ b/api/tests/opentrons/protocol_engine/commands/test_load_pipette.py @@ -2,8 +2,8 @@ import pytest from decoy import Decoy -from opentrons_shared_data.pipette.dev_types import PipetteNameType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.pipette.types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import MountType, Point from opentrons.protocol_engine.errors import InvalidSpecificationForRobotTypeError diff --git a/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_blow_out_in_place.py b/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_blow_out_in_place.py new file mode 100644 index 00000000000..f25d8d06169 --- /dev/null +++ b/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_blow_out_in_place.py @@ -0,0 +1,49 @@ +"""Test blow-out-in-place commands.""" +from decoy import Decoy + +from opentrons.types import MountType +from opentrons.protocol_engine.state import StateView +from opentrons.protocol_engine.commands.unsafe.unsafe_blow_out_in_place import ( + UnsafeBlowOutInPlaceParams, + UnsafeBlowOutInPlaceResult, + UnsafeBlowOutInPlaceImplementation, +) +from opentrons.protocol_engine.commands.command import SuccessData +from opentrons.protocol_engine.execution import ( + PipettingHandler, +) +from opentrons.protocol_engine.state.motion import PipetteLocationData +from opentrons.hardware_control import OT3HardwareControlAPI +from opentrons.hardware_control.types import Axis + + +async def test_blow_out_in_place_implementation( + decoy: Decoy, + state_view: StateView, + ot3_hardware_api: OT3HardwareControlAPI, + pipetting: PipettingHandler, +) -> None: + """Test UnsafeBlowOut command execution.""" + subject = UnsafeBlowOutInPlaceImplementation( + state_view=state_view, + hardware_api=ot3_hardware_api, + pipetting=pipetting, + ) + + data = UnsafeBlowOutInPlaceParams( + pipetteId="pipette-id", + flowRate=1.234, + ) + + decoy.when( + state_view.motion.get_pipette_location(pipette_id="pipette-id") + ).then_return(PipetteLocationData(mount=MountType.LEFT, critical_point=None)) + + result = await subject.execute(data) + + assert result == SuccessData(public=UnsafeBlowOutInPlaceResult(), private=None) + + decoy.verify( + await ot3_hardware_api.update_axis_position_estimations([Axis.P_L]), + await pipetting.blow_out_in_place(pipette_id="pipette-id", flow_rate=1.234), + ) diff --git a/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_drop_tip_in_place.py b/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_drop_tip_in_place.py new file mode 100644 index 00000000000..3659dd2db31 --- /dev/null +++ b/api/tests/opentrons/protocol_engine/commands/unsafe/test_unsafe_drop_tip_in_place.py @@ -0,0 +1,53 @@ +"""Test unsafe drop tip in place commands.""" +import pytest +from decoy import Decoy + +from opentrons.types import MountType +from opentrons.protocol_engine.state import StateView + +from opentrons.protocol_engine.execution import TipHandler + +from opentrons.protocol_engine.commands.command import SuccessData +from opentrons.protocol_engine.commands.unsafe.unsafe_drop_tip_in_place import ( + UnsafeDropTipInPlaceParams, + UnsafeDropTipInPlaceResult, + UnsafeDropTipInPlaceImplementation, +) +from opentrons.protocol_engine.state.motion import PipetteLocationData +from opentrons.hardware_control import OT3HardwareControlAPI +from opentrons.hardware_control.types import Axis + + +@pytest.fixture +def mock_tip_handler(decoy: Decoy) -> TipHandler: + """Get a mock TipHandler.""" + return decoy.mock(cls=TipHandler) + + +async def test_drop_tip_implementation( + decoy: Decoy, + mock_tip_handler: TipHandler, + state_view: StateView, + ot3_hardware_api: OT3HardwareControlAPI, +) -> None: + """A DropTip command should have an execution implementation.""" + subject = UnsafeDropTipInPlaceImplementation( + tip_handler=mock_tip_handler, + state_view=state_view, + hardware_api=ot3_hardware_api, + ) + + params = UnsafeDropTipInPlaceParams(pipetteId="abc", homeAfter=False) + decoy.when(state_view.motion.get_pipette_location(pipette_id="abc")).then_return( + PipetteLocationData(mount=MountType.LEFT, critical_point=None) + ) + + result = await subject.execute(params) + + assert result == SuccessData(public=UnsafeDropTipInPlaceResult(), private=None) + + decoy.verify( + await ot3_hardware_api.update_axis_position_estimations([Axis.P_L]), + await mock_tip_handler.drop_tip(pipette_id="abc", home_after=False), + times=1, + ) diff --git a/api/tests/opentrons/protocol_engine/commands/unsafe/test_update_position_estimators.py b/api/tests/opentrons/protocol_engine/commands/unsafe/test_update_position_estimators.py new file mode 100644 index 00000000000..da7ffe75012 --- /dev/null +++ b/api/tests/opentrons/protocol_engine/commands/unsafe/test_update_position_estimators.py @@ -0,0 +1,54 @@ +"""Test update-position-estimator commands.""" +from decoy import Decoy + +from opentrons.protocol_engine.commands.unsafe.update_position_estimators import ( + UpdatePositionEstimatorsParams, + UpdatePositionEstimatorsResult, + UpdatePositionEstimatorsImplementation, +) +from opentrons.protocol_engine.commands.command import SuccessData +from opentrons.protocol_engine.execution import GantryMover +from opentrons.protocol_engine.types import MotorAxis +from opentrons.hardware_control import OT3HardwareControlAPI +from opentrons.hardware_control.types import Axis + + +async def test_update_position_estimators_implementation( + decoy: Decoy, ot3_hardware_api: OT3HardwareControlAPI, gantry_mover: GantryMover +) -> None: + """Test UnsafeBlowOut command execution.""" + subject = UpdatePositionEstimatorsImplementation( + hardware_api=ot3_hardware_api, gantry_mover=gantry_mover + ) + + data = UpdatePositionEstimatorsParams( + axes=[MotorAxis.LEFT_Z, MotorAxis.LEFT_PLUNGER, MotorAxis.X, MotorAxis.Y] + ) + + decoy.when(gantry_mover.motor_axis_to_hardware_axis(MotorAxis.LEFT_Z)).then_return( + Axis.Z_L + ) + decoy.when( + gantry_mover.motor_axis_to_hardware_axis(MotorAxis.LEFT_PLUNGER) + ).then_return(Axis.P_L) + decoy.when(gantry_mover.motor_axis_to_hardware_axis(MotorAxis.X)).then_return( + Axis.X + ) + decoy.when(gantry_mover.motor_axis_to_hardware_axis(MotorAxis.Y)).then_return( + Axis.Y + ) + decoy.when( + await ot3_hardware_api.update_axis_position_estimations( + [Axis.Z_L, Axis.P_L, Axis.X, Axis.Y] + ) + ).then_return(None) + + result = await subject.execute(data) + + assert result == SuccessData(public=UpdatePositionEstimatorsResult(), private=None) + + decoy.verify( + await ot3_hardware_api.update_axis_position_estimations( + [Axis.Z_L, Axis.P_L, Axis.X, Axis.Y] + ), + ) diff --git a/api/tests/opentrons/protocol_engine/conftest.py b/api/tests/opentrons/protocol_engine/conftest.py index df02b53ce38..a44548e0de3 100644 --- a/api/tests/opentrons/protocol_engine/conftest.py +++ b/api/tests/opentrons/protocol_engine/conftest.py @@ -7,7 +7,7 @@ from opentrons_shared_data import load_shared_data from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data.labware import load_definition from opentrons_shared_data.pipette import pipette_definition from opentrons.protocols.models import LabwareDefinition diff --git a/api/tests/opentrons/protocol_engine/execution/test_command_executor.py b/api/tests/opentrons/protocol_engine/execution/test_command_executor.py index b55bdc24c18..bc2e8a0a8fe 100644 --- a/api/tests/opentrons/protocol_engine/execution/test_command_executor.py +++ b/api/tests/opentrons/protocol_engine/execution/test_command_executor.py @@ -140,9 +140,11 @@ def command_note_tracker_provider(decoy: Decoy) -> CommandNoteTrackerProvider: @pytest.fixture -def error_recovery_policy(decoy: Decoy) -> ErrorRecoveryPolicy: +def error_recovery_policy(state_store: StateStore, decoy: Decoy) -> ErrorRecoveryPolicy: """Get a mock error recovery policy.""" - return decoy.mock(cls=ErrorRecoveryPolicy) + mock = decoy.mock(cls=ErrorRecoveryPolicy) + decoy.when(state_store.commands.get_error_recovery_policy()).then_return(mock) + return mock def get_next_tracker( @@ -182,7 +184,6 @@ def subject( status_bar: StatusBarHandler, model_utils: ModelUtils, command_note_tracker_provider: CommandNoteTrackerProvider, - error_recovery_policy: ErrorRecoveryPolicy, ) -> CommandExecutor: """Get a CommandExecutor test subject with its dependencies mocked out.""" return CommandExecutor( @@ -200,7 +201,6 @@ def subject( rail_lights=rail_lights, status_bar=status_bar, command_note_tracker_provider=command_note_tracker_provider, - error_recovery_policy=error_recovery_policy, ) diff --git a/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py b/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py index 068ad6cfbd6..d28ebe700ca 100644 --- a/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py +++ b/api/tests/opentrons/protocol_engine/execution/test_equipment_handler.py @@ -6,9 +6,9 @@ from decoy import Decoy, matchers from typing import Any, Optional, cast, Dict -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.pipette import pipette_definition -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons.calibration_storage.helpers import uri_from_details from opentrons.types import Mount as HwMount, MountType, DeckSlotName, Point diff --git a/api/tests/opentrons/protocol_engine/execution/test_heater_shaker_movement_flagger.py b/api/tests/opentrons/protocol_engine/execution/test_heater_shaker_movement_flagger.py index 18d49e233c5..91afa9f023c 100644 --- a/api/tests/opentrons/protocol_engine/execution/test_heater_shaker_movement_flagger.py +++ b/api/tests/opentrons/protocol_engine/execution/test_heater_shaker_movement_flagger.py @@ -32,7 +32,7 @@ HeaterShakerModuleSubState, ) from opentrons.types import DeckSlotName -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType @pytest.fixture diff --git a/api/tests/opentrons/protocol_engine/pipette_fixtures.py b/api/tests/opentrons/protocol_engine/pipette_fixtures.py index 146a0cb12d1..6f5a44286bd 100644 --- a/api/tests/opentrons/protocol_engine/pipette_fixtures.py +++ b/api/tests/opentrons/protocol_engine/pipette_fixtures.py @@ -5,7 +5,7 @@ from opentrons.types import Point from opentrons.hardware_control.nozzle_manager import NozzleMap -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.pipette.pipette_definition import ValidNozzleMaps diff --git a/api/tests/opentrons/protocol_engine/resources/test_deck_configuration_provider.py b/api/tests/opentrons/protocol_engine/resources/test_deck_configuration_provider.py index 12b324955be..174c101f8b1 100644 --- a/api/tests/opentrons/protocol_engine/resources/test_deck_configuration_provider.py +++ b/api/tests/opentrons/protocol_engine/resources/test_deck_configuration_provider.py @@ -5,7 +5,7 @@ from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.types import DeckSlotName diff --git a/api/tests/opentrons/protocol_engine/resources/test_deck_data_provider.py b/api/tests/opentrons/protocol_engine/resources/test_deck_data_provider.py index bd720777ed6..0caa1d52ac5 100644 --- a/api/tests/opentrons/protocol_engine/resources/test_deck_data_provider.py +++ b/api/tests/opentrons/protocol_engine/resources/test_deck_data_provider.py @@ -3,7 +3,7 @@ from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] from decoy import Decoy -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.protocols.models import LabwareDefinition from opentrons.types import DeckSlotName diff --git a/api/tests/opentrons/protocol_engine/resources/test_labware_data_provider.py b/api/tests/opentrons/protocol_engine/resources/test_labware_data_provider.py index ef73fa61b61..92718c70d89 100644 --- a/api/tests/opentrons/protocol_engine/resources/test_labware_data_provider.py +++ b/api/tests/opentrons/protocol_engine/resources/test_labware_data_provider.py @@ -1,7 +1,7 @@ """Functional tests for the LabwareDataProvider.""" from typing import cast -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.calibration_storage.helpers import hash_labware_def from opentrons.protocols.models import LabwareDefinition from opentrons.protocol_api.labware import get_labware_definition diff --git a/api/tests/opentrons/protocol_engine/resources/test_pipette_data_provider.py b/api/tests/opentrons/protocol_engine/resources/test_pipette_data_provider.py index 4b792376368..4fb2f6a2fd3 100644 --- a/api/tests/opentrons/protocol_engine/resources/test_pipette_data_provider.py +++ b/api/tests/opentrons/protocol_engine/resources/test_pipette_data_provider.py @@ -2,7 +2,7 @@ from typing import Dict from sys import maxsize import pytest -from opentrons_shared_data.pipette.dev_types import PipetteNameType, PipetteModel +from opentrons_shared_data.pipette.types import PipetteNameType, PipetteModel from opentrons_shared_data.pipette import pipette_definition, types as pip_types from opentrons_shared_data.pipette.pipette_definition import ( PipetteBoundingBoxOffsetDefinition, diff --git a/api/tests/opentrons/protocol_engine/state/command_fixtures.py b/api/tests/opentrons/protocol_engine/state/command_fixtures.py index 98ee48e724d..845b33f18d8 100644 --- a/api/tests/opentrons/protocol_engine/state/command_fixtures.py +++ b/api/tests/opentrons/protocol_engine/state/command_fixtures.py @@ -3,7 +3,7 @@ from pydantic import BaseModel from typing import Optional, cast -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType from opentrons.protocols.models import LabwareDefinition from opentrons.protocol_engine import ErrorOccurrence, commands as cmd @@ -603,3 +603,41 @@ def create_reload_labware_command( params=params, result=result, ) + + +def create_unsafe_blow_out_in_place_command( + pipette_id: str, + flow_rate: float, +) -> cmd.unsafe.UnsafeBlowOutInPlace: + """Create a completed UnsafeBlowOutInPlace command.""" + params = cmd.unsafe.UnsafeBlowOutInPlaceParams( + pipetteId=pipette_id, flowRate=flow_rate + ) + result = cmd.unsafe.UnsafeBlowOutInPlaceResult() + + return cmd.unsafe.UnsafeBlowOutInPlace( + id="command-id", + key="command-key", + status=cmd.CommandStatus.SUCCEEDED, + createdAt=datetime.now(), + params=params, + result=result, + ) + + +def create_unsafe_drop_tip_in_place_command( + pipette_id: str, +) -> cmd.unsafe.UnsafeDropTipInPlace: + """Get a completed UnsafeDropTipInPlace command.""" + params = cmd.unsafe.UnsafeDropTipInPlaceParams(pipetteId=pipette_id) + + result = cmd.unsafe.UnsafeDropTipInPlaceResult() + + return cmd.unsafe.UnsafeDropTipInPlace( + id="command-id", + key="command-key", + status=cmd.CommandStatus.SUCCEEDED, + createdAt=datetime.now(), + params=params, + result=result, + ) diff --git a/api/tests/opentrons/protocol_engine/state/test_addressable_area_state.py b/api/tests/opentrons/protocol_engine/state/test_addressable_area_state.py index 66fa692fe25..987db0dcba3 100644 --- a/api/tests/opentrons/protocol_engine/state/test_addressable_area_state.py +++ b/api/tests/opentrons/protocol_engine/state/test_addressable_area_state.py @@ -4,7 +4,7 @@ implementation detail. """ -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.protocol_engine.actions.actions import SetDeckConfigurationAction from opentrons.protocol_engine.state.addressable_areas import ( diff --git a/api/tests/opentrons/protocol_engine/state/test_addressable_area_store.py b/api/tests/opentrons/protocol_engine/state/test_addressable_area_store.py index fcadb43940e..9c098cf1c96 100644 --- a/api/tests/opentrons/protocol_engine/state/test_addressable_area_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_addressable_area_store.py @@ -7,7 +7,7 @@ import pytest -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data.labware.labware_definition import Parameters from opentrons.protocols.models import LabwareDefinition from opentrons.types import DeckSlotName diff --git a/api/tests/opentrons/protocol_engine/state/test_addressable_area_view.py b/api/tests/opentrons/protocol_engine/state/test_addressable_area_view.py index 3d1cbe9be1a..07552aa4273 100644 --- a/api/tests/opentrons/protocol_engine/state/test_addressable_area_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_addressable_area_view.py @@ -11,8 +11,8 @@ from decoy import Decoy from typing import Dict, Set, Optional, cast -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.types import Point, DeckSlotName from opentrons.protocol_engine.errors import ( diff --git a/api/tests/opentrons/protocol_engine/state/test_command_state.py b/api/tests/opentrons/protocol_engine/state/test_command_state.py index d0a05260abb..9ebb338d85c 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_state.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_state.py @@ -5,6 +5,7 @@ """ from datetime import datetime +from typing import Any from unittest.mock import sentinel import pytest @@ -14,7 +15,10 @@ from opentrons.hardware_control.types import DoorState from opentrons.ordered_set import OrderedSet from opentrons.protocol_engine import actions, commands, errors -from opentrons.protocol_engine.actions.actions import PlayAction +from opentrons.protocol_engine.actions.actions import ( + PlayAction, + SetErrorRecoveryPolicyAction, +) from opentrons.protocol_engine.commands.command import CommandIntent from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryType from opentrons.protocol_engine.errors.error_occurrence import ErrorOccurrence @@ -39,9 +43,18 @@ def _make_config() -> Config: ) +def _placeholder_error_recovery_policy(*args: object, **kwargs: object) -> Any: + """A placeholder `ErrorRecoveryPolicy` for tests that don't care about it.""" + raise NotImplementedError() + + def test_queue_command_action() -> None: """It should translate a command request into a queued command and add it.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) id = "command-id" @@ -70,7 +83,11 @@ def test_queue_command_action() -> None: def test_latest_protocol_command_hash() -> None: """It should return the latest protocol command's hash.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) # The initial hash should be None. @@ -106,7 +123,11 @@ def test_latest_protocol_command_hash() -> None: @pytest.mark.parametrize("error_recovery_type", ErrorRecoveryType) def test_command_failure(error_recovery_type: ErrorRecoveryType) -> None: """It should store an error and mark the command if it fails.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) command_id = "command-id" @@ -175,7 +196,11 @@ def test_command_failure(error_recovery_type: ErrorRecoveryType) -> None: def test_command_failure_clears_queues() -> None: """It should clear the command queue on command failure.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + is_door_open=False, + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) queue_1 = actions.QueueCommandAction( @@ -228,7 +253,11 @@ def test_setup_command_failure_only_clears_setup_command_queue() -> None: This test queues up a non-setup command followed by two setup commands, then runs and fails the first setup command. """ - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) queue_1 = actions.QueueCommandAction( @@ -298,7 +327,11 @@ def test_nonfatal_command_failure() -> None: The queue status should be "awaiting-recovery." """ - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject_view = CommandView(subject.state) queue_1 = actions.QueueCommandAction( @@ -348,6 +381,7 @@ def test_door_during_setup_phase() -> None: """Test behavior when the door is opened during the setup phase.""" subject = CommandStore( is_door_open=False, + error_recovery_policy=_placeholder_error_recovery_policy, config=Config( block_on_door_open=True, # Choice of robot and deck type are arbitrary. @@ -380,6 +414,7 @@ def test_door_during_protocol_phase() -> None: """Test behavior when the door is opened during the main protocol phase.""" subject = CommandStore( is_door_open=False, + error_recovery_policy=_placeholder_error_recovery_policy, config=Config( block_on_door_open=True, # Choice of robot and deck type are arbitrary. @@ -429,6 +464,7 @@ def test_door_during_error_recovery() -> None: """Test behavior when the door is opened during error recovery.""" subject = CommandStore( is_door_open=False, + error_recovery_policy=_placeholder_error_recovery_policy, config=Config( block_on_door_open=True, # Choice of robot and deck type are arbitrary. @@ -515,6 +551,7 @@ def test_door_initially_open( """Test open-door blocking behavior given different initial door states.""" subject = CommandStore( is_door_open=door_initially_open, + error_recovery_policy=_placeholder_error_recovery_policy, config=Config( block_on_door_open=True, # Choice of robot and deck type are arbitrary. @@ -533,7 +570,11 @@ def test_door_initially_open( def test_error_recovery_type_tracking() -> None: """It should keep track of each failed command's error recovery type.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) subject.handle_action( actions.QueueCommandAction( @@ -593,7 +634,11 @@ def test_error_recovery_type_tracking() -> None: def test_recovery_target_tracking() -> None: """It should keep track of the command currently undergoing error recovery.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) subject_view = CommandView(subject.state) queue_1 = actions.QueueCommandAction( @@ -687,7 +732,11 @@ def test_recovery_target_tracking() -> None: def test_final_state_after_estop() -> None: """Test the final state of the run after it's E-stopped.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) subject_view = CommandView(subject.state) error_details = actions.FinishErrorDetails( @@ -716,7 +765,11 @@ def test_final_state_after_estop() -> None: def test_final_state_after_stop() -> None: """Test the final state of the run after it's stopped.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) subject_view = CommandView(subject.state) subject.handle_action(actions.StopAction()) @@ -749,7 +802,11 @@ def test_final_state_after_error_recovery_stop() -> None: We still want to count this as "stopped," not "failed." """ - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) subject_view = CommandView(subject.state) # Fail a command to put the subject in recovery mode. @@ -802,3 +859,18 @@ def test_final_state_after_error_recovery_stop() -> None: assert subject_view.get_status() == EngineStatus.STOPPED assert subject_view.get_recovery_target() is None assert subject_view.get_error() is None + + +def test_set_and_get_error_recovery_policy() -> None: + """Test storage of `ErrorRecoveryPolicy`s.""" + initial_policy = sentinel.initial_policy + new_policy = sentinel.new_policy + subject = CommandStore( + config=_make_config(), + error_recovery_policy=initial_policy, + is_door_open=False, + ) + subject_view = CommandView(subject.state) + assert subject_view.get_error_recovery_policy() is initial_policy + subject.handle_action(SetErrorRecoveryPolicyAction(sentinel.new_policy)) + assert subject_view.get_error_recovery_policy() is new_policy diff --git a/api/tests/opentrons/protocol_engine/state/test_command_store_old.py b/api/tests/opentrons/protocol_engine/state/test_command_store_old.py index 49edce8312f..92fba9b4851 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_store_old.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_store_old.py @@ -5,8 +5,10 @@ """ +from decoy import matchers import pytest from datetime import datetime +from typing import Any from opentrons_shared_data.errors import ErrorCodes @@ -50,6 +52,15 @@ def _make_config(block_on_door_open: bool = False) -> Config: ) +def _placeholder_error_recovery_policy(*args: object, **kwargs: object) -> Any: + """A placeholder `ErrorRecoveryPolicy` for tests that don't care about it. + + That should be all the tests in this file, since error recovery was added + after this file was deprecated. + """ + raise NotImplementedError() + + def test_command_queue_and_unqueue() -> None: """It should queue on QueueCommandAction and dequeue on RunCommandAction.""" queue_1 = QueueCommandAction( @@ -77,7 +88,11 @@ def test_command_queue_and_unqueue() -> None: command=create_succeeded_command(command_id="command-id-2"), ) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(queue_1) assert subject.state.command_history.get_queue_ids() == OrderedSet(["command-id-1"]) @@ -126,7 +141,11 @@ def test_setup_command_queue_and_unqueue() -> None: command=create_succeeded_command(command_id="command-id-2"), ) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(queue_1) assert subject.state.command_history.get_setup_queue_ids() == OrderedSet( @@ -170,7 +189,11 @@ def test_setup_queue_action_updates_command_intent() -> None: intent=commands.CommandIntent.SETUP, ) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(queue_cmd) assert subject.state.command_history.get("command-id-1") == CommandEntry( @@ -195,7 +218,11 @@ def test_running_command_id() -> None: command=create_succeeded_command(command_id="command-id-1"), ) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(queue) assert subject.state.command_history.get_running_command() is None @@ -222,7 +249,11 @@ def test_command_store_keeps_commands_in_queue_order() -> None: params=commands.CommentParams(message="hello world"), ) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action( QueueCommandAction( @@ -285,7 +316,11 @@ def test_command_store_keeps_commands_in_queue_order() -> None: @pytest.mark.parametrize("pause_source", PauseSource) def test_command_store_handles_pause_action(pause_source: PauseSource) -> None: """It should clear the running flag on pause.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PauseAction(source=pause_source)) assert subject.state == CommandState( @@ -302,13 +337,18 @@ def test_command_store_handles_pause_action(pause_source: PauseSource) -> None: recovery_target_command_id=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) @pytest.mark.parametrize("pause_source", PauseSource) def test_command_store_handles_play_action(pause_source: PauseSource) -> None: """It should set the running flag on play.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) assert subject.state == CommandState( @@ -325,6 +365,7 @@ def test_command_store_handles_play_action(pause_source: PauseSource) -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -334,7 +375,11 @@ def test_command_store_handles_play_action(pause_source: PauseSource) -> None: def test_command_store_handles_finish_action() -> None: """It should change to a succeeded state with FinishAction.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) subject.handle_action(FinishAction()) @@ -353,6 +398,7 @@ def test_command_store_handles_finish_action() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -362,7 +408,11 @@ def test_command_store_handles_finish_action() -> None: def test_command_store_handles_finish_action_with_stopped() -> None: """It should change to a stopped state if FinishAction has set_run_status=False.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) subject.handle_action(FinishAction(set_run_status=False)) @@ -378,7 +428,11 @@ def test_command_store_handles_stop_action( from_estop: bool, expected_run_result: RunResult ) -> None: """It should mark the engine as non-gracefully stopped on StopAction.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) subject.handle_action(StopAction(from_estop=from_estop)) @@ -397,6 +451,7 @@ def test_command_store_handles_stop_action( run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=from_estop, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -406,7 +461,11 @@ def test_command_store_handles_stop_action( def test_command_store_handles_stop_action_when_awaiting_recovery() -> None: """It should mark the engine as non-gracefully stopped on StopAction.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) @@ -428,6 +487,7 @@ def test_command_store_handles_stop_action_when_awaiting_recovery() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -437,7 +497,11 @@ def test_command_store_handles_stop_action_when_awaiting_recovery() -> None: def test_command_store_cannot_restart_after_should_stop() -> None: """It should reject a play action after finish.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(FinishAction()) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) @@ -455,6 +519,7 @@ def test_command_store_cannot_restart_after_should_stop() -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -464,7 +529,11 @@ def test_command_store_cannot_restart_after_should_stop() -> None: def test_command_store_save_started_completed_run_timestamp() -> None: """It should save started and completed timestamps.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) start_time = datetime(year=2021, month=1, day=1) hardware_stopped_time = datetime(year=2022, month=2, day=2) @@ -481,7 +550,11 @@ def test_command_store_save_started_completed_run_timestamp() -> None: def test_timestamps_are_latched() -> None: """It should not change startedAt or completedAt once set.""" - subject = CommandStore(config=_make_config(), is_door_open=False) + subject = CommandStore( + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + is_door_open=False, + ) play_time_1 = datetime(year=2021, month=1, day=1) play_time_2 = datetime(year=2022, month=2, day=2) @@ -512,7 +585,11 @@ def test_command_store_wraps_unknown_errors() -> None: The wrapping EnumeratedError should be an UnexpectedProtocolError for errors that happened in the main part of the protocol run, or a PythonException for errors that happened elsewhere. """ - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action( FinishAction( @@ -587,6 +664,7 @@ def test_command_store_wraps_unknown_errors() -> None: recovery_target_command_id=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -601,7 +679,11 @@ class MyCustomError(errors.ProtocolEngineError): def __init__(self, message: str) -> None: super().__init__(ErrorCodes.PIPETTE_NOT_PRESENT, message) - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action( FinishAction( @@ -650,6 +732,7 @@ def __init__(self, message: str) -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -659,7 +742,11 @@ def __init__(self, message: str) -> None: def test_command_store_ignores_stop_after_graceful_finish() -> None: """It should no-op on stop if already gracefully finished.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) subject.handle_action(FinishAction()) @@ -679,6 +766,7 @@ def test_command_store_ignores_stop_after_graceful_finish() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -688,7 +776,11 @@ def test_command_store_ignores_stop_after_graceful_finish() -> None: def test_command_store_ignores_finish_after_non_graceful_stop() -> None: """It should no-op on finish if already ungracefully stopped.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) subject.handle_action(PlayAction(requested_at=datetime(year=2021, month=1, day=1))) subject.handle_action(StopAction()) @@ -708,6 +800,7 @@ def test_command_store_ignores_finish_after_non_graceful_stop() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -717,7 +810,11 @@ def test_command_store_ignores_finish_after_non_graceful_stop() -> None: def test_handles_hardware_stopped() -> None: """It should mark the hardware as stopped on HardwareStoppedAction.""" - subject = CommandStore(is_door_open=False, config=_make_config()) + subject = CommandStore( + is_door_open=False, + config=_make_config(), + error_recovery_policy=_placeholder_error_recovery_policy, + ) completed_at = datetime(year=2021, day=1, month=1) subject.handle_action( HardwareStoppedAction(completed_at=completed_at, finish_error_details=None) @@ -737,6 +834,7 @@ def test_handles_hardware_stopped() -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + error_recovery_policy=matchers.Anything(), ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] diff --git a/api/tests/opentrons/protocol_engine/state/test_command_view_old.py b/api/tests/opentrons/protocol_engine/state/test_command_view_old.py index 1d1da8e193c..f64f4a09d2d 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_view_old.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_view_old.py @@ -8,7 +8,7 @@ import pytest from contextlib import nullcontext as does_not_raise from datetime import datetime -from typing import Dict, List, NamedTuple, Optional, Sequence, Type, Union +from typing import Any, Dict, List, NamedTuple, Optional, Sequence, Type, Union from opentrons.protocol_engine import EngineStatus, commands as cmd, errors from opentrons.protocol_engine.actions import ( @@ -46,6 +46,15 @@ ) +def _placeholder_error_recovery_policy(*args: object, **kwargs: object) -> Any: + """A placeholder `ErrorRecoveryPolicy` for tests that don't care about it. + + That should be all the tests in this file, since error recovery was added + after this file was deprecated. + """ + raise NotImplementedError() + + def get_command_view( # noqa: C901 queue_status: QueueStatus = QueueStatus.SETUP, run_completed_at: Optional[datetime] = None, @@ -99,6 +108,7 @@ def get_command_view( # noqa: C901 run_started_at=run_started_at, latest_protocol_command_hash=latest_command_hash, stopped_by_estop=False, + error_recovery_policy=_placeholder_error_recovery_policy, ) return CommandView(state=state) diff --git a/api/tests/opentrons/protocol_engine/state/test_geometry_view.py b/api/tests/opentrons/protocol_engine/state/test_geometry_view.py index 9887a4ef76c..1f085b526f1 100644 --- a/api/tests/opentrons/protocol_engine/state/test_geometry_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_geometry_view.py @@ -7,14 +7,14 @@ from typing import cast, List, Tuple, Optional, NamedTuple from datetime import datetime -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.pipette import pipette_definition from opentrons.calibration_storage.helpers import uri_from_details from opentrons.protocols.models import LabwareDefinition from opentrons.types import Point, DeckSlotName, MountType -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.labware.labware_definition import ( Dimensions as LabwareDimensions, Parameters as LabwareDefinitionParameters, diff --git a/api/tests/opentrons/protocol_engine/state/test_labware_store.py b/api/tests/opentrons/protocol_engine/state/test_labware_store.py index 960ce423194..cb651fc37a7 100644 --- a/api/tests/opentrons/protocol_engine/state/test_labware_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_labware_store.py @@ -4,7 +4,7 @@ from datetime import datetime from opentrons.calibration_storage.helpers import uri_from_details -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.protocols.models import LabwareDefinition from opentrons.types import DeckSlotName diff --git a/api/tests/opentrons/protocol_engine/state/test_labware_view.py b/api/tests/opentrons/protocol_engine/state/test_labware_view.py index 0f8086de606..43c69594422 100644 --- a/api/tests/opentrons/protocol_engine/state/test_labware_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_labware_view.py @@ -5,8 +5,8 @@ from contextlib import nullcontext as does_not_raise from opentrons_shared_data.deck import load as load_deck -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.deck.types import DeckDefinitionV5 +from opentrons_shared_data.pipette.types import LabwareUri from opentrons_shared_data.labware import load_definition from opentrons_shared_data.labware.labware_definition import ( Parameters, diff --git a/api/tests/opentrons/protocol_engine/state/test_module_store.py b/api/tests/opentrons/protocol_engine/state/test_module_store.py index 0dabf508483..f052056aa35 100644 --- a/api/tests/opentrons/protocol_engine/state/test_module_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_module_store.py @@ -2,8 +2,8 @@ from typing import List, Set, cast, Dict, Optional import pytest -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5 from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] from opentrons.types import DeckSlotName diff --git a/api/tests/opentrons/protocol_engine/state/test_module_view.py b/api/tests/opentrons/protocol_engine/state/test_module_view.py index e308c09407d..c7c67aa7e61 100644 --- a/api/tests/opentrons/protocol_engine/state/test_module_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_module_view.py @@ -17,8 +17,8 @@ cast, ) -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons_shared_data import load_shared_data from opentrons.types import DeckSlotName, MountType diff --git a/api/tests/opentrons/protocol_engine/state/test_motion_view.py b/api/tests/opentrons/protocol_engine/state/test_motion_view.py index 61ec01262f3..278fff82023 100644 --- a/api/tests/opentrons/protocol_engine/state/test_motion_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_motion_view.py @@ -5,7 +5,7 @@ import pytest from decoy import Decoy -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import Point, MountType, DeckSlotName from opentrons.hardware_control.types import CriticalPoint from opentrons import motion_planning @@ -29,7 +29,7 @@ from opentrons.protocol_engine.state.motion import MotionView from opentrons.protocol_engine.state.modules import ModuleView from opentrons.protocol_engine.state.module_substates import HeaterShakerModuleId -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType @pytest.fixture diff --git a/api/tests/opentrons/protocol_engine/state/test_pipette_store.py b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py index 8ccfc06fd07..c8d60395b3b 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py @@ -3,7 +3,7 @@ from datetime import datetime from typing import Optional, Union -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.pipette import pipette_definition from opentrons.types import DeckSlotName, MountType, Point @@ -50,6 +50,7 @@ create_pick_up_tip_command, create_drop_tip_command, create_drop_tip_in_place_command, + create_unsafe_drop_tip_in_place_command, create_touch_tip_command, create_move_to_well_command, create_blow_out_command, @@ -58,6 +59,7 @@ create_move_to_coordinates_command, create_move_relative_command, create_prepare_to_aspirate_command, + create_unsafe_blow_out_in_place_command, ) from ..pipette_fixtures import get_default_nozzle_map @@ -176,6 +178,42 @@ def test_handles_drop_tip_in_place(subject: PipetteStore) -> None: assert subject.state.aspirated_volume_by_id["xyz"] is None +def test_handles_unsafe_drop_tip_in_place(subject: PipetteStore) -> None: + """It should clear tip and volume details after a drop tip in place.""" + load_pipette_command = create_load_pipette_command( + pipette_id="xyz", + pipette_name=PipetteNameType.P300_SINGLE, + mount=MountType.LEFT, + ) + + pick_up_tip_command = create_pick_up_tip_command( + pipette_id="xyz", tip_volume=42, tip_length=101, tip_diameter=8.0 + ) + + unsafe_drop_tip_in_place_command = create_unsafe_drop_tip_in_place_command( + pipette_id="xyz", + ) + + subject.handle_action( + SucceedCommandAction(private_result=None, command=load_pipette_command) + ) + subject.handle_action( + SucceedCommandAction(private_result=None, command=pick_up_tip_command) + ) + assert subject.state.attached_tip_by_id["xyz"] == TipGeometry( + volume=42, length=101, diameter=8.0 + ) + assert subject.state.aspirated_volume_by_id["xyz"] == 0 + + subject.handle_action( + SucceedCommandAction( + private_result=None, command=unsafe_drop_tip_in_place_command + ) + ) + assert subject.state.attached_tip_by_id["xyz"] is None + assert subject.state.aspirated_volume_by_id["xyz"] is None + + @pytest.mark.parametrize( "aspirate_command", [ @@ -261,6 +299,7 @@ def test_dispense_subtracts_volume( [ create_blow_out_command("pipette-id", 1.23), create_blow_out_in_place_command("pipette-id", 1.23), + create_unsafe_blow_out_in_place_command("pipette-id", 1.23), ], ) def test_blow_out_clears_volume( @@ -330,6 +369,7 @@ def test_blow_out_clears_volume( public=OverpressureError( id="error-id", createdAt=datetime.now(), + errorInfo={"retryLocation": (0, 0, 0)}, ), private=OverpressureErrorInternalData( position=DeckPoint(x=0, y=0, z=0) @@ -425,6 +465,43 @@ def test_blow_out_clears_volume( well_name="move-to-well-well-name", ), ), + ( + FailCommandAction( + running_command=cmd.Dispense( + params=cmd.DispenseParams( + pipetteId="pipette-id", + labwareId="dispense-labware-id", + wellName="dispense-well-name", + volume=50, + flowRate=1.23, + ), + id="command-id", + key="command-key", + createdAt=datetime.now(), + status=cmd.CommandStatus.RUNNING, + ), + error=DefinedErrorData( + public=OverpressureError( + id="error-id", + createdAt=datetime.now(), + errorInfo={"retryLocation": (0, 0, 0)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=0, y=0, z=0) + ), + ), + command_id="command-id", + error_id="error-id", + failed_at=datetime.now(), + notes=[], + type=ErrorRecoveryType.WAIT_FOR_RECOVERY, + ), + CurrentWell( + pipette_id="pipette-id", + labware_id="dispense-labware-id", + well_name="dispense-well-name", + ), + ), ), ) def test_movement_commands_update_current_well( @@ -818,6 +895,7 @@ def test_add_pipette_config( id="error-id", detail="error-detail", createdAt=datetime.now(), + errorInfo={"retryLocation": (11, 22, 33)}, ), private=OverpressureErrorInternalData( position=DeckPoint(x=11, y=22, z=33) @@ -900,6 +978,95 @@ def test_add_pipette_config( ), private_result=None, ), + FailCommandAction( + running_command=cmd.Dispense( + params=cmd.DispenseParams( + pipetteId="pipette-id", + labwareId="labware-id", + wellName="well-name", + volume=125, + flowRate=1.23, + ), + id="command-id", + key="command-key", + createdAt=datetime.now(), + status=cmd.CommandStatus.RUNNING, + ), + error=DefinedErrorData( + public=OverpressureError( + id="error-id", + detail="error-detail", + createdAt=datetime.now(), + errorInfo={"retryLocation": (11, 22, 33)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=11, y=22, z=33) + ), + ), + command_id="command-id", + error_id="error-id", + failed_at=datetime.now(), + notes=[], + type=ErrorRecoveryType.WAIT_FOR_RECOVERY, + ), + FailCommandAction( + running_command=cmd.AspirateInPlace( + params=cmd.AspirateInPlaceParams( + pipetteId="pipette-id", + volume=125, + flowRate=1.23, + ), + id="command-id", + key="command-key", + createdAt=datetime.now(), + status=cmd.CommandStatus.RUNNING, + ), + error=DefinedErrorData( + public=OverpressureError( + id="error-id", + detail="error-detail", + createdAt=datetime.now(), + errorInfo={"retryLocation": (11, 22, 33)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=11, y=22, z=33) + ), + ), + command_id="command-id", + error_id="error-id", + failed_at=datetime.now(), + notes=[], + type=ErrorRecoveryType.WAIT_FOR_RECOVERY, + ), + FailCommandAction( + running_command=cmd.DispenseInPlace( + params=cmd.DispenseInPlaceParams( + pipetteId="pipette-id", + volume=125, + flowRate=1.23, + ), + id="command-id", + key="command-key", + createdAt=datetime.now(), + status=cmd.CommandStatus.RUNNING, + ), + error=DefinedErrorData( + public=OverpressureError( + id="error-id", + detail="error-detail", + createdAt=datetime.now(), + errorInfo={"retryLocation": (11, 22, 33)}, + ), + private=OverpressureErrorInternalData( + position=DeckPoint(x=11, y=22, z=33) + ), + ), + command_id="command-id", + error_id="error-id", + failed_at=datetime.now(), + notes=[], + type=ErrorRecoveryType.WAIT_FOR_RECOVERY, + ), ), ) def test_movement_commands_update_deck_point( diff --git a/api/tests/opentrons/protocol_engine/state/test_pipette_view.py b/api/tests/opentrons/protocol_engine/state/test_pipette_view.py index 1942a9a04e1..e8823c3c6ad 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_view.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_view.py @@ -4,7 +4,7 @@ import pytest from typing import cast, Dict, List, Optional, Tuple, NamedTuple -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.pipette import pipette_definition from opentrons_shared_data.pipette.pipette_definition import ValidNozzleMaps diff --git a/api/tests/opentrons/protocol_engine/state/test_state_store.py b/api/tests/opentrons/protocol_engine/state/test_state_store.py index 26f50515317..6cd24564795 100644 --- a/api/tests/opentrons/protocol_engine/state/test_state_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_state_store.py @@ -1,11 +1,11 @@ """Tests for the top-level StateStore/StateView.""" -from typing import Callable, Union +from typing import Any, Callable, Union from datetime import datetime import pytest from decoy import Decoy -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from opentrons.util.change_notifier import ChangeNotifier from opentrons.protocol_engine.actions import PlayAction @@ -36,6 +36,10 @@ def subject( engine_config: Config, ) -> StateStore: """Get a StateStore test subject.""" + + def placeholder_error_recovery_policy(*args: object, **kwargs: object) -> Any: + raise NotImplementedError() + return StateStore( config=engine_config, deck_definition=ot2_standard_deck_def, @@ -49,6 +53,7 @@ def subject( deck_fixed_labware=[], change_notifier=change_notifier, is_door_open=False, + error_recovery_policy=placeholder_error_recovery_policy, ) diff --git a/api/tests/opentrons/protocol_engine/state/test_tip_state.py b/api/tests/opentrons/protocol_engine/state/test_tip_state.py index 3572f81567f..da570c940cd 100644 --- a/api/tests/opentrons/protocol_engine/state/test_tip_state.py +++ b/api/tests/opentrons/protocol_engine/state/test_tip_state.py @@ -20,7 +20,7 @@ LoadedStaticPipetteData, ) from opentrons.types import Point -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from ..pipette_fixtures import ( NINETY_SIX_MAP, NINETY_SIX_COLS, @@ -109,6 +109,17 @@ def drop_tip_in_place_command() -> commands.DropTipInPlace: ) +@pytest.fixture +def unsafe_drop_tip_in_place_command() -> commands.unsafe.UnsafeDropTipInPlace: + """Get an unsafe drop-tip-in-place command.""" + return commands.unsafe.UnsafeDropTipInPlace.construct( # type: ignore[call-arg] + params=commands.unsafe.UnsafeDropTipInPlaceParams.construct( + pipetteId="pipette-id" + ), + result=commands.unsafe.UnsafeDropTipInPlaceResult.construct(), + ) + + @pytest.mark.parametrize( "labware_definition", [ @@ -903,6 +914,7 @@ def test_drop_tip( pick_up_tip_command: commands.PickUpTip, drop_tip_command: commands.DropTip, drop_tip_in_place_command: commands.DropTipInPlace, + unsafe_drop_tip_in_place_command: commands.unsafe.UnsafeDropTipInPlace, supported_tip_fixture: pipette_definition.SupportedTipsDefinition, ) -> None: """It should be clear tip length when a tip is dropped.""" @@ -968,6 +980,20 @@ def test_drop_tip( result = TipView(subject.state).get_tip_length("pipette-id") assert result == 0 + subject.handle_action( + actions.SucceedCommandAction(private_result=None, command=pick_up_tip_command) + ) + result = TipView(subject.state).get_tip_length("pipette-id") + assert result == 1.23 + + subject.handle_action( + actions.SucceedCommandAction( + private_result=None, command=unsafe_drop_tip_in_place_command + ) + ) + result = TipView(subject.state).get_tip_length("pipette-id") + assert result == 0 + @pytest.mark.parametrize( argnames=["nozzle_map", "expected_channels"], diff --git a/api/tests/opentrons/protocol_engine/test_create_protocol_engine.py b/api/tests/opentrons/protocol_engine/test_create_protocol_engine.py index d3bfd4843bc..b5733cda6b8 100644 --- a/api/tests/opentrons/protocol_engine/test_create_protocol_engine.py +++ b/api/tests/opentrons/protocol_engine/test_create_protocol_engine.py @@ -2,8 +2,8 @@ import pytest from pytest_lazyfixture import lazy_fixture # type: ignore[import-untyped] -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.deck.types import DeckDefinitionV5 +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.deck import load as load_deck from opentrons.calibration_storage.helpers import uri_from_details diff --git a/api/tests/opentrons/protocol_engine/test_protocol_engine.py b/api/tests/opentrons/protocol_engine/test_protocol_engine.py index c2f56d13a2e..2669640e649 100644 --- a/api/tests/opentrons/protocol_engine/test_protocol_engine.py +++ b/api/tests/opentrons/protocol_engine/test_protocol_engine.py @@ -7,8 +7,9 @@ import pytest from decoy import Decoy -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType +from opentrons.protocol_engine.actions.actions import SetErrorRecoveryPolicyAction from opentrons.types import DeckSlotName from opentrons.hardware_control import HardwareControlAPI, OT2HardwareControlAPI from opentrons.hardware_control.modules import MagDeck, TempDeck @@ -32,7 +33,6 @@ PostRunHardwareState, AddressableAreaLocation, ) -from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from opentrons.protocol_engine.execution import ( QueueWorker, HardwareStopper, @@ -117,12 +117,6 @@ def module_data_provider(decoy: Decoy) -> ModuleDataProvider: return decoy.mock(cls=ModuleDataProvider) -@pytest.fixture -def error_recovery_policy(decoy: Decoy) -> ErrorRecoveryPolicy: - """Get a mock ErrorRecoveryPolicy.""" - return decoy.mock(cls=ErrorRecoveryPolicy) - - @pytest.fixture(autouse=True) def _mock_slot_standardization_module( decoy: Decoy, monkeypatch: pytest.MonkeyPatch @@ -146,7 +140,6 @@ def _mock_hash_command_params_module( def subject( hardware_api: HardwareControlAPI, state_store: StateStore, - error_recovery_policy: ErrorRecoveryPolicy, action_dispatcher: ActionDispatcher, plugin_starter: PluginStarter, queue_worker: QueueWorker, @@ -159,7 +152,6 @@ def subject( return ProtocolEngine( hardware_api=hardware_api, state_store=state_store, - error_recovery_policy=error_recovery_policy, action_dispatcher=action_dispatcher, plugin_starter=plugin_starter, queue_worker=queue_worker, @@ -1198,3 +1190,13 @@ def test_reset_tips( action_dispatcher.dispatch(ResetTipsAction(labware_id="cool-labware")), times=1, ) + + +async def test_set_error_recovery_policy( + decoy: Decoy, action_dispatcher: ActionDispatcher, subject: ProtocolEngine +) -> None: + """It should set the error recovery policy by dispatching an action.""" + subject.set_error_recovery_policy(sentinel.new_policy) + decoy.verify( + action_dispatcher.dispatch(SetErrorRecoveryPolicyAction(sentinel.new_policy)) + ) diff --git a/api/tests/opentrons/protocol_engine/test_slot_standardization.py b/api/tests/opentrons/protocol_engine/test_slot_standardization.py index b93fb0a8ad8..f97d09af242 100644 --- a/api/tests/opentrons/protocol_engine/test_slot_standardization.py +++ b/api/tests/opentrons/protocol_engine/test_slot_standardization.py @@ -2,7 +2,7 @@ import pytest -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.types import DeckSlotName from opentrons.protocol_engine import ( diff --git a/api/tests/opentrons/protocol_reader/test_file_identifier.py b/api/tests/opentrons/protocol_reader/test_file_identifier.py index f992e5d727b..6cde2a00db2 100644 --- a/api/tests/opentrons/protocol_reader/test_file_identifier.py +++ b/api/tests/opentrons/protocol_reader/test_file_identifier.py @@ -7,7 +7,7 @@ import pytest from opentrons_shared_data import load_shared_data -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocols import parse from opentrons.protocols.api_support.types import APIVersion diff --git a/api/tests/opentrons/protocol_runner/smoke_tests/test_legacy_command_mapper.py b/api/tests/opentrons/protocol_runner/smoke_tests/test_legacy_command_mapper.py index f6c743ba68b..a652d76eac3 100644 --- a/api/tests/opentrons/protocol_runner/smoke_tests/test_legacy_command_mapper.py +++ b/api/tests/opentrons/protocol_runner/smoke_tests/test_legacy_command_mapper.py @@ -25,7 +25,7 @@ ) from opentrons.protocol_runner.legacy_command_mapper import LegacyCommandParams from opentrons.types import MountType, DeckSlotName -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType async def simulate_and_get_commands(protocol_file: Path) -> List[commands.Command]: diff --git a/api/tests/opentrons/protocol_runner/smoke_tests/test_protocol_runner.py b/api/tests/opentrons/protocol_runner/smoke_tests/test_protocol_runner.py index d7e19fa6966..1a8da30bd76 100644 --- a/api/tests/opentrons/protocol_runner/smoke_tests/test_protocol_runner.py +++ b/api/tests/opentrons/protocol_runner/smoke_tests/test_protocol_runner.py @@ -12,7 +12,7 @@ from decoy import matchers from pathlib import Path -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType, DeckSlotName from opentrons.protocol_engine import ( DeckSlotLocation, diff --git a/api/tests/opentrons/protocol_runner/test_json_translator.py b/api/tests/opentrons/protocol_runner/test_json_translator.py index 10669b30047..0c65274ad9a 100644 --- a/api/tests/opentrons/protocol_runner/test_json_translator.py +++ b/api/tests/opentrons/protocol_runner/test_json_translator.py @@ -29,7 +29,7 @@ Pipette, Robot, ) -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import DeckSlotName, MountType from opentrons.protocol_runner.json_translator import JsonTranslator from opentrons.protocol_engine import ( diff --git a/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py b/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py index 011b790da85..ed171280d17 100644 --- a/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py +++ b/api/tests/opentrons/protocol_runner/test_legacy_command_mapper.py @@ -34,9 +34,9 @@ LegacyContextCommandError, LegacyCommandMapper, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.module.dev_types import ModuleDefinitionV3 -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.module.types import ModuleDefinitionV3 +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import DeckSlotName, Mount, MountType diff --git a/api/tests/opentrons/protocol_runner/test_legacy_context_plugin.py b/api/tests/opentrons/protocol_runner/test_legacy_context_plugin.py index 368a34a297f..620b7afa1ba 100644 --- a/api/tests/opentrons/protocol_runner/test_legacy_context_plugin.py +++ b/api/tests/opentrons/protocol_runner/test_legacy_context_plugin.py @@ -24,7 +24,7 @@ from opentrons.types import DeckSlotName -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionDict, ) diff --git a/api/tests/opentrons/protocol_runner/test_protocol_runner.py b/api/tests/opentrons/protocol_runner/test_protocol_runner.py index 3f9f702d825..e975e90fa73 100644 --- a/api/tests/opentrons/protocol_runner/test_protocol_runner.py +++ b/api/tests/opentrons/protocol_runner/test_protocol_runner.py @@ -9,16 +9,17 @@ from typing import List, cast, Union, Type from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.labware.dev_types import ( +from opentrons_shared_data.labware.types import ( LabwareDefinition as LabwareDefinitionTypedDict, ) from opentrons_shared_data.protocol.models import ProtocolSchemaV6, ProtocolSchemaV7 -from opentrons_shared_data.protocol.dev_types import ( +from opentrons_shared_data.protocol.types import ( JsonProtocol as LegacyJsonProtocolDict, ) from opentrons.hardware_control import API as HardwareAPI from opentrons.legacy_broker import LegacyBroker from opentrons.protocol_api import ProtocolContext +from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryType from opentrons.protocol_engine.types import PostRunHardwareState from opentrons.protocols.api_support.types import APIVersion from opentrons.protocols.parse import PythonParseMode @@ -409,8 +410,14 @@ async def test_run_json_runner_stop_requested_stops_enqueuing( createdAt=datetime(year=2021, month=1, day=1), error=pe_errors.ProtocolEngineError(), ), + status=pe_commands.CommandStatus.FAILED, ) ) + decoy.when( + protocol_engine.state_view.commands.get_error_recovery_type( + "protocol-command-id" + ) + ).then_return(ErrorRecoveryType.FAIL_RUN) await json_runner_subject.load(json_protocol_source) @@ -638,6 +645,7 @@ async def test_load_legacy_python( legacy_protocol_source, python_parse_mode=PythonParseMode.ALLOW_LEGACY_METADATA_AND_REQUIREMENTS, run_time_param_values=None, + run_time_param_files=None, ) run_func_captor = matchers.Captor() @@ -719,6 +727,7 @@ async def test_load_python_with_pe_papi_core( protocol_source, python_parse_mode=PythonParseMode.ALLOW_LEGACY_METADATA_AND_REQUIREMENTS, run_time_param_values=None, + run_time_param_files=None, ) decoy.verify(protocol_engine.add_plugin(matchers.IsA(LegacyContextPlugin)), times=0) @@ -781,6 +790,7 @@ async def test_load_legacy_json( legacy_protocol_source, python_parse_mode=PythonParseMode.ALLOW_LEGACY_METADATA_AND_REQUIREMENTS, run_time_param_values=None, + run_time_param_files=None, ) run_func_captor = matchers.Captor() diff --git a/api/tests/opentrons/protocol_runner/test_run_orchestrator.py b/api/tests/opentrons/protocol_runner/test_run_orchestrator.py index 3746368d8a8..2c5e243c3ec 100644 --- a/api/tests/opentrons/protocol_runner/test_run_orchestrator.py +++ b/api/tests/opentrons/protocol_runner/test_run_orchestrator.py @@ -8,6 +8,7 @@ from decoy import Decoy from typing import Union, Generator +from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy from opentrons.protocol_engine.errors import RunStoppedError from opentrons.protocol_engine.state import StateStore from opentrons.protocols.api_support.types import APIVersion @@ -336,6 +337,7 @@ async def test_load_json( await json_protocol_subject.load( protocol_source=protocol_source, run_time_param_values=None, + run_time_param_files=None, parse_mode=ParseMode.NORMAL, ) @@ -362,6 +364,7 @@ async def test_load_python( protocol_source=protocol_source, parse_mode=ParseMode.NORMAL, run_time_param_values=None, + run_time_param_files=None, ) decoy.verify( @@ -369,6 +372,7 @@ async def test_load_python( protocol_source=protocol_source, python_parse_mode=PythonParseMode.NORMAL, run_time_param_values=None, + run_time_param_files=None, ) ) @@ -392,6 +396,7 @@ async def test_load_json_raises_no_protocol( await live_protocol_subject.load( protocol_source=protocol_source, run_time_param_values=None, + run_time_param_files=None, parse_mode=ParseMode.NORMAL, ) @@ -518,3 +523,14 @@ def get_next_to_execute() -> Generator[str, None, None]: async for command in live_protocol_subject.command_generator(): assert command == f"command-id-{index}" index = index + 1 + + +async def test_create_error_recovery_policy( + decoy: Decoy, + mock_protocol_engine: ProtocolEngine, + live_protocol_subject: RunOrchestrator, +) -> None: + """Should call PE set_error_recovery_policy.""" + policy = decoy.mock(cls=ErrorRecoveryPolicy) + live_protocol_subject.set_error_recovery_policy(policy) + decoy.verify(mock_protocol_engine.set_error_recovery_policy(policy)) diff --git a/api/tests/opentrons/protocols/api_support/test_labware_like.py b/api/tests/opentrons/protocols/api_support/test_labware_like.py index e911b3224a9..2e7fef6ef00 100644 --- a/api/tests/opentrons/protocols/api_support/test_labware_like.py +++ b/api/tests/opentrons/protocols/api_support/test_labware_like.py @@ -9,7 +9,7 @@ from opentrons.protocol_api.core.legacy.deck import Deck from opentrons.types import Location -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition @pytest.fixture(scope="session") diff --git a/api/tests/opentrons/protocols/execution/test_execute_python.py b/api/tests/opentrons/protocols/execution/test_execute_python.py index e663ea1bd69..f02db136836 100644 --- a/api/tests/opentrons/protocols/execution/test_execute_python.py +++ b/api/tests/opentrons/protocols/execution/test_execute_python.py @@ -98,6 +98,7 @@ def run(ctx): assert "Exception [line 5]: hi" in str(e.value) +# TODO (spp, 2024-7-16): add a test for CSV rtp extraction @pytest.mark.ot2_only @pytest.mark.parametrize("protocol_file", ["testosaur_with_rtp.py"]) def test_rtp_extraction(protocol, protocol_file) -> None: @@ -109,4 +110,5 @@ def test_rtp_extraction(protocol, protocol_file) -> None: protocol=proto, parameter_context=parameter_context, run_time_param_overrides=run_time_param_overrides, + run_time_param_file_overrides={}, ).get_all() == {"sample_count": 2, "mount": "left"} diff --git a/api/tests/opentrons/protocols/parameters/test_csv_parameter_definition.py b/api/tests/opentrons/protocols/parameters/test_csv_parameter_definition.py index 70933cc326c..0bb257cabfd 100644 --- a/api/tests/opentrons/protocols/parameters/test_csv_parameter_definition.py +++ b/api/tests/opentrons/protocols/parameters/test_csv_parameter_definition.py @@ -6,7 +6,7 @@ import pytest from decoy import Decoy -from opentrons.protocol_engine.types import CSVParameter, FileId +from opentrons.protocol_engine.types import CSVParameter, FileInfo from opentrons.protocols.parameters import validation as mock_validation from opentrons.protocols.parameters.csv_parameter_definition import ( create_csv_parameter, @@ -49,7 +49,7 @@ def test_create_csv_parameter(decoy: Decoy) -> None: assert result.variable_name == "my_cool_csv" assert result._description == "Comma Separated Value" assert result.value is None - assert result.id is None + assert result.file_info is None def test_set_csv_value( @@ -75,13 +75,13 @@ def test_csv_parameter_as_protocol_engine_type( file=None, ) - csv_parameter_subject.id = "123abc" + csv_parameter_subject.file_info = FileInfo(id="123abc", name="") result = csv_parameter_subject.as_protocol_engine_type() assert result == CSVParameter( displayName="My cool CSV", variableName="my_cool_csv", description="Comma Separated Value", - file=FileId(id="123abc"), + file=FileInfo(id="123abc", name=""), ) diff --git a/api/tests/opentrons/protocols/test_parse.py b/api/tests/opentrons/protocols/test_parse.py index 11a39507238..2013b67c410 100644 --- a/api/tests/opentrons/protocols/test_parse.py +++ b/api/tests/opentrons/protocols/test_parse.py @@ -3,7 +3,7 @@ from typing import Any, Callable, Optional, Union, Literal import pytest -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocols.parse import ( PythonParseMode, diff --git a/api/tests/opentrons/test_execute.py b/api/tests/opentrons/test_execute.py index 77563083337..b259f6bada3 100644 --- a/api/tests/opentrons/test_execute.py +++ b/api/tests/opentrons/test_execute.py @@ -12,7 +12,7 @@ from _pytest.fixtures import SubRequest from opentrons_shared_data import get_shared_data_root, load_shared_data -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel from opentrons_shared_data.pipette import ( pipette_load_name_conversions as pipette_load_name, load_data as load_pipette_data, diff --git a/api/tests/opentrons/util/test_entrypoint_util.py b/api/tests/opentrons/util/test_entrypoint_util.py index c30351dec3b..cf1de7741ac 100644 --- a/api/tests/opentrons/util/test_entrypoint_util.py +++ b/api/tests/opentrons/util/test_entrypoint_util.py @@ -3,7 +3,7 @@ from pathlib import Path from typing import Callable -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.util.entrypoint_util import ( FoundLabware, labware_from_paths, diff --git a/api/tests/opentrons/util/test_performance_helpers.py b/api/tests/opentrons/util/test_performance_helpers.py index 88181091c34..1b5b2007558 100644 --- a/api/tests/opentrons/util/test_performance_helpers.py +++ b/api/tests/opentrons/util/test_performance_helpers.py @@ -3,7 +3,7 @@ from pathlib import Path from opentrons.util.performance_helpers import ( _StubbedTracker, - _get_robot_context_tracker, + _get_robot_activity_tracker, ) @@ -19,6 +19,6 @@ def func_to_track() -> None: def test_singleton_tracker() -> None: """Test that the tracker is a singleton.""" - tracker = _get_robot_context_tracker() - tracker2 = _get_robot_context_tracker() + tracker = _get_robot_activity_tracker() + tracker2 = _get_robot_activity_tracker() assert tracker is tracker2 diff --git a/app-shell-odd/src/config/__fixtures__/index.ts b/app-shell-odd/src/config/__fixtures__/index.ts index 5365d1cb075..d670234ebbc 100644 --- a/app-shell-odd/src/config/__fixtures__/index.ts +++ b/app-shell-odd/src/config/__fixtures__/index.ts @@ -11,6 +11,7 @@ import type { ConfigV21, ConfigV22, ConfigV23, + ConfigV24, } from '@opentrons/app/src/redux/config/types' const PKG_VERSION: string = _PKG_VERSION_ @@ -159,3 +160,14 @@ export const MOCK_CONFIG_V23: ConfigV23 = { hasDismissedQuickTransferIntro: false, }, } + +export const MOCK_CONFIG_V24: ConfigV24 = { + ...(() => { + const { support, ...rest } = MOCK_CONFIG_V23 + return rest + })(), + version: 24, + userInfo: { + userId: 'MOCK_UUIDv4', + }, +} diff --git a/app-shell-odd/src/config/__tests__/migrate.test.ts b/app-shell-odd/src/config/__tests__/migrate.test.ts index 1ef817da3be..dcc8eb03708 100644 --- a/app-shell-odd/src/config/__tests__/migrate.test.ts +++ b/app-shell-odd/src/config/__tests__/migrate.test.ts @@ -1,5 +1,7 @@ // config migration tests -import { describe, it, expect } from 'vitest' +import { describe, it, expect, beforeEach, vi } from 'vitest' +import uuid from 'uuid/v4' + import { MOCK_CONFIG_V12, MOCK_CONFIG_V13, @@ -13,18 +15,26 @@ import { MOCK_CONFIG_V21, MOCK_CONFIG_V22, MOCK_CONFIG_V23, + MOCK_CONFIG_V24, } from '../__fixtures__' import { migrate } from '../migrate' -const NEWEST_VERSION = 23 +vi.mock('uuid/v4') + +const NEWEST_VERSION = 24 +const NEWEST_MOCK_CONFIG = MOCK_CONFIG_V24 describe('config migration', () => { + beforeEach(() => { + vi.mocked(uuid).mockReturnValue('MOCK_UUIDv4') + }) + it('should migrate version 12 to latest', () => { const v12Config = MOCK_CONFIG_V12 const result = migrate(v12Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 13 to latest', () => { @@ -32,7 +42,7 @@ describe('config migration', () => { const result = migrate(v13Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 14 to latest', () => { @@ -40,7 +50,7 @@ describe('config migration', () => { const result = migrate(v14Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 15 to latest', () => { @@ -48,7 +58,7 @@ describe('config migration', () => { const result = migrate(v15Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 16 to latest', () => { @@ -56,7 +66,7 @@ describe('config migration', () => { const result = migrate(v16Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 17 to latest', () => { @@ -64,7 +74,7 @@ describe('config migration', () => { const result = migrate(v17Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 18 to latest', () => { @@ -72,7 +82,7 @@ describe('config migration', () => { const result = migrate(v18Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 19 to latest', () => { @@ -80,7 +90,7 @@ describe('config migration', () => { const result = migrate(v19Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 20 to latest', () => { @@ -88,27 +98,34 @@ describe('config migration', () => { const result = migrate(v20Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 21 to latest', () => { const v21Config = MOCK_CONFIG_V21 const result = migrate(v21Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 22 to latest', () => { const v22Config = MOCK_CONFIG_V22 const result = migrate(v22Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) - it('should keep version 23', () => { + it('should migrate version 23 to latest', () => { const v23Config = MOCK_CONFIG_V23 const result = migrate(v23Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(v23Config) + expect(result).toEqual(NEWEST_MOCK_CONFIG) + }) + it('should keep version 24', () => { + const v24Config = MOCK_CONFIG_V24 + const result = migrate(v24Config) + + expect(result.version).toBe(NEWEST_VERSION) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) }) diff --git a/app-shell-odd/src/config/migrate.ts b/app-shell-odd/src/config/migrate.ts index c6d667a8fc3..d1e9103d430 100644 --- a/app-shell-odd/src/config/migrate.ts +++ b/app-shell-odd/src/config/migrate.ts @@ -16,6 +16,7 @@ import type { ConfigV21, ConfigV22, ConfigV23, + ConfigV24, } from '@opentrons/app/src/redux/config/types' // format // base config v12 defaults @@ -214,6 +215,17 @@ const toVersion23 = (prevConfig: ConfigV22): ConfigV23 => { return nextConfig } +const toVersion24 = (prevConfig: ConfigV23): ConfigV24 => { + const { support, ...rest } = prevConfig + return { + ...rest, + version: 24 as const, + userInfo: { + userId: uuid(), + }, + } +} + const MIGRATIONS: [ (prevConfig: ConfigV12) => ConfigV13, (prevConfig: ConfigV13) => ConfigV14, @@ -225,7 +237,8 @@ const MIGRATIONS: [ (prevConfig: ConfigV19) => ConfigV20, (prevConfig: ConfigV20) => ConfigV21, (prevConfig: ConfigV21) => ConfigV22, - (prevConfig: ConfigV22) => ConfigV23 + (prevConfig: ConfigV22) => ConfigV23, + (prevConfig: ConfigV23) => ConfigV24 ] = [ toVersion13, toVersion14, @@ -238,6 +251,7 @@ const MIGRATIONS: [ toVersion21, toVersion22, toVersion23, + toVersion24, ] export const DEFAULTS: Config = migrate(DEFAULTS_V12) @@ -256,6 +270,7 @@ export function migrate( | ConfigV21 | ConfigV22 | ConfigV23 + | ConfigV24 ): Config { let result = prevConfig // loop through the migrations, skipping any migrations that are unnecessary diff --git a/app-shell-odd/src/usb.ts b/app-shell-odd/src/usb.ts index 6736c93f1ae..44252c6a339 100644 --- a/app-shell-odd/src/usb.ts +++ b/app-shell-odd/src/usb.ts @@ -199,11 +199,17 @@ const getLatestMassStorageCsvFiles = ( filePaths: string[], dispatch: Dispatch ): void => { - // Note (kk:06/28/2024) The following regex is mostly for Resource fork file ex ._test.csv - // Resource fork file would be on a usb flash drive if a user uses macOS. - const regex = /._\w/gm - const csvFilePaths = - filePaths.filter(path => !path.match(regex) && path.endsWith('.csv')) ?? [] + // Note (kk:07/29/2024) get all files' last path + // remove Resource fork that starts "._" + // remove hidden file that starts "." + const csvFilePaths = filePaths.filter(path => { + const fileName = path.split('/').pop() || '' + return ( + !fileName.startsWith('._') && + !fileName.startsWith('.') && + fileName.endsWith('.csv') + ) + }) dispatch(sendFilePaths(csvFilePaths)) } diff --git a/app-shell/Makefile b/app-shell/Makefile index 13e15e6e439..5daafd82f44 100644 --- a/app-shell/Makefile +++ b/app-shell/Makefile @@ -71,6 +71,15 @@ electron := yarn electron . \ --ui.url.path="localhost:$(PORT)" \ --python.pathToPythonOverride=$(shell cd ../api && pipenv --venv) +electron-dist := yarn electron . \ + --devtools \ + --log.level.console="debug" \ + --disable_ui.webPreferences.webSecurity \ + --ui.url.protocol="file:" \ + --ui.url.path="$(ui_dir)/dist/index.html" + --python.pathToPythonOverride=$(shell cd ../api && pipenv --venv) + + # standard targets ##################################################################### @@ -182,6 +191,13 @@ dev: dev-app-update-file vite build $(electron) +.PHONY: dev-dist +dev: export NODE_ENV := development +dev-dist: export OPENTRONS_PROJECT := $(OPENTRONS_PROJECT) +dev-dist: package-deps + vite build + $(electron-dist) + .PHONY: test test: $(MAKE) -C .. test-js-app-shell diff --git a/app-shell/build/license_en.txt b/app-shell/build/license_en.txt index cf847badf81..c764c7b3e8a 100644 --- a/app-shell/build/license_en.txt +++ b/app-shell/build/license_en.txt @@ -2,9 +2,9 @@ Opentrons End-User License Agreement Last updated: July 10, 2024 -THIS END-USER LICENSE AGREEMENT (“EULA”) is a legal agreement between you (“User”), either as an individual or on behalf of an entity, and Opentrons Labworks Inc. (“Opentrons”) regarding your use of Opentrons robots, modules, software, and associated documentation (“Opentrons Products”) including, but not limited to, the Opentrons OT-2 robot and associated modules, the Opentrons Flex robot and associated modules, the Opentrons App, the Opentrons API, the Opentrons Protocol Designer and Protocol Library, the Opentrons Labware Library, and the Opentrons Website. By installing or using the Opentrons Products, you agree to be bound by the terms and conditions of this EULA. If you do not agree to the terms of this EULA, you must immediately cease use of the Opentrons Products. +THIS END-USER LICENSE AGREEMENT ("EULA") is a legal agreement between you ("User"), either as an individual or on behalf of an entity, and Opentrons Labworks Inc. ("Opentrons") regarding your use of Opentrons robots, modules, software, and associated documentation ("Opentrons Products") including, but not limited to, the Opentrons OT-2 robot and associated modules, the Opentrons Flex robot and associated modules, the Opentrons App, the Opentrons API, the Opentrons Protocol Designer and Protocol Library, the Opentrons Labware Library, and the Opentrons Website. By installing or using the Opentrons Products, you agree to be bound by the terms and conditions of this EULA. If you do not agree to the terms of this EULA, you must immediately cease use of the Opentrons Products. -License Grant. Opentrons grants User a revocable, non-exclusive, non-transferable, limited license to access and use the Opentrons Products strictly in accordance with the terms and conditions of this EULA, the Opentrons Terms and Conditions of Sale, the Opentrons Privacy Policy, and any other agreements between User and Opentrons (collectively “Related Agreements”). +License Grant. Opentrons grants User a revocable, non-exclusive, non-transferable, limited license to access and use the Opentrons Products strictly in accordance with the terms and conditions of this EULA, the Opentrons Terms and Conditions of Sale, the Opentrons Privacy Policy, and any other agreements between User and Opentrons (collectively "Related Agreements"). Use of Opentrons Products. Permitted Use. User shall use the Opentrons Products strictly in accordance with the terms of the EULA and Related Agreements. User shall use Opentrons Product software only in conjunction with Opentrons Product hardware. Restrictions on Use. Unless otherwise specified in a separate agreement entered into between Opentrons and User, User may not, and may not permit others to: @@ -25,15 +25,15 @@ Intellectual Property Rights. Opentrons retains all rights, title, and interest Ownership. All worldwide patents, copyrights, trade secrets, and other intellectual property rights related to the Opentrons Products are the exclusive property of Opentrons. Feedback. Any feedback or suggestions provided by User regarding the Opentrons Products may be used by Opentrons without any obligation to User, and User hereby provides Opentrons a perpetual, global, fully paid-up, royalty free license to use such feedback or suggestions. Privacy Notices. The Opentrons Products may automatically communicate with Opentrons servers and transmit data to Opentrons for various purposes including, but not limited to: 1. updating Opentrons Product software; 2. sending error reports to Opentrons; and 3. sending Opentrons Product usage data to Opentrons. The collection and use of such data is governed by the Opentrons Privacy Policy. By agreeing to this EULA you also acknowledge and agree to the Opentrons Privacy Policy. If User does not agree to the Opentrons Privacy or to the collection of Opentrons Product data, User must immediately cease all use of Opentrons Products and destroy all copies of Opentrons Product software. -Disclaimer of Warranties. THE OPENTRONS PRODUCTS ARE PROVIDED ON AN “AS IS” BASIS AND NO WARRANTY, EITHER EXPRESS OR IMPLIED, IS GIVEN. OPENTRONS DISCLAIMS ALL REPRESENTATIONS, WARRANTIES AND CONDITIONS, EXPRESS, IMPLIED OR COLLATERAL, INCLUDING AS TO OWNERSHIP AND NON-INFRINGEMENT, THE IMPLIED WARRANTIES OR CONDITIONS OF MERCHANTABILITY, MERCHANTABLE QUALITY, FITNESS FOR A PARTICULAR PURPOSE, AND THOSE ARISING BY STATUTE OR OTHERWISE IN LAW, OR FROM THE COURSE OF DEALING OR USAGE OF TRADE. WITHOUT LIMITING THE FOREGOING, OPENTRONS DOES NOT REPRESENT OR WARRANT THAT THE OPENTRONS PRODUCTS WILL MEET ANY OR ALL OF YOUR PARTICULAR REQUIREMENTS, THAT THE OPERATION OF THE OPENTRONS PRODUCTS WILL BE ERROR FREE OR UNINTERRUPTED OR THAT ALL PROGRAMMING ERRORS IN THE OPENTRONS PRODUCTS CAN BE FOUND IN ORDER TO BE CORRECTED. -Limitation of Liability. To the maximum extent permitted by applicable law, in no event shall Opentrons, its affiliates, shareholders, directors, officers, employees and agents be liable for any special, incidental, indirect, exemplary, or consequential damages whatsoever (including, without limitation, damages for loss of business profits, business interruption, loss of business information, or any other pecuniary loss) arising out of your use or inability to use the Opentrons Products, whether or not the damages were foreseeable and whether or not Opentrons was advised of the possibility of such damages. In any case, without limiting the foregoing, Opentrons’ entire liability arising from or under any provision of this EULA or from the use of the Opentrons Products shall be limited to fifty dollars ($50.00). The foregoing limitations will apply even if the above stated remedy fails in its essential purpose. +Disclaimer of Warranties. THE OPENTRONS PRODUCTS ARE PROVIDED ON AN "AS IS" BASIS AND NO WARRANTY, EITHER EXPRESS OR IMPLIED, IS GIVEN. OPENTRONS DISCLAIMS ALL REPRESENTATIONS, WARRANTIES AND CONDITIONS, EXPRESS, IMPLIED OR COLLATERAL, INCLUDING AS TO OWNERSHIP AND NON-INFRINGEMENT, THE IMPLIED WARRANTIES OR CONDITIONS OF MERCHANTABILITY, MERCHANTABLE QUALITY, FITNESS FOR A PARTICULAR PURPOSE, AND THOSE ARISING BY STATUTE OR OTHERWISE IN LAW, OR FROM THE COURSE OF DEALING OR USAGE OF TRADE. WITHOUT LIMITING THE FOREGOING, OPENTRONS DOES NOT REPRESENT OR WARRANT THAT THE OPENTRONS PRODUCTS WILL MEET ANY OR ALL OF YOUR PARTICULAR REQUIREMENTS, THAT THE OPERATION OF THE OPENTRONS PRODUCTS WILL BE ERROR FREE OR UNINTERRUPTED OR THAT ALL PROGRAMMING ERRORS IN THE OPENTRONS PRODUCTS CAN BE FOUND IN ORDER TO BE CORRECTED. +Limitation of Liability. To the maximum extent permitted by applicable law, in no event shall Opentrons, its affiliates, shareholders, directors, officers, employees and agents be liable for any special, incidental, indirect, exemplary, or consequential damages whatsoever (including, without limitation, damages for loss of business profits, business interruption, loss of business information, or any other pecuniary loss) arising out of your use or inability to use the Opentrons Products, whether or not the damages were foreseeable and whether or not Opentrons was advised of the possibility of such damages. In any case, without limiting the foregoing, Opentrons' entire liability arising from or under any provision of this EULA or from the use of the Opentrons Products shall be limited to fifty dollars ($50.00). The foregoing limitations will apply even if the above stated remedy fails in its essential purpose. General Provisions. User Representations. User represents and warrants that User is not located in a country that is subject to a United States government embargo, or that has been designated by the United States government as a "terrorist supporting" country. User represents and warrants that User is not listed on any United States government list of prohibited or restricted parties. User represents and warrants that they will comply with all United States export laws and regulations applicable to their possession and use of the Opentrons Products. Amendment; Waiver. This EULA shall not be modified or amended except by a written document executed by the parties. No waiver by Opentrons or any failure by Opentrons to keep or perform any provision, covenant or condition of this EULA shall be deemed to be a waiver of any preceding or succeeding breach of the same or of any other provision, covenant, or condition. Any waiver to be granted by Opentrons shall not be effective unless it is set forth in a written instrument signed by Opentrons. Assignment; Successors and Assigns. User may not assign this EULA or any rights, interests, claims or obligations under this EULA without the prior written consent of Opentrons. This EULA shall be binding upon and shall inure to the benefit of the parties and their respective successors, representatives and permitted assigns. Governing Law; Venue. This EULA shall be construed and governed by the laws of the State of New York without regard to any conflicts of law provisions or rules that would operate to cause the application of the laws of any other jurisdiction. The exclusive jurisdiction and venue for all actions under this EULA will be in the state or federal courts of competent jurisdiction in New York County, NY. Survival. All provisions of this EULA reasonably expected to survive the termination or expiration of this EULA shall do so. -Severability. Whenever possible, each provision of this EULA shall be interpreted in such manner as to be effective and valid under applicable law, but if any provision of this EULA is held to be invalid, illegal or unenforceable in any respect under any applicable law or rule in any jurisdiction, such invalidity, illegality or unenforceability shall not affect any other provision of this EULA or the parties’ rights and obligations under this EULA in any other jurisdiction. Instead, this EULA shall be reformed, construed and enforced in such jurisdiction to include an amended or modified version of the provision held to be invalid, illegal, or unenforceable or, if amendment or modification is impossible, as if such invalid, illegal or unenforceable provision had never been contained herein. +Severability. Whenever possible, each provision of this EULA shall be interpreted in such manner as to be effective and valid under applicable law, but if any provision of this EULA is held to be invalid, illegal or unenforceable in any respect under any applicable law or rule in any jurisdiction, such invalidity, illegality or unenforceability shall not affect any other provision of this EULA or the parties' rights and obligations under this EULA in any other jurisdiction. Instead, this EULA shall be reformed, construed and enforced in such jurisdiction to include an amended or modified version of the provision held to be invalid, illegal, or unenforceable or, if amendment or modification is impossible, as if such invalid, illegal or unenforceable provision had never been contained herein. Captions. The captions or section headings used in this EULA are for convenience only and shall not affect the construction, interpretation or meaning of any term or provision of this EULA. Amendments. Opentrons reserves the right to amend this EULA at any time by providing notice to the User. Continued use of the Opentrons Products following such notice constitutes acceptance of the amended EULA. diff --git a/app-shell/build/release-notes-internal.md b/app-shell/build/release-notes-internal.md index c2193890ead..6967672777e 100644 --- a/app-shell/build/release-notes-internal.md +++ b/app-shell/build/release-notes-internal.md @@ -1,15 +1,31 @@ For more details about this release, please see the full [technical changelog][]. [technical change log]: https://github.com/Opentrons/opentrons/releases +## Internal Release 2.0.0-alpha.3 + +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. + +- [Opentrons changes since the latest stable release](https://github.com/Opentrons/opentrons/compare/v7.3.1...ot3@2.0.0-alpha.3) +- [Opentrons changes since the last internal release](https://github.com/Opentrons/opentrons/compare/ot3@2.0.0-alpha.2...ot3@2.0.0-alpha.3) +- [Flex changes](https://github.com/Opentrons/oe-core/compare/internal@2.0.0-alpha.2...internal@2.0.0-alpha.3) +- [Flex firmware changes](https://github.com/Opentrons/ot3-firmware/compare/internal@v9...internal@v10) +- [OT2 changes](https://github.com/Opentrons/buildroot/compare/v1.17.7...internal@2.0.0-alpha.0) + +## Internal Release 2.0.0-alpha.2 + +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. + + + ## Internal Release 2.0.0-alpha.1 -This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. Usage may require a robot factory reset to restore robot stability. +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. ## Internal Release 2.0.0-alpha.0 -This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. Usage may require a robot factory reset to restore robot stability. +This internal release, pulled from the `edge` branch, contains features being developed for 8.0.0. It's for internal testing only. diff --git a/app-shell/build/release-notes.md b/app-shell/build/release-notes.md index 60adacf1a4a..ffdf4fad357 100644 --- a/app-shell/build/release-notes.md +++ b/app-shell/build/release-notes.md @@ -6,11 +6,11 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- -## Opentrons App Changes in 7.4.0 +## Opentrons App Changes in 7.5.0 -Welcome to the v7.4.0 release of the Opentrons App! +Welcome to the v7.5.0 release of the Opentrons App! -This release adds support for the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module). +There are no changes to the Opentrons App in v7.5.0, but it is required for updating the robot software to support the [Opentrons Flex HEPA/UV Module](https://opentrons.com/products/opentrons-flex-hepa-uv-module) and the latest Flex Gripper model (serial numbers beginning `GRPV13`). --- diff --git a/app-shell/src/__fixtures__/config.ts b/app-shell/src/__fixtures__/config.ts index c118630c09d..23ef4f56f90 100644 --- a/app-shell/src/__fixtures__/config.ts +++ b/app-shell/src/__fixtures__/config.ts @@ -23,6 +23,7 @@ import type { ConfigV21, ConfigV22, ConfigV23, + ConfigV24, } from '@opentrons/app/src/redux/config/types' export const MOCK_CONFIG_V0: ConfigV0 = { @@ -290,3 +291,14 @@ export const MOCK_CONFIG_V23: ConfigV23 = { hasDismissedQuickTransferIntro: false, }, } + +export const MOCK_CONFIG_V24: ConfigV24 = { + ...(() => { + const { support, ...rest } = MOCK_CONFIG_V23 + return rest + })(), + version: 24, + userInfo: { + userId: 'MOCK_UUIDv4', + }, +} diff --git a/app-shell/src/config/__tests__/migrate.test.ts b/app-shell/src/config/__tests__/migrate.test.ts index d95109d8661..dee16e0dae4 100644 --- a/app-shell/src/config/__tests__/migrate.test.ts +++ b/app-shell/src/config/__tests__/migrate.test.ts @@ -1,5 +1,7 @@ // config migration tests -import { describe, it, expect } from 'vitest' +import { describe, it, expect, beforeEach, vi } from 'vitest' +import uuid from 'uuid/v4' + import { MOCK_CONFIG_V0, MOCK_CONFIG_V1, @@ -25,18 +27,26 @@ import { MOCK_CONFIG_V21, MOCK_CONFIG_V22, MOCK_CONFIG_V23, + MOCK_CONFIG_V24, } from '../../__fixtures__' import { migrate } from '../migrate' -const NEWEST_VERSION = 23 +vi.mock('uuid/v4') + +const NEWEST_VERSION = 24 +const NEWEST_MOCK_CONFIG = MOCK_CONFIG_V24 describe('config migration', () => { + beforeEach(() => { + vi.mocked(uuid).mockReturnValue('MOCK_UUIDv4') + }) + it('should migrate version 0 to latest', () => { const v0Config = MOCK_CONFIG_V0 const result = migrate(v0Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 1 to latest', () => { @@ -44,7 +54,7 @@ describe('config migration', () => { const result = migrate(v1Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 2 to latest', () => { @@ -52,7 +62,7 @@ describe('config migration', () => { const result = migrate(v2Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 3 to latest', () => { @@ -60,7 +70,7 @@ describe('config migration', () => { const result = migrate(v3Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 4 to latest', () => { @@ -68,7 +78,7 @@ describe('config migration', () => { const result = migrate(v4Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 5 to latest', () => { @@ -76,7 +86,7 @@ describe('config migration', () => { const result = migrate(v5Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 6 to latest', () => { @@ -84,7 +94,7 @@ describe('config migration', () => { const result = migrate(v6Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 7 to latest', () => { @@ -92,7 +102,7 @@ describe('config migration', () => { const result = migrate(v7Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 8 to latest', () => { @@ -100,7 +110,7 @@ describe('config migration', () => { const result = migrate(v8Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 9 to latest', () => { @@ -108,7 +118,7 @@ describe('config migration', () => { const result = migrate(v9Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 10 to latest', () => { @@ -116,7 +126,7 @@ describe('config migration', () => { const result = migrate(v10Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 11 to latest', () => { @@ -124,7 +134,7 @@ describe('config migration', () => { const result = migrate(v11Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 12 to latest', () => { @@ -132,7 +142,7 @@ describe('config migration', () => { const result = migrate(v12Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 13 to latest', () => { @@ -140,7 +150,7 @@ describe('config migration', () => { const result = migrate(v13Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 14 to latest', () => { @@ -148,7 +158,7 @@ describe('config migration', () => { const result = migrate(v14Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 15 to latest', () => { @@ -156,7 +166,7 @@ describe('config migration', () => { const result = migrate(v15Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 16 to latest', () => { @@ -164,7 +174,7 @@ describe('config migration', () => { const result = migrate(v16Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 17 to latest', () => { @@ -172,48 +182,55 @@ describe('config migration', () => { const result = migrate(v17Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migrate version 18 to latest', () => { const v18Config = MOCK_CONFIG_V18 const result = migrate(v18Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should keep migrate version 19 to latest', () => { const v19Config = MOCK_CONFIG_V19 const result = migrate(v19Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 20 to latest', () => { const v20Config = MOCK_CONFIG_V20 const result = migrate(v20Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 21 to latest', () => { const v21Config = MOCK_CONFIG_V21 const result = migrate(v21Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) it('should migration version 22 to latest', () => { const v22Config = MOCK_CONFIG_V22 const result = migrate(v22Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) - it('should keep version 23', () => { + it('should migrate version 23 to latest', () => { const v23Config = MOCK_CONFIG_V23 const result = migrate(v23Config) expect(result.version).toBe(NEWEST_VERSION) - expect(result).toEqual(MOCK_CONFIG_V23) + expect(result).toEqual(NEWEST_MOCK_CONFIG) + }) + it('should keep version 24', () => { + const v24Config = MOCK_CONFIG_V24 + const result = migrate(v24Config) + + expect(result.version).toBe(NEWEST_VERSION) + expect(result).toEqual(NEWEST_MOCK_CONFIG) }) }) diff --git a/app-shell/src/config/migrate.ts b/app-shell/src/config/migrate.ts index d9c4b063213..fa9ed4a91dd 100644 --- a/app-shell/src/config/migrate.ts +++ b/app-shell/src/config/migrate.ts @@ -27,6 +27,7 @@ import type { ConfigV21, ConfigV22, ConfigV23, + ConfigV24, } from '@opentrons/app/src/redux/config/types' // format // base config v0 defaults @@ -418,6 +419,17 @@ const toVersion23 = (prevConfig: ConfigV22): ConfigV23 => { return nextConfig } +const toVersion24 = (prevConfig: ConfigV23): ConfigV24 => { + const { support, ...rest } = prevConfig + return { + ...rest, + version: 24 as const, + userInfo: { + userId: uuid(), + }, + } +} + const MIGRATIONS: [ (prevConfig: ConfigV0) => ConfigV1, (prevConfig: ConfigV1) => ConfigV2, @@ -441,7 +453,8 @@ const MIGRATIONS: [ (prevConfig: ConfigV19) => ConfigV20, (prevConfig: ConfigV20) => ConfigV21, (prevConfig: ConfigV21) => ConfigV22, - (prevConfig: ConfigV22) => ConfigV23 + (prevConfig: ConfigV22) => ConfigV23, + (prevConfig: ConfigV23) => ConfigV24 ] = [ toVersion1, toVersion2, @@ -466,6 +479,7 @@ const MIGRATIONS: [ toVersion21, toVersion22, toVersion23, + toVersion24, ] export const DEFAULTS: Config = migrate(DEFAULTS_V0) @@ -496,6 +510,7 @@ export function migrate( | ConfigV21 | ConfigV22 | ConfigV23 + | ConfigV24 ): Config { const prevVersion = prevConfig.version let result = prevConfig diff --git a/app-shell/src/menu.ts b/app-shell/src/menu.ts index 52f04978934..7b3abc186a1 100644 --- a/app-shell/src/menu.ts +++ b/app-shell/src/menu.ts @@ -8,6 +8,8 @@ import { LOG_DIR } from './log' const PRODUCT_NAME: string = _PKG_PRODUCT_NAME_ const BUGS_URL: string = _PKG_BUGS_URL_ +const EULA_URL = 'https://opentrons.com/eula' as const + // file or application menu const firstMenu: MenuItemConstructorOptions = { role: process.platform === 'darwin' ? 'appMenu' : 'fileMenu', @@ -42,6 +44,16 @@ const helpMenu: MenuItemConstructorOptions = { shell.openExternal(BUGS_URL) }, }, + { + label: 'View Privacy Policy', + click: () => { + shell.openExternal(EULA_URL).catch((e: Error) => { + console.error( + `could not open end user license agreement: ${e.message}` + ) + }) + }, + }, ], } diff --git a/app/Makefile b/app/Makefile index 92d0343507a..51305c9b3b0 100644 --- a/app/Makefile +++ b/app/Makefile @@ -74,6 +74,11 @@ dev-server: export OPENTRONS_PROJECT := $(OPENTRONS_PROJECT) dev-server: vite serve +.PHONY: dev-dist +dev-dist: export NODE_ENV := development +dev-dist: + $(MAKE) -C $(shell_dir) dev-dist OPENTRONS_PROJECT=$(OPENTRONS_PROJECT) + .PHONY: dev-shell dev-shell: $(MAKE) -C $(shell_dir) dev OPENTRONS_PROJECT=$(OPENTRONS_PROJECT) diff --git a/app/src/App/DesktopApp.tsx b/app/src/App/DesktopApp.tsx index ab0b91f7c9c..c6eeadc278f 100644 --- a/app/src/App/DesktopApp.tsx +++ b/app/src/App/DesktopApp.tsx @@ -144,7 +144,7 @@ export const DesktopApp = (): JSX.Element => { /> ) })} - } /> + } /> diff --git a/app/src/assets/images/change-pip/1_and_8_channel.png b/app/src/assets/images/change-pip/1_and_8_channel.png index 15c15bbacdc..d9091b899ad 100644 Binary files a/app/src/assets/images/change-pip/1_and_8_channel.png and b/app/src/assets/images/change-pip/1_and_8_channel.png differ diff --git a/app/src/assets/localization/en/drop_tip_wizard.json b/app/src/assets/localization/en/drop_tip_wizard.json index 16a9fab24b6..4bedd4bc8e6 100644 --- a/app/src/assets/localization/en/drop_tip_wizard.json +++ b/app/src/assets/localization/en/drop_tip_wizard.json @@ -17,13 +17,15 @@ "getting_ready": "Getting ready…", "go_back": "go back", "jog_too_far": "Jog too far?", - "start_over": "Start over", + "liquid_damages_pipette": "Homing the pipette with liquid in the tips may damage it. You must remove all tips before using the pipette again.", + "liquid_damages_this_pipette": "Homing the {{mount}} pipette with liquid in the tips may damage it. You must remove all tips before using the pipette again.", "move_to_slot": "move to slot", "no_proceed_to_drop_tip": "No, proceed to tip removal", "position_and_blowout": "Ensure that the pipette tip is centered above and level with where you want the liquid to be blown out. If it isn't, use the controls below or your keyboard to jog the pipette until it is properly aligned.", "position_and_drop_tip": "Ensure that the pipette tip is centered above and level with where you want to drop the tips. If it isn't, use the controls below or your keyboard to jog the pipette until it is properly aligned.", "position_the_pipette": "position the pipette", - "remove_the_tips": "You may want to remove the tips from the {{mount}} Pipette before using it again in a protocol.", + "remove_any_attached_tips": "Remove any attached tips", + "remove_attached_tips": "Remove any attached tips", "remove_the_tips_from_pipette": "You may want to remove the tips from the pipette before using it again in a protocol.", "remove_the_tips_manually": "Remove the tips manually. Then home the gantry. Homing with tips attached could pull liquid into the pipette and damage it.", "remove_tips": "Remove tips", @@ -35,7 +37,6 @@ "stand_back_blowing_out": "Stand back, robot is blowing out liquid", "stand_back_dropping_tips": "Stand back, robot is dropping tips", "stand_back_robot_in_motion": "Stand back, robot is in motion", - "tips_are_attached": "Tips are attached", - "tips_may_be_attached": "Tips may be attached.", + "start_over": "Start over", "yes_blow_out_liquid": "Yes, blow out liquid in labware" } diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index c139f21acd2..f7eb5c5a565 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -2,6 +2,7 @@ "are_you_sure_you_want_to_cancel": "Are you sure you want to cancel?", "at_step": "At step", "back_to_menu": "Back to menu", + "another_app_controlling_robot": "The robot’s touchscreen or another computer with the app is currently controlling this robot.", "before_you_begin": "Before you begin", "begin_removal": "Begin removal", "blowout_failed": "Blowout failed", @@ -16,11 +17,13 @@ "continue_run_now": "Continue run now", "continue_to_drop_tip": "Continue to drop tip", "error": "Error", + "error_details": "Error details", "error_on_robot": "Error on {{robot}}", - "failed_dispense_step_not_completed": "The failed dispense step will not be completed. The run will continue from the next step.Close the robot door before proceeding.", + "failed_dispense_step_not_completed": "The failed dispense step will not be completed. The run will continue from the next step with the attached tips.Close the robot door before proceeding.", "failed_step": "Failed step", "first_take_any_necessary_actions": "First, take any necessary actions to prepare the robot to retry the failed step.Then, close the robot door before proceeding.", "go_back": "Go back", + "homing_pipette_dangerous": "Homing the {{mount}} pipette with liquid in the tips may damage it. You must remove all tips before using the pipette again.", "if_tips_are_attached": "If tips are attached, you can choose to blow out any aspirated liquid and drop tips before the run is terminated.", "ignore_all_errors_of_this_type": "Ignore all errors of this type", "ignore_error_and_skip": "Ignore error and skip to next step", @@ -40,6 +43,7 @@ "recovery_action_failed": "{{action}} failed", "recovery_mode": "Recovery Mode", "recovery_mode_explanation": "Recovery Mode provides you with guided and manual controls for handling errors at runtime.
You can make changes to ensure the step in progress when the error occurred can be completed or choose to cancel the protocol. When changes are made and no subsequent errors are detected, the method completes. Depending on the conditions that caused the error, you will only be provided with appropriate options.", + "remove_any_attached_tips": "Remove any attached tips", "replace_tips_and_select_location": "It's best to replace tips and select the last location used for tip pickup.", "replace_used_tips_in_rack_location": "Replace used tips in rack location {{location}} in slot {{slot}}", "replace_with_new_tip_rack": "Replace with new tip rack in slot {{slot}}", @@ -52,13 +56,15 @@ "return_to_menu": "Return to menu", "return_to_the_menu": "Return to the menu to choose how to proceed.", "robot_door_is_open": "Robot door is open", + "robot_is_canceling_run": "Robot is canceling the run", + "robot_is_in_recovery_mode": "Robot is in recovery mode", "robot_will_not_check_for_liquid": "The robot will not check for liquid again. The run will continue from the next step.Close the robot door before proceeding.", "robot_will_retry_with_new_tips": "The robot will retry the failed step with the new tips.Close the robot door before proceeding.", "robot_will_retry_with_same_tips": "The robot will retry the failed step with the same tips.Close the robot door before proceeding.", "robot_will_retry_with_tips": "The robot will retry the failed step with new tips.", "run_paused": "Run paused", "select_tip_pickup_location": "Select tip pick-up location", - "skip_removal": "Skip removal", + "skip": "Skip", "skip_to_next_step": "Skip to next step", "skip_to_next_step_new_tips": "Skip to next step with new tips", "skip_to_next_step_same_tips": "Skip to next step with same tips", @@ -68,10 +74,10 @@ "stand_back_resuming": "Stand back, resuming current step", "stand_back_retrying": "Stand back, retrying failed step", "stand_back_skipping_to_next_step": "Stand back, skipping to next step", + "terminate_remote_activity": "Terminate remote activity", "tip_drop_failed": "Tip drop failed", "tip_not_detected": "Tip not detected", "view_error_details": "View error details", "view_recovery_options": "View recovery options", - "you_can_still_drop_tips": "You can still drop the attached tips before proceeding to tip selection.", - "remove_tips_from_pipette": "Remove tips from {{mount}} pipette before canceling the run?" + "you_can_still_drop_tips": "You can still drop the attached tips before proceeding to tip selection." } diff --git a/app/src/assets/localization/en/run_details.json b/app/src/assets/localization/en/run_details.json index 9209e9e5fc2..ab9a8114f5d 100644 --- a/app/src/assets/localization/en/run_details.json +++ b/app/src/assets/localization/en/run_details.json @@ -87,6 +87,7 @@ "protocol_title": "Protocol - {{protocol_name}}", "resume_run": "Resume run", "return_to_dashboard": "Return to dashboard", + "return_to_quick_transfer": "Return to quick transfer", "right": "Right", "robot_has_previous_offsets": "This robot has stored Labware Offset data from previous protocol runs. Do you want to apply that data to this protocol run? You can still adjust any offsets with Labware Position Check.", "robot_was_recalibrated": "This robot was recalibrated after this Labware Offset data was stored.", diff --git a/app/src/assets/localization/en/shared.json b/app/src/assets/localization/en/shared.json index 996ed8326d2..0b580a612e8 100644 --- a/app/src/assets/localization/en/shared.json +++ b/app/src/assets/localization/en/shared.json @@ -2,6 +2,7 @@ "a_software_update_is_available": "A software update is available for this robot. Update to run protocols.", "add": "add", "alphabetical": "Alphabetical", + "another_app_controlling_robot": "The robot's touchscreen or another app may be controlling this robot.", "back": "Back", "before_you_begin": "Before you begin", "browse": "browse", @@ -9,15 +10,15 @@ "change_protocol": "Change protocol", "change_robot": "Change robot", "clear_data": "clear data", - "close_robot_door": "Close the robot door before starting the run.", "close": "close", + "close_robot_door": "Close the robot door before starting the run.", + "confirm": "Confirm", "confirm_placement": "Confirm placement", "confirm_position": "Confirm position", "confirm_values": "Confirm values", - "confirm": "Confirm", + "continue": "continue", "continue_activity": "Continue activity", "continue_to_param": "Continue to parameters", - "continue": "continue", "delete": "Delete", "did_pipette_pick_up_tip": "Did pipette pick up tip successfully?", "disabled_cannot_connect": "Cannot connect to robot", @@ -28,8 +29,8 @@ "drag_and_drop": "Drag and drop or browse your files", "empty": "empty", "ending": "ending", - "error_encountered": "Error encountered", "error": "error", + "error_encountered": "Error encountered", "exit": "exit", "extension_mount": "extension mount", "flow_complete": "{{flowName}} complete!", @@ -39,8 +40,8 @@ "instruments": "instruments", "loading": "Loading...", "next": "Next", - "no_data": "no data", "no": "no", + "no_data": "no data", "none": "None", "not_used": "Not Used", "off": "Off", @@ -50,18 +51,18 @@ "proceed_to_setup": "Proceed to setup", "protocol_run_general_error_msg": "Protocol run could not be created on the robot.", "reanalyze": "Reanalyze", - "refresh_list": "Refresh list", "refresh": "refresh", + "refresh_list": "Refresh list", "remember_my_selection_and_do_not_ask_again": "Remember my selection and don't ask again", - "reset_all": "Reset all", "reset": "Reset", + "reset_all": "Reset all", "restart": "restart", "resume": "resume", "return": "return", "reverse": "Reverse alphabetical", "robot_is_analyzing": "Robot is analyzing", - "robot_is_busy_no_protocol_run_allowed": "This robot is busy and can’t run this protocol right now. Go to Robot", "robot_is_busy": "Robot is busy", + "robot_is_busy_no_protocol_run_allowed": "This robot is busy and can’t run this protocol right now. Go to Robot", "robot_is_reachable_but_not_responding": "This robot's API server is not responding correctly to requests at IP address {{hostname}}", "robot_was_seen_but_is_unreachable": "This robot has been seen recently, but is currently not reachable at IP address {{hostname}}", "save": "save", @@ -72,11 +73,11 @@ "starting": "starting", "step": "Step {{current}} / {{max}}", "stop": "stop", - "terminate_activity": "Terminate activity", "terminate": "Terminate remote activity", + "terminate_activity": "Terminate activity", "try_again": "try again", - "unknown_error": "An unknown error occurred", "unknown": "unknown", + "unknown_error": "An unknown error occurred", "update": "Update", "view_latest_release_notes": "View latest release notes on", "yes": "yes", diff --git a/app/src/atoms/buttons/RadioButton.tsx b/app/src/atoms/buttons/RadioButton.tsx index f22c19ef6e7..7876866d56d 100644 --- a/app/src/atoms/buttons/RadioButton.tsx +++ b/app/src/atoms/buttons/RadioButton.tsx @@ -20,6 +20,7 @@ interface RadioButtonProps extends StyleProps { isSelected?: boolean radioButtonType?: 'large' | 'small' subButtonLabel?: string + id?: string } export function RadioButton(props: RadioButtonProps): JSX.Element { @@ -31,6 +32,7 @@ export function RadioButton(props: RadioButtonProps): JSX.Element { onChange, radioButtonType = 'large', subButtonLabel, + id = buttonLabel, } = props const isLarge = radioButtonType === 'large' @@ -84,12 +86,12 @@ export function RadioButton(props: RadioButtonProps): JSX.Element { - + ) => { return renderWithProviders()[0] @@ -20,6 +20,7 @@ describe('RadioButton', () => { buttonValue: 1, } }) + it('renders the large button', () => { props = { ...props, @@ -30,6 +31,7 @@ describe('RadioButton', () => { expect(label).toHaveStyle(`background-color: ${COLORS.blue35}`) expect(label).toHaveStyle(`padding: ${SPACING.spacing24}`) }) + it('renders the large selected button', () => { props = { ...props, @@ -41,6 +43,7 @@ describe('RadioButton', () => { expect(label).toHaveStyle(`background-color: ${COLORS.blue50}`) expect(label).toHaveStyle(`padding: ${SPACING.spacing24}`) }) + it('renders the small button', () => { props = { ...props, @@ -51,6 +54,7 @@ describe('RadioButton', () => { expect(label).toHaveStyle(`background-color: ${COLORS.blue35}`) expect(label).toHaveStyle(`padding: ${SPACING.spacing20}`) }) + it('renders the small selected button', () => { props = { ...props, @@ -62,4 +66,23 @@ describe('RadioButton', () => { expect(label).toHaveStyle(`background-color: ${COLORS.blue50}`) expect(label).toHaveStyle(`padding: ${SPACING.spacing20}`) }) + + it('renders id instead of buttonLabel when id is set', () => { + props = { + ...props, + id: 'mock-radio-button-id', + } + render(props) + const getById = queryByAttribute.bind(null, 'id') + const idRadioButton = getById( + render(props).container, + 'mock-radio-button-id' + ) + expect(idRadioButton).toBeInTheDocument() + const buttonLabelIdRadioButton = getById( + render(props).container, + props.buttonLabel + ) + expect(buttonLabelIdRadioButton).not.toBeInTheDocument() + }) }) diff --git a/app/src/index.tsx b/app/src/index.tsx index b8fe832abdc..cf4fcbfc44c 100644 --- a/app/src/index.tsx +++ b/app/src/index.tsx @@ -2,7 +2,7 @@ import React from 'react' import ReactDom from 'react-dom/client' import { Provider } from 'react-redux' -import { BrowserRouter } from 'react-router-dom' +import { HashRouter } from 'react-router-dom' import { ApiClientProvider } from '@opentrons/react-api-client' @@ -32,10 +32,10 @@ if (container == null) throw new Error('Failed to find the root element') const root = ReactDom.createRoot(container) root.render( - + - + ) diff --git a/app/src/molecules/Command/hooks/useCommandTextString/index.tsx b/app/src/molecules/Command/hooks/useCommandTextString/index.tsx index 34df2f33c7f..093593dc911 100644 --- a/app/src/molecules/Command/hooks/useCommandTextString/index.tsx +++ b/app/src/molecules/Command/hooks/useCommandTextString/index.tsx @@ -216,6 +216,7 @@ export function useCommandTextString( commandText: utils.getCustomCommandText({ ...fullParams, command }), } + case undefined: case null: return { commandText: '' } diff --git a/app/src/molecules/InterventionModal/index.tsx b/app/src/molecules/InterventionModal/index.tsx index 3faa3b34f2c..aec8c9fea22 100644 --- a/app/src/molecules/InterventionModal/index.tsx +++ b/app/src/molecules/InterventionModal/index.tsx @@ -179,6 +179,8 @@ const ICON_STYLE = css` width: ${SPACING.spacing16}; height: ${SPACING.spacing16}; margin: ${SPACING.spacing4}; + cursor: pointer; + @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { width: ${SPACING.spacing32}; height: ${SPACING.spacing32}; diff --git a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx index 2c3e27b43df..8c35b7e5c04 100644 --- a/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx +++ b/app/src/organisms/CalibrationTaskList/__tests__/CalibrationTaskList.test.tsx @@ -26,7 +26,7 @@ import { import { mockLeftProtoPipette } from '../../../redux/pipettes/__fixtures__' vi.mock('../../Devices/hooks') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') const render = (robotName: string = 'otie') => { return renderWithProviders( diff --git a/app/src/organisms/CalibrationTaskList/index.tsx b/app/src/organisms/CalibrationTaskList/index.tsx index 301a6d1e2b8..d72a5ced341 100644 --- a/app/src/organisms/CalibrationTaskList/index.tsx +++ b/app/src/organisms/CalibrationTaskList/index.tsx @@ -25,7 +25,7 @@ import { useCalibrationTaskList, useRunHasStarted, } from '../Devices/hooks' -import { useCurrentRunId } from '../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../resources/runs' import type { DashboardCalOffsetInvoker } from '../../pages/Devices/CalibrationDashboard/hooks/useDashboardCalibratePipOffset' import type { DashboardCalTipLengthInvoker } from '../../pages/Devices/CalibrationDashboard/hooks/useDashboardCalibrateTipLength' diff --git a/app/src/organisms/ChooseProtocolSlideout/index.tsx b/app/src/organisms/ChooseProtocolSlideout/index.tsx index cd82b1a48ba..a5d5a293256 100644 --- a/app/src/organisms/ChooseProtocolSlideout/index.tsx +++ b/app/src/organisms/ChooseProtocolSlideout/index.tsx @@ -220,7 +220,7 @@ export function ChooseProtocolSlideoutComponent( : acc, {} ) - Promise.all( + void Promise.all( Object.entries(dataFilesForProtocolMap).map(([key, file]) => { const fileResponse = uploadCsvFile(file) const varName = Promise.resolve(key) diff --git a/app/src/organisms/ChooseRobotSlideout/AvailableRobotOption.tsx b/app/src/organisms/ChooseRobotSlideout/AvailableRobotOption.tsx index abcdc5b903d..7953a0cd353 100644 --- a/app/src/organisms/ChooseRobotSlideout/AvailableRobotOption.tsx +++ b/app/src/organisms/ChooseRobotSlideout/AvailableRobotOption.tsx @@ -22,7 +22,7 @@ import { getNetworkInterfaces, fetchStatus } from '../../redux/networking' import { appShellRequestor } from '../../redux/shell/remote' import OT2_PNG from '../../assets/images/OT2-R_HERO.png' import FLEX_PNG from '../../assets/images/FLEX.png' -import { useNotifyAllRunsQuery } from '../../resources/runs' +import { useCurrentRunId, useNotifyRunQuery } from '../../resources/runs' import type { IconName } from '@opentrons/components' import type { Runs } from '@opentrons/api-client' @@ -59,14 +59,15 @@ export function AvailableRobotOption( getRobotModelByName(state, robotName) ) - const { data: runsData } = useNotifyAllRunsQuery( - { pageLength: 0 }, + const [isBusy, setIsBusy] = React.useState(true) + + const currentRunId = useCurrentRunId( { onSuccess: data => { - if ((data as Runs)?.links?.current != null) - registerRobotBusyStatus({ type: 'robotIsBusy', robotName }) - else { + const definitelyIdle = (data as Runs)?.links?.current == null + if (definitelyIdle) { registerRobotBusyStatus({ type: 'robotIsIdle', robotName }) + setIsBusy(false) } }, }, @@ -75,7 +76,28 @@ export function AvailableRobotOption( requestor: ip === OPENTRONS_USB ? appShellRequestor : undefined, } ) - const robotHasCurrentRun = runsData?.links?.current != null + + useNotifyRunQuery( + currentRunId, + { + onSuccess: data => { + const busy = data?.data != null && data.data.completedAt == null + registerRobotBusyStatus({ + type: busy ? 'robotIsBusy' : 'robotIsIdle', + robotName, + }) + setIsBusy(busy) + }, + onError: () => { + registerRobotBusyStatus({ type: 'robotIsIdle', robotName }) + setIsBusy(false) + }, + }, + { + hostname: ip, + requestor: ip === OPENTRONS_USB ? appShellRequestor : undefined, + } + ) const { ethernet, wifi } = useSelector((state: State) => getNetworkInterfaces(state, robotName) @@ -95,7 +117,7 @@ export function AvailableRobotOption( // eslint-disable-next-line react-hooks/exhaustive-deps }, []) - return showIdleOnly && robotHasCurrentRun ? null : ( + return showIdleOnly && isBusy ? null : ( <> @@ -88,7 +87,7 @@ describe('ChooseRobotToRunProtocolSlideout', () => { isClosingCurrentRun: false, closeCurrentRun: mockCloseCurrentRun, }) - vi.mocked(useCurrentRunId).mockReturnValue(null) + provideNullCurrentRunIdFor(mockConnectableRobot.ip) vi.mocked(useCurrentRunStatus).mockReturnValue(null) when(vi.mocked(useCreateRunFromProtocol)) .calledWith( @@ -191,6 +190,7 @@ describe('ChooseRobotToRunProtocolSlideout', () => { { ...mockConnectableRobot, name: 'otherRobot', ip: 'otherIp' }, mockConnectableRobot, ]) + provideNullCurrentRunIdFor('otherIp') render({ storedProtocolData: storedProtocolDataFixture, onCloseClick: vi.fn(), @@ -372,6 +372,7 @@ describe('ChooseRobotToRunProtocolSlideout', () => { mockConnectableRobot, { ...mockConnectableRobot, name: 'otherRobot', ip: 'otherIp' }, ]) + provideNullCurrentRunIdFor('otherIp') render({ storedProtocolData: storedProtocolDataFixture, onCloseClick: vi.fn(), @@ -387,7 +388,7 @@ describe('ChooseRobotToRunProtocolSlideout', () => { fireEvent.click(proceedButton) fireEvent.click(screen.getByRole('button', { name: 'Confirm values' })) expect(vi.mocked(useCreateRunFromProtocol)).nthCalledWith( - 2, + 3, expect.any(Object), { hostname: '127.0.0.1' }, [ @@ -450,3 +451,21 @@ describe('ChooseRobotToRunProtocolSlideout', () => { ) }) }) + +const provideNullCurrentRunIdFor = (hostname: string): void => { + let once = true + when(vi.mocked(useCurrentRunId)) + .calledWith(expect.any(Object), { + hostname, + requestor: undefined, + }) + .thenDo(options => { + void (options?.onSuccess != null && once + ? options.onSuccess({ + links: { current: null }, + } as any) + : {}) + once = false + return null + }) +} diff --git a/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts b/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts index f28b01da886..24d4d5d0eba 100644 --- a/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts +++ b/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts @@ -98,7 +98,12 @@ export function useCreateRunFromProtocol( return { createRunFromProtocolSource: ( - { files: srcFiles, protocolKey, runTimeParameterValues }, + { + files: srcFiles, + protocolKey, + runTimeParameterValues, + runTimeParameterFiles, + }, ...args ) => { resetRunMutation() @@ -107,6 +112,7 @@ export function useCreateRunFromProtocol( files: [...srcFiles, ...customLabwareFiles], protocolKey, runTimeParameterValues, + runTimeParameterFiles, }, ...args ) diff --git a/app/src/organisms/Devices/DownloadCsvFileLink.tsx b/app/src/organisms/Devices/DownloadCsvFileLink.tsx new file mode 100644 index 00000000000..4975db0ce11 --- /dev/null +++ b/app/src/organisms/Devices/DownloadCsvFileLink.tsx @@ -0,0 +1,47 @@ +import * as React from 'react' +import { useTranslation } from 'react-i18next' + +import { + ALIGN_CENTER, + Flex, + Icon, + LegacyStyledText, + Link, + SPACING, + TYPOGRAPHY, +} from '@opentrons/components' +import { useCsvFileRawQuery } from '@opentrons/react-api-client' +import { downloadFile } from './utils' + +interface DownloadCsvFileLinkProps { + fileId: string + fileName: string +} +export function DownloadCsvFileLink( + props: DownloadCsvFileLinkProps +): JSX.Element { + const { fileId, fileName } = props + const { t } = useTranslation('run_details') + const { data: csvFileRaw } = useCsvFileRawQuery(fileId) + + return ( + { + if (csvFileRaw != null) { + downloadFile(csvFileRaw, fileName) + } + }} + > + + {t('download')} + + + + ) +} diff --git a/app/src/organisms/Devices/HistoricalProtocolRun.tsx b/app/src/organisms/Devices/HistoricalProtocolRun.tsx index 217e4e9380d..5f8d8a9547e 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRun.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRun.tsx @@ -44,7 +44,7 @@ export function HistoricalProtocolRun( const [drawerOpen, setDrawerOpen] = React.useState(false) const { data: protocolFileData } = useAllCsvFilesQuery(run.protocolId ?? '') const allProtocolDataFiles = - protocolFileData != null ? protocolFileData.data.files : [] + protocolFileData != null ? protocolFileData.data : [] const runStatus = run.status const runDisplayName = formatTimestamp(run.createdAt) let duration = EMPTY_TIMESTAMP diff --git a/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx b/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx index bb4133eeb03..e3b6eadbfe6 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRunDrawer.tsx @@ -12,11 +12,9 @@ import { DeckInfoLabel, DIRECTION_COLUMN, Flex, - Icon, InfoScreen, JUSTIFY_FLEX_START, LegacyStyledText, - Link, OVERFLOW_HIDDEN, SPACING, TYPOGRAPHY, @@ -28,6 +26,7 @@ import { getModuleDisplayName, } from '@opentrons/shared-data' import { useAllCsvFilesQuery } from '@opentrons/react-api-client' +import { DownloadCsvFileLink } from './DownloadCsvFileLink' import { useFeatureFlag } from '../../redux/config' import { Banner } from '../../atoms/Banner' import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' @@ -49,7 +48,7 @@ export function HistoricalProtocolRunDrawer( (a, b) => new Date(b.createdAt).getTime() - new Date(a.createdAt).getTime() ) const { data } = useAllCsvFilesQuery(run.protocolId ?? '') - const allProtocolDataFiles = data != null ? data.data.files : [] + const allProtocolDataFiles = data != null ? data.data : [] const uniqueLabwareOffsets = allLabwareOffsets?.filter( (offset, index, array) => { return ( @@ -97,7 +96,7 @@ export function HistoricalProtocolRunDrawer( ) : null const protocolFilesData = - allProtocolDataFiles.length === 0 ? ( + allProtocolDataFiles.length === 1 ? ( ) : ( @@ -137,7 +136,7 @@ export function HistoricalProtocolRunDrawer( {allProtocolDataFiles.map((fileData, index) => { - const { createdAt, name } = fileData + const { createdAt, name: fileName, id: fileId } = fileData return ( - {name} + {fileName} @@ -169,22 +168,7 @@ export function HistoricalProtocolRunDrawer( - {}} // TODO (nd: 06/18/2024) get file and download - > - - - {t('download')} - - - - + ) diff --git a/app/src/organisms/Devices/InstrumentsAndModules.tsx b/app/src/organisms/Devices/InstrumentsAndModules.tsx index d9216144750..fd81eeb0267 100644 --- a/app/src/organisms/Devices/InstrumentsAndModules.tsx +++ b/app/src/organisms/Devices/InstrumentsAndModules.tsx @@ -22,7 +22,7 @@ import { import { Banner } from '../../atoms/Banner' import { PipetteRecalibrationWarning } from './PipetteCard/PipetteRecalibrationWarning' -import { useCurrentRunId } from '../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../resources/runs' import { ModuleCard } from '../ModuleCard' import { useIsFlex, useIsRobotViewable, useRunStatuses } from './hooks' import { getShowPipetteCalibrationWarning } from './utils' diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolDropTipBanner.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolDropTipBanner.tsx index 7344d160510..65458700d95 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolDropTipBanner.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolDropTipBanner.tsx @@ -23,19 +23,19 @@ export function ProtocolDropTipBanner(props: { return ( - {t('tips_may_be_attached')} + {t('remove_attached_tips')} - {t('remove_the_tips_from_pipette')} + {t('liquid_damages_pipette')} void + /* True if the most recent run is the current run */ + isMostRecentRunCurrent: boolean +} + +interface UseProtocolDropTipModalResult { + showDTModal: boolean + onDTModalSkip: () => void + onDTModalRemoval: () => void +} + +// Wraps functionality required for rendering the related modal. +export function useProtocolDropTipModal({ + areTipsAttached, + toggleDTWiz, + isMostRecentRunCurrent, +}: UseProtocolDropTipModalProps): UseProtocolDropTipModalResult { + const [showDTModal, setShowDTModal] = React.useState(areTipsAttached) + + React.useEffect(() => { + if (isMostRecentRunCurrent) { + setShowDTModal(areTipsAttached) + } else { + setShowDTModal(false) + } + }, [areTipsAttached, isMostRecentRunCurrent]) + + const onDTModalSkip = (): void => { + setShowDTModal(false) + } + + const onDTModalRemoval = (): void => { + toggleDTWiz() + } + + return { showDTModal, onDTModalSkip, onDTModalRemoval } +} + +interface ProtocolDropTipModalProps { + onSkip: UseProtocolDropTipModalResult['onDTModalSkip'] + onBeginRemoval: UseProtocolDropTipModalResult['onDTModalRemoval'] + mount?: PipetteData['mount'] +} + +export function ProtocolDropTipModal({ + onSkip, + onBeginRemoval, + mount, +}: ProtocolDropTipModalProps): JSX.Element { + const { t } = useTranslation('drop_tip_wizard') + + const buildIcon = (): IconProps => { + return { + name: 'information', + color: COLORS.red50, + size: SPACING.spacing20, + marginRight: SPACING.spacing8, + } + } + + const buildHeader = (): JSX.Element => { + return ( + + ) + } + + return ( + + + + , + }} + /> + + + + + {t('begin_removal')} + + + + + ) +} + +const MODAL_STYLE = css` + width: 500px; +` diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx index 9b090a67259..a35276c7e8c 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx @@ -67,10 +67,7 @@ import { } from '../../../redux/analytics' import { getIsHeaterShakerAttached } from '../../../redux/config' import { Tooltip } from '../../../atoms/Tooltip' -import { - useCloseCurrentRun, - useCurrentRunId, -} from '../../../organisms/ProtocolUpload/hooks' +import { useCloseCurrentRun } from '../../../organisms/ProtocolUpload/hooks' import { ConfirmCancelModal } from '../../../organisms/RunDetails/ConfirmCancelModal' import { HeaterShakerIsRunningModal } from '../HeaterShakerIsRunningModal' import { @@ -103,11 +100,16 @@ import { getIsFixtureMismatch } from '../../../resources/deck_configuration/util import { useDeckConfigurationCompatibility } from '../../../resources/deck_configuration/hooks' import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useMostRecentRunId } from '../../ProtocolUpload/hooks/useMostRecentRunId' -import { useNotifyRunQuery } from '../../../resources/runs' +import { useNotifyRunQuery, useCurrentRunId } from '../../../resources/runs' import { useErrorRecoveryFlows, ErrorRecoveryFlows, } from '../../ErrorRecoveryFlows' +import { useRecoveryAnalytics } from '../../ErrorRecoveryFlows/hooks' +import { + useProtocolDropTipModal, + ProtocolDropTipModal, +} from './ProtocolDropTipModal' import type { Run, RunError, RunStatus } from '@opentrons/api-client' import type { IconName } from '@opentrons/components' @@ -148,6 +150,7 @@ export function ProtocolRunHeader({ protocolKey, isProtocolAnalyzing, } = useProtocolDetailsForRun(runId) + const { reportRecoveredRunResult } = useRecoveryAnalytics() const { trackProtocolRunEvent } = useTrackProtocolRunEvent(runId, robotName) const robotAnalyticsData = useRobotAnalyticsData(robotName) @@ -161,6 +164,7 @@ export function ProtocolRunHeader({ const { startedAt, stoppedAt, completedAt } = useRunTimestamps(runId) const [showRunFailedModal, setShowRunFailedModal] = React.useState(false) const [showDropTipBanner, setShowDropTipBanner] = React.useState(true) + const [enteredER, setEnteredER] = React.useState(false) const isResetRunLoadingRef = React.useRef(false) const { data: runRecord } = useNotifyRunQuery(runId, { staleTime: Infinity }) const highestPriorityError = @@ -210,6 +214,15 @@ export function ProtocolRunHeader({ host, isFlex, }) + const { + showDTModal, + onDTModalSkip, + onDTModalRemoval, + } = useProtocolDropTipModal({ + areTipsAttached, + toggleDTWiz, + isMostRecentRunCurrent: mostRecentRunId === runId, + }) React.useEffect(() => { if (isFlex) { @@ -234,6 +247,7 @@ export function ProtocolRunHeader({ // Side effects dependent on the current run state. React.useEffect(() => { + reportRecoveredRunResult(runStatus, enteredER) // After a user-initiated stopped run, close the run current run automatically. if (runStatus === RUN_STATUS_STOPPED && isRunCurrent && runId != null) { trackProtocolRunEvent({ @@ -244,6 +258,9 @@ export function ProtocolRunHeader({ }) closeCurrentRun() } + if (runStatus === RUN_STATUS_AWAITING_RECOVERY) { + setEnteredER(true) + } }, [runStatus, isRunCurrent, runId, closeCurrentRun]) const startedAtTimestamp = @@ -297,7 +314,6 @@ export function ProtocolRunHeader({ <> {isERActive ? ( ) : null} + {showDTModal ? ( + + ) : null} setTipStatusResolved().then(toggleDTWiz)} /> diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx index 9c1ee6e8ac5..651657d4597 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx @@ -61,10 +61,8 @@ const LabwareRow = styled.div` border-width: 1px; border-color: ${COLORS.grey30}; border-radius: ${BORDERS.borderRadius4}; - padding: ${(SPACING.spacing12, - SPACING.spacing16, - SPACING.spacing12, - SPACING.spacing24)}; + padding: ${SPACING.spacing12} ${SPACING.spacing16} ${SPACING.spacing12} + ${SPACING.spacing24}; ` interface LabwareListItemProps extends LabwareSetupItem { diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipBanner.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipBanner.test.tsx index 0f8b391aa57..295a1bea3f6 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipBanner.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipBanner.test.tsx @@ -24,8 +24,10 @@ describe('Module Update Banner', () => { it('displays appropriate banner text', () => { render(props) - screen.getByText('Tips may be attached.') - screen.queryByText('You may want to remove tips') + screen.getByText('Remove any attached tips') + screen.queryByText( + /Homing the .* pipette with liquid in the tips may damage it\. You must remove all tips before using the pipette again\./ + ) screen.getByText('Remove tips') }) diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipModal.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipModal.test.tsx new file mode 100644 index 00000000000..0e9ce19fc5f --- /dev/null +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolDropTipModal.test.tsx @@ -0,0 +1,107 @@ +import * as React from 'react' +import { describe, it, vi, expect, beforeEach } from 'vitest' +import { renderHook, act, screen, fireEvent } from '@testing-library/react' + +import { + useProtocolDropTipModal, + ProtocolDropTipModal, +} from '../ProtocolDropTipModal' +import { renderWithProviders } from '../../../../__testing-utils__' +import { i18n } from '../../../../i18n' + +describe('useProtocolDropTipModal', () => { + let props: Parameters[0] + + beforeEach(() => { + props = { + areTipsAttached: true, + toggleDTWiz: vi.fn(), + isMostRecentRunCurrent: true, + } + }) + + it('should return initial values', () => { + const { result } = renderHook(() => useProtocolDropTipModal(props)) + + expect(result.current).toEqual({ + showDTModal: true, + onDTModalSkip: expect.any(Function), + onDTModalRemoval: expect.any(Function), + }) + }) + + it('should update showDTModal when areTipsAttached changes', () => { + const { result, rerender } = renderHook(() => + useProtocolDropTipModal(props) + ) + + expect(result.current.showDTModal).toBe(true) + + props.areTipsAttached = false + rerender() + + expect(result.current.showDTModal).toBe(false) + }) + + it('should not show modal when isMostRecentRunCurrent is false', () => { + props.isMostRecentRunCurrent = false + const { result } = renderHook(() => useProtocolDropTipModal(props)) + + expect(result.current.showDTModal).toBe(false) + }) + + it('should call toggleDTWiz when onDTModalRemoval is called', () => { + const { result } = renderHook(() => useProtocolDropTipModal(props)) + + act(() => { + result.current.onDTModalRemoval() + }) + + expect(props.toggleDTWiz).toHaveBeenCalled() + }) +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + })[0] +} + +describe('ProtocolDropTipModal', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + onSkip: vi.fn(), + onBeginRemoval: vi.fn(), + mount: 'left', + } + }) + + it('renders the modal with correct content', () => { + render(props) + + screen.getByText('Remove any attached tips') + screen.queryByText( + /Homing the .* pipette with liquid in the tips may damage it\. You must remove all tips before using the pipette again\./ + ) + screen.getByText('Begin removal') + screen.getByText('Skip') + }) + + it('calls onSkip when skip button is clicked', () => { + render(props) + + fireEvent.click(screen.getByText('Skip')) + + expect(props.onSkip).toHaveBeenCalled() + }) + + it('calls onBeginRemoval when begin removal button is clicked', () => { + render(props) + + fireEvent.click(screen.getByText('Begin removal')) + + expect(props.onBeginRemoval).toHaveBeenCalled() + }) +}) diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunHeader.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunHeader.test.tsx index b090b284a34..157538c9ff8 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunHeader.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunHeader.test.tsx @@ -34,10 +34,7 @@ import { import { renderWithProviders } from '../../../../__testing-utils__' import { i18n } from '../../../../i18n' -import { - useCloseCurrentRun, - useCurrentRunId, -} from '../../../../organisms/ProtocolUpload/hooks' +import { useCloseCurrentRun } from '../../../../organisms/ProtocolUpload/hooks' import { ConfirmCancelModal } from '../../../../organisms/RunDetails/ConfirmCancelModal' import { useRunTimestamps, @@ -87,7 +84,7 @@ import { getIsFixtureMismatch } from '../../../../resources/deck_configuration/u import { useDeckConfigurationCompatibility } from '../../../../resources/deck_configuration/hooks' import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useMostRecentRunId } from '../../../ProtocolUpload/hooks/useMostRecentRunId' -import { useNotifyRunQuery } from '../../../../resources/runs' +import { useNotifyRunQuery, useCurrentRunId } from '../../../../resources/runs' import { useDropTipWizardFlows, useTipAttachmentStatus, @@ -96,6 +93,10 @@ import { useErrorRecoveryFlows, ErrorRecoveryFlows, } from '../../../ErrorRecoveryFlows' +import { + ProtocolDropTipModal, + useProtocolDropTipModal, +} from '../ProtocolDropTipModal' import type { UseQueryResult } from 'react-query' import type { NavigateFunction } from 'react-router-dom' @@ -151,6 +152,7 @@ vi.mock('../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') vi.mock('../../../ProtocolUpload/hooks/useMostRecentRunId') vi.mock('../../../../resources/runs') vi.mock('../../../ErrorRecoveryFlows') +vi.mock('../ProtocolDropTipModal') const ROBOT_NAME = 'otie' const RUN_ID = '95e67900-bc9f-4fbf-92c6-cc4d7226a51b' @@ -374,6 +376,14 @@ describe('ProtocolRunHeader', () => { vi.mocked(ErrorRecoveryFlows).mockReturnValue(
MOCK_ERROR_RECOVERY
) + vi.mocked(useProtocolDropTipModal).mockReturnValue({ + onDTModalRemoval: vi.fn(), + onDTModalSkip: vi.fn(), + showDTModal: false, + } as any) + vi.mocked(ProtocolDropTipModal).mockReturnValue( +
MOCK_DROP_TIP_MODAL
+ ) }) afterEach(() => { @@ -1018,8 +1028,23 @@ describe('ProtocolRunHeader', () => { render() await waitFor(() => { - screen.getByText('Tips may be attached.') + screen.getByText('Remove any attached tips') + screen.getByText( + 'Homing the pipette with liquid in the tips may damage it. You must remove all tips before using the pipette again.' + ) + }) + }) + + it('renders the drop tip modal initially when the run ends if tips are attached', () => { + vi.mocked(useProtocolDropTipModal).mockReturnValue({ + onDTModalRemoval: vi.fn(), + onDTModalSkip: vi.fn(), + showDTModal: true, }) + + render() + + screen.getByText('MOCK_DROP_TIP_MODAL') }) it('does not render the drop tip banner when the run is not over', async () => { diff --git a/app/src/organisms/Devices/RecentProtocolRuns.tsx b/app/src/organisms/Devices/RecentProtocolRuns.tsx index d52575b1cb9..af685895ffa 100644 --- a/app/src/organisms/Devices/RecentProtocolRuns.tsx +++ b/app/src/organisms/Devices/RecentProtocolRuns.tsx @@ -16,10 +16,9 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { useCurrentRunId } from '../ProtocolUpload/hooks' import { HistoricalProtocolRun } from './HistoricalProtocolRun' import { useIsRobotViewable, useRunStatuses } from './hooks' -import { useNotifyAllRunsQuery } from '../../resources/runs' +import { useNotifyAllRunsQuery, useCurrentRunId } from '../../resources/runs' import { useFeatureFlag } from '../../redux/config' interface RecentProtocolRunsProps { diff --git a/app/src/organisms/Devices/RobotCard.tsx b/app/src/organisms/Devices/RobotCard.tsx index d92ea790435..246cfea101f 100644 --- a/app/src/organisms/Devices/RobotCard.tsx +++ b/app/src/organisms/Devices/RobotCard.tsx @@ -41,6 +41,10 @@ import { useIsFlex } from './hooks' import { ReachableBanner } from './ReachableBanner' import { RobotOverflowMenu } from './RobotOverflowMenu' import { RobotStatusHeader } from './RobotStatusHeader' +import { + ErrorRecoveryBanner, + useErrorRecoveryBanner, +} from '../ErrorRecoveryBanner' import type { GripperData } from '@opentrons/api-client' import type { GripperModel } from '@opentrons/shared-data' @@ -59,6 +63,8 @@ export function RobotCard(props: RobotCardProps): JSX.Element | null { getRobotModelByName(state, robotName) ) + const { showRecoveryBanner, recoveryIntent } = useErrorRecoveryBanner() + return robot != null ? ( + {showRecoveryBanner ? ( + + ) : null} getRobotAddressesByName(state, robot?.name ?? '') @@ -113,6 +119,12 @@ export function RobotOverview({
+ {showRecoveryBanner ? ( + + ) : null} { @@ -39,7 +39,7 @@ vi.mock('../../ModuleCard') vi.mock('../PipetteCard') vi.mock('../PipetteCard/FlexPipetteCard') vi.mock('../PipetteCard/PipetteRecalibrationWarning') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') vi.mock('../../../atoms/Banner') vi.mock('../utils') vi.mock('../../RunTimeControl/hooks') diff --git a/app/src/organisms/Devices/__tests__/RobotCard.test.tsx b/app/src/organisms/Devices/__tests__/RobotCard.test.tsx index 5bb7d2cce4d..d4cf00a61d7 100644 --- a/app/src/organisms/Devices/__tests__/RobotCard.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotCard.test.tsx @@ -31,6 +31,10 @@ import { UpdateRobotBanner } from '../../UpdateRobotBanner' import { RobotOverflowMenu } from '../RobotOverflowMenu' import { RobotStatusHeader } from '../RobotStatusHeader' import { RobotCard } from '../RobotCard' +import { + ErrorRecoveryBanner, + useErrorRecoveryBanner, +} from '../../ErrorRecoveryBanner' import type { State } from '../../../redux/types' @@ -41,6 +45,7 @@ vi.mock('../../UpdateRobotBanner') vi.mock('../../../redux/config') vi.mock('../RobotOverflowMenu') vi.mock('../RobotStatusHeader') +vi.mock('../../ErrorRecoveryBanner') const OT2_PNG_FILE_NAME = '/app/src/assets/images/OT2-R_HERO.png' const FLEX_PNG_FILE_NAME = '/app/src/assets/images/FLEX.png' @@ -127,6 +132,13 @@ describe('RobotCard', () => { when(getRobotModelByName) .calledWith(MOCK_STATE, mockConnectableRobot.name) .thenReturn('OT-2') + vi.mocked(ErrorRecoveryBanner).mockReturnValue( +
MOCK_RECOVERY_BANNER
+ ) + vi.mocked(useErrorRecoveryBanner).mockReturnValue({ + showRecoveryBanner: false, + recoveryIntent: 'recovering', + }) }) it('renders an OT-2 image when robot model is OT-2', () => { @@ -161,4 +173,15 @@ describe('RobotCard', () => { render(props) screen.getByText('Mock RobotStatusHeader') }) + + it('renders the error recovery banner when another user is performing error recovery', () => { + vi.mocked(useErrorRecoveryBanner).mockReturnValue({ + showRecoveryBanner: true, + recoveryIntent: 'recovering', + }) + + render(props) + + screen.getByText('MOCK_RECOVERY_BANNER') + }) }) diff --git a/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx index 6227cbd5675..868e14cf171 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverflowMenu.test.tsx @@ -5,7 +5,7 @@ import { describe, it, vi, beforeEach, expect } from 'vitest' import '@testing-library/jest-dom/vitest' import { renderWithProviders } from '../../../__testing-utils__' import { i18n } from '../../../i18n' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { ChooseProtocolSlideout } from '../../ChooseProtocolSlideout' import { RobotOverflowMenu } from '../RobotOverflowMenu' import { getRobotUpdateDisplayInfo } from '../../../redux/robot-update' @@ -17,7 +17,7 @@ import { } from '../../../redux/discovery/__fixtures__' vi.mock('../../../redux/robot-update/selectors') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') vi.mock('../../ChooseProtocolSlideout') vi.mock('../hooks') vi.mock('../../../resources/devices/hooks/useIsEstopNotDisengaged') diff --git a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx index 66f6d18b7d0..e800f9741bf 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx @@ -9,7 +9,7 @@ import * as DiscoveryClientFixtures from '../../../../../discovery-client/src/fi import { useAuthorization } from '@opentrons/react-api-client' import { i18n } from '../../../i18n' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { mockConnectableRobot } from '../../../redux/discovery/__fixtures__' import { getRobotUpdateDisplayInfo } from '../../../redux/robot-update' import { getConfig, useFeatureFlag } from '../../../redux/config' @@ -44,6 +44,10 @@ import { UpdateRobotBanner } from '../../UpdateRobotBanner' import { RobotStatusHeader } from '../RobotStatusHeader' import { RobotOverview } from '../RobotOverview' import { RobotOverviewOverflowMenu } from '../RobotOverviewOverflowMenu' +import { + ErrorRecoveryBanner, + useErrorRecoveryBanner, +} from '../../ErrorRecoveryBanner' import type { Config } from '../../../redux/config/types' import type { DiscoveryClientRobotAddress } from '../../../redux/discovery/types' @@ -61,11 +65,12 @@ vi.mock('../../../redux/robot-controls') vi.mock('../../../redux/robot-update/selectors') vi.mock('../../../redux/config') vi.mock('../../../redux/discovery/selectors') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') vi.mock('../hooks') vi.mock('../RobotStatusHeader') vi.mock('../../UpdateRobotBanner') vi.mock('../RobotOverviewOverflowMenu') +vi.mock('../../ErrorRecoveryBanner') const OT2_PNG_FILE_NAME = '/app/src/assets/images/OT2-R_HERO.png' const FLEX_PNG_FILE_NAME = '/app/src/assets/images/FLEX.png' @@ -151,7 +156,7 @@ describe('RobotOverview', () => { .calledWith(MOCK_STATE, mockConnectableRobot.name) .thenReturn([]) vi.mocked(getConfig).mockReturnValue({ - support: { userId: 'opentrons-robot-user' }, + userInfo: { userId: 'opentrons-robot-user' }, } as Config) when(useAuthorization) .calledWith({ @@ -164,6 +169,13 @@ describe('RobotOverview', () => { registrationToken: { token: 'my.registration.jwt' }, }) vi.mocked(useIsRobotViewable).mockReturnValue(true) + vi.mocked(ErrorRecoveryBanner).mockReturnValue( +
MOCK_RECOVERY_BANNER
+ ) + vi.mocked(useErrorRecoveryBanner).mockReturnValue({ + showRecoveryBanner: false, + recoveryIntent: 'recovering', + }) }) it('renders an OT-2 image', () => { @@ -367,4 +379,15 @@ describe('RobotOverview', () => { agentId: 'opentrons-robot-user', }) }) + + it('renders the error recovery banner when another user is performing error recovery', () => { + vi.mocked(useErrorRecoveryBanner).mockReturnValue({ + showRecoveryBanner: true, + recoveryIntent: 'recovering', + }) + + render(props) + + screen.getByText('MOCK_RECOVERY_BANNER') + }) }) diff --git a/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx index d39aa5d6f61..2cbcab8b99c 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverviewOverflowMenu.test.tsx @@ -18,7 +18,7 @@ import { import { useCanDisconnect } from '../../../resources/networking/hooks' import { DisconnectModal } from '../../../organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal' import { ChooseProtocolSlideout } from '../../ChooseProtocolSlideout' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { useIsRobotBusy } from '../hooks' import { handleUpdateBuildroot } from '../RobotSettings/UpdateBuildroot' import { useIsEstopNotDisengaged } from '../../../resources/devices/hooks/useIsEstopNotDisengaged' @@ -35,7 +35,7 @@ vi.mock( '../../../organisms/Devices/RobotSettings/ConnectNetwork/DisconnectModal' ) vi.mock('../../ChooseProtocolSlideout') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') vi.mock('../RobotSettings/UpdateBuildroot') vi.mock('../../../resources/devices/hooks/useIsEstopNotDisengaged') diff --git a/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx b/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx index c93299ebf85..b7fee94c37f 100644 --- a/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotStatusHeader.test.tsx @@ -9,7 +9,6 @@ import { renderWithProviders } from '../../../__testing-utils__' import { useProtocolQuery } from '@opentrons/react-api-client' import { i18n } from '../../../i18n' -import { useCurrentRunId } from '../../../organisms/ProtocolUpload/hooks' import { useCurrentRunStatus } from '../../../organisms/RunTimeControl/hooks' import { getRobotAddressesByName, @@ -19,14 +18,13 @@ import { import { getNetworkInterfaces } from '../../../redux/networking' import { useIsFlex } from '../hooks' import { RobotStatusHeader } from '../RobotStatusHeader' -import { useNotifyRunQuery } from '../../../resources/runs' +import { useNotifyRunQuery, useCurrentRunId } from '../../../resources/runs' import type { DiscoveryClientRobotAddress } from '../../../redux/discovery/types' import type { SimpleInterfaceStatus } from '../../../redux/networking/types' import type { State } from '../../../redux/types' vi.mock('@opentrons/react-api-client') -vi.mock('../../../organisms/ProtocolUpload/hooks') vi.mock('../../../organisms/RunTimeControl/hooks') vi.mock('../../../redux/discovery') vi.mock('../../../redux/networking') diff --git a/app/src/organisms/Devices/hooks/__tests__/useRunStartedOrLegacySessionInProgress.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useRunStartedOrLegacySessionInProgress.test.tsx index 2354c31c6a8..e659e24930a 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useRunStartedOrLegacySessionInProgress.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useRunStartedOrLegacySessionInProgress.test.tsx @@ -2,7 +2,7 @@ import { useAllSessionsQuery } from '@opentrons/react-api-client' import { RUN_STATUS_IDLE, RUN_STATUS_RUNNING } from '@opentrons/api-client' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' -import { useCurrentRunId } from '../../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../../resources/runs' import { useRunStatus } from '../../../RunTimeControl/hooks' import { useRunStartedOrLegacySessionInProgress } from '..' @@ -10,7 +10,7 @@ import type { UseQueryResult } from 'react-query' import type { Sessions } from '@opentrons/api-client' vi.mock('@opentrons/react-api-client') -vi.mock('../../../ProtocolUpload/hooks') +vi.mock('../../../../resources/runs') vi.mock('../../../RunTimeControl/hooks') describe('useRunStartedOrLegacySessionInProgress', () => { diff --git a/app/src/organisms/Devices/hooks/__tests__/useRunStatuses.test.tsx b/app/src/organisms/Devices/hooks/__tests__/useRunStatuses.test.tsx index 6c805c7ca39..9277ddafd10 100644 --- a/app/src/organisms/Devices/hooks/__tests__/useRunStatuses.test.tsx +++ b/app/src/organisms/Devices/hooks/__tests__/useRunStatuses.test.tsx @@ -8,11 +8,11 @@ import { } from '@opentrons/api-client' import { vi, it, expect, describe, beforeEach, afterEach } from 'vitest' -import { useCurrentRunId } from '../../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../../resources/runs' import { useRunStatus } from '../../../RunTimeControl/hooks' import { useRunStatuses } from '..' -vi.mock('../../../ProtocolUpload/hooks') +vi.mock('../../../../resources/runs') vi.mock('../../../RunTimeControl/hooks') describe(' useRunStatuses ', () => { diff --git a/app/src/organisms/Devices/hooks/useRunStartedOrLegacySessionInProgress.ts b/app/src/organisms/Devices/hooks/useRunStartedOrLegacySessionInProgress.ts index f272c322bc2..e8678518847 100644 --- a/app/src/organisms/Devices/hooks/useRunStartedOrLegacySessionInProgress.ts +++ b/app/src/organisms/Devices/hooks/useRunStartedOrLegacySessionInProgress.ts @@ -1,6 +1,6 @@ import { useAllSessionsQuery } from '@opentrons/react-api-client' import { RUN_STATUS_IDLE } from '@opentrons/api-client' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { useRunStatus } from '../../RunTimeControl/hooks' export function useRunStartedOrLegacySessionInProgress(): boolean { diff --git a/app/src/organisms/Devices/hooks/useRunStatuses.ts b/app/src/organisms/Devices/hooks/useRunStatuses.ts index c93b1fc070f..bf1c550efa0 100644 --- a/app/src/organisms/Devices/hooks/useRunStatuses.ts +++ b/app/src/organisms/Devices/hooks/useRunStatuses.ts @@ -6,7 +6,7 @@ import { RUN_STATUS_PAUSED, RUN_STATUS_RUNNING, } from '@opentrons/api-client' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { useRunStatus } from '../../RunTimeControl/hooks' import type { RunStatus } from '@opentrons/api-client' diff --git a/app/src/organisms/Devices/utils.ts b/app/src/organisms/Devices/utils.ts index c5302e62208..718bf976c63 100644 --- a/app/src/organisms/Devices/utils.ts +++ b/app/src/organisms/Devices/utils.ts @@ -33,9 +33,10 @@ export function onDeviceDisplayFormatTimestamp(timestamp: string): string { : timestamp } -export function downloadFile(data: object, fileName: string): void { +export function downloadFile(data: object | string, fileName: string): void { // Create a blob with the data we want to download as a file - const blob = new Blob([JSON.stringify(data)], { type: 'text/json' }) + const blobContent = typeof data === 'string' ? data : JSON.stringify(data) + const blob = new Blob([blobContent], { type: 'text/json' }) // Create an anchor element and dispatch a click event on it // to trigger a download const a = document.createElement('a') diff --git a/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx b/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx index b5cd5730e52..329ec38d199 100644 --- a/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx +++ b/app/src/organisms/DropTipWizardFlows/DropTipWizard.tsx @@ -318,10 +318,14 @@ export const DropTipWizardContent = ( } function buildSuccess(): JSX.Element { + const { tipDropComplete } = fixitCommandTypeUtils?.buttonOverrides ?? {} + // Route to the drop tip route if user is at the blowout success screen, otherwise proceed conditionally. const handleProceed = (): void => { if (currentStep === BLOWOUT_SUCCESS) { void proceedToRoute(DT_ROUTES.DROP_TIP) + } else if (tipDropComplete != null) { + tipDropComplete() } else { proceedWithConditionalClose() } diff --git a/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx b/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx index c71b537cde3..0cb1872b196 100644 --- a/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx +++ b/app/src/organisms/DropTipWizardFlows/TipsAttachedModal.tsx @@ -46,9 +46,9 @@ const TipsAttachedModal = NiceModal.create( const { showDTWiz, toggleDTWiz } = useDropTipWizardFlows() const tipsAttachedHeader: ModalHeaderBaseProps = { - title: t('tips_are_attached'), + title: t('remove_any_attached_tips'), iconName: 'ot-alert', - iconColor: COLORS.yellow50, + iconColor: COLORS.red50, } const cleanUpAndClose = (): void => { @@ -66,7 +66,7 @@ const TipsAttachedModal = NiceModal.create( { const btn = screen.getByTestId('testButton') fireEvent.click(btn) - screen.getByText('Tips are attached') - screen.queryByText(`${LEFT} Pipette`) + screen.getByText('Remove any attached tips') + screen.queryByText( + /Homing the .* pipette with liquid in the tips may damage it\. You must remove all tips before using the pipette again\./ + ) }) it('clicking the skip button properly closes the modal', () => { render(MOCK_PIPETTES_WITH_TIP) diff --git a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType/useDropTipCommands.ts b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType/useDropTipCommands.ts index 12946613b58..4bb8f6c96ae 100644 --- a/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType/useDropTipCommands.ts +++ b/app/src/organisms/DropTipWizardFlows/hooks/useDropTipWithType/useDropTipCommands.ts @@ -5,12 +5,16 @@ import { useDeleteMaintenanceRunMutation } from '@opentrons/react-api-client' import { MANAGED_PIPETTE_ID, POSITION_AND_BLOWOUT } from '../../constants' import { getAddressableAreaFromConfig } from '../../getAddressableAreaFromConfig' import { useNotifyDeckConfigurationQuery } from '../../../../resources/deck_configuration' - import type { CreateCommand, AddressableAreaName, PipetteModelSpecs, + BlowoutInPlaceCreateCommand, + UnsafeBlowoutInPlaceCreateCommand, + DropTipInPlaceCreateCommand, + UnsafeDropTipInPlaceCreateCommand, } from '@opentrons/shared-data' +import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' import type { CommandData, PipetteData } from '@opentrons/api-client' import type { Axis, @@ -61,6 +65,7 @@ export function useDropTipCommands({ robotType, fixitCommandTypeUtils, }: UseDropTipSetupCommandsParams): UseDropTipCommandsResult { + const isFlex = robotType === FLEX_ROBOT_TYPE const [hasSeenClose, setHasSeenClose] = React.useState(false) const { deleteMaintenanceRun } = useDeleteMaintenanceRunMutation({ @@ -114,8 +119,12 @@ export function useDropTipCommands({ if (addressableAreaFromConfig != null) { const moveToAACommand = buildMoveToAACommand(addressableAreaFromConfig) - - return chainRunCommands([moveToAACommand], true) + return chainRunCommands( + isFlex + ? [UPDATE_ESTIMATORS_EXCEPT_PLUNGERS, moveToAACommand] + : [moveToAACommand], + true + ) .then((commandData: CommandData[]) => { const error = commandData[0].data.error if (error != null) { @@ -177,10 +186,12 @@ export function useDropTipCommands({ proceed: () => void ): Promise => { return new Promise((resolve, reject) => { - const blowoutCommand = buildBlowoutInPlaceCommand(instrumentModelSpecs) - chainRunCommands( - [currentStep === POSITION_AND_BLOWOUT ? blowoutCommand : DROP_TIP], + [ + currentStep === POSITION_AND_BLOWOUT + ? buildBlowoutInPlaceCommand(instrumentModelSpecs, isFlex) + : buildDropTipInPlaceCommand(isFlex), + ], true ) .then((commandData: CommandData[]) => { @@ -260,22 +271,43 @@ const HOME_EXCEPT_PLUNGERS: CreateCommand = { params: { axes: ['leftZ', 'rightZ', 'x', 'y'] }, } -const DROP_TIP: CreateCommand = { - commandType: 'dropTipInPlace', - params: { pipetteId: MANAGED_PIPETTE_ID }, +const UPDATE_ESTIMATORS_EXCEPT_PLUNGERS: CreateCommand = { + commandType: 'unsafe/updatePositionEstimators' as const, + params: { axes: ['leftZ', 'rightZ', 'x', 'y'] }, } +const buildDropTipInPlaceCommand = ( + isFlex: boolean +): DropTipInPlaceCreateCommand | UnsafeDropTipInPlaceCreateCommand => + isFlex + ? { + commandType: 'unsafe/dropTipInPlace', + params: { pipetteId: MANAGED_PIPETTE_ID }, + } + : { + commandType: 'dropTipInPlace', + params: { pipetteId: MANAGED_PIPETTE_ID }, + } + const buildBlowoutInPlaceCommand = ( - specs: PipetteModelSpecs -): CreateCommand => { - return { - commandType: 'blowOutInPlace', - params: { - pipetteId: MANAGED_PIPETTE_ID, - flowRate: specs.defaultBlowOutFlowRate.value, - }, - } -} + specs: PipetteModelSpecs, + isFlex: boolean +): BlowoutInPlaceCreateCommand | UnsafeBlowoutInPlaceCreateCommand => + isFlex + ? { + commandType: 'unsafe/blowOutInPlace', + params: { + pipetteId: MANAGED_PIPETTE_ID, + flowRate: specs.defaultBlowOutFlowRate.value, + }, + } + : { + commandType: 'blowOutInPlace', + params: { + pipetteId: MANAGED_PIPETTE_ID, + flowRate: specs.defaultBlowOutFlowRate.value, + }, + } const buildMoveToAACommand = ( addressableAreaFromConfig: AddressableAreaName diff --git a/app/src/organisms/DropTipWizardFlows/types.ts b/app/src/organisms/DropTipWizardFlows/types.ts index d7a8309b60b..f4aa36266ae 100644 --- a/app/src/organisms/DropTipWizardFlows/types.ts +++ b/app/src/organisms/DropTipWizardFlows/types.ts @@ -27,6 +27,7 @@ interface ErrorOverrides { interface ButtonOverrides { goBackBeforeBeginning: () => void + tipDropComplete: (() => void) | null } export interface FixitCommandTypeUtils { diff --git a/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx b/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx new file mode 100644 index 00000000000..78c5da162c3 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryBanner/__tests__/ErrorRecoveryBanner.test.tsx @@ -0,0 +1,48 @@ +import * as React from 'react' +import { describe, it, vi, beforeEach } from 'vitest' +import { screen } from '@testing-library/react' + +import { renderWithProviders } from '../../../__testing-utils__' +import { i18n } from '../../../i18n' +import { useErrorRecoveryBanner, ErrorRecoveryBanner } from '..' + +vi.mock('..', async importOriginal => { + const actualReact = await importOriginal() + return { + ...actualReact, + useErrorRecoveryBanner: vi.fn(), + } +}) + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + })[0] +} + +describe('ErrorRecoveryBanner', () => { + beforeEach(() => { + vi.mocked(useErrorRecoveryBanner).mockReturnValue({ + showRecoveryBanner: true, + recoveryIntent: 'recovering', + }) + }) + + it('renders banner with correct content for recovering intent', () => { + render({ recoveryIntent: 'recovering' }) + + screen.getByText('Robot is in recovery mode') + screen.getByText( + 'The robot’s touchscreen or another computer with the app is currently controlling this robot.' + ) + }) + + it('renders banner with correct content for canceling intent', () => { + render({ recoveryIntent: 'canceling' }) + + screen.getByText('Robot is canceling the run') + screen.getByText( + 'The robot’s touchscreen or another computer with the app is currently controlling this robot.' + ) + }) +}) diff --git a/app/src/organisms/ErrorRecoveryBanner/__tests__/useErrorRecoveryBanner.test.ts b/app/src/organisms/ErrorRecoveryBanner/__tests__/useErrorRecoveryBanner.test.ts new file mode 100644 index 00000000000..6eedd264499 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryBanner/__tests__/useErrorRecoveryBanner.test.ts @@ -0,0 +1,54 @@ +import { beforeEach, describe, expect, it, vi } from 'vitest' +import { useSelector } from 'react-redux' +import { getUserId } from '../../../redux/config' +import { useClientDataRecovery } from '../../../resources/client_data' +import { renderHook } from '@testing-library/react' +import { useErrorRecoveryBanner } from '../index' + +vi.mock('react-redux', () => ({ + useSelector: vi.fn(), +})) +vi.mock('../../../redux/config') +vi.mock('../../../resources/client_data') + +describe('useErrorRecoveryBanner', () => { + beforeEach(() => { + vi.mocked(useSelector).mockReturnValue('thisUserId') + vi.mocked(getUserId).mockReturnValue('thisUserId') + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: null, + intent: null, + }) + }) + + it('should return initial values', () => { + const { result } = renderHook(() => useErrorRecoveryBanner()) + + expect(result.current).toEqual({ + showRecoveryBanner: false, + recoveryIntent: 'recovering', + }) + }) + + it('should show banner when userId is different', () => { + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: 'otherUserId', + intent: null, + }) + + const { result } = renderHook(() => useErrorRecoveryBanner()) + + expect(result.current.showRecoveryBanner).toBe(true) + }) + + it('should return correct intent when provided', () => { + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: 'otherUserId', + intent: 'canceling', + }) + + const { result } = renderHook(() => useErrorRecoveryBanner()) + + expect(result.current.recoveryIntent).toBe('canceling') + }) +}) diff --git a/app/src/organisms/ErrorRecoveryBanner/index.tsx b/app/src/organisms/ErrorRecoveryBanner/index.tsx new file mode 100644 index 00000000000..504cf2fc979 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryBanner/index.tsx @@ -0,0 +1,75 @@ +import * as React from 'react' +import { useTranslation } from 'react-i18next' +import { useSelector } from 'react-redux' + +import { + Flex, + DIRECTION_COLUMN, + SPACING, + StyledText, +} from '@opentrons/components' + +import { getUserId } from '../../redux/config' +import { useClientDataRecovery } from '../../resources/client_data' +import { Banner } from '../../atoms/Banner' + +import type { RecoveryIntent } from '../../resources/client_data' +import type { StyleProps } from '@opentrons/components' + +const CLIENT_DATA_INTERVAL_MS = 5000 + +export interface UseErrorRecoveryBannerResult { + showRecoveryBanner: boolean + recoveryIntent: RecoveryIntent +} + +export function useErrorRecoveryBanner(): UseErrorRecoveryBannerResult { + const { userId, intent } = useClientDataRecovery({ + refetchInterval: CLIENT_DATA_INTERVAL_MS, + }) + const thisUserId = useSelector(getUserId) + + return { + showRecoveryBanner: userId !== null && thisUserId !== userId, + recoveryIntent: intent ?? 'recovering', + } +} + +export interface ErrorRecoveryBannerProps extends StyleProps { + recoveryIntent: RecoveryIntent +} + +export function ErrorRecoveryBanner({ + recoveryIntent, + ...styleProps +}: ErrorRecoveryBannerProps): JSX.Element { + const { t } = useTranslation(['error_recovery', 'shared']) + + const buildTitleText = (): string => { + switch (recoveryIntent) { + case 'canceling': + return t('robot_is_canceling_run') + case 'recovering': + default: + return t('robot_is_in_recovery_mode') + } + } + + return ( + + + + {buildTitleText()} + + + + {t('another_app_controlling_robot')} + + + + + ) +} diff --git a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx index 73b7a1a5ea1..96fcd2209ac 100644 --- a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx @@ -1,5 +1,6 @@ import * as React from 'react' import { useTranslation } from 'react-i18next' +import { css } from 'styled-components' import { StyledText } from '@opentrons/components' @@ -28,13 +29,13 @@ import { RECOVERY_MAP } from './constants' import type { RobotType } from '@opentrons/shared-data' import type { RecoveryContentProps } from './types' -import type { ERUtilsResults } from './hooks' +import type { ERUtilsResults, UseRecoveryAnalyticsResult } from './hooks' import type { ErrorRecoveryFlowsProps } from '.' -interface UseERWizardResult { +export interface UseERWizardResult { hasLaunchedRecovery: boolean showERWizard: boolean - toggleERWizard: (hasLaunchedER: boolean) => Promise + toggleERWizard: (isActive: boolean, hasLaunchedER?: boolean) => Promise } export function useERWizard(): UseERWizardResult { @@ -44,9 +45,14 @@ export function useERWizard(): UseERWizardResult { // when recovery mode has not been launched. const [hasLaunchedRecovery, setHasLaunchedRecovery] = React.useState(false) - const toggleERWizard = (hasLaunchedER: boolean): Promise => { - setHasLaunchedRecovery(hasLaunchedER) - setShowERWizard(!showERWizard) + const toggleERWizard = ( + isActive: boolean, + hasLaunchedER?: boolean + ): Promise => { + if (hasLaunchedER !== undefined) { + setHasLaunchedRecovery(hasLaunchedER) + } + setShowERWizard(isActive) return Promise.resolve() } @@ -58,6 +64,7 @@ export type ErrorRecoveryWizardProps = ErrorRecoveryFlowsProps & robotType: RobotType isOnDevice: boolean isDoorOpen: boolean + analytics: UseRecoveryAnalyticsResult } export function ErrorRecoveryWizard( @@ -83,11 +90,23 @@ export function ErrorRecoveryWizard( export function ErrorRecoveryComponent( props: RecoveryContentProps ): JSX.Element { - const { recoveryMap, hasLaunchedRecovery, isDoorOpen, isOnDevice } = props + const { + recoveryMap, + hasLaunchedRecovery, + isDoorOpen, + isOnDevice, + analytics, + } = props const { route, step } = recoveryMap const { t } = useTranslation('error_recovery') const { showModal, toggleModal } = useErrorDetailsModal() + React.useEffect(() => { + if (showModal) { + analytics.reportViewErrorDetailsEvent(route, step) + } + }, [analytics, route, showModal, step]) + const buildTitleHeading = (): JSX.Element => { const titleText = hasLaunchedRecovery ? t('recovery_mode') : t('cancel_run') return ( @@ -101,12 +120,18 @@ export function ErrorRecoveryComponent( } const buildIconHeading = (): JSX.Element => ( - + {t('view_error_details')} ) - // TODO(jh, 07-16-24): Revisit making RecoveryDoorOpen a route. + // TODO(jh, 07-29-24): Make RecoveryDoorOpen render logic equivalent to RecoveryTakeover. Do not nest it in RecoveryWizard. const buildInterventionContent = (): JSX.Element => { if (isDoorOpen) { return @@ -119,6 +144,7 @@ export function ErrorRecoveryComponent( !isDoorOpen && route === RECOVERY_MAP.DROP_TIP_FLOWS.ROUTE && step !== RECOVERY_MAP.DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL + const desktopType = isLargeDesktopStyle ? 'desktop-large' : 'desktop-small' return ( {showModal ? ( - + ) : null} {buildInterventionContent()} diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx index 96de863c965..8f49634a3ad 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx @@ -1,6 +1,5 @@ import * as React from 'react' import { useTranslation } from 'react-i18next' -import { css } from 'styled-components' import { ALIGN_CENTER, @@ -10,10 +9,13 @@ import { Icon, SPACING, StyledText, - RESPONSIVENESS, } from '@opentrons/components' -import { RECOVERY_MAP } from '../constants' +import { + FLEX_WIDTH_ALERT_INFO_STYLE, + ICON_SIZE_ALERT_INFO_STYLE, + RECOVERY_MAP, +} from '../constants' import { RecoveryFooterButtons, RecoverySingleColumnContentWrapper, @@ -71,11 +73,11 @@ function CancelRunConfirmation({ gridGap={SPACING.spacing16} padding={`${SPACING.spacing32} ${SPACING.spacing16}`} height="100%" - css={FLEX_WIDTH} + css={FLEX_WIDTH_ALERT_INFO_STYLE} > @@ -147,19 +149,3 @@ export function useOnCancelRun({ return { showBtnLoadingState, handleCancelRunClick } } - -const FLEX_WIDTH = css` - width: 41.625rem; - @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { - width: 53rem; - } -` - -const ICON_SIZE = css` - width: ${SPACING.spacing40}; - height: ${SPACING.spacing40}; - @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { - width: ${SPACING.spacing60}; - height: ${SPACING.spacing60}; - } -` diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx index 87a195a12a8..6dbc6924559 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx @@ -1,7 +1,6 @@ import * as React from 'react' -import { useTranslation } from 'react-i18next' +import { Trans, useTranslation } from 'react-i18next' import head from 'lodash/head' -import { css } from 'styled-components' import { DIRECTION_COLUMN, @@ -9,28 +8,25 @@ import { SPACING, Flex, StyledText, - RESPONSIVENESS, + ALIGN_CENTER, + Icon, } from '@opentrons/components' -import { FLEX_ROBOT_TYPE, OT2_ROBOT_TYPE } from '@opentrons/shared-data' -import { RadioButton } from '../../../atoms/buttons' import { - ODD_SECTION_TITLE_STYLE, RECOVERY_MAP, - ODD_ONLY, - DESKTOP_ONLY, + FLEX_WIDTH_ALERT_INFO_STYLE, + ICON_SIZE_ALERT_INFO_STYLE, } from '../constants' import { RecoveryFooterButtons, RecoverySingleColumnContentWrapper, - RecoveryRadioGroup, } from '../shared' import { DropTipWizardFlows } from '../../DropTipWizardFlows' import { DT_ROUTES } from '../../DropTipWizardFlows/constants' import { SelectRecoveryOption } from './SelectRecoveryOption' import type { PipetteWithTip } from '../../DropTipWizardFlows' -import type { RecoveryContentProps } from '../types' +import type { RecoveryContentProps, RecoveryRoute, RouteStep } from '../types' import type { FixitCommandTypeUtils } from '../../DropTipWizardFlows/types' // The Drop Tip flow entry point. Includes entry from SelectRecoveryOption and CancelRun. @@ -57,8 +53,6 @@ export function ManageTips(props: RecoveryContentProps): JSX.Element { return buildContent() } -type RemovalOptions = 'begin-removal' | 'skip' - export function BeginRemoval({ tipStatusUtils, routeUpdateActions, @@ -77,108 +71,70 @@ export function BeginRemoval({ const { ROBOT_CANCELING, RETRY_NEW_TIPS } = RECOVERY_MAP const mount = head(pipettesWithTip)?.mount - const [selected, setSelected] = React.useState( - 'begin-removal' - ) - const primaryOnClick = (): void => { - if (selected === 'begin-removal') { - void proceedNextStep() - } else { - if (selectedRecoveryOption === RETRY_NEW_TIPS.ROUTE) { - void proceedToRouteAndStep( - RETRY_NEW_TIPS.ROUTE, - RETRY_NEW_TIPS.STEPS.REPLACE_TIPS - ) - } else { - void setRobotInMotion(true, ROBOT_CANCELING.ROUTE).then(() => { - cancelRun() - }) - } - } + void proceedNextStep() } - const DESKTOP_ONLY_GRID_GAP = css` - @media not (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { - gap: 0rem; - } - ` - - const RADIO_GROUP_STYLE = css` - @media not (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { - color: ${COLORS.black90}; - margin-left: 0.5rem; + const secondaryOnClick = (): void => { + if (selectedRecoveryOption === RETRY_NEW_TIPS.ROUTE) { + void proceedToRouteAndStep( + RETRY_NEW_TIPS.ROUTE, + RETRY_NEW_TIPS.STEPS.REPLACE_TIPS + ) + } else { + void setRobotInMotion(true, ROBOT_CANCELING.ROUTE).then(() => { + cancelRun() + }) } - ` + } return ( - - - {t('remove_tips_from_pipette', { mount })} - + - { - setSelected('begin-removal') - }} - isSelected={selected === 'begin-removal'} - /> - { - setSelected('skip') - }} - isSelected={selected === 'skip'} - /> - - - ) => { - setSelected(e.currentTarget.value as RemovalOptions) - }} - options={[ - { - value: 'begin-removal', - children: ( - - {t('begin_removal')} - - ), - }, - { - value: 'skip', - children: ( - - {t('skip_removal')} - - ), - }, - ]} + + + {t('remove_any_attached_tips')} + + + , + }} + /> + - + ) } @@ -187,10 +143,10 @@ function DropTipFlowsContainer( props: RecoveryContentProps ): JSX.Element | null { const { + robotType, tipStatusUtils, routeUpdateActions, recoveryCommands, - isFlex, currentRecoveryOptionUtils, } = props const { DROP_TIP_FLOWS, ROBOT_CANCELING, RETRY_NEW_TIPS } = RECOVERY_MAP @@ -229,7 +185,7 @@ function DropTipFlowsContainer( return ( { switch (selectedRecoveryOption) { case RETRY_NEW_TIPS.ROUTE: + case SKIP_STEP_WITH_NEW_TIPS.ROUTE: return t('proceed_to_tip_selection') default: return t('proceed_to_cancel') } } + const buildTipDropCompleteRouting = (): (() => void) | null => { + const routeTo = (selectedRoute: RecoveryRoute, step: RouteStep): void => { + void proceedToRouteAndStep(selectedRoute, step) + } + + switch (selectedRecoveryOption) { + case RETRY_NEW_TIPS.ROUTE: + return () => { + routeTo(selectedRecoveryOption, RETRY_NEW_TIPS.STEPS.REPLACE_TIPS) + } + case SKIP_STEP_WITH_NEW_TIPS.ROUTE: + return () => { + routeTo( + selectedRecoveryOption, + SKIP_STEP_WITH_NEW_TIPS.STEPS.REPLACE_TIPS + ) + } + default: + return null + } + } + const buildCopyOverrides = (): FixitCommandTypeUtils['copyOverrides'] => { return { tipDropCompleteBtnCopy: buildTipDropCompleteBtn(), @@ -303,6 +283,7 @@ export function useDropTipFlowUtils({ goBackBeforeBeginning: () => { return proceedToRouteAndStep(DROP_TIP_FLOWS.ROUTE) }, + tipDropComplete: buildTipDropCompleteRouting(), } } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx index ed9de710db8..7fba59a0a6f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx @@ -51,6 +51,7 @@ export function SelectRecoveryOptionHome({ tipStatusUtils, currentRecoveryOptionUtils, getRecoveryOptionCopy, + analytics, ...rest }: RecoveryContentProps): JSX.Element | null { const { t } = useTranslation('error_recovery') @@ -68,6 +69,7 @@ export function SelectRecoveryOptionHome({ { + analytics.reportActionSelectedEvent(selectedRoute) setSelectedRecoveryOption(selectedRoute) void proceedToRouteAndStep(selectedRoute as RecoveryRoute) }, diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx index ed58d7e597a..362d30e2860 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx @@ -90,25 +90,27 @@ describe('ManageTips', () => { it(`renders BeginRemoval with correct copy when the step is ${DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL}`, () => { render(props) - screen.getByText('Remove tips from left pipette before canceling the run?') + screen.getByText('Remove any attached tips') + screen.queryByText( + /Homing the .* pipette with liquid in the tips may damage it\. You must remove all tips before using the pipette again\./ + ) screen.queryAllByText('Begin removal') screen.queryAllByText('Skip') - expect(screen.getAllByText('Continue').length).toBe(2) }) it('routes correctly when continuing on BeginRemoval', () => { render(props) const beginRemovalBtn = screen.queryAllByText('Begin removal')[0] - const skipBtn = screen.queryAllByText('Skip removal')[0] + const skipBtn = screen.queryAllByText('Skip')[0] fireEvent.click(beginRemovalBtn) - clickButtonLabeled('Continue') + clickButtonLabeled('Begin removal') expect(mockProceedNextStep).toHaveBeenCalled() fireEvent.click(skipBtn) - clickButtonLabeled('Continue') + clickButtonLabeled('Skip') expect(mockSetRobotInMotion).toHaveBeenCalled() }) @@ -122,10 +124,10 @@ describe('ManageTips', () => { } render(props) - const skipBtn = screen.queryAllByText('Skip removal')[0] + const skipBtn = screen.queryAllByText('Skip')[0] fireEvent.click(skipBtn) - clickButtonLabeled('Continue') + clickButtonLabeled('Skip') expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( RETRY_NEW_TIPS.ROUTE, @@ -264,11 +266,34 @@ describe('useDropTipFlowUtils', () => { }) it('should return the correct button overrides', () => { - const { result } = renderHook(() => useDropTipFlowUtils(mockProps)) + const { result } = renderHook(() => + useDropTipFlowUtils({ + ...mockProps, + recoveryMap: { + route: RETRY_NEW_TIPS.ROUTE, + step: RETRY_NEW_TIPS.STEPS.DROP_TIPS, + }, + currentRecoveryOptionUtils: { + selectedRecoveryOption: RETRY_NEW_TIPS.ROUTE, + } as any, + }) + ) + const { tipDropComplete } = result.current.buttonOverrides result.current.buttonOverrides.goBackBeforeBeginning() expect(mockProceedToRouteAndStep).toHaveBeenCalledWith(DROP_TIP_FLOWS.ROUTE) + + expect(tipDropComplete).toBeDefined() + + if (tipDropComplete != null) { + tipDropComplete() + } + + expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( + RETRY_NEW_TIPS.ROUTE, + RETRY_NEW_TIPS.STEPS.REPLACE_TIPS + ) }) it(`should return correct route overrides when the route is ${DROP_TIP_FLOWS.STEPS.CHOOSE_TIP_DROP}`, () => { diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx new file mode 100644 index 00000000000..3c4d0a7b261 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryTakeover.tsx @@ -0,0 +1,171 @@ +import * as React from 'react' +import { useTranslation } from 'react-i18next' +import { css } from 'styled-components' + +import { + DIRECTION_COLUMN, + Flex, + SPACING, + COLORS, + Icon, + StyledText, + AlertPrimaryButton, + ALIGN_CENTER, + JUSTIFY_CENTER, + TEXT_ALIGN_CENTER, + JUSTIFY_SPACE_BETWEEN, +} from '@opentrons/components' +import { + RUN_STATUS_AWAITING_RECOVERY, + RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + RUN_STATUS_AWAITING_RECOVERY_PAUSED, +} from '@opentrons/api-client' + +import { useUpdateClientDataRecovery } from '../../resources/client_data' +import { TakeoverModal } from '../TakeoverModal/TakeoverModal' +import { RecoveryInterventionModal } from './shared' + +import type { + ClientDataRecovery, + UseUpdateClientDataRecoveryResult, +} from '../../resources/client_data' +import type { ErrorRecoveryFlowsProps } from '.' + +// The takeover view, functionally similar to MaintenanceRunTakeover +export function RecoveryTakeover(props: { + intent: ClientDataRecovery['intent'] + runStatus: ErrorRecoveryFlowsProps['runStatus'] + robotName: string + isOnDevice: boolean +}): JSX.Element { + const { runStatus } = props + const { t } = useTranslation('error_recovery') + const { clearClientData } = useUpdateClientDataRecovery() + + // TODO(jh, 07-29-24): This is likely sufficient for most edge cases, but this does not account for + // all terminal commands as it should. Revisit this. + const isTerminateDisabled = !( + runStatus === RUN_STATUS_AWAITING_RECOVERY || + runStatus === RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR || + runStatus === RUN_STATUS_AWAITING_RECOVERY_PAUSED + ) + + const buildRecoveryTakeoverProps = ( + intent: ClientDataRecovery['intent'] + ): RecoveryTakeoverProps => { + switch (intent) { + case 'canceling': + return { + title: t('robot_is_canceling_run'), + isRunStatusAwaitingRecovery: isTerminateDisabled, + clearClientData, + ...props, + } + case 'recovering': + default: + return { + title: t('robot_is_in_recovery_mode'), + isRunStatusAwaitingRecovery: isTerminateDisabled, + clearClientData, + ...props, + } + } + } + + return ( + + ) +} + +interface RecoveryTakeoverProps { + title: string + /* Do not let other users terminate activity if run is not awaiting recovery. Ex, the run is "recovery canceling." */ + isRunStatusAwaitingRecovery: boolean + robotName: string + isOnDevice: boolean + clearClientData: UseUpdateClientDataRecoveryResult['clearClientData'] +} + +export function RecoveryTakeoverComponent( + props: RecoveryTakeoverProps +): JSX.Element { + return props.isOnDevice ? ( + + ) : ( + + ) +} + +export function RecoveryTakeoverODD({ + title, + clearClientData, + isRunStatusAwaitingRecovery, +}: RecoveryTakeoverProps): JSX.Element { + const [showConfirmation, setShowConfirmation] = React.useState(false) + + return ( + + ) +} + +export function RecoveryTakeoverDesktop({ + title, + robotName, + clearClientData, + isRunStatusAwaitingRecovery, +}: RecoveryTakeoverProps): JSX.Element { + const { t } = useTranslation('error_recovery') + + return ( + + + + + {title} + + {t('another_app_controlling_robot')} + + + + + {t('terminate_remote_activity')} + + + + + ) +} + +const CONTAINER_STYLE = css` + flex-direction: ${DIRECTION_COLUMN}; + justify-content: ${JUSTIFY_SPACE_BETWEEN}; + align-items: ${ALIGN_CENTER}; + padding-top: ${SPACING.spacing12}; +` + +const CONTENT_STYLE = css` + flex-direction: ${DIRECTION_COLUMN}; + justify-content: ${JUSTIFY_CENTER}; + align-items: ${ALIGN_CENTER}; + + text-align: ${TEXT_ALIGN_CENTER}; + padding: ${SPACING.spacing40} ${SPACING.spacing40}; + grid-gap: ${SPACING.spacing16}; +` diff --git a/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx b/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx index f9d253719ed..db66388deaa 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx @@ -20,7 +20,6 @@ import { StyledText, JUSTIFY_END, PrimaryButton, - ALIGN_FLEX_END, SecondaryButton, } from '@opentrons/components' @@ -36,8 +35,11 @@ import { import type { RobotType } from '@opentrons/shared-data' import type { ErrorRecoveryFlowsProps } from '.' -import type { ERUtilsResults } from './hooks' -import { useHost } from '@opentrons/react-api-client' +import type { + ERUtilsResults, + UseRecoveryAnalyticsResult, + UseRecoveryTakeoverResult, +} from './hooks' export function useRunPausedSplash( isOnDevice: boolean, @@ -45,7 +47,11 @@ export function useRunPausedSplash( ): boolean { // Don't show the splash when desktop ER wizard is active, // but always show it on the ODD (with or without the wizard rendered above it). - return !(!isOnDevice && showERWizard) + if (isOnDevice) { + return true + } else { + return !showERWizard + } } type RunPausedSplashProps = ERUtilsResults & { @@ -53,35 +59,49 @@ type RunPausedSplashProps = ERUtilsResults & { failedCommand: ErrorRecoveryFlowsProps['failedCommand'] protocolAnalysis: ErrorRecoveryFlowsProps['protocolAnalysis'] robotType: RobotType - toggleERWiz: (launchER: boolean) => Promise + robotName: string + toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] + analytics: UseRecoveryAnalyticsResult } export function RunPausedSplash( props: RunPausedSplashProps ): JSX.Element | null { - const { isOnDevice, toggleERWiz, routeUpdateActions, failedCommand } = props + const { + isOnDevice, + toggleERWizAsActiveUser, + routeUpdateActions, + failedCommand, + analytics, + robotName, + } = props const { t } = useTranslation('error_recovery') const errorKind = getErrorKind(failedCommand) const title = useErrorName(errorKind) - const host = useHost() const { proceedToRouteAndStep } = routeUpdateActions + const { reportInitialActionEvent } = analytics const buildTitleHeadingDesktop = (): JSX.Element => { return ( - {t('error_on_robot', { robot: host?.robotName ?? '' })} + {t('error_on_robot', { robot: robotName })} ) } // Do not launch error recovery, but do utilize the wizard's cancel route. const onCancelClick = (): Promise => { - return toggleERWiz(false).then(() => - proceedToRouteAndStep(RECOVERY_MAP.CANCEL_RUN.ROUTE) - ) + return toggleERWizAsActiveUser(true, false).then(() => { + reportInitialActionEvent('cancel-run') + void proceedToRouteAndStep(RECOVERY_MAP.CANCEL_RUN.ROUTE) + }) } - const onLaunchERClick = (): Promise => toggleERWiz(true) + const onLaunchERClick = (): Promise => { + return toggleERWizAsActiveUser(true, true).then(() => { + reportInitialActionEvent('launch-recovery') + }) + } // TODO(jh 05-22-24): The hardcoded Z-indexing is non-ideal but must be done to keep the splash page above // several components in the RunningProtocol page. Investigate why these components have seemingly arbitrary zIndex values @@ -115,7 +135,7 @@ export function RunPausedSplash( {title} {}, + reportErrorEvent: () => {}, + reportViewErrorDetailsEvent: () => {}, + reportActionSelectedEvent: () => {}, + reportInitialActionEvent: () => {}, + reportActionSelectedResult: () => {}, + }, } diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx index 427bb8d00cd..54462e62f22 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx @@ -2,6 +2,7 @@ import * as React from 'react' import { vi, describe, expect, it, beforeEach } from 'vitest' import { screen, renderHook } from '@testing-library/react' +import { useHost } from '@opentrons/react-api-client' import { RUN_STATUS_AWAITING_RECOVERY, RUN_STATUS_RUNNING, @@ -16,8 +17,10 @@ import { useCurrentlyRecoveringFrom, useERUtils, useShowDoorInfo, + useRecoveryAnalytics, + useRecoveryTakeover, } from '../hooks' -import { useFeatureFlag } from '../../../redux/config' +import { getIsOnDevice } from '../../../redux/config' import { useERWizard, ErrorRecoveryWizard } from '../ErrorRecoveryWizard' import { useRunPausedSplash, RunPausedSplash } from '../RunPausedSplash' @@ -28,6 +31,14 @@ vi.mock('../hooks') vi.mock('../useRecoveryCommands') vi.mock('../../../redux/config') vi.mock('../RunPausedSplash') +vi.mock('@opentrons/react-api-client') +vi.mock('react-redux', async () => { + const actual = await vi.importActual('react-redux') + return { + ...actual, + useSelector: vi.fn(), + } +}) describe('useErrorRecoveryFlows', () => { beforeEach(() => { @@ -121,7 +132,7 @@ const render = (props: React.ComponentProps) => { })[0] } -describe('ErrorRecovery', () => { +describe('ErrorRecoveryFlows', () => { let props: React.ComponentProps beforeEach(() => { @@ -129,14 +140,12 @@ describe('ErrorRecovery', () => { runStatus: RUN_STATUS_AWAITING_RECOVERY, failedCommand: mockFailedCommand, runId: 'MOCK_RUN_ID', - isFlex: true, protocolAnalysis: {} as any, } vi.mocked(ErrorRecoveryWizard).mockReturnValue(
MOCK WIZARD
) vi.mocked(RunPausedSplash).mockReturnValue(
MOCK RUN PAUSED SPLASH
) - vi.mocked(useFeatureFlag).mockReturnValue(true) vi.mocked(useERWizard).mockReturnValue({ hasLaunchedRecovery: true, toggleERWizard: () => Promise.resolve(), @@ -145,9 +154,20 @@ describe('ErrorRecovery', () => { vi.mocked(useRunPausedSplash).mockReturnValue(true) vi.mocked(useERUtils).mockReturnValue({ routeUpdateActions: {} } as any) vi.mocked(useShowDoorInfo).mockReturnValue(false) + vi.mocked(useRecoveryAnalytics).mockReturnValue({ + reportErrorEvent: vi.fn(), + } as any) + vi.mocked(useHost).mockReturnValue({ robotName: 'MockRobot' } as any) + vi.mocked(getIsOnDevice).mockReturnValue(false) + vi.mocked(useRecoveryTakeover).mockReturnValue({ + toggleERWizAsActiveUser: vi.fn(), + isActiveUser: true, + intent: 'recovering', + showTakeover: false, + }) }) - it('renders the wizard when the wizard is toggled on', () => { + it('renders the wizard when showERWizard is true', () => { render(props) screen.getByText('MOCK WIZARD') }) @@ -164,25 +184,43 @@ describe('ErrorRecovery', () => { screen.getByText('MOCK WIZARD') }) - it('does not render the wizard when the wizard is toggled off', () => { + it('does not render the wizard when showERWizard is false and isDoorOpen is false', () => { vi.mocked(useERWizard).mockReturnValue({ hasLaunchedRecovery: true, toggleERWizard: () => Promise.resolve(), showERWizard: false, }) + vi.mocked(useShowDoorInfo).mockReturnValue(false) render(props) expect(screen.queryByText('MOCK WIZARD')).not.toBeInTheDocument() }) - it('renders the splash when the showSplash is true', () => { + it('renders the splash when showSplash is true', () => { render(props) screen.getByText('MOCK RUN PAUSED SPLASH') }) - it('does not render the splash when the showSplash is false', () => { + it('does not render the splash when showSplash is false', () => { vi.mocked(useRunPausedSplash).mockReturnValue(false) render(props) expect(screen.queryByText('MOCK RUN PAUSED SPLASH')).not.toBeInTheDocument() }) + + it('calls reportErrorEvent with failedCommand on mount and when failedCommand changes', () => { + const mockReportErrorEvent = vi.fn() + vi.mocked(useRecoveryAnalytics).mockReturnValue({ + reportErrorEvent: mockReportErrorEvent, + } as any) + + const { rerender } = render(props) + expect(mockReportErrorEvent).toHaveBeenCalledWith(mockFailedCommand) + + const newProps = { + ...props, + failedCommand: null, + } + rerender() + expect(mockReportErrorEvent).toHaveBeenCalledWith(newProps.failedCommand) + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx index 3c5174bf54d..1e80b53779d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryWizard.test.tsx @@ -44,28 +44,48 @@ vi.mock('../shared', async importOriginal => { } }) describe('useERWizard', () => { - it('has correct initial values', () => { - const { result } = renderHook(() => useERWizard()) - expect(result.current.showERWizard).toBe(false) - expect(result.current.hasLaunchedRecovery).toBe(false) - }) + describe('useERWizard', () => { + it('has correct initial values', () => { + const { result } = renderHook(() => useERWizard()) + expect(result.current.showERWizard).toBe(false) + expect(result.current.hasLaunchedRecovery).toBe(false) + }) - it('correctly toggles showERWizard and updates hasLaunchedRecovery as expected', async () => { - const { result } = renderHook(() => useERWizard()) + it('correctly toggles showERWizard and updates hasLaunchedRecovery when hasLaunchedER is provided', async () => { + const { result } = renderHook(() => useERWizard()) - await act(async () => { - await result.current.toggleERWizard(true) - }) + await act(async () => { + await result.current.toggleERWizard(true, true) + }) - expect(result.current.showERWizard).toBe(true) - expect(result.current.hasLaunchedRecovery).toBe(true) + expect(result.current.showERWizard).toBe(true) + expect(result.current.hasLaunchedRecovery).toBe(true) - await act(async () => { - await result.current.toggleERWizard(false) + await act(async () => { + await result.current.toggleERWizard(false, false) + }) + + expect(result.current.showERWizard).toBe(false) + expect(result.current.hasLaunchedRecovery).toBe(false) }) - expect(result.current.showERWizard).toBe(false) - expect(result.current.hasLaunchedRecovery).toBe(false) + it('does not update hasLaunchedRecovery when hasLaunchedER is undefined', async () => { + const { result } = renderHook(() => useERWizard()) + + await act(async () => { + await result.current.toggleERWizard(true) + }) + + expect(result.current.showERWizard).toBe(true) + expect(result.current.hasLaunchedRecovery).toBe(false) + + await act(async () => { + await result.current.toggleERWizard(false) + }) + + expect(result.current.showERWizard).toBe(false) + expect(result.current.hasLaunchedRecovery).toBe(false) + }) }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx new file mode 100644 index 00000000000..0caeffbf89f --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RecoveryTakeover.test.tsx @@ -0,0 +1,122 @@ +import * as React from 'react' +import { describe, it, vi, expect, beforeEach } from 'vitest' +import { screen } from '@testing-library/react' + +import { + RUN_STATUS_AWAITING_RECOVERY, + RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + RUN_STATUS_AWAITING_RECOVERY_PAUSED, +} from '@opentrons/api-client' + +import { renderWithProviders } from '../../../__testing-utils__' +import { i18n } from '../../../i18n' +import { RecoveryTakeover, RecoveryTakeoverDesktop } from '../RecoveryTakeover' +import { useUpdateClientDataRecovery } from '../../../resources/client_data' +import { clickButtonLabeled } from './util' + +import type { Mock } from 'vitest' + +vi.mock('../../../resources/client_data') + +const render = (props: React.ComponentProps) => { + return renderWithProviders(, { + i18nInstance: i18n, + })[0] +} + +describe('RecoveryTakeover', () => { + let props: React.ComponentProps + let mockClearClientData: Mock + + beforeEach(() => { + mockClearClientData = vi.fn() + vi.mocked(useUpdateClientDataRecovery).mockReturnValue({ + clearClientData: mockClearClientData, + } as any) + + props = { + intent: 'recovering', + runStatus: RUN_STATUS_AWAITING_RECOVERY, + robotName: 'TestRobot', + isOnDevice: false, + } + }) + + it('renders RecoveryTakeoverComponent with correct props for recovering intent on desktop', () => { + render(props) + screen.getByText('Error on TestRobot') + screen.getByText('Robot is in recovery mode') + screen.getByText( + 'The robot’s touchscreen or another computer with the app is currently controlling this robot.' + ) + }) + + it('renders RecoveryTakeoverComponent with correct props for canceling intent on desktop', () => { + render({ ...props, intent: 'canceling' }) + screen.getByText('Error on TestRobot') + screen.getByText('Robot is canceling the run') + screen.getByText( + 'The robot’s touchscreen or another computer with the app is currently controlling this robot.' + ) + }) + ;[ + RUN_STATUS_AWAITING_RECOVERY, + RUN_STATUS_AWAITING_RECOVERY_PAUSED, + RUN_STATUS_AWAITING_RECOVERY_BLOCKED_BY_OPEN_DOOR, + ].forEach(runStatus => { + it(`renders the terminate button as enabled when run status is ${runStatus}`, () => { + render({ ...props, runStatus }) + + expect(screen.getByText('Terminate remote activity')).toBeEnabled() + }) + }) + + it('renders RecoveryTakeoverComponent with correct props for recovering intent on ODD', () => { + render({ ...props, isOnDevice: true }) + screen.getByText('Robot is in recovery mode') + screen.getByText( + 'A computer with the Opentrons App is currently controlling this robot.' + ) + screen.getByText('Terminate remote activity') + }) + + it('renders RecoveryTakeoverComponent with correct props for canceling intent on ODD', () => { + render({ ...props, isOnDevice: true, intent: 'canceling' }) + screen.getByText('Robot is canceling the run') + screen.getByText( + 'A computer with the Opentrons App is currently controlling this robot.' + ) + screen.getByText('Terminate remote activity') + }) +}) + +describe('RecoveryTakeoverDesktop', () => { + let props: React.ComponentProps + + beforeEach(() => { + props = { + title: 'Test Title', + isRunStatusAwaitingRecovery: false, + robotName: 'TestRobot', + isOnDevice: false, + clearClientData: vi.fn(), + } + }) + + it('calls clearClientData when terminate button is clicked', () => { + renderWithProviders(, { + i18nInstance: i18n, + }) + clickButtonLabeled('Terminate remote activity') + + expect(props.clearClientData).toHaveBeenCalled() + }) + + it('disables terminate button when isRunStatusAwaitingRecovery is true', () => { + props.isRunStatusAwaitingRecovery = true + renderWithProviders(, { + i18nInstance: i18n, + }) + expect(screen.getByText('Terminate remote activity')).toBeDisabled() + }) +}) diff --git a/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx b/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx index ff3c8d1cc34..9968012efeb 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/RunPausedSplash.test.tsx @@ -78,7 +78,8 @@ describe('RunPausedSplash', () => { beforeEach(() => { props = { ...mockRecoveryContentProps, - toggleERWiz: mockToggleERWiz, + robotName: 'testRobot', + toggleERWizAsActiveUser: mockToggleERWiz, routeUpdateActions: mockRouteUpdateActions, } @@ -126,7 +127,7 @@ describe('RunPausedSplash', () => { expect(mockToggleERWiz).toHaveBeenCalledTimes(1) }) await waitFor(() => { - expect(mockToggleERWiz).toHaveBeenCalledWith(false) + expect(mockToggleERWiz).toHaveBeenCalledWith(true, false) }) await waitFor(() => { expect(mockProceedToRouteAndStep).toHaveBeenCalledTimes(1) @@ -141,7 +142,7 @@ describe('RunPausedSplash', () => { expect(mockToggleERWiz).toHaveBeenCalledTimes(2) }) await waitFor(() => { - expect(mockToggleERWiz).toHaveBeenCalledWith(true) + expect(mockToggleERWiz).toHaveBeenCalledWith(true, true) }) }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/constants.ts b/app/src/organisms/ErrorRecoveryFlows/constants.ts index 846f7e2efc0..2af09a3bf97 100644 --- a/app/src/organisms/ErrorRecoveryFlows/constants.ts +++ b/app/src/organisms/ErrorRecoveryFlows/constants.ts @@ -1,6 +1,6 @@ import { css } from 'styled-components' -import { SPACING, TYPOGRAPHY, RESPONSIVENESS } from '@opentrons/components' +import { RESPONSIVENESS, SPACING } from '@opentrons/components' import type { StepOrder } from './types' @@ -204,10 +204,6 @@ export const INVALID = 'INVALID' as const * Styling */ -export const BODY_TEXT_STYLE = css` - ${TYPOGRAPHY.bodyTextRegular}; -` - export const ODD_SECTION_TITLE_STYLE = css` margin-bottom: ${SPACING.spacing16}; ` @@ -222,3 +218,17 @@ export const DESKTOP_ONLY = css` display: none; } ` +export const FLEX_WIDTH_ALERT_INFO_STYLE = css` + width: 41.625rem; + @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { + width: 53rem; + } +` +export const ICON_SIZE_ALERT_INFO_STYLE = css` + width: ${SPACING.spacing40}; + height: ${SPACING.spacing40}; + @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { + width: ${SPACING.spacing60}; + height: ${SPACING.spacing60}; + } +` diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts index d75387a99d4..a55f3ef43f2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts @@ -4,6 +4,7 @@ import { renderHook, act } from '@testing-library/react' import { useResumeRunFromRecoveryMutation, useStopRunMutation, + useUpdateErrorRecoveryPolicy, } from '@opentrons/react-api-client' import { useChainRunCommands } from '../../../../resources/runs' @@ -11,34 +12,50 @@ import { useRecoveryCommands, HOME_PIPETTE_Z_AXES, buildPickUpTips, + buildIgnorePolicyRules, } from '../useRecoveryCommands' import { RECOVERY_MAP } from '../../constants' vi.mock('@opentrons/react-api-client') vi.mock('../../../../resources/runs') -const mockFailedCommand = { - id: 'MOCK_ID', - commandType: 'mockCommandType', - params: { test: 'mock_param' }, -} as any -const mockRunId = '123' -const mockFailedLabwareUtils = { - selectedTipLocations: { A1: null }, - pickUpTipLabware: { id: 'MOCK_LW_ID' }, -} as any -const mockProceedToRouteAndStep = vi.fn() -const mockRouteUpdateActions = { - proceedToRouteAndStep: mockProceedToRouteAndStep, -} as any - describe('useRecoveryCommands', () => { + const mockFailedCommand = { + id: 'MOCK_ID', + commandType: 'mockCommandType', + params: { test: 'mock_param' }, + } as any + const mockRunId = '123' + const mockFailedLabwareUtils = { + selectedTipLocations: { A1: null }, + pickUpTipLabware: { id: 'MOCK_LW_ID' }, + } as any + const mockProceedToRouteAndStep = vi.fn() + const mockRouteUpdateActions = { + proceedToRouteAndStep: mockProceedToRouteAndStep, + } as any const mockMakeSuccessToast = vi.fn() const mockResumeRunFromRecovery = vi.fn(() => Promise.resolve(mockMakeSuccessToast()) ) const mockStopRun = vi.fn() const mockChainRunCommands = vi.fn().mockResolvedValue([]) + const mockReportActionSelectedResult = vi.fn() + const mockReportRecoveredRunResult = vi.fn() + const mockUpdateErrorRecoveryPolicy = vi.fn() + + const props = { + runId: mockRunId, + failedCommand: mockFailedCommand, + failedLabwareUtils: mockFailedLabwareUtils, + routeUpdateActions: mockRouteUpdateActions, + recoveryToastUtils: { makeSuccessToast: mockMakeSuccessToast } as any, + analytics: { + reportActionSelectedResult: mockReportActionSelectedResult, + reportRecoveredRunResult: mockReportRecoveredRunResult, + } as any, + selectedRecoveryOption: RECOVERY_MAP.RETRY_NEW_TIPS.ROUTE, + } beforeEach(() => { vi.mocked(useResumeRunFromRecoveryMutation).mockReturnValue({ @@ -50,21 +67,16 @@ describe('useRecoveryCommands', () => { vi.mocked(useChainRunCommands).mockReturnValue({ chainRunCommands: mockChainRunCommands, } as any) + vi.mocked(useUpdateErrorRecoveryPolicy).mockReturnValue({ + updateErrorRecoveryPolicy: mockUpdateErrorRecoveryPolicy, + } as any) }) it('should call chainRunRecoveryCommands with continuePastCommandFailure set to false', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { - await result.current.homePipetteZAxes() // can use any result returned command + await result.current.homePipetteZAxes() }) expect(mockChainRunCommands).toHaveBeenCalledWith( @@ -79,15 +91,7 @@ describe('useRecoveryCommands', () => { chainRunCommands: vi.fn().mockRejectedValue(mockError), } as any) - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { await expect(result.current.homePipetteZAxes()).rejects.toThrow( @@ -106,15 +110,7 @@ describe('useRecoveryCommands', () => { params: mockFailedCommand.params, } - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { await result.current.retryFailedCommand() @@ -125,17 +121,67 @@ describe('useRecoveryCommands', () => { false ) }) + ;([ + 'aspirateInPlace', + 'dispenseInPlace', + 'blowOutInPlace', + 'dropTipInPlace', + 'prepareToAspirate', + ] as const).forEach(inPlaceCommandType => { + it(`Should move to retryLocation if failed command is ${inPlaceCommandType} and error is appropriate when retrying`, async () => { + const { result } = renderHook(() => + useRecoveryCommands({ + runId: mockRunId, + failedCommand: { + ...mockFailedCommand, + commandType: inPlaceCommandType, + params: { + pipetteId: 'mock-pipette-id', + }, + error: { + errorType: 'overpressure', + errorCode: '3006', + isDefined: true, + errorInfo: { + retryLocation: [1, 2, 3], + }, + }, + }, + failedLabwareUtils: mockFailedLabwareUtils, + routeUpdateActions: mockRouteUpdateActions, + recoveryToastUtils: {} as any, + analytics: { + reportActionSelectedResult: mockReportActionSelectedResult, + reportRecoveredRunResult: mockReportRecoveredRunResult, + } as any, + selectedRecoveryOption: RECOVERY_MAP.RETRY_NEW_TIPS.ROUTE, + }) + ) + await act(async () => { + await result.current.retryFailedCommand() + }) + expect(mockChainRunCommands).toHaveBeenLastCalledWith( + [ + { + commandType: 'moveToCoordinates', + intent: 'fixit', + params: { + pipetteId: 'mock-pipette-id', + coordinates: { x: 1, y: 2, z: 3 }, + }, + }, + { + commandType: inPlaceCommandType, + params: { pipetteId: 'mock-pipette-id' }, + }, + ], + false + ) + }) + }) it('should call resumeRun with runId and show success toast on success', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: { makeSuccessToast: mockMakeSuccessToast } as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { await result.current.resumeRun() @@ -146,15 +192,7 @@ describe('useRecoveryCommands', () => { }) it('should call cancelRun with runId', () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) result.current.cancelRun() @@ -162,15 +200,7 @@ describe('useRecoveryCommands', () => { }) it('should call homePipetteZAxes with the appropriate command', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { await result.current.homePipetteZAxes() @@ -198,18 +228,16 @@ describe('useRecoveryCommands', () => { mockFailedLabware ) - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCmdWithPipetteId, - failedLabwareUtils: { - ...mockFailedLabwareUtils, - failedLabware: mockFailedLabware, - }, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const testProps = { + ...props, + failedCommand: mockFailedCmdWithPipetteId, + failedLabwareUtils: { + ...mockFailedLabwareUtils, + failedLabware: mockFailedLabware, + }, + } + + const { result } = renderHook(() => useRecoveryCommands(testProps)) await act(async () => { await result.current.pickUpTips() @@ -217,20 +245,12 @@ describe('useRecoveryCommands', () => { expect(mockChainRunCommands).toHaveBeenCalledWith( [buildPickUpTipsCmd], - true + false ) }) it('should call skipFailedCommand and show success toast on success', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: { makeSuccessToast: mockMakeSuccessToast } as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) await act(async () => { await result.current.skipFailedCommand() @@ -240,26 +260,46 @@ describe('useRecoveryCommands', () => { expect(mockMakeSuccessToast).toHaveBeenCalled() }) - it('should call ignoreErrorKindThisRun and resolve immediately', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + it('should call updateErrorRecoveryPolicy with correct policy rules when failedCommand has an error', async () => { + const mockFailedCommandWithError = { + ...mockFailedCommand, + commandType: 'aspirateInPlace', + error: { + errorType: 'mockErrorType', + }, + } - const consoleSpy = vi.spyOn(console, 'log') + const testProps = { + ...props, + failedCommand: mockFailedCommandWithError, + } + + const { result } = renderHook(() => useRecoveryCommands(testProps)) await act(async () => { await result.current.ignoreErrorKindThisRun() }) - expect(consoleSpy).toHaveBeenCalledWith( - 'IGNORING ALL ERRORS OF THIS KIND THIS RUN' + const expectedPolicyRules = buildIgnorePolicyRules( + 'aspirateInPlace', + 'mockErrorType' + ) + + expect(mockUpdateErrorRecoveryPolicy).toHaveBeenCalledWith( + expectedPolicyRules + ) + }) + + it('should reject with an error when failedCommand or error is null', async () => { + const testProps = { + ...props, + failedCommand: null, + } + + const { result } = renderHook(() => useRecoveryCommands(testProps)) + + await expect(result.current.ignoreErrorKindThisRun()).rejects.toThrow( + 'Could not execute command. No failed command.' ) - expect(result.current.ignoreErrorKindThisRun()).resolves.toBeUndefined() }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryTakeover.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryTakeover.test.ts new file mode 100644 index 00000000000..309ea6f0c93 --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryTakeover.test.ts @@ -0,0 +1,123 @@ +import { describe, it, vi, expect, beforeEach } from 'vitest' +import { renderHook, act } from '@testing-library/react' +import { useSelector } from 'react-redux' + +import { useRecoveryTakeover } from '../useRecoveryTakeover' +import { getUserId } from '../../../../redux/config' +import { + useClientDataRecovery, + useUpdateClientDataRecovery, +} from '../../../../resources/client_data' + +import type { Mock } from 'vitest' + +vi.mock('react-redux', () => ({ + useSelector: vi.fn(), +})) +vi.mock('../../../../redux/config') +vi.mock('../../../../resources/client_data') + +describe('useRecoveryTakeover', () => { + let mockToggleERWiz: Mock + let mockUpdateWithIntent: Mock + let mockClearClientData: Mock + + beforeEach(() => { + mockToggleERWiz = vi.fn(() => Promise.resolve()) + mockUpdateWithIntent = vi.fn() + mockClearClientData = vi.fn() + + vi.mocked(useSelector).mockReturnValue('thisUserId') + vi.mocked(getUserId).mockReturnValue('thisUserId') + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: null, + intent: null, + }) + vi.mocked(useUpdateClientDataRecovery).mockReturnValue({ + updateWithIntent: mockUpdateWithIntent, + clearClientData: mockClearClientData, + } as any) + }) + + it('should return initial values', () => { + const { result } = renderHook(() => useRecoveryTakeover(mockToggleERWiz)) + + expect(result.current).toEqual({ + showTakeover: false, + intent: null, + toggleERWizAsActiveUser: expect.any(Function), + isActiveUser: false, + }) + }) + + it('should show takeover when activeId is different from thisUserId', () => { + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: 'otherUserId', + intent: null, + }) + + const { result } = renderHook(() => useRecoveryTakeover(mockToggleERWiz)) + + expect(result.current.showTakeover).toBe(true) + }) + + it('should not show takeover when activeId is null', () => { + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: null, + intent: null, + }) + + const { result } = renderHook(() => useRecoveryTakeover(mockToggleERWiz)) + + expect(result.current.showTakeover).toBe(false) + }) + + it('should update active user status and intent when toggleERWizAsActiveUser is called', async () => { + const { result } = renderHook(() => useRecoveryTakeover(mockToggleERWiz)) + + await act(async () => { + await result.current.toggleERWizAsActiveUser(true, true) + }) + + expect(result.current.isActiveUser).toBe(true) + expect(mockUpdateWithIntent).toHaveBeenCalledWith('recovering') + expect(mockToggleERWiz).toHaveBeenCalledWith(true, true) + }) + + it('should clear client data when toggleERWizAsActiveUser is called to deactivate', async () => { + const { result } = renderHook(() => useRecoveryTakeover(mockToggleERWiz)) + + await act(async () => { + await result.current.toggleERWizAsActiveUser(true, true) + }) + + await act(async () => { + await result.current.toggleERWizAsActiveUser(false, false) + }) + + expect(result.current.isActiveUser).toBe(false) + expect(mockClearClientData).toHaveBeenCalled() + expect(mockToggleERWiz).toHaveBeenCalledWith(false, false) + }) + + it('should update isActiveUser when activeId changes', () => { + const { result, rerender } = renderHook(() => + useRecoveryTakeover(mockToggleERWiz) + ) + + act(() => { + result.current.toggleERWizAsActiveUser(true, true) + }) + + expect(result.current.isActiveUser).toBe(true) + + vi.mocked(useClientDataRecovery).mockReturnValue({ + userId: 'otherUserId', + intent: null, + }) + + rerender() + + expect(result.current.isActiveUser).toBe(false) + }) +}) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts index 3b3cf7bc9f1..b8fd72d06df 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRouteUpdateActions.test.ts @@ -23,7 +23,7 @@ describe('useRouteUpdateActions', () => { useRouteUpdateActionsParams = { hasLaunchedRecovery: true, - toggleERWizard: mockToggleERWizard, + toggleERWizAsActiveUser: mockToggleERWizard, recoveryMap: { route: RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, step: RECOVERY_MAP.RETRY_FAILED_COMMAND.STEPS.CONFIRM_RETRY, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts index 31d9ebb4367..34da52e7224 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts @@ -5,9 +5,13 @@ export { useShowDoorInfo } from './useShowDoorInfo' export { useRecoveryCommands } from './useRecoveryCommands' export { useRouteUpdateActions } from './useRouteUpdateActions' export { useERUtils } from './useERUtils' +export { useRecoveryAnalytics } from './useRecoveryAnalytics' +export { useRecoveryTakeover } from './useRecoveryTakeover' export type { UseRouteUpdateActionsResult } from './useRouteUpdateActions' export type { UseRecoveryCommandsResult } from './useRecoveryCommands' export type { RecoveryTipStatusUtils } from './useRecoveryTipStatus' export type { ERUtilsResults } from './useERUtils' export type { UseFailedLabwareUtilsResult } from './useFailedLabwareUtils' +export type { UseRecoveryAnalyticsResult } from './useRecoveryAnalytics' +export type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts index ff05642ff18..965abf761bc 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts @@ -1,4 +1,5 @@ import { useInstrumentsQuery } from '@opentrons/react-api-client' +import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' import { useRouteUpdateActions } from './useRouteUpdateActions' import { useRecoveryCommands } from './useRecoveryCommands' @@ -28,12 +29,15 @@ import type { UseDeckMapUtilsResult } from './useDeckMapUtils' import type { CurrentRecoveryOptionUtils } from './useRecoveryRouting' import type { RecoveryActionMutationResult } from './useRecoveryActionMutation' import type { StepCounts } from '../../../resources/protocols/hooks' +import type { UseRecoveryAnalyticsResult } from './useRecoveryAnalytics' +import type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' type ERUtilsProps = ErrorRecoveryFlowsProps & { - toggleERWizard: (launchER: boolean) => Promise + toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] hasLaunchedRecovery: boolean isOnDevice: boolean robotType: RobotType + analytics: UseRecoveryAnalyticsResult } export interface ERUtilsResults { @@ -56,14 +60,14 @@ export interface ERUtilsResults { const SUBSEQUENT_COMMAND_DEPTH = 2 // Builds various Error Recovery utilities. export function useERUtils({ - isFlex, failedCommand, runId, - toggleERWizard, + toggleERWizAsActiveUser, hasLaunchedRecovery, protocolAnalysis, isOnDevice, robotType, + analytics, }: ERUtilsProps): ERUtilsResults { const { data: attachedInstruments } = useInstrumentsQuery() const { data: runRecord } = useNotifyRunQuery(runId) @@ -96,7 +100,7 @@ export function useERUtils({ const tipStatusUtils = useRecoveryTipStatus({ runId, - isFlex, + isFlex: robotType === FLEX_ROBOT_TYPE, runRecord, attachedInstruments, }) @@ -104,7 +108,7 @@ export function useERUtils({ const routeUpdateActions = useRouteUpdateActions({ hasLaunchedRecovery, recoveryMap, - toggleERWizard, + toggleERWizAsActiveUser, setRecoveryMap: setRM, }) @@ -128,6 +132,8 @@ export function useERUtils({ failedLabwareUtils, routeUpdateActions, recoveryToastUtils, + analytics, + selectedRecoveryOption: currentRecoveryOptionUtils.selectedRecoveryOption, }) const deckMapUtils = useDeckMapUtils({ diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts index 5de95fe90da..927b867752b 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts @@ -57,7 +57,7 @@ export function useFailedLabwareUtils({ }: UseFailedLabwareUtilsProps): UseFailedLabwareUtilsResult { const recentRelevantFailedLabwareCmd = React.useMemo( () => getRelevantFailedLabwareCmdFrom({ failedCommand, runCommands }), - [failedCommand, runCommands] + [failedCommand?.error?.errorType, runCommands] ) const tipSelectionUtils = useTipSelectionUtils(recentRelevantFailedLabwareCmd) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryAnalytics.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryAnalytics.ts new file mode 100644 index 00000000000..a94c8f290ad --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryAnalytics.ts @@ -0,0 +1,127 @@ +import { RUN_STATUS_FAILED, RUN_STATUS_SUCCEEDED } from '@opentrons/api-client' + +import { + ANALYTICS_RECOVERY_ACTION_RESULT, + ANALYTICS_RECOVERY_ACTION_SELECTED, + ANALYTICS_RECOVERY_ERROR_EVENT, + ANALYTICS_RECOVERY_INITIAL_ACTION, + ANALYTICS_RECOVERY_RUN_RESULT, + ANALYTICS_RECOVERY_VIEW_ERROR_DETAILS, + useTrackEvent, +} from '../../../redux/analytics' + +import type { RunStatus } from '@opentrons/api-client' +import type { FailedCommand, RecoveryRoute, RouteStep } from '../types' + +type InitialActionType = 'cancel-run' | 'launch-recovery' +type CommandResult = 'succeeded' | 'failed' + +export interface UseRecoveryAnalyticsResult { + /* Report the error which occurs error recovery is currently handling. */ + reportErrorEvent: (failedCommand: FailedCommand | null) => void + /* Report which action the user selected on the recovery splash screen. */ + reportInitialActionEvent: (initialAction: InitialActionType) => void + /* Report which recovery option the user selected. */ + reportActionSelectedEvent: (selectedRecoveryOption: RecoveryRoute) => void + /* Report when the user views the error details and where they currently are in Error Recovery. */ + reportViewErrorDetailsEvent: (route: RecoveryRoute, step: RouteStep) => void + /* Report the ultimate result of a selected recovery action, ie, does it result in the run resuming or does the action fail? */ + reportActionSelectedResult: ( + selectedRecoveryOption: RecoveryRoute | null, + result: CommandResult + ) => void + /* Report whether the run succeeds or fails if the run entered error recovery at least once. */ + reportRecoveredRunResult: ( + runStatus: RunStatus | null, + enteredER: boolean + ) => void +} + +export function useRecoveryAnalytics(): UseRecoveryAnalyticsResult { + const doTrackEvent = useTrackEvent() + + const reportErrorEvent = (failedCommand: FailedCommand | null): void => { + if (failedCommand != null) { + doTrackEvent({ + name: ANALYTICS_RECOVERY_ERROR_EVENT, + properties: { + errorEvent: failedCommand.commandType, + errorString: failedCommand.error?.detail, + }, + }) + } + } + + const reportInitialActionEvent = (initialAction: InitialActionType): void => { + doTrackEvent({ + name: ANALYTICS_RECOVERY_INITIAL_ACTION, + properties: { + initialAction, + }, + }) + } + + const reportActionSelectedEvent = ( + selectedRecoveryOption: RecoveryRoute + ): void => { + doTrackEvent({ + name: ANALYTICS_RECOVERY_ACTION_SELECTED, + properties: { + selectedUserAction: selectedRecoveryOption, + }, + }) + } + + const reportViewErrorDetailsEvent = ( + route: RecoveryRoute, + step: RouteStep + ): void => { + doTrackEvent({ + name: ANALYTICS_RECOVERY_VIEW_ERROR_DETAILS, + properties: { + route, + step, + }, + }) + } + + const reportActionSelectedResult = ( + selectedRecoveryOption: RecoveryRoute | null, + result: CommandResult + ): void => { + if (selectedRecoveryOption != null) { + doTrackEvent({ + name: ANALYTICS_RECOVERY_ACTION_RESULT, + properties: { + selectedUserAction: selectedRecoveryOption, + result, + }, + }) + } + } + + const reportRecoveredRunResult = ( + runStatus: RunStatus | null, + enteredER: boolean + ): void => { + if (runStatus === RUN_STATUS_SUCCEEDED || runStatus === RUN_STATUS_FAILED) { + if (enteredER) { + doTrackEvent({ + name: ANALYTICS_RECOVERY_RUN_RESULT, + properties: { + result: runStatus, + }, + }) + } + } + } + + return { + reportActionSelectedEvent, + reportActionSelectedResult, + reportErrorEvent, + reportInitialActionEvent, + reportViewErrorDetailsEvent, + reportRecoveredRunResult, + } +} diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts index 3ef8f5b3809..803bdf18f6a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -4,18 +4,33 @@ import head from 'lodash/head' import { useResumeRunFromRecoveryMutation, useStopRunMutation, + useUpdateErrorRecoveryPolicy, } from '@opentrons/react-api-client' import { useChainRunCommands } from '../../../resources/runs' import { RECOVERY_MAP } from '../constants' -import type { CreateCommand, LoadedLabware } from '@opentrons/shared-data' -import type { CommandData } from '@opentrons/api-client' +import type { + CreateCommand, + LoadedLabware, + MoveToCoordinatesCreateCommand, + AspirateInPlaceRunTimeCommand, + BlowoutInPlaceRunTimeCommand, + DispenseInPlaceRunTimeCommand, + DropTipInPlaceRunTimeCommand, + PrepareToAspirateRunTimeCommand, +} from '@opentrons/shared-data' +import type { + CommandData, + RecoveryPolicyRulesParams, +} from '@opentrons/api-client' import type { WellGroup } from '@opentrons/components' import type { FailedCommand } from '../types' import type { UseFailedLabwareUtilsResult } from './useFailedLabwareUtils' import type { UseRouteUpdateActionsResult } from './useRouteUpdateActions' import type { RecoveryToasts } from './useRecoveryToasts' +import type { UseRecoveryAnalyticsResult } from './useRecoveryAnalytics' +import type { CurrentRecoveryOptionUtils } from './useRecoveryRouting' interface UseRecoveryCommandsParams { runId: string @@ -23,6 +38,8 @@ interface UseRecoveryCommandsParams { failedLabwareUtils: UseFailedLabwareUtilsResult routeUpdateActions: UseRouteUpdateActionsResult recoveryToastUtils: RecoveryToasts + analytics: UseRecoveryAnalyticsResult + selectedRecoveryOption: CurrentRecoveryOptionUtils['selectedRecoveryOption'] } export interface UseRecoveryCommandsResult { /* A terminal recovery command that causes ER to exit as the run status becomes "running" */ @@ -40,6 +57,8 @@ export interface UseRecoveryCommandsResult { /* A non-terminal recovery command */ pickUpTips: () => Promise } + +// TODO(jh, 07-24-24): Create tighter abstractions for terminal vs. non-terminal commands. // Returns commands with a "fixit" intent. Commands may or may not terminate Error Recovery. See each command docstring for details. export function useRecoveryCommands({ runId, @@ -47,6 +66,8 @@ export function useRecoveryCommands({ failedLabwareUtils, routeUpdateActions, recoveryToastUtils, + analytics, + selectedRecoveryOption, }: UseRecoveryCommandsParams): UseRecoveryCommandsResult { const { proceedToRouteAndStep } = routeUpdateActions const { chainRunCommands } = useChainRunCommands(runId, failedCommand?.id) @@ -54,8 +75,52 @@ export function useRecoveryCommands({ mutateAsync: resumeRunFromRecovery, } = useResumeRunFromRecoveryMutation() const { stopRun } = useStopRunMutation() + const { updateErrorRecoveryPolicy } = useUpdateErrorRecoveryPolicy(runId) const { makeSuccessToast } = recoveryToastUtils + const buildRetryPrepMove = (): MoveToCoordinatesCreateCommand | null => { + type InPlaceCommand = + | AspirateInPlaceRunTimeCommand + | BlowoutInPlaceRunTimeCommand + | DispenseInPlaceRunTimeCommand + | DropTipInPlaceRunTimeCommand + | PrepareToAspirateRunTimeCommand + const IN_PLACE_COMMAND_TYPES = [ + 'aspirateInPlace', + 'dispenseInPlace', + 'blowOutInPlace', + 'dropTipInPlace', + 'prepareToAspirate', + ] as const + const isInPlace = ( + failedCommand: FailedCommand + ): failedCommand is InPlaceCommand => + IN_PLACE_COMMAND_TYPES.includes( + (failedCommand as InPlaceCommand).commandType + ) + return failedCommand != null + ? isInPlace(failedCommand) + ? failedCommand.error?.isDefined && + failedCommand.error?.errorType === 'overpressure' && + // Paranoia: this value comes from the wire and may be unevenly implemented + typeof failedCommand.error?.errorInfo?.retryLocation?.at(0) === + 'number' + ? { + commandType: 'moveToCoordinates', + intent: 'fixit', + params: { + pipetteId: failedCommand.params?.pipetteId, + coordinates: { + x: failedCommand.error.errorInfo.retryLocation[0], + y: failedCommand.error.errorInfo.retryLocation[1], + z: failedCommand.error.errorInfo.retryLocation[2], + }, + }, + } + : null + : null + : null + } const chainRunRecoveryCommands = React.useCallback( ( commands: CreateCommand[], @@ -63,6 +128,7 @@ export function useRecoveryCommands({ ): Promise => chainRunCommands(commands, continuePastFailure).catch(e => { console.warn(`Error executing "fixit" command: ${e}`) + analytics.reportActionSelectedResult(selectedRecoveryOption, 'failed') // the catch never occurs if continuePastCommandFailure is "true" void proceedToRouteAndStep(RECOVERY_MAP.ERROR_WHILE_RECOVERING.ROUTE) return Promise.reject(new Error(`Could not execute command: ${e}`)) @@ -72,10 +138,13 @@ export function useRecoveryCommands({ const retryFailedCommand = React.useCallback((): Promise => { const { commandType, params } = failedCommand as FailedCommand // Null case is handled before command could be issued. - - return chainRunRecoveryCommands([ - { commandType, params }, - ] as CreateCommand[]) // the created command is the same command that failed + return chainRunRecoveryCommands( + [ + // move back to the location of the command if it is an in-place command + buildRetryPrepMove(), + { commandType, params }, // retry the command that failed + ].filter(c => c != null) as CreateCommand[] + ) // the created command is the same command that failed }, [chainRunRecoveryCommands, failedCommand]) // Homes the Z-axis of all attached pipettes. @@ -84,7 +153,6 @@ export function useRecoveryCommands({ }, [chainRunRecoveryCommands]) // Pick up the user-selected tips - // TODO(jh, 06-14-24): Do not ignore pickUpTip errors once Pipettes can support tip pick up. const pickUpTips = React.useCallback((): Promise => { const { selectedTipLocations, failedLabware } = failedLabwareUtils @@ -97,30 +165,44 @@ export function useRecoveryCommands({ if (pickUpTipCmd == null) { return Promise.reject(new Error('Invalid use of pickUpTips command')) } else { - return chainRunRecoveryCommands([pickUpTipCmd], true) + return chainRunRecoveryCommands([pickUpTipCmd]) } }, [chainRunRecoveryCommands, failedCommand, failedLabwareUtils]) const resumeRun = React.useCallback((): void => { void resumeRunFromRecovery(runId).then(() => { + analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') makeSuccessToast() }) }, [runId, resumeRunFromRecovery, makeSuccessToast]) const cancelRun = React.useCallback((): void => { + analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') stopRun(runId) }, [runId]) const skipFailedCommand = React.useCallback((): void => { void resumeRunFromRecovery(runId).then(() => { + analytics.reportActionSelectedResult(selectedRecoveryOption, 'succeeded') makeSuccessToast() }) }, [runId, resumeRunFromRecovery, makeSuccessToast]) const ignoreErrorKindThisRun = React.useCallback((): Promise => { - console.log('IGNORING ALL ERRORS OF THIS KIND THIS RUN') - return Promise.resolve() - }, []) + if (failedCommand?.error != null) { + const ignorePolicyRules = buildIgnorePolicyRules( + failedCommand.commandType, + failedCommand.error.errorType + ) + + updateErrorRecoveryPolicy(ignorePolicyRules) + return Promise.resolve() + } else { + return Promise.reject( + new Error('Could not execute command. No failed command.') + ) + } + }, [failedCommand?.error?.errorType, failedCommand?.commandType]) return { resumeRun, @@ -164,3 +246,16 @@ export const buildPickUpTips = ( } } } + +export const buildIgnorePolicyRules = ( + commandType: FailedCommand['commandType'], + errorType: string +): RecoveryPolicyRulesParams => { + return [ + { + commandType, + errorType, + ifMatch: 'ignoreAndContinue', + }, + ] +} diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts new file mode 100644 index 00000000000..3008a9066df --- /dev/null +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryTakeover.ts @@ -0,0 +1,99 @@ +import * as React from 'react' +import { useSelector } from 'react-redux' + +import { getUserId } from '../../../redux/config' +import { + useClientDataRecovery, + useUpdateClientDataRecovery, +} from '../../../resources/client_data' + +import type { ClientDataRecovery } from '../../../resources/client_data' +import type { UseERWizardResult } from '../ErrorRecoveryWizard' + +const CLIENT_DATA_INTERVAL_MS = 5000 + +export interface UseRecoveryTakeoverResult { + /* Whether to show the takeover modal. */ + showTakeover: boolean + /* Whether the user is permitted to enter *any* ER flows, including the cancel flow. */ + isActiveUser: boolean + /* Wraps toggleERWiz with the ability to claim immediate active user status and inform the network of the client's active user status. */ + toggleERWizAsActiveUser: ( + isActive: boolean, + launchER: boolean + ) => Promise + /* Indicates the active user's recovery intent. */ + intent: ClientDataRecovery['intent'] +} + +/** + * A client is the active user when actively engaging in Error Recovery. A client claims the active user status via a CTA + * (ex, clicking an option from the splash screen), and then informs other clients of their active user status. + * + * An active user may be made inactive through three methods: + * 1) If a different client revokes the active user status (ex "terminate remote activity"). + * In this instance, the client will not be the active user, but may *not* see a takeover modal if a different user has + * yet to become the active user. + * 2) The client yields their active status by returning to the splash page (the client is not actively using error recovery). + * 3) Completing a recovery flow. + */ +export function useRecoveryTakeover( + toggleERWiz: UseERWizardResult['toggleERWizard'] +): UseRecoveryTakeoverResult { + const [isActiveUser, setIsActiveUser] = React.useState(false) + + const thisUserId = useSelector(getUserId) + const { userId: activeId, intent } = useClientDataRecovery({ + refetchInterval: CLIENT_DATA_INTERVAL_MS, + }) + const { updateWithIntent, clearClientData } = useUpdateClientDataRecovery() + + // Update the client's active user status implicitly if revoked by a different client. + React.useEffect(() => { + if (isActiveUser && activeId !== thisUserId) { + setIsActiveUser(false) + } + }, [activeId]) // Not all dependencies added for intended behavior! + + // If Error Recovery unrenders and this client is the active user, revoke the client's active user status. + React.useEffect(() => { + return () => { + if (isActiveUser) { + clearClientData() + } + } + }, [isActiveUser]) + + const showTakeover = !(activeId == null || thisUserId === activeId) + + const toggleERWizAsActiveUser = ( + isActive: boolean, + launchER: boolean + ): Promise => { + const newIsActiveUser = !isActiveUser + setIsActiveUser(newIsActiveUser) + + if (newIsActiveUser) { + const intent: ClientDataRecovery['intent'] = launchER + ? 'recovering' + : 'canceling' + + updateWithIntent(intent) + } + // If the client is in a takeover and then presses "go back" enough to get back to the splash, revoke the client's active status. + else if (isActiveUser && !newIsActiveUser) { + void toggleERWiz(false).then(() => { + clearClientData() + }) + } + + return toggleERWiz(isActive, launchER) + } + + return { + showTakeover, + intent, + toggleERWizAsActiveUser, + isActiveUser, + } +} diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts index 44489f190e8..74c8186bf79 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRouteUpdateActions.ts @@ -10,10 +10,11 @@ import type { RobotMovingRoute, RouteStep, } from '../types' +import type { UseRecoveryTakeoverResult } from './useRecoveryTakeover' export interface GetRouteUpdateActionsParams { hasLaunchedRecovery: boolean - toggleERWizard: (launchER: boolean) => Promise + toggleERWizAsActiveUser: UseRecoveryTakeoverResult['toggleERWizAsActiveUser'] recoveryMap: IRecoveryMap setRecoveryMap: (recoveryMap: IRecoveryMap) => void } @@ -165,7 +166,7 @@ export function getRecoveryRouteNavigation( return { getNextStep, getPrevStep } } -interface DetermineRecoveryRoutingParams extends GetRouteUpdateActionsParams { +type DetermineRecoveryRoutingParams = GetRouteUpdateActionsParams & { updatedStep: string currentRoute: RecoveryRoute } @@ -175,7 +176,7 @@ interface DetermineRecoveryRoutingParams extends GetRouteUpdateActionsParams { // is the fallback route as opposed to SelectRecoveryOption (ex, accessed by pressing "go back" enough times). function determineRecoveryRouting({ hasLaunchedRecovery, - toggleERWizard, + toggleERWizAsActiveUser, setRecoveryMap, updatedStep, currentRoute, @@ -189,7 +190,7 @@ function determineRecoveryRouting({ }) if (!hasLaunchedRecovery) { - void toggleERWizard(false) + void toggleERWizAsActiveUser(false, false) } } else { setRecoveryMap({ route: currentRoute, step: updatedStep }) diff --git a/app/src/organisms/ErrorRecoveryFlows/index.tsx b/app/src/organisms/ErrorRecoveryFlows/index.tsx index 6e4e2bf1fd3..76b5d6b07bc 100644 --- a/app/src/organisms/ErrorRecoveryFlows/index.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/index.tsx @@ -14,13 +14,17 @@ import { RUN_STATUS_SUCCEEDED, } from '@opentrons/api-client' import { OT2_ROBOT_TYPE } from '@opentrons/shared-data' +import { useHost } from '@opentrons/react-api-client' import { getIsOnDevice } from '../../redux/config' import { ErrorRecoveryWizard, useERWizard } from './ErrorRecoveryWizard' import { RunPausedSplash, useRunPausedSplash } from './RunPausedSplash' +import { RecoveryTakeover } from './RecoveryTakeover' import { useCurrentlyRecoveringFrom, useERUtils, + useRecoveryAnalytics, + useRecoveryTakeover, useShowDoorInfo, } from './hooks' @@ -106,40 +110,61 @@ export interface ErrorRecoveryFlowsProps { runId: string runStatus: RunStatus | null failedCommand: FailedCommand | null - isFlex: boolean protocolAnalysis: CompletedProtocolAnalysis | null } export function ErrorRecoveryFlows( props: ErrorRecoveryFlowsProps ): JSX.Element | null { - const { protocolAnalysis, runStatus } = props + const { protocolAnalysis, runStatus, failedCommand } = props - const { hasLaunchedRecovery, toggleERWizard, showERWizard } = useERWizard() + const analytics = useRecoveryAnalytics() + React.useEffect(() => { + analytics.reportErrorEvent(failedCommand) + }, [failedCommand?.error?.detail]) + const { hasLaunchedRecovery, toggleERWizard, showERWizard } = useERWizard() const isOnDevice = useSelector(getIsOnDevice) const robotType = protocolAnalysis?.robotType ?? OT2_ROBOT_TYPE - const showSplash = useRunPausedSplash(isOnDevice, showERWizard) + const robotName = useHost()?.robotName ?? 'robot' const isDoorOpen = useShowDoorInfo(runStatus) + const { + showTakeover, + isActiveUser, + intent, + toggleERWizAsActiveUser, + } = useRecoveryTakeover(toggleERWizard) + const renderWizard = isActiveUser && (showERWizard || isDoorOpen) + const showSplash = useRunPausedSplash(isOnDevice, renderWizard) const recoveryUtils = useERUtils({ ...props, hasLaunchedRecovery, - toggleERWizard, + toggleERWizAsActiveUser, isOnDevice, robotType, + analytics, }) return ( <> - {showERWizard || isDoorOpen ? ( + {showTakeover ? ( + + ) : null} + {renderWizard ? ( ) : null} {showSplash ? ( @@ -147,8 +172,10 @@ export function ErrorRecoveryFlows( {...props} {...recoveryUtils} robotType={robotType} + robotName={robotName} isOnDevice={isOnDevice} - toggleERWiz={toggleERWizard} + toggleERWizAsActiveUser={toggleERWizAsActiveUser} + analytics={analytics} /> ) : null} diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx index de4829d937f..f1921b83d02 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/ErrorDetailsModal.tsx @@ -1,9 +1,11 @@ import * as React from 'react' import { useTranslation } from 'react-i18next' import { createPortal } from 'react-dom' +import { css } from 'styled-components' import { Flex, + StyledText, SPACING, COLORS, BORDERS, @@ -12,16 +14,22 @@ import { import { useErrorName } from '../hooks' import { Modal } from '../../../molecules/Modal' -import { getTopPortalEl } from '../../../App/portal' +import { getModalPortalEl, getTopPortalEl } from '../../../App/portal' import { ERROR_KINDS } from '../constants' import { InlineNotification } from '../../../atoms/InlineNotification' import { StepInfo } from './StepInfo' import { getErrorKind } from '../utils' +import { + LegacyModalShell, + LegacyModalHeader, +} from '../../../molecules/LegacyModal' import type { RobotType } from '@opentrons/shared-data' +import type { IconProps } from '@opentrons/components' import type { ModalHeaderBaseProps } from '../../../molecules/Modal/types' import type { ERUtilsResults } from '../hooks' import type { ErrorRecoveryFlowsProps } from '..' +import type { DesktopSizeType } from '../types' export function useErrorDetailsModal(): { showModal: boolean @@ -41,6 +49,7 @@ type ErrorDetailsModalProps = ErrorRecoveryFlowsProps & toggleModal: () => void isOnDevice: boolean robotType: RobotType + desktopType: DesktopSizeType } export function ErrorDetailsModal(props: ErrorDetailsModalProps): JSX.Element { @@ -64,7 +73,101 @@ export function ErrorDetailsModal(props: ErrorDetailsModalProps): JSX.Element { hasExitIcon: true, } - return createPortal( + const buildModal = (): JSX.Element => { + if (isOnDevice) { + return createPortal( + + {getIsOverpressureErrorKind() ? : null} + , + getTopPortalEl() + ) + } else { + return createPortal( + + {getIsOverpressureErrorKind() ? : null} + , + getModalPortalEl() + ) + } + } + + return buildModal() +} + +type ErrorDetailsModalType = ErrorDetailsModalProps & { + children: React.ReactNode + modalHeader: ModalHeaderBaseProps + toggleModal: () => void + desktopType: DesktopSizeType +} + +export function ErrorDetailsModalDesktop( + props: ErrorDetailsModalType +): JSX.Element { + const { children, modalHeader, toggleModal, desktopType } = props + const { t } = useTranslation('error_recovery') + + const buildIcon = (): IconProps => { + return { + name: 'information', + color: COLORS.grey60, + size: SPACING.spacing20, + marginRight: SPACING.spacing8, + } + } + + const buildHeader = (): JSX.Element => { + return ( + + ) + } + + return ( + + + + {modalHeader.title} + + {children} + + + + + + ) +} + +export function ErrorDetailsModalODD( + props: ErrorDetailsModalType +): JSX.Element { + const { children, modalHeader, toggleModal } = props + + return ( - {getIsOverpressureErrorKind() ? ( - - ) : null} + {children} - + - , - getTopPortalEl() + ) } -export function OverpressureBanner(props: { - isOnDevice: boolean -}): JSX.Element | null { +export function OverpressureBanner(): JSX.Element | null { const { t } = useTranslation('error_recovery') - if (props.isOnDevice) { - return ( - - ) - } else { - return null - } + return ( + + ) } + +// TODO(jh, 07-24-24): Using shared height/width constants for intervention modal sizing and the ErrorDetailsModal sizing +// would be ideal. +const DESKTOP_STEP_INFO_STYLE = css` + background-color: ${COLORS.grey30}; + grid-gap: ${SPACING.spacing10}; + border-radius: ${BORDERS.borderRadius4}; + padding: ${SPACING.spacing6} ${SPACING.spacing24} ${SPACING.spacing6} + ${SPACING.spacing12}; +` + +const DESKTOP_MODAL_STYLE_BASE = css` + width: 47rem; +` + +const DESKTOP_MODAL_STYLE_SMALL = css` + ${DESKTOP_MODAL_STYLE_BASE} + height: 26rem; +` +const DESKTOP_MODAL_STYLE_LARGE = css` + ${DESKTOP_MODAL_STYLE_BASE} + height: 31rem; +` diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx index ea78376da4e..c9042cfc55f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryFooterButtons.tsx @@ -20,16 +20,19 @@ import { SmallButton, TextOnlyButton } from '../../../atoms/buttons' interface RecoveryFooterButtonProps { primaryBtnOnClick: () => void - /* The "Go back" button */ - secondaryBtnOnClick?: () => void primaryBtnTextOverride?: string primaryBtnDisabled?: boolean /* If true, render pressed state and a spinner icon for the primary button. */ isLoadingPrimaryBtnAction?: boolean + /* Typically the "Go back" button */ + secondaryBtnOnClick?: () => void + secondaryBtnTextOverride?: string /* To the left of the primary button. */ tertiaryBtnOnClick?: () => void tertiaryBtnText?: string tertiaryBtnDisabled?: boolean + /* Use the style of the secondary button in the position typically used by the tertiary button. */ + secondaryAsTertiary?: boolean } export function RecoveryFooterButtons( props: RecoveryFooterButtonProps @@ -42,20 +45,24 @@ export function RecoveryFooterButtons( alignItems={ALIGN_FLEX_END} gridGap={SPACING.spacing8} > - + {!props.secondaryAsTertiary && }
) } function RecoveryGoBackButton({ + secondaryBtnTextOverride, secondaryBtnOnClick, }: RecoveryFooterButtonProps): JSX.Element | null { const showGoBackBtn = secondaryBtnOnClick != null const { t } = useTranslation('error_recovery') return showGoBackBtn ? ( - + ) : ( @@ -63,10 +70,17 @@ function RecoveryGoBackButton({ } function PrimaryButtonGroup(props: RecoveryFooterButtonProps): JSX.Element { - const { tertiaryBtnOnClick, tertiaryBtnText } = props + const { + tertiaryBtnOnClick, + tertiaryBtnText, + secondaryAsTertiary, + secondaryBtnOnClick, + } = props const renderTertiaryBtn = - tertiaryBtnOnClick != null || tertiaryBtnText != null + tertiaryBtnOnClick != null || + tertiaryBtnText != null || + (secondaryBtnOnClick != null && secondaryAsTertiary) if (!renderTertiaryBtn) { return ( @@ -76,8 +90,15 @@ function PrimaryButtonGroup(props: RecoveryFooterButtonProps): JSX.Element { ) } else { return ( - - + + {secondaryAsTertiary ? ( + + ) : ( + + )} ) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx index 00a853ee99a..e044d46054f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx @@ -8,13 +8,14 @@ import { InterventionModal } from '../../../molecules/InterventionModal' import { getModalPortalEl, getTopPortalEl } from '../../../App/portal' import type { ModalType } from '../../../molecules/InterventionModal' +import type { DesktopSizeType } from '../types' export type RecoveryInterventionModalProps = Omit< React.ComponentProps, 'type' > & { /* If on desktop, specifies the hard-coded dimensions height of the modal. */ - desktopType: 'desktop-small' | 'desktop-large' + desktopType: DesktopSizeType isOnDevice: boolean } diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx index 0b6f66aa484..d4012670c27 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/SelectTips.tsx @@ -12,7 +12,12 @@ import { TipSelection } from './TipSelection' import type { RecoveryContentProps } from '../types' export function SelectTips(props: RecoveryContentProps): JSX.Element | null { - const { failedPipetteInfo, routeUpdateActions, recoveryCommands } = props + const { + failedPipetteInfo, + routeUpdateActions, + recoveryCommands, + isOnDevice, + } = props const { ROBOT_PICKING_UP_TIPS } = RECOVERY_MAP const { pickUpTips } = recoveryCommands const { @@ -33,6 +38,22 @@ export function SelectTips(props: RecoveryContentProps): JSX.Element | null { setShowTipSelectModal(!showTipSelectModal) } + const buildTertiaryBtnProps = (): { + tertiaryBtnDisabled?: boolean + tertiaryBtnOnClick?: () => void + tertiaryBtnText?: string + } => { + if (isOnDevice) { + return { + tertiaryBtnDisabled: failedPipetteInfo?.data.channels === 96, + tertiaryBtnOnClick: toggleModal, + tertiaryBtnText: t('change_location'), + } + } else { + return {} + } + } + return ( <> {showTipSelectModal && ( @@ -50,15 +71,13 @@ export function SelectTips(props: RecoveryContentProps): JSX.Element | null { type="location" bannerText={t('replace_tips_and_select_location')} /> - + diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx index 54fb2464124..13cb6f3a702 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/StepInfo.tsx @@ -2,7 +2,7 @@ import * as React from 'react' import { useTranslation } from 'react-i18next' -import { Flex, DISPLAY_INLINE, LegacyStyledText } from '@opentrons/components' +import { Flex, DISPLAY_INLINE, StyledText } from '@opentrons/components' import { CommandText } from '../../../molecules/Command' @@ -10,15 +10,17 @@ import type { StyleProps } from '@opentrons/components' import type { RecoveryContentProps } from '../types' interface StepInfoProps extends StyleProps { - textStyle: React.ComponentProps['as'] stepCounts: RecoveryContentProps['stepCounts'] failedCommand: RecoveryContentProps['failedCommand'] robotType: RecoveryContentProps['robotType'] protocolAnalysis: RecoveryContentProps['protocolAnalysis'] + desktopStyle?: React.ComponentProps['desktopStyle'] + oddStyle?: React.ComponentProps['oddStyle'] } export function StepInfo({ - textStyle, + desktopStyle, + oddStyle, stepCounts, failedCommand, robotType, @@ -35,18 +37,27 @@ export function StepInfo({ const currentCopy = currentStepNumber ?? '?' const totalCopy = totalStepCount ?? '?' + const desktopStyleDefaulted = desktopStyle ?? 'bodyDefaultRegular' + const oddStyleDefaulted = oddStyle ?? 'bodyTextRegular' + return ( - + {`${t('at_step')} ${currentCopy}/${totalCopy}: `} - +
{analysisCommand != null && protocolAnalysis != null ? ( ) : null} diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx index 3eb590f1a35..b63464b4382 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/ErrorDetailsModal.test.tsx @@ -44,27 +44,6 @@ describe('useErrorDetailsModal', () => { }) }) -describe('ErrorDetailsModal', () => { - let props: React.ComponentProps - - beforeEach(() => { - props = { - ...mockRecoveryContentProps, - toggleModal: vi.fn(), - robotType: 'OT-3 Standard', - } - - vi.mocked(StepInfo).mockReturnValue(
MOCK_STEP_INFO
) - }) - - it('renders ErrorDetailsModal', () => { - renderWithProviders(, { - i18nInstance: i18n, - }) - expect(screen.getByText('MOCK_STEP_INFO')).toBeInTheDocument() - }) -}) - const render = (props: React.ComponentProps) => { return renderWithProviders(, { i18nInstance: i18n, @@ -79,6 +58,7 @@ describe('ErrorDetailsModal', () => { ...mockRecoveryContentProps, toggleModal: vi.fn(), robotType: 'OT-3 Standard', + desktopType: 'desktop-small', } vi.mocked(StepInfo).mockReturnValue(
MOCK_STEP_INFO
) @@ -87,7 +67,9 @@ describe('ErrorDetailsModal', () => { ) }) - it('renders the modal with the correct content', () => { + const IS_ODD = [true, false] + + it('renders the ODD modal with the correct content', () => { render(props) expect(vi.mocked(Modal)).toHaveBeenCalledWith( expect.objectContaining({ @@ -102,21 +84,30 @@ describe('ErrorDetailsModal', () => { expect(screen.getByText('MOCK_STEP_INFO')).toBeInTheDocument() }) - it('renders the OverpressureBanner when the error kind is an overpressure error', () => { - props.failedCommand = { - ...props.failedCommand, - commandType: 'aspirate', - error: { isDefined: true, errorType: 'overpressure' }, - } as any - render(props) + it('renders the desktop modal with the correct content', () => { + render({ ...props, isOnDevice: false }) - screen.getByText('MOCK_INLINE_NOTIFICATION') + screen.getByText('MOCK_STEP_INFO') + screen.getByText('Error details') }) - it('does not render the OverpressureBanner when the error kind is not an overpressure error', () => { - render(props) + IS_ODD.forEach(isOnDevice => { + it('renders the OverpressureBanner when the error kind is an overpressure error', () => { + props.failedCommand = { + ...props.failedCommand, + commandType: 'aspirate', + error: { isDefined: true, errorType: 'overpressure' }, + } as any + render({ ...props, isOnDevice }) + + screen.getByText('MOCK_INLINE_NOTIFICATION') + }) + + it('does not render the OverpressureBanner when the error kind is not an overpressure error', () => { + render({ ...props, isOnDevice }) - expect(screen.queryByText('MOCK_INLINE_NOTIFICATION')).toBeNull() + expect(screen.queryByText('MOCK_INLINE_NOTIFICATION')).toBeNull() + }) }) }) @@ -128,7 +119,7 @@ describe('OverpressureBanner', () => { }) it('renders the InlineNotification', () => { - renderWithProviders(, { + renderWithProviders(, { i18nInstance: i18n, }) expect(vi.mocked(InlineNotification)).toHaveBeenCalledWith( diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx index b4e2b260715..3e4e9045c1a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/RecoveryFooterButtons.test.tsx @@ -139,4 +139,59 @@ describe('RecoveryFooterButtons', () => { expect(btn).toBeDisabled() }) }) + + it('renders the secondary button as tertiary when secondaryAsTertiary is true', () => { + props = { + ...props, + secondaryAsTertiary: true, + secondaryBtnOnClick: mockSecondaryBtnOnClick, + } + render(props) + + const secondaryBtn = screen.getAllByRole('button', { name: 'Go back' }) + expect(secondaryBtn.length).toBe(1) + + secondaryBtn.forEach(btn => { + mockSecondaryBtnOnClick.mockReset() + fireEvent.click(btn) + expect(mockSecondaryBtnOnClick).toHaveBeenCalled() + }) + }) + + it('renders secondary button with custom text when secondaryBtnTextOverride is provided', () => { + props = { + ...props, + secondaryBtnTextOverride: 'Custom Back', + } + render(props) + + const secondaryBtns = screen.getAllByRole('button', { name: 'Custom Back' }) + expect(secondaryBtns.length).toBe(1) + + secondaryBtns.forEach(btn => { + mockSecondaryBtnOnClick.mockReset() + fireEvent.click(btn) + expect(mockSecondaryBtnOnClick).toHaveBeenCalled() + }) + }) + + it('renders secondary button as tertiary with custom text', () => { + props = { + ...props, + secondaryAsTertiary: true, + secondaryBtnTextOverride: 'Custom Tertiary', + } + render(props) + + const secondaryBtns = screen.getAllByRole('button', { + name: 'Custom Tertiary', + }) + expect(secondaryBtns.length).toBe(1) + + secondaryBtns.forEach(btn => { + mockSecondaryBtnOnClick.mockReset() + fireEvent.click(btn) + expect(mockSecondaryBtnOnClick).toHaveBeenCalled() + }) + }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx index 4e7e8b393fa..9396fcf8f7d 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/__tests__/StepInfo.test.tsx @@ -25,7 +25,7 @@ describe('StepInfo', () => { ...mockRecoveryContentProps, protocolAnalysis: { commands: [mockFailedCommand] } as any, }, - textStyle: 'h4', + desktopStyle: 'bodyDefaultRegular', stepCounts: { currentStepNumber: 5, totalStepCount: 10, diff --git a/app/src/organisms/ErrorRecoveryFlows/types.ts b/app/src/organisms/ErrorRecoveryFlows/types.ts index c1f0ea49329..747000f2dbb 100644 --- a/app/src/organisms/ErrorRecoveryFlows/types.ts +++ b/app/src/organisms/ErrorRecoveryFlows/types.ts @@ -63,3 +63,5 @@ export type RecoveryContentProps = ErrorRecoveryWizardProps & { errorKind: ErrorKind isOnDevice: boolean } + +export type DesktopSizeType = 'desktop-small' | 'desktop-large' diff --git a/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts index c3406ee6391..adad317fd2a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/utils/__tests__/getErrorKind.test.ts @@ -55,10 +55,10 @@ describe('getErrorKind', () => { it(`returns ${ERROR_KINDS.GENERAL_ERROR} for defined errors not handled explicitly`, () => { const result = getErrorKind({ commandType: 'aspirate', - error: { + error: ({ isDefined: true, errorType: 'someHithertoUnknownDefinedErrorType', - } as RunCommandError, + } as unknown) as RunCommandError, } as RunTimeCommand) expect(result).toEqual(ERROR_KINDS.GENERAL_ERROR) }) diff --git a/app/src/organisms/LabwareDetails/Gallery.tsx b/app/src/organisms/LabwareDetails/Gallery.tsx index 3444bc149dc..6cc4e10c85e 100644 --- a/app/src/organisms/LabwareDetails/Gallery.tsx +++ b/app/src/organisms/LabwareDetails/Gallery.tsx @@ -8,8 +8,8 @@ import { JUSTIFY_SPACE_EVENLY, LabwareRender, RobotWorkSpace, - SPACING, SPACING_AUTO, + SPACING, } from '@opentrons/components' import { labwareImages } from './labware-images' @@ -51,10 +51,10 @@ export function Gallery(props: GalleryProps): JSX.Element { const images = staticImages != null ? [render, ...staticImages] : [render] return ( - + diff --git a/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx b/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx index 28744a218cf..872927bf170 100644 --- a/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx +++ b/app/src/organisms/ModuleCard/ModuleOverflowMenu.tsx @@ -12,7 +12,7 @@ import { import { MenuList } from '../../atoms/MenuList' import { Tooltip } from '../../atoms/Tooltip' import { MenuItem } from '../../atoms/MenuList/MenuItem' -import { useCurrentRunId } from '../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../resources/runs' import { useIsFlex, useRunStatuses, diff --git a/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx b/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx index 57c51f4d2f9..e78b0a2d0e7 100644 --- a/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/ModuleOverflowMenu.test.tsx @@ -16,14 +16,14 @@ import { useIsLegacySessionInProgress, useIsFlex, } from '../../Devices/hooks' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { ModuleOverflowMenu } from '../ModuleOverflowMenu' import type { TemperatureStatus } from '@opentrons/api-client' vi.mock('../../Devices/hooks') vi.mock('../../RunTimeControl/hooks') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') const render = (props: React.ComponentProps) => { return renderWithProviders(, { diff --git a/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx b/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx index 854b41b24a5..ffeb8eb0e84 100644 --- a/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx +++ b/app/src/organisms/ModuleCard/__tests__/hooks.test.tsx @@ -19,7 +19,7 @@ import { } from '../../../redux/modules/__fixtures__' import { useIsRobotBusy, useRunStatuses } from '../../Devices/hooks' import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { useCurrentRunId } from '../../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { useLatchControls, useModuleOverflowMenu, @@ -31,7 +31,7 @@ import type { State } from '../../../redux/types' vi.mock('@opentrons/react-api-client') vi.mock('../../LabwarePositionCheck/useMostRecentCompletedAnalysis') -vi.mock('../../ProtocolUpload/hooks') +vi.mock('../../../resources/runs') vi.mock('../../Devices/hooks') const mockCloseLatchHeaterShaker = { diff --git a/app/src/organisms/ModuleCard/hooks.tsx b/app/src/organisms/ModuleCard/hooks.tsx index 5f515b75a8d..88f5ae69a02 100644 --- a/app/src/organisms/ModuleCard/hooks.tsx +++ b/app/src/organisms/ModuleCard/hooks.tsx @@ -11,7 +11,7 @@ import { import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { MenuItem } from '../../atoms/MenuList/MenuItem' import { Tooltip } from '../../atoms/Tooltip' -import { useCurrentRunId } from '../ProtocolUpload/hooks' +import { useCurrentRunId } from '../../resources/runs' import type { HeaterShakerCloseLatchCreateCommand, diff --git a/app/src/organisms/OnDeviceDisplay/RunningProtocol/ConfirmCancelRunModal.tsx b/app/src/organisms/OnDeviceDisplay/RunningProtocol/ConfirmCancelRunModal.tsx index 76efff81b8b..6a674b6835b 100644 --- a/app/src/organisms/OnDeviceDisplay/RunningProtocol/ConfirmCancelRunModal.tsx +++ b/app/src/organisms/OnDeviceDisplay/RunningProtocol/ConfirmCancelRunModal.tsx @@ -14,6 +14,7 @@ import { } from '@opentrons/components' import { useStopRunMutation, + useDeleteRunMutation, useDismissCurrentRunMutation, } from '@opentrons/react-api-client' @@ -31,6 +32,7 @@ interface ConfirmCancelRunModalProps { runId: string setShowConfirmCancelRunModal: (showConfirmCancelRunModal: boolean) => void isActiveRun: boolean + isQuickTransfer: boolean protocolId?: string | null } @@ -38,14 +40,27 @@ export function ConfirmCancelRunModal({ runId, setShowConfirmCancelRunModal, isActiveRun, + isQuickTransfer, protocolId, }: ConfirmCancelRunModalProps): JSX.Element { const { t } = useTranslation(['run_details', 'shared']) const { stopRun } = useStopRunMutation() + const { deleteRun } = useDeleteRunMutation({ + onError: error => { + setIsCanceling(false) + console.error('Error deleting quick transfer run', error) + }, + }) const { dismissCurrentRun, isLoading: isDismissing, - } = useDismissCurrentRunMutation() + } = useDismissCurrentRunMutation({ + onSuccess: () => { + if (isQuickTransfer && !isActiveRun) { + deleteRun(runId) + } + }, + }) const runStatus = useRunStatus(runId) const localRobot = useSelector(getLocalRobot) const robotName = localRobot?.name ?? '' @@ -74,7 +89,11 @@ export function ConfirmCancelRunModal({ trackProtocolRunEvent({ name: ANALYTICS_PROTOCOL_RUN_ACTION.CANCEL }) dismissCurrentRun(runId) if (!isActiveRun) { - if (protocolId != null) { + if (isQuickTransfer && protocolId != null) { + navigate(`/quick-transfer/${protocolId}`) + } else if (isQuickTransfer) { + navigate('/quick-transfer') + } else if (protocolId != null) { navigate(`/protocols/${protocolId}`) } else { navigate('/protocols') diff --git a/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx b/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx index 9ae25e466f4..358436283aa 100644 --- a/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx +++ b/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/ConfirmCancelRunModal.test.tsx @@ -7,6 +7,7 @@ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { RUN_STATUS_IDLE, RUN_STATUS_STOPPED } from '@opentrons/api-client' import { useStopRunMutation, + useDeleteRunMutation, useDismissCurrentRunMutation, } from '@opentrons/react-api-client' @@ -31,6 +32,7 @@ vi.mock('../CancelingRunModal') vi.mock('../../../../redux/discovery') const mockNavigate = vi.fn() const mockStopRun = vi.fn() +const mockDeleteRun = vi.fn() const mockDismissCurrentRun = vi.fn() const mockTrackEvent = vi.fn() const mockTrackProtocolRunEvent = vi.fn( @@ -69,11 +71,15 @@ describe('ConfirmCancelRunModal', () => { isActiveRun: true, runId: RUN_ID, setShowConfirmCancelRunModal: mockFn, + isQuickTransfer: false, } vi.mocked(useStopRunMutation).mockReturnValue({ stopRun: mockStopRun, } as any) + vi.mocked(useDeleteRunMutation).mockReturnValue({ + deleteRun: mockDeleteRun, + } as any) vi.mocked(useDismissCurrentRunMutation).mockReturnValue({ dismissCurrentRun: mockDismissCurrentRun, isLoading: false, @@ -152,4 +158,16 @@ describe('ConfirmCancelRunModal', () => { expect(mockTrackProtocolRunEvent).toHaveBeenCalled() expect(mockNavigate).toHaveBeenCalledWith('/protocols') }) + it('when quick transfer run is stopped, the run is dismissed and you return to quick transfer', () => { + props = { + ...props, + isActiveRun: false, + isQuickTransfer: true, + } + when(useRunStatus).calledWith(RUN_ID).thenReturn(RUN_STATUS_STOPPED) + render(props) + + expect(mockDismissCurrentRun).toHaveBeenCalled() + expect(mockNavigate).toHaveBeenCalledWith('/quick-transfer') + }) }) diff --git a/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/RunFailedModal.test.tsx b/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/RunFailedModal.test.tsx index b57a09cc8aa..629c0b1adea 100644 --- a/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/RunFailedModal.test.tsx +++ b/app/src/organisms/OnDeviceDisplay/RunningProtocol/__tests__/RunFailedModal.test.tsx @@ -20,39 +20,39 @@ const mockErrors = [ { id: 'd0245210-dfb9-4f1c-8ad0-3416b603a7ba', errorType: 'generalError', - isDefined: false, + isDefined: false as const, createdAt: '2023-04-09T21:41:51.333171+00:00', detail: 'Error with code 4000 (lowest priority)', errorInfo: {}, - errorCode: '4000', + errorCode: '4000' as const, wrappedErrors: [ { id: 'd0245210-dfb9-4f1c-8ad0-3416b603a7ba', errorType: 'roboticsInteractionError', - isDefined: false, + isDefined: false as const, createdAt: '2023-04-09T21:41:51.333171+00:00', detail: 'Error with code 3000 (second lowest priortiy)', errorInfo: {}, - errorCode: '3000', + errorCode: '3000' as const, wrappedErrors: [], }, { id: 'd0245210-dfb9-4f1c-8ad0-3416b603a7ba', errorType: 'roboticsControlError', - isDefined: false, + isDefined: false as const, createdAt: '2023-04-09T21:41:51.333171+00:00', detail: 'Error with code 2000 (second highest priority)', errorInfo: {}, - errorCode: '2000', + errorCode: '2000' as const, wrappedErrors: [ { id: 'd0245210-dfb9-4f1c-8ad0-3416b603a7ba', errorType: 'hardwareCommunicationError', - isDefined: false, + isDefined: false as const, createdAt: '2023-04-09T21:41:51.333171+00:00', detail: 'Error with code 1000 (highest priority)', errorInfo: {}, - errorCode: '1000', + errorCode: '1000' as const, wrappedErrors: [], }, ], @@ -62,11 +62,11 @@ const mockErrors = [ { id: 'd0245210-dfb9-4f1c-8ad0-3416b603a7ba', errorType: 'roboticsInteractionError', - isDefined: false, + isDefined: false as const, createdAt: '2023-04-09T21:41:51.333171+00:00', detail: 'Error with code 2001 (second highest priortiy)', errorInfo: {}, - errorCode: '2001', + errorCode: '2001' as const, wrappedErrors: [], }, ] diff --git a/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx b/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx index e663d4dc9b9..e985386d461 100644 --- a/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx +++ b/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx @@ -83,6 +83,7 @@ export function LabwareMapViewModal( handleLabwareClick(topLabwareDefinition, topLabwareId) } : undefined, + highlightLabware: true, moduleChildren: null, } }) @@ -105,6 +106,7 @@ export function LabwareMapViewModal( handleLabwareClick(topLabwareDefinition, topLabwareId) }, labwareChildren: null, + highlight: true, } } ) diff --git a/app/src/organisms/ProtocolSetupLabware/index.tsx b/app/src/organisms/ProtocolSetupLabware/index.tsx index b504ac528bd..9c962f55d7b 100644 --- a/app/src/organisms/ProtocolSetupLabware/index.tsx +++ b/app/src/organisms/ProtocolSetupLabware/index.tsx @@ -27,6 +27,7 @@ import { getDeckDefFromRobotType, getLabwareDefURI, getLabwareDisplayName, + getModuleDisplayName, HEATERSHAKER_MODULE_TYPE, } from '@opentrons/shared-data' import { parseInitialLoadedLabwareByAdapter } from '@opentrons/api-client' @@ -170,16 +171,7 @@ export function ProtocolSetupLabware({ module.moduleId === selectedLabware.location.moduleId ) if (matchedModule != null) { - location = ( - <> - - - - ) + location = } } else if ( selectedLabware != null && @@ -200,18 +192,7 @@ export function ProtocolSetupLabware({ module => module.moduleId === adapterLocation.moduleId ) if (moduleUnderAdapter != null) { - location = ( - <> - - - - ) + location = } } } @@ -488,7 +469,10 @@ function RowLabware({ commands, }: RowLabwareProps): JSX.Element | null { const { definition, initialLocation, nickName } = labware - const { t } = useTranslation('protocol_command_text') + const { t, i18n } = useTranslation([ + 'protocol_command_text', + 'protocol_setup', + ]) const matchedModule = initialLocation !== 'offDeck' && @@ -509,7 +493,9 @@ function RowLabware({ let slotName: string = '' let location: JSX.Element | string | null = null if (initialLocation === 'offDeck') { - location = t('off_deck') + location = ( + + ) } else if ('slotName' in initialLocation) { slotName = initialLocation.slotName location = @@ -521,7 +507,6 @@ function RowLabware({ location = ( <> - ) } else if ('labwareId' in initialLocation) { @@ -542,18 +527,7 @@ function RowLabware({ ) if (moduleUnderAdapter != null) { slotName = moduleUnderAdapter.slotName - location = ( - <> - - - - ) + location = } } } @@ -568,6 +542,9 @@ function RowLabware({ > {location} + {nestedLabwareInfo != null || matchedModule != null ? ( + + ) : null} - {nestedLabwareInfo != null ? ( - - ) : null} {nestedLabwareInfo != null && nestedLabwareInfo?.sharedSlotId === slotName ? ( - - + + + + {nestedLabwareInfo.nestedLabwareDisplayName} + + + {nestedLabwareInfo.nestedLabwareNickName} + + + + ) : null} + {matchedModule != null ? ( + <> + + - {nestedLabwareInfo.nestedLabwareDisplayName} - - - {nestedLabwareInfo.nestedLabwareNickName} - - + + + + {getModuleDisplayName(matchedModule.moduleDef.model)} + + {matchingHeaterShaker != null ? ( + + {t('protocol_setup:labware_latch_instructions')} + + ) : null} + + + ) : null} {matchingHeaterShaker != null ? ( diff --git a/app/src/organisms/ProtocolSetupParameters/ChooseCsvFile.tsx b/app/src/organisms/ProtocolSetupParameters/ChooseCsvFile.tsx index 5c0d5202c28..4964b48d98a 100644 --- a/app/src/organisms/ProtocolSetupParameters/ChooseCsvFile.tsx +++ b/app/src/organisms/ProtocolSetupParameters/ChooseCsvFile.tsx @@ -10,8 +10,8 @@ import { DIRECTION_COLUMN, DIRECTION_ROW, Flex, - SPACING, LegacyStyledText, + SPACING, TYPOGRAPHY, } from '@opentrons/components' import { useAllCsvFilesQuery } from '@opentrons/react-api-client' @@ -46,8 +46,30 @@ export function ChooseCsvFile({ const { t } = useTranslation('protocol_setup') const csvFilesOnUSB = useSelector(getShellUpdateDataFiles) ?? [] + const csvFilesOnRobot = (useAllCsvFilesQuery(protocolId).data?.data ?? + []) as CsvFileData[] + const sortedCsvFilesOnUSB = csvFilesOnUSB.sort((a, b) => { + const regex = /^(.*\/)?(.+?)(\d*)\.csv$/ + const aMatch = a.match(regex) + const bMatch = b.match(regex) + + if (!aMatch || !bMatch) { + console.error('Invalid filename format:', !aMatch ? a : b) + return 0 + } + + const [, , aText, aNum] = aMatch + const [, , bText, bNum] = bMatch + + if (aText !== bText) { + return aText.localeCompare(bText) + } - const csvFilesOnRobot = useAllCsvFilesQuery(protocolId).data?.data.files ?? [] + return ( + (aNum === '' ? 0 : parseInt(aNum, 10)) - + (bNum === '' ? 0 : parseInt(bNum, 10)) + ) + }) const initialFileObject: CsvFileParameterFileData = parameter.file ?? {} const [ @@ -57,7 +79,7 @@ export function ChooseCsvFile({ const handleBackButton = (): void => { if (!isEqual(csvFileSelected, initialFileObject)) { - setParameter(csvFileSelected, parameter.variableName) + setParameter(csvFileSelected, parameter.variableName as string) } handleGoBack() } @@ -94,16 +116,17 @@ export function ChooseCsvFile({ {csvFilesOnRobot.length !== 0 ? ( csvFilesOnRobot.map((csv: CsvFileData) => ( - { - setCsvFileSelected({ id: csv.id, fileName: csv.name }) - }} - isSelected={csvFileSelected?.id === csv.id} - /> + + { + setCsvFileSelected({ id: csv.id, fileName: csv.name }) + }} + id={`${csv.id}-on-robot`} + isSelected={csvFileSelected?.id === csv.id} + /> + )) ) : ( @@ -115,22 +138,22 @@ export function ChooseCsvFile({ {t('csv_files_on_usb')} - {csvFilesOnUSB.length !== 0 ? ( - csvFilesOnUSB.map(csvFilePath => { + {sortedCsvFilesOnUSB.length !== 0 ? ( + sortedCsvFilesOnUSB.map(csvFilePath => { const fileName = last(csvFilePath.split('/')) return ( {csvFilePath.length !== 0 && fileName !== undefined ? ( { setCsvFileSelected({ filePath: csvFilePath, - fileName: fileName, + fileName, }) }} + id={`${csvFilePath.replace('/', '-')}}-on-usb`} isSelected={csvFileSelected?.filePath === csvFilePath} /> ) : null} diff --git a/app/src/organisms/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx b/app/src/organisms/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx index 918f5838084..404ee1059f9 100644 --- a/app/src/organisms/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx +++ b/app/src/organisms/ProtocolSetupParameters/__tests__/ChooseCsvFile.test.tsx @@ -32,20 +32,18 @@ const mockUsbData = [ const mockDataOnRobot = { data: { - data: { - files: [ - { - id: '1', - createdAt: '2024-06-07T19:19:56.268029+00:00', - name: 'rtp_mock_file1.csv', - }, - { - id: '2', - createdAt: '2024-06-17T19:19:56.268029+00:00', - name: 'rtp_mock_file2.csv', - }, - ], - }, + data: [ + { + id: '1', + createdAt: '2024-06-07T19:19:56.268029+00:00', + name: 'rtp_mock_file1.csv', + }, + { + id: '2', + createdAt: '2024-06-17T19:19:56.268029+00:00', + name: 'rtp_mock_file2.csv', + }, + ], }, } @@ -72,6 +70,7 @@ describe('ChooseCsvFile', () => { .calledWith(PROTOCOL_ID) .thenReturn([] as any) }) + it('should render text and buttons', () => { render(props) screen.getByText('Choose CSV file') diff --git a/app/src/organisms/ProtocolUpload/hooks/index.ts b/app/src/organisms/ProtocolUpload/hooks/index.ts index 52b34474d94..c53b3d97ce0 100644 --- a/app/src/organisms/ProtocolUpload/hooks/index.ts +++ b/app/src/organisms/ProtocolUpload/hooks/index.ts @@ -2,7 +2,6 @@ export * from './useCloseCurrentRun' export * from './useCurrentProtocol' export * from './useCurrentRun' export * from './useCurrentRunCommands' -export * from './useCurrentRunId' export * from './useCloneRun' export * from './useRestartRun' export * from './useRunCommands' diff --git a/app/src/organisms/ProtocolUpload/hooks/useCloseCurrentRun.ts b/app/src/organisms/ProtocolUpload/hooks/useCloseCurrentRun.ts index 811ac95e223..58512ba1e9d 100644 --- a/app/src/organisms/ProtocolUpload/hooks/useCloseCurrentRun.ts +++ b/app/src/organisms/ProtocolUpload/hooks/useCloseCurrentRun.ts @@ -1,7 +1,7 @@ import * as React from 'react' import { useDismissCurrentRunMutation } from '@opentrons/react-api-client' -import { useCurrentRunId } from './useCurrentRunId' +import { useCurrentRunId } from '../../../resources/runs' import type { UseDismissCurrentRunMutationOptions } from '@opentrons/react-api-client/src/runs/useDismissCurrentRunMutation' diff --git a/app/src/organisms/ProtocolUpload/hooks/useCurrentRun.ts b/app/src/organisms/ProtocolUpload/hooks/useCurrentRun.ts index 6510f7e672e..0c19e86c097 100644 --- a/app/src/organisms/ProtocolUpload/hooks/useCurrentRun.ts +++ b/app/src/organisms/ProtocolUpload/hooks/useCurrentRun.ts @@ -1,5 +1,4 @@ -import { useCurrentRunId } from './useCurrentRunId' -import { useNotifyRunQuery } from '../../../resources/runs' +import { useNotifyRunQuery, useCurrentRunId } from '../../../resources/runs' import type { Run } from '@opentrons/api-client' diff --git a/app/src/organisms/ProtocolUpload/hooks/useCurrentRunCommands.ts b/app/src/organisms/ProtocolUpload/hooks/useCurrentRunCommands.ts index 59ca41a4b23..b6cc00709f9 100644 --- a/app/src/organisms/ProtocolUpload/hooks/useCurrentRunCommands.ts +++ b/app/src/organisms/ProtocolUpload/hooks/useCurrentRunCommands.ts @@ -1,4 +1,4 @@ -import { useCurrentRunId } from './useCurrentRunId' +import { useCurrentRunId } from '../../../resources/runs' import { useRunCommands } from './useRunCommands' import type { UseQueryOptions } from 'react-query' import type { diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/AirGap.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/AirGap.tsx index 1101cf3b9e3..16732019986 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/AirGap.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/AirGap.tsx @@ -109,13 +109,13 @@ export function AirGap(props: AirGapProps): JSX.Element { // after each aspirate action, so we need to halve the available capacity for single path // to get the amount available, assuming a min of 2 aspirates per dispense maxAvailableCapacity = - (Math.min(maxPipetteVolume, tipVolume) - state.volume) / 2 + (Math.min(maxPipetteVolume, tipVolume) - 2 * state.volume) / 2 } else { // aspirate air gap for multi dispense occurs once per asprirate and // available volume is max capacity - volume*3 assuming a min of 2 dispenses // per aspirate plus 1x the volume for disposal maxAvailableCapacity = - Math.min(maxPipetteVolume, tipVolume) - state.volume / 3 + Math.min(maxPipetteVolume, tipVolume) - state.volume * 3 } } diff --git a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx index 5d79f2a85e5..7ec21c7c857 100644 --- a/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx +++ b/app/src/organisms/QuickTransferFlow/QuickTransferAdvancedSettings/index.tsx @@ -156,7 +156,9 @@ export function QuickTransferAdvancedSettings( reps: state.mixOnAspirate?.repititions, }) : '', - enabled: state.transferType === 'transfer', + enabled: + state.transferType === 'transfer' || + state.transferType === 'distribute', onClick: () => { if (state.transferType === 'transfer') { setSelectedSetting('aspirate_mix') @@ -234,7 +236,9 @@ export function QuickTransferAdvancedSettings( reps: state.mixOnDispense?.repititions, }) : '', - enabled: state.transferType === 'transfer', + enabled: + state.transferType === 'transfer' || + state.transferType === 'consolidate', onClick: () => { if (state.transferType === 'transfer') { setSelectedSetting('dispense_mix') diff --git a/app/src/organisms/QuickTransferFlow/TipManagement/ChangeTip.tsx b/app/src/organisms/QuickTransferFlow/TipManagement/ChangeTip.tsx index 32a2ea9111c..af10ba03431 100644 --- a/app/src/organisms/QuickTransferFlow/TipManagement/ChangeTip.tsx +++ b/app/src/organisms/QuickTransferFlow/TipManagement/ChangeTip.tsx @@ -35,9 +35,13 @@ export function ChangeTip(props: ChangeTipProps): JSX.Element { ) { allowedChangeTipOptions.push('always') } - if (state.path === 'single' && state.transferType === 'distribute') { + if ( + state.path === 'single' && + state.transferType === 'distribute' && + state.destinationWells.length <= 96 + ) { allowedChangeTipOptions.push('perDest') - } else if (state.path === 'single') { + } else if (state.path === 'single' && state.sourceWells.length <= 96) { allowedChangeTipOptions.push('perSource') } diff --git a/app/src/organisms/RunProgressMeter/index.tsx b/app/src/organisms/RunProgressMeter/index.tsx index 2914e07cf53..c1c674d03b3 100644 --- a/app/src/organisms/RunProgressMeter/index.tsx +++ b/app/src/organisms/RunProgressMeter/index.tsx @@ -31,7 +31,7 @@ import { } from '@opentrons/api-client' import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getTopPortalEl } from '../../App/portal' +import { getModalPortalEl } from '../../App/portal' import { Tooltip } from '../../atoms/Tooltip' import { CommandText } from '../../molecules/Command' import { useRunStatus } from '../RunTimeControl/hooks' @@ -175,7 +175,7 @@ export function RunProgressMeter(props: RunProgressMeterProps): JSX.Element { run={runData} analysis={analysis} />, - getTopPortalEl() + getModalPortalEl() ) : null} diff --git a/app/src/organisms/RunTimeControl/__tests__/hooks.test.tsx b/app/src/organisms/RunTimeControl/__tests__/hooks.test.tsx index 0fe4351caa7..8107a236383 100644 --- a/app/src/organisms/RunTimeControl/__tests__/hooks.test.tsx +++ b/app/src/organisms/RunTimeControl/__tests__/hooks.test.tsx @@ -4,11 +4,7 @@ import { describe, it, expect, vi, beforeEach } from 'vitest' import '@testing-library/jest-dom/vitest' import { useRunActionMutations } from '@opentrons/react-api-client' -import { - useCloneRun, - useCurrentRunId, - useRunCommands, -} from '../../ProtocolUpload/hooks' +import { useCloneRun, useRunCommands } from '../../ProtocolUpload/hooks' import { useRunControls, useRunStatus, @@ -16,7 +12,7 @@ import { useRunTimestamps, useRunErrors, } from '../hooks' -import { useNotifyRunQuery } from '../../../resources/runs' +import { useNotifyRunQuery, useCurrentRunId } from '../../../resources/runs' import { RUN_ID_2, diff --git a/app/src/organisms/RunTimeControl/hooks.ts b/app/src/organisms/RunTimeControl/hooks.ts index 4339a3a3eee..606e5852f36 100644 --- a/app/src/organisms/RunTimeControl/hooks.ts +++ b/app/src/organisms/RunTimeControl/hooks.ts @@ -16,12 +16,8 @@ import { } from '@opentrons/api-client' import { useRunActionMutations } from '@opentrons/react-api-client' -import { - useCloneRun, - useCurrentRunId, - useRunCommands, -} from '../ProtocolUpload/hooks' -import { useNotifyRunQuery } from '../../resources/runs' +import { useCloneRun, useRunCommands } from '../ProtocolUpload/hooks' +import { useNotifyRunQuery, useCurrentRunId } from '../../resources/runs' import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' import type { UseQueryOptions } from 'react-query' diff --git a/app/src/organisms/TakeoverModal/MaintenanceRunTakeover.tsx b/app/src/organisms/TakeoverModal/MaintenanceRunTakeover.tsx index 54b6e3d5d2b..b4cef390203 100644 --- a/app/src/organisms/TakeoverModal/MaintenanceRunTakeover.tsx +++ b/app/src/organisms/TakeoverModal/MaintenanceRunTakeover.tsx @@ -1,5 +1,8 @@ import * as React from 'react' +import { useTranslation } from 'react-i18next' + import { useDeleteMaintenanceRunMutation } from '@opentrons/react-api-client' + import { TakeoverModal } from './TakeoverModal' import { MaintenanceRunStatusProvider } from './MaintenanceRunStatusProvider' import { useMaintenanceRunTakeover } from './useMaintenanceRunTakeover' @@ -25,6 +28,7 @@ interface MaintenanceRunTakeoverModalProps { export function MaintenanceRunTakeoverModal( props: MaintenanceRunTakeoverModalProps ): JSX.Element { + const { i18n, t } = useTranslation(['shared', 'branded']) const [isLoading, setIsLoading] = React.useState(false) const [ showConfirmTerminateModal, @@ -58,6 +62,7 @@ export function MaintenanceRunTakeoverModal( <> {desktopMaintenanceRunInProgress && ( > confirmTerminate: () => void @@ -27,12 +30,13 @@ interface TakeoverModalProps { export function TakeoverModal(props: TakeoverModalProps): JSX.Element { const { + title, showConfirmTerminateModal, setShowConfirmTerminateModal, confirmTerminate, terminateInProgress, } = props - const { i18n, t } = useTranslation(['shared', 'branded']) + const { t } = useTranslation(['shared', 'branded']) const terminateHeader: ModalHeaderBaseProps = { title: t('terminate') + '?', @@ -95,7 +99,7 @@ export function TakeoverModal(props: TakeoverModalProps): JSX.Element { marginBottom={SPACING.spacing4} fontWeight={TYPOGRAPHY.fontWeightBold} > - {i18n.format(t('robot_is_busy'), 'capitalize')} + {title} {t('branded:computer_in_app_is_controlling_robot')} diff --git a/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx b/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx index 0e09b3096a3..eff8b68c101 100644 --- a/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx +++ b/app/src/organisms/TakeoverModal/__tests__/TakeoverModal.test.tsx @@ -20,6 +20,7 @@ describe('TakeoverModal', () => { setShowConfirmTerminateModal: vi.fn(), confirmTerminate: vi.fn(), terminateInProgress: false, + title: 'Robot is busy', } }) diff --git a/app/src/pages/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx b/app/src/pages/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx index a5a1df6d218..c5543f06d8c 100644 --- a/app/src/pages/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx +++ b/app/src/pages/Devices/ProtocolRunDetails/__tests__/ProtocolRunDetails.test.tsx @@ -18,7 +18,7 @@ import { ProtocolRunModuleControls } from '../../../../organisms/Devices/Protoco import { ProtocolRunSetup } from '../../../../organisms/Devices/ProtocolRun/ProtocolRunSetup' import { RunPreviewComponent } from '../../../../organisms/RunPreview' import { ProtocolRunRuntimeParameters } from '../../../../organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters' -import { useCurrentRunId } from '../../../../organisms/ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../../resources/runs' import { mockRobotSideAnalysis } from '../../../../molecules/Command/__fixtures__' import { useFeatureFlag } from '../../../../redux/config' import { ProtocolRunDetails } from '..' @@ -33,7 +33,7 @@ vi.mock('../../../../organisms/Devices/ProtocolRun/ProtocolRunHeader') vi.mock('../../../../organisms/Devices/ProtocolRun/ProtocolRunSetup') vi.mock('../../../../organisms/RunPreview') vi.mock('../../../../organisms/Devices/ProtocolRun/ProtocolRunModuleControls') -vi.mock('../../../../organisms/ProtocolUpload/hooks') +vi.mock('../../../../resources/runs') vi.mock( '../../../../organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters' ) diff --git a/app/src/pages/Devices/ProtocolRunDetails/index.tsx b/app/src/pages/Devices/ProtocolRunDetails/index.tsx index e77841ec980..8d21ae21203 100644 --- a/app/src/pages/Devices/ProtocolRunDetails/index.tsx +++ b/app/src/pages/Devices/ProtocolRunDetails/index.tsx @@ -33,7 +33,7 @@ import { RunPreview } from '../../../organisms/RunPreview' import { ProtocolRunSetup } from '../../../organisms/Devices/ProtocolRun/ProtocolRunSetup' import { ProtocolRunModuleControls } from '../../../organisms/Devices/ProtocolRun/ProtocolRunModuleControls' import { ProtocolRunRuntimeParameters } from '../../../organisms/Devices/ProtocolRun/ProtocolRunRunTimeParameters' -import { useCurrentRunId } from '../../../organisms/ProtocolUpload/hooks' +import { useCurrentRunId } from '../../../resources/runs' import { OPENTRONS_USB } from '../../../redux/discovery' import { fetchProtocols } from '../../../redux/protocol-storage' import { appShellRequestor } from '../../../redux/shell/remote' diff --git a/app/src/pages/ProtocolSetup/index.tsx b/app/src/pages/ProtocolSetup/index.tsx index 8954e7d0b01..36ce4220bcb 100644 --- a/app/src/pages/ProtocolSetup/index.tsx +++ b/app/src/pages/ProtocolSetup/index.tsx @@ -813,6 +813,9 @@ function PrepareToRun({ {showConfirmCancelModal ? ( { + if (isQuickTransfer) { + deleteRun(runId) + } + } + + const { reset, isResetRunLoading } = useRunControls(runId, onCloneRunSuccess) const trackEvent = useTrackEvent() const { closeCurrentRun, isClosingCurrentRun } = useCloseCurrentRun() const robotAnalyticsData = useRobotAnalyticsData(robotName) @@ -148,6 +159,20 @@ export function RunSummary(): JSX.Element { closeCurrentRun() navigate('/') } + // TODO(jh, 07-24-24): After EXEC-504, add reportRecoveredRunResult here. + + const returnToQuickTransfer = (): void => { + if (!isRunCurrent) { + deleteRun(runId) + } else { + closeCurrentRun({ + onSuccess: () => { + deleteRun(runId) + }, + }) + } + navigate('/quick-transfer') + } // TODO(jh, 05-30-24): EXEC-487. Refactor reset() so we can redirect to the setup page, showing the shimmer skeleton instead. const runAgain = (): void => { @@ -177,6 +202,8 @@ export function RunSummary(): JSX.Element { host, pipettesWithTip, }) + } else if (isQuickTransfer) { + returnToQuickTransfer() } else { returnToDash() } @@ -323,7 +350,11 @@ export function RunSummary(): JSX.Element { onClick={() => { handleReturnToDash(pipettesWithTip) }} - buttonText={t('return_to_dashboard')} + buttonText={ + isQuickTransfer + ? t('return_to_quick_transfer') + : t('return_to_dashboard') + } height="17rem" /> diff --git a/app/src/redux/analytics/constants.ts b/app/src/redux/analytics/constants.ts index 9cb3af568ac..e2dc4d58473 100644 --- a/app/src/redux/analytics/constants.ts +++ b/app/src/redux/analytics/constants.ts @@ -60,3 +60,16 @@ export const ANALYTICS_STATE_ROBOT_UPDATE = { } as const export const ANALYTICS_ROBOT_UPDATE_VIEW = 'robotUpdateView' export const ANALYTICS_ROBOT_UPDATE_CHANGE_LOG_VIEW = 'robotUpdateChangeLogView' + +/** + * Error Recovery Analytics + */ + +export const ANALYTICS_RECOVERY_ERROR_EVENT = 'recoveryErrorEvent' +export const ANALYTICS_RECOVERY_INITIAL_ACTION = 'recoveryInitialAction' +export const ANALYTICS_RECOVERY_ACTION_SELECTED = + 'recoverySelectedRecoveryAction' +export const ANALYTICS_RECOVERY_VIEW_ERROR_DETAILS = 'recoveryViewErrorDetails' +export const ANALYTICS_RECOVERY_ACTION_RESULT = + 'recoverySelectedRecoveryActionResult' +export const ANALYTICS_RECOVERY_RUN_RESULT = 'recoveryRunResultAfterError' diff --git a/app/src/redux/config/__tests__/selectors.test.ts b/app/src/redux/config/__tests__/selectors.test.ts index 3ba6c0ea6cf..18262108c0a 100644 --- a/app/src/redux/config/__tests__/selectors.test.ts +++ b/app/src/redux/config/__tests__/selectors.test.ts @@ -282,4 +282,20 @@ describe('shell selectors', () => { expect(Selectors.getApplyHistoricOffsets(state)).toEqual(true) }) }) + + describe('getUserId', () => { + it('should return userId if it exists in config', () => { + const state: State = { + config: { + userInfo: { userId: 'test-user-id' }, + }, + } as any + expect(Selectors.getUserId(state)).toEqual('test-user-id') + }) + + it('should return an empty string if config is null', () => { + const state: State = { config: null } as any + expect(Selectors.getUserId(state)).toEqual('') + }) + }) }) diff --git a/app/src/redux/config/schema-types.ts b/app/src/redux/config/schema-types.ts index ae554217e11..ae83dbabe7e 100644 --- a/app/src/redux/config/schema-types.ts +++ b/app/src/redux/config/schema-types.ts @@ -268,4 +268,11 @@ export type ConfigV23 = Omit & { } } -export type Config = ConfigV23 +export type ConfigV24 = Omit & { + version: 24 + userInfo: { + userId: string + } +} + +export type Config = ConfigV24 diff --git a/app/src/redux/config/selectors.ts b/app/src/redux/config/selectors.ts index f7de3783290..76749e0a869 100644 --- a/app/src/redux/config/selectors.ts +++ b/app/src/redux/config/selectors.ts @@ -145,3 +145,8 @@ export const getOnDeviceDisplaySettings: ( unfinishedUnboxingFlowRoute: '/welcome', } }) + +export const getUserId: (state: State) => string = createSelector( + getConfig, + config => config?.userInfo.userId ?? '' +) diff --git a/app/src/redux/shell/types.ts b/app/src/redux/shell/types.ts index 1016f632594..aeee1fe72c6 100644 --- a/app/src/redux/shell/types.ts +++ b/app/src/redux/shell/types.ts @@ -144,6 +144,7 @@ export type NotifyTopic = | `robot-server/runs/${string}` | 'robot-server/deck_configuration' | `robot-server/runs/pre_serialized_commands/${string}` + | `robot-server/clientData/${string}` export interface NotifySubscribeAction { type: 'shell:NOTIFY_SUBSCRIBE' diff --git a/app/src/resources/client_data/constants.ts b/app/src/resources/client_data/constants.ts new file mode 100644 index 00000000000..c267f686ab9 --- /dev/null +++ b/app/src/resources/client_data/constants.ts @@ -0,0 +1,9 @@ +/** + * Keys + * + * Keys must be in URL-compliant format + */ + +export const KEYS = { + ERROR_RECOVERY: 'ot-error-recovery-v1', +} as const diff --git a/app/src/resources/client_data/index.ts b/app/src/resources/client_data/index.ts new file mode 100644 index 00000000000..55ab3c895b4 --- /dev/null +++ b/app/src/resources/client_data/index.ts @@ -0,0 +1 @@ +export * from './recovery' diff --git a/app/src/resources/client_data/recovery/index.ts b/app/src/resources/client_data/recovery/index.ts new file mode 100644 index 00000000000..7afe196b56a --- /dev/null +++ b/app/src/resources/client_data/recovery/index.ts @@ -0,0 +1,4 @@ +export * from './useClientDataRecovery' +export * from './useUpdateClientDataRecovery' + +export * from './types' diff --git a/app/src/resources/client_data/recovery/types.ts b/app/src/resources/client_data/recovery/types.ts new file mode 100644 index 00000000000..fef5848641a --- /dev/null +++ b/app/src/resources/client_data/recovery/types.ts @@ -0,0 +1,7 @@ +export type RecoveryIntent = 'recovering' | 'canceling' + +// The shape of the client data at the error recovery key. +export interface ClientDataRecovery { + userId: string | null + intent: RecoveryIntent | null +} diff --git a/app/src/resources/client_data/recovery/useClientDataRecovery.ts b/app/src/resources/client_data/recovery/useClientDataRecovery.ts new file mode 100644 index 00000000000..e9bfa324578 --- /dev/null +++ b/app/src/resources/client_data/recovery/useClientDataRecovery.ts @@ -0,0 +1,22 @@ +import { useNotifyClientDataRecovery } from './useNotifyClientDataRecovery' + +import type { UseQueryOptions } from 'react-query' +import type { AxiosError } from 'axios' +import type { ClientDataResponse } from '@opentrons/api-client' +import type { ClientDataRecovery } from './types' + +// Returns the client data store value associated with the error recovery key, if any. +export function useClientDataRecovery( + options: UseQueryOptions< + ClientDataResponse, + AxiosError + > = {} +): ClientDataRecovery { + const { data } = useNotifyClientDataRecovery(options) + + const { userId: userIdResponse, intent } = data?.data ?? {} + const userId = + userIdResponse != null && userIdResponse.length > 0 ? userIdResponse : null + + return { userId, intent: intent ?? null } +} diff --git a/app/src/resources/client_data/recovery/useNotifyClientDataRecovery.ts b/app/src/resources/client_data/recovery/useNotifyClientDataRecovery.ts new file mode 100644 index 00000000000..a93553fa3fa --- /dev/null +++ b/app/src/resources/client_data/recovery/useNotifyClientDataRecovery.ts @@ -0,0 +1,32 @@ +import { useClientData } from '@opentrons/react-api-client' + +import { KEYS } from '../constants' +import { useNotifyDataReady } from '../../useNotifyDataReady' + +import type { UseQueryOptions, UseQueryResult } from 'react-query' +import type { AxiosError } from 'axios' +import type { ClientDataResponse } from '@opentrons/api-client' +import type { ClientDataRecovery } from './types' + +export function useNotifyClientDataRecovery( + options: UseQueryOptions< + ClientDataResponse, + AxiosError + > = {} +): UseQueryResult, AxiosError> { + const { notifyOnSettled, shouldRefetch } = useNotifyDataReady({ + topic: `robot-server/clientData/${KEYS.ERROR_RECOVERY}`, + options, + }) + + const httpQueryResult = useClientData( + KEYS.ERROR_RECOVERY, + { + ...options, + enabled: options?.enabled !== false && shouldRefetch, + onSettled: notifyOnSettled, + } + ) + + return httpQueryResult +} diff --git a/app/src/resources/client_data/recovery/useUpdateClientDataRecovery.ts b/app/src/resources/client_data/recovery/useUpdateClientDataRecovery.ts new file mode 100644 index 00000000000..da5164e2b85 --- /dev/null +++ b/app/src/resources/client_data/recovery/useUpdateClientDataRecovery.ts @@ -0,0 +1,46 @@ +import { useUpdateClientData } from '@opentrons/react-api-client' +import { useSelector } from 'react-redux' + +import { getUserId } from '../../../redux/config' +import { KEYS } from '../constants' + +import type { + UseUpdateClientDataMutationOptions, + UseUpdateClientDataMutationResult, +} from '@opentrons/react-api-client' +import type { ClientDataRecovery } from './types' + +export type UseUpdateClientDataRecoveryResult = Omit< + UseUpdateClientDataMutationResult, + 'updateClientData' +> & { + /* Update the server with the user's id and a recovery intent. */ + updateWithIntent: (intent: ClientDataRecovery['intent']) => void + /* Clear the clientData store at the error recovery key. */ + clearClientData: () => void +} + +// Update the client data store value associated with the error recovery key. +export function useUpdateClientDataRecovery( + options: UseUpdateClientDataMutationOptions = {} +): UseUpdateClientDataRecoveryResult { + const { + updateClientData, + ...mutate + } = useUpdateClientData(KEYS.ERROR_RECOVERY, options) + const thisUserId = useSelector(getUserId) + + const updateWithIntent = (intent: ClientDataRecovery['intent']): void => { + updateClientData({ userId: thisUserId, intent }) + } + + const clearClientData = (): void => { + updateClientData({ userId: null, intent: null }) + } + + return { + ...mutate, + updateWithIntent, + clearClientData, + } +} diff --git a/app/src/organisms/ProtocolUpload/hooks/__tests__/useCurrentRunId.test.tsx b/app/src/resources/runs/__tests__/useCurrentRunId.test.tsx similarity index 92% rename from app/src/organisms/ProtocolUpload/hooks/__tests__/useCurrentRunId.test.tsx rename to app/src/resources/runs/__tests__/useCurrentRunId.test.tsx index af4c9edf012..b10695789f1 100644 --- a/app/src/organisms/ProtocolUpload/hooks/__tests__/useCurrentRunId.test.tsx +++ b/app/src/resources/runs/__tests__/useCurrentRunId.test.tsx @@ -3,9 +3,9 @@ import { renderHook } from '@testing-library/react' import { describe, it, afterEach, expect, vi } from 'vitest' import { useCurrentRunId } from '../useCurrentRunId' -import { useNotifyAllRunsQuery } from '../../../../resources/runs' +import { useNotifyAllRunsQuery } from '../useNotifyAllRunsQuery' -vi.mock('../../../../resources/runs') +vi.mock('../useNotifyAllRunsQuery') describe('useCurrentRunId hook', () => { afterEach(() => { diff --git a/app/src/resources/runs/index.ts b/app/src/resources/runs/index.ts index a69aba067aa..b9023f3f702 100644 --- a/app/src/resources/runs/index.ts +++ b/app/src/resources/runs/index.ts @@ -4,3 +4,4 @@ export * from './useNotifyAllRunsQuery' export * from './useNotifyRunQuery' export * from './useNotifyAllCommandsQuery' export * from './useNotifyAllCommandsAsPreSerializedList' +export * from './useCurrentRunId' diff --git a/app/src/organisms/ProtocolUpload/hooks/useCurrentRunId.ts b/app/src/resources/runs/useCurrentRunId.ts similarity index 60% rename from app/src/organisms/ProtocolUpload/hooks/useCurrentRunId.ts rename to app/src/resources/runs/useCurrentRunId.ts index 6ae83907681..88efba892fc 100644 --- a/app/src/organisms/ProtocolUpload/hooks/useCurrentRunId.ts +++ b/app/src/resources/runs/useCurrentRunId.ts @@ -1,13 +1,19 @@ -import { useNotifyAllRunsQuery } from '../../../resources/runs' +import { useNotifyAllRunsQuery } from './useNotifyAllRunsQuery' import type { AxiosError } from 'axios' import type { UseAllRunsQueryOptions } from '@opentrons/react-api-client/src/runs/useAllRunsQuery' -import type { QueryOptionsWithPolling } from '../../../resources/useNotifyDataReady' +import type { QueryOptionsWithPolling } from '../useNotifyDataReady' +import type { HostConfig } from '@opentrons/api-client' export function useCurrentRunId( - options: QueryOptionsWithPolling = {} + options: QueryOptionsWithPolling = {}, + hostOverride?: HostConfig | null ): string | null { - const { data: allRuns } = useNotifyAllRunsQuery({ pageLength: 0 }, options) + const { data: allRuns } = useNotifyAllRunsQuery( + { pageLength: 0 }, + options, + hostOverride + ) const currentRunLink = allRuns?.links?.current ?? null return currentRunLink != null && typeof currentRunLink !== 'string' && diff --git a/app/src/resources/runs/useNotifyRunQuery.ts b/app/src/resources/runs/useNotifyRunQuery.ts index 1b0a99e5f41..003cfeabf94 100644 --- a/app/src/resources/runs/useNotifyRunQuery.ts +++ b/app/src/resources/runs/useNotifyRunQuery.ts @@ -3,26 +3,32 @@ import { useRunQuery } from '@opentrons/react-api-client' import { useNotifyDataReady } from '../useNotifyDataReady' import type { UseQueryResult } from 'react-query' -import type { Run } from '@opentrons/api-client' +import type { Run, HostConfig } from '@opentrons/api-client' import type { QueryOptionsWithPolling } from '../useNotifyDataReady' import type { NotifyTopic } from '../../redux/shell/types' export function useNotifyRunQuery( runId: string | null, - options: QueryOptionsWithPolling = {} + options: QueryOptionsWithPolling = {}, + hostOverride?: HostConfig | null ): UseQueryResult { const isEnabled = options.enabled !== false && runId != null const { notifyOnSettled, shouldRefetch } = useNotifyDataReady({ topic: `robot-server/runs/${runId}` as NotifyTopic, options: { ...options, enabled: options.enabled != null && runId != null }, + hostOverride, }) - const httpResponse = useRunQuery(runId, { - ...options, - enabled: isEnabled && shouldRefetch, - onSettled: notifyOnSettled, - }) + const httpResponse = useRunQuery( + runId, + { + ...options, + enabled: isEnabled && shouldRefetch, + onSettled: notifyOnSettled, + }, + hostOverride + ) return httpResponse } diff --git a/components/src/hardware-sim/BaseDeck/BaseDeck.tsx b/components/src/hardware-sim/BaseDeck/BaseDeck.tsx index f7a3baaf715..50791c42b3c 100644 --- a/components/src/hardware-sim/BaseDeck/BaseDeck.tsx +++ b/components/src/hardware-sim/BaseDeck/BaseDeck.tsx @@ -54,6 +54,7 @@ export interface LabwareOnDeck { /** generic prop to render self-positioned children for each labware */ labwareChildren?: React.ReactNode onLabwareClick?: () => void + highlight?: boolean } export interface ModuleOnDeck { @@ -65,6 +66,7 @@ export interface ModuleOnDeck { /** generic prop to render self-positioned children for each module */ moduleChildren?: React.ReactNode onLabwareClick?: () => void + highlightLabware?: boolean } interface BaseDeckProps { deckConfig: DeckConfiguration @@ -240,6 +242,7 @@ export function BaseDeck(props: BaseDeckProps): JSX.Element { innerProps, moduleChildren, onLabwareClick, + highlightLabware, }) => { const slotPosition = getPositionFromSlotId( moduleLocation.slotName, @@ -266,6 +269,7 @@ export function BaseDeck(props: BaseDeckProps): JSX.Element { inferModuleOrientationFromXCoordinate(slotPosition[0]) === 'left' && moduleModel === HEATERSHAKER_MODULE_V1 } + highlight={highlightLabware} /> ) : null} {moduleChildren} @@ -281,6 +285,7 @@ export function BaseDeck(props: BaseDeckProps): JSX.Element { wellFill, missingTips, onLabwareClick, + highlight, }) => { if ( labwareLocation === 'offDeck' || @@ -308,6 +313,7 @@ export function BaseDeck(props: BaseDeckProps): JSX.Element { onLabwareClick={onLabwareClick} wellFill={wellFill ?? undefined} missingTips={missingTips} + highlight={highlight} /> {labwareChildren} diff --git a/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx b/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx index ffd6e85ce55..14a33b00aa0 100644 --- a/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx +++ b/components/src/hardware-sim/Labware/labwareInternals/LabwareOutline.tsx @@ -46,6 +46,7 @@ export function LabwareOutline(props: LabwareOutlineProps): JSX.Element { + {/* TODO(bh, 2024-07-22): adjust gaussian blur for stacks */} str: diff --git a/hardware/opentrons_hardware/firmware_bindings/constants.py b/hardware/opentrons_hardware/firmware_bindings/constants.py index f0202307647..960419d334c 100644 --- a/hardware/opentrons_hardware/firmware_bindings/constants.py +++ b/hardware/opentrons_hardware/firmware_bindings/constants.py @@ -241,6 +241,9 @@ class MessageId(int, Enum): gear_write_motor_driver_request = 0x506 gear_read_motor_driver_request = 0x507 + max_sensor_value_request = 0x70 + max_sensor_value_response = 0x71 + read_sensor_request = 0x82 write_sensor_request = 0x83 baseline_sensor_request = 0x84 diff --git a/hardware/opentrons_hardware/firmware_bindings/messages/message_definitions.py b/hardware/opentrons_hardware/firmware_bindings/messages/message_definitions.py index 82ba3040928..12f65f80f81 100644 --- a/hardware/opentrons_hardware/firmware_bindings/messages/message_definitions.py +++ b/hardware/opentrons_hardware/firmware_bindings/messages/message_definitions.py @@ -522,6 +522,28 @@ class ReadLimitSwitchResponse(BaseMessage): # noqa: D101 message_id: Literal[MessageId.limit_sw_response] = MessageId.limit_sw_response +@dataclass +class MaxSensorValueRequest(BaseMessage): # noqa: D101 + payload: payloads.ReadFromSensorRequestPayload + payload_type: Type[ + payloads.ReadFromSensorRequestPayload + ] = payloads.ReadFromSensorRequestPayload + message_id: Literal[ + MessageId.max_sensor_value_request + ] = MessageId.max_sensor_value_request + + +@dataclass +class MaxSensorValueResponse(BaseMessage): # noqa: D101 + payload: payloads.ReadFromSensorRequestPayload + payload_type: Type[ + payloads.ReadFromSensorRequestPayload + ] = payloads.ReadFromSensorRequestPayload + message_id: Literal[ + MessageId.max_sensor_value_response + ] = MessageId.max_sensor_value_response + + @dataclass class ReadFromSensorRequest(BaseMessage): # noqa: D101 payload: payloads.ReadFromSensorRequestPayload diff --git a/hardware/opentrons_hardware/firmware_bindings/messages/messages.py b/hardware/opentrons_hardware/firmware_bindings/messages/messages.py index 772bb6c7e86..0249ddec69e 100644 --- a/hardware/opentrons_hardware/firmware_bindings/messages/messages.py +++ b/hardware/opentrons_hardware/firmware_bindings/messages/messages.py @@ -66,6 +66,8 @@ defs.FirmwareUpdateStartApp, defs.ReadLimitSwitchRequest, defs.ReadLimitSwitchResponse, + defs.MaxSensorValueRequest, + defs.MaxSensorValueResponse, defs.ReadFromSensorRequest, defs.WriteToSensorRequest, defs.BaselineSensorRequest, diff --git a/labware-library/Makefile b/labware-library/Makefile index 92fe4785cda..2ccca5f45a8 100644 --- a/labware-library/Makefile +++ b/labware-library/Makefile @@ -30,7 +30,6 @@ dist: # development assets server .PHONY: dev dev: export NODE_ENV := development -dev: export NODE_OPTIONS := --openssl-legacy-provider dev: vite serve --host=:: diff --git a/labware-library/README.md b/labware-library/README.md index bae2741a422..06982723d7d 100644 --- a/labware-library/README.md +++ b/labware-library/README.md @@ -43,7 +43,7 @@ make -C labware-library all make -C labware-library clean dist # start a hot-reloading development server -# (optional) specify port with PORT; default is 8080 +# (optional) specify port with PORT; default is 5173 make -C labware-library dev make -C labware-library dev PORT=8081 @@ -53,11 +53,10 @@ make -C labware-library serve make -C labware-library serve PORT=9091 ``` -### webpack setup +### vite setup -This project (along with our other front-end projects) uses [webpack][] to generate artifacts. +This project (along with our other front-end projects) uses [vite][] to generate artifacts. -- Extends our [base webpack config][base-config] - Entry point is [`labware-library/src/index.tsx`][entry] - [Handlebars][] HTML template is [`labware-library/src/index.hbs`][template] - Post-build, the site is crawled and prerendered with [react-snap][] @@ -68,10 +67,10 @@ This project (along with our other front-end projects) uses [webpack][] to gener [handlebars]: https://handlebarsjs.com/ [css modules]: https://github.com/css-modules/css-modules [react-snap]: https://github.com/stereobooster/react-snap -[base-config]: ../webpack-config [entry]: ./src/index.js [template]: ./src/index.hbs [global-style]: ./src/global.module.css +[vite]: https://vitejs.dev/ ### environment variables diff --git a/labware-library/src/labware-creator/components/sections/StackingOffsets.tsx b/labware-library/src/labware-creator/components/sections/StackingOffsets.tsx index 54daa88e229..ee5c6097658 100644 --- a/labware-library/src/labware-creator/components/sections/StackingOffsets.tsx +++ b/labware-library/src/labware-creator/components/sections/StackingOffsets.tsx @@ -64,6 +64,7 @@ export function StackingOffsets(): JSX.Element | null { const isFlatBottom = values.wellBottomShape === 'flat' const isCircular = values.wellShape === 'circular' const isReservoir = values.labwareType === 'reservoir' + const isWellPlate = values.labwareType === 'wellPlate' const labwareHeight = values.labwareZDimension const has12Columns = values.gridColumns != null && parseInt(values.gridColumns) === 12 @@ -84,7 +85,7 @@ export function StackingOffsets(): JSX.Element | null { definition.parameters.loadName === 'opentrons_96_well_aluminum_block' ) } - if (isFlatBottom && isReservoir) { + if (isFlatBottom && (isReservoir || isWellPlate)) { modifiedAdapterDefinitions = adapterDefinitions.filter( definition => definition.parameters.loadName === @@ -92,6 +93,7 @@ export function StackingOffsets(): JSX.Element | null { definition.parameters.loadName === 'opentrons_universal_flat_adapter' ) } + if ( isFlatBottom && isCircular && diff --git a/opentrons-ai-server/Pipfile b/opentrons-ai-server/Pipfile index 947046431b3..511aaae2b61 100644 --- a/opentrons-ai-server/Pipfile +++ b/opentrons-ai-server/Pipfile @@ -14,6 +14,8 @@ ddtrace = "==2.9.2" pydantic-settings = "==2.3.4" pyjwt = {extras = ["crypto"], version = "*"} python-json-logger = "==2.0.7" +beautifulsoup4 = "==4.12.3" +markdownify = "==0.13.1" [dev-packages] docker = "==7.1.0" @@ -27,6 +29,7 @@ boto3-stubs = "==1.34.114" rich = "==13.7.1" cryptography = "==42.0.7" types-docker = "==7.0.0.20240528" +types-beautifulsoup4 = "*" [requires] python_version = "3.12" diff --git a/opentrons-ai-server/Pipfile.lock b/opentrons-ai-server/Pipfile.lock index 3ca2c236564..5b55e8c1ea6 100644 --- a/opentrons-ai-server/Pipfile.lock +++ b/opentrons-ai-server/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "cb12ec3ffaa00df575c6e86536481c155a47af29a9a638cba8999b39f7bc28a6" + "sha256": "e2bc86bbccb5f6ac73c2e1e72eb1c92164fb6712e5672b31b60c97ea53cf94fd" }, "pipfile-spec": 6, "requires": { @@ -136,6 +136,7 @@ "sha256:74e3d1928edc070d21748185c46e3fb33490f22f52a3addee9aee0f4f7781051", "sha256:b80878c9f40111313e55da8ba20bdba06d8fa3969fc68304167741bbf9e082ed" ], + "index": "pypi", "markers": "python_full_version >= '3.6.0'", "version": "==4.12.3" }, @@ -157,11 +158,11 @@ }, "certifi": { "hashes": [ - "sha256:3cd43f1c6fa7dedc5899d69d3ad0398fd018ad1a17fba83ddaf78aa46c747516", - "sha256:ddc6c8ce995e6987e7faf5e3f1b02b302836a0e5d98ece18392cb1a36c72ad56" + "sha256:5a1e7645bc0ec61a09e26c36f6106dd4cf40c6db3a1fb6352b0244e7fb057c7b", + "sha256:c198e21b1289c2ab85ee4e67bb4b4ef3ead0892059901a8d5b622f24a1101e90" ], "markers": "python_version >= '3.6'", - "version": "==2024.6.2" + "version": "==2024.7.4" }, "cffi": { "hashes": [ @@ -327,40 +328,35 @@ }, "cryptography": { "hashes": [ - "sha256:013629ae70b40af70c9a7a5db40abe5d9054e6f4380e50ce769947b73bf3caad", - "sha256:2346b911eb349ab547076f47f2e035fc8ff2c02380a7cbbf8d87114fa0f1c583", - "sha256:2f66d9cd9147ee495a8374a45ca445819f8929a3efcd2e3df6428e46c3cbb10b", - "sha256:2f88d197e66c65be5e42cd72e5c18afbfae3f741742070e3019ac8f4ac57262c", - "sha256:31f721658a29331f895a5a54e7e82075554ccfb8b163a18719d342f5ffe5ecb1", - "sha256:343728aac38decfdeecf55ecab3264b015be68fc2816ca800db649607aeee648", - "sha256:5226d5d21ab681f432a9c1cf8b658c0cb02533eece706b155e5fbd8a0cdd3949", - "sha256:57080dee41209e556a9a4ce60d229244f7a66ef52750f813bfbe18959770cfba", - "sha256:5a94eccb2a81a309806027e1670a358b99b8fe8bfe9f8d329f27d72c094dde8c", - "sha256:6b7c4f03ce01afd3b76cf69a5455caa9cfa3de8c8f493e0d3ab7d20611c8dae9", - "sha256:7016f837e15b0a1c119d27ecd89b3515f01f90a8615ed5e9427e30d9cdbfed3d", - "sha256:81884c4d096c272f00aeb1f11cf62ccd39763581645b0812e99a91505fa48e0c", - "sha256:81d8a521705787afe7a18d5bfb47ea9d9cc068206270aad0b96a725022e18d2e", - "sha256:8d09d05439ce7baa8e9e95b07ec5b6c886f548deb7e0f69ef25f64b3bce842f2", - "sha256:961e61cefdcb06e0c6d7e3a1b22ebe8b996eb2bf50614e89384be54c48c6b63d", - "sha256:9c0c1716c8447ee7dbf08d6db2e5c41c688544c61074b54fc4564196f55c25a7", - "sha256:a0608251135d0e03111152e41f0cc2392d1e74e35703960d4190b2e0f4ca9c70", - "sha256:a0c5b2b0585b6af82d7e385f55a8bc568abff8923af147ee3c07bd8b42cda8b2", - "sha256:ad803773e9df0b92e0a817d22fd8a3675493f690b96130a5e24f1b8fabbea9c7", - "sha256:b297f90c5723d04bcc8265fc2a0f86d4ea2e0f7ab4b6994459548d3a6b992a14", - "sha256:ba4f0a211697362e89ad822e667d8d340b4d8d55fae72cdd619389fb5912eefe", - "sha256:c4783183f7cb757b73b2ae9aed6599b96338eb957233c58ca8f49a49cc32fd5e", - "sha256:c9bb2ae11bfbab395bdd072985abde58ea9860ed84e59dbc0463a5d0159f5b71", - "sha256:cafb92b2bc622cd1aa6a1dce4b93307792633f4c5fe1f46c6b97cf67073ec961", - "sha256:d45b940883a03e19e944456a558b67a41160e367a719833c53de6911cabba2b7", - "sha256:dc0fdf6787f37b1c6b08e6dfc892d9d068b5bdb671198c72072828b80bd5fe4c", - "sha256:dea567d1b0e8bc5764b9443858b673b734100c2871dc93163f58c46a97a83d28", - "sha256:dec9b018df185f08483f294cae6ccac29e7a6e0678996587363dc352dc65c842", - "sha256:e3ec3672626e1b9e55afd0df6d774ff0e953452886e06e0f1eb7eb0c832e8902", - "sha256:e599b53fd95357d92304510fb7bda8523ed1f79ca98dce2f43c115950aa78801", - "sha256:fa76fbb7596cc5839320000cdd5d0955313696d9511debab7ee7278fc8b5c84a", - "sha256:fff12c88a672ab9c9c1cf7b0c80e3ad9e2ebd9d828d955c126be4fd3e5578c9e" - ], - "version": "==42.0.8" + "sha256:0663585d02f76929792470451a5ba64424acc3cd5227b03921dab0e2f27b1709", + "sha256:08a24a7070b2b6804c1940ff0f910ff728932a9d0e80e7814234269f9d46d069", + "sha256:232ce02943a579095a339ac4b390fbbe97f5b5d5d107f8a08260ea2768be8cc2", + "sha256:2905ccf93a8a2a416f3ec01b1a7911c3fe4073ef35640e7ee5296754e30b762b", + "sha256:299d3da8e00b7e2b54bb02ef58d73cd5f55fb31f33ebbf33bd00d9aa6807df7e", + "sha256:2c6d112bf61c5ef44042c253e4859b3cbbb50df2f78fa8fae6747a7814484a70", + "sha256:31e44a986ceccec3d0498e16f3d27b2ee5fdf69ce2ab89b52eaad1d2f33d8778", + "sha256:3d9a1eca329405219b605fac09ecfc09ac09e595d6def650a437523fcd08dd22", + "sha256:3dcdedae5c7710b9f97ac6bba7e1052b95c7083c9d0e9df96e02a1932e777895", + "sha256:47ca71115e545954e6c1d207dd13461ab81f4eccfcb1345eac874828b5e3eaaf", + "sha256:4a997df8c1c2aae1e1e5ac49c2e4f610ad037fc5a3aadc7b64e39dea42249431", + "sha256:51956cf8730665e2bdf8ddb8da0056f699c1a5715648c1b0144670c1ba00b48f", + "sha256:5bcb8a5620008a8034d39bce21dc3e23735dfdb6a33a06974739bfa04f853947", + "sha256:64c3f16e2a4fc51c0d06af28441881f98c5d91009b8caaff40cf3548089e9c74", + "sha256:6e2b11c55d260d03a8cf29ac9b5e0608d35f08077d8c087be96287f43af3ccdc", + "sha256:7b3f5fe74a5ca32d4d0f302ffe6680fcc5c28f8ef0dc0ae8f40c0f3a1b4fca66", + "sha256:844b6d608374e7d08f4f6e6f9f7b951f9256db41421917dfb2d003dde4cd6b66", + "sha256:9a8d6802e0825767476f62aafed40532bd435e8a5f7d23bd8b4f5fd04cc80ecf", + "sha256:aae4d918f6b180a8ab8bf6511a419473d107df4dbb4225c7b48c5c9602c38c7f", + "sha256:ac1955ce000cb29ab40def14fd1bbfa7af2017cca696ee696925615cafd0dce5", + "sha256:b88075ada2d51aa9f18283532c9f60e72170041bba88d7f37e49cbb10275299e", + "sha256:cb013933d4c127349b3948aa8aaf2f12c0353ad0eccd715ca789c8a0f671646f", + "sha256:cc70b4b581f28d0a254d006f26949245e3657d40d8857066c2ae22a61222ef55", + "sha256:e9c5266c432a1e23738d178e51c2c7a5e2ddf790f248be939448c0ba2021f9d1", + "sha256:ea9e57f8ea880eeea38ab5abf9fbe39f923544d7884228ec67d666abd60f5a47", + "sha256:ee0c405832ade84d4de74b9029bedb7b31200600fa524d218fc29bfa371e97f5", + "sha256:fdcb265de28585de5b859ae13e3846a8e805268a823a12a4da2597f1f5afc9f0" + ], + "version": "==43.0.0" }, "dataclasses-json": { "hashes": [ @@ -486,11 +482,11 @@ }, "envier": { "hashes": [ - "sha256:b45ef6051fea33d0c32a64e186bff2cfb446e2242d6781216c9bc9ce708c5909", - "sha256:bd5ccf707447973ea0f4125b7df202ba415ad888bcdcb8df80e0b002ee11ffdb" + "sha256:4e7e398cb09a8dd360508ef7e12511a152355426d2544b8487a34dad27cc20ad", + "sha256:65099cf3aa9b3b3b4b92db2f7d29e2910672e085b76f7e587d2167561a834add" ], "markers": "python_version >= '3.7'", - "version": "==0.5.1" + "version": "==0.5.2" }, "fastapi": { "hashes": [ @@ -781,11 +777,11 @@ }, "llama-index-agent-openai": { "hashes": [ - "sha256:13ce535f03e32c821763c01e26af4222f3981178622414d3868013a1946e8124", - "sha256:34be65011a508dd8cab0c9a606594f28075b98b0cebe69e3c543adc8564fee0d" + "sha256:d7f0fd4c87124781acd783be603871f8808b1a3969e876a9c96e2ed0844d46ac", + "sha256:debe86da6d9d983db32b445ddca7c798ac140fe59573bafded73595b3995f3d5" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.2.7" + "version": "==0.2.9" }, "llama-index-cli": { "hashes": [ @@ -805,19 +801,19 @@ }, "llama-index-embeddings-openai": { "hashes": [ - "sha256:1bc1fc9b46773a12870c5d3097d3735d7ca33805f12462a8e35ae8a6e5ce1cf6", - "sha256:c3cfa83b537ded34d035fc172a945dd444c87fb58a89b02dfbf785b675f9f681" + "sha256:6025e229e375201788a9b14d6ebe470329907576cba5f6b7b832c3d68f39db30", + "sha256:e20806fc4baff6b8f5274decf2c1ca7c5c737648e01865475ffada164e32e173" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.10" + "version": "==0.1.11" }, "llama-index-indices-managed-llama-cloud": { "hashes": [ - "sha256:30c73a77fc54fa83c4a183fcdc3b5138a6b709a6fefc9539d0cb0c6315b0f2fc", - "sha256:9a3db075878bc7adf798a74ec4d6220dec5421f46c0675702a94894934d17a7a" + "sha256:35ecc8c4d6098588e911b4bf990f8786765dab94fcd311729de2c2c7b3237bf3", + "sha256:8ee6b11e47f2c5e7302a749b6e2baf893ab602967d86b13c84455b8ee1fb1404" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.2.2" + "version": "==0.2.4" }, "llama-index-legacy": { "hashes": [ @@ -829,19 +825,19 @@ }, "llama-index-llms-openai": { "hashes": [ - "sha256:9031bd155c303f89cc51cfcc75d7d6f12fffa4274f2f9c7f67d4140350d13d56", - "sha256:c7b71cd34765e2d080d5eaf23c602877cc74fea162b59d53965273b2d4c4a56a" + "sha256:08a408cd53af4cd4623dd5807be4cbbd5e5b3ca01272128cd678d667343e4d5d", + "sha256:1ad8e4eb02f9410c2091749d4d9aa9db4452646b595eb5eb937edbc496fb65fe" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.24" + "version": "==0.1.26" }, "llama-index-multi-modal-llms-openai": { "hashes": [ - "sha256:0b6950a6cf98d16ade7d3b9dd0821ecfe457ca103819ae6c3e66cfc9634ca646", - "sha256:10de75a877a444af35306385faad9b9f0624391e55309970564114a080a0578c" + "sha256:16ae72ac3c5201ebd1d4b62203930c1768149ec85c3e477e5e51ed2ef8db1067", + "sha256:5e2c94a6415a2509cad035ccea34461959ae327a5900d3e820417e9ebb9a13ec" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.6" + "version": "==0.1.8" }, "llama-index-program-openai": { "hashes": [ @@ -861,27 +857,27 @@ }, "llama-index-readers-file": { "hashes": [ - "sha256:238ddd98aa377d6a44322013eb848056037c80ad84571ea5bf451a640fff4d5c", - "sha256:bc659e432d441c445e110580340675aa60abae1d82add4f65e559dfe8add541b" + "sha256:32f40465f2a8a65fa5773e03c9f4dd55164be934ae67fad62113680436787d91", + "sha256:d5f6cdd4685ee73103c68b9bc0dfb0d05439033133fc6bd45ef31ff41519e723" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.25" + "version": "==0.1.30" }, "llama-index-readers-llama-parse": { "hashes": [ - "sha256:78608b193c818894aefeee0aa303f02b7f80f2e4caf13866c2fd3b0b1023e2c0", - "sha256:c4914b37d12cceee56fbd185cca80f87d60acbf8ea7a73f9719610180be1fcdd" + "sha256:04f2dcfbb0fb87ce70890f5a2f4f89941d79be6a818b43738f053560e4b451cf", + "sha256:71d445a2357ce4c632e0fada7c913ac62790e77c062f12d916dd86378380ff1f" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.1.4" + "version": "==0.1.6" }, "llama-parse": { "hashes": [ - "sha256:08a48bcf4af5b623bf26fa6266038572b95409f7be64746067db8d38f6927fe5", - "sha256:a68fc91a2b0bce98a4960b8f709ca3c2f90b421da66e0d8522f0ea45b78846b9" + "sha256:657f8fa5f7d399f14c0454fc05cae6034da0373f191df6cfca17a1b4a704ef87", + "sha256:71974a57a73d642608cc406942bee4e7fc1a713fa410f51df67da509479ba544" ], "markers": "python_version < '4.0' and python_full_version >= '3.8.1'", - "version": "==0.4.5" + "version": "==0.4.9" }, "markdown-it-py": { "hashes": [ @@ -891,6 +887,14 @@ "markers": "python_version >= '3.8'", "version": "==3.0.0" }, + "markdownify": { + "hashes": [ + "sha256:1d181d43d20902bcc69d7be85b5316ed174d0dda72ff56e14ae4c95a4a407d22", + "sha256:ab257f9e6bd4075118828a28c9d02f8a4bfeb7421f558834aa79b2dfeb32a098" + ], + "index": "pypi", + "version": "==0.13.1" + }, "markupsafe": { "hashes": [ "sha256:00e046b6dd71aa03a41079792f8473dc494d564611a8f89bbbd7cb93295ebdcf", @@ -1162,55 +1166,60 @@ }, "orjson": { "hashes": [ - "sha256:03b565c3b93f5d6e001db48b747d31ea3819b89abf041ee10ac6988886d18e01", - "sha256:099e81a5975237fda3100f918839af95f42f981447ba8f47adb7b6a3cdb078fa", - "sha256:10c0eb7e0c75e1e486c7563fe231b40fdd658a035ae125c6ba651ca3b07936f5", - "sha256:1146bf85ea37ac421594107195db8bc77104f74bc83e8ee21a2e58596bfb2f04", - "sha256:1670fe88b116c2745a3a30b0f099b699a02bb3482c2591514baf5433819e4f4d", - "sha256:185c394ef45b18b9a7d8e8f333606e2e8194a50c6e3c664215aae8cf42c5385e", - "sha256:1ad1de7fef79736dde8c3554e75361ec351158a906d747bd901a52a5c9c8d24b", - "sha256:235dadefb793ad12f7fa11e98a480db1f7c6469ff9e3da5e73c7809c700d746b", - "sha256:28afa96f496474ce60d3340fe8d9a263aa93ea01201cd2bad844c45cd21f5268", - "sha256:2d97531cdfe9bdd76d492e69800afd97e5930cb0da6a825646667b2c6c6c0211", - "sha256:338fd4f071b242f26e9ca802f443edc588fa4ab60bfa81f38beaedf42eda226c", - "sha256:36a10f43c5f3a55c2f680efe07aa93ef4a342d2960dd2b1b7ea2dd764fe4a37c", - "sha256:3d21b9983da032505f7050795e98b5d9eee0df903258951566ecc358f6696969", - "sha256:51bbcdea96cdefa4a9b4461e690c75ad4e33796530d182bdd5c38980202c134a", - "sha256:53ed1c879b10de56f35daf06dbc4a0d9a5db98f6ee853c2dbd3ee9d13e6f302f", - "sha256:545d493c1f560d5ccfc134803ceb8955a14c3fcb47bbb4b2fee0232646d0b932", - "sha256:584c902ec19ab7928fd5add1783c909094cc53f31ac7acfada817b0847975f26", - "sha256:5a35455cc0b0b3a1eaf67224035f5388591ec72b9b6136d66b49a553ce9eb1e6", - "sha256:5df58d206e78c40da118a8c14fc189207fffdcb1f21b3b4c9c0c18e839b5a214", - "sha256:64c9cc089f127e5875901ac05e5c25aa13cfa5dbbbd9602bda51e5c611d6e3e2", - "sha256:68f85ecae7af14a585a563ac741b0547a3f291de81cd1e20903e79f25170458f", - "sha256:6970ed7a3126cfed873c5d21ece1cd5d6f83ca6c9afb71bbae21a0b034588d96", - "sha256:6b68742c469745d0e6ca5724506858f75e2f1e5b59a4315861f9e2b1df77775a", - "sha256:7a5baef8a4284405d96c90c7c62b755e9ef1ada84c2406c24a9ebec86b89f46d", - "sha256:7d10cc1b594951522e35a3463da19e899abe6ca95f3c84c69e9e901e0bd93d38", - "sha256:85c89131d7b3218db1b24c4abecea92fd6c7f9fab87441cfc342d3acc725d807", - "sha256:8a11d459338f96a9aa7f232ba95679fc0c7cedbd1b990d736467894210205c09", - "sha256:8c13ca5e2ddded0ce6a927ea5a9f27cae77eee4c75547b4297252cb20c4d30e6", - "sha256:9cd684927af3e11b6e754df80b9ffafd9fb6adcaa9d3e8fdd5891be5a5cad51e", - "sha256:b2efbd67feff8c1f7728937c0d7f6ca8c25ec81373dc8db4ef394c1d93d13dc5", - "sha256:b39e006b00c57125ab974362e740c14a0c6a66ff695bff44615dcf4a70ce2b86", - "sha256:b6c8e30adfa52c025f042a87f450a6b9ea29649d828e0fec4858ed5e6caecf63", - "sha256:be79e2393679eda6a590638abda16d167754393f5d0850dcbca2d0c3735cebe2", - "sha256:c05f16701ab2a4ca146d0bca950af254cb7c02f3c01fca8efbbad82d23b3d9d4", - "sha256:c4057c3b511bb8aef605616bd3f1f002a697c7e4da6adf095ca5b84c0fd43595", - "sha256:c4a65310ccb5c9910c47b078ba78e2787cb3878cdded1702ac3d0da71ddc5228", - "sha256:ca0b3a94ac8d3886c9581b9f9de3ce858263865fdaa383fbc31c310b9eac07c9", - "sha256:cc28e90a7cae7fcba2493953cff61da5a52950e78dc2dacfe931a317ee3d8de7", - "sha256:cdf7365063e80899ae3a697def1277c17a7df7ccfc979990a403dfe77bb54d40", - "sha256:d69858c32f09c3e1ce44b617b3ebba1aba030e777000ebdf72b0d8e365d0b2b3", - "sha256:dbead71dbe65f959b7bd8cf91e0e11d5338033eba34c114f69078d59827ee139", - "sha256:dcbe82b35d1ac43b0d84072408330fd3295c2896973112d495e7234f7e3da2e1", - "sha256:dfc91d4720d48e2a709e9c368d5125b4b5899dced34b5400c3837dadc7d6271b", - "sha256:eded5138cc565a9d618e111c6d5c2547bbdd951114eb822f7f6309e04db0fb47", - "sha256:f4324929c2dd917598212bfd554757feca3e5e0fa60da08be11b4aa8b90013c1", - "sha256:fb66215277a230c456f9038d5e2d84778141643207f85336ef8d2a9da26bd7ca" + "sha256:03c95484d53ed8e479cade8628c9cea00fd9d67f5554764a1110e0d5aa2de96e", + "sha256:05ac3d3916023745aa3b3b388e91b9166be1ca02b7c7e41045da6d12985685f0", + "sha256:0943e4c701196b23c240b3d10ed8ecd674f03089198cf503105b474a4f77f21f", + "sha256:1335d4ef59ab85cab66fe73fd7a4e881c298ee7f63ede918b7faa1b27cbe5212", + "sha256:1c680b269d33ec444afe2bdc647c9eb73166fa47a16d9a75ee56a374f4a45f43", + "sha256:227df19441372610b20e05bdb906e1742ec2ad7a66ac8350dcfd29a63014a83b", + "sha256:30b0a09a2014e621b1adf66a4f705f0809358350a757508ee80209b2d8dae219", + "sha256:3722fddb821b6036fd2a3c814f6bd9b57a89dc6337b9924ecd614ebce3271394", + "sha256:446dee5a491b5bc7d8f825d80d9637e7af43f86a331207b9c9610e2f93fee22a", + "sha256:450e39ab1f7694465060a0550b3f6d328d20297bf2e06aa947b97c21e5241fbd", + "sha256:49e3bc615652617d463069f91b867a4458114c5b104e13b7ae6872e5f79d0844", + "sha256:4bbc6d0af24c1575edc79994c20e1b29e6fb3c6a570371306db0993ecf144dc5", + "sha256:5410111d7b6681d4b0d65e0f58a13be588d01b473822483f77f513c7f93bd3b2", + "sha256:55d43d3feb8f19d07e9f01e5b9be4f28801cf7c60d0fa0d279951b18fae1932b", + "sha256:57985ee7e91d6214c837936dc1608f40f330a6b88bb13f5a57ce5257807da143", + "sha256:61272a5aec2b2661f4fa2b37c907ce9701e821b2c1285d5c3ab0207ebd358d38", + "sha256:633a3b31d9d7c9f02d49c4ab4d0a86065c4a6f6adc297d63d272e043472acab5", + "sha256:64c81456d2a050d380786413786b057983892db105516639cb5d3ee3c7fd5148", + "sha256:66680eae4c4e7fc193d91cfc1353ad6d01b4801ae9b5314f17e11ba55e934183", + "sha256:697a35a083c4f834807a6232b3e62c8b280f7a44ad0b759fd4dce748951e70db", + "sha256:6eeb13218c8cf34c61912e9df2de2853f1d009de0e46ea09ccdf3d757896af0a", + "sha256:7275664f84e027dcb1ad5200b8b18373e9c669b2a9ec33d410c40f5ccf4b257e", + "sha256:738dbe3ef909c4b019d69afc19caf6b5ed0e2f1c786b5d6215fbb7539246e4c6", + "sha256:79b9b9e33bd4c517445a62b90ca0cc279b0f1f3970655c3df9e608bc3f91741a", + "sha256:874ce88264b7e655dde4aeaacdc8fd772a7962faadfb41abe63e2a4861abc3dc", + "sha256:95a0cce17f969fb5391762e5719575217bd10ac5a189d1979442ee54456393f3", + "sha256:960db0e31c4e52fa0fc3ecbaea5b2d3b58f379e32a95ae6b0ebeaa25b93dfd34", + "sha256:965a916373382674e323c957d560b953d81d7a8603fbeee26f7b8248638bd48b", + "sha256:9c1c4b53b24a4c06547ce43e5fee6ec4e0d8fe2d597f4647fc033fd205707365", + "sha256:a2debd8ddce948a8c0938c8c93ade191d2f4ba4649a54302a7da905a81f00b56", + "sha256:a6ea7afb5b30b2317e0bee03c8d34c8181bc5a36f2afd4d0952f378972c4efd5", + "sha256:ac3045267e98fe749408eee1593a142e02357c5c99be0802185ef2170086a863", + "sha256:b1ec490e10d2a77c345def52599311849fc063ae0e67cf4f84528073152bb2ba", + "sha256:b6f3d167d13a16ed263b52dbfedff52c962bfd3d270b46b7518365bcc2121eed", + "sha256:bb1f28a137337fdc18384079fa5726810681055b32b92253fa15ae5656e1dddb", + "sha256:bf2fbbce5fe7cd1aa177ea3eab2b8e6a6bc6e8592e4279ed3db2d62e57c0e1b2", + "sha256:c27bc6a28ae95923350ab382c57113abd38f3928af3c80be6f2ba7eb8d8db0b0", + "sha256:c2c116072a8533f2fec435fde4d134610f806bdac20188c7bd2081f3e9e0133f", + "sha256:caff75b425db5ef8e8f23af93c80f072f97b4fb3afd4af44482905c9f588da28", + "sha256:d27456491ca79532d11e507cadca37fb8c9324a3976294f68fb1eff2dc6ced5a", + "sha256:d40f839dddf6a7d77114fe6b8a70218556408c71d4d6e29413bb5f150a692ff7", + "sha256:df25d9271270ba2133cc88ee83c318372bdc0f2cd6f32e7a450809a111efc45c", + "sha256:e060748a04cccf1e0a6f2358dffea9c080b849a4a68c28b1b907f272b5127e9b", + "sha256:e54b63d0a7c6c54a5f5f726bc93a2078111ef060fec4ecbf34c5db800ca3b3a7", + "sha256:ea2977b21f8d5d9b758bb3f344a75e55ca78e3ff85595d248eee813ae23ecdfb", + "sha256:eadc8fd310edb4bdbd333374f2c8fec6794bbbae99b592f448d8214a5e4050c0", + "sha256:f215789fb1667cdc874c1b8af6a84dc939fd802bf293a8334fce185c79cd359b", + "sha256:f710f346e4c44a4e8bdf23daa974faede58f83334289df80bc9cd12fe82573c7", + "sha256:f759503a97a6ace19e55461395ab0d618b5a117e8d0fbb20e70cfd68a47327f2", + "sha256:fb0ee33124db6eaa517d00890fc1a55c3bfe1cf78ba4a8899d71a06f2d6ff5c7", + "sha256:fd502f96bf5ea9a61cbc0b2b5900d0dd68aa0da197179042bdd2be67e51a1e4b" ], "markers": "python_version >= '3.8'", - "version": "==3.10.5" + "version": "==3.10.6" }, "packaging": { "hashes": [ @@ -1490,11 +1499,11 @@ }, "pypdf": { "hashes": [ - "sha256:dc035581664e0ad717e3492acebc1a5fc23dba759e788e3d4a9fc9b1a32e72c1", - "sha256:fe63f3f7d1dcda1c9374421a94c1bba6c6f8c4a62173a59b64ffd52058f846b1" + "sha256:64b31da97eda0771ef22edb1bfecd5deee4b72c3d1736b7df2689805076d6418", + "sha256:b2f37fe9a3030aa97ca86067a56ba3f9d3565f9a791b305c7355d8392c30d91b" ], "markers": "python_version >= '3.6'", - "version": "==4.2.0" + "version": "==4.3.1" }, "python-dateutil": { "hashes": [ @@ -1697,11 +1706,11 @@ }, "setuptools": { "hashes": [ - "sha256:b8b8060bb426838fbe942479c90296ce976249451118ef566a5a0b7d8b78fb05", - "sha256:bd63e505105011b25c3c11f753f7e3b8465ea739efddaccef8f0efac2137bac1" + "sha256:032d42ee9fb536e33087fb66cac5f840eb9391ed05637b3f2a76a7c8fb477936", + "sha256:33874fdc59b3188304b2e7c80d9029097ea31627180896fb549c578ceb8a0855" ], "markers": "python_version >= '3.12'", - "version": "==70.2.0" + "version": "==71.1.0" }, "shellingham": { "hashes": [ @@ -1810,11 +1819,11 @@ }, "tenacity": { "hashes": [ - "sha256:9e6f7cf7da729125c7437222f8a522279751cdfbe6b67bfe64f75d3a348661b2", - "sha256:cd80a53a79336edba8489e767f729e4f391c896956b57140b5d7511a64bbd3ef" + "sha256:8bc6c0c8a09b31e6cad13c47afbed1a567518250a9a171418582ed8d9c20ca78", + "sha256:b594c2a5945830c267ce6b79a166228323ed52718f30302c1359836112346687" ], "markers": "python_version >= '3.8'", - "version": "==8.4.2" + "version": "==8.5.0" }, "tiktoken": { "hashes": [ @@ -1994,11 +2003,11 @@ "standard" ], "hashes": [ - "sha256:cd17daa7f3b9d7a24de3617820e634d0933b69eed8e33a516071174427238c81", - "sha256:d46cd8e0fd80240baffbcd9ec1012a712938754afcf81bce56c024c1656aece8" + "sha256:0d114d0831ff1adbf231d358cbf42f17333413042552a624ea6a9b4c33dcfd81", + "sha256:94a3608da0e530cea8f69683aa4126364ac18e3826b6630d1a65f4638aade503" ], "markers": "python_version >= '3.8'", - "version": "==0.30.1" + "version": "==0.30.3" }, "uvloop": { "hashes": [ @@ -2432,27 +2441,27 @@ }, "botocore": { "hashes": [ - "sha256:7f7135178692b39143c8f152a618d2a3b71065a317569a7102d2306d4946f42f", - "sha256:c63fe9032091fb9e9477706a3ebfa4d0c109b807907051d892ed574f9b573e61" + "sha256:3fd4782362bd29c192704ebf859c5c8c5189ad05719e391eefe23088434427ae", + "sha256:849cb8e54e042443aeabcd7822b5f2b76cb5cfe33fe3a71f91c7c069748a869c" ], "markers": "python_version >= '3.8'", - "version": "==1.34.136" + "version": "==1.34.146" }, "botocore-stubs": { "hashes": [ - "sha256:b97bece44abc9f16c1fcb022d3f03c22d411ea3dcb4730207a318eb8e6ef63e7", - "sha256:f8a8716d9f1b8387acce6ef35d9428ed4ef1584dbeb6cce3f2f95ba6ff8120f7" + "sha256:a7d990c165e3151180a8c1c754183c2f687520117c918bdd6057f26baad4e1d2", + "sha256:f77385f2b6af64b04e00dffc0f684879f308c48ce05deefabbec46b290841fc1" ], "markers": "python_version >= '3.8' and python_version < '4.0'", - "version": "==1.34.136" + "version": "==1.34.146" }, "certifi": { "hashes": [ - "sha256:3cd43f1c6fa7dedc5899d69d3ad0398fd018ad1a17fba83ddaf78aa46c747516", - "sha256:ddc6c8ce995e6987e7faf5e3f1b02b302836a0e5d98ece18392cb1a36c72ad56" + "sha256:5a1e7645bc0ec61a09e26c36f6106dd4cf40c6db3a1fb6352b0244e7fb057c7b", + "sha256:c198e21b1289c2ab85ee4e67bb4b4ef3ead0892059901a8d5b622f24a1101e90" ], "markers": "python_version >= '3.6'", - "version": "==2024.6.2" + "version": "==2024.7.4" }, "cffi": { "hashes": [ @@ -2618,40 +2627,35 @@ }, "cryptography": { "hashes": [ - "sha256:013629ae70b40af70c9a7a5db40abe5d9054e6f4380e50ce769947b73bf3caad", - "sha256:2346b911eb349ab547076f47f2e035fc8ff2c02380a7cbbf8d87114fa0f1c583", - "sha256:2f66d9cd9147ee495a8374a45ca445819f8929a3efcd2e3df6428e46c3cbb10b", - "sha256:2f88d197e66c65be5e42cd72e5c18afbfae3f741742070e3019ac8f4ac57262c", - "sha256:31f721658a29331f895a5a54e7e82075554ccfb8b163a18719d342f5ffe5ecb1", - "sha256:343728aac38decfdeecf55ecab3264b015be68fc2816ca800db649607aeee648", - "sha256:5226d5d21ab681f432a9c1cf8b658c0cb02533eece706b155e5fbd8a0cdd3949", - "sha256:57080dee41209e556a9a4ce60d229244f7a66ef52750f813bfbe18959770cfba", - "sha256:5a94eccb2a81a309806027e1670a358b99b8fe8bfe9f8d329f27d72c094dde8c", - "sha256:6b7c4f03ce01afd3b76cf69a5455caa9cfa3de8c8f493e0d3ab7d20611c8dae9", - "sha256:7016f837e15b0a1c119d27ecd89b3515f01f90a8615ed5e9427e30d9cdbfed3d", - "sha256:81884c4d096c272f00aeb1f11cf62ccd39763581645b0812e99a91505fa48e0c", - "sha256:81d8a521705787afe7a18d5bfb47ea9d9cc068206270aad0b96a725022e18d2e", - "sha256:8d09d05439ce7baa8e9e95b07ec5b6c886f548deb7e0f69ef25f64b3bce842f2", - "sha256:961e61cefdcb06e0c6d7e3a1b22ebe8b996eb2bf50614e89384be54c48c6b63d", - "sha256:9c0c1716c8447ee7dbf08d6db2e5c41c688544c61074b54fc4564196f55c25a7", - "sha256:a0608251135d0e03111152e41f0cc2392d1e74e35703960d4190b2e0f4ca9c70", - "sha256:a0c5b2b0585b6af82d7e385f55a8bc568abff8923af147ee3c07bd8b42cda8b2", - "sha256:ad803773e9df0b92e0a817d22fd8a3675493f690b96130a5e24f1b8fabbea9c7", - "sha256:b297f90c5723d04bcc8265fc2a0f86d4ea2e0f7ab4b6994459548d3a6b992a14", - "sha256:ba4f0a211697362e89ad822e667d8d340b4d8d55fae72cdd619389fb5912eefe", - "sha256:c4783183f7cb757b73b2ae9aed6599b96338eb957233c58ca8f49a49cc32fd5e", - "sha256:c9bb2ae11bfbab395bdd072985abde58ea9860ed84e59dbc0463a5d0159f5b71", - "sha256:cafb92b2bc622cd1aa6a1dce4b93307792633f4c5fe1f46c6b97cf67073ec961", - "sha256:d45b940883a03e19e944456a558b67a41160e367a719833c53de6911cabba2b7", - "sha256:dc0fdf6787f37b1c6b08e6dfc892d9d068b5bdb671198c72072828b80bd5fe4c", - "sha256:dea567d1b0e8bc5764b9443858b673b734100c2871dc93163f58c46a97a83d28", - "sha256:dec9b018df185f08483f294cae6ccac29e7a6e0678996587363dc352dc65c842", - "sha256:e3ec3672626e1b9e55afd0df6d774ff0e953452886e06e0f1eb7eb0c832e8902", - "sha256:e599b53fd95357d92304510fb7bda8523ed1f79ca98dce2f43c115950aa78801", - "sha256:fa76fbb7596cc5839320000cdd5d0955313696d9511debab7ee7278fc8b5c84a", - "sha256:fff12c88a672ab9c9c1cf7b0c80e3ad9e2ebd9d828d955c126be4fd3e5578c9e" - ], - "version": "==42.0.8" + "sha256:0663585d02f76929792470451a5ba64424acc3cd5227b03921dab0e2f27b1709", + "sha256:08a24a7070b2b6804c1940ff0f910ff728932a9d0e80e7814234269f9d46d069", + "sha256:232ce02943a579095a339ac4b390fbbe97f5b5d5d107f8a08260ea2768be8cc2", + "sha256:2905ccf93a8a2a416f3ec01b1a7911c3fe4073ef35640e7ee5296754e30b762b", + "sha256:299d3da8e00b7e2b54bb02ef58d73cd5f55fb31f33ebbf33bd00d9aa6807df7e", + "sha256:2c6d112bf61c5ef44042c253e4859b3cbbb50df2f78fa8fae6747a7814484a70", + "sha256:31e44a986ceccec3d0498e16f3d27b2ee5fdf69ce2ab89b52eaad1d2f33d8778", + "sha256:3d9a1eca329405219b605fac09ecfc09ac09e595d6def650a437523fcd08dd22", + "sha256:3dcdedae5c7710b9f97ac6bba7e1052b95c7083c9d0e9df96e02a1932e777895", + "sha256:47ca71115e545954e6c1d207dd13461ab81f4eccfcb1345eac874828b5e3eaaf", + "sha256:4a997df8c1c2aae1e1e5ac49c2e4f610ad037fc5a3aadc7b64e39dea42249431", + "sha256:51956cf8730665e2bdf8ddb8da0056f699c1a5715648c1b0144670c1ba00b48f", + "sha256:5bcb8a5620008a8034d39bce21dc3e23735dfdb6a33a06974739bfa04f853947", + "sha256:64c3f16e2a4fc51c0d06af28441881f98c5d91009b8caaff40cf3548089e9c74", + "sha256:6e2b11c55d260d03a8cf29ac9b5e0608d35f08077d8c087be96287f43af3ccdc", + "sha256:7b3f5fe74a5ca32d4d0f302ffe6680fcc5c28f8ef0dc0ae8f40c0f3a1b4fca66", + "sha256:844b6d608374e7d08f4f6e6f9f7b951f9256db41421917dfb2d003dde4cd6b66", + "sha256:9a8d6802e0825767476f62aafed40532bd435e8a5f7d23bd8b4f5fd04cc80ecf", + "sha256:aae4d918f6b180a8ab8bf6511a419473d107df4dbb4225c7b48c5c9602c38c7f", + "sha256:ac1955ce000cb29ab40def14fd1bbfa7af2017cca696ee696925615cafd0dce5", + "sha256:b88075ada2d51aa9f18283532c9f60e72170041bba88d7f37e49cbb10275299e", + "sha256:cb013933d4c127349b3948aa8aaf2f12c0353ad0eccd715ca789c8a0f671646f", + "sha256:cc70b4b581f28d0a254d006f26949245e3657d40d8857066c2ae22a61222ef55", + "sha256:e9c5266c432a1e23738d178e51c2c7a5e2ddf790f248be939448c0ba2021f9d1", + "sha256:ea9e57f8ea880eeea38ab5abf9fbe39f923544d7884228ec67d666abd60f5a47", + "sha256:ee0c405832ade84d4de74b9029bedb7b31200600fa524d218fc29bfa371e97f5", + "sha256:fdcb265de28585de5b859ae13e3846a8e805268a823a12a4da2597f1f5afc9f0" + ], + "version": "==43.0.0" }, "docker": { "hashes": [ @@ -2867,11 +2871,20 @@ }, "types-awscrt": { "hashes": [ - "sha256:0beabdde0205dc1da679ea464fd3f98b570ef4f0fc825b155a974fb51b21e8d9", - "sha256:521ce54cc4dad9fe6480556bb0f8315a508106938ba1f2a0baccfcea7d4a4dee" + "sha256:0839fe12f0f914d8f7d63ed777c728cb4eccc2d5d79a26e377d12b0604e7bf0e", + "sha256:84a9f4f422ec525c314fdf54c23a1e73edfbcec968560943ca2d41cfae623b38" ], "markers": "python_version >= '3.7' and python_version < '4.0'", - "version": "==0.20.12" + "version": "==0.21.2" + }, + "types-beautifulsoup4": { + "hashes": [ + "sha256:004f6096fdd83b19cdbf6cb10e4eae57b10205eccc365d0a69d77da836012e28", + "sha256:7ceda66a93ba28d759d5046d7fec9f4cad2f563a77b3a789efc90bcadafeefd1" + ], + "index": "pypi", + "markers": "python_version >= '3.8'", + "version": "==4.12.0.20240511" }, "types-docker": { "hashes": [ @@ -2882,14 +2895,22 @@ "markers": "python_version >= '3.8'", "version": "==7.0.0.20240528" }, + "types-html5lib": { + "hashes": [ + "sha256:22736b7299e605ec4ba539d48691e905fd0c61c3ea610acc59922232dc84cede", + "sha256:af5de0125cb0fe5667543b158db83849b22e25c0e36c9149836b095548bf1020" + ], + "markers": "python_version >= '3.8'", + "version": "==1.1.11.20240228" + }, "types-requests": { "hashes": [ - "sha256:97bac6b54b5bd4cf91d407e62f0932a74821bc2211f22116d9ee1dd643826caf", - "sha256:ed5e8a412fcc39159d6319385c009d642845f250c63902718f605cd90faade31" + "sha256:90c079ff05e549f6bf50e02e910210b98b8ff1ebdd18e19c873cd237737c1358", + "sha256:f754283e152c752e46e70942fa2a146b5bc70393522257bb85bd1ef7e019dcc3" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==2.32.0.20240622" + "version": "==2.32.0.20240712" }, "types-s3transfer": { "hashes": [ diff --git a/opentrons-ai-server/api/data/python_api_219_docs.md b/opentrons-ai-server/api/data/python_api_219_docs.md new file mode 100644 index 00000000000..31d3ce47404 --- /dev/null +++ b/opentrons-ai-server/api/data/python_api_219_docs.md @@ -0,0 +1,6212 @@ +# [Python Protocol API v2](#) + +Python Protocol API + +### Table of Contents + +- [Welcome](index.html#document-index) +- [Tutorial](index.html#document-tutorial) +- [Versioning](index.html#document-versioning) +- [Labware](index.html#document-new_labware) +- [Moving Labware](index.html#document-moving_labware) +- [Hardware Modules](index.html#document-new_modules) +- [Deck Slots](index.html#document-deck_slots) +- [Pipettes](index.html#document-new_pipette) +- [Building Block Commands](index.html#document-new_atomic_commands) +- [Complex Commands](index.html#document-new_complex_commands) +- [Labware and Deck Positions](index.html#document-robot_position) +- [Runtime Parameters](index.html#document-runtime_parameters) +- [Advanced Control](index.html#document-new_advanced_running) +- [Protocol Examples](index.html#document-new_examples) +- [Adapting OT\-2 Protocols for Flex](index.html#document-adapting_ot2_flex) +- [API Version 2 Reference](index.html#document-new_protocol_api) + +--- + +- [OT\-2 Python API v1](../v1/index.html) + +### Related Topics + +- [Documentation overview](#) + +# Welcome + +## Tutorial + +### Introduction + +This tutorial will guide you through creating a Python protocol file from scratch. At the end of this process you’ll have a complete protocol that can run on a Flex or an OT\-2 robot. If you don’t have a Flex or an OT\-2 (or if you’re away from your lab, or if your robot is in use), you can use the same file to simulate the protocol on your computer instead. + +#### What You’ll Automate + +The lab task that you’ll automate in this tutorial is serial dilution: taking a solution and progressively diluting it by transferring it stepwise across a plate from column 1 to column 12\. With just a dozen or so lines of code, you can instruct your robot to perform the hundreds of individual pipetting actions necessary to fill an entire 96\-well plate. And all of those liquid transfers will be done automatically, so you’ll have more time to do other work in your lab. + +#### Before You Begin + +You’re going to write some Python code, but you don’t need to be a Python expert to get started writing Opentrons protocols. You should know some basic Python syntax, like how it uses [indentation](https://docs.python.org/3/reference/lexical_analysis.html#indentation) to group blocks of code, dot notation for [calling methods](https://docs.python.org/3/tutorial/classes.html#method-objects), and the format of [lists](https://docs.python.org/3/tutorial/introduction.html#lists) and [dictionaries](https://docs.python.org/3/tutorial/datastructures.html#dictionaries). You’ll also be using [common control structures](https://docs.python.org/3/tutorial/controlflow.html#if-statements) like `if` statements and `for` loops. + +You should write your code in your favorite plaintext editor or development environment and save it in a file with a `.py` extension, like `dilution-tutorial.py`. + +To simulate your code, you’ll need [Python 3\.10](https://www.python.org/downloads/) and the [pip package installer](https://pip.pypa.io/en/stable/getting-started/). Newer versions of Python aren’t yet supported by the Python Protocol API. If you don’t use Python 3\.10 as your system Python, we recommend using [pyenv](https://github.com/pyenv/pyenv) to manage multiple Python versions. + +#### Hardware and Labware + +Before running a protocol, you’ll want to have the right kind of hardware and labware ready for your Flex or OT\-2\. + +- **Flex users** should review Chapter 2: Installation and Relocation in the [instruction manual](https://insights.opentrons.com/hubfs/Products/Flex/Opentrons%20Flex%20Manual.pdf). Specifically, see the pipette information in the “Instrument Installation and Calibration” section. You can use either a 1\-channel or 8\-channel pipette for this tutorial. Most Flex code examples will use a [Flex 1\-Channel 1000 μL pipette](https://shop.opentrons.com/opentrons-flex-1-channel-pipette/). +- **OT\-2 users** should review the robot setup and pipette information on the [Get Started page](https://support.opentrons.com/s/ot2-get-started). Specifically, see [attaching pipettes](https://support.opentrons.com/s/article/Get-started-Attach-pipettes) and [initial calibration](https://support.opentrons.com/s/article/Get-started-Calibrate-the-deck). You can use either a single\-channel or 8\-channel pipette for this tutorial. Most OT\-2 code examples will use a [P300 Single\-Channel GEN2](https://shop.opentrons.com/single-channel-electronic-pipette-p20/) pipette. + +The Flex and OT\-2 use similar labware for serial dilution. The tutorial code will use the labware listed in the table below, but as long as you have labware of each type you can modify the code to run with your labware. + +| Labware type | Labware name | API load name | +| -------------- | ----------------------------------------------------------------------------------------------- | --------------------------------- | +| Reservoir | [NEST 12 Well Reservoir 15 mL](https://labware.opentrons.com/nest_12_reservoir_15ml) | `nest_12_reservoir_15ml` | +| Well plate | [NEST 96 Well Plate 200 µL Flat](https://labware.opentrons.com/nest_96_wellplate_200ul_flat) | `nest_96_wellplate_200ul_flat` | +| Flex tip rack | [Opentrons Flex Tips, 200 µL](https://shop.opentrons.com/opentrons-flex-tips-200-l/) | `opentrons_flex_96_tiprack_200ul` | +| OT\-2 tip rack | [Opentrons 96 Tip Rack](https://labware.opentrons.com/?category=tipRack&manufacturer=Opentrons) | `opentrons_96_tiprack_300ul` | + +For the liquids, you can use plain water as the diluent and water dyed with food coloring as the solution. + +### Create a Protocol File + +Let’s start from scratch to create your serial dilution protocol. Open up a new file in your editor and start with the line: + +``` +from opentrons import protocol_api + +``` + +Throughout this documentation, you’ll see protocols that begin with the `import` statement shown above. It identifies your code as an Opentrons protocol. This statement is not required, but including it is a good practice and allows most code editors to provide helpful autocomplete suggestions. + +Everything else in the protocol file is required. Next, you’ll specify the version of the API you’re using. Then comes the core of the protocol: defining a single `run()` function that provides the locations of your labware, states which kind of pipettes you’ll use, and finally issues the commands that the robot will perform. + +For this tutorial, you’ll write very little Python outside of the `run()` function. But for more complex applications it’s worth remembering that your protocol file _is_ a Python script, so any Python code that can run on your robot can be a part of a protocol. + +#### Metadata + +Every protocol needs to have a metadata dictionary with information about the protocol. At minimum, you need to specify what [version of the API](index.html#version-table) the protocol requires. The [scripts](https://github.com/Opentrons/opentrons/blob/edge/api/docs/v2/example_protocols/) for this tutorial were validated against API version 2\.16, so specify: + +``` +metadata = {"apiLevel": "2.16"} + +``` + +You can include any other information you like in the metadata dictionary. The fields `protocolName`, `description`, and `author` are all displayed in the Opentrons App, so it’s a good idea to expand the dictionary to include them: + +``` +metadata = { + "apiLevel": "2.16", + "protocolName": "Serial Dilution Tutorial", + "description": """This protocol is the outcome of following the + Python Protocol API Tutorial located at + https://docs.opentrons.com/v2/tutorial.html. It takes a + solution and progressively dilutes it by transferring it + stepwise across a plate.""", + "author": "New API User" + } + +``` + +Note, if you have a Flex, or are using an OT\-2 with API v2\.15 (or higher), we recommend adding a `requirements` section to your code. See the Requirements section below. + +#### Requirements + +The `requirements` code block can appear before _or_ after the `metadata` code block in a Python protocol. It uses the following syntax and accepts two arguments: `robotType` and `apiLevel`. + +Whether you need a `requirements` block depends on your robot model and API version. + +- **Flex:** The `requirements` block is always required. And, the API version does not go in the `metadata` section. The API version belongs in the `requirements`. For example: + +``` +requirements = {"robotType": "Flex", "apiLevel": "2.16"} + +``` + +- **OT\-2:** The `requirements` block is optional, but including it is a recommended best practice, particularly if you’re using API version 2\.15 or greater. If you do use it, remember to remove the API version from the `metadata`. For example: + +``` +requirements = {"robotType": "OT-2", "apiLevel": "2.16"} + +``` + +With the metadata and requirements defined, you can move on to creating the `run()` function for your protocol. + +#### The `run()` function + +Now it’s time to actually instruct the Flex or OT\-2 how to perform serial dilution. All of this information is contained in a single Python function, which has to be named `run`. This function takes one argument, which is the _protocol context_. Many examples in these docs use the argument name `protocol`, and sometimes they specify the argument’s type: + +``` +def run(protocol: protocol_api.ProtocolContext): + +``` + +With the protocol context argument named and typed, you can start calling methods on `protocol` to add labware and hardware. + +##### Labware + +For serial dilution, you need to load a tip rack, reservoir, and 96\-well plate on the deck of your Flex or OT\-2\. Loading labware is done with the [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') method of the protocol context, which takes two arguments: the standard labware name as defined in the [Opentrons Labware Library](https://labware.opentrons.com/), and the position where you’ll place the labware on the robot’s deck. + +### Flex + +Here’s how to load the labware on a Flex in slots D1, D2, and D3 (repeating the `def` statement from above to show proper indenting): + +``` +def run(protocol: protocol_api.ProtocolContext): + tips = protocol.load_labware("opentrons_flex_96_tiprack_200ul", "D1") + reservoir = protocol.load_labware("nest_12_reservoir_15ml", "D2") + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", "D3") + +``` + +If you’re using a different model of labware, find its name in the Labware Library and replace it in your code. + +Now the robot will expect to find labware in a configuration that looks like this: + +### OT-2 + +Here’s how to load the labware on an OT\-2 in slots 1, 2, and 3 (repeating the `def` statement from above to show proper indenting): + +``` +def run(protocol: protocol_api.ProtocolContext): + tips = protocol.load_labware("opentrons_96_tiprack_300ul", 1) + reservoir = protocol.load_labware("nest_12_reservoir_15ml", 2) + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", 3) + +``` + +If you’re using a different model of labware, find its name in the Labware Library and replace it in your code. + +Now the robot will expect to find labware in a configuration that looks like this: + +You may notice that these deck maps don’t show where the liquids will be at the start of the protocol. Liquid definitions aren’t required in Python protocols, unlike protocols made in [Protocol Designer](https://designer.opentrons.com/). If you want to identify liquids, see [Labeling Liquids in Wells](https://docs.opentrons.com/v2/new_labware.html#labeling-liquids-in-wells). (Sneak peek: you’ll put the diluent in column 1 of the reservoir and the solution in column 2 of the reservoir.) + +##### Trash Bin + +Flex and OT\-2 both come with a trash bin for disposing used tips. + +The OT\-2 trash bin is fixed in slot 12\. Since it can’t go anywhere else on the deck, you don’t need to write any code to tell the API where it is. Skip ahead to the Pipettes section below. + +Flex lets you put a [trash bin](index.html#configure-trash-bin) in multiple locations on the deck. You can even have more than one trash bin, or none at all (if you use the [waste chute](index.html#configure-waste-chute) instead, or if your protocol never trashes any tips). For serial dilution, you’ll need to dispose used tips, so you also need to tell the API where the trash container is located on your robot. Loading a trash bin on Flex is done with the [`load_trash_bin()`](index.html#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin') method, which takes one argument: its location. Here’s how to load the trash in slot A3: + +``` +trash = protocol.load_trash_bin("A3") + +``` + +##### Pipettes + +Next you’ll specify what pipette to use in the protocol. Loading a pipette is done with the [`load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') method, which takes three arguments: the name of the pipette, the mount it’s installed in, and the tip racks it should use when performing transfers. Load whatever pipette you have installed in your robot by using its [standard pipette name](index.html#new-pipette-models). Here’s how to load the pipette in the left mount and instantiate it as a variable named `left_pipette`: + +``` +# Flex +left_pipette = protocol.load_instrument("flex_1channel_1000", "left", tip_racks=[tips]) + +``` + +``` +# OT-2 +left_pipette = protocol.load_instrument("p300_single_gen2", "left", tip_racks=[tips]) + +``` + +Since the pipette is so fundamental to the protocol, it might seem like you should have specified it first. But there’s a good reason why pipettes are loaded after labware: you need to have already loaded `tips` in order to tell the pipette to use it. And now you won’t have to reference `tips` again in your code — it’s assigned to the `left_pipette` and the robot will know to use it when commanded to pick up tips. + +Note + +You may notice that the value of `tip_racks` is in brackets, indicating that it’s a list. This serial dilution protocol only uses one tip rack, but some protocols require more tips, so you can assign them to a pipette all at once, like `tip_racks=[tips1, tips2]`. + +##### Commands + +Finally, all of your labware and hardware is in place, so it’s time to give the robot pipetting commands. The required steps of the serial dilution process break down into three main phases: + +1. Measure out equal amounts of diluent from the reservoir to every well on the plate. +2. Measure out equal amounts of solution from the reservoir into wells in the first column of the plate. +3. Move a portion of the combined liquid from column 1 to 2, then from column 2 to 3, and so on all the way to column 12\. + +Thanks to the flexibility of the API’s [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') method, which combines many [building block commands](index.html#v2-atomic-commands) into one call, each of these phases can be accomplished with a single line of code! You’ll just have to write a few more lines of code to repeat the process for as many rows as you want to fill. + +Let’s start with the diluent. This phase takes a larger quantity of liquid and spreads it equally to many wells. `transfer()` can handle this all at once, because it accepts either a single well or a list of wells for its source and destination: + +``` +left_pipette.transfer(100, reservoir["A1"], plate.wells()) + +``` + +Breaking down these single lines of code shows the power of [complex commands](index.html#v2-complex-commands). The first argument is the amount to transfer to each destination, 100 µL. The second argument is the source, column 1 of the reservoir (which is still specified with grid\-style coordinates as `A1` — a reservoir only has an A row). The third argument is the destination. Here, calling the [`wells()`](index.html#opentrons.protocol_api.Labware.wells 'opentrons.protocol_api.Labware.wells') method of `plate` returns a list of _every well_, and the command will apply to all of them. + +In plain English, you’ve instructed the robot, “For every well on the plate, aspirate 100 µL of fluid from column 1 of the reservoir and dispense it in the well.” That’s how we understand this line of code as scientists, yet the robot will understand and execute it as nearly 200 discrete actions. + +Now it’s time to start mixing in the solution. To do this row by row, nest the commands in a `for` loop: + +``` +for i in range(8): + row = plate.rows()[i] + +``` + +Using Python’s built\-in [`range`](https://docs.python.org/3/library/stdtypes.html#range '(in Python v3.12)') class is an easy way to repeat this block 8 times, once for each row. This also lets you use the repeat index `i` with `plate.rows()` to keep track of the current row. + +In each row, you first need to add solution. This will be similar to what you did with the diluent, but putting it only in column 1 of the plate. It’s best to mix the combined solution and diluent thoroughly, so add the optional `mix_after` argument to `transfer()`: + +``` +left_pipette.transfer(100, reservoir["A2"], row[0], mix_after=(3, 50)) + +``` + +As before, the first argument specifies to transfer 100 µL. The second argument is the source, column 2 of the reservoir. The third argument is the destination, the element at index 0 of the current `row`. Since Python lists are zero\-indexed, but columns on labware start numbering at 1, this will be well A1 on the first time through the loop, B1 the second time, and so on. The fourth argument specifies to mix 3 times with 50 µL of fluid each time. + +Finally, it’s time to dilute the solution down the row. One approach would be to nest another `for` loop here, but instead let’s use another feature of the `transfer()` method, taking lists as the source and destination arguments: + +``` +left_pipette.transfer(100, row[:11], row[1:], mix_after=(3, 50)) + +``` + +There’s some Python shorthand here, so let’s unpack it. You can get a range of indices from a list using the colon `:` operator, and omitting it at either end means “from the beginning” or “until the end” of the list. So the source is `row[:11]`, from the beginning of the row until its 11th item. And the destination is `row[1:]`, from index 1 (column 2!) until the end. Since both of these lists have 11 items, `transfer()` will _step through them in parallel_, and they’re constructed so when the source is 0, the destination is 1; when the source is 1, the destination is 2; and so on. This condenses all of the subsequent transfers down the row into a single line of code. + +All that remains is for the loop to repeat these steps, filling each row down the plate. + +That’s it! If you’re using a single\-channel pipette, you’re ready to try out your protocol. + +##### 8\-Channel Pipette + +If you’re using an 8\-channel pipette, you’ll need to make a couple tweaks to the single\-channel code from above. Most importantly, whenever you target a well in row A of a plate with an 8\-channel pipette, it will move its topmost tip to row A, lining itself up over the entire column. + +Thus, when adding the diluent, instead of targeting every well on the plate, you should only target the top row: + +``` +left_pipette.transfer(100, reservoir["A1"], plate.rows()[0]) + +``` + +And by accessing an entire column at once, the 8\-channel pipette effectively implements the `for` loop in hardware, so you’ll need to remove it: + +``` +row = plate.rows()[0] +left_pipette.transfer(100, reservoir["A2"], row[0], mix_after=(3, 50)) +left_pipette.transfer(100, row[:11], row[1:], mix_after=(3, 50)) + +``` + +Instead of tracking the current row in the `row` variable, this code sets it to always be row A (index 0\). + +### Try Your Protocol + +There are two ways to try out your protocol: simulation on your computer, or a live run on a Flex or OT\-2\. Even if you plan to run your protocol on a robot, it’s a good idea to check the simulation output first. + +If you get any errors in simulation, or you don’t get the outcome you expected when running your protocol, you can check your code against these reference protocols on GitHub: + +- [Flex: Single\-channel serial dilution](https://github.com/Opentrons/opentrons/blob/edge/api/docs/v2/example_protocols/dilution_tutorial_flex.py) +- [Flex: 8\-channel serial dilution](https://github.com/Opentrons/opentrons/blob/edge/api/docs/v2/example_protocols/dilution_tutorial_multi_flex.py) +- [OT\-2: Single\-channel serial dilution](https://github.com/Opentrons/opentrons/blob/edge/api/docs/v2/example_protocols/dilution_tutorial.py) +- [OT\-2: 8\-channel serial dilution](https://github.com/Opentrons/opentrons/blob/edge/api/docs/v2/example_protocols/dilution_tutorial_multi.py) + +#### In Simulation + +Simulation doesn’t require having a robot connected to your computer. You just need to install the [Opentrons Python module](https://pypi.org/project/opentrons/) using pip (`pip install opentrons`). This will give you access to the `opentrons_simulate` command\-line utility (`opentrons_simulate.exe` on Windows). + +To see a text preview of the steps your Flex or OT\-2 will take, use the change directory (`cd`) command to navigate to the location of your saved protocol file and run: + +``` +opentrons_simulate dilution-tutorial.py + +``` + +This should generate a lot of output! As written, the protocol has about 1000 steps. In fact, using a single\-channel pipette for serial dilution across the whole plate will take about half an hour — plenty of time to grab a coffee while your robot pipettes for you! ☕️ + +If that’s too long, you can always cancel your run partway through or modify `for i in range(8)` to loop through fewer rows. + +#### On a Robot + +The simplest way to run your protocol on a Flex or OT\-2 is to use the [Opentrons App](https://opentrons.com/ot-app). When you first launch the Opentrons App, you will see the Protocols screen. (Click **Protocols** in the left sidebar to access it at any other time.) Click **Import** in the top right corner to reveal the Import a Protocol pane. Then click **Choose File** and find your protocol in the system file picker, or drag and drop your protocol file into the well. + +You should see “Protocol \- Serial Dilution Tutorial” (or whatever `protocolName` you entered in the metadata) in the list of protocols. Click the three\-dot menu (⋮) for your protocol and choose **Start setup**. + +If you have any remaining calibration tasks to do, you can finish them up here. Below the calibration section is a preview of the initial deck state. Optionally you can run Labware Position Check, or you can go ahead and click **Proceed to Run**. + +On the Run tab, you can double\-check the Run Preview, which is similar to the command\-line simulation output. Make sure all your labware and liquids are in the right place, and then click **Start run**. The run log will update in real time as your robot proceeds through the steps. + +When it’s all done, check the results of your serial dilution procedure — you should have a beautiful dye gradient running across the plate! + +### Next Steps + +This tutorial has relied heavily on the `transfer()` method, but there’s much more that the Python Protocol API can do. Many advanced applications use [building block commands](index.html#v2-atomic-commands) for finer control over the robot. These commands let you aspirate and dispense separately, add air gaps, blow out excess liquid, move the pipette to any location, and more. For protocols that use [Opentrons hardware modules](index.html#new-modules), there are methods to control their behavior. And all of the API’s classes and methods are catalogued in the [API Reference](index.html#protocol-api-reference). + +## Versioning + +The Python Protocol API has its own versioning system, which is separate from the versioning system used for the robot software and the Opentrons App. This allows protocols to run on newer robot software versions without modification. + +### Major and Minor Versions + +The API uses a major and minor version number and does not use patch version numbers. For instance, major version 2 and minor version 0 is written as `2.0`. Versions are not decimal numbers, so `2.10` indicates major version 2 and minor version 10\. The Python Protocol API version will only increase based on changes that affect protocol behavior. + +The major version of the API increases whenever there are significant structural or behavioral changes to protocols. For instance, major version 2 of the API was introduced because it required protocols to have a `run` function that takes a `protocol` argument rather than importing the `robot`, `instruments`, and `labware` modules. Protocols written with major version 1 of the API will not run without modification in major version 2\. A similar level of structural change would require a major version 3\. This documentation only deals with features found in major version 2 of the API; see the [archived version 1 documentation](https://docs.opentrons.com/v1/index.html) for information on older protocols. + +The minor version of the API increases whenever there is new functionality that might change the way a protocol is written, or when a behavior changes in one aspect of the API but does not affect all protocols. For instance, adding support for a new hardware module, adding new parameters for a function, or deprecating a feature would increase the minor version of the API. + +### Specifying Versions + +You must specify the API version you are targeting in your Python protocol. In all minor versions, you can do this with the `apiLevel` key in the `metadata` dictionary, alongside any other metadata elements: + +``` + from opentrons import protocol_api + + metadata = { + "apiLevel": "2.19", + "author": "A. Biologist"} + + def run(protocol: protocol_api.ProtocolContext): + protocol.comment("Hello, world!") + +``` + +From version 2\.15 onward, you can specify `apiLevel` in the `requirements` dictionary instead: + +``` + from opentrons import protocol_api + + metadata = {"author": "A. Biologist"} + requirements = {"apiLevel": "2.19", "robotType": "Flex"} + + def run(protocol: protocol_api.ProtocolContext): + protocol.comment("Hello, Flex!") + +``` + +Choose only one of these places to specify `apiLevel`. If you put it in neither or both places, you will not be able to simulate or run your protocol. + +The version you specify determines the features and behaviors available to your protocol. For example, support for the Heater\-Shaker Module was added in version 2\.13, so you can’t specify a lower version and then call `HeaterShakerContext` methods without causing an error. This protects you from accidentally using features not present in your specified API version, and keeps your protocol portable between API versions. + +When choosing an API level, consider what features you need and how widely you plan to share your protocol. Throughout the Python Protocol API documentation, there are version statements indicating when elements (features, function calls, available properties, etc.) were introduced. Keep these in mind when specifying your protocol’s API version. Version statements look like this: + +New in version 2\.0\. + +On the one hand, using the highest available version will give your protocol access to all the latest [features and fixes](#version-notes). On the other hand, using the lowest possible version lets the protocol work on a wider range of robot software versions. For example, a protocol that uses the Heater\-Shaker and specifies version 2\.13 of the API should work equally well on a robot running version 6\.1\.0 or 6\.2\.0 of the robot software. Specifying version 2\.14 would limit the protocol to robots running 6\.2\.0 or higher. + +### Maximum Supported Versions + +The maximum supported API version for your robot is listed in the Opentrons App under **Robots** \> your robot \> **Robot Settings** \> **Advanced**. Before version 6\.0\.0 of the app, the same information was listed on your robot’s **Information** card. + +If you upload a protocol that specifies a higher API level than the maximum supported, your robot won’t be able to analyze or run your protocol. You can increase the maximum supported version by updating your robot software and Opentrons App. + +Opentrons robots running the latest software (7\.3\.0\) support the following version ranges: + +> - **Flex:** version 2\.15–2\.19\. +> - **OT\-2:** versions 2\.0–2\.19\. + +### API and Robot Software Versions + +This table lists the correspondence between Protocol API versions and robot software versions. + +| API Version | Introduced in Robot Software | +| ----------- | ---------------------------- | +| 2\.19 | 7\.3\.1 | +| 2\.18 | 7\.3\.0 | +| 2\.17 | 7\.2\.0 | +| 2\.16 | 7\.1\.0 | +| 2\.15 | 7\.0\.0 | +| 2\.14 | 6\.3\.0 | +| 2\.13 | 6\.1\.0 | +| 2\.12 | 5\.0\.0 | +| 2\.11 | 4\.4\.0 | +| 2\.10 | 4\.3\.0 | +| 2\.9 | 4\.1\.0 | +| 2\.8 | 4\.0\.0 | +| 2\.7 | 3\.21\.0 | +| 2\.6 | 3\.20\.0 | +| 2\.5 | 3\.19\.0 | +| 2\.4 | 3\.17\.1 | +| 2\.3 | 3\.17\.0 | +| 2\.2 | 3\.16\.0 | +| 2\.1 | 3\.15\.2 | +| 2\.0 | 3\.14\.0 | +| 1\.0 | 3\.0\.0 | + +### Changes in API Versions + +#### Version 2\.19 + +Opentrons recommends updating protocols from `apiLevel` 2\.18 to 2\.19 to take advantage of improved pipetting behavior. + +- This version uses new values for how much a tip overlaps with the pipette nozzle when the pipette picks up tips. This can correct errors caused by the robot positioning the tip slightly lower than intended, potentially making contact with labware. See [`pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') for additional details. + +#### Version 2\.18 + +- Define customizable parameters with the new `add_parameters()` function, and access their values on the [`ProtocolContext.params`](index.html#opentrons.protocol_api.ProtocolContext.params 'opentrons.protocol_api.ProtocolContext.params') object during a protocol run. See [Runtime Parameters](index.html#runtime-parameters) and related pages for more information. +- Move the pipette to positions relative to the top of a trash container. See [Position Relative to Trash Containers](index.html#position-relative-trash). The default behavior of [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') also accounts for this new possibility. +- [`set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset') has been restored to the API with new behavior that applies to labware type–location pairs. +- Automatic tip tracking is now available for all nozzle configurations. + +#### Version 2\.17 + +- [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') now raises an error if you try to dispense more than [`InstrumentContext.current_volume`](index.html#opentrons.protocol_api.InstrumentContext.current_volume 'opentrons.protocol_api.InstrumentContext.current_volume'). + +#### Version 2\.16 + +This version introduces new features for Flex and adds and improves methods for aspirating and dispensing. Note that when updating Flex protocols to version 2\.16, you _must_ load a trash container before dropping tips. + +- New features + + - Use [`configure_nozzle_layout()`](index.html#opentrons.protocol_api.InstrumentContext.configure_nozzle_layout 'opentrons.protocol_api.InstrumentContext.configure_nozzle_layout') to pick up a single column of tips with the 96\-channel pipette. See [Partial Tip Pickup](index.html#partial-tip-pickup). + - Specify the trash containers attached to your Flex with [`load_waste_chute()`](index.html#opentrons.protocol_api.ProtocolContext.load_waste_chute 'opentrons.protocol_api.ProtocolContext.load_waste_chute') and [`load_trash_bin()`](index.html#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin'). + - Dispense, blow out, drop tips, and dispose labware in the waste chute. Disposing labware requires the gripper and calling [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') with `use_gripper=True`. + - Perform actions in staging area slots by referencing slots A4 through D4\. See [Deck Slots](index.html#deck-slots). + - Explicitly command a pipette to [`prepare_to_aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.prepare_to_aspirate 'opentrons.protocol_api.InstrumentContext.prepare_to_aspirate'). The API usually prepares pipettes to aspirate automatically, but this is useful for certain applications, like pre\-wetting routines. + +- Improved features + + - [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'), [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'), and [`mix()`](index.html#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix') will not move any liquid when called with `volume=0`. + +- Other changes + + - [`ProtocolContext.fixed_trash`](index.html#opentrons.protocol_api.ProtocolContext.fixed_trash 'opentrons.protocol_api.ProtocolContext.fixed_trash') and [`InstrumentContext.trash_container`](index.html#opentrons.protocol_api.InstrumentContext.trash_container 'opentrons.protocol_api.InstrumentContext.trash_container') now return [`TrashBin`](index.html#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin') objects instead of [`Labware`](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') objects. + - Flex will no longer automatically drop tips in the trash at the end of a protocol. You can add a [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') command to your protocol or use the Opentrons App to drop the tips. + +#### Version 2\.15 + +This version introduces support for the Opentrons Flex robot, instruments, modules, and labware. + +- Flex features + + - Write protocols for Opentrons Flex by declaring `"robotType": "Flex"` in the new `requirements` dictionary. See the [examples in the Tutorial](index.html#tutorial-requirements). + - [`load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') supports loading Flex 1\-, 8\-, and 96\-channel pipettes. See [Loading Pipettes](index.html#new-create-pipette). + - The new [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') method can move labware automatically using the Flex Gripper. You can also move labware manually on Flex. + - [`load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') supports loading the [Magnetic Block](index.html#magnetic-block). + - The API does not enforce placement restrictions for the Heater\-Shaker module on Flex, because it is installed below\-deck in a module caddy. Pipetting restrictions are still in place when the Heater\-Shaker is shaking or its labware latch is open. + - The new [`configure_for_volume()`](index.html#opentrons.protocol_api.InstrumentContext.configure_for_volume 'opentrons.protocol_api.InstrumentContext.configure_for_volume') method can place Flex 50 µL pipettes in a low\-volume mode for dispensing very small volumes of liquid. See [Volume Modes](index.html#pipette-volume-modes). + +- Flex and OT\-2 features + + - Optionally specify `apiLevel` in the new `requirements` dictionary (otherwise, specify it in `metadata`). + - Optionally specify `"robotType": "OT-2"` in `requirements`. + - Use coordinates or numbers to specify [deck slots](index.html#deck-slots). These formats match physical labels on Flex and OT\-2, but you can use either system, regardless of `robotType`. + - The new module context `load_adapter()` methods let you load adapters and labware separately on modules, and [`ProtocolContext.load_adapter()`](index.html#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') lets you load adapters directly in deck slots. See [Loading Labware on Adapters](index.html#labware-on-adapters). + - Move labware manually using [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware'), without having to stop your protocol. + - Manual labware moves support moving to or from the new [`OFF_DECK`](index.html#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK') location (outside of the robot). + - [`ProtocolContext.load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') also accepts [`OFF_DECK`](index.html#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK') as a location. This lets you prepare labware to be moved onto the deck later in a protocol. + - The new `push_out` parameter of the [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') method helps ensure that the pipette dispenses all of its liquid when working with very small volumes. + - By default, repeated calls to [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') cycle through multiple locations above the trash bin to prevent tips from stacking up. + +- Bug fixes + + - [`InstrumentContext.starting_tip`](index.html#opentrons.protocol_api.InstrumentContext.starting_tip 'opentrons.protocol_api.InstrumentContext.starting_tip') is now respected on the second and subsequent calls to [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') with no argument. + +#### Version 2\.14 + +This version introduces a new protocol runtime that offers more reliable run control +and builds a strong foundation for future Protocol API improvements. + +Several older parts of the Protocol API were deprecated as part of this switchover. +If you specify an API version of `2.13` or lower, your protocols will continue to execute on the old runtime. + +- Feature additions + + - [`ProtocolContext.define_liquid()`](index.html#opentrons.protocol_api.ProtocolContext.define_liquid 'opentrons.protocol_api.ProtocolContext.define_liquid') and [`Well.load_liquid()`](index.html#opentrons.protocol_api.Well.load_liquid 'opentrons.protocol_api.Well.load_liquid') added + to define different liquid types and add them to wells, respectively. + +- Bug fixes + + - [`Labware`](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') and [`Well`](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') now adhere to the protocol’s API level setting. + Prior to this version, they incorrectly ignored the setting. + - [`InstrumentContext.touch_tip()`](index.html#opentrons.protocol_api.InstrumentContext.touch_tip 'opentrons.protocol_api.InstrumentContext.touch_tip') will end with the pipette tip in the center of the well + instead of on the edge closest to the front of the machine. + - [`ProtocolContext.load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') now prefers loading user\-provided labware definitions + rather than built\-in definitions if no explicit `namespace` is specified. + - [`ProtocolContext.pause()`](index.html#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause') will now properly wait until you resume the protocol before moving on. + In previous versions, the run will not pause until the first call to a different `ProtocolContext` method. + - Motion planning has been improved to avoid certain erroneous downward movements, + especially when using [`InstrumentContext.aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'). + - [`Labware.reset()`](index.html#opentrons.protocol_api.Labware.reset 'opentrons.protocol_api.Labware.reset') and [`Labware.tip_length`](index.html#opentrons.protocol_api.Labware.tip_length 'opentrons.protocol_api.Labware.tip_length') will raise useful errors if called on labware that is not a tip rack. + +- Removals + + - The `presses` and `increment` arguments of [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') were deprecated. + Configure your pipette pick\-up settings with the Opentrons App, instead. + - `InstrumentContext.speed` property was removed. + This property tried to allow setting a pipette’s **plunger** speed in mm/s. + However, it could only approximately set the plunger speed, + because the plunger’s speed is a stepwise function of the volume. + Use [`InstrumentContext.flow_rate`](index.html#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate') to set the flow rate in µL/s, instead. + - `load_labware_object()` was removed from module contexts as an unnecessary internal method. + - `geometry` was removed from module contexts in favor of + `model` and `type` attributes. + - `Well.geometry` was removed as unnecessary. + - `MagneticModuleContext.calibrate` was removed since it was never needed nor implemented. + - The `height` parameter of [`MagneticModuleContext.engage()`](index.html#opentrons.protocol_api.MagneticModuleContext.engage 'opentrons.protocol_api.MagneticModuleContext.engage') was removed. + Use `offset` or `height_from_base` instead. + - `Labware.separate_calibration` and [`Labware.set_calibration()`](index.html#opentrons.protocol_api.Labware.set_calibration 'opentrons.protocol_api.Labware.set_calibration') were removed, + since they were holdovers from a calibration system that no longer exists. + - Various methods and setters were removed that could modify tip state outside of + calls to [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') and [`InstrumentContext.drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip'). + This change allows the robot to track tip usage more completely and reliably. + You may still use [`Labware.reset()`](index.html#opentrons.protocol_api.Labware.reset 'opentrons.protocol_api.Labware.reset') and [`InstrumentContext.reset_tipracks()`](index.html#opentrons.protocol_api.InstrumentContext.reset_tipracks 'opentrons.protocol_api.InstrumentContext.reset_tipracks') + to reset your tip racks’ state. + + > - The [`Well.has_tip`](index.html#opentrons.protocol_api.Well.has_tip 'opentrons.protocol_api.Well.has_tip') **setter** was removed. **The getter is still supported.** + > - Internal methods `Labware.use_tips`, `Labware.previous_tip`, and `Labware.return_tips` + > were removed. + + - The `configuration` argument of [`ProtocolContext.load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') was removed + because it made unsafe modifications to the protocol’s geometry system, + and the Thermocycler’s “semi” configuration is not officially supported. + +- Known limitations + + - [`Labware.set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset') is not yet supported on this API version. + Run protocols via the Opentrons App, instead. + - [`ProtocolContext.max_speeds`](index.html#opentrons.protocol_api.ProtocolContext.max_speeds 'opentrons.protocol_api.ProtocolContext.max_speeds') is not yet supported on the API version. + Use [`InstrumentContext.default_speed`](index.html#opentrons.protocol_api.InstrumentContext.default_speed 'opentrons.protocol_api.InstrumentContext.default_speed') or the per\-method speed argument, instead. + - [`InstrumentContext.starting_tip`](index.html#opentrons.protocol_api.InstrumentContext.starting_tip 'opentrons.protocol_api.InstrumentContext.starting_tip') is not respected on the second and subsequent calls to [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') with no argument. + +#### Version 2\.13 + +- Adds [`HeaterShakerContext`](index.html#opentrons.protocol_api.HeaterShakerContext 'opentrons.protocol_api.HeaterShakerContext') to support the Heater\-Shaker Module. You can use the load name `heaterShakerModuleV1` with [`ProtocolContext.load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') to add a Heater\-Shaker to a protocol. +- [`InstrumentContext.drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') now has a `prep_after` parameter. +- [`InstrumentContext.home()`](index.html#opentrons.protocol_api.InstrumentContext.home 'opentrons.protocol_api.InstrumentContext.home') may home _both_ pipettes as needed to avoid collision risks. +- [`InstrumentContext.aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') and [`InstrumentContext.dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') will avoid interacting directly with modules. + +#### Version 2\.12 + +- [`ProtocolContext.resume()`](index.html#opentrons.protocol_api.ProtocolContext.resume 'opentrons.protocol_api.ProtocolContext.resume') has been deprecated. +- [`Labware.set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset') has been added to apply labware offsets to protocols run (exclusively) outside of the Opentrons App (Jupyter Notebook and SSH). + +#### Version 2\.11 + +- Attempting to aspirate from or dispense to tip racks will raise an error. + +#### Version 2\.10 + +- Moving to the same well twice in a row with different pipettes no longer results in strange diagonal movements. + +#### Version 2\.9 + +- You can now access certain geometry data regarding a labware’s well via a Well Object. See [Well Dimensions](index.html#new-labware-well-properties) for more information. + +#### Version 2\.8 + +- You can now pass in a list of volumes to distribute and consolidate. See [List of Volumes](index.html#distribute-consolidate-volume-list) for more information. + + - Passing in a zero volume to any [complex command](index.html#v2-complex-commands) will result in no actions taken for aspirate or dispense + +- [`Well.from_center_cartesian()`](index.html#opentrons.protocol_api.Well.from_center_cartesian 'opentrons.protocol_api.Well.from_center_cartesian') can be used to find a point within a well using normalized distance from the center in each axis. + + - Note that you will need to create a location object to use this function in a protocol. See [Labware](index.html#protocol-api-labware) for more information. + +- You can now pass in a blowout location to transfer, distribute, and consolidate + with the `blowout_location` parameter. See [`InstrumentContext.transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') for more detail! + +#### Version 2\.7 + +- Added `InstrumentContext.pair_with()`, an experimental feature for moving both pipettes simultaneously. + +Note + +This feature has been removed from the Python Protocol API. + +- Calling [`InstrumentContext.has_tip()`](index.html#opentrons.protocol_api.InstrumentContext.has_tip 'opentrons.protocol_api.InstrumentContext.has_tip') will return whether a particular instrument + has a tip attached or not. + +#### Version 2\.6 + +- GEN2 Single pipettes now default to flow rates equivalent to 10 mm/s plunger + speeds + + + Protocols that manually configure pipette flow rates will be unaffected + + For a comparison between API Versions, see [OT\-2 Pipette Flow Rates](index.html#ot2-flow-rates) + +#### Version 2\.5 + +- New [utility commands](index.html#new-utility-commands) were added: + + - [`ProtocolContext.set_rail_lights()`](index.html#opentrons.protocol_api.ProtocolContext.set_rail_lights 'opentrons.protocol_api.ProtocolContext.set_rail_lights'): turns robot rail lights on or off + - [`ProtocolContext.rail_lights_on`](index.html#opentrons.protocol_api.ProtocolContext.rail_lights_on 'opentrons.protocol_api.ProtocolContext.rail_lights_on'): describes whether or not the rail lights are on + - [`ProtocolContext.door_closed`](index.html#opentrons.protocol_api.ProtocolContext.door_closed 'opentrons.protocol_api.ProtocolContext.door_closed'): describes whether the robot door is closed + +#### Version 2\.4 + +- The following improvements were made to the `touch_tip` command: + + - The speed for `touch_tip` can now be lowered down to 1 mm/s + - `touch_tip` no longer moves diagonally from the X direction \-\> Y direction + - Takes into account geometry of the deck and modules + +#### Version 2\.3 + +- Magnetic Module GEN2 and Temperature Module GEN2 are now supported; you can load them with the names `"magnetic module gen2"` and `"temperature module gen2"`, respectively. +- All pipettes will return tips to tip racks from a higher position to avoid + possible collisions. +- During a [`mix()`](index.html#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix'), the pipette will no longer move up to clear the liquid in + between every dispense and following aspirate. +- You can now access the Temperature Module’s status via [`TemperatureModuleContext.status`](index.html#opentrons.protocol_api.TemperatureModuleContext.status 'opentrons.protocol_api.TemperatureModuleContext.status'). + +#### Version 2\.2 + +- You should now specify Magnetic Module engage height using the + `height_from_base` parameter, which specifies the height of the top of the + magnet from the base of the labware. For more, see [Engaging and Disengaging](index.html#magnetic-module-engage). +- Return tip will now use pre\-defined heights from hardware testing. For more information, see [Returning a Tip](index.html#pipette-return-tip). +- When using the return tip function, tips are no longer added back into the tip tracker. For more information, see [Returning a Tip](index.html#pipette-return-tip). + +#### Version 2\.1 + +- When loading labware onto a module, you can now specify a label with the `label` parameter of + [`MagneticModuleContext.load_labware()`](index.html#opentrons.protocol_api.MagneticModuleContext.load_labware 'opentrons.protocol_api.MagneticModuleContext.load_labware'), + [`TemperatureModuleContext.load_labware()`](index.html#opentrons.protocol_api.TemperatureModuleContext.load_labware 'opentrons.protocol_api.TemperatureModuleContext.load_labware'), or + [`ThermocyclerContext.load_labware()`](index.html#opentrons.protocol_api.ThermocyclerContext.load_labware 'opentrons.protocol_api.ThermocyclerContext.load_labware'), + just like you can when loading labware onto the deck with [`ProtocolContext.load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). + +#### Version 2\.0 + +Version 2 of the API is a new way to write Python protocols, with support for new modules like the Thermocycler. To transition your protocols from version 1 to version 2 of the API, follow this [migration guide](http://support.opentrons.com/en/articles/3425727-switching-your-protocols-from-api-version-1-to-version-2). + +We’ve also published a [more in\-depth discussion](http://support.opentrons.com/en/articles/3418212-opentrons-protocol-api-version-2) of why we developed version 2 of the API and how it differs from version 1\. + +## Labware + +Labware are the durable or consumable items that you work with, reuse, or discard while running a protocol on a Flex or OT\-2\. Items such as pipette tips, well plates, tubes, and reservoirs are all examples of labware. This section provides a brief overview of default labware, custom labware, and how to use basic labware API methods when creating a protocol for your robot. + +Note + +Code snippets use coordinate deck slot locations (e.g. `"D1"`, `"D2"`), like those found on Flex. If you have an OT\-2 and are using API version 2\.14 or earlier, replace the coordinate with its numeric OT\-2 equivalent. For example, slot D1 on Flex corresponds to slot 1 on an OT\-2\. See [Deck Slots](index.html#deck-slots) for more information. + +### Labware Types + +#### Default Labware + +Default labware is everything listed in the [Opentrons Labware Library](https://labware.opentrons.com/). When used in a protocol, your Flex or OT\-2 knows how to work with default labware. However, you must first inform the API about the labware you will place on the robot’s deck. Search the library when you’re looking for the API load names of the labware you want to use. You can copy the load names from the library and pass them to the [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') method in your protocol. + +#### Custom Labware + +Custom labware is labware that is not listed the Labware Library. If your protocol needs something that’s not in the library, you can create it with the [Opentrons Labware Creator](https://labware.opentrons.com/create/). However, before using the Labware Creator, you should take a moment to review the support article [Creating Custom Labware Definitions](https://support.opentrons.com/s/article/Creating-Custom-Labware-Definitions). + +After you’ve created your labware, save it as a `.json` file and add it to the Opentrons App. See [Using Labware in Your Protocols](https://support.opentrons.com/s/article/Using-labware-in-your-protocols) for instructions. + +If other people need to use your custom labware definition, they must also add it to their Opentrons App. + +### Loading Labware + +Throughout this section, we’ll use the labware listed in the following table. + +| Labware type | Labware name | API load name | +| -------------- | --------------------------------------------------------------------------------------------------- | --------------------------------- | +| Well plate | [Corning 96 Well Plate 360 µL Flat](https://labware.opentrons.com/corning_96_wellplate_360ul_flat/) | `corning_96_wellplate_360ul_flat` | +| Flex tip rack | [Opentrons Flex 96 Tips 200 µL](https://shop.opentrons.com/opentrons-flex-tips-200-l/) | `opentrons_flex_96_tiprack_200ul` | +| OT\-2 tip rack | [Opentrons 96 Tip Rack 300 µL](https://labware.opentrons.com/opentrons_96_tiprack_300ul) | `opentrons_96_tiprack_300ul` | + +Similar to the code sample in [How the API Works](index.html#overview-section-v2), here’s how you use the [`ProtocolContext.load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') method to load labware on either Flex or OT\-2\. + +``` +#Flex +tiprack = protocol.load_labware("opentrons_flex_96_tiprack_200ul", "D1") +plate = protocol.load_labware("corning_96_wellplate_360ul_flat", "D2") + +``` + +``` +#OT-2 +tiprack = protocol.load_labware("opentrons_96_tiprack_300ul", "1") +plate = protocol.load_labware("corning_96_wellplate_360ul_flat", "2") + +``` + +New in version 2\.0\. + +When the `load_labware` method loads labware into your protocol, it returns a [`Labware`](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware') object. + +Tip + +The `load_labware` method includes an optional `label` argument. You can use it to identify labware with a descriptive name. If used, the label value is displayed in the Opentrons App. For example: + +``` +tiprack = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1", + label="any-name-you-want") + +``` + +#### Loading Labware on Adapters + +The previous section demonstrates loading labware directly into a deck slot. But you can also load labware on top of an adapter that either fits on a module or goes directly on the deck. The ability to combine labware with adapters adds functionality and flexibility to your robot and protocols. + +You can either load the adapter first and the labware second, or load both the adapter and labware all at once. + +##### Loading Separately + +The `load_adapter()` method is available on `ProtocolContext` and module contexts. It behaves similarly to `load_labware()`, requiring the load name and location for the desired adapter. Load a module, adapter, and labware with separate calls to specify each layer of the physical stack of components individually: + +``` +hs_mod = protocol.load_module("heaterShakerModuleV1", "D1") +hs_adapter = hs_mod.load_adapter("opentrons_96_flat_bottom_adapter") +hs_plate = hs_adapter.load_labware("nest_96_wellplate_200ul_flat") + +``` + +New in version 2\.15: The `load_adapter()` method. + +##### Loading Together + +Use the `adapter` argument of `load_labware()` to load an adapter at the same time as labware. For example, to load the same 96\-well plate and adapter from the previous section at once: + +``` +hs_plate = hs_mod.load_labware( + name="nest_96_wellplate_200ul_flat", + adapter="opentrons_96_flat_bottom_adapter" +) + +``` + +New in version 2\.15: The `adapter` parameter. + +The API also has some “combination” labware definitions, which treat the adapter and labware as a unit: + +``` +hs_combo = hs_mod.load_labware( + "opentrons_96_flat_bottom_adapter_nest_wellplate_200ul_flat" +) + +``` + +Loading labware this way prevents you from [moving the labware](index.html#moving-labware) onto or off of the adapter, so it’s less flexible than loading the two separately. Avoid using combination definitions unless your protocol specifies an `apiLevel` of 2\.14 or lower. + +### Accessing Wells in Labware + +#### Well Ordering + +You need to select which wells to transfer liquids to and from over the course of a protocol. + +Rows of wells on a labware have labels that are capital letters starting with A. For instance, an 96\-well plate has 8 rows, labeled `"A"` through `"H"`. + +Columns of wells on a labware have labels that are numbers starting with 1\. For instance, a 96\-well plate has columns `"1"` through `"12"`. + +All well\-accessing functions start with the well at the top left corner of the labware. The ending well is in the bottom right. The order of travel from top left to bottom right depends on which function you use. + +The code in this section assumes that `plate` is a 24\-well plate. For example: + +``` +plate = protocol.load_labware("corning_24_wellplate_3.4ml_flat", location="D1") + +``` + +#### Accessor Methods + +The API provides many different ways to access wells inside labware. Different methods are useful in different contexts. The table below lists out the methods available to access wells and their differences. + +| Method | Returns | Example | +| ----------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------- | ---------------------------------------------------------------------- | +| [`Labware.wells()`](index.html#opentrons.protocol_api.Labware.wells 'opentrons.protocol_api.Labware.wells') | List of all wells. | `[labware:A1, labware:B1, labware:C1...]` | +| [`Labware.rows()`](index.html#opentrons.protocol_api.Labware.rows 'opentrons.protocol_api.Labware.rows') | List of lists grouped by row. | `[[labware:A1, labware:A2...], [labware:B1, labware:B2...]]` | +| [`Labware.columns()`](index.html#opentrons.protocol_api.Labware.columns 'opentrons.protocol_api.Labware.columns') | List of lists grouped by column. | `[[labware:A1, labware:B1...], [labware:A2, labware:B2...]]` | +| [`Labware.wells_by_name()`](index.html#opentrons.protocol_api.Labware.wells_by_name 'opentrons.protocol_api.Labware.wells_by_name') | Dictionary with well names as keys. | `{"A1": labware:A1, "B1": labware:B1}` | +| [`Labware.rows_by_name()`](index.html#opentrons.protocol_api.Labware.rows_by_name 'opentrons.protocol_api.Labware.rows_by_name') | Dictionary with row names as keys. | `{"A": [labware:A1, labware:A2...], "B": [labware:B1, labware:B2...]}` | +| [`Labware.columns_by_name()`](index.html#opentrons.protocol_api.Labware.columns_by_name 'opentrons.protocol_api.Labware.columns_by_name') | Dictionary with column names as keys. | `{"1": [labware:A1, labware:B1...], "2": [labware:A2, labware:B2...]}` | + +#### Accessing Individual Wells + +##### Dictionary Access + +The simplest way to refer to a single well is by its [`well_name`](index.html#opentrons.protocol_api.Well.well_name 'opentrons.protocol_api.Well.well_name'), like A1 or D6\. Referencing a particular key in the result of [`Labware.wells_by_name()`](index.html#opentrons.protocol_api.Labware.wells_by_name 'opentrons.protocol_api.Labware.wells_by_name') accomplishes this. This is such a common task that the API also has an equivalent shortcut: dictionary indexing. + +``` +a1 = plate.wells_by_name()["A1"] +d6 = plate["D6"] # dictionary indexing + +``` + +If a well does not exist in the labware, such as `plate["H12"]` on a 24\-well plate, the API will raise a `KeyError`. In contrast, it would be a valid reference on a standard 96\-well plate. + +New in version 2\.0\. + +##### List Access From `wells` + +In addition to referencing wells by name, you can also reference them with zero\-indexing. The first well in a labware is at position 0\. + +``` +plate.wells()[0] # well A1 +plate.wells()[23] # well D6 + +``` + +Tip + +You may find coordinate well names like `"B3"` easier to reason with, especially when working with irregular labware, e.g. +`opentrons_10_tuberack_falcon_4x50ml_6x15ml_conical` (see the [Opentrons 10 Tube Rack](https://labware.opentrons.com/opentrons_10_tuberack_falcon_4x50ml_6x15ml_conical) in the Labware Library). Whichever well access method you use, your protocol will be most maintainable if you use only one access method consistently. + +New in version 2\.0\. + +#### Accessing Groups of Wells + +When handling liquid, you can provide a group of wells as the source or destination. Alternatively, you can take a group of wells and loop (or iterate) through them, with each liquid\-handling command inside the loop accessing the loop index. + +Use [`Labware.rows_by_name()`](index.html#opentrons.protocol_api.Labware.rows_by_name 'opentrons.protocol_api.Labware.rows_by_name') to access a specific row of wells or [`Labware.columns_by_name()`](index.html#opentrons.protocol_api.Labware.columns_by_name 'opentrons.protocol_api.Labware.columns_by_name') to access a specific column of wells on a labware. These methods both return a dictionary with the row or column name as the keys: + +``` +row_dict = plate.rows_by_name()["A"] +row_list = plate.rows()[0] # equivalent to the line above +column_dict = plate.columns_by_name()["1"] +column_list = plate.columns()[0] # equivalent to the line above + +print('Column "1" has', len(column_dict), 'wells') # Column "1" has 4 wells +print('Row "A" has', len(row_dict), 'wells') # Row "A" has 6 wells + +``` + +Since these methods return either lists or dictionaries, you can iterate through them as you would regular Python data structures. + +For example, to transfer 50 µL of liquid from the first well of a reservoir to each of the wells of row `"A"` on a plate: + +``` +for well in plate.rows()[0]: + pipette.transfer(reservoir["A1"], well, 50) + +``` + +Equivalently, using `rows_by_name`: + +``` +for well in plate.rows_by_name()["A"].values(): + pipette.transfer(reservoir["A1"], well, 50) + +``` + +New in version 2\.0\. + +### Labeling Liquids in Wells + +Optionally, you can specify the liquids that should be in various wells at the beginning of your protocol. Doing so helps you identify well contents by name and volume, and adds corresponding labels to a single well, or group of wells, in well plates and reservoirs. You can view the initial liquid setup: + +- For Flex protocols, on the touchscreen. +- For Flex or OT\-2 protocols, in the Opentrons App (v6\.3\.0 or higher). + +To use these optional methods, first create a liquid object with [`ProtocolContext.define_liquid()`](index.html#opentrons.protocol_api.ProtocolContext.define_liquid 'opentrons.protocol_api.ProtocolContext.define_liquid') and then label individual wells by calling [`Well.load_liquid()`](index.html#opentrons.protocol_api.Well.load_liquid 'opentrons.protocol_api.Well.load_liquid'). + +Let’s examine how these two methods work. The following examples demonstrate how to define colored water samples for a well plate and reservoir. + +#### Defining Liquids + +This example uses `define_liquid` to create two liquid objects and instantiates them with the variables `greenWater` and `blueWater`, respectively. The arguments for `define_liquid` are all required, and let you name the liquid, describe it, and assign it a color: + +``` +greenWater = protocol.define_liquid( + name="Green water", + description="Green colored water for demo", + display_color="#00FF00", +) +blueWater = protocol.define_liquid( + name="Blue water", + description="Blue colored water for demo", + display_color="#0000FF", +) + +``` + +New in version 2\.14\. + +The `display_color` parameter accepts a hex color code, which adds a color to that liquid’s label when you import your protocol into the Opentrons App. The `define_liquid` method accepts standard 3\-, 4\-, 6\-, and 8\-character hex color codes. + +#### Labeling Wells and Reservoirs + +This example uses `load_liquid` to label the initial well location, contents, and volume (in µL) for the liquid objects created by `define_liquid`. Notice how values of the `liquid` argument use the variable names `greenWater` and `blueWater` (defined above) to associate each well with a particular liquid: + +``` +well_plate["A1"].load_liquid(liquid=greenWater, volume=50) +well_plate["A2"].load_liquid(liquid=greenWater, volume=50) +well_plate["B1"].load_liquid(liquid=blueWater, volume=50) +well_plate["B2"].load_liquid(liquid=blueWater, volume=50) +reservoir["A1"].load_liquid(liquid=greenWater, volume=200) +reservoir["A2"].load_liquid(liquid=blueWater, volume=200) + +``` + +New in version 2\.14\. + +This information is available after you import your protocol to the app or send it to Flex. A summary of liquids appears on the protocol detail page, and well\-by\-well detail is available on the run setup page (under Initial Liquid Setup in the app, or under Liquids on Flex). + +Note + +`load_liquid` does not validate volume for your labware nor does it prevent you from adding multiple liquids to each well. For example, you could label a 40 µL well with `greenWater`, `volume=50`, and then also add blue water to the well. The API won’t stop you. It’s your responsibility to ensure the labels you use accurately reflect the amounts and types of liquid you plan to place into wells and reservoirs. + +#### Labeling vs Handling Liquids + +The `load_liquid` arguments include a volume amount (`volume=n` in µL). This amount is just a label. It isn’t a command or function that manipulates liquids. It only tells you how much liquid should be in a well at the start of the protocol. You need to use a method like [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') to physically move liquids from a source to a destination. + +### Well Dimensions + +The functions in the [Accessing Wells in Labware](#new-well-access) section above return a single [`Well`](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') object or a larger object representing many wells. [`Well`](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') objects have attributes that provide information about their physical shape, such as the depth or diameter, as specified in their corresponding labware definition. These properties can be used for different applications, such as calculating the volume of a well or a [position relative to the well](index.html#position-relative-labware). + +#### Depth + +Use [`Well.depth`](index.html#opentrons.protocol_api.Well.depth 'opentrons.protocol_api.Well.depth') to get the distance in mm between the very top of the well and the very bottom. For example, a conical well’s depth is measured from the top center to the bottom center of the well. + +``` +plate = protocol.load_labware("corning_96_wellplate_360ul_flat", "D1") +depth = plate["A1"].depth # 10.67 + +``` + +#### Diameter + +Use [`Well.diameter`](index.html#opentrons.protocol_api.Well.diameter 'opentrons.protocol_api.Well.diameter') to get the diameter of a given well in mm. Since diameter is a circular measurement, this attribute is only present on labware with circular wells. If the well is not circular, the value will be `None`. Use length and width (see below) for non\-circular wells. + +``` +plate = protocol.load_labware("corning_96_wellplate_360ul_flat", "D1") +diameter = plate["A1"].diameter # 6.86 + +``` + +#### Length + +Use [`Well.length`](index.html#opentrons.protocol_api.Well.length 'opentrons.protocol_api.Well.length') to get the length of a given well in mm. Length is defined as the distance along the robot’s x\-axis (left to right). This attribute is only present on rectangular wells. If the well is not rectangular, the value will be `None`. Use diameter (see above) for circular wells. + +``` +plate = protocol.load_labware("nest_12_reservoir_15ml", "D1") +length = plate["A1"].length # 8.2 + +``` + +#### Width + +Use [`Well.width`](index.html#opentrons.protocol_api.Well.width 'opentrons.protocol_api.Well.width') to get the width of a given well in mm. Width is defined as the distance along the y\-axis (front to back). This attribute is only present on rectangular wells. If the well is not rectangular, the value will be `None`. Use diameter (see above) for circular wells. + +``` +plate = protocol.load_labware("nest_12_reservoir_15ml", "D1") +width = plate["A1"].width # 71.2 + +``` + +New in version 2\.9\. + +## Moving Labware + +You can move an entire labware (and all of its contents) from one deck slot to another at any point during your protocol. On Flex, you can either use the gripper or move the labware manually. On OT\-2, you can can only move labware manually, since it doesn’t have a gripper instrument. + +### Basic Movement + +Use the [`ProtocolContext.move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') method to initiate a move, regardless of whether it uses the gripper. + +``` +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", "D1") + protocol.move_labware(labware=plate, new_location="D2") + +``` + +New in version 2\.15\. + +The required arguments of `move_labware()` are the `labware` you want to move and its `new_location`. You don’t need to specify where the move begins, since that information is already stored in the [`Labware`](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware') object — `plate` in this example. The destination of the move can be any empty deck slot, or a module that’s ready to have labware added to it (see [Movement with Modules](#movement-modules) below). Movement to an occupied location, including the labware’s current location, will raise an error. + +When the move step is complete, the API updates the labware’s location, so you can move the plate multiple times: + +``` +protocol.move_labware(labware=plate, new_location="D2") +protocol.move_labware(labware=plate, new_location="D3") + +``` + +For the first move, the API knows to find the plate in its initial load location, slot D1\. For the second move, the API knows to find the plate in D2\. + +### Automatic vs Manual Moves + +There are two ways to move labware: + +- Automatically, with the Opentrons Flex Gripper. +- Manually, by pausing the protocol until a user confirms that they’ve moved the labware. + +The `use_gripper` parameter of [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') determines whether a movement is automatic or manual. Set its value to `True` for an automatic move. The default value is `False`, so if you don’t specify a value, the protocol will pause for a manual move. + +``` +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", "D1") + + # have the gripper move the plate from D1 to D2 + protocol.move_labware(labware=plate, new_location="D2", use_gripper=True) + + # pause to move the plate manually from D2 to D3 + protocol.move_labware(labware=plate, new_location="D3", use_gripper=False) + + # pause to move the plate manually from D3 to C1 + protocol.move_labware(labware=plate, new_location="C1") + +``` + +New in version 2\.15\. + +Note + +Don’t add a `pause()` command before `move_labware()`. When `use_gripper` is unset or `False`, the protocol pauses when it reaches the movement step. The Opentrons App or the touchscreen on Flex shows an animation of the labware movement that you need to perform manually. The protocol only resumes when you press **Confirm and resume**. + +The above example is a complete and valid `run()` function. You don’t have to load the gripper as an instrument, and there is no `InstrumentContext` for the gripper. All you have to do to specify that a protocol requires the gripper is to include at least one `move_labware()` command with `use_gripper=True`. + +If you attempt to use the gripper to move labware in an OT\-2 protocol, the API will raise an error. + +### Supported Labware + +You can manually move any standard or custom labware. Using the gripper to move the following labware is fully supported by Opentrons: + +| Labware Type | API Load Names | +| --------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| Full\-skirt PCR plates | _ `armadillo_96_wellplate_200ul_pcr_full_skirt` _ `opentrons_96_wellplate_200ul_pcr_full_skirt` | +| NEST well plates | _ `nest_96_wellplate_200ul_flat` _ `nest_96_wellplate_2ml_deep` | +| Opentrons Flex 96 Tip Racks | _ `opentrons_flex_96_tiprack_50ul` _ `opentrons_flex_96_tiprack_200ul` _ `opentrons_flex_96_tiprack_1000ul` _ `opentrons_flex_96_filtertiprack_50ul` _ `opentrons_flex_96_filtertiprack_200ul` _ `opentrons_flex_96_filtertiprack_1000ul` | + +The gripper may work with other ANSI/SLAS standard labware, but this is not recommended. + +Note + +The labware definitions listed above include information about the position and force that the gripper uses to pick up the labware. The gripper uses default values for labware definitions that don’t include position and force information. The Python Protocol API won’t raise a warning or error if you try to grip and move other types of labware. + +### Movement with Modules + +Moving labware on and off of modules lets you precisely control when the labware is in contact with the hot, cold, or magnetic surfaces of the modules — all within a single protocol. + +When moving labware anywhere that isn’t an empty deck slot, consider what physical object the labware will rest on following the move. That object should be the value of `new_location`, and you need to make sure it’s already loaded before the move. For example, if you want to move a 96\-well flat plate onto a Heater\-Shaker module, you actually want to have it rest on top of the Heater\-Shaker’s 96 Flat Bottom Adapter. Pass the adapter, not the module or the slot, as the value of `new_location`: + +``` +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", "D1") + hs_mod = protocol.load_module("heaterShakerModuleV1", "C1") + hs_adapter = hs_mod.load_adapter("opentrons_96_flat_bottom_adapter") + hs_mod.open_labware_latch() + protocol.move_labware( + labware=plate, new_location=hs_adapter, use_gripper=True + ) + +``` + +New in version 2\.15\. + +If you try to move the plate to slot C1 or the Heater\-Shaker module, the API will raise an error, because C1 is occupied by the Heater\-Shaker, and the Heater\-Shaker is occupied by the adapter. Only the adapter, as the topmost item in that stack, is unoccupied. + +Also note the `hs_mod.open_labware_latch()` command in the above example. To move labware onto or off of a module, you have to make sure that it’s physically accessible: + +> - For the Heater\-Shaker, use [`open_labware_latch()`](index.html#opentrons.protocol_api.HeaterShakerContext.open_labware_latch 'opentrons.protocol_api.HeaterShakerContext.open_labware_latch'). +> - For the Thermocycler, use [`open_lid()`](index.html#opentrons.protocol_api.ThermocyclerContext.open_lid 'opentrons.protocol_api.ThermocyclerContext.open_lid'). + +If the labware is inaccessible, the API will raise an error. + +### Movement into the Waste Chute + +Move used tip racks and well plates to the waste chute to dispose of them. This requires you to first [configure the waste chute](index.html#configure-waste-chute) in your protocol. Then use the loaded [`WasteChute`](index.html#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.WasteChute') object as the value of `new_location`: + +``` +chute = protocol.load_waste_chute() +protocol.move_labware( + labware=plate, new_location=chute, use_gripper=True +) + +``` + +New in version 2\.16\. + +This will pick up `plate` from its current location and drop it into the chute. + +Always specify `use_gripper=True` when moving labware into the waste chute. The chute is not designed for manual movement. You can still manually move labware to other locations, including off\-deck, with the chute installed. + +### The Off\-Deck Location + +In addition to moving labware around the deck, [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') can also prompt you to move labware off of or onto the deck. + +Remove labware from the deck to perform tasks like retrieving samples or discarding a spent tip rack. The destination location for such moves is the special constant [`OFF_DECK`](index.html#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK'): + +``` +protocol.move_labware(labware=plate, new_location=protocol_api.OFF_DECK) + +``` + +New in version 2\.15\. + +Moving labware off\-deck always requires user intervention, because the gripper can’t reach outside of the robot. Omit the `use_gripper` parameter or explicitly set it to `False`. If you try to move labware off\-deck with `use_gripper=True`, the API will raise an error. + +You can also load labware off\-deck, in preparation for a `move_labware()` command that brings it _onto_ the deck. For example, you could assign two tip racks to a pipette — one on\-deck, and one off\-deck — and then swap out the first rack for the second one: + +> ``` +> from opentrons import protocol_api +> +> metadata = {"apiLevel": "2.19", "protocolName": "Tip rack replacement"} +> requirements = {"robotType": "OT-2"} +> +> +> def run(protocol: protocol_api.ProtocolContext): +> tips1 = protocol.load_labware("opentrons_96_tiprack_1000ul", 1) +> # load another tip rack but don't put it in a slot yet +> tips2 = protocol.load_labware( +> "opentrons_96_tiprack_1000ul", protocol_api.OFF_DECK +> ) +> pipette = protocol.load_instrument( +> "p1000_single_gen2", "left", tip_racks=[tips1, tips2] +> ) +> # use all the on-deck tips +> for i in range(96): +> pipette.pick_up_tip() +> pipette.drop_tip() +> # pause to move the spent tip rack off-deck +> protocol.move_labware(labware=tips1, new_location=protocol_api.OFF_DECK) +> # pause to move the fresh tip rack on-deck +> protocol.move_labware(labware=tips2, new_location=1) +> pipette.pick_up_tip() +> +> ``` + +Using the off\-deck location to remove or replace labware lets you continue your workflow in a single protocol, rather than needing to end a protocol, reset the deck, and start a new protocol run. + +## Hardware Modules + +### Module Setup + +#### Loading Modules onto the Deck + +Similar to labware and pipettes, you must inform the API about the modules you want to use in your protocol. Even if you don’t use the module anywhere else in your protocol, the Opentrons App and the robot won’t let you start the protocol run until all loaded modules that use power are connected via USB and turned on. + +Use [`ProtocolContext.load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') to load a module. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # Load a Heater-Shaker Module GEN1 in deck slot D1. + heater_shaker = protocol.load_module( + module_name="heaterShakerModuleV1", location="D1") + + # Load a Temperature Module GEN2 in deck slot D3. + temperature_module = protocol.load_module( + module_name="temperature module gen2", location="D3") + +``` + +After the `load_module()` method loads the modules into your protocol, it returns the [`HeaterShakerContext`](index.html#opentrons.protocol_api.HeaterShakerContext 'opentrons.protocol_api.HeaterShakerContext') and [`TemperatureModuleContext`](index.html#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.TemperatureModuleContext') objects. + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # Load a Magnetic Module GEN2 in deck slot 1. + magnetic_module = protocol.load_module( + module_name="magnetic module gen2", location=1) + + # Load a Temperature Module GEN1 in deck slot 3. + temperature_module = protocol.load_module( + module_name="temperature module", location=3) + +``` + +After the `load_module()` method loads the modules into your protocol, it returns the [`MagneticModuleContext`](index.html#opentrons.protocol_api.MagneticModuleContext 'opentrons.protocol_api.MagneticModuleContext') and [`TemperatureModuleContext`](index.html#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.TemperatureModuleContext') objects. + +New in version 2\.0\. + +##### Available Modules + +The first parameter of [`ProtocolContext.load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') is the module’s _API load name_. The load name tells your robot which module you’re going to use in a protocol. The table below lists the API load names for the currently available modules. + +| Module | API Load Name | Introduced in API Version | +| -------------------------- | ---------------------------------------------------- | ------------------------- | +| Temperature Module GEN1 | `temperature module` or `tempdeck` | 2\.0 | +| Temperature Module GEN2 | `temperature module gen2` | 2\.3 | +| Magnetic Module GEN1 | `magnetic module` or `magdeck` | 2\.0 | +| Magnetic Module GEN2 | `magnetic module gen2` | 2\.3 | +| Thermocycler Module GEN1 | `thermocycler module` or `thermocycler` | 2\.0 | +| Thermocycler Module GEN2 | `thermocycler module gen2` or `thermocyclerModuleV2` | 2\.13 | +| Heater\-Shaker Module GEN1 | `heaterShakerModuleV1` | 2\.13 | +| Magnetic Block GEN1 | `magneticBlockV1` | 2\.15 | + +Some modules were added to our Python API later than others, and others span multiple hardware generations. When writing a protocol that requires a module, make sure your `requirements` or `metadata` code block specifies an [API version](index.html#v2-versioning) high enough to support all the module generations you want to use. + +#### Loading Labware onto a Module + +Use the `load_labware()` method on the module context to load labware on a module. For example, to load the [Opentrons 24 Well Aluminum Block](https://labware.opentrons.com/opentrons_24_aluminumblock_generic_2ml_screwcap?category=aluminumBlock) on top of a Temperature Module: + +``` +def run(protocol: protocol_api.ProtocolContext): + temp_mod = protocol.load_module( + module_name="temperature module gen2", + location="D1") + temp_labware = temp_mod.load_labware( + name="opentrons_24_aluminumblock_generic_2ml_screwcap", + label="Temperature-Controlled Tubes") + +``` + +New in version 2\.0\. + +When you load labware on a module, you don’t need to specify the deck slot. In the above example, the `load_module()` method already specifies where the module is on the deck: `location= "D1"`. + +Any [custom labware](index.html#v2-custom-labware) added to your Opentrons App is also accessible when loading labware onto a module. You can find and copy its load name by going to its card on the Labware page. + +New in version 2\.1\. + +##### Module and Labware Compatibility + +It’s your responsibility to ensure the labware and module combinations you load together work together. The Protocol API won’t raise a warning or error if you load an unusual combination, like placing a tube rack on a Thermocycler. See [What labware can I use with my modules?](https://support.opentrons.com/s/article/What-labware-can-I-use-with-my-modules) for more information about labware/module combinations. + +##### Additional Labware Parameters + +In addition to the mandatory `load_name` argument, you can also specify additional parameters. For example, if you specify a `label`, this name will appear in the Opentrons App and the run log instead of the load name. For labware that has multiple definitions, you can specify `version` and `namespace` (though most of the time you won’t have to). The [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') methods of all module contexts accept these additional parameters. + +### Heater\-Shaker Module + +The Heater\-Shaker Module provides on\-deck heating and orbital shaking. The module can heat from 37 to 95 °C, and can shake samples from 200 to 3000 rpm. + +The Heater\-Shaker Module is represented in code by a [`HeaterShakerContext`](index.html#opentrons.protocol_api.HeaterShakerContext 'opentrons.protocol_api.HeaterShakerContext') object. For example: + +``` +hs_mod = protocol.load_module( + module_name="heaterShakerModuleV1", location="D1" +) + +``` + +New in version 2\.13\. + +#### Deck Slots + +The supported deck slot positions for the Heater\-Shaker depend on the robot you’re using. + +| Robot Model | Heater\-Shaker Deck Placement | +| ----------- | ----------------------------------------------------------------------------------------------------------- | +| Flex | In any deck slot in column 1 or 3\. The module can go in slot A3, but you need to move the trash bin first. | +| OT\-2 | In deck slot 1, 3, 4, 6, 7, or 10\. | + +#### OT\-2 Placement Restrictions + +On OT\-2, you need to restrict placement of other modules and labware around the Heater\-Shaker. On Flex, the module is installed below\-deck in a caddy and there is more space between deck slots, so these restrictions don’t apply. + +In general, it’s best to leave all slots adjacent to the Heater\-Shaker empty. If your protocol requires filling those slots, observe the following restrictions to avoid physical crashes involving the Heater\-Shaker. + +##### Adjacent Modules + +Do not place other modules next to the Heater\-Shaker. Keeping adjacent deck slots clear helps prevents collisions during shaking and while opening the labware latch. Loading a module next to the Heater\-Shaker on OT\-2 will raise a `DeckConflictError`. + +##### Tall Labware + +Do not place labware taller than 53 mm to the left or right of the Heater\-Shaker. This prevents the Heater\-Shaker’s latch from colliding with the adjacent labware. Common labware that exceed the height limit include Opentrons tube racks and Opentrons 1000 µL tip racks. Loading tall labware to the right or left of the Heater\-Shaker on OT\-2 will raise a `DeckConflictError`. + +##### 8\-Channel Pipettes + +You can’t perform pipetting actions in any slots adjacent to the Heater\-Shaker if you’re using a GEN2 or GEN1 8\-channel pipette. This prevents the pipette ejector from crashing on the module housing or labware latch. Using an 8\-channel pipette will raise a `PipetteMovementRestrictedByHeaterShakerError`. + +There is one exception: to the front or back of the Heater\-Shaker, an 8\-channel pipette can access tip racks only. Attempting to pipette to non\-tip\-rack labware will also raise a `PipetteMovementRestrictedByHeaterShakerError`. + +#### Latch Control + +To add and remove labware from the Heater\-Shaker, control the module’s labware latch from your protocol using [`open_labware_latch()`](index.html#opentrons.protocol_api.HeaterShakerContext.open_labware_latch 'opentrons.protocol_api.HeaterShakerContext.open_labware_latch') and [`close_labware_latch()`](index.html#opentrons.protocol_api.HeaterShakerContext.close_labware_latch 'opentrons.protocol_api.HeaterShakerContext.close_labware_latch'). Shaking requires the labware latch to be closed, so you may want to issue a close command before the first shake command in your protocol: + +``` +hs_mod.close_labware_latch() +hs_mod.set_and_wait_for_shake_speed(500) + +``` + +If the labware latch is already closed, `close_labware_latch()` will succeed immediately; you don’t have to check the status of the latch before opening or closing it. + +To prepare the deck before running a protocol, use the labware latch controls in the Opentrons App or run these methods in Jupyter notebook. + +#### Loading Labware + +Use the Heater\-Shaker’s [`load_adapter()`](index.html#opentrons.protocol_api.HeaterShakerContext.load_adapter 'opentrons.protocol_api.HeaterShakerContext.load_adapter') and [`load_labware()`](index.html#opentrons.protocol_api.HeaterShakerContext.load_labware 'opentrons.protocol_api.HeaterShakerContext.load_labware') methods to specify what you will place on the module. For the Heater\-Shaker, use one of the thermal adapters listed below and labware that fits on the adapter. See [Loading Labware on Adapters](index.html#labware-on-adapters) for examples of loading labware on modules. + +The [Opentrons Labware Library](https://labware.opentrons.com/) includes definitions for both standalone adapters and adapter–labware combinations. These labware definitions help make the Heater\-Shaker ready to use right out of the box. + +Note + +If you plan to [move labware](index.html#moving-labware) onto or off of the Heater\-Shaker during your protocol, you must use a standalone adapter definition, not an adapter–labware combination definiton. + +##### Standalone Adapters + +You can use these standalone adapter definitions to load Opentrons verified or custom labware on top of the Heater\-Shaker. + +| Adapter Type | API Load Name | +| ----------------------------------------------- | ---------------------------------- | +| Opentrons Universal Flat Heater\-Shaker Adapter | `opentrons_universal_flat_adapter` | +| Opentrons 96 PCR Heater\-Shaker Adapter | `opentrons_96_pcr_adapter` | +| Opentrons 96 Deep Well Heater\-Shaker Adapter | `opentrons_96_deep_well_adapter` | +| Opentrons 96 Flat Bottom Heater\-Shaker Adapter | `opentrons_96_flat_bottom_adapter` | + +For example, these commands load a well plate on top of the flat bottom adapter: + +``` +hs_adapter = hs_mod.load_adapter("opentrons_96_flat_bottom_adapter") +hs_plate = hs_adapter.load_labware("nest_96_wellplate_200ul_flat") + +``` + +New in version 2\.15: The `load_adapter()` method. + +##### Pre\-configured Combinations + +The Heater\-Shaker supports these thermal adapter and labware combinations for backwards compatibility. If your protocol specifies an `apiLevel` of 2\.15 or higher, you should use the standalone adapter definitions instead. + +| Adapter/Labware Combination | API Load Name | +| ------------------------------------------------------------------------ | ------------------------------------------------------------------- | +| Opentrons 96 Deep Well Adapter with NEST Deep Well Plate 2 mL | `opentrons_96_deep_well_adapter_nest_wellplate_2ml_deep` | +| Opentrons 96 Flat Bottom Adapter with NEST 96 Well Plate 200 µL Flat | `opentrons_96_flat_bottom_adapter_nest_wellplate_200ul_flat` | +| Opentrons 96 PCR Adapter with Armadillo Well Plate 200 µL | `opentrons_96_pcr_adapter_armadillo_wellplate_200ul` | +| Opentrons 96 PCR Adapter with NEST Well Plate 100 µL | `opentrons_96_pcr_adapter_nest_wellplate_100ul_pcr_full_skirt` | +| Opentrons Universal Flat Adapter with Corning 384 Well Plate 112 µL Flat | `opentrons_universal_flat_adapter_corning_384_wellplate_112ul_flat` | + +This command loads the same physical adapter and labware as the example in the previous section, but it is also compatible with API versions 2\.13 and 2\.14: + +``` +hs_combo = hs_mod.load_labware( + "opentrons_96_flat_bottom_adapter_nest_wellplate_200ul_flat" +) + +``` + +New in version 2\.13\. + +##### Custom Flat\-Bottom Labware + +Custom flat\-bottom labware can be used with the Universal Flat Adapter. See the support article [Requesting a Custom Labware Definition](https://support.opentrons.com/s/article/Requesting-a-custom-labware-definition) if you need assistance creating custom labware definitions for the Heater\-Shaker. + +#### Heating and Shaking + +The API treats heating and shaking as separate, independent activities due to the amount of time they take. + +Increasing or reducing shaking speed takes a few seconds, so the API treats these actions as _blocking_ commands. All other commands cannot run until the module reaches the required speed. + +Heating the module, or letting it passively cool, takes more time than changing the shaking speed. As a result, the API gives you the flexibility to perform other pipetting actions while waiting for the module to reach a target temperature. When holding at temperature, you can design your protocol to run in a blocking or non\-blocking manner. + +Note + +Since API version 2\.13, only the Heater\-Shaker Module supports non\-blocking command execution. All other modules’ methods are blocking commands. + +##### Blocking commands + +This example uses a blocking command and shakes a sample for one minute. No other commands will execute until a minute has elapsed. The three commands in this example start the shake, wait for one minute, and then stop the shake: + +``` +hs_mod.set_and_wait_for_shake_speed(500) +protocol.delay(minutes=1) +hs_mod.deactivate_shaker() + +``` + +These actions will take about 65 seconds total. Compare this with similar\-looking commands for holding a sample at a temperature for one minute: + +``` +hs_mod.set_and_wait_for_temperature(75) +protocol.delay(minutes=1) +hs_mod.deactivate_heater() + +``` + +This may take much longer, depending on the thermal block used, the volume and type of liquid contained in the labware, and the initial temperature of the module. + +##### Non\-blocking commands + +To pipette while the Heater\-Shaker is heating, use [`set_target_temperature()`](index.html#opentrons.protocol_api.HeaterShakerContext.set_target_temperature 'opentrons.protocol_api.HeaterShakerContext.set_target_temperature') and [`wait_for_temperature()`](index.html#opentrons.protocol_api.HeaterShakerContext.wait_for_temperature 'opentrons.protocol_api.HeaterShakerContext.wait_for_temperature') instead of [`set_and_wait_for_temperature()`](index.html#opentrons.protocol_api.HeaterShakerContext.set_and_wait_for_temperature 'opentrons.protocol_api.HeaterShakerContext.set_and_wait_for_temperature'): + +``` +hs_mod.set_target_temperature(75) +pipette.pick_up_tip() +pipette.aspirate(50, plate["A1"]) +pipette.dispense(50, plate["B1"]) +pipette.drop_tip() +hs_mod.wait_for_temperature() +protocol.delay(minutes=1) +hs_mod.deactivate_heater() + +``` + +This example would likely take just as long as the blocking version above; it’s unlikely that one aspirate and one dispense action would take longer than the time for the module to heat. However, be careful when putting a lot of commands between a `set_target_temperature()` call and a `delay()` call. In this situation, you’re relying on `wait_for_temperature()` to resume execution of commands once heating is complete. But if the temperature has already been reached, the delay will begin later than expected and the Heater\-Shaker will hold at its target temperature longer than intended. + +Additionally, if you want to pipette while the module holds a temperature for a certain length of time, you need to track the holding time yourself. One of the simplest ways to do this is with Python’s `time` module. First, add `import time` at the start of your protocol. Then, use [`time.monotonic()`](https://docs.python.org/3/library/time.html#time.monotonic '(in Python v3.12)') to set a reference time when the target is reached. Finally, add a delay that calculates how much holding time is remaining after the pipetting actions: + +``` +hs_mod.set_and_wait_for_temperature(75) +start_time = time.monotonic() # set reference time +pipette.pick_up_tip() +pipette.aspirate(50, plate["A1"]) +pipette.dispense(50, plate["B1"]) +pipette.drop_tip() +# delay for the difference between now and 60 seconds after the reference time +protocol.delay(max(0, start_time+60 - time.monotonic())) +hs_mod.deactivate_heater() + +``` + +Provided that the parallel pipetting actions don’t take more than one minute, this code will deactivate the heater one minute after its target was reached. If more than one minute has elapsed, the value passed to `protocol.delay()` will equal 0, and the protocol will continue immediately. + +#### Deactivating + +Deactivating the heater and shaker are done separately using the [`deactivate_heater()`](index.html#opentrons.protocol_api.HeaterShakerContext.deactivate_heater 'opentrons.protocol_api.HeaterShakerContext.deactivate_heater') and [`deactivate_shaker()`](index.html#opentrons.protocol_api.HeaterShakerContext.deactivate_shaker 'opentrons.protocol_api.HeaterShakerContext.deactivate_shaker') methods, respectively. There is no method to deactivate both simultaneously. Call the two methods in sequence if you need to stop both heating and shaking. + +Note + +The robot will not automatically deactivate the Heater\-Shaker at the end of a protocol. If you need to deactivate the module after a protocol is completed or canceled, use the Heater\-Shaker module controls on the device detail page in the Opentrons App or run these methods in Jupyter notebook. + +### Magnetic Block + +Note + +The Magnetic Block is compatible with Opentrons Flex only. If you have an OT\-2, use the [Magnetic Module](index.html#magnetic-module). + +The Magnetic Block is an unpowered, 96\-well plate that holds labware close to its high\-strength neodymium magnets. This module is suitable for many magnetic bead\-based protocols, but does not move beads up or down in solution. + +Because the Magnetic Block is unpowered, neither your robot nor the Opentrons App aware of this module. You “control” it via protocols to load labware onto the module and use the Opentrons Flex Gripper to move labware on and off the module. See [Moving Labware](index.html#moving-labware) for more information. + +The Magnetic Block is represented by a [`MagneticBlockContext`](index.html#opentrons.protocol_api.MagneticBlockContext 'opentrons.protocol_api.MagneticBlockContext') object which lets you load labware on top of the module. + +``` +# Load the Magnetic Block in deck slot D1 +magnetic_block = protocol.load_module( + module_name="magneticBlockV1", location="D1" +) + +# Load a 96-well plate on the magnetic block +mag_plate = magnetic_block.load_labware( + name="biorad_96_wellplate_200ul_pcr" +) + +# Use the Gripper to move labware +protocol.move_labware(mag_plate, new_location="B2", use_gripper=True) + +``` + +New in version 2\.15\. + +### Magnetic Module + +Note + +The Magnetic Module is compatible with the OT\-2 only. If you have a Flex, use the [Magnetic Block](index.html#magnetic-block). + +The Magnetic Module controls a set of permanent magnets which can move vertically to induce a magnetic field in the labware loaded on the module. + +The Magnetic Module is represented by a [`MagneticModuleContext`](index.html#opentrons.protocol_api.MagneticModuleContext 'opentrons.protocol_api.MagneticModuleContext') object, which has methods for engaging (raising) and disengaging (lowering) its magnets. + +The examples in this section apply to an OT\-2 with a Magnetic Module GEN2 loaded in slot 6: + +``` +def run(protocol: protocol_api.ProtocolContext): + mag_mod = protocol.load_module( + module_name="magnetic module gen2", + location="6") + plate = mag_mod.load_labware( + name="nest_96_wellplate_100ul_pcr_full_skirt") + +``` + +New in version 2\.3\. + +#### Loading Labware + +Like with all modules, use the Magnetic Module’s [`load_labware()`](index.html#opentrons.protocol_api.MagneticModuleContext.load_labware 'opentrons.protocol_api.MagneticModuleContext.load_labware') method to specify what you will place on the module. The Magnetic Module supports 96\-well PCR plates and deep well plates. For the best compatibility, use a labware definition that specifies how far the magnets should move when engaging with the labware. The following plates in the [Opentrons Labware Library](https://labware.opentrons.com/) include this measurement: + +| Labware Name | API Load Name | +| -------------------------------------------- | ------------------------------------------ | +| Bio\-Rad 96 Well Plate 200 µL PCR | `biorad_96_wellplate_200ul_pcr` | +| NEST 96 Well Plate 100 µL PCR Full Skirt | `nest_96_wellplate_100ul_pcr_full_skirt` | +| NEST 96 Deep Well Plate 2mL | `nest_96_wellplate_2ml_deep` | +| Thermo Scientific Nunc 96 Well Plate 1300 µL | `thermoscientificnunc_96_wellplate_1300ul` | +| Thermo Scientific Nunc 96 Well Plate 2000 µL | `thermoscientificnunc_96_wellplate_2000ul` | +| USA Scientific 96 Deep Well Plate 2\.4 mL | `usascientific_96_wellplate_2.4ml_deep` | + +To check whether a custom labware definition specifies this measurement, load the labware and query its [`magdeck_engage_height`](index.html#opentrons.protocol_api.Labware.magdeck_engage_height 'opentrons.protocol_api.Labware.magdeck_engage_height') property. If has a numerical value, the labware is ready for use with the Magnetic Module. + +#### Engaging and Disengaging + +Raise and lower the module’s magnets with the [`engage()`](index.html#opentrons.protocol_api.MagneticModuleContext.engage 'opentrons.protocol_api.MagneticModuleContext.engage') and [`disengage()`](index.html#opentrons.protocol_api.MagneticModuleContext.disengage 'opentrons.protocol_api.MagneticModuleContext.disengage') functions, respectively. + +If your loaded labware is fully compatible with the Magnetic Module, you can call `engage()` with no argument: + +> ``` +> mag_mod.engage() +> +> ``` +> +> New in version 2\.0\. + +This will move the magnets upward to the default height for the labware, which should be close to the bottom of the labware’s wells. If your loaded labware doesn’t specify a default height, this will raise an `ExceptionInProtocolError`. + +For certain applications, you may want to move the magnets to a different height. The recommended way is to use the `height_from_base` parameter, which represents the distance above the base of the labware (its lowest point, where it rests on the module). Setting `height_from_base=0` should move the tops of the magnets level with the base of the labware. Alternatively, you can use the `offset` parameter, which represents the distance above _or below_ the labware’s default position (close to the bottom of its wells). Like using `engage()` with no argument, this will raise an error if there is no default height for the loaded labware. + +Note + +There is up to 1 mm of manufacturing variance across Magnetic Module units, so observe the exact position and adjust as necessary before running your protocol. + +Here are some examples of where the magnets will move when using the different parameters in combination with the loaded NEST PCR plate, which specifies a default height of 20 mm: + +> ``` +> mag_mod.engage(height_from_base=13.5) # 13.5 mm +> mag_mod.engage(offset=-2) # 15.5 mm +> +> ``` + +Note that `offset` takes into account the fact that the magnets’ home position is measured as −2\.5 mm for GEN2 modules. + +> New in version 2\.0\. +> +> Changed in version 2\.2: Added the `height_from_base` parameter. + +When you need to retract the magnets back to their home position, call [`disengage()`](index.html#opentrons.protocol_api.MagneticModuleContext.disengage 'opentrons.protocol_api.MagneticModuleContext.disengage'). + +> ``` +> mag_mod.disengage() # -2.5 mm +> +> ``` + +New in version 2\.0\. + +If at any point you need to check whether the magnets are engaged or not, use the [`status`](index.html#opentrons.protocol_api.MagneticModuleContext.status 'opentrons.protocol_api.MagneticModuleContext.status') property. This will return either the string `engaged` or `disengaged`, not the exact height of the magnets. + +Note + +The OT\-2 will not automatically deactivate the Magnetic Module at the end of a protocol. If you need to deactivate the module after a protocol is completed or canceled, use the Magnetic Module controls on the device detail page in the Opentrons App or run `deactivate()` in Jupyter notebook. + +#### Changes with the GEN2 Magnetic Module + +The GEN2 Magnetic Module uses smaller magnets than the GEN1 version. This change helps mitigate an issue with the magnets attracting beads from their retracted position, but it also takes longer for the GEN2 module to attract beads. The recommended attraction time is 5 minutes for liquid volumes up to 50 µL and 7 minutes for volumes greater than 50 µL. If your application needs additional magnetic strength to attract beads within these timeframes, use the available [Adapter Magnets](https://support.opentrons.com/s/article/Adapter-magnets). + +### Temperature Module + +The Temperature Module acts as both a cooling and heating device. It can control the temperature of its deck between 4 °C and 95 °C with a resolution of 1 °C. + +The Temperature Module is represented in code by a [`TemperatureModuleContext`](index.html#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.TemperatureModuleContext') object, which has methods for setting target temperatures and reading the module’s status. This example demonstrates loading a Temperature Module GEN2 and loading a well plate on top of it. + +``` +temp_mod = protocol.load_module( + module_name="temperature module gen2", location="D3" +) + +``` + +New in version 2\.3\. + +#### Loading Labware + +Use the Temperature Module’s [`load_adapter()`](index.html#opentrons.protocol_api.TemperatureModuleContext.load_adapter 'opentrons.protocol_api.TemperatureModuleContext.load_adapter') and [`load_labware()`](index.html#opentrons.protocol_api.TemperatureModuleContext.load_labware 'opentrons.protocol_api.TemperatureModuleContext.load_labware') methods to specify what you will place on the module. You may use one or both of the methods, depending on the labware you’re using. See [Loading Labware on Adapters](index.html#labware-on-adapters) for examples of loading labware on modules. + +The [Opentrons Labware Library](https://labware.opentrons.com/) includes definitions for both standalone adapters and adapter–labware combinations. These labware definitions help make the Temperature Module ready to use right out of the box. + +##### Standalone Adapters + +You can use these standalone adapter definitions to load Opentrons verified or custom labware on top of the Temperature Module. + +| Adapter Type | API Load Name | +| ------------------------------------ | -------------------------------------- | +| Opentrons Aluminum Flat Bottom Plate | `opentrons_aluminum_flat_bottom_plate` | +| Opentrons 96 Well Aluminum Block | `opentrons_96_well_aluminum_block` | + +For example, these commands load a PCR plate on top of the 96\-well block: + +``` +temp_adapter = temp_mod.load_adapter( + "opentrons_96_well_aluminum_block" +) +temp_plate = temp_adapter.load_labware( + "nest_96_wellplate_100ul_pcr_full_skirt" +) + +``` + +New in version 2\.15: The `load_adapter()` method. + +Note + +You can also load labware directly onto the Temperature Module. In API version 2\.14 and earlier, this was the correct way to load labware on top of the flat bottom plate. In API version 2\.15 and later, you should load both the adapter and the labware with separate commands. + +##### Block\-and\-tube combinations + +You can use these combination labware definitions to load various types of tubes into the 24\-well thermal block on top of the Temperature Module. There is no standalone definition for the 24\-well block. + +| Tube Type | API Load Name | +| ---------------------- | ------------------------------------------------- | +| Generic 2 mL screw cap | `opentrons_24_aluminumblock_generic_2ml_screwcap` | +| NEST 0\.5 mL screw cap | `opentrons_24_aluminumblock_nest_0.5ml_screwcap` | +| NEST 1\.5 mL screw cap | `opentrons_24_aluminumblock_nest_1.5ml_screwcap` | +| NEST 1\.5 mL snap cap | `opentrons_24_aluminumblock_nest_1.5ml_snapcap` | +| NEST 2 mL screw cap | `opentrons_24_aluminumblock_nest_2ml_screwcap` | +| NEST 2 mL snap cap | `opentrons_24_aluminumblock_nest_2ml_snapcap` | + +For example, this command loads the 24\-well block with generic 2 mL tubes: + +``` +temp_tubes = temp_mod.load_labware( + "opentrons_24_aluminumblock_generic_2ml_screwcap" +) + +``` + +New in version 2\.0\. + +##### Block\-and\-plate combinations + +The Temperature Module supports these 96\-well block and labware combinations for backwards compatibility. If your protocol specifies an `apiLevel` of 2\.15 or higher, you should use the standalone 96\-well block definition instead. + +| 96\-well block contents | API Load Name | +| -------------------------- | ---------------------------------------------------- | +| Bio\-Rad well plate 200 μL | `opentrons_96_aluminumblock_biorad_wellplate_200uL` | +| Generic PCR strip 200 μL | `opentrons_96_aluminumblock_generic_pcr_strip_200uL` | +| NEST well plate 100 μL | `opentrons_96_aluminumblock_nest_wellplate_100uL` | + +This command loads the same physical adapter and labware as the example in the Standalone Adapters section above, but it is also compatible with earlier API versions: + +``` +temp_combo = temp_mod.load_labware( + "opentrons_96_aluminumblock_nest_wellplate_100uL" +) + +``` + +New in version 2\.0\. + +#### Temperature Control + +The primary function of the module is to control the temperature of its deck, using [`set_temperature()`](index.html#opentrons.protocol_api.TemperatureModuleContext.set_temperature 'opentrons.protocol_api.TemperatureModuleContext.set_temperature'), which takes one parameter: `celsius`. For example, to set the Temperature Module to 4 °C: + +``` +temp_mod.set_temperature(celsius=4) + +``` + +When using `set_temperature()`, your protocol will wait until the target temperature is reached before proceeding to further commands. In other words, you can pipette to or from the Temperature Module when it is holding at a temperature or idle, but not while it is actively changing temperature. Whenever the module reaches its target temperature, it will hold the temperature until you set a different target or call [`deactivate()`](index.html#opentrons.protocol_api.TemperatureModuleContext.deactivate 'opentrons.protocol_api.TemperatureModuleContext.deactivate'), which will stop heating or cooling and will turn off the fan. + +Note + +Your robot will not automatically deactivate the Temperature Module at the end of a protocol. If you need to deactivate the module after a protocol is completed or canceled, use the Temperature Module controls on the device detail page in the Opentrons App or run `deactivate()` in Jupyter notebook. + +New in version 2\.0\. + +#### Temperature Status + +If you need to confirm in software whether the Temperature Module is holding at a temperature or is idle, use the [`status`](index.html#opentrons.protocol_api.TemperatureModuleContext.status 'opentrons.protocol_api.TemperatureModuleContext.status') property: + +``` +temp_mod.set_temperature(celsius=90) +temp_mod.status # "holding at target" +temp_mod.deactivate() +temp_mod.status # "idle" + +``` + +If you don’t need to use the status value in your code, and you have physical access to the module, you can read its status and temperature from the LED and display on the module. + +New in version 2\.0\. + +#### Changes with the GEN2 Temperature Module + +All methods of [`TemperatureModuleContext`](index.html#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.TemperatureModuleContext') work with both the GEN1 and GEN2 Temperature Module. Physically, the GEN2 module has a plastic insulating rim around the plate, and plastic insulating shrouds designed to fit over Opentrons aluminum blocks. This mitigates an issue where the GEN1 module would have trouble cooling to very low temperatures, especially if it shared the deck with a running Thermocycler. + +### Thermocycler Module + +The Thermocycler Module provides on\-deck, fully automated thermocycling, and can heat and cool very quickly during operation. The module’s block can reach and maintain temperatures between 4 and 99 °C. The module’s lid can heat up to 110 °C. + +The Thermocycler is represented in code by a [`ThermocyclerContext`](index.html#opentrons.protocol_api.ThermocyclerContext 'opentrons.protocol_api.ThermocyclerContext') object, which has methods for controlling the lid, controlling the block, and setting _profiles_ — timed heating and cooling routines that can be repeated automatically. + +The examples in this section will use a Thermocycler Module GEN2 loaded as follows: + +``` +tc_mod = protocol.load_module(module_name="thermocyclerModuleV2") +plate = tc_mod.load_labware(name="nest_96_wellplate_100ul_pcr_full_skirt") + +``` + +New in version 2\.13\. + +#### Lid Control + +The Thermocycler can control the position and temperature of its lid. + +To change the lid position, use [`open_lid()`](index.html#opentrons.protocol_api.ThermocyclerContext.open_lid 'opentrons.protocol_api.ThermocyclerContext.open_lid') and [`close_lid()`](index.html#opentrons.protocol_api.ThermocyclerContext.close_lid 'opentrons.protocol_api.ThermocyclerContext.close_lid'). When the lid is open, the pipettes can access the loaded labware. + +You can also control the temperature of the lid. Acceptable target temperatures are between 37 and 110 °C. Use [`set_lid_temperature()`](index.html#opentrons.protocol_api.ThermocyclerContext.set_lid_temperature 'opentrons.protocol_api.ThermocyclerContext.set_lid_temperature'), which takes one parameter: the target `temperature` (in degrees Celsius) as an integer. For example, to set the lid to 50 °C: + +``` +tc_mod.set_lid_temperature(temperature=50) + +``` + +The protocol will only proceed once the lid temperature reaches 50 °C. This is the case whether the previous temperature was lower than 50 °C (in which case the lid will actively heat) or higher than 50 °C (in which case the lid will passively cool). + +You can turn off the lid heater at any time with [`deactivate_lid()`](index.html#opentrons.protocol_api.ThermocyclerContext.deactivate_lid 'opentrons.protocol_api.ThermocyclerContext.deactivate_lid'). + +Note + +Lid temperature is not affected by Thermocycler profiles. Therefore you should set an appropriate lid temperature to hold during your profile _before_ executing it. See [Thermocycler Profiles](#thermocycler-profiles) for more information on defining and executing profiles. + +New in version 2\.0\. + +#### Block Control + +The Thermocycler can control its block temperature, including holding at a temperature and adjusting for the volume of liquid held in its loaded plate. + +##### Temperature + +To set the block temperature inside the Thermocycler, use [`set_block_temperature()`](index.html#opentrons.protocol_api.ThermocyclerContext.set_block_temperature 'opentrons.protocol_api.ThermocyclerContext.set_block_temperature'). At minimum you have to specify a `temperature` in degrees Celsius: + +``` +tc_mod.set_block_temperature(temperature=4) + +``` + +If you don’t specify any other parameters, the Thermocycler will hold this temperature until a new temperature is set, [`deactivate_block()`](index.html#opentrons.protocol_api.ThermocyclerContext.deactivate_block 'opentrons.protocol_api.ThermocyclerContext.deactivate_block') is called, or the module is powered off. + +New in version 2\.0\. + +##### Hold Time + +You can optionally instruct the Thermocycler to hold its block temperature for a specific amount of time. You can specify `hold_time_minutes`, `hold_time_seconds`, or both (in which case they will be added together). For example, this will set the block to 4 °C for 4 minutes and 15 seconds: + +``` +tc_mod.set_block_temperature( + temperature=4, + hold_time_minutes=4, + hold_time_seconds=15) + +``` + +Note + +Your protocol will not proceed to further commands while holding at a temperature. If you don’t specify a hold time, the protocol will proceed as soon as the target temperature is reached. + +New in version 2\.0\. + +##### Block Max Volume + +The Thermocycler’s block temperature controller varies its behavior based on the amount of liquid in the wells of its labware. Accurately specifying the liquid volume allows the Thermocycler to more precisely control the temperature of the samples. You should set the `block_max_volume` parameter to the amount of liquid in the _fullest_ well, measured in µL. If not specified, the Thermocycler will assume samples of 25 µL. + +It is especially important to specify `block_max_volume` when holding at a temperature. For example, say you want to hold larger samples at a temperature for a short time: + +``` +tc_mod.set_block_temperature( + temperature=4, + hold_time_seconds=20, + block_max_volume=80) + +``` + +If the Thermocycler assumes these samples are 25 µL, it may not cool them to 4 °C before starting the 20\-second timer. In fact, with such a short hold time they may not reach 4 °C at all! + +New in version 2\.0\. + +#### Thermocycler Profiles + +In addition to executing individual temperature commands, the Thermocycler can automatically cycle through a sequence of block temperatures to perform heat\-sensitive reactions. These sequences are called _profiles_, which are defined in the Protocol API as lists of dictionaries. Each dictionary within the profile should have a `temperature` key, which specifies the temperature of the step, and either or both of `hold_time_seconds` and `hold_time_minutes`, which specify the duration of the step. + +For example, this profile commands the Thermocycler to reach 10 °C and hold for 30 seconds, and then to reach 60 °C and hold for 45 seconds: + +``` +profile = [ + {"temperature":10, "hold_time_seconds":30}, + {"temperature":60, "hold_time_seconds":45} +] + +``` + +Once you have written the steps of your profile, execute it with [`execute_profile()`](index.html#opentrons.protocol_api.ThermocyclerContext.execute_profile 'opentrons.protocol_api.ThermocyclerContext.execute_profile'). This function executes your profile steps multiple times depending on the `repetitions` parameter. It also takes a `block_max_volume` parameter, which is the same as that of the [`set_block_temperature()`](index.html#opentrons.protocol_api.ThermocyclerContext.set_block_temperature 'opentrons.protocol_api.ThermocyclerContext.set_block_temperature') function. + +For instance, a PCR prep protocol might define and execute a profile like this: + +``` +profile = [ + {"temperature":95, "hold_time_seconds":30}, + {"temperature":57, "hold_time_seconds":30}, + {"temperature":72, "hold_time_seconds":60} +] +tc_mod.execute_profile(steps=profile, repetitions=20, block_max_volume=32) + +``` + +In terms of the actions that the Thermocycler performs, this would be equivalent to nesting `set_block_temperature` commands in a `for` loop: + +``` +for i in range(20): + tc_mod.set_block_temperature(95, hold_time_seconds=30, block_max_volume=32) + tc_mod.set_block_temperature(57, hold_time_seconds=30, block_max_volume=32) + tc_mod.set_block_temperature(72, hold_time_seconds=60, block_max_volume=32) + +``` + +However, this code would generate 60 lines in the protocol’s run log, while executing a profile is summarized in a single line. Additionally, you can set a profile once and execute it multiple times (with different numbers of repetitions and maximum volumes, if needed). + +Note + +Temperature profiles only control the temperature of the block in the Thermocycler. You should set a lid temperature before executing the profile using [`set_lid_temperature()`](index.html#opentrons.protocol_api.ThermocyclerContext.set_lid_temperature 'opentrons.protocol_api.ThermocyclerContext.set_lid_temperature'). + +New in version 2\.0\. + +#### Changes with the GEN2 Thermocycler Module + +All methods of [`ThermocyclerContext`](index.html#opentrons.protocol_api.ThermocyclerContext 'opentrons.protocol_api.ThermocyclerContext') work with both the GEN1 and GEN2 Thermocycler. One practical difference is that the GEN2 module has a plate lift feature to make it easier to remove the plate manually or with the Opentrons Flex Gripper. To activate the plate lift, press the button on the Thermocycler for three seconds while the lid is open. If you need to do this in the middle of a run, call [`pause()`](index.html#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause'), lift and move the plate, and then resume the run. + +### Multiple Modules of the Same Type + +You can use multiple modules of the same type within a single protocol. The exception is the Thermocycler Module, which has only one supported deck location because of its size. Running protocols with multiple modules of the same type requires version 4\.3 or newer of the Opentrons App and robot server. + +When working with multiple modules of the same type, load them in your protocol according to their USB port number. Deck coordinates are required by the [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') method, but location does not determine which module loads first. Your robot will use the module with the lowest USB port number _before_ using a module of the same type that’s connected to higher numbered USB port. The USB port number (not deck location) determines module load sequence, starting with the lowest port number first. + +### Flex + +In this example, `temperature_module_1` loads first because it’s connected to USB port 2\. `temperature_module_2` loads next because it’s connected to USB port 6\. + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # Load Temperature Module 1 in deck slot D1 on USB port 2 + temperature_module_1 = protocol.load_module( + module_name="temperature module gen2", + location="D1") + + # Load Temperature Module 2 in deck slot C1 on USB port 6 + temperature_module_2 = protocol.load_module( + module_name="temperature module gen2", + location="C1") + +``` + +The Temperature Modules are connected as shown here: + +### OT-2 + +In this example, `temperature_module_1` loads first because it’s connected to USB port 1\. `temperature_module_2` loads next because it’s connected to USB port 3\. + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + + +def run(protocol: protocol_api.ProtocolContext): + # Load Temperature Module 1 in deck slot C1 on USB port 1 + temperature_module_1 = protocol.load_module( + load_name="temperature module gen2", location="1" + ) + + # Load Temperature Module 2 in deck slot D3 on USB port 2 + temperature_module_2 = protocol.load_module( + load_name="temperature module gen2", location="3" + ) + +``` + +The Temperature Modules are connected as shown here: + +Before running your protocol, it’s a good idea to use the module controls in the Opentrons App to check that commands are being sent where you expect. + +See the support article [Using Modules of the Same Type](https://support.opentrons.com/s/article/Using-modules-of-the-same-type-on-the-OT-2) for more information. + +Hardware modules are powered and unpowered deck\-mounted peripherals. The Flex and OT\-2 are aware of deck\-mounted powered modules when they’re attached via a USB connection and used in an uploaded protocol. The robots do not know about unpowered modules until you use one in a protocol and upload it to the Opentrons App. + +Powered modules include the Heater\-Shaker Module, Magnetic Module, Temperature Module, and Thermocycler Module. The 96\-well Magnetic Block is an unpowered module. + +Pages in this section of the documentation cover: + +> - [Setting up modules and their labware](index.html#module-setup). +> - Working with the module contexts for each type of module. +> +> > - [Heater\-Shaker Module](index.html#heater-shaker-module) +> > - [Magnetic Block](index.html#magnetic-block) +> > - [Magnetic Module](index.html#magnetic-module) +> > - [Temperature Module](index.html#temperature-module) +> > - [Thermocycler Module](index.html#thermocycler-module) +> +> - Working with [multiple modules of the same type](index.html#moam) in a single protocol. + +Note + +Throughout these pages, most code examples use coordinate deck slot locations (e.g. `"D1"`, `"D2"`), like those found on Flex. If you have an OT\-2 and are using API version 2\.14 or earlier, replace the coordinate with its numeric OT\-2 equivalent. For example, slot D1 on Flex corresponds to slot 1 on an OT\-2\. See [Deck Slots](index.html#deck-slots) for more information. + +## Deck Slots + +Deck slots are where you place hardware items on the deck surface of your Opentrons robot. In the API, you load the corresponding items into your protocol with methods like [`ProtocolContext.load_labware`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'), [`ProtocolContext.load_module`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'), or [`ProtocolContext.load_trash_bin`](index.html#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin'). When you call these methods, you need to specify which slot to load the item in. + +### Physical Deck Labels + +Flex uses a coordinate labeling system for slots A1 (back left) through D4 (front right). Columns 1 through 3 are in the _working area_ and are accessible by pipettes and the gripper. Column 4 is in the _staging area_ and is only accessible by the gripper. For more information on staging area slots, see [Deck Configuration](#deck-configuration) below. + +OT\-2 uses a numeric labeling system for slots 1 (front left) through 11 (back center). The back right slot is occupied by the fixed trash. + +### API Deck Labels + +The API accepts values that correspond to the physical deck slot labels on a Flex or OT\-2 robot. Specify a slot in either format: + +- A coordinate like `"A1"`. This format must be a string. +- A number like `"10"` or `10`. This format can be a string or an integer. + +As of API version 2\.15, the Flex and OT\-2 formats are interchangeable. You can use either format, regardless of which robot your protocol is for. You could even mix and match formats within a protocol, although this is not recommended. + +For example, these two `load_labware()` commands are equivalent: + +``` +protocol.load_labware("nest_96_wellplate_200ul_flat", "A1") + +``` + +New in version 2\.15\. + +``` +protocol.load_labware("nest_96_wellplate_200ul_flat", 10) + +``` + +New in version 2\.0\. + +Both of these commands would require you to load the well plate in the back left slot of the robot. + +The correspondence between deck labels is based on the relative locations of the slots. The full list of slot equivalencies is as follows: + +| Flex | A1 | A2 | A3 | B1 | B2 | B3 | C1 | C2 | C3 | D1 | D2 | D3 | +| ----- | --- | --- | ----- | --- | --- | --- | --- | --- | --- | --- | --- | --- | +| OT\-2 | 10 | 11 | Trash | 7 | 8 | 9 | 4 | 5 | 6 | 1 | 2 | 3 | + +Slots A4, B4, C4, and D4 on Flex have no equivalent on OT\-2\. + +### Deck Configuration + +A Flex running robot system version 7\.1\.0 or higher lets you specify its deck configuration on the touchscreen or in the Opentrons App. This tells the robot the positions of unpowered _deck fixtures_: items that replace standard deck slots. The following table lists currently supported deck fixtures and their allowed deck locations. + +| Fixture | Slots | +| ------------------ | ------------- | +| Staging area slots | A3–D3 | +| Trash bin | A1–D1, A3\-D3 | +| Waste chute | D3 | + +Which fixtures you need to configure depend on both load methods and the effects of other methods called in your protocol. The following sections explain how to configure each type of fixture. + +#### Staging Area Slots + +Slots A4 through D4 are the staging area slots. Pipettes can’t reach the staging area, but these slots are always available in the API for loading and moving labware. Using a slot in column 4 as the `location` argument of [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') or the `new_location` argument of [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') will require the corresponding staging area slot in the robot’s deck configuration: + +``` +plate_1 = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="C3" +) # no staging slots required +plate_2 = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="D4" +) # one staging slot required +protocol.move_labware( + labware=plate_1, new_location="C4" +) # two staging slots required + +``` + +New in version 2\.16\. + +Since staging area slots also include a standard deck slot in column 3, they are physically incompatible with powered modules in the same row of column 3\. For example, if you try to load a module in C3 and labware in C4, the API will raise an error: + +``` +temp_mod = protocol.load_module( + module_name="temperature module gen2", + location="C3" +) +staging_plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="C4" +) # deck conflict error + +``` + +It is possible to use slot D4 along with the waste chute. See the [Waste Chute](#configure-waste-chute) section below for details. + +#### Trash Bin + +In version 2\.15 of the API, Flex can only have a single trash bin in slot A3\. You do not have to (and cannot) load the trash in version 2\.15 protocols. + +Starting in API version 2\.16, you must load trash bin fixtures in your protocol in order to use them. Use [`load_trash_bin()`](index.html#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin') to load a movable trash bin. This example loads a single bin in the default location: + +``` +default_trash = protocol.load_trash_bin(location = "A3") + +``` + +New in version 2\.16\. + +Call `load_trash_bin()` multiple times to add more than one bin. See [Adding Trash Containers](index.html#pipette-trash-containers) for more information on using pipettes with multiple trash bins. + +#### Waste Chute + +The waste chute accepts various materials from Flex pipettes or the Flex Gripper and uses gravity to transport them outside of the robot for disposal. Pipettes can dispose of liquid or drop tips into the chute. The gripper can drop tip racks and other labware into the chute. + +To use the waste chute, first use [`load_waste_chute()`](index.html#opentrons.protocol_api.ProtocolContext.load_waste_chute 'opentrons.protocol_api.ProtocolContext.load_waste_chute') to load it in slot D3: + +``` +chute = protocol.load_waste_chute() + +``` + +New in version 2\.16\. + +The `load_waste_chute()` method takes no arguments, since D3 is the only valid location for the chute. However, there are multiple variant configurations of the waste chute, depending on how other methods in your protocol use it. + +The waste chute is installed either on a standard deck plate adapter or on a deck plate adapter with a staging area. If any [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') or [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') calls in your protocol reference slot D4, you have to use the deck plate adapter with staging area. + +The waste chute has a removable cover with a narrow opening which helps prevent aerosols and droplets from contaminating the working area. 1\- and 8\-channel pipettes can dispense liquid, blow out, or drop tips through the opening in the cover. Any of the following require you to remove the cover. + +> - [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'), [`blow_out()`](index.html#opentrons.protocol_api.InstrumentContext.blow_out 'opentrons.protocol_api.InstrumentContext.blow_out'), or [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') with a 96\-channel pipette. +> - [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') with the chute as `new_location` and `use_gripper=True`. + +If your protocol _does not_ call any of these methods, your deck configuration should include the cover. + +In total, there are four possible deck configurations for the waste chute.\* Waste chute only + +- Waste chute with cover +- Waste chute with staging area slot +- Waste chute with staging area slot and cover + +### Deck Conflicts + +A deck conflict check occurs when preparing to run a Python protocol on a Flex running robot system version 7\.1\.0 or higher. The Opentrons App and touchscreen will prevent you from starting the protocol run until any conflicts are resolved. You can resolve them one of two ways: + +> - Physically move hardware around the deck, and update the deck configuration. +> - Alter your protocol to work with the current deck configuration, and resend the protocol to your Flex. + +## Pipettes + +### Loading Pipettes + +When writing a protocol, you must inform the Protocol API about the pipettes you will be using on your robot. The [`ProtocolContext.load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') function provides this information and returns an [`InstrumentContext`](index.html#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.InstrumentContext') object. + +As noted above, you call the [`load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') method to load a pipette. This method also requires the [pipette’s API load name](#new-pipette-models), its left or right mount position, and (optionally) a list of associated tip racks. Even if you don’t use the pipette anywhere else in your protocol, the Opentrons App and the robot won’t let you start the protocol run until all pipettes loaded by `load_instrument()` are attached properly. + +#### API Load Names + +The pipette’s API load name (`instrument_name`) is the first parameter of the `load_instrument()` method. It tells your robot which attached pipette you’re going to use in a protocol. The tables below list the API load names for the currently available Flex and OT\-2 pipettes. + +### Flex Pipettes + +| Pipette Model | Volume (µL) | API Load Name | | +| ------------------------ | -------------------- | --------------------- | --- | +| Flex 1\-Channel Pipette | 1–50 | `flex_1channel_50` | | +| 5–1000 | `flex_1channel_1000` | | +| Flex 8\-Channel Pipette | 1–50 | `flex_8channel_50` | | +| 5–1000 | `flex_8channel_1000` | | +| Flex 96\-Channel Pipette | 5–1000 | `flex_96channel_1000` | | + +### OT-2 Pipettes + +| Pipette Model | Volume (µL) | API Load Name | +| -------------------------- | ----------------- | ------------------- | +| P20 Single\-Channel GEN2 | 1\-20 | `p20_single_gen2` | +| P20 Multi\-Channel GEN2 | `p20_multi_gen2` | +| P300 Single\-Channel GEN2 | 20\-300 | `p300_single_gen2` | +| P300 Multi\-Channel GEN2 | `p300_multi_gen2` | +| P1000 Single\-Channel GEN2 | 100\-1000 | `p1000_single_gen2` | + +See the [OT\-2 Pipette Generations](index.html#ot2-pipette-generations) section if you’re using GEN1 pipettes on an OT\-2\. The GEN1 family includes the P10, P50, and P300 single\- and multi\-channel pipettes, along with the P1000 single\-channel model. + +#### Loading Flex 1\- and 8\-Channel Pipettes + +This code sample loads a Flex 1\-Channel Pipette in the left mount and a Flex 8\-Channel Pipette in the right mount. Both pipettes are 1000 µL. Each pipette uses its own 1000 µL tip rack. + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel":"2.19"} + +def run(protocol: protocol_api.ProtocolContext): + tiprack1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="D1") + tiprack2 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="C1") + left = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack1]) + right = protocol.load_instrument( + instrument_name="flex_8channel_1000", + mount="right", + tip_racks=[tiprack2]) + +``` + +If you’re writing a protocol that uses the Flex Gripper, you might think that this would be the place in your protocol to declare that. However, the gripper doesn’t require `load_instrument`! Whether your gripper requires a protocol is determined by the presence of [`ProtocolContext.move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') commands. See [Moving Labware](index.html#moving-labware) for more details. + +#### Loading a Flex 96\-Channel Pipette + +This code sample loads the Flex 96\-Channel Pipette. Because of its size, the Flex 96\-Channel Pipette requires the left _and_ right pipette mounts. You cannot use this pipette with 1\- or 8\-Channel Pipette in the same protocol or when these instruments are attached to the robot. Load the 96\-channel pipette as follows: + +``` +def run(protocol: protocol_api.ProtocolContext): + pipette = protocol.load_instrument( + instrument_name="flex_96channel_1000" + ) + +``` + +In protocols specifying API version 2\.15, also include `mount="left"` as a parameter of `load_instrument()`. + +New in version 2\.15\. + +Changed in version 2\.16: The `mount` parameter is optional. + +#### Loading OT\-2 Pipettes + +This code sample loads a P1000 Single\-Channel GEN2 pipette in the left mount and a P300 Single\-Channel GEN2 pipette in the right mount. Each pipette uses its own 1000 µL tip rack. + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + tiprack1 = protocol.load_labware( + load_name="opentrons_96_tiprack_1000ul", location=1) + tiprack2 = protocol.load_labware( + load_name="opentrons_96_tiprack_1000ul", location=2) + left = protocol.load_instrument( + instrument_name="p1000_single_gen2", + mount="left", + tip_racks=[tiprack1]) + right = protocol.load_instrument( + instrument_name="p300_multi_gen2", + mount="right", + tip_racks=[tiprack1]) + +``` + +New in version 2\.0\. + +#### Adding Tip Racks + +The `load_instrument()` method includes the optional argument `tip_racks`. This parameter accepts a list of tip rack labware objects, which lets you to specify as many tip racks as you want. You can also edit a pipette’s tip racks after loading it by setting its [`InstrumentContext.tip_racks`](index.html#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks') property. + +Note + +Some methods, like [`configure_nozzle_layout()`](index.html#opentrons.protocol_api.InstrumentContext.configure_nozzle_layout 'opentrons.protocol_api.InstrumentContext.configure_nozzle_layout'), reset a pipette’s tip racks. See [Partial Tip Pickup](index.html#partial-tip-pickup) for more information. + +The advantage of using `tip_racks` is twofold. First, associating tip racks with your pipette allows for automatic tip tracking throughout your protocol. Second, it removes the need to specify tip locations in the [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') method. For example, let’s start by loading loading some labware and instruments like this: + +``` +def run(protocol: protocol_api.ProtocolContext): + tiprack_left = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", location="D1") + tiprack_right = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", location="D2") + left_pipette = protocol.load_instrument( + instrument_name="flex_8channel_1000", mount="left") + right_pipette = protocol.load_instrument( + instrument_name="flex_8channel_1000", + mount="right", + tip_racks=[tiprack_right]) + +``` + +Let’s pick up a tip with the left pipette. We need to specify the location as an argument of `pick_up_tip()`, since we loaded the left pipette without a `tip_racks` argument. + +``` +left_pipette.pick_up_tip(tiprack_left["A1"]) +left_pipette.drop_tip() + +``` + +But now you have to specify `tiprack_left` every time you call `pick_up_tip`, which means you’re doing all your own tip tracking: + +``` +left_pipette.pick_up_tip(tiprack_left["A2"]) +left_pipette.drop_tip() +left_pipette.pick_up_tip(tiprack_left["A3"]) +left_pipette.drop_tip() + +``` + +However, because you specified a tip rack location for the right pipette, the robot will automatically pick up from location `A1` of its associated tiprack: + +``` +right_pipette.pick_up_tip() +right_pipette.drop_tip() + +``` + +Additional calls to `pick_up_tip` will automatically progress through the tips in the right rack: + +``` +right_pipette.pick_up_tip() # picks up from A2 +right_pipette.drop_tip() +right_pipette.pick_up_tip() # picks up from A3 +right_pipette.drop_tip() + +``` + +New in version 2\.0\. + +See also [Building Block Commands](index.html#v2-atomic-commands) and [Complex Commands](index.html#v2-complex-commands). + +#### Adding Trash Containers + +The API automatically assigns a [`trash_container`](index.html#opentrons.protocol_api.InstrumentContext.trash_container 'opentrons.protocol_api.InstrumentContext.trash_container') to pipettes, if one is available in your protocol. The `trash_container` is where the pipette will dispose tips when you call [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') with no arguments. You can change the trash container, if you don’t want to use the default. + +One example of when you might want to change the trash container is a Flex protocol that goes through a lot of tips. In a case where the protocol uses two pipettes, you could load two trash bins and assign one to each pipette: + +``` +left_pipette = protocol.load_instrument( + instrument_name="flex_8channel_1000", mount="left" +) +right_pipette = protocol.load_instrument( + instrument_name="flex_8channel_50", mount="right" +) +left_trash = load_trash_bin("A3") +right_trash = load_trash_bin("B3") +left_pipette.trash_container = left_trash +right_pipette.trash_container = right_trash + +``` + +Another example is a Flex protocol that uses a waste chute. Say you want to only dispose labware in the chute, and you want the pipette to drop tips in a trash bin. You can implicitly get the trash bin to be the pipette’s `trash_container` based on load order, or you can ensure it by setting it after all the load commands: + +``` +pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left" +) +chute = protocol.load_waste_chute() # default because loaded first +trash = protocol.load_trash_bin("A3") +pipette.trash_container = trash # overrides default + +``` + +New in version 2\.0\. + +Changed in version 2\.16: Added support for `TrashBin` and `WasteChute` objects. + +### Pipette Characteristics + +Each Opentrons pipette has different capabilities, which you’ll want to take advantage of in your protocols. This page covers some fundamental pipette characteristics. + +[Multi\-Channel Movement](#new-multichannel-pipettes) gives examples of how multi\-channel pipettes move around the deck by using just one of their channels as a reference point. Taking this into account is important for commanding your pipettes to perform actions in the correct locations. + +[Pipette Flow Rates](#new-plunger-flow-rates) discusses how quickly each type of pipette can handle liquids. The defaults are designed to operate quickly, based on the pipette’s hardware and assuming that you’re handling aqueous liquids. You can speed up or slow down a pipette’s flow rate to suit your protocol’s needs. + +Finally, the volume ranges of pipettes affect what you can do with them. The volume ranges for current pipettes are listed on the [Loading Pipettes](index.html#loading-pipettes) page. The [OT\-2 Pipette Generations](#ot2-pipette-generations) section of this page describes how the API behaves when running protocols that specify older OT\-2 pipettes. + +#### Multi\-Channel Movement + +All [building block](index.html#v2-atomic-commands) and [complex commands](index.html#v2-complex-commands) work with single\- and multi\-channel pipettes. + +To keep the protocol API consistent when using single\- and multi\-channel pipettes, commands treat the back left channel of a multi\-channel pipette as its _primary channel_. Location arguments of pipetting commands use the primary channel. The [`InstrumentContext.configure_nozzle_layout()`](index.html#opentrons.protocol_api.InstrumentContext.configure_nozzle_layout 'opentrons.protocol_api.InstrumentContext.configure_nozzle_layout') method can change the pipette’s primary channel, using its `start` parameter. See [Partial Tip Pickup](index.html#partial-tip-pickup) for more information. + +With a pipette’s default settings, you can generally access the wells indicated in the table below. Moving to any other well may cause the pipette to crash. + +| Channels | 96\-well plate | 384\-well plate | +| -------- | ---------------- | ---------------- | +| 1 | Any well, A1–H12 | Any well, A1–P24 | +| 8 | A1–A12 | A1–B24 | +| 96 | A1 only | A1–B2 | + +Also, you should apply any location offset, such as [`Well.top()`](index.html#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') or [`Well.bottom()`](index.html#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom'), to the well accessed by the primary channel. Since all of the pipette’s channels move together, each channel will have the same offset relative to the well that it is over. + +Finally, because each multi\-channel pipette has only one motor, they always aspirate and dispense on all channels simultaneously. + +##### 8\-Channel, 96\-Well Plate Example + +To demonstrate these concepts, let’s write a protocol that uses a Flex 8\-Channel Pipette and a 96\-well plate. We’ll then aspirate and dispense a liquid to different locations on the same well plate. To start, let’s load a pipette in the right mount and add our labware. + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel":"2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # Load a tiprack for 1000 µL tips + tiprack1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="D1") + # Load a 96-well plate + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="C1") + # Load an 8-channel pipette on the right mount + right = protocol.load_instrument( + instrument_name="flex_8channel_1000", + mount="right", + tip_racks=[tiprack1]) + +``` + +After loading our instruments and labware, let’s tell the robot to pick up a pipette tip from location `A1` in `tiprack1`: + +``` +right.pick_up_tip() + +``` + +With the backmost pipette channel above location A1 on the tip rack, all eight channels are above the eight tip rack wells in column 1\. + +After picking up a tip, let’s tell the robot to aspirate 300 µL from the well plate at location `A2`: + +``` +right.aspirate(volume=300, location=plate["A2"]) + +``` + +With the backmost pipette tip above location A2 on the well plate, all eight channels are above the eight wells in column 2\. + +Finally, let’s tell the robot to dispense 300 µL into the well plate at location `A3`: + +``` +right.dispense(volume=300, location=plate["A3"].top()) + +``` + +With the backmost pipette tip above location A3, all eight channels are above the eight wells in column 3\. The pipette will dispense liquid into all the wells simultaneously. + +##### 8\-Channel, 384\-Well Plate Example + +In general, you should specify wells in the first row of a well plate when using multi\-channel pipettes. An exception to this rule is when using 384\-well plates. The greater well density means the nozzles of a multi\-channel pipette can only access every other well in a column. Specifying well A1 accesses every other well starting with the first (rows A, C, E, G, I, K, M, and O). Similarly, specifying well B1 also accesses every other well, but starts with the second (rows B, D, F, H, J, L, N, and P). + +To demonstrate these concepts, let’s write a protocol that uses a Flex 8\-Channel Pipette and a 384\-well plate. We’ll then aspirate and dispense a liquid to different locations on the same well plate. To start, let’s load a pipette in the right mount and add our labware. + +``` +def run(protocol: protocol_api.ProtocolContext): + # Load a tiprack for 200 µL tips + tiprack1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", location="D1") + # Load a well plate + plate = protocol.load_labware( + load_name="corning_384_wellplate_112ul_flat", location="D2") + # Load an 8-channel pipette on the right mount + right = protocol.load_instrument( + instrument_name="flex_8channel_1000", + mount="right", + tip_racks=[tiprack1]) + +``` + +After loading our instruments and labware, let’s tell the robot to pick up a pipette tip from location `A1` in `tiprack1`: + +``` +right.pick_up_tip() + +``` + +With the backmost pipette channel above location A1 on the tip rack, all eight channels are above the eight tip rack wells in column 1\. + +After picking up a tip, let’s tell the robot to aspirate 100 µL from the well plate at location `A1`: + +``` +right.aspirate(volume=100, location=plate["A1"]) + +``` + +The eight pipette channels will only aspirate from every other well in the column: A1, C1, E1, G1, I1, K1, M1, and O1\. + +Finally, let’s tell the robot to dispense 100 µL into the well plate at location `B1`: + +``` +right.dispense(volume=100, location=plate["B1"]) + +``` + +The eight pipette channels will only dispense into every other well in the column: B1, D1, F1, H1, J1, L1, N1, and P1\. + +#### Pipette Flow Rates + +Measured in µL/s, the flow rate determines how much liquid a pipette can aspirate, dispense, and blow out. Opentrons pipettes have their own default flow rates. The API lets you change the flow rate on a loaded [`InstrumentContext`](index.html#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.InstrumentContext') by altering the [`InstrumentContext.flow_rate`](index.html#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate') properties listed below. + +- Aspirate: `InstrumentContext.flow_rate.aspirate` +- Dispense: `InstrumentContext.flow_rate.dispense` +- Blow out: `InstrumentContext.flow_rate.blow_out` + +These flow rate properties operate independently. This means you can specify different flow rates for each property within the same protocol. For example, let’s load a simple protocol and set different flow rates for the attached pipette. + +``` +def run(protocol: protocol_api.ProtocolContext): + tiprack1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="D1") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack1]) + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="D3") + pipette.pick_up_tip() + +``` + +Let’s tell the robot to aspirate, dispense, and blow out the liquid using default flow rates. Notice how you don’t need to specify a `flow_rate` attribute to use the defaults: + +``` +pipette.aspirate(200, plate["A1"]) # 160 µL/s +pipette.dispense(200, plate["A2"]) # 160 µL/s +pipette.blow_out() # 80 µL/s + +``` + +Now let’s change the flow rates for each action: + +``` +pipette.flow_rate.aspirate = 50 +pipette.flow_rate.dispense = 100 +pipette.flow_rate.blow_out = 75 +pipette.aspirate(200, plate["A1"]) # 50 µL/s +pipette.dispense(200, plate["A2"]) # 100 µL/s +pipette.blow_out() # 75 µL/s + +``` + +These flow rates will remain in effect until you change the `flow_rate` attribute again _or_ call `configure_for_volume()`. Calling `configure_for_volume()` always resets all pipette flow rates to the defaults for the mode that it sets. + +Note + +In API version 2\.13 and earlier, [`InstrumentContext.speed`](index.html#opentrons.protocol_api.InstrumentContext.speed 'opentrons.protocol_api.InstrumentContext.speed') offered similar functionality to `.flow_rate`. It attempted to set the plunger speed in mm/s. Due to technical limitations, that speed could only be approximate. You must use `.flow_rate` in version 2\.14 and later, and you should consider replacing older code that sets `.speed`. + +New in version 2\.0\. + +##### Flex Pipette Flow Rates + +The default flow rates for Flex pipettes depend on the maximum volume of the pipette and the capacity of the currently attached tip. For each pipette–tip configuration, the default flow rate is the same for aspirate, dispense, and blowout actions. + +| Pipette Model | Tip Capacity (µL) | Flow Rate (µL/s) | +| ----------------------------------- | ----------------- | ---------------- | +| 50 µL (1\- and 8\-channel) | All capacities | 57 | +| 1000 µL (1\-, 8\-, and 96\-channel) | 50 | 478 | +| 1000 µL (1\-, 8\-, and 96\-channel) | 200 | 716 | +| 1000 µL (1\-, 8\-, and 96\-channel) | 1000 | 716 | + +Additionally, all Flex pipettes have a well bottom clearance of 1 mm for aspirate and dispense actions. + +##### OT\-2 Pipette Flow Rates + +The following table provides data on the default aspirate, dispense, and blowout flow rates (in µL/s) for OT\-2 GEN2 pipettes. Default flow rates are the same across all three actions. + +| Pipette Model | Volume (µL) | Flow Rates (µL/s) | +| -------------------------- | ----------- | ----------------------------------------------------------- | +| P20 Single\-Channel GEN2 | 1–20 | _ API v2\.6 or higher: 7\.56 _ API v2\.5 or lower: 3\.78 | +| P300 Single\-Channel GEN2 | 20–300 | _ API v2\.6 or higher: 92\.86 _ API v2\.5 or lower: 46\.43 | +| P1000 Single\-Channel GEN2 | 100–1000 | _ API v2\.6 or higher: 274\.7 _ API v2\.5 or lower: 137\.35 | +| P20 Multi\-Channel GEN2 | 1–20 | 7\.6 | +| P300 Multi\-Channel GEN2 | 20–300 | 94 | + +Additionally, all OT\-2 GEN2 pipettes have a default head speed of 400 mm/s and a well bottom clearance of 1 mm for aspirate and dispense actions. + +#### OT\-2 Pipette Generations + +The OT\-2 works with the GEN1 and GEN2 pipette models. The newer GEN2 pipettes have different volume ranges than the older GEN1 pipettes. With some exceptions, the volume ranges for GEN2 pipettes overlap those used by the GEN1 models. If your protocol specifies a GEN1 pipette, but you have a GEN2 pipette with a compatible volume range, you can still run your protocol. The OT\-2 will consider the GEN2 pipette to have the same minimum volume as the GEN1 pipette. The following table lists the volume compatibility between the GEN2 and GEN1 pipettes. + +| GEN2 Pipette | GEN1 Pipette | GEN1 Volume | +| -------------------------- | -------------------------- | ------------ | +| P20 Single\-Channel GEN2 | P10 Single\-Channel GEN1 | 1\-10 µL | +| P20 Multi\-Channel GEN2 | P10 Multi\-Channel GEN1 | 1\-10 µL | +| P300 Single\-Channel GEN2 | P300 Single\-Channel GEN1 | 30\-300 µL | +| P300 Multi\-Channel GEN2 | P300 Multi\-Channel GEN1 | 20\-200 µL | +| P1000 Single\-Channel GEN2 | P1000 Single\-Channel GEN1 | 100\-1000 µL | + +The single\- and multi\-channel P50 GEN1 pipettes are the exceptions here. If your protocol uses a P50 GEN1 pipette, there is no backward compatibility with a related GEN2 pipette. To replace a P50 GEN1 with a corresponding GEN2 pipette, edit your protocol to load a P20 Single\-Channel GEN2 (for volumes below 20 µL) or a P300 Single\-Channel GEN2 (for volumes between 20 and 50 µL). + +### Partial Tip Pickup + +The 96\-channel pipette occupies both pipette mounts on Flex, so it’s not possible to attach another pipette at the same time. Partial tip pickup lets you perform some of the same actions that you would be able to perform with a second pipette. As of version 2\.16 of the API, you can configure the 96\-channel pipette to pick up a single column of tips, similar to the behavior of an 8\-channel pipette. + +#### Nozzle Layout + +Use the [`configure_nozzle_layout()`](index.html#opentrons.protocol_api.InstrumentContext.configure_nozzle_layout 'opentrons.protocol_api.InstrumentContext.configure_nozzle_layout') method to choose how many tips the 96\-channel pipette will pick up. The method’s `style` parameter accepts special layout constants. You must import these constants at the top of your protocol, or you won’t be able to configure the pipette for partial tip pickup. + +At minimum, import the API from the `opentrons` package: + +``` +from opentrons import protocol_api + +``` + +Then when you call `configure_nozzle_layout` later in your protocol, you can set `style=protocol_api.COLUMN`. + +For greater convenience, also import the individual layout constants that you plan to use in your protocol: + +``` +from opentrons.protocol_api import COLUMN, ALL + +``` + +Then when you call `configure_nozzle_layout` later in your protocol, you can set `style=COLUMN`. + +Here is the start of a protocol that performs both imports, loads a 96\-channel pipette, and sets it to pick up a single column of tips. + +``` +from opentrons import protocol_api +from opentrons.protocol_api import COLUMN, ALL + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + column_rack = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", + location="D3" + ) + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument("flex_96channel_1000") + pipette.configure_nozzle_layout( + style=COLUMN, + start="A12", + tip_racks=[column_rack] + ) + +``` + +New in version 2\.16\. + +Let’s unpack some of the details of this code. + +First, we’ve given a special name to the tip rack, `column_rack`. You can name your tip racks whatever you like, but if you’re performing full pickup and partial pickup in the same protocol, you’ll need to keep them separate. See [Tip Rack Adapters](#partial-tip-rack-adapters) below. + +Next, we load the 96\-channel pipette. Note that [`load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') only has a single argument. The 96\-channel pipette occupies both mounts, so `mount` is omissible. The `tip_racks` argument is always optional. But it would have no effect to declare it here, because every call to `configure_nozzle_layout()` resets the pipette’s [`InstrumentContext.tip_racks`](index.html#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks') property. + +Finally, we configure the nozzle layout, with three arguments. + +> - The `style` parameter directly accepts the `COLUMN` constant, since we imported it at the top of the protocol. +> - The `start` parameter accepts a nozzle name, representing the back\-left nozzle in the layout, as a string. `"A12"` tells the pipette to use its rightmost column of nozzles for pipetting. +> - The `tip_racks` parameter tells the pipette which racks to use for tip tracking, just like [adding tip racks](index.html#pipette-tip-racks) when loading a pipette. + +In this configuration, pipetting actions will use a single column: + +``` +# configured in COLUMN mode +pipette.pick_up_tip() # picks up A1-H1 from tip rack +pipette.drop_tip() +pipette.pick_up_tip() # picks up A2-H2 from tip rack + +``` + +Warning + +[`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') always accepts a `location` argument, regardless of nozzle configuration. Do not pass a value that would lead the pipette to line up over more unused tips than specified by the current layout. For example, setting `COLUMN` layout and then calling `pipette.pick_up_tip(tip_rack["A2"])` on a full tip rack will lead to unexpected pipetting behavior and potential crashes. + +#### Tip Rack Adapters + +You can use both partial and full tip pickup in the same protocol. This requires having some tip racks directly on the deck, and some tip racks in the tip rack adapter. + +Do not use a tip rack adapter when performing partial tip pickup. Instead, place the tip rack on the deck. During partial tip pickup, the 96\-channel pipette lowers onto the tip rack in a horizontally offset position. If the tip rack were in its adapter, the pipette would collide with the adapter’s posts, which protrude above the top of the tip rack. If you configure the pipette for partial pickup and then call `pick_up_tip()` on a tip rack that’s loaded onto an adapter, the API will raise an error. + +On the other hand, you must use the tip rack adapter for full tip pickup. If the 96\-channel pipette is in a full layout, either by default or by configuring `style=ALL`, and you then call `pick_up_tip()` on a tip rack that’s not in an adapter, the API will raise an error. + +When switching between full and partial pickup, you may want to organize your tip racks into lists, depending on whether they’re loaded on adapters or not. + +``` +tips_1 = protocol.load_labware( + "opentrons_flex_96_tiprack_1000ul", "C1" +) +tips_2 = protocol.load_labware( + "opentrons_flex_96_tiprack_1000ul", "D1" +) +tips_3 = protocol.load_labware( + "opentrons_flex_96_tiprack_1000ul", "C3", + adapter="opentrons_flex_96_tiprack_adapter" +) +tips_4 = protocol.load_labware( + "opentrons_flex_96_tiprack_1000ul", "D3", + adapter="opentrons_flex_96_tiprack_adapter" +) + +partial_tip_racks = [tips_1, tips_2] +full_tip_racks = [tips_3, tips_4] + +``` + +Tip + +It’s also good practice to keep separate lists of tip racks when using multiple partial tip pickup configurations (i.e., using both column 1 and column 12 in the same protocol). This improves positional accuracy when picking up tips. Additionally, use Labware Position Check in the Opentrons App to ensure that the partial configuration is well\-aligned to the rack. + +Now, when you configure the nozzle layout, you can reference the appropriate list as the value of `tip_racks`: + +``` +pipette.configure_nozzle_layout( + style=COLUMN, + start="A12", + tip_racks=partial_tip_racks +) +# partial pipetting commands go here + +pipette.configure_nozzle_layout( + style=ALL, + tip_racks=full_tip_racks +) +pipette.pick_up_tip() # picks up full rack in C1 + +``` + +This keeps tip tracking consistent across each type of pickup. And it reduces the risk of errors due to the incorrect presence or absence of a tip rack adapter. + +#### Tip Pickup and Conflicts + +During partial tip pickup, 96\-channel pipette moves into spaces above adjacent slots. To avoid crashes, the API prevents you from performing partial tip pickup when there is tall labware in these spaces. The current nozzle layout determines which labware can safely occupy adjacent slots. + +The API will raise errors for potential labware crashes when using a column nozzle configuration. Nevertheless, it’s a good idea to do the following when working with partial tip pickup: + +> - Plan your deck layout carefully. Make a diagram and visualize everywhere the pipette will travel. +> - Simulate your protocol and compare the run preview to your expectations of where the pipette will travel. +> - Perform a dry run with only tip racks on the deck. Have the Emergency Stop Pendant handy in case you see an impending crash. + +For column pickup, Opentrons recommends using the nozzles in column 12 of the pipette: + +``` +pipette.configure_nozzle_layout( + style=COLUMN, + start="A12", +) + +``` + +When using column 12, the pipette overhangs space to the left of wherever it is picking up tips or pipetting. For this reason, it’s a good idea to organize tip racks front to back on the deck. If you place them side by side, the rack to the right will be inaccessible. For example, let’s load three tip racks in the front left corner of the deck: + +``` +tips_C1 = protocol.load_labware("opentrons_flex_96_tiprack_1000ul", "C1") +tips_D1 = protocol.load_labware("opentrons_flex_96_tiprack_1000ul", "D1") +tips_D2 = protocol.load_labware("opentrons_flex_96_tiprack_1000ul", "D2") + +``` + +Now the pipette will be able to access the racks in column 1 only. `pick_up_tip(tips_D2["A1"])` will raise an error due to the tip rack immediately to its left, in slot D1\. There a couple of ways to avoid this error: + +> - Load the tip rack in a different slot, with no tall labware to its left. +> - Use all the tips in slot D1 first, and then use [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') to make space for the pipette before picking up tips from D2\. + +You would get a similar error trying to aspirate from or dispense into a well plate in slot D3, since there is a tip rack to the left. + +Tip + +When using column 12 for partial tip pickup and pipetting, generally organize your deck with the shortest labware on the left side of the deck, and the tallest labware on the right side. + +If your application can’t accommodate a deck layout that works well with column 12, you can configure the 96\-channel pipette to pick up tips with column 1: + +``` +pipette.configure_nozzle_layout( + style=COLUMN, + start="A1", +) + +``` + +Note + +When using a column 1 layout, the pipette can’t reach the rightmost portion of labware in slots A3–D3\. Any well that is within 29 mm of the right edge of the slot may be inaccessible. Use a column 12 layout if you need to pipette in that area. + +### Volume Modes + +The Flex 1\-Channel 50 µL and Flex 8\-Channel 50 µL pipettes must operate in a low\-volume mode to accurately dispense very small volumes of liquid. Set the volume mode by calling [`InstrumentContext.configure_for_volume()`](index.html#opentrons.protocol_api.InstrumentContext.configure_for_volume 'opentrons.protocol_api.InstrumentContext.configure_for_volume') with the amount of liquid you plan to aspirate, in µL: + +``` +pipette50.configure_for_volume(1) +pipette50.pick_up_tip() +pipette50.aspirate(1, plate["A1"]) + +``` + +New in version 2\.15\. + +Passing different values to `configure_for_volume()` changes the minimum and maximum volume of Flex 50 µL pipettes as follows: + +| Value | Minimum Volume (µL) | Maximum Volume (µL) | +| ------ | ------------------- | ------------------- | +| 1–4\.9 | 1 | 30 | +| 5–50 | 5 | 50 | + +Note + +The pipette must not contain liquid when you call `configure_for_volume()`, or the API will raise an error. + +Also, if the pipette is in a well location that may contain liquid, it will move upward to ensure it is not immersed in liquid before changing its mode. Calling `configure_for_volume()` _before_ `pick_up_tip()` helps to avoid this situation. + +In a protocol that handles many different volumes, it’s a good practice to call `configure_for_volume()` once for each [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') or [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'), specifying the volume that you are about to handle. When operating with a list of volumes, nest `configure_for_volume()` inside a `for` loop to ensure that the pipette is properly configured for each volume: + +``` +volumes = [1, 2, 3, 4, 1, 5, 2, 8] +sources = plate.columns()[0] +destinations = plate.columns()[1] +for i in range(8): + pipette50.configure_for_volume(volumes[i]) + pipette50.pick_up_tip() + pipette50.aspirate(volume=volumes[i], location=sources[i]) + pipette50.dispense(location=destinations[i]) + pipette50.drop_tip() + +``` + +If you know that all your liquid handling will take place in a specific mode, then you can call `configure_for_volume()` just once with a representative volume. Or if all the volumes correspond to the pipette’s default mode, you don’t have to call `configure_for_volume()` at all. + +Opentrons pipettes are configurable devices used to move liquids throughout the working area during the execution of protocols. Flex and OT\-2 each have their own pipettes, which are available for use in the Python API. + +Pages in this section of the documentation cover: + +> - [Loading pipettes](index.html#loading-pipettes) into your protocol. +> - [Pipette characteristics](index.html#pipette-characteristics), such as how fast they can move liquid and how they move around the deck. +> - The [partial tip pickup](index.html#partial-tip-pickup) configuration for the Flex 96\-Channel Pipette, which uses only 8 channels for pipetting. Full and partial tip pickup can be combined in a single protocol. +> - The [volume modes](index.html#pipette-volume-modes) of Flex 50 µL pipettes, which must operate in low\-volume mode to accurately dispense very small volumes of liquid. + +For information about liquid handling, see [Building Block Commands](index.html#v2-atomic-commands) and [Complex Commands](index.html#v2-complex-commands). + +## Building Block Commands + +### Manipulating Pipette Tips + +Your robot needs to attach a disposable tip to the pipette before it can aspirate or dispense liquids. The API provides three basic functions that help the robot attach and manage pipette tips during a protocol run. These methods are [`InstrumentContext.pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'), [`InstrumentContext.drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip'), and [`InstrumentContext.return_tip()`](index.html#opentrons.protocol_api.InstrumentContext.return_tip 'opentrons.protocol_api.InstrumentContext.return_tip'). Respectively, these methods tell the robot to pick up a tip from a tip rack, drop a tip into the trash (or another location), and return a tip to its location in the tip rack. + +The following sections demonstrate how to use each method and include sample code. The examples used here assume that you’ve loaded the pipettes and labware from the basic [protocol template](index.html#protocol-template). + +#### Picking Up a Tip + +To pick up a tip, call the [`pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') method without any arguments: + +``` +pipette.pick_up_tip() + +``` + +When added to the protocol template, this simple statement works because the API knows which tip rack is associated with `pipette`, as indicated by `tip_racks=[tiprack_1]` in the [`load_instrument()`](index.html#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') call. And it knows the on\-deck location of the tip rack (slot D3 on Flex, slot 3 on OT\-2\) from the `location` argument of [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). Given this information, the robot moves to the tip rack and picks up a tip from position A1 in the rack. On subsequent calls to `pick_up_tip()`, the robot will use the next available tip. For example: + +``` +pipette.pick_up_tip() # picks up tip from rack location A1 +pipette.drop_tip() # drops tip in trash bin +pipette.pick_up_tip() # picks up tip from rack location B1 +pipette.drop_tip() # drops tip in trash bin + +``` + +If you omit the `tip_rack` argument from the `pipette` variable, the API will raise an error. In that case, you must pass the tip rack’s location to `pick_up_tip` like this: + +``` +pipette.pick_up_tip(tiprack_1["A1"]) +pipette.drop_tip() +pipette.pick_up_tip(tiprack_1["B1"]) + +``` + +In most cases, it’s best to associate tip racks with a pipette and let the API automatically track pickup location for you. This also makes it easy to pick up tips when iterating over a loop, as shown in the next section. + +New in version 2\.0\. + +#### Automating Tip Pick Up + +When used with Python’s [`range`](https://docs.python.org/3/library/stdtypes.html#range '(in Python v3.12)') class, a `for` loop brings automation to the tip pickup and tracking process. It also eliminates the need to call `pick_up_tip()` multiple times. For example, this snippet tells the robot to sequentially use all the tips in a 96\-tip rack: + +``` +for i in range(96): + pipette.pick_up_tip() + # liquid handling commands + pipette.drop_tip() + +``` + +If your protocol requires a lot of tips, add a second tip rack to the protocol. Then, associate it with your pipette and increase the number of repetitions in the loop. The robot will work through both racks. + +First, add another tip rack to the sample protocol: + +``` +tiprack_2 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", + location="C3" +) + +``` + +Next, change the pipette’s `tip_rack` property to include the additional rack: + +``` +pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1, tiprack_2], +) + +``` + +Finally, iterate over a larger range: + +``` +for i in range(192): + pipette.pick_up_tip() + # liquid handling commands + pipette.drop_tip() + +``` + +For a more advanced “real\-world” example, review the [off\-deck location protocol](index.html#off-deck-location) on the [Moving Labware](index.html#moving-labware) page. This example also uses a `for` loop to iterate through a tip rack, but it includes other commands that pause the protocol and let you replace an on\-deck tip rack with another rack stored in an off\-deck location. + +#### Dropping a Tip + +To drop a tip in the pipette’s trash container, call the [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') method with no arguments: + +``` +pipette.pick_up_tip() + +``` + +You can specify where to drop the tip by passing in a location. For example, this code drops a tip in the trash bin and returns another tip to to a previously used well in a tip rack: + +``` +pipette.pick_up_tip() # picks up tip from rack location A1 +pipette.drop_tip() # drops tip in default trash container +pipette.pick_up_tip() # picks up tip from rack location B1 +pipette.drop_tip(tiprack["A1"]) # drops tip in rack location A1 + +``` + +New in version 2\.0\. + +Another use of the `location` parameter is to drop a tip in a specific trash container. For example, calling `pipette.drop_tip(chute)` will dispose tips in the waste chute, even if the pipette’s default trash container is a trash bin: + +``` +pipette.pick_up_tip() # picks up tip from rack location A1 +pipette.drop_tip() # drops tip in default trash container +pipette.pick_up_tip() # picks up tip from rack location B1 +pipette.drop_tip(chute) # drops tip in waste chute + +``` + +New in version 2\.16\. + +#### Returning a Tip + +To return a tip to its original location, call the [`return_tip()`](index.html#opentrons.protocol_api.InstrumentContext.return_tip 'opentrons.protocol_api.InstrumentContext.return_tip') method with no arguments: + +``` +pipette.return_tip() + +``` + +New in version 2\.0\. + +Note + +You can’t return tips with a pipette that’s configured to use [partial tip pickup](index.html#partial-tip-pickup). This restriction ensures that the pipette has clear access to unused tips. For example, a 96\-channel pipette in column configuration can’t reach column 2 unless column 1 is empty. + +If you call `return_tip()` while using partial tip pickup, the API will raise an error. Use `drop_tip()` to dispose the tips instead. + +#### Working With Used Tips + +Currently, the API considers tips as “used” after being picked up. For example, if the robot picked up a tip from rack location A1 and then returned it to the same location, it will not attempt to pick up this tip again, unless explicitly specified. Instead, the robot will pick up a tip starting from rack location B1\. For example: + +``` +pipette.pick_up_tip() # picks up tip from rack location A1 +pipette.return_tip() # drops tip in rack location A1 +pipette.pick_up_tip() # picks up tip from rack location B1 +pipette.drop_tip() # drops tip in trash bin +pipette.pick_up_tip(tiprack_1["A1"]) # picks up tip from rack location A1 + +``` + +Early API versions treated returned tips as unused items. They could be picked up again without an explicit argument. For example: + +``` +pipette.pick_up_tip() # picks up tip from rack location A1 +pipette.return_tip() # drops tip in rack location A1 +pipette.pick_up_tip() # picks up tip from rack location A1 + +``` + +Changed in version 2\.2\. + +### Liquid Control + +After attaching a tip, your robot is ready to aspirate, dispense, and perform other liquid handling tasks. The API includes methods that help you perform these actions and the following sections show how to use them. The examples used here assume that you’ve loaded the pipettes and labware from the basic [protocol template](index.html#protocol-template). + +#### Aspirate + +To draw liquid up into a pipette tip, call the [`InstrumentContext.aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') method. Using this method, you can specify the aspiration volume in µL, the well location, and pipette flow rate. Other parameters let you position the pipette within a well. For example, this snippet tells the robot to aspirate 200 µL from well location A1\. + +``` +pipette.pick_up_tip() +pipette.aspirate(200, plate["A1"]) + +``` + +If the pipette doesn’t move, you can specify an additional aspiration action without including a location. To demonstrate, this code snippet pauses the protocol, automatically resumes it, and aspirates a second time from `plate["A1"]`). + +``` +pipette.pick_up_tip() +pipette.aspirate(200, plate["A1"]) +protocol.delay(seconds=5) # pause for 5 seconds +pipette.aspirate(100) # aspirate 100 µL at current position + +``` + +Now our pipette holds 300 µL. + +##### Aspirate by Well or Location + +The [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') method includes a `location` parameter that accepts either a [`Well`](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or a [`Location`](index.html#opentrons.types.Location 'opentrons.types.Location'). + +If you specify a well, like `plate["A1"]`, the pipette will aspirate from a default position 1 mm above the bottom center of that well. To change the default clearance, first set the `aspirate` attribute of [`well_bottom_clearance`](index.html#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance'): + +``` +pipette.pick_up_tip +pipette.well_bottom_clearance.aspirate = 2 # tip is 2 mm above well bottom +pipette.aspirate(200, plate["A1"]) + +``` + +You can also aspirate from a location along the center vertical axis within a well using the [`Well.top()`](index.html#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') and [`Well.bottom()`](index.html#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom') methods. These methods move the pipette to a specified distance relative to the top or bottom center of a well: + +``` +pipette.pick_up_tip() +depth = plate["A1"].bottom(z=2) # tip is 2 mm above well bottom +pipette.aspirate(200, depth) + +``` + +See also: + +- [Default Positions](index.html#new-default-op-positions) for information about controlling pipette height for a particular pipette. +- [Position Relative to Labware](index.html#position-relative-labware) for information about controlling pipette height from within a well. +- [Move To](index.html#move-to) for information about moving a pipette to any reachable deck location. + +##### Aspiration Flow Rates + +Flex and OT\-2 pipettes aspirate at [default flow rates](index.html#new-plunger-flow-rates) measured in µL/s. Specifying the `rate` parameter multiplies the flow rate by that value. As a best practice, don’t set the flow rate higher than 3x the default. For example, this code causes the pipette to aspirate at twice its normal rate: + +``` +pipette.aspirate(200, plate["A1"], rate=2.0) + +``` + +New in version 2\.0\. + +#### Dispense + +To dispense liquid from a pipette tip, call the [`InstrumentContext.dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') method. Using this method, you can specify the dispense volume in µL, the well location, and pipette flow rate. Other parameters let you position the pipette within a well. For example, this snippet tells the robot to dispense 200 µL into well location B1\. + +``` +pipette.dispense(200, plate["B1"]) + +``` + +Note + +In API version 2\.16 and earlier, you could pass a `volume` argument to `dispense()` greater than what was aspirated into the pipette. In this case, the API would ignore `volume` and dispense the pipette’s [`current_volume`](index.html#opentrons.protocol_api.InstrumentContext.current_volume 'opentrons.protocol_api.InstrumentContext.current_volume'). The robot _would not_ move the plunger lower as a result. + +In version 2\.17 and later, passing such values raises an error. + +To move the plunger a small extra amount, add a [push out](#push-out-dispense). Or to move it a large amount, use [blow out](#blow-out). + +If the pipette doesn’t move, you can specify an additional dispense action without including a location. To demonstrate, this code snippet pauses the protocol, automatically resumes it, and dispense a second time from location B1\. + +``` +pipette.dispense(100, plate["B1"]) +protocol.delay(seconds=5) # pause for 5 seconds +pipette.dispense(100) # dispense 100 µL at current position + +``` + +##### Dispense by Well or Location + +The [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') method includes a `location` parameter that accepts either a [`Well`](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or a [`Location`](index.html#opentrons.types.Location 'opentrons.types.Location'). + +If you specify a well, like `plate["B1"]`, the pipette will dispense from a default position 1 mm above the bottom center of that well. To change the default clearance, you would call [`well_bottom_clearance`](index.html#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance'): + +``` +pipette.well_bottom_clearance.dispense=2 # tip is 2 mm above well bottom +pipette.dispense(200, plate["B1"]) + +``` + +You can also dispense from a location along the center vertical axis within a well using the [`Well.top()`](index.html#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') and [`Well.bottom()`](index.html#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom') methods. These methods move the pipette to a specified distance relative to the top or bottom center of a well: + +``` +depth = plate["B1"].bottom(z=2) # tip is 2 mm above well bottom +pipette.dispense(200, depth) + +``` + +See also: + +- [Default Positions](index.html#new-default-op-positions) for information about controlling pipette height for a particular pipette. +- [Position Relative to Labware](index.html#position-relative-labware) for formation about controlling pipette height from within a well. +- [Move To](index.html#move-to) for information about moving a pipette to any reachable deck location. + +##### Dispense Flow Rates + +Flex and OT\-2 pipettes dispense at [default flow rates](index.html#new-plunger-flow-rates) measured in µL/s. Adding a number to the `rate` parameter multiplies the flow rate by that value. As a best practice, don’t set the flow rate higher than 3x the default. For example, this code causes the pipette to dispense at twice its normal rate: + +``` +pipette.dispense(200, plate["B1"], rate=2.0) + +``` + +New in version 2\.0\. + +##### Push Out After Dispense + +The optional `push_out` parameter of `dispense()` helps ensure all liquid leaves the tip. Use `push_out` for applications that require moving the pipette plunger lower than the default, without performing a full [blow out](#blow-out). + +For example, this dispense action moves the plunger the equivalent of an additional 5 µL beyond where it would stop if `push_out` was set to zero or omitted: + +``` +pipette.pick_up_tip() +pipette.aspirate(100, plate["A1"]) +pipette.dispense(100, plate["B1"], push_out=5) +pipette.drop_tip() + +``` + +New in version 2\.15\. + +#### Blow Out + +To blow an extra amount of air through the pipette’s tip, call the [`InstrumentContext.blow_out()`](index.html#opentrons.protocol_api.InstrumentContext.blow_out 'opentrons.protocol_api.InstrumentContext.blow_out') method. You can use a specific well in a well plate or reservoir as the blowout location. If no location is specified, the pipette will blowout from its current well position: + +``` +pipette.blow_out() + +``` + +You can also specify a particular well as the blowout location: + +``` +pipette.blow_out(plate["B1"]) + +``` + +Many protocols use a trash container for blowing out the pipette. You can specify the pipette’s current trash container as the blowout location by using the [`InstrumentContext.trash_container`](index.html#opentrons.protocol_api.InstrumentContext.trash_container 'opentrons.protocol_api.InstrumentContext.trash_container') property: + +``` +pipette.blow_out(pipette.trash_container) + +``` + +New in version 2\.0\. + +Changed in version 2\.16: Added support for `TrashBin` and `WasteChute` locations. + +#### Touch Tip + +The [`InstrumentContext.touch_tip()`](index.html#opentrons.protocol_api.InstrumentContext.touch_tip 'opentrons.protocol_api.InstrumentContext.touch_tip') method moves the pipette so the tip touches each wall of a well. A touch tip procedure helps knock off any droplets that might cling to the pipette’s tip. This method includes optional arguments that allow you to control where the tip will touch the inner walls of a well and the touch speed. Calling [`touch_tip()`](index.html#opentrons.protocol_api.InstrumentContext.touch_tip 'opentrons.protocol_api.InstrumentContext.touch_tip') without arguments causes the pipette to touch the well walls from its current location: + +``` +pipette.touch_tip() + +``` + +##### Touch Location + +These optional location arguments give you control over where the tip will touch the side of a well. + +This example demonstrates touching the tip in a specific well: + +``` +pipette.touch_tip(plate["B1"]) + +``` + +This example uses an offset to set the touch tip location 2mm below the top of the current well: + +``` +pipette.touch_tip(v_offset=-2) + +``` + +This example moves the pipette 75% of well’s total radius and 2 mm below the top of well: + +``` +pipette.touch_tip(plate["B1"], + radius=0.75, + v_offset=-2) + +``` + +The `touch_tip` feature allows the pipette to touch the edges of a well gently instead of crashing into them. It includes the `radius` argument. When `radius=1` the robot moves the centerline of the pipette’s plunger axis to the edge of a well. This means a pipette tip may sometimes touch the well wall too early, causing it to bend inwards. A smaller radius helps avoid premature wall collisions and a lower speed produces gentler motion. Different liquid droplets behave differently, so test out these parameters in a single well before performing a full protocol run. + +Warning + +_Do not_ set the `radius` value greater than `1.0`. When `radius` is \> `1.0`, the robot will forcibly move the pipette tip across a well wall or edge. This type of aggressive movement can damage the pipette tip and the pipette. + +##### Touch Speed + +Touch speed controls how fast the pipette moves in mm/s during a touch tip step. The default movement speed is 60 mm/s, the minimum is 1 mm/s, and the maximum is 80 mm/s. Calling `touch_tip` without any arguments moves a tip at the default speed in the current well: + +``` +pipette.touch_tip() + +``` + +This example specifies a well location and sets the speed to 20 mm/s: + +``` +pipette.touch_tip(plate["B1"], speed=20) + +``` + +This example uses the current well and sets the speed to 80 mm/s: + +``` +pipette.touch_tip(speed=80) + +``` + +New in version 2\.0\. + +Changed in version 2\.4: Lowered minimum speed to 1 mm/s. + +#### Mix + +The [`mix()`](index.html#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix') method aspirates and dispenses repeatedly in a single location. It’s designed to mix the contents of a well together using a single command rather than using multiple `aspirate()` and `dispense()` calls. This method includes arguments that let you specify the number of times to mix, the volume (in µL) of liquid, and the well that contains the liquid you want to mix. + +This example draws 100 µL from the current well and mixes it three times: + +``` +pipette.mix(repetitions=3, volume=100) + +``` + +This example draws 100 µL from well B1 and mixes it three times: + +``` +pipette.mix(3, 100, plate["B1"]) + +``` + +This example draws an amount equal to the pipette’s maximum rated volume and mixes it three times: + +``` +pipette.mix(repetitions=3) + +``` + +Note + +In API versions 2\.2 and earlier, during a mix, the pipette moves up and out of the target well. In API versions 2\.3 and later, the pipette does not move while mixing. + +New in version 2\.0\. + +#### Air Gap + +The [`InstrumentContext.air_gap()`](index.html#opentrons.protocol_api.InstrumentContext.air_gap 'opentrons.protocol_api.InstrumentContext.air_gap') method tells the pipette to draw in air before or after a liquid. Creating an air gap helps keep liquids from seeping out of a pipette after drawing it from a well. This method includes arguments that give you control over the amount of air to aspirate and the pipette’s height (in mm) above the well. By default, the pipette moves 5 mm above a well before aspirating air. Calling [`air_gap()`](index.html#opentrons.protocol_api.InstrumentContext.air_gap 'opentrons.protocol_api.InstrumentContext.air_gap') with no arguments uses the entire remaining volume in the pipette. + +This example aspirates 200 µL of air 5 mm above the current well: + +``` +pipette.air_gap(volume=200) + +``` + +This example aspirates 200 µL of air 20 mm above the the current well: + +``` +pipette.air_gap(volume=200, height=20) + +``` + +This example aspirates enough air to fill the remaining volume in a pipette: + +``` +pipette.air_gap() + +``` + +New in version 2\.0\. + +### Utility Commands + +With utility commands, you can control various robot functions such as pausing or delaying a protocol, checking the robot’s door, turning robot lights on/off, and more. The following sections show you how to these utility commands and include sample code. The examples used here assume that you’ve loaded the pipettes and labware from the basic [protocol template](index.html#protocol-template). + +#### Delay and Resume + +Call the [`ProtocolContext.delay()`](index.html#opentrons.protocol_api.ProtocolContext.delay 'opentrons.protocol_api.ProtocolContext.delay') method to insert a timed delay into your protocol. This method accepts time increments in seconds, minutes, or combinations of both. Your protocol resumes automatically after the specified time expires. + +This example delays a protocol for 10 seconds: + +``` +protocol.delay(seconds=10) + +``` + +This example delays a protocol for 5 minutes: + +``` +protocol.delay(minutes=5) + +``` + +This example delays a protocol for 5 minutes and 10 seconds: + +``` +protocol.delay(minutes=5, seconds=10) + +``` + +#### Pause Until Resumed + +Call the [`ProtocolContext.pause()`](index.html#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause') method to stop a protocol at a specific step. Unlike a delay, [`pause()`](index.html#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause') does not restart your protocol automatically. To resume, you’ll respond to a prompt on the touchscreen or in the Opentrons App. This method also lets you specify an optional message that provides on\-screen or in\-app instructions on how to proceed. This example inserts a pause and includes a brief message: + +``` +protocol.pause("Remember to get more pipette tips") + +``` + +New in version 2\.0\. + +#### Homing + +Homing commands the robot to move the gantry, a pipette, or a pipette plunger to a defined position. For example, homing the gantry moves it to the back right of the working area. With the available homing methods you can home the gantry, home the mounted pipette and plunger, and home the pipette plunger. These functions take no arguments. + +To home the gantry, call [`ProtocolContext.home()`](index.html#opentrons.protocol_api.ProtocolContext.home 'opentrons.protocol_api.ProtocolContext.home'): + +``` +protocol.home() + +``` + +To home a specific pipette’s Z axis and plunger, call [`InstrumentContext.home()`](index.html#opentrons.protocol_api.InstrumentContext.home 'opentrons.protocol_api.InstrumentContext.home'): + +``` +pipette = protocol.load_instrument("flex_1channel_1000", "right") +pipette.home() + +``` + +To home a specific pipette’s plunger only, you can call [`InstrumentContext.home_plunger()`](index.html#opentrons.protocol_api.InstrumentContext.home_plunger 'opentrons.protocol_api.InstrumentContext.home_plunger'): + +``` +pipette = protocol.load_instrument("flex_1channel_1000", "right") +pipette.home_plunger() + +``` + +New in version 2\.0\. + +#### Comment + +Call the [`ProtocolContext.comment()`](index.html#opentrons.protocol_api.ProtocolContext.comment 'opentrons.protocol_api.ProtocolContext.comment') method if you want to write and display a brief message in the Opentrons App during a protocol run: + +``` +protocol.comment("Hello, world!") + +``` + +New in version 2\.0\. + +#### Control and Monitor Robot Rail Lights + +Call the [`ProtocolContext.set_rail_lights()`](index.html#opentrons.protocol_api.ProtocolContext.set_rail_lights 'opentrons.protocol_api.ProtocolContext.set_rail_lights') method to turn the robot’s rail lights on or off during a protocol. This method accepts Boolean `True` (lights on) or `False` (lights off) arguments. Rail lights are off by default. + +This example turns the rail lights on: + +``` +protocol.set_rail_lights(True) + +``` + +This example turns the rail lights off: + +``` +protocol.set_rail_lights(False) + +``` + +New in version 2\.5\. + +You can also check whether the rail lights are on or off in the protocol by using [`ProtocolContext.rail_lights_on`](index.html#opentrons.protocol_api.ProtocolContext.rail_lights_on 'opentrons.protocol_api.ProtocolContext.rail_lights_on'). This method returns `True` when lights are on and `False` when the lights are off. + +New in version 2\.5\. + +#### OT\-2 Door Safety Switch + +Introduced with [robot software version](index.html#version-table) 3\.19, the safety switch feature prevents the OT\-2, and your protocol, from running if the door is open. To operate properly, the front door and top window of your OT\-2 must be closed. You can toggle the door safety switch on or off from **Robot Settings \> Advanced \> Usage Settings**. + +To check if the robot’s door is closed at a specific point during a protocol run, call [`ProtocolContext.door_closed`](index.html#opentrons.protocol_api.ProtocolContext.door_closed 'opentrons.protocol_api.ProtocolContext.door_closed'). It returns a Boolean `True` (door closed) or `False` (door open) response. + +``` +protocol.door_closed + +``` + +Warning + +[`door_closed`](index.html#opentrons.protocol_api.ProtocolContext.door_closed 'opentrons.protocol_api.ProtocolContext.door_closed') is a status check only. It does not control the robot’s behavior. If you wish to implement a custom method to pause or resume a protocol using `door_closed`, disable the door safety feature first (not recommended). + +New in version 2\.5\. + +Building block commands execute some of the most basic actions that your robot can complete. But basic doesn’t mean these commands lack capabilities. They perform important tasks in your protocols. They’re also foundational to the [complex commands](index.html#v2-complex-commands) that help you combine multiple actions into fewer lines of code. + +Pages in this section of the documentation cover: + +- [Manipulating Pipette Tips](index.html#pipette-tips): Get started with commands for picking up pipette tips, dropping tips, returning tips, and working with used tips. +- [Liquid Control](index.html#liquid-control): Learn about aspirating and dispensing liquids, blow out and touch tip procedures, mixing, and creating air gaps. +- [Utility Commands](index.html#new-utility-commands): Control various robot functions such as pausing or delaying a protocol, checking the robot’s door, turning robot lights on/off, and more. + +## Complex Commands + +### Sources and Destinations + +The [`InstrumentContext.transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer'), [`InstrumentContext.distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute'), and [`InstrumentContext.consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') methods form the family of complex liquid handling commands. These methods require `source` and `dest` (destination) arguments to move liquid from one well, or group of wells, to another. In contrast, the [building block commands](index.html#v2-atomic-commands) [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') and [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') only operate in a single location. + +For example, this command performs a simple transfer between two wells on a plate: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=plate["A2"], +) + +``` + +New in version 2\.0\. + +This page covers the restrictions on sources and destinations for complex commands, their different patterns of aspirating and dispensing, and how to optimize them for different use cases. + +#### Source and Destination Arguments + +As noted above, the [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer'), [`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute'), and [`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') methods require `source` and `dest` (destination) arguments to aspirate and dispense liquid. However, each method handles liquid sources and destinations differently. Understanding how complex commands work with source and destination wells is essential to using these methods effectively. + +[`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') is the most versatile complex liquid handling function, because it has the fewest restrictions on what wells it can operate on. You will likely use transfer commands in many of your protocols. + +Certain liquid handling cases focus on moving liquid to or from a single well. [`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') limits its source to a single well, while [`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') limits its destination to a single well. Distribute commands also make changes to liquid\-handling behavior to improve the accuracy of dispensing. + +The following table summarizes the source and destination restrictions for each method. + +| Method | Accepted wells | +| --------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------ | +| `transfer()` | _ **Source:** Any number of wells. _ **Destination:** Any number of wells. \* The larger group of wells must be evenly divisible by the smaller group. | +| `distribute()` | _ **Source:** Exactly one well. _ **Destination:** Any number of wells. | +| `consolidate()` | _ **Source:** Any number of wells. _ **Destination:** Exactly one well. | + +A single well can be passed by itself or as a list with one item: `source=plate["A1"]` and `source=[plate["A1"]]` are equivalent. + +The section on [many\-to\-many transfers](#many-to-many) below covers how `transfer()` works when specifying sources and destinations of different sizes. However, if they don’t meet the even divisibility requirement, the API will raise an error. You can work around such situations by making multiple calls to `transfer()` in sequence or by using a [list of volumes](index.html#complex-list-volumes) to skip certain wells. + +For distributing and consolidating, the API will not raise an error if you use a list of wells as the argument that is limited to exactly one well. Instead, the API will ignore everything except the first well in the list. For example, the following command will only aspirate from well A1: + +``` +pipette.distribute( + volume=100, + source=[plate["A1"], plate["A2"]], # A2 ignored + dest=plate.columns()[1], +) + +``` + +On the other hand, a transfer command with the same arguments would aspirate from both A1 and A2\. The next section examines the exact order of aspiration and dispensing for all three methods. + +#### Transfer Patterns + +Each complex command uses a different pattern of aspiration and dispensing. In addition, when you provide multiple wells as both the source and destination for `transfer()`, it maps the source list onto the destination list in a certain way. + +##### Aspirating and Dispensing + +`transfer()` always alternates between aspirating and dispensing, regardless of how many wells are in the source and destination. Its default behavior is: + +> 1. Pick up a tip. +> 2. Aspirate from the first source well. +> 3. Dispense in the first destination well. +> 4. Repeat the pattern of aspirating and dispensing, as needed. +> 5. Drop the tip in the trash. + +This transfer aspirates six times and dispenses six times. + +`distribute()` always fills the tip with as few aspirations as possible, and then dispenses to the destination wells in order. Its default behavior is: + +> 1. Pick up a tip. +> 2. Aspirate enough to dispense in all the destination wells. This aspirate includes a disposal volume. +> 3. Dispense in the first destination well. +> 4. Continue to dispense in destination wells. +> 5. Drop the tip in the trash. + +See [Tip Refilling](index.html#complex-tip-refilling) below for cases where the total amount to be dispensed is greater than the capacity of the tip. + +This distribute aspirates one time and dispenses three times. + +`consolidate()` aspirates multiple times in a row, and then dispenses as few times as possible in the destination well. Its default behavior is: + +> 1. Pick up a tip. +> 2. Aspirate from the first source well. +> 3. Continue aspirating from source wells. +> 4. Dispense in the destination well. +> 5. Drop the tip in the trash. + +See [Tip Refilling](index.html#complex-tip-refilling) below for cases where the total amount to be aspirated is greater than the capacity of the tip. + +This consolidate aspirates three times and dispenses one time. + +Note + +By default, all three commands begin by picking up a tip and conclude by dropping a tip. In general, don’t call [`pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') just before a complex command, or the API will raise an error. You can override this behavior with the [tip handling complex parameter](index.html#param-tip-handling), by setting `new_tip="never"`. + +##### Many\-to\-Many + +`transfer()` lets you specify both `source` and `dest` arguments that contain multiple wells. This section covers how the method determines which wells to aspirate from and dispense to in these cases. + +When the source and destination both contain the same number of wells, the mapping between wells is straightforward. You can imagine writing out the two lists one above each other, with each unique well in the source list paired to a unique well in the destination list. For example, here is the code for using one row as the source and another row as the destination, and the resulting correspondence between wells: + +``` +pipette.transfer( + volume=50, + source=plate.rows()[0], + dest=plate.rows()[1], +) + +``` + +| Source | A1 | A2 | A3 | A4 | A5 | A6 | A7 | A8 | A9 | A10 | A11 | A12 | +| ----------- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | +| Destination | B1 | B2 | B3 | B4 | B5 | B6 | B7 | B8 | B9 | B10 | B11 | B12 | + +There’s no requirement that the source and destination lists be mutually exclusive. In fact, this command adapted from the [tutorial](index.html#tutorial) deliberately uses slices of the same list, saved to the variable `row`, with the effect that each aspiration happens in the same location as the previous dispense: + +``` +row = plate.rows()[0] +pipette.transfer( + volume=50, + source=row[:11], + dest=row[1:], +) + +``` + +| Source | A1 | A2 | A3 | A4 | A5 | A6 | A7 | A8 | A9 | A10 | A11 | +| ----------- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | +| Destination | A2 | A3 | A4 | A5 | A6 | A7 | A8 | A9 | A10 | A11 | A12 | + +When the source and destination lists contain different numbers of wells, `transfer()` will always aspirate and dispense as many times as there are wells in the _longer_ list. The shorter list will be “stretched” to cover the length of the longer list. Here is an example of transferring from 3 wells to a full row of 12 wells: + +``` +pipette.transfer( + volume=50, + source=[plate["A1"], plate["A2"], plate["A3"]], + dest=plate.rows()[1], +) + +``` + +| Source | A1 | A1 | A1 | A1 | A2 | A2 | A2 | A2 | A3 | A3 | A3 | A3 | +| ----------- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | --- | +| Destination | B1 | B2 | B3 | B4 | B5 | B6 | B7 | B8 | B9 | B10 | B11 | B12 | + +This is why the longer list must be evenly divisible by the shorter list. Changing the destination in this example to a column instead of a row will cause the API to raise an error, because 8 is not evenly divisible by 3: + +``` +pipette.transfer( + volume=50, + source=[plate["A1"], plate["A2"], plate["A3"]], + dest=plate.columns()[3], # labware column 4 +) +# error: source and destination lists must be divisible + +``` + +The API raises this error rather than presuming which wells to aspirate from three times and which only two times. If you want to aspirate three times from A1, three times from A2, and two times from A3, use multiple `transfer()` commands in sequence: + +``` +pipette.transfer(50, plate["A1"], plate.columns()[3][:3]) +pipette.transfer(50, plate["A2"], plate.columns()[3][3:6]) +pipette.transfer(50, plate["A3"], plate.columns()[3][6:]) + +``` + +Finally, be aware of the ordering of source and destination lists when constructing them with [well accessor methods](index.html#well-accessor-methods). For example, at first glance this code may appear to take liquid from each well in the first row of a plate and move it to each of the other wells in the same column: + +``` +pipette.transfer( + volume=20, + source=plate.rows()[0], + dest=plate.rows()[1:], +) + +``` + +However, because the well ordering of [`Labware.rows()`](index.html#opentrons.protocol_api.Labware.rows 'opentrons.protocol_api.Labware.rows') goes _across_ the plate instead of _down_ the plate, liquid from A1 will be dispensed in B1–B7, liquid from A2 will be dispensed in B8–C2, etc. The intended task is probably better accomplished by repeating transfers in a `for` loop: + +``` +for i in range(12): + pipette.transfer( + volume=20, + source=plate.rows()[0][i], + dest=plate.columns()[i][1:], + ) + +``` + +Here the repeat index `i` picks out: + +> - The individual well in the first row, for the source. +> - The corresponding column, which is sliced to form the destination. + +##### Optimizing Patterns + +Choosing the right complex command optimizes gantry movement and helps save time in your protocol. For example, say you want to take liquid from a reservoir and put 50 µL in each well of the first row of a plate. You could use `transfer()`, like this: + +``` +pipette.transfer( + volume=50, + source=reservoir["A1"], + destination=plate.rows()[0], +) + +``` + +This will produce 12 aspirate steps and 12 dispense steps. The steps alternate, with the pipette moving back and forth between the reservoir and plate each time. Using `distribute()` with the same arguments is more optimal in this scenario: + +``` +pipette.distribute( + volume=50, + source=reservoir["A1"], + destination=plate.rows()[0], +) + +``` + +This will produce _just 1_ aspirate step and 12 dispense steps (when using a 1000 µL pipette). The pipette will aspirate enough liquid to fill all the wells, plus a disposal volume. Then it will move to A1 of the plate, dispense, move the short distance to A2, dispense, and so on. This greatly reduces gantry movement and the time to perform this action. And even if you’re using a smaller pipette, `distribute()` will fill the pipette, dispense as many times as possible, and only then return to the reservoir to refill (see [Tip Refilling](index.html#complex-tip-refilling) for more information). + +### Order of Operations + +Complex commands perform a series of [building block commands](index.html#v2-atomic-commands) in order. In fact, the run preview for your protocol in the Opentrons App lists all of these commands as separate steps. This lets you examine what effect your complex commands will have before running them. + +This page describes what steps you should expect the robot to perform when using different complex commands with different required and [optional](index.html#complex-params) parameters. + +#### Step Sequence + +The order of steps is fixed within complex commands. Aspiration and dispensing are the only required actions. You can enable or disable all of the other actions with [complex liquid handling parameters](index.html#complex-params). A complex command designed to perform every possible action will proceed in this order: + +> 1. Pick up tip +> 2. Mix at source +> 3. Aspirate from source +> 4. Touch tip at source +> 5. Air gap +> 6. Dispense into destination +> 7. Mix at destination +> 8. Touch tip at destination +> 9. Blow out +> 10. Drop tip + +The command may repeat some or all of these steps in order to move liquid as requested. [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') repeats as many times as there are wells in the longer of its `source` or `dest` argument. [`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') and [`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') try to repeat as few times as possible. See [Tip Refilling](#complex-tip-refilling) below for how they behave when they do need to repeat. + +#### Example Orders + +The smallest possible number of steps in a complex command is just two: aspirating and dispensing. This is possible by omitting the tip pickup and drop steps: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=plate["B1"], + new_tip="never", +) + +``` + +Here’s another example, a distribute command that adds touch tip steps (and does not turn off tip handling). The code for this command is: + +``` +pipette.distribute( + volume=100, + source=[plate["A1"]], + dest=[plate["B1"], plate["B2"]], + touch_tip=True, +) + +``` + +Compared to the list of all possible actions, this code will only perform the following: + +> 1. Pick up tip +> 2. Aspirate from source +> 3. Touch tip at source +> 4. Dispense into destination +> 5. Touch tip at destination +> 6. Blow out +> 7. Drop tip + +Let’s unpack this. Picking up and dropping tips is default behavior for `distribute()`. Specifying `touch_tip=True` adds two steps, as it is performed at both the source and destination. And it’s also default behavior for `distribute()` to aspirate a disposal volume, which is blown out before dropping the tip. The exact order of steps in the run preview should look similar to this: + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 220.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Touching tip +Dispensing 100.0 uL into B1 of well plate on 2 at 92.86 uL/sec +Touching tip +Dispensing 100.0 uL into B2 of well plate on 2 at 92.86 uL/sec +Touching tip +Blowing out at A1 of Opentrons Fixed Trash on 12 +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +Since dispensing and touching the tip are both associated with the destination wells, those steps are performed at each of the two destination wells. + +#### Tip Refilling + +One factor that affects the exact order of steps for a complex command is whether the amount of liquid being moved can fit in the tip at once. If it won’t fit, you don’t have to adjust your command. The API will handle it for you by including additional steps to refill the tip when needed. + +For example, say you need to move 100 µL of liquid from one well to another, but you only have a 50 µL pipette attached to your robot. To accomplish this with building block commands, you’d need multiple aspirates and dispenses. `aspirate(volume=100)` would raise an error, since it exceeds the tip’s volume. But you can accomplish this with a single transfer command: + +``` +pipette50.transfer( + volume=100, + source=plate["A1"], + dest=plate["B1"], +) + +``` + +To effect the transfer, the API will aspirate and dispense the maximum volume of the pipette (50 µL) twice: + +``` +Picking up tip from A1 of tip rack on D3 +Aspirating 50.0 uL from A1 of well plate on D2 at 57 uL/sec +Dispensing 50.0 uL into B1 of well plate on D2 at 57 uL/sec +Aspirating 50.0 uL from A1 of well plate on D2 at 57 uL/sec +Dispensing 50.0 uL into B1 of well plate on D2 at 57 uL/sec +Dropping tip into A1 of Opentrons Fixed Trash on A3 + +``` + +You can change `volume` to any value (above the minimum volume of the pipette) and the API will automatically calculate how many times the pipette needs to aspirate and dispense. `volume=50` would require just one repetition. `volume=75` would require two, split into 50 µL and 25 µL. `volume=1000` would repeat 20 times — not very efficient, but perhaps more useful than having to swap to a different pipette! + +Remember that `distribute()` includes a disposal volume by default, and this can affect the number of times the pipette refills its tip. Say you want to distribute 80 µL to each of the 12 wells in row A of a plate. That’s 960 µL total — less than the capacity of the pipette — but the 100 µL disposal volume will cause the pipette to refill. + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 980.0 uL from A1 of well plate on 2 at 274.7 uL/sec +Dispensing 80.0 uL into B1 of well plate on 2 at 274.7 uL/sec +Dispensing 80.0 uL into B2 of well plate on 2 at 274.7 uL/sec +... +Dispensing 80.0 uL into B11 of well plate on 2 at 274.7 uL/sec +Blowing out at A1 of Opentrons Fixed Trash on 12 +Aspirating 180.0 uL from A1 of well plate on 2 at 274.7 uL/sec +Dispensing 80.0 uL into B12 of well plate on 2 at 274.7 uL/sec +Blowing out at A1 of Opentrons Fixed Trash on 12 +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +This command will blow out 200 total µL of liquid in the trash. If you need to conserve liquid, use [complex liquid handling parameters](index.html#complex-params) to reduce or eliminate the [disposal volume](index.html#param-disposal-volume), or to [blow out](index.html#param-blow-out) in a location other than the trash. + +#### List of Volumes + +Complex commands can aspirate or dispense different amounts for different wells, rather than the same amount across all wells. To do this, set the `volume` parameter to a list of volumes instead of a single number. The list must be the same length as the longer of `source` or `dest`, or the API will raise an error. For example, this command transfers a different amount of liquid into each of wells B1, B2, and B3: + +``` +pipette.transfer( + volume=[20, 40, 60], + source=plate["A1"], + dest=[plate["B1"], plate["B2"], plate["B3"]], +) + +``` + +Setting any item in the list to `0` will skip aspirating and dispensing for the corresponding well. This example takes the command from above and skips B2: + +``` +pipette.transfer( + volume=[20, 0, 60], + source=plate["A1"], + dest=[plate["B1"], plate["B2"], plate["B3"]], +) + +``` + +The pipette dispenses in B1 and B3, and does not move to B2 at all. + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 20.0 uL from A1 of well plate on 2 at 274.7 uL/sec +Dispensing 20.0 uL into B1 of well plate on 2 at 274.7 uL/sec +Aspirating 60.0 uL from A1 of well plate on 2 at 274.7 uL/sec +Dispensing 60.0 uL into B3 of well plate on 2 at 274.7 uL/sec +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +This is such a simple example that you might prefer to use two `transfer()` commands instead. Lists of volumes become more useful when they are longer than a couple elements. For example, you can specify `volume` as a list with 96 items and `dest=plate.wells()` to individually control amounts to dispense (and wells to skip) across an entire plate. + +Note + +When the optional `new_tip` parameter is set to `"always"`, the pipette will pick up and drop a tip even for skipped wells. If you don’t want to waste tips, pre\-process your list of sources or destinations and use the result as the argument of your complex command. + +New in version 2\.0: Skip wells for `transfer()` and `distribute()`. + +New in version 2\.8: Skip wells for `consolidate()`. + +### Complex Liquid Handling Parameters + +Complex commands accept a number of optional parameters that give you greater control over the exact steps they perform. + +This page describes the accepted values and behavior of each parameter. The parameters are organized in the order that they first add a step. Some parameters, such as `touch_tip`, add multiple steps. See [Order of Operations](index.html#complex-command-order) for more details on the sequence of steps performed by complex commands. + +The API reference entry for [`InstrumentContext.transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') also lists the parameters and has more information on their implementation as keyword arguments. + +#### Tip Handling + +The `new_tip` parameter controls if and when complex commands pick up new tips from the pipette’s tip racks. It has three possible values: + +| Value | Behavior | +| ---------- | ------------------------------------------------------------------------------------------------------------------------------ | +| `"once"` | _ Pick up a tip at the start of the command. _ Use the tip for all liquid handling. \* Drop the tip at the end of the command. | +| `"always"` | Pick up and drop a tip for each set of aspirate and dispense steps. | +| `"never"` | Do not pick up or drop tips at all. | + +`"once"` is the default behavior for all complex commands. + +New in version 2\.0\. + +##### Tip Handling Requirements + +`"once"` and `"always"` require that the pipette has an [associated tip rack](index.html#pipette-tip-racks), or the API will raise an error (because it doesn’t know where to pick up a tip from). If the pipette already has a tip attached, the API will also raise an error when it tries to pick up a tip. + +``` +pipette.pick_up_tip() +pipette.transfer( + volume=100, + source=plate["A1"], + dest=[plate["B1"], plate["B2"], plate["B3"]], + new_tip="never", # "once", "always", or None will error +) + +``` + +Conversely, `"never"` requires that the pipette has picked up a tip, or the API will raise an error (because it will attempt to aspirate without a tip attached). + +##### Avoiding Cross\-Contamination + +One reason to set `new_tip="always"` is to avoid cross\-contamination between wells. However, you should always do a dry run of your protocol to test that the pipette is picking up and dropping tips in the way that your application requires. + +[`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') will pick up a new tip before _every_ aspirate when `new_tip="always"`. This includes when [tip refilling](index.html#complex-tip-refilling) requires multiple aspirations from a single source well. + +[`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') and [`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') only pick up one tip, even when `new_tip="always"`. For example, this distribute command returns to the source well a second time, because the amount to be distributed (400 µL total plus disposal volume) exceeds the pipette capacity (300 μL): + +``` +pipette.distribute( + volume=200, + source=plate["A1"], + dest=[plate["B1"], plate["B2"]], + new_tip="always", +) + +``` + +But it _does not_ pick up a new tip after dispensing into B1: + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 220.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Dispensing 200.0 uL into B1 of well plate on 2 at 92.86 uL/sec +Blowing out at A1 of Opentrons Fixed Trash on 12 +Aspirating 220.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Dispensing 200.0 uL into B2 of well plate on 2 at 92.86 uL/sec +Blowing out at A1 of Opentrons Fixed Trash on 12 +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +If this poses a contamination risk, you can work around it in a few ways: + +> - Use `transfer()` with `new_tip="always"` instead. +> - Set [`well_bottom_clearance`](index.html#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance') high enough that the tip doesn’t contact liquid in the destination well. +> - Use [building block commands](index.html#v2-atomic-commands) instead of complex commands. + +#### Mix Before + +The `mix_before` parameter controls mixing in source wells before each aspiration. Its value must be a [`tuple`](https://docs.python.org/3/library/stdtypes.html#tuple '(in Python v3.12)') with two numeric values. The first value is the number of repetitions, and the second value is the amount of liquid to mix in µL. + +For example, this transfer command will mix 50 µL of liquid 3 times before each of its aspirations: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=[plate["B1"], plate["B2"]], + mix_before=(3, 50), +) + +``` + +New in version 2\.0\. + +Mixing occurs before every aspiration, including when [tip refilling](index.html#complex-tip-refilling) is required. + +Note + +[`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') ignores any value of `mix_before`. Mixing on the second and subsequent aspirations of a consolidate command would defeat its purpose: to aspirate multiple times in a row, from different wells, _before_ dispensing. + +#### Disposal Volume + +The `disposal_volume` parameter controls how much extra liquid is aspirated as part of a [`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') command. Including a disposal volume can improve the accuracy of each dispense. The pipette blows out the disposal volume of liquid after dispensing. To skip aspirating and blowing out extra liquid, set `disposal_volume=0`. + +By default, `disposal_volume` is the [minimum volume](index.html#new-pipette-models) of the pipette, but you can set it to any amount: + +``` +pipette.distribute( + volume=100, + source=plate["A1"], + dest=[plate["B1"], plate["B2"]], + disposal_volume=10, # reduce from default 20 µL to 10 µL +) + +``` + +New in version 2\.0\. + +If the amount to aspirate plus the disposal volume exceeds the tip’s capacity, `distribute()` will use a [tip refilling strategy](index.html#complex-tip-refilling). In such cases, the pipette will aspirate and blow out the disposal volume _for each aspiration_. For example, this command will require tip refilling with a 1000 µL pipette: + +``` +pipette.distribute( + volume=120, + source=reservoir["A1"], + dest=[plate.columns()[0]], + disposal_volume=50, +) + +``` + +The amount to dispense in the destination is 960 µL (120 µL for each of 8 wells in the column). Adding the 50 µL disposal volume exceeds the 1000 µL capacity of the tip. The command will be split across two aspirations, each with the full disposal volume of 50 µL. The pipette will dispose _a total of 100 µL_ during the command. + +Note + +[`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') will not aspirate additional liquid if you set `disposal_volume`. However, it will perform a very small blow out after each dispense. + +[`consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') ignores `disposal_volume` completely. + +#### Touch Tip + +The `touch_tip` parameter accepts a Boolean value. When `True`, a touch tip step occurs after every aspirate and dispense. + +For example, this transfer command aspirates, touches the tip at the source, dispenses, and touches the tip at the destination: + +``` +pipette.transfer( + volume=100, + dest=plate["A1"], + source=plate["B1"], + touch_tip=True, +) + +``` + +New in version 2\.0\. + +Touch tip occurs after every aspiration, including when [tip refilling](index.html#complex-tip-refilling) is required. + +This parameter always uses default motion behavior for touch tip. Use the [touch tip building block command](index.html#touch-tip) if you need to: + +> - Only touch the tip after aspirating or dispensing, but not both. +> - Control the speed, radius, or height of the touch tip motion. + +#### Air Gap + +The `air_gap` parameter controls how much air to aspirate and hold in the bottom of the tip when it contains liquid. The parameter’s value is the amount of air to aspirate in µL. + +Air\-gapping behavior is different for each complex command. The different behaviors all serve the same purpose, which is to never leave the pipette holding liquid at the very bottom of the tip. This helps keep liquids from seeping out of the pipette. + +| Method | Air\-gapping behavior | +| --------------- | ----------------------------------------------------------------------------------------------------------------------- | +| `transfer()` | _ Air gap after each aspiration. _ Pipette is empty after dispensing. | +| `distribute()` | _ Air gap after each aspiration. _ Air gap after dispensing if the pipette isn’t empty. | +| `consolidate()` | _ Air gap after each aspiration. This may create multiple air gaps within the tip. _ Pipette is empty after dispensing. | + +For example, this transfer command will create a 20 µL air gap after each of its aspirations. When dispensing, it will clear the air gap and dispense the full 100 µL of liquid: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=plate["B1"], + air_gap=20, +) + +``` + +New in version 2\.0\. + +When consolidating, air gaps still occur after every aspiration. In this example, the tip will use 210 µL of its capacity (50 µL of liquid followed by 20 µL of air, repeated three times): + +``` +pipette.consolidate( + volume=50, + source=[plate["A1"], plate["A2"], plate["A3"]], + dest=plate["B1"], + air_gap=20, +) + +``` + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 50.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Air gap + Aspirating 20.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Aspirating 50.0 uL from A2 of well plate on 2 at 92.86 uL/sec +Air gap + Aspirating 20.0 uL from A2 of well plate on 2 at 92.86 uL/sec +Aspirating 50.0 uL from A3 of well plate on 2 at 92.86 uL/sec +Air gap + Aspirating 20.0 uL from A3 of well plate on 2 at 92.86 uL/sec +Dispensing 210.0 uL into B1 of well plate on 2 at 92.86 uL/sec +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +If adding an air gap would exceed the pipette’s maximum volume, the complex command will use a [tip refilling strategy](index.html#complex-tip-refilling). For example, this command uses a 300 µL pipette to transfer 300 µL of liquid plus an air gap: + +``` +pipette.transfer( + volume=300, + source=plate["A1"], + dest=plate["B1"], + air_gap=20, +) + +``` + +As a result, the transfer is split into two aspirates of 150 µL, each with their own 20 µL air gap: + +``` +Picking up tip from A1 of tip rack on 3 +Aspirating 150.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Air gap + Aspirating 20.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Dispensing 170.0 uL into B1 of well plate on 2 at 92.86 uL/sec +Aspirating 150.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Air gap + Aspirating 20.0 uL from A1 of well plate on 2 at 92.86 uL/sec +Dispensing 170.0 uL into B1 of well plate on 2 at 92.86 uL/sec +Dropping tip into A1 of Opentrons Fixed Trash on 12 + +``` + +#### Mix After + +The `mix_after` parameter controls mixing in source wells after each dispense. Its value must be a [`tuple`](https://docs.python.org/3/library/stdtypes.html#tuple '(in Python v3.12)') with two numeric values. The first value is the number of repetitions, and the second value is the amount of liquid to mix in µL. + +For example, this transfer command will mix 50 µL of liquid 3 times after each of its dispenses: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=[plate["B1"], plate["B2"]], + mix_after=(3, 50), +) + +``` + +New in version 2\.0\. + +Note + +[`distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') ignores any value of `mix_after`. Mixing after dispensing would combine (and potentially contaminate) the remaining source liquid with liquid present at the destination. + +#### Blow Out + +There are two parameters that control whether and where the pipette blows out liquid. The `blow_out` parameter accepts a Boolean value. When `True`, the pipette blows out remaining liquid when the tip is empty or only contains the disposal volume. The `blowout_location` parameter controls in which of three locations these blowout actions occur. The default blowout location is the trash. Blowout behavior is different for each complex command. + +| Method | Blowout behavior and location | +| --------------- | --------------------------------------------------------------------------------------------------- | +| `transfer()` | _ Blow out after each dispense. _ Valid locations: `"trash"`, `"source well"`, `"destination well"` | +| `distribute()` | _ Blow out after the final dispense. _ Valid locations: `"trash"`, `"source well"` | +| `consolidate()` | _ Blow out after the only dispense. _ Valid locations: `"trash"`, `"destination well"` | + +For example, this transfer command will blow out liquid in the trash twice, once after each dispense into a destination well: + +``` +pipette.transfer( + volume=100, + source=[plate["A1"], plate["A2"]], + dest=[plate["B1"], plate["B2"]], + blow_out=True, +) + +``` + +New in version 2\.0\. + +Set `blowout_location` when you don’t want to waste any liquid by blowing it out into the trash. For example, you may want to make sure that every last bit of a sample is moved into a destination well. Or you may want to return every last bit of an expensive reagent to the source for use in later pipetting. + +If you need to blow out in a different well, or at a specific location within a well, use the [blow out building block command](index.html#blow-out) instead. + +When setting a blowout location, you _must_ also set `blow_out=True`, or the location will be ignored: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=plate["B1"], + blow_out=True, # required to set location + blowout_location="destination well", +) + +``` + +New in version 2\.8\. + +With `transfer()`, the pipette will not blow out at all if you only set `blowout_location`. + +`blow_out=True` is also required for distribute commands that blow out by virtue of having a disposal volume: + +``` +pipette.distribute( + volume=100, + source=plate["A1"], + dest=[plate["B1"], plate["B2"]], + disposal_volume=50, # causes blow out + blow_out=True, # still required to set location! + blowout_location="source well", +) + +``` + +With `distribute()`, the pipette will still blow out if you only set `blowout_location`, but in the default location of the trash. + +Note + +If the tip already contains liquid before the complex command, the default blowout location will shift away from the trash. `transfer()` and `distribute()` shift to the source well, and `consolidate()` shifts to the destination well. For example, this transfer command will blow out in well B1 because it’s the source: + +``` +pipette.pick_up_tip() +pipette.aspirate(100, plate["A1"]) +pipette.transfer( + volume=100, + source=plate["B1"], + dest=plate["C1"], + new_tip="never", + blow_out=True, + # no blowout_location +) +pipette.drop_tip() + +``` + +This only occurs when you aspirate and then perform a complex command with `new_tip="never"` and `blow_out=True`. + +#### Trash Tips + +The `trash` parameter controls what the pipette does with tips at the end of complex commands. When `True`, the pipette drops tips into the trash. When `False`, the pipette returns tips to their original locations in their tip rack. + +The default is `True`, so you only have to set `trash` when you want the tip\-returning behavior: + +``` +pipette.transfer( + volume=100, + source=plate["A1"], + dest=plate["B1"], + trash=False, +) + +``` + +New in version 2\.0\. + +Complex liquid handling commands combine multiple [building block commands](index.html#v2-atomic-commands) into a single method call. These commands make it easier to handle larger groups of wells and repeat actions without having to write your own control flow code. They integrate tip\-handling behavior and can pick up, use, and drop multiple tips depending on how you want to handle your liquids. They can optionally perform other actions, like adding air gaps, knocking droplets off the tip, mixing, and blowing out excess liquid from the tip. + +There are three complex liquid handling commands, each optimized for a different liquid handling scenario: + +> - [`InstrumentContext.transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') +> - [`InstrumentContext.distribute()`](index.html#opentrons.protocol_api.InstrumentContext.distribute 'opentrons.protocol_api.InstrumentContext.distribute') +> - [`InstrumentContext.consolidate()`](index.html#opentrons.protocol_api.InstrumentContext.consolidate 'opentrons.protocol_api.InstrumentContext.consolidate') + +Pages in this section of the documentation cover: + +> - [Sources and Destinations](index.html#complex-source-dest): Which wells complex commands aspirate from and dispense to. +> - [Order of Operations](index.html#complex-command-order): The order of basic commands that are part of a complex commmand. +> - [Complex Liquid Handling Parameters](index.html#complex-params): Additional keyword arguments that affect complex command behavior. + +Code samples throughout these pages assume that you’ve loaded the pipettes and labware from the [basic protocol template](index.html#protocol-template). + +## Labware and Deck Positions + +The API automatically determines how the robot needs to move when working with the instruments and labware in your protocol. But sometimes you need direct control over these activities. The API lets you do just that. Specifically, you can control movements relative to labware and deck locations. You can also manage the gantry’s speed and trajectory as it traverses the working area. This document explains how to use API commands to take direct control of the robot and position it exactly where you need it. + +### Position Relative to Labware + +When the robot positions itself relative to a piece of labware, where it moves is determined by the labware definition, the actions you want it to perform, and the labware offsets for a specific deck slot. This section describes how these positional components are calculated and how to change them. + +#### Top, Bottom, and Center + +Every well on every piece of labware has three addressable positions: top, bottom, and center. The position is determined by the labware definition and what the labware is loaded on top of. You can use these positions as\-is or calculate other positions relative to them. + +##### Top + +Let’s look at the [`Well.top()`](index.html#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') method. It returns a position level with the top of the well, centered in both horizontal directions. + +``` +plate["A1"].top() # the top center of the well + +``` + +This is a good position to use for a [blow out operation](index.html#new-blow-out) or an activity where you don’t want the tip to contact the liquid. In addition, you can adjust the height of this position with the optional argument `z`, which is measured in mm. Positive `z` numbers move the position up, negative `z` numbers move it down. + +``` +plate["A1"].top(z=1) # 1 mm above the top center of the well +plate["A1"].top(z=-1) # 1 mm below the top center of the well + +``` + +New in version 2\.0\. + +##### Bottom + +Let’s look at the [`Well.bottom()`](index.html#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom') method. It returns a position level with the bottom of the well, centered in both horizontal directions. + +``` +plate["A1"].bottom() # the bottom center of the well + +``` + +This is a good position for [aspirating liquid](index.html#new-aspirate) or an activity where you want the tip to contact the liquid. Similar to the `Well.top()` method, you can adjust the height of this position with the optional argument `z`, which is measured in mm. Positive `z` numbers move the position up, negative `z` numbers move it down. + +``` +plate["A1"].bottom(z=1) # 1 mm above the bottom center of the well +plate["A1"].bottom(z=-1) # 1 mm below the bottom center of the well + # this may be dangerous! + +``` + +Warning + +Negative `z` arguments to `Well.bottom()` will cause the pipette tip to collide with the bottom of the well. Collisions may bend the tip (affecting liquid handling) and the pipette may be higher than expected on the z\-axis until it picks up another tip. + +Flex can detect collisions, and even gentle contact may trigger an overpressure error and cause the protocol to fail. Avoid `z` values less than 1, if possible. + +The OT\-2 has no sensors to detect contact with a well bottom. The protocol will continue even after a collision. + +New in version 2\.0\. + +##### Center + +Let’s look at the [`Well.center()`](index.html#opentrons.protocol_api.Well.center 'opentrons.protocol_api.Well.center') method. It returns a position centered in the well both vertically and horizontally. This can be a good place to start for precise control of positions within the well for unusual or custom labware. + +``` +plate["A1"].center() # the vertical and horizontal center of the well + +``` + +New in version 2\.0\. + +#### Default Positions + +By default, your robot will aspirate and dispense 1 mm above the bottom of wells. This default clearance may not be suitable for some labware geometries, liquids, or protocols. You can change this value by using the [`Well.bottom()`](index.html#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom') method with the `z` argument, though it can be cumbersome to do so repeatedly. + +If you need to change the aspiration or dispensing height for multiple operations, specify the distance in mm from the well bottom with the [`InstrumentContext.well_bottom_clearance`](index.html#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance') object. It has two attributes: `well_bottom_clearance.aspirate` and `well_bottom_clearance.dispense`. These change the aspiration height and dispense height, respectively. + +Modifying these attributes will affect all subsequent aspirate and dispense actions performed by the attached pipette, even those executed as part of a [`transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') operation. This snippet from a sample protocol demonstrates how to work with and change the default clearance: + +``` +# aspirate 1 mm above the bottom of the well (default) +pipette.aspirate(50, plate["A1"]) +# dispense 1 mm above the bottom of the well (default) +pipette.dispense(50, plate["A1"]) + +# change clearance for aspiration to 2 mm +pipette.well_bottom_clearance.aspirate = 2 +# aspirate 2 mm above the bottom of the well +pipette.aspirate(50, plate["A1"]) +# still dispensing 1 mm above the bottom +pipette.dispense(50, plate["A1"]) + +pipette.aspirate(50, plate["A1"]) +# change clearance for dispensing to 10 mm +pipette.well_bottom_clearance.dispense = 10 +# dispense high above the well +pipette.dispense(50, plate["A1"]) + +``` + +New in version 2\.0\. + +### Using Labware Position Check + +All positions relative to labware are adjusted automatically based on labware offset data. Calculate labware offsets by running Labware Position Check during protocol setup, either in the Opentrons App or on the Flex touchscreen. Version 6\.0\.0 and later of the robot software can apply previously calculated offsets on the same robot for the same labware type and deck slot, even across different protocols. + +You should only adjust labware offsets in your Python code if you plan to run your protocol in Jupyter Notebook or from the command line. See [Setting Labware Offsets](index.html#using-lpc) in the Advanced Control article for information. + +### Position Relative to Trash Containers + +Movement to [`TrashBin`](index.html#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin') or [`WasteChute`](index.html#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.WasteChute') objects is based on the horizontal _center_ of the pipette. This is different than movement to labware, which is based on the primary channel (the back channel on 8\-channel pipettes, and the back\-left channel on 96\-channel pipettes in default configuration). Using the center of the pipette ensures that all attached tips are over the trash container for blowing out, dropping tips, or other disposal operations. + +Note + +In API version 2\.15 and earlier, trash containers are [`Labware`](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') objects that have a single well. See [`fixed_trash`](index.html#opentrons.protocol_api.ProtocolContext.fixed_trash 'opentrons.protocol_api.ProtocolContext.fixed_trash') and [Position Relative to Labware](#position-relative-labware) above. + +You can adjust the position of the pipette center with the [`TrashBin.top()`](index.html#opentrons.protocol_api.TrashBin.top 'opentrons.protocol_api.TrashBin.top') and [`WasteChute.top()`](index.html#opentrons.protocol_api.WasteChute.top 'opentrons.protocol_api.WasteChute.top') methods. These methods allow adjustments along the x\-, y\-, and z\-axes. In contrast, `Well.top()`, [covered above](#well-top), only allows z\-axis adjustment. With no adjustments, the “top” position is centered on the x\- and y\-axes and is just below the opening of the trash container. + +``` +trash = protocol.load_trash_bin("A3") + +trash # pipette center just below trash top center +trash.top() # same position +trash.top(z=10) # 10 mm higher +trash.top(y=10) # 10 mm towards back, default height + +``` + +New in version 2\.18\. + +Another difference between the trash container `top()` methods and `Well.top()` is that they return an object of the same type, not a [`Location`](index.html#opentrons.types.Location 'opentrons.types.Location'). This helps prevent performing undesired actions in trash containers. For example, you can [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') at a location or from a well, but not from a trash container. On the other hand, you can [`blow_out()`](index.html#opentrons.protocol_api.InstrumentContext.blow_out 'opentrons.protocol_api.InstrumentContext.blow_out') at a location, well, trash bin, or waste chute. + +### Position Relative to the Deck + +The robot’s base coordinate system is known as _deck coordinates_. Many API functions use this coordinate system, and you can also reference it directly. It is a right\-handed coordinate system always specified in mm, with the origin `(0, 0, 0)` at the front left of the robot. The positive `x` direction is to the right, the positive `y` direction is to the back, and the positive `z` direction is up. + +You can identify a point in this coordinate system with a [`types.Location`](index.html#opentrons.types.Location 'opentrons.types.Location') object, either as a standard Python [`tuple`](https://docs.python.org/3/library/stdtypes.html#tuple '(in Python v3.12)') of three floats, or as an instance of the [`namedtuple`](https://docs.python.org/3/library/collections.html#collections.namedtuple '(in Python v3.12)') [`types.Point`](index.html#opentrons.types.Point 'opentrons.types.Point'). + +Note + +There are technically multiple vertical axes. For example, `z` is the axis of the left pipette mount and `a` is the axis of the right pipette mount. There are also pipette plunger axes: `b` (left) and `c` (right). You usually don’t have to refer to these axes directly, since most motion commands are issued to a particular pipette and the robot automatically selects the correct axis to move. Similarly, [`types.Location`](index.html#opentrons.types.Location 'opentrons.types.Location') only deals with `x`, `y`, and `z` values. + +### Independent Movement + +For convenience, many methods have location arguments and incorporate movement automatically. This section will focus on moving the pipette independently, without performing other actions like `aspirate()` or `dispense()`. + +#### Move To + +The [`InstrumentContext.move_to()`](index.html#opentrons.protocol_api.InstrumentContext.move_to 'opentrons.protocol_api.InstrumentContext.move_to') method moves a pipette to any reachable location on the deck. If the pipette has picked up a tip, it will move the end of the tip to that position; if it hasn’t, it will move the pipette nozzle to that position. + +The [`move_to()`](index.html#opentrons.protocol_api.InstrumentContext.move_to 'opentrons.protocol_api.InstrumentContext.move_to') method requires the [`Location`](index.html#opentrons.types.Location 'opentrons.types.Location') argument. The location can be automatically generated by methods like `Well.top()` and `Well.bottom()` or one you’ve created yourself, but you can’t move a pipette to a well directly: + +``` +pipette.move_to(plate["A1"]) # error; can't move to a well itself +pipette.move_to(plate["A1"].bottom()) # move to the bottom of well A1 +pipette.move_to(plate["A1"].top()) # move to the top of well A1 +pipette.move_to(plate["A1"].bottom(z=2)) # move to 2 mm above the bottom of well A1 +pipette.move_to(plate["A1"].top(z=-2)) # move to 2 mm below the top of well A1 + +``` + +When using `move_to()`, by default the pipette will move in an arc: first upwards, then laterally to a position above the target location, and finally downwards to the target location. If you have a reason for doing so, you can force the pipette to move in a straight line to the target location: + +``` +pipette.move_to(plate["A1"].top(), force_direct=True) + +``` + +Warning + +Moving without an arc runs the risk of the pipette colliding with objects on the deck. Be very careful when using this option, especially when moving longer distances. + +Small, direct movements can be useful for working inside of a well, without having the tip exit and re\-enter the well. This code sample demonstrates how to move the pipette to a well, make direct movements inside that well, and then move on to a different well: + +``` +pipette.move_to(plate["A1"].top()) +pipette.move_to(plate["A1"].bottom(1), force_direct=True) +pipette.move_to(plate["A1"].top(-2), force_direct=True) +pipette.move_to(plate["A2"].top()) + +``` + +New in version 2\.0\. + +#### Points and Locations + +When instructing the robot to move, it’s important to consider the difference between the [`Point`](index.html#opentrons.types.Point 'opentrons.types.Point') and [`Location`](index.html#opentrons.types.Location 'opentrons.types.Location') types. + +- Points are ordered tuples or named tuples: `Point(10, 20, 30)`, `Point(x=10, y=20, z=30)`, and `Point(z=30, y=20, x=10)` are all equivalent. +- Locations are a higher\-order tuple that combines a point with a reference object: a well, a piece of labware, or `None` (the deck). + +This distinction is important for the [`Location.move()`](index.html#opentrons.types.Location.move 'opentrons.types.Location.move') method, which operates on a location, takes a point as an argument, and outputs an updated location. To use this method, include `from opentrons import types` at the start of your protocol. The `move()` method does not mutate the location it is called on, so to perform an action at the updated location, use it as an argument of another method or save it to a variable. For example: + +``` +# get the location at the center of well A1 +center_location = plate["A1"].center() + +# get a location 1 mm right, 1 mm back, and 1 mm up from the center of well A1 +adjusted_location = center_location.move(types.Point(x=1, y=1, z=1)) + +# aspirate 1 mm right, 1 mm back, and 1 mm up from the center of well A1 +pipette.aspirate(50, adjusted_location) + +# dispense at the same location +pipette.dispense(50, center_location.move(types.Point(x=1, y=1, z=1))) + +``` + +Note + +The additional `z` arguments of the `top()` and `bottom()` methods (see [Position Relative to Labware](#position-relative-labware) above) are shorthand for adjusting the top and bottom locations with `move()`. You still need to use `move()` to adjust these positions along the x\- or y\-axis: + +``` +# the following are equivalent +pipette.move_to(plate["A1"].bottom(z=2)) +pipette.move_to(plate["A1"].bottom().move(types.Point(z=2))) + +# adjust along the y-axis +pipette.move_to(plate["A1"].bottom().move(types.Point(y=2))) + +``` + +New in version 2\.0\. + +### Movement Speeds + +In addition to instructing the robot where to move a pipette, you can also control the speed at which it moves. Speed controls can be applied either to all pipette motions or to movement along a particular axis. + +Note + +Like all mechanical systems, Opentrons robots have resonant frequencies that depend on their construction and current configuration. It’s possible to set a speed that causes your robot to resonate, producing louder sounds than typical operation. This is safe, but if you find it annoying, increase or decrease the speed slightly. + +#### Gantry Speed + +The robot’s gantry usually moves as fast as it can given its construction. The default speed for Flex varies between 300 and 350 mm/s. The OT\-2 default is 400 mm/s. However, some experiments or liquids may require slower movements. In this case, you can reduce the gantry speed for a specific pipette by setting [`InstrumentContext.default_speed`](index.html#opentrons.protocol_api.InstrumentContext.default_speed 'opentrons.protocol_api.InstrumentContext.default_speed') like this: + +``` +pipette.move_to(plate["A1"].top()) # move to the first well at default speed +pipette.default_speed = 100 # reduce pipette speed +pipette.move_to(plate["D6"].top()) # move to the last well at the slower speed + +``` + +Warning + +These default speeds were chosen because they’re the maximum speeds that Opentrons knows will work with the gantry. Your robot may be able to move faster, but you shouldn’t increase this value unless instructed by Opentrons Support. + +New in version 2\.0\. + +#### Axis Speed Limits + +In addition to controlling the overall gantry speed, you can set speed limits for each of the individual axes: `x` (gantry left/right motion), `y` (gantry forward/back motion), `z` (left pipette up/down motion), and `a` (right pipette up/down motion). Unlike `default_speed`, which is a pipette property, axis speed limits are stored in a protocol property [`ProtocolContext.max_speeds`](index.html#opentrons.protocol_api.ProtocolContext.max_speeds 'opentrons.protocol_api.ProtocolContext.max_speeds'); therefore the `x` and `y` values affect all movements by both pipettes. This property works like a dictionary, where the keys are axes, assigning a value to a key sets a max speed, and deleting a key or setting it to `None` resets that axis’s limit to the default: + +``` + protocol.max_speeds["x"] = 50 # limit x-axis to 50 mm/s + del protocol.max_speeds["x"] # reset x-axis limit + protocol.max_speeds["a"] = 10 # limit a-axis to 10 mm/s + protocol.max_speeds["a"] = None # reset a-axis limit + +``` + +Note that `max_speeds` can’t set limits for the pipette plunger axes (`b` and `c`); instead, set the flow rates or plunger speeds as described in [Pipette Flow Rates](index.html#new-plunger-flow-rates). + +New in version 2\.0\. + +## Runtime Parameters + +### Choosing Good Parameters + +The first decision you need to make when adding parameters to your protocol is “What should be parameterized?” Your goals in adding parameters should be the following: + +1. **Add flexibility.** Accommodate changes from run to run or from lab to lab. +2. **Work efficiently.** Don’t burden run setup with too many choices or confusing options. +3. **Avoid errors.** Ensure that every combination of parameters produces an analyzable, runnable protocol. + +The trick to choosing good parameters is reasoning through the choices the protocol’s users may make. If any of them lead to nonsensical outcomes or errors, adjust the parameters — or how your protocol [uses parameter values](index.html#using-rtp) — to avoid those situations. + +#### Build on a Task + +Consider what scientific task is at the heart of your protocol, and build parameters that contribute to, rather than diverge from it. + +For example, it makes sense to add a parameter for number of samples to a DNA prep protocol that uses a particular reagent kit. But it wouldn’t make sense to add a parameter for _which reagent kit_ to use for DNA prep. That kind of parameter would affect so many aspects of the protocol that it would make more sense to maintain a separate protocol for each kit. + +Also consider how a small number of parameters can combine to produce many useful outputs. Take the serial dilution task from the [Tutorial](index.html#tutorial) as an example. We could add just three parameters to it: number of dilutions, dilution factor, and number of rows. Now that single protocol can produce a whole plate that gradually dilutes, a 2×4 grid that rapidly dilutes, and _thousands_ of other combinations. + +#### Consider Contradictions + +Here’s a common time\-saving use of parameters: your protocol requires a 1\-channel pipette and an 8\-channel pipette, but it doesn’t matter which mount they’re attached to. Without parameters, you would have to assign the mounts in your protocol. Then if the robot is set up in the reverse configuration, you’d have to either physically swap the pipettes or modify your protocol. + +One way to get this information is to ask which mount the 1\-channel pipette is on, and which mount the 8\-channel pipette is on. But if a technician answers “left” to both questions — even by accident — the API will raise an error, because you can’t load two pipettes on a single mount. It’s no better to flip things around by asking which pipette is on the left mount, and which pipette is on the right mount. Now the technician can say that both mounts have a 1\-channel pipette. This is even more dangerous, because it _might not_ raise any errors in analysis. The protocol could run “successfully” on a robot with two 1\-channel pipettes, but produce completely unintended results. + +The best way to avoid these contradictions is to collapse the two questions into one, with limited choices. Where are the pipettes mounted? Either the 1\-channel is on the left and the 8\-channel on the right, or the 8\-channel is on the left and the 1\-channel is on the right. This approach is best for several reasons: + +- It avoids analysis errors. +- It avoids potentially dangerous execution errors. +- It only requires answering one question instead of two. +- The [phrasing of the question and answer](index.html#rtp-style) makes it clear that the protocol requires exactly one of each pipette type. + +#### Set Boundaries + +Numerical parameters support minimum and maximum values, which you should set to avoid incorrect inputs that are outside of your protocol’s possibile actions. + +Consider our earlier example of parameterizing serial dilution. Each of the three numerical parameters have logical upper and lower bounds, which we need to enforce to get sensible results. + +- _Number of dilutions_ must be between 0 and 11 on a 96\-well plate. And it may make sense to require at least 1 dilution. +- _Dilution factor_ is a ratio, which we can express as a decimal number that must be between 0 and 1\. +- _Number of rows_ must be between 1 and 8 on a 96\-well plate. + +What if you wanted to perform a dilution with 20 repetitions? It’s possible with two 96\-well plates, or with a 384\-well plate. You could set the maximum for the number of dilutions to 24 and allow for these possibilities — either switching the plate type or loading an additional plate based on the provided value. + +But what if the technician wanted to do just 8 repetitions on a 384\-well plate? That would require an additional parameter, an additional choice by the technician, and additional logic in your protocol code. It’s up to you as the protocol author to decide if adding more parameters will make protocol setup overly difficult. Sometimes it’s more efficient to work with two or three simple protocols rather than one that’s long and complex. + +### Defining Parameters + +To use parameters, you need to define them in [a separate function](#add-parameters) within your protocol. Each parameter definition has two main purposes: to specify acceptable values, and to inform the protocol user what the parameter does. + +Depending on the [type of parameter](#rtp-types), you’ll need to specify some or all of the following. + +| Attribute | Details | +| ----------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| `variable_name` | _ A unique name for [referencing the parameter value](index.html#using-rtp) elsewhere in the protocol. _ Must meet the usual requirements for [naming objects in Python](https://docs.python.org/3/reference/lexical_analysis.html#identifiers). | +| `display_name` | _ A label for the parameter shown in the Opentrons App or on the touchscreen. _ Maximum 30 characters. | +| `description` | _ An optional longer explanation of what the parameter does, or how its values will affect the execution of the protocol. _ Maximum 100 characters. | +| `default` | \* The value the parameter will have if the technician makes no changes to it during run setup. | +| `minimum` and `maximum` | _ For numeric parameters only. _ Allows free entry of any value within the range (inclusive). _ Both values are required. _ Can’t be used at the same time as `choices`. | +| `choices` | _ For numeric or string parameters. _ Provides a fixed list of values to choose from. _ Each choice has its own display name and value. _ Can’t be used at the same time as `minimum` and `maximum`. | +| `units` | _ Optional, for numeric parameters with `minimum` and `maximum` only. _ Displays after the number during run setup. _ Does not affect the parameter’s value or protocol execution. _ Maximum 10 characters. | + +#### The `add_parameters()` Function + +All parameter definitions are contained in a Python function, which must be named `add_parameters` and takes a single argument. Define `add_parameters()` before the `run()` function that contains protocol commands. + +The examples on this page assume the following definition, which uses the argument name `parameters`. The type specification of the argument is optional. + +``` +def add_parameters(parameters: protocol_api.Parameters): + +``` + +Within this function definition, call methods on `parameters` to define parameters. The next section demonstrates how each type of parameter has its own method. + +#### Types of Parameters + +The API supports four types of parameters: Boolean ([`bool`](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')), integer ([`int`](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')), floating point number ([`float`](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')), and string ([`str`](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')). It is not possible to mix types within a single parameter. + +##### Boolean Parameters + +Boolean parameters are `True` or `False` only. + +``` +parameters.add_bool( + variable_name="dry_run", + display_name="Dry Run", + description="Skip incubation delays and shorten mix steps.", + default=False +) + +``` + +During run setup, the technician can toggle between the two values. In the Opentrons App, Boolean parameters appear as a toggle switch. On the touchscreen, they appear as _On_ or _Off_, for `True` and `False` respectively. + +New in version 2\.18\. + +##### Integer Parameters + +Integer parameters either accept a range of numbers or a list of numbers. You must specify one or the other; you can’t create an open\-ended prompt that accepts any integer. + +To specify a range, include `minimum` and `maximum`. + +``` +parameters.add_int( + variable_name="volume", + display_name="Aspirate volume", + description="How much to aspirate from each sample.", + default=20, + minimum=10, + maximum=100, + unit="µL" +) + +``` + +During run setup, the technician can enter any integer value from the minimum up to the maximum. Entering a value outside of the range will show an error. At that point, they can correct their custom value or restore the default value. + +To specify a list of numbers, include `choices`. Each choice is a dictionary with entries for display name and value. The display names let you briefly explain the effect each choice will have. + +``` +parameters.add_int( + variable_name="volume", + display_name="Aspirate volume", + description="How much to aspirate from each sample.", + default=20, + choices=[ + {"display_name": "Low (10 µL)", "value": 10}, + {"display_name": "Medium (20 µL)", "value": 20}, + {"display_name": "High (50 µL)", "value": 50}, + ] +) + +``` + +During run setup, the technician can choose from a menu of the provided choices. + +New in version 2\.18\. + +##### Float Parameters + +Float parameters either accept a range of numbers or a list of numbers. You must specify one or the other; you can’t create an open\-ended prompt that accepts any floating point number. + +Specifying a range or list is done exactly the same as in the integer examples above. The only difference is that all values must be floating point numbers. + +``` +parameters.add_float( + variable_name="volume", + display_name="Aspirate volume", + description="How much to aspirate from each sample.", + default=5.0, + choices=[ + {"display_name": "Low (2.5 µL)", "value": 2.5}, + {"display_name": "Medium (5 µL)", "value": 5.0}, + {"display_name": "High (10 µL)", "value": 10.0}, + ] +) + +``` + +New in version 2\.18\. + +##### String Parameters + +String parameters only accept a list of values. You can’t currently prompt for free text entry of a string value. + +To specify a list of strings, include `choices`. Each choice is a dictionary with entries for display name and value. Only the display name will appear during run setup. + +A common use for string display names is to provide an easy\-to\-read version of an API load name. You can also use them to briefly explain the effect each choice will have. + +``` +parameters.add_str( + variable_name="pipette", + display_name="Pipette type", + choices=[ + {"display_name": "1-Channel 50 µL", "value": "flex_1channel_50"}, + {"display_name": "8-Channel 50 µL", "value": "flex_8channel_50"}, + ], + default="flex_1channel_50", +) + +``` + +During run setup, the technician can choose from a menu of the provided choices. + +New in version 2\.18\. + +### Using Parameters + +Once you’ve [defined parameters](index.html#defining-rtp), their values are accessible anywhere within the `run()` function of your protocol. + +#### The `params` Object + +Protocols with parameters have a [`ProtocolContext.params`](index.html#opentrons.protocol_api.ProtocolContext.params 'opentrons.protocol_api.ProtocolContext.params') object, which contains the values of all parameters as set during run setup. Each attribute of `params` corresponds to the `variable_name` of a parameter. + +For example, consider a protocol that defines the following three parameters: + +- `add_bool` with `variable_name="dry_run"` +- `add_int` with `variable_name="sample_count"` +- `add_float` with `variable_name="volume"` + +Then `params` will gain three attributes: `params.dry_run`, `params.sample_count`, and `params.volume`. You can use these attributes anywhere you want to access their values, including directly as arguments of methods. + +``` +if protocol.params.dry_run is False: + pipette.mix(repetitions=10, volume=protocol.params.volume) + +``` + +You can also save parameter values to variables with names of your choosing. + +#### Parameter Types + +Each attribute of `params` has the type corresponding to its parameter definition. Keep in mind the parameter’s type when using its value in different contexts. + +Say you wanted to add a comment to the run log, stating how many samples the protocol will process. Since `sample_count` is an `int`, you’ll need to cast it to a `str` or the API will raise an error. + +``` +protocol.comment( + "Processing " + str(protocol.params.sample_count) + " samples." +) + +``` + +Also be careful with `int` types when performing calculations: dividing an `int` by an `int` with the `/` operator always produces a `float`, even if there is no remainder. The [sample count use case](index.html#use-case-sample-count) converts a sample count to a column count by dividing by 8 — but it uses the `//` integer division operator, so the result can be used for creating ranges, slicing lists, and as `int` argument values without having to cast it in those contexts. + +#### Limitations + +Since `params` is only available within the `run()` function, there are certain aspects of a protocol that parameter values can’t affect. These include, but are not limited to the following: + +| Information | Location | +| -------------------------------- | ----------------------------------------------- | +| `import` statements | At the beginning of the protocol. | +| Robot type (Flex or OT\-2\) | In the `requirements` dictionary. | +| API version | In the `requirements` or `metadata` dictionary. | +| Protocol name | In the `metadata` dictionary. | +| Protocol description | In the `metadata` dictionary. | +| Protocol author | In the `metadata` dictionary. | +| Other runtime parameters | In the `add_parameters()` function. | +| Non\-nested function definitions | Anywhere outside of `run()`. | + +Additionally, keep in mind that updated parameter values are applied by reanalyzing the protocol. This means you can’t depend on updated values for any action that takes place _prior to reanalysis_. + +An example of such an action is applying labware offset data. Say you have a parameter that changes the type of well plate you load in a particular slot: + +``` +# within add_parameters() +parameters.add_str( + variable_name="plate_type", + display_name="Well plate type", + choices=[ + {"display_name": "Corning", "value": "corning_96_wellplate_360ul_flat"}, + {"display_name": "NEST", "value": "nest_96_wellplate_200ul_flat"}, + ], + default="corning_96_wellplate_360ul_flat", +) + +# within run() +plate = protocol.load_labware( + load_name=protocol.params.plate_type, location="D2" +) + +``` + +When performing run setup, you’re prompted to apply offsets before selecting parameter values. This is your only opportunity to apply offsets, so they’re applied for the default parameter values — in this case, the Corning plate. If you then change the “Well plate type” parameter to the NEST plate, the NEST plate will have default offset values (0\.0 on all axes). You can fix this by running Labware Position Check, since it takes place after reanalysis, or by using [`Labware.set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset') in your protocol. + +### Parameter Use Case – Sample Count + +Choosing how many samples to process is important for efficient automation. This use case explores how a single parameter for sample count can have pervasive effects throughout a protocol. The examples are adapted from an actual parameterized protocol for DNA prep. The sample code will use 8\-channel pipettes to process 8, 16, 24, or 32 samples. + +At first glance, it might seem like sample count would primarily affect liquid transfers to and from sample wells. But when using the Python API’s full range of capabilities, it affects: + +- How many tip racks to load. +- The initial volume and placement of reagents. +- Pipetting to and from samples. +- If and when tip racks need to be replaced. + +To keep things as simple as possible, this use case only focuses on setting up and using the value of the sample count parameter, which is just one of several parameters present in the full protocol. + +#### From Samples to Columns + +First of all, we need to set up the sample count parameter so it’s both easy for technicians to understand during protocol setup and easy for us to use in the protocol’s `run()` function. + +We want to limit the number of samples to 8, 16, 24, or 32, so we’ll use an integer parameter with choices: + +``` +def add_parameters(parameters): + + parameters.add_int( + variable_name="sample_count", + display_name="Sample count", + description="Number of input DNA samples.", + default=24, + choices=[ + {"display_name": "8", "value": 8}, + {"display_name": "16", "value": 16}, + {"display_name": "24", "value": 24}, + {"display_name": "32", "value": 32}, + ] + ) + +``` + +All of the possible values are multiples of 8, because the protocol will use an 8\-channel pipette to process an entire column of samples at once. Considering how 8\-channel pipettes access wells, it may be more useful to operate with a _column count_ in code. We can set a `column_count` very early in the `run()` function by accessing the value of `params.sample_count` and dividing it by 8: + +``` +def run(protocol): + + column_count = protocol.params.sample_count // 8 + +``` + +Most examples below will use `column_count`, rather than redoing (and retyping!) this calculation multiple times. + +#### Loading Tip Racks + +Tip racks come first in most protocols. To ensure that the protocol runs to completion, we need to load enough tip racks to avoid running out of tips. + +We could load as many tip racks as are needed for our maximum number of samples, but that would be suboptimal. Run setup is faster when the technician doesn’t have to load extra items onto the deck. So it’s best to examine the protocol’s steps and determine how many racks are needed for each value of `sample_count`. + +In the case of this DNA prep protocol, we can create formulas for the number of 200 µL and 50 µL tip racks needed. The following factors go into these computations: + +- 50 µL tips + - 1 fixed action that picks up once per protocol. + - 7 variable actions that pick up once per sample column. +- 200 µL tips + - 2 fixed actions that pick up once per protocol. + - 11 variable actions that pick up once per sample column. + +Since each tip rack has 12 columns, divide the number of pickup actions by 12 to get the number of racks needed. And we always need to round up — performing 13 pickups requires 2 racks. The [`math.ceil()`](https://docs.python.org/3/library/math.html#math.ceil '(in Python v3.12)') method rounds up to the nearest integer. We’ll add `from math import ceil` at the top of the protocol and then calculate the number of tip racks as follows: + +``` +tip_rack_50_count = ceil((1 + 7 * column_count) / 12) +tip_rack_200_count = ceil((2 + 13 * column_count) / 12) + +``` + +Running the numbers shows that the maximum combined number of tip racks is 7\. Now we have to decide where to load up to 7 racks, working around the modules and other labware on the deck. Assuming we’re running this protocol on a Flex with staging area slots, they’ll all fit! (If you don’t have staging area slots, you can load labware off\-deck instead.) We’ll reserve these slots for the different size racks: + +``` +tip_rack_50_slots = ["B3", "C3", "B4"] +tip_rack_200_slots = ["A2", "B2", "A3", "A4"] + +``` + +Finally, we can combine this information to call [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). Depending on the number of racks needed, we’ll slice that number of elements from the slot list and use a [list comprehension](https://docs.python.org/2/tutorial/datastructures.html#list-comprehensions) to gather up the loaded tip racks. For the 50 µL tips, this would look like: + +``` +tip_racks_50 = [ + protocol.load_labware( + load_name="opentrons_flex_96_tiprack_50ul", + location=slot + ) + for slot in tip_rack_50_slots[:tip_rack_50_count] +] + +``` + +Then we can associate those lists of tip racks directly with each pipette as we load them. All together, the start of our `run()` function looks like this: + +``` +# calculate column count from sample count +column_count = protocol.params.sample_count // 8 + +# calculate number of required tip racks +tip_rack_50_count = ceil((1 + 7 * column_count) / 12) +tip_rack_200_count = ceil((2 + 13 * column_count) / 12) + +# assign tip rack locations (maximal case) +tip_rack_50_slots = ["B3", "C3", "B4"] +tip_rack_200_slots = ["A2", "B2", "A3", "A4"] + +# create lists of loaded tip racks +# limit to number of needed racks for each type +tip_racks_50 = [ + protocol.load_labware( + load_name="opentrons_flex_96_tiprack_50ul", + location=slot + ) + for slot in tip_rack_50_slots[:tip_rack_50_count] +] +tip_racks_200 = [ + protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location=slot + ) + for slot in tip_rack_200_slots[:tip_rack_200_count] +] + +pipette_50 = protocol.load_instrument( + instrument_name="flex_8channel_50", + mount="right", + tip_racks=tip_racks_50 +) +pipette_1000 = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=tip_racks_200 +) + +``` + +This code will load as few as 3 tip racks and as many as 7, and associate them with the correct pipettes — all based on a single choice from a dropdown menu at run setup. + +#### Loading Liquids + +Next come the reagents, samples, and the labware that holds them. + +The required volume of each reagent is dependent on the sample count. While the full protocol defines more than ten liquids, we’ll show three reagents plus the samples here. + +First, let’s load a reservoir and [define](index.html#defining-liquids) the three example liquids. Definitions only specify the name, description, and display color, so our sample count parameter doesn’t come into play yet: + +``` +# labware to hold reagents +reservoir = protocol.load_labware( + load_name="nest_12_reservoir_15ml", location="C2" +) + +# reagent liquid definitions +ampure_liquid = protocol.define_liquid( + name="AMPure", description="AMPure Beads", display_color="#704848" +) +tagstop_liquid = protocol.define_liquid( + name="TAGSTOP", description="Tagmentation Stop", display_color="#FF0000" +) +twb_liquid = protocol.define_liquid( + name="TWB", description="Tagmentation Wash Buffer", display_color="#FFA000" +) + +``` + +Now we’ll bring sample count into consideration as we [load the liquids](index.html#loading-liquids). The application requires the following volumes for each column of samples: + +| Liquid | Volume (µL per column) | +| ------------------------ | ---------------------- | +| AMPure Beads | 180 | +| Tagmentation Stop | 10 | +| Tagmentation Wash Buffer | 900 | + +To calculate the total volume for each liquid, we’ll multiply these numbers by `column_count` and by 1\.1 (to ensure that the pipette can aspirate the required volume without drawing in air at the bottom of the well). This calculation can be done inline as the `volume` value of [`load_liquid()`](index.html#opentrons.protocol_api.Well.load_liquid 'opentrons.protocol_api.Well.load_liquid'): + +``` +reservoir["A1"].load_liquid( + liquid=ampure_liquid, volume=180 * column_count * 1.1 +) +reservoir["A2"].load_liquid( + liquid=tagstop_liquid, volume=10 * column_count * 1.1 +) +reservoir["A4"].load_liquid( + liquid=twb_liquid, volume=900 * column_count * 1.1 +) + +``` + +Now, for example, the volume of AMPure beads to load will vary from 198 µL for a single sample column up to 792 µL for four columns. + +Tip + +Does telling a technician to load 792 µL of a liquid seem overly precise? Remember that you can perform any calculation you like to set the value of `volume`! For example, you could round the AMPure volume up to the nearest 10 µL: + +``` +volume=ceil((180 * column_count * 1.1) / 10) * 10 + +``` + +Finally, it’s good practice to label the wells where the samples reside. The sample plate starts out atop the Heater\-Shaker Module: + +``` +hs_mod = protocol.load_module( + module_name="heaterShakerModuleV1", location="D1" +) +hs_adapter = hs_mod.load_adapter(name="opentrons_96_pcr_adapter") +sample_plate = hs_adapter.load_labware( + name="opentrons_96_wellplate_200ul_pcr_full_skirt", + label="Sample Plate", +) + +``` + +Now we can construct a `for` loop to label each sample well with `load_liquid()`. The simplest way to do this is to combine our original _sample count_ with the fact that the [`Labware.wells()`](index.html#opentrons.protocol_api.Labware.wells 'opentrons.protocol_api.Labware.wells') accessor returns wells top\-to\-bottom, left\-to\-right: + +``` +# define sample liquid +sample_liquid = protocol.define_liquid( + name="Samples", description=None, display_color="#52AAFF" +) + +# load 40 µL in each sample well +for w in range(protocol.params.sample_count): + sample_plate.wells()[w].load_liquid(liquid=sample_liquid, volume=40) + +``` + +#### Processing Samples + +When it comes time to process the samples, we’ll return to working by column, since the protocol uses an 8\-channel pipette. There are many pipetting stages in the full protocol, but this section will examine just the stage for adding the Tagmentation Stop liquid. The same techniques would apply to similar stages. + +For pipetting in the original sample locations, we’ll command the 50 µL pipette to move to some or all of A1–A4 on the sample plate. Similar to when we loaded tip racks earlier, we can use `column_count` to slice a list containing these well names, and then iterate over that list with a `for` loop: + +``` +for w in ["A1", "A2", "A3", "A4"][:column_count]: + pipette_50.pick_up_tip() + pipette_50.aspirate(volume=13, location=reservoir["A2"].bottom()) + pipette_50.dispense(volume=3, location=reservoir["A2"].bottom()) + pipette_50.dispense(volume=10, location=sample_plate[w].bottom()) + pipette_50.move_to(location=sample_plate[w].bottom()) + pipette_50.mix(repetitions=10, volume=20) + pipette_50.blow_out(location=sample_plate[w].top(z=-2)) + pipette_50.drop_tip() + +``` + +Each time through the loop, the pipette will fill from the same well of the reservoir and then dispense (and mix and blow out) in a different column of the sample plate. + +Later steps of the protocol will move intermediate samples to the middle of the plate (columns 5–8\) and final samples to the right side of the plate (columns 9–12\). When moving directly from one set of columns to another, we have to track _both lists_ with the `for` loop. The [`zip()`](https://docs.python.org/3/library/functions.html#zip '(in Python v3.12)') function lets us pair up the lists of well names and step through them in parallel: + +``` +for initial, intermediate in zip( + ["A1", "A2", "A3", "A4"][:column_count], + ["A5", "A6", "A7", "A8"][:column_count], +): + pipette_50.pick_up_tip() + pipette_50.aspirate(volume=13, location=sample_plate[initial]) + pipette_50.dispense(volume=13, location=sample_plate[intermediate]) + pipette_50.drop_tip() + +``` + +This will transfer from column 1 to 5, 2 to 6, and so on — depending on the number of samples chosen during run setup. + +#### Replenishing Tips + +For the higher values of `protocol.params.sample_count`, the protocol will load tip racks in the staging area slots (column 4\). Since pipettes can’t reach these slots, we need to move these tip racks into the working area (columns 1–3\) before issuing a pipetting command that targets them, or the API will raise an error. + +A protocol without parameters will always run out of tips at the same time — just add a [`move_labware()`](index.html#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') command when that happens. But as we saw in the Processing Samples section above, our parameterized protocol will go through tips at a different rate depending on the sample count. + +In our simplified example, we know that when the sample count is 32, the first 200 µL tip rack will be exhausted after three stages of pipetting using the 1000 µL pipette. So, after that step, we could add: + +``` +if protocol.params.sample_count == 32: + protocol.move_labware( + labware=tip_racks_200[0], + new_location=chute, + use_gripper=True, + ) + protocol.move_labware( + labware=tip_racks_200[-1], + new_location="A2", + use_gripper=True, + ) + +``` + +This will replace the first 200 µL tip rack (in slot A2\) with the last 200 µL tip rack (in the staging area). + +However, in the full protocol, sample count is not the only parameter that affects the rate of tip use. It would be unwieldy to calculate in advance all the permutations of when tip replenishment is necessary. Instead, before each stage of the protocol, we could use [`Well.has_tip()`](index.html#opentrons.protocol_api.Well.has_tip 'opentrons.protocol_api.Well.has_tip') to check whether the first tip rack is empty. If the _last well_ of the rack is empty, we can assume that the entire rack is empty and needs to be replaced: + +``` +if tip_racks_200[0].wells()[-1].has_tip is False: + # same move_labware() steps as above + +``` + +For a protocol that uses tips at a faster rate than this one — such that it might exhaust a tip rack in a single `for` loop of pipetting steps — you may have to perform such checks even more frequently. You can even define a function that counts tips or performs `has_tip` checks in combination with picking up a tip, and use that instead of [`pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') every time you pipette. The built\-in capabilities of Python and the methods of the Python Protocol API give you the flexibility to add this kind of smart behavior to your protocols. + +### Parameter Use Case – Dry Run + +When testing out a new protocol, it’s common to perform a dry run to watch your robot go through all the steps without actually handling samples or reagents. This use case explores how to add a single Boolean parameter for whether you’re performing a dry run. + +The code examples will show how this single value can control: + +- Skipping module actions and long delays. +- Reducing mix repetitions to save time. +- Returning tips (that never touched any liquid) to their racks. + +To keep things as simple as possible, this use case only focuses on setting up and using the value of the dry run parameter, which could be just one of many parameters in a complete protocol. + +#### Dry Run Definition + +First, we need to set up the dry run parameter. We want to set up a simple yes/no choice for the technician running the protocol, so we’ll use a Boolean parameter: + +``` +def add_parameters(parameters): + + parameters.add_bool( + variable_name="dry_run", + display_name="Dry Run", + description=( + "Skip delays," + " shorten mix steps," + " and return tips to their racks." + ), + default=False + ) + +``` + +This parameter is set to `False` by default, assuming that most runs will be live runs. In other words, during run setup the technician will have to change the parameter setting to perform a dry run. If they leave it as is, the robot will perform a live run. + +Additionally, since “dry run” can have different meanings in different contexts, it’s important to include a `description` that indicates exactly what the parameter will control — in this case, three things. The following sections will show how to accomplish each of those when the dry run parameter is set to `True`. + +#### Skipping Delays + +Many protocols have built\-in delays, either for a module to work or to let a reaction happen passively. Lengthy delays just get in the way when verifying a protocol with a dry run. So wherever the protocol calls for a delay, we can check the value of `protocol.params.dry_run` and make the protocol behave accordingly. + +To start, let’s consider a simple [`delay()`](index.html#opentrons.protocol_api.ProtocolContext.delay 'opentrons.protocol_api.ProtocolContext.delay') command. We can wrap it in an `if` statement such that the delay will only execute when the run is _not_ a dry run: + +``` +if protocol.params.dry_run is False: + protocol.delay(minutes=5) + +``` + +You can extend this approach to more complex situations, like module interactions. For example, in a protocol that moves a plate to the Thermocycler for an incubation, you’ll want to perform all the movement steps — opening and closing the module lid, and moving the plate to and from the block — but skip the heating and cooling time. The simplest way to do this is, like in the delay example above, to wrap each skippable command: + +``` +protocol.move_labware(labware=plate, new_location=tc_mod, use_gripper=True) +if protocol.params.dry_run is False: + tc_mod.set_block_temperature(4) + tc_mod.set_lid_temperature(100) +tc_mod.close_lid() +pcr_profile = [ + {"temperature": 68, "hold_time_seconds": 180}, + {"temperature": 98, "hold_time_seconds": 180}, +] +if protocol.params.dry_run is False: + tc_mod.execute_profile( + steps=pcr_profile, repetitions=1, block_max_volume=50 + ) +tc_mod.open_lid() + +``` + +#### Shortening Mix Steps + +Similar to delays, mix steps can take a long time because they are inherently repetitive actions. Mixing ten times takes ten times as long as mixing once! To save time, set a mix repetitions variable based on the value of `protocol.params.dry_run` and pass that to [`mix()`](index.html#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix'): + +``` +if protocol.params.dry_run is True: + mix_reps = 1 +else: + mix_reps = 10 +pipette.mix(repetitions=mix_reps, volume=50, location=plate["A1"].bottom()) + +``` + +Note that this checks whether the dry run parameter is `True`. If you prefer to set up all your `if` statements to check whether it’s `False`, you can reverse the logic: + +``` +if protocol.params.dry_run is False: + mix_reps = 10 +else: + mix_reps = 1 + +``` + +#### Returning Tips + +Tips used in a dry run should be reusable — for another dry run, if nothing else. It doesn’t make sense to dispose of them in a trash container, unless you specifically need to test movement to the trash. You can choose whether to use [`drop_tip()`](index.html#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') or [`return_tip()`](index.html#opentrons.protocol_api.InstrumentContext.return_tip 'opentrons.protocol_api.InstrumentContext.return_tip') based on the value of `protocol.params.dry_run`. If the protocol doesn’t have too many tip drop actions, you can use an `if` statement each time: + +``` +if protocol.params.dry_run is True: + pipette.return_tip() +else: + pipette.drop_tip() + +``` + +However, repeating this block every time you handle tips could significantly clutter your code. Instead, you could define it as a function: + +``` +def return_or_drop(pipette): + if protocol.params.dry_run is True: + pipette.return_tip() + else: + pipette.drop_tip() + +``` + +Then call that function throughout your protocol: + +``` +pipette.pick_up_tip() +return_or_drop(pipette) + +``` + +Note + +It’s generally better to define a standalone function, rather than adding a method to the [`InstrumentContext`](index.html#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.InstrumentContext') class. This makes your custom, parameterized commands stand out from API methods in your code. + +Additionally, if your protocol uses enough tips that you have to replenish tip racks, you’ll need separate behavior for dry runs and live runs. In a live run, once you’ve used all the tips, the rack is empty, because the tips are in the trash. In a dry run, once you’ve used all the tips in a rack, the rack is _full_, because you returned the tips. + +The API has methods to handle both of these situations. To continue using the same tip rack without physically replacing it, call [`reset_tipracks()`](index.html#opentrons.protocol_api.InstrumentContext.reset_tipracks 'opentrons.protocol_api.InstrumentContext.reset_tipracks'). In the live run, move the empty tip rack off the deck and move a full one into place: + +``` +if protocol.params.dry_run is True: + pipette.reset_tipracks() +else: + protocol.move_labware( + labware=tips_1, new_location=chute, use_gripper=True + ) + protocol.move_labware( + labware=tips_2, new_location="C3", use_gripper=True + ) + +``` + +You can modify this code for similar cases. You may be moving tip racks by hand, rather than with the gripper. Or you could even mix the two, moving the used (but full) rack off\-deck by hand — instead of dropping it down the chute, spilling all the tips — and have the gripper move a new rack into place. Ultimately, it’s up to you to fine\-tune your dry run behavior, and communicate it to your protocol’s users with your parameter descriptions. + +### Parameter Style Guide + +It’s important to write clear names and descriptions when you [define parameters](index.html#defining-rtp) in your protocols. Clarity improves the user experience for the technicians who run your protocols. They rely on your parameter names and descriptions to understand how the robot will function when running your protocol. + +Adopting the advice of this guide will help make your protocols clear, consistent, and ultimately easy to use. It also aligns them with protocols in the [Opentrons Protocol Library](https://library.opentrons.com), which can help others access and replicate your science. + +#### General Guidance + +**Parameter names are nouns.** Parameters should be discrete enough that you can describe them in a single word or short noun phrase. `display_name` is limited to 30 characters, and you can add more context in the description. + +Don’t ask questions or put other sentence punctuation in parameter names. For example: + +| ✅ Dry run | ❌ Dry run? | +| -------------------- | -------------------------------- | +| ✅ Sample count | ❌ How many samples? | +| ✅ Number of samples | ❌ Number of samples to process. | + +**Parameter descriptions explain actions.** In one or two clauses or sentences, state when and how the parameter value is used in the protocol. Don’t merely restate the parameter name. + +Punctuate descriptions as sentences, even if they aren’t complete sentences. For example: + +| Parameter name | Parameter description | +| --------------- | --------------------------------------------------------------------------------------------------------------------------------------------- | +| Dry run | _ ✅ Skip incubation delays and shorten mix steps. _ ❌ Whether to do a dry run. | +| Aspirate volume | _ ✅ How much to aspirate from each sample. _ ❌ Volume that the pipette will aspirate | +| Dilution factor | _ ✅ Each step uses this ratio of total liquid to original solution. Express the ratio as a decimal. _ ❌ total/diluent ratio for the process | + +Not every parameter requires a description! For example, in a protocol that uses only one pipette, it would be difficult to explain a parameter named “Pipette type” without repeating yourself. In a protocol that offers parameters for two different pipettes, it may be useful to summarize what steps each pipette performs. + +**Use sentence case for readability**. Sentence case means adding a capital letter to _only_ the first word of the name and description. This gives your parameters a professional appearance. Keep proper names capitalized as they would be elsewhere in a sentence. For example: + +| ✅ Number of samples | ❌ number of samples | +| -------------------------- | -------------------------- | +| ✅ Temperature Module slot | ❌ Temperature module slot | +| ✅ Dilution factor | ❌ Dilution Factor | + +**Use numerals for all numbers.** In a scientific context, this includes single\-digit numbers. Additionally, punctuate numbers according to the needs of your protocol’s users. If you plan to share your protocol widely, consider using American English number punctuation (comma for thousands separator; period for decimal separator). + +**Order choices logically.** Place items within the `choices` attribute in the order that makes sense for your application. + +Numeric choices should either ascend or descend. Consider an offset parameter with choices. Sorting according to value is easy to use in either direction, but sorting by absolute value is difficult: + +| ✅ \-3, \-2, \-1, 0, 1, 2, 3 | ❌ 0, 1, \-1, 2, \-2, 3, \-3 | +| ---------------------------- | ---------------------------- | +| ✅ 3, 2, 1, 0, \-1, \-2, \-3 | | + +String choices may have an intrinsic ordering. If they don’t, fall back to alphabetical order. + +| Parameter name | Parameter description | +| -------------- | ------------------------------------------------------------------------------------------- | +| Liquid color | _ ✅ Red, Orange, Yellow, Green, Blue, Violet _ ❌ Blue, Green, Orange, Red, Violet, Yellow | +| Tube brand | _ ✅ Eppendorf, Falcon, Generic, NEST _ ❌ Falcon, NEST, Eppendorf, Generic | + +#### Type\-Specific Guidance + +##### Booleans + +The `True` value of a Boolean corresponds to the word _On_ and the `False` value corresponds to the word _Off_. + +**Avoid double negatives.** These are difficult to understand and may lead to a technician making an incorrect choice. Remember that negation can be part of a word’s meaning! For example, it’s difficult to reason about what will happen when a parameter named “Deactivate module” is set to “Off”. + +**When in doubt, clarify in the description.** If you feel like you need to add extra clarity to your Boolean choices, use the phrase “When on” or “When off” at the beginning of your description. For example, a parameter named “Dry run” could have the description “When on, skip protocol delays and return tips instead of trashing them.” + +##### Number Choices + +**Don’t repeat text in choices.** Rely on the name and description to indicate what the number refers to. It’s OK to add units to the display names of numeric choices, because the `unit` attribute is ignored when you specify `choices`. + +| Parameter name | Parameter description | +| ----------------- | -------------------------------------------------------------------------------------------------------------------- | +| Number of columns | _ ✅ 1, 2, 3 _ ❌ 1 column, 2 columns, 3 columns | +| Aspirate volume | _ ✅ 10 µL, 20 µL, 50 µL _ ✅ Low (10 µL), Medium (20 µL), High (50 µL) \* ❌ Low volume, Medium volume, High volume | + +**Use a range instead of choices when all values are acceptable.** It’s faster and easier to enter a numeric value than to choose from a long list. For example, a “Number of columns” parameter that accepts any number 1 through 12 should specify a `minimum` and `maximum`, rather than `choices`. However, if the application requires that the parameter only accepts even numbers, you need to specify choices (2, 4, 6, 8, 10, 12\). + +##### Strings + +**Avoid strings that are synonymous with “yes” and “no”.** When presenting exactly two string choices, consider their meaning. Can they be rephrased in terms of “yes/no”, “true/false”, or “on/off”? If no, then a string parameter is appropriate. If yes, it’s better to use a Boolean, which appears in run setup as a toggle rather than a dropdown menu. + +> - ✅ Blue, Red +> - ✅ Left\-to\-right, Right\-to\-left +> - ❌ Include, Exclude +> - ❌ Yes, No + +Runtime parameters let you define user\-customizable variables in your Python protocols. This gives you greater flexibility and puts extra control in the hands of the technician running the protocol — without forcing them to switch between lots of protocol files or write code themselves. + +This section begins with the fundamentals of runtime parameters: + +- Preliminary advice on how to [choose good parameters](index.html#good-rtps), before you start writing code. +- The syntax for [defining parameters](index.html#defining-rtp) with boolean, numeric, and string values. +- How to [use parameter values](index.html#using-rtp) in your protocol, building logic and API calls that implement the technician’s choices. + +It continues with a selection of use cases and some overall style guidance. When adding parameters, you are in charge of the user experience when it comes time to set up the protocol! These pages outline best practices for making your protocols reliable and easy to use. + +- [Use case – sample count](index.html#use-case-sample-count): Change behavior throughout a protocol based on how many samples you plan to process. Setting sample count exactly saves time, tips, and reagents. +- [Use case – dry run](index.html#use-case-dry-run): Test your protocol, rather than perform a live run, just by flipping a toggle. +- [Style and usage](index.html#rtp-style): When you’re a protocol author, you write code. When you’re a parameter author, you write words. Follow this advice to make things as clear as possible for the technicians who will run your protocol. + +## Advanced Control + +As its name implies, the Python Protocol API is primarily designed for creating protocols that you upload via the Opentrons App and execute on the robot as a unit. But sometimes it’s more convenient to control the robot outside of the app. For example, you might want to have variables in your code that change based on user input or the contents of a CSV file. Or you might want to only execute part of your protocol at a time, especially when developing or debugging a new protocol. + +The Python API offers two ways of issuing commands to the robot outside of the app: through Jupyter Notebook or on the command line with `opentrons_execute`. + +### Jupyter Notebook + +The Flex and OT\-2 run [Jupyter Notebook](https://jupyter.org) servers on port 48888, which you can connect to with your web browser. This is a convenient environment for writing and debugging protocols, since you can define different parts of your protocol in different notebook cells and run a single cell at a time. + +Access your robot’s Jupyter Notebook by either: + +- Going to the **Advanced** tab of Robot Settings and clicking **Launch Jupyter Notebook**. +- Going directly to `http://:48888` in your web browser (if you know your robot’s IP address). + +Once you’ve launched Jupyter Notebook, you can create a notebook file or edit an existing one. These notebook files are stored on the the robot. If you want to save code from a notebook to your computer, go to **File \> Download As** in the notebook interface. + +#### Protocol Structure + +Jupyter Notebook is structured around cells: discrete chunks of code that can be run individually. This is nearly the opposite of Opentrons protocols, which bundle all commands into a single `run` function. Therefore, to take full advantage of Jupyter Notebook, you have to restructure your protocol. + +Rather than writing a `run` function and embedding commands within it, start your notebook by importing `opentrons.execute` and calling [`opentrons.execute.get_protocol_api()`](index.html#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api'). This function also replaces the `metadata` block of a standalone protocol by taking the minimum [API version](index.html#v2-versioning) as its argument. Then you can call [`ProtocolContext`](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.ProtocolContext') methods in subsequent lines or cells: + +``` +import opentrons.execute +protocol = opentrons.execute.get_protocol_api("2.19") +protocol.home() + +``` + +The first command you execute should always be [`home()`](index.html#opentrons.protocol_api.ProtocolContext.home 'opentrons.protocol_api.ProtocolContext.home'). If you try to execute other commands first, you will get a `MustHomeError`. (When running protocols through the Opentrons App, the robot homes automatically.) + +You should use the same [`ProtocolContext`](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.ProtocolContext') throughout your notebook, unless you need to start over from the beginning of your protocol logic. In that case, call [`get_protocol_api()`](index.html#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api') again to get a new [`ProtocolContext`](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.ProtocolContext'). + +#### Running a Previously Written Protocol + +You can also use Jupyter to run a protocol that you have already written. To do so, first copy the entire text of the protocol into a cell and run that cell: + +``` +import opentrons.execute +from opentrons import protocol_api +def run(protocol: protocol_api.ProtocolContext): + # the contents of your previously written protocol go here + +``` + +Since a typical protocol only defines the `run` function but doesn’t call it, this won’t immediately cause the robot to move. To begin the run, instantiate a [`ProtocolContext`](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.ProtocolContext') and pass it to the `run` function you just defined: + +``` +protocol = opentrons.execute.get_protocol_api("2.19") +run(protocol) # your protocol will now run + +``` + +### Setting Labware Offsets + +All positions relative to labware are adjusted automatically based on labware offset data. When you’re running your code in Jupyter Notebook or with `opentrons_execute`, you need to set your own offsets because you can’t perform run setup and Labware Position Check in the Opentrons App or on the Flex touchscreen. + +#### Creating a Dummy Protocol + +For advanced control applications, do the following to calculate and apply labware offsets: + +> 1. Create a “dummy” protocol that loads your labware and has each used pipette pick up a tip from a tip rack. +> 2. Import the dummy protocol to the Opentrons App. +> 3. Run Labware Position Check from the app or touchscreen. +> 4. Add the offsets to your code with [`set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset'). + +Creating the dummy protocol requires you to: + +> 1. Use the `metadata` or `requirements` dictionary to specify the API version. (See [Versioning](index.html#v2-versioning) for details.) Use the same API version as you did in [`opentrons.execute.get_protocol_api()`](index.html#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api'). +> 2. Define a `run()` function. +> 3. Load all of your labware in their initial locations. +> 4. Load your smallest capacity pipette and specify its `tip_racks`. +> 5. Call `pick_up_tip()`. Labware Position Check can’t run if you don’t pick up a tip. + +For example, the following dummy protocol will use a P300 Single\-Channel GEN2 pipette to enable Labware Position Check for an OT\-2 tip rack, NEST reservoir, and NEST flat well plate. + +``` +metadata = {"apiLevel": "2.13"} + + def run(protocol): + tiprack = protocol.load_labware("opentrons_96_tiprack_300ul", 1) + reservoir = protocol.load_labware("nest_12_reservoir_15ml", 2) + plate = protocol.load_labware("nest_96_wellplate_200ul_flat", 3) + p300 = protocol.load_instrument("p300_single_gen2", "left", tip_racks=[tiprack]) + p300.pick_up_tip() + p300.return_tip() + +``` + +After importing this protocol to the Opentrons App, run Labware Position Check to get the x, y, and z offsets for the tip rack and labware. When complete, you can click **Get Labware Offset Data** to view automatically generated code that uses [`set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset') to apply the offsets to each piece of labware. + +``` +labware_1 = protocol.load_labware("opentrons_96_tiprack_300ul", location="1") +labware_1.set_offset(x=0.00, y=0.00, z=0.00) + +labware_2 = protocol.load_labware("nest_12_reservoir_15ml", location="2") +labware_2.set_offset(x=0.10, y=0.20, z=0.30) + +labware_3 = protocol.load_labware("nest_96_wellplate_200ul_flat", location="3") +labware_3.set_offset(x=0.10, y=0.20, z=0.30) + +``` + +This automatically generated code uses generic names for the loaded labware. If you want to match the labware names already in your protocol, change the labware names to match your original code: + +``` +reservoir = protocol.load_labware("nest_12_reservoir_15ml", "2") +reservoir.set_offset(x=0.10, y=0.20, z=0.30) + +``` + +New in version 2\.12\. + +Once you’ve executed this code in Jupyter Notebook, all subsequent positional calculations for this reservoir in slot 2 will be adjusted 0\.1 mm to the right, 0\.2 mm to the back, and 0\.3 mm up. + +Keep in mind that `set_offset()` commands will override any labware offsets set by running Labware Position Check in the Opentrons App. And you should follow the behavior of Labware Position Check, i.e., _do not_ reuse offset measurements unless they apply to the _same labware type_ in the _same deck slot_ on the _same robot_. + +Warning + +Improperly reusing offset data may cause your robot to move to an unexpected position or crash against labware, which can lead to incorrect protocol execution or damage your equipment. When in doubt: run Labware Position Check again and update your code! + +#### Labware Offset Behavior + +How the API applies labware offsets varies depending on the API level of your protocol. This section describes the latest behavior. For details on how offsets work in earlier API versions, see the API reference entry for [`set_offset()`](index.html#opentrons.protocol_api.Labware.set_offset 'opentrons.protocol_api.Labware.set_offset'). + +In the latest API version, offsets apply to labware type–location combinations. For example, if you use `set_offset()` on a tip rack, use all the tips, and replace the rack with a fresh one of the same type in the same location, the offsets will apply to the fresh tip rack: + +``` +tiprack = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="D3" +) +tiprack2 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", + location=protocol_api.OFF_DECK, +) +tiprack.set_offset(x=0.1, y=0.1, z=0.1) +protocol.move_labware( + labware=tiprack, new_location=protocol_api.OFF_DECK +) # tiprack has no offset while off-deck +protocol.move_labware( + labware=tiprack2, new_location="D3" +) # tiprack2 now has offset 0.1, 0.1, 0.1 + +``` + +Because offsets apply to combinations of labware type and location, if you want an offset to apply to a piece of labware as it moves around the deck, call `set_offset()` again after each movement: + +``` +plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="D2" +) +plate.set_offset( + x=-0.1, y=-0.2, z=-0.3 +) # plate now has offset -0.1, -0.2, -0.3 +protocol.move_labware( + labware=plate, new_location="D3" +) # plate now has offset 0, 0, 0 +plate.set_offset( + x=-0.1, y=-0.2, z=-0.3 +) # plate again has offset -0.1, -0.2, -0.3 + +``` + +### Using Custom Labware + +If you have custom labware definitions you want to use with Jupyter, make a new directory called `labware` in Jupyter and put the definitions there. These definitions will be available when you call [`load_labware()`](index.html#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). + +### Using Modules + +If your protocol uses [modules](index.html#new-modules), you need to take additional steps to make sure that Jupyter Notebook doesn’t send commands that conflict with the robot server. Sending commands to modules while the robot server is running will likely cause errors, and the module commands may not execute as expected. + +To disable the robot server, open a Jupyter terminal session by going to **New \> Terminal** and run `systemctl stop opentrons-robot-server`. Then you can run code from cells in your notebook as usual. When you are done using Jupyter Notebook, you should restart the robot server with `systemctl start opentrons-robot-server`. + +Note + +While the robot server is stopped, the robot will display as unavailable in the Opentrons App. If you need to control the robot or its attached modules through the app, you need to restart the robot server and wait for the robot to appear as available in the app. + +### Command Line + +The robot’s command line is accessible either by going to **New \> Terminal** in Jupyter or [via SSH](https://support.opentrons.com/s/article/Connecting-to-your-OT-2-with-SSH). + +To execute a protocol from the robot’s command line, copy the protocol file to the robot with `scp` and then run the protocol with `opentrons_execute`: + +``` +opentrons_execute /data/my_protocol.py + +``` + +By default, `opentrons_execute` will print out the same run log shown in the Opentrons App, as the protocol executes. It also prints out internal logs at the level `warning` or above. Both of these behaviors can be changed. Run `opentrons_execute --help` for more information. + +## Protocol Examples + +This page provides simple, ready\-made protocols for Flex and OT\-2\. Feel free to copy and modify these examples to create unique protocols that help automate your laboratory workflows. Also, experimenting with these protocols is another way to build upon the skills you’ve learned from working through the [tutorial](index.html#tutorial). Try adding different hardware, labware, and commands to a sample protocol and test its validity after importing it into the Opentrons App. + +### Using These Protocols + +These sample protocols are designed for anyone using an Opentrons Flex or OT\-2 liquid handling robot. For our users with little to no Python experience, we’ve taken some liberties with the syntax and structure of the code to make it easier to understand. For example, we’ve formatted the samples with line breaks to show method arguments clearly and to avoid horizontal scrolling. Additionally, the methods use [named arguments](https://en.wikipedia.org/wiki/Named_parameter) instead of positional arguments. For example: + +``` +# This code uses named arguments +tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + +# This code uses positional arguments +tiprack_1 = protocol.load_labware("opentrons_flex_96_tiprack_200ul", "D2") + +``` + +Both examples instantiate the variable `tiprack_1` with a Flex tip rack, but the former is more explicit. It shows the parameter name and its value together (e.g. `location="D2"`), which may be helpful when you’re unsure about what’s going on in a protocol code sample. + +Python developers with more experience should feel free to ignore the code styling used here and work with these examples as you like. + +### Instruments and Labware + +The sample protocols all use the following pipettes: + +- Flex 1\-Channel Pipette (5–1000 µL). The API load name for this pipette is `flex_1channel_1000`. +- P300 Single\-Channel GEN2 pipette for the OT\-2\. The API load name for this pipette is `p300_single_gen2`. + +They also use the labware listed below: + +| Labware type | Labware name | API load name | +| -------------- | --------------------------------------- | --------------------------------- | +| Reservoir | USA Scientific 12\-Well Reservoir 22 mL | `usascientific_12_reservoir_22ml` | +| Well plate | Corning 96\-Well Plate 360 µL Flat | `corning_96_wellplate_360ul_flat` | +| Flex tip rack | Opentrons Flex 96 Tip Rack 200 µL | `opentrons_flex_96_tiprack_200ul` | +| OT\-2 tip rack | Opentrons 96 Tip Rack 300 µL | `opentrons_96_tiprack_300ul` | + +### Protocol Template + +This code only loads the instruments and labware listed above, and performs no other actions. Many code snippets from elsewhere in the documentation will run without modification when added at the bottom of this template. You can also use it to start writing and testing your own code. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # load tip rack in deck slot D3 + tiprack = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", location="D3" + ) + # attach pipette to left mount + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack] + ) + # load well plate in deck slot D2 + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location="D2" + ) + # load reservoir in deck slot D1 + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", location="D1" + ) + # load trash bin in deck slot A3 + trash = protocol.load_trash_bin(location="A3") + # Put protocol commands here + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + # load tip rack in deck slot 3 + tiprack = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", location=3 + ) + # attach pipette to left mount + pipette = protocol.load_instrument( + instrument_name="p300_single_gen2", + mount="left", + tip_racks=[tiprack] + ) + # load well plate in deck slot 2 + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", location=2 + ) + # load reservoir in deck slot 1 + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", location=1 + ) + # Put protocol commands here + +``` + +### Transferring Liquids + +These protocols demonstrate how to move 100 µL of liquid from one well to another. + +#### Basic Method + +This protocol uses some [building block commands](index.html#v2-atomic-commands) to tell the robot, explicitly, where to go to aspirate and dispense liquid. These commands include the [`pick_up_tip()`](index.html#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'), [`aspirate()`](index.html#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'), and [`dispense()`](index.html#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') methods. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel":"2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1]) + + pipette.pick_up_tip() + pipette.aspirate(100, plate["A1"]) + pipette.dispense(100, plate["B1"]) + pipette.drop_tip() + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="left", + tip_racks=[tiprack_1]) + + p300.pick_up_tip() + p300.aspirate(100, plate["A1"]) + p300.dispense(100, plate["B1"]) + p300.drop_tip() + +``` + +#### Advanced Method + +This protocol accomplishes the same thing as the previous example, but does it a little more efficiently. Notice how it uses the [`InstrumentContext.transfer()`](index.html#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') method to move liquid between well plates. The source and destination well arguments (e.g., `plate["A1"], plate["B1"]`) are part of `transfer()` method parameters. You don’t need separate calls to `aspirate` or `dispense` here. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1]) + # transfer 100 µL from well A1 to well B1 + pipette.transfer(100, plate["A1"], plate["B1"]) + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="left", + tip_racks=[tiprack_1]) + # transfer 100 µL from well A1 to well B1 + p300.transfer(100, plate["A1"], plate["B1"]) + +``` + +### Loops + +In Python, a loop is an instruction that keeps repeating an action until a specific condition is met. + +When used in a protocol, loops automate repetitive steps such as aspirating and dispensing liquids from a reservoir to a a range of wells, or all the wells, in a well plate. For example, this code sample loops through the numbers 0 to 7, and uses the loop’s current value to transfer liquid from all the wells in a reservoir to all the wells in a 96\-well plate. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel":"2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location="D3") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1]) + + # distribute 20 µL from reservoir:A1 -> plate:row:1 + # distribute 20 µL from reservoir:A2 -> plate:row:2 + # etc... + # range() starts at 0 and stops before 8, creating a range of 0-7 + for i in range(8): + pipette.distribute(200, reservoir.wells()[i], plate.rows()[i]) + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location=4) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="left", + tip_racks=[tiprack_1]) + + # distribute 20 µL from reservoir:A1 -> plate:row:1 + # distribute 20 µL from reservoir:A2 -> plate:row:2 + # etc... + # range() starts at 0 and stops before 8, creating a range of 0-7 + for i in range(8): + p300.distribute(200, reservoir.wells()[i], plate.rows()[i]) + +``` + +Notice here how Python’s [`range`](https://docs.python.org/3/library/stdtypes.html#range '(in Python v3.12)') class (e.g., `range(8)`) determines how many times the code loops. Also, in Python, a range of numbers is _exclusive_ of the end value and counting starts at 0, not 1\. For the Corning 96\-well plate used here, this means well A1\=0, B1\=1, C1\=2, and so on to the last well in the row, which is H1\=7\. + +### Multiple Air Gaps + +Opentrons electronic pipettes can do some things that a human cannot do with a pipette, like accurately alternate between liquid and air aspirations that create gaps within the same tip. The protocol shown below shows you how to aspirate from the first five wells in the reservoir and create an air gap between each sample. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel":"2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_1000ul", + location="D2") + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location="D3") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1]) + + pipette.pick_up_tip() + + # aspirate from the first 5 wells + for well in reservoir.wells()[:5]: + pipette.aspirate(volume=35, location=well) + pipette.air_gap(10) + + pipette.dispense(225, plate["A1"]) + + pipette.return_tip() + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location=3) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="right", + tip_racks=[tiprack_1]) + + p300.pick_up_tip() + + # aspirate from the first 5 wells + for well in reservoir.wells()[:5]: + p300.aspirate(volume=35, location=well) + p300.air_gap(10) + + p300.dispense(225, plate["A1"]) + + p300.return_tip() + +``` + +Notice here how Python’s [`slice`](https://docs.python.org/3/library/functions.html#slice '(in Python v3.12)') functionality (in the code sample as `[:5]`) lets us select the first five wells of the well plate only. Also, in Python, a range of numbers is _exclusive_ of the end value and counting starts at 0, not 1\. For the USA Scientific 12\-well reservoir used here, this means well A1\=0, A2\=1, A3\=2, and so on to the last well used, which is A5\=4\. See also, the [Commands](index.html#tutorial-commands) section of the Tutorial. + +### Dilution + +This protocol dispenses diluent to all wells of a Corning 96\-well plate. Next, it dilutes 8 samples from the reservoir across all 8 columns of the plate. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + tiprack_2 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D3") + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location="C1") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="left", + tip_racks=[tiprack_1, tiprack_2]) + # Dispense diluent + pipette.distribute(50, reservoir["A12"], plate.wells()) + + # loop through each row + for i in range(8): + # save the source well and destination column to variables + source = reservoir.wells()[i] + row = plate.rows()[i] + + # transfer 30 µL of source to first well in column + pipette.transfer(30, source, row[0], mix_after=(3, 25)) + + # dilute the sample down the column + pipette.transfer( + 30, row[:11], row[1:], + mix_after=(3, 25)) + +``` + +### OT-2 + +``` +from opentrons import protocol_api + +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + tiprack_2 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=3) + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location=4) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="right", + tip_racks=[tiprack_1, tiprack_2]) + # Dispense diluent + p300.distribute(50, reservoir["A12"], plate.wells()) + + # loop through each row + for i in range(8): + # save the source well and destination column to variables + source = reservoir.wells()[i] + source = reservoir.wells()[i] + row = plate.rows()[i] + + # transfer 30 µL of source to first well in column + p300.transfer(30, source, row[0], mix_after=(3, 25)) + + # dilute the sample down the column + p300.transfer( + 30, row[:11], row[1:], + mix_after=(3, 25)) + +``` + +Notice here how the code sample loops through the rows and uses slicing to distribute the diluent. For information about these features, see the Loops and Air Gaps examples above. See also, the [Commands](index.html#tutorial-commands) section of the Tutorial. + +### Plate Mapping + +This protocol dispenses different volumes of liquids to a well plate and automatically refills the pipette when empty. + +### Flex + +``` +from opentrons import protocol_api + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location="D1") + tiprack_1 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D2") + tiprack_2 = protocol.load_labware( + load_name="opentrons_flex_96_tiprack_200ul", + location="D3") + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location="C1") + trash = protocol.load_trash_bin("A3") + pipette = protocol.load_instrument( + instrument_name="flex_1channel_1000", + mount="right", + tip_racks=[tiprack_1, tiprack_2]) + + # Volume amounts are for demonstration purposes only + water_volumes = [ + 1, 2, 3, 4, 5, 6, 7, 8, + 9, 10, 11, 12, 13, 14, 15, 16, + 17, 18, 19, 20, 21, 22, 23, 24, + 25, 26, 27, 28, 29, 30, 31, 32, + 33, 34, 35, 36, 37, 38, 39, 40, + 41, 42, 43, 44, 45, 46, 47, 48, + 49, 50, 51, 52, 53, 54, 55, 56, + 57, 58, 59, 60, 61, 62, 63, 64, + 65, 66, 67, 68, 69, 70, 71, 72, + 73, 74, 75, 76, 77, 78, 79, 80, + 81, 82, 83, 84, 85, 86, 87, 88, + 89, 90, 91, 92, 93, 94, 95, 96 + ] + + pipette.distribute(water_volumes, reservoir["A12"], plate.wells()) + +``` + +### OT-2 + +``` +from opentrons import protocol_api +metadata = {"apiLevel": "2.19"} + +def run(protocol: protocol_api.ProtocolContext): + plate = protocol.load_labware( + load_name="corning_96_wellplate_360ul_flat", + location=1) + tiprack_1 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=2) + tiprack_2 = protocol.load_labware( + load_name="opentrons_96_tiprack_300ul", + location=3) + reservoir = protocol.load_labware( + load_name="usascientific_12_reservoir_22ml", + location=4) + p300 = protocol.load_instrument( + instrument_name="p300_single", + mount="right", + tip_racks=[tiprack_1, tiprack_2]) + + # Volume amounts are for demonstration purposes only + water_volumes = [ + 1, 2, 3, 4, 5, 6, 7, 8, + 9, 10, 11, 12, 13, 14, 15, 16, + 17, 18, 19, 20, 21, 22, 23, 24, + 25, 26, 27, 28, 29, 30, 31, 32, + 33, 34, 35, 36, 37, 38, 39, 40, + 41, 42, 43, 44, 45, 46, 47, 48, + 49, 50, 51, 52, 53, 54, 55, 56, + 57, 58, 59, 60, 61, 62, 63, 64, + 65, 66, 67, 68, 69, 70, 71, 72, + 73, 74, 75, 76, 77, 78, 79, 80, + 81, 82, 83, 84, 85, 86, 87, 88, + 89, 90, 91, 92, 93, 94, 95, 96 + ] + + p300.distribute(water_volumes, reservoir["A12"], plate.wells()) + +``` + +## Adapting OT\-2 Protocols for Flex + +Python protocols designed to run on the OT\-2 can’t be directly run on Flex without some modifications. This page describes the minimal steps that you need to take to get OT\-2 protocols analyzing and running on Flex. + +Adapting a protocol for Flex lets you have parity across different Opentrons robots in your lab, or you can extend older protocols to take advantage of new features only available on Flex. Depending on your application, you may need to do additional verification of your adapted protocol. + +Examples on this page are in tabs so you can quickly move back and forth to see the differences between OT\-2 and Flex code. + +### Metadata and Requirements + +Flex requires you to specify an `apiLevel` of 2\.15 or higher. If your OT\-2 protocol specified `apiLevel` in the `metadata` dictionary, it’s best to move it to the `requirements` dictionary. You can’t specify it in both places, or the API will raise an error. + +Note + +Consult the [list of changes in API versions](index.html#version-notes) to see what effect raising the `apiLevel` will have. If you increased it by multiple minor versions to get your protocol running on Flex, make sure that your protocol isn’t using removed commands or commands whose behavior has changed in a way that may affect your scientific results. + +You also need to specify `"robotType": "Flex"`. If you omit `robotType` in the `requirements` dictionary, the API will assume the protocol is designed for the OT\-2\. + +### Original OT-2 code + +``` +from opentrons import protocol_api + +metadata = { + "protocolName": "My Protocol", + "description": "This protocol uses the OT-2", + "apiLevel": "2.19" +} + +``` + +### Updated Flex code + +``` +from opentrons import protocol_api + +metadata = { + "protocolName": "My Protocol", + "description": "This protocol uses the Flex", +} + +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +``` + +### Pipettes and Tip\-rack Load Names + +Flex uses different types of pipettes and tip racks than OT\-2, which have their own load names in the API. If possible, load Flex pipettes of the same capacity or larger than the OT\-2 pipettes. See the [list of pipette API load names](index.html#new-pipette-models) for the valid values of `instrument_name` in Flex protocols. And check [Labware Library](https://labware.opentrons.com) or the Opentrons App for the load names of Flex tip racks. + +Note + +If you use smaller capacity tips than in the OT\-2 protocol, you may need to make further adjustments to avoid running out of tips. Also, the protocol may have more steps and take longer to execute. + +This example converts OT\-2 code that uses a P300 Single\-Channel GEN2 pipette and 300 µL tips to Flex code that uses a Flex 1\-Channel 1000 µL pipette and 1000 µL tips. + +### Original OT-2 code + +``` +def run(protocol: protocol_api.ProtocolContext): + tips = protocol.load_labware("opentrons_96_tiprack_300ul", 1) + left_pipette = protocol.load_instrument( + "p300_single_gen2", "left", tip_racks=[tips] + ) + +``` + +### Updated Flex code + +``` +def run(protocol: protocol_api.ProtocolContext): + tips = protocol.load_labware("opentrons_flex_96_tiprack_1000ul", "D1") + left_pipette = protocol.load_instrument( + "flex_1channel_1000", "left", tip_racks[tips] + ) + +``` + +### Trash Container + +OT\-2 protocols always have a [`fixed_trash`](index.html#opentrons.protocol_api.ProtocolContext.fixed_trash 'opentrons.protocol_api.ProtocolContext.fixed_trash') in slot 12\. In Flex protocols specifying API version 2\.16 or later, you need to [load a trash bin](index.html#configure-trash-bin). Put it in slot A3 to match the physical position of the OT\-2 fixed trash: + +``` +trash = protocol.load_trash_bin("A3") + +``` + +### Deck Slot Labels + +It’s good practice to update numeric labels for [deck slots](index.html#deck-slots) (which match the labels on an OT\-2\) to coordinate ones (which match the labels on Flex). This is an optional step, since the two formats are interchangeable. + +For example, the code in the previous section changed the location of the tip rack from `1` to `"D1"`. + +### Module Load Names + +If your OT\-2 protocol uses older generations of the Temperature Module or Thermocycler Module, update the load names you pass to [`load_module()`](index.html#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module') to ones compatible with Flex: + +> - `temperature module gen2` +> - `thermocycler module gen2` or `thermocyclerModuleV2` + +The Heater\-Shaker Module only has one generation, `heaterShakerModuleV1`, which is compatible with Flex and OT\-2\. + +The Magnetic Module is not compatible with Flex. For protocols that load `magnetic module`, `magdeck`, or `magnetic module gen2`, you will need to make further modifications to use the [Magnetic Block](index.html#magnetic-block) and Flex Gripper instead. This will require reworking some of your protocol steps, and you should verify that your new protocol design achieves similar results. + +This simplified example, taken from a DNA extraction protocol, shows how using the Flex Gripper and the Magnetic Block can save time. Instead of pipetting an entire plate’s worth of liquid from the Heater\-Shaker to the Magnetic Module and then engaging the module, the gripper moves the plate to the Magnetic Block in one step. + +### Original OT-2 code + +``` +hs_mod.set_and_wait_for_shake_speed(2000) +protocol.delay(minutes=5) +hs_mod.deactivate_shaker() + +for i in sample_plate.wells(): + # mix, transfer, and blow-out all samples + pipette.pick_up_tip() + pipette.aspirate(100,hs_plate[i]) + pipette.dispense(100,hs_plate[i]) + pipette.aspirate(100,hs_plate[i]) + pipette.air_gap(10) + pipette.dispense(pipette.current_volume,mag_plate[i]) + pipette.aspirate(50,hs_plate[i]) + pipette.air_gap(10) + pipette.dispense(pipette.current_volume,mag_plate[i]) + pipette.blow_out(mag_plate[i].bottom(0.5)) + pipette.drop_tip() + +mag_mod.engage() + +# perform elution steps + +``` + +### Updated Flex code + +``` +hs_mod.set_and_wait_for_shake_speed(2000) +protocol.delay(minutes=5) +hs_mod.deactivate_shaker() + +# move entire plate +# no pipetting from Heater-Shaker needed +hs_mod.open_labware_latch() +protocol.move_labware(sample_plate, mag_block, use_gripper=True) + +# perform elution steps + +``` + +The Opentrons Python Protocol API is a Python framework designed to make it easy to write automated biology lab protocols. Python protocols can control Opentrons Flex and OT\-2 robots, their pipettes, and optional hardware modules. We’ve designed the API to be accessible to anyone with basic Python and wet\-lab skills. + +As a bench scientist, you should be able to code your protocols in a way that reads like a lab notebook. You can write a fully functional protocol just by listing the equipment you’ll use (modules, labware, and pipettes) and the exact sequence of movements the robot should make. + +As a programmer, you can leverage the full power of Python for advanced automation in your protocols. Perform calculations, manage external data, use built\-in and imported Python modules, and more to implement your custom lab workflow. + +## Getting Started + +**New to Python protocols?** Check out the [Tutorial](index.html#tutorial) to learn about the different parts of a protocol file and build a working protocol from scratch. + +If you want to **dive right into code**, take a look at our [Protocol Examples](index.html#new-examples) and the comprehensive [API Version 2 Reference](index.html#protocol-api-reference). + +When you’re ready to **try out a protocol**, download the [Opentrons App](https://www.opentrons.com/ot-app), import the protocol file, and run it on your robot. + +## How the API Works + +The design goal of this API is to make code readable and easy to understand. A protocol, in its most basic form: + +1. Provides some information about who made the protocol and what it is for. +2. Specifies which type of robot the protocol should run on. +3. Tells the robot where to find labware, pipettes, and (optionally) hardware modules. +4. Commands the robot to manipulate its attached hardware. + +For example, if we wanted to transfer liquid from well A1 to well B1 on a plate, our protocol would look like: + +### Flex + +``` +from opentrons import protocol_api + +# metadata +metadata = { + "protocolName": "My Protocol", + "author": "Name ", + "description": "Simple protocol to get started using the Flex", +} + +# requirements +requirements = {"robotType": "Flex", "apiLevel": "2.19"} + +# protocol run function +def run(protocol: protocol_api.ProtocolContext): + # labware + plate = protocol.load_labware( + "corning_96_wellplate_360ul_flat", location="D1" + ) + tiprack = protocol.load_labware( + "opentrons_flex_96_tiprack_200ul", location="D2" + ) + trash = protocol.load_trash_bin(location="A3") + + # pipettes + left_pipette = protocol.load_instrument( + "flex_1channel_1000", mount="left", tip_racks=[tiprack] + ) + + # commands + left_pipette.pick_up_tip() + left_pipette.aspirate(100, plate["A1"]) + left_pipette.dispense(100, plate["B2"]) + left_pipette.drop_tip() + +``` + +This example proceeds completely linearly. Following it line\-by\-line, you can see that it has the following effects: + +1. Gives the name, contact information, and a brief description for the protocol. +2. Indicates the protocol should run on a Flex robot, using API version 2\.19\. +3. Tells the robot that there is: + 1. A 96\-well flat plate in slot D1\. + 2. A rack of 300 µL tips in slot D2\. + 3. A 1\-channel 1000 µL pipette attached to the left mount, which should pick up tips from the aforementioned rack. +4. Tells the robot to act by: + 1. Picking up the first tip from the tip rack. + 2. Aspirating 100 µL of liquid from well A1 of the plate. + 3. Dispensing 100 µL of liquid into well B1 of the plate. + 4. Dropping the tip in the trash. + +### OT-2 + +``` +from opentrons import protocol_api + +# metadata +metadata = { + "protocolName": "My Protocol", + "author": "Name ", + "description": "Simple protocol to get started using the OT-2", +} + +# requirements +requirements = {"robotType": "OT-2", "apiLevel": "2.19"} + +# protocol run function +def run(protocol: protocol_api.ProtocolContext): + # labware + plate = protocol.load_labware( + "corning_96_wellplate_360ul_flat", location="1" + ) + tiprack = protocol.load_labware( + "opentrons_96_tiprack_300ul", location="2" + ) + + # pipettes + left_pipette = protocol.load_instrument( + "p300_single", mount="left", tip_racks=[tiprack] + ) + + # commands + left_pipette.pick_up_tip() + left_pipette.aspirate(100, plate["A1"]) + left_pipette.dispense(100, plate["B2"]) + left_pipette.drop_tip() + +``` + +This example proceeds completely linearly. Following it line\-by\-line, you can see that it has the following effects: + +1. Gives the name, contact information, and a brief description for the protocol. +2. Indicates the protocol should run on an OT\-2 robot, using API version 2\.19\. +3. Tells the robot that there is: + 1. A 96\-well flat plate in slot 1\. + 2. A rack of 300 µL tips in slot 2\. + 3. A single\-channel 300 µL pipette attached to the left mount, which should pick up tips from the aforementioned rack. +4. Tells the robot to act by: + 1. Picking up the first tip from the tip rack. + 2. Aspirating 100 µL of liquid from well A1 of the plate. + 3. Dispensing 100 µL of liquid into well B1 of the plate. + 4. Dropping the tip in the trash. + +There is much more that Opentrons robots and the API can do! The [Building Block Commands](index.html#v2-atomic-commands), [Complex Commands](index.html#v2-complex-commands), and [Hardware Modules](index.html#new-modules) pages cover many of these functions. + +## More Resources + +### Opentrons App + +The [Opentrons App](https://opentrons.com/ot-app/) is the easiest way to run your Python protocols. The app runs on the latest versions of macOS, Windows, and Ubuntu. + +### Support + +Questions about setting up your robot, using Opentrons software, or troubleshooting? Check out our [support articles](https://support.opentrons.com/s/) or [contact Opentrons Support directly](mailto:support%40opentrons.com). + +### Custom Protocol Service + +Don’t have the time or resources to write your own protocols? Our [custom protocol development service](https://opentrons.com/instrument-services/) can get you set up in two weeks. + +### Contributing + +Opentrons software, including the Python API and this documentation, is open source. If you have an improvement or an interesting idea, you can create an issue on GitHub by following our [guidelines](https://github.com/Opentrons/opentrons/blob/edge/CONTRIBUTING.md#opening-issues). + +That guide also includes more information on how to [directly contribute code](https://github.com/Opentrons/opentrons/blob/edge/CONTRIBUTING.md). diff --git a/opentrons-ai-server/api/data/python_api_219_reference.md b/opentrons-ai-server/api/data/python_api_219_reference.md new file mode 100644 index 00000000000..be4b965abc7 --- /dev/null +++ b/opentrons-ai-server/api/data/python_api_219_reference.md @@ -0,0 +1,3313 @@ +## API Version 2 Reference + +### Protocols + +_class_ opentrons.protocol*api.ProtocolContext(\_api_version: APIVersion*, _core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _broker: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[LegacyBroker] \= None_, _core_map: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[LoadedCoreMap] \= None_, _deck: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[Deck] \= None_, _bundled_data: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Dict](https://docs.python.org/3/library/typing.html#typing.Dict '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), [bytes](https://docs.python.org/3/library/stdtypes.html#bytes '(in Python v3.12)')]] \= None_) +A context for the state of a protocol. + +The `ProtocolContext` class provides the objects, attributes, and methods that +allow you to configure and control the protocol. + +Methods generally fall into one of two categories. + +> - They can change the state of the `ProtocolContext` object, such as adding +> pipettes, hardware modules, or labware to your protocol. +> - They can control the flow of a running protocol, such as pausing, displaying +> messages, or controlling built\-in robot hardware like the ambient lighting. + +Do not instantiate a `ProtocolContext` directly. +The `run()` function of your protocol does that for you. +See the [Tutorial](index.html#run-function) for more information. + +Use [`opentrons.execute.get_protocol_api()`](#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api') to instantiate a `ProtocolContext` when +using Jupyter Notebook. See [Advanced Control](index.html#advanced-control). + +New in version 2\.0\. + +_property_ api_version*: APIVersion* +Return the API version specified for this protocol context. + +This value is set when the protocol context +is initialized. + +> - When the context is the argument of `run()`, the `"apiLevel"` key of the +> [metadata](index.html#tutorial-metadata) or [requirements](index.html#tutorial-requirements) dictionary determines `api_version`. +> - When the context is instantiated with +> [`opentrons.execute.get_protocol_api()`](#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api') or +> [`opentrons.simulate.get_protocol_api()`](#opentrons.simulate.get_protocol_api 'opentrons.simulate.get_protocol_api'), the value of its `version` +> argument determines `api_version`. + +It may be lower than the [maximum version](index.html#max-version) supported by the +robot software, which is accessible via the +`protocol_api.MAX_SUPPORTED_VERSION` constant. + +New in version 2\.0\. + +_property_ bundled_data*: [Dict](https://docs.python.org/3/library/typing.html#typing.Dict '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), [bytes](https://docs.python.org/3/library/stdtypes.html#bytes '(in Python v3.12)')]* +Accessor for data files bundled with this protocol, if any. + +This is a dictionary mapping the filenames of bundled datafiles to their +contents. The filename keys are formatted with extensions but without paths. For +example, a file stored in the bundle as `data/mydata/aspirations.csv` will +have the key `"aspirations.csv"`. The values are [`bytes`](https://docs.python.org/3/library/stdtypes.html#bytes '(in Python v3.12)') objects +representing the contents of the files. + +New in version 2\.0\. + +commands(_self_) → 'List\[str]' +Return the run log. + +This is a list of human\-readable strings representing what’s been done in the protocol so +far. For example, “Aspirating 123 µL from well A1 of 96 well plate in slot 1\.” + +The exact format of these entries is not guaranteed. The format here may differ from other +places that show the run log, such as the Opentrons App or touchscreen. + +New in version 2\.0\. + +comment(_self_, _msg: 'str'_) → 'None' +Add a user\-readable message to the run log. + +The message is visible anywhere you can view the run log, including the Opentrons App and the touchscreen on Flex. + +Note + +The value of the message is computed during protocol analysis, +so `comment()` can’t communicate real\-time information during the +actual protocol run. + +New in version 2\.0\. + +_property_ deck*: Deck* +An interface to provide information about what’s currently loaded on the deck. +This object is useful for determining if a slot on the deck is free. + +This object behaves like a dictionary whose keys are the [deck slot](index.html#deck-slots) names. +For instance, `deck[1]`, `deck["1"]`, and `deck["D1"]` +will all return the object loaded in the front\-left slot. + +The value for each key depends on what is loaded in the slot:\* A [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') if the slot contains a labware. + +- A module context if the slot contains a hardware module. +- `None` if the slot doesn’t contain anything. + +A module that occupies multiple slots is set as the value for all of the +relevant slots. Currently, the only multiple\-slot module is the Thermocycler. +When loaded, the [`ThermocyclerContext`](#opentrons.protocol_api.ThermocyclerContext 'opentrons.protocol_api.ThermocyclerContext') object is the value for +`deck` keys `"A1"` and `"B1"` on Flex, and `7`, `8`, `10`, and +`11` on OT\-2\. In API version 2\.13 and earlier, only slot 7 keyed to the +Thermocycler object, and slots 8, 10, and 11 keyed to `None`. + +Rather than filtering the objects in the deck map yourself, +you can also use [`loaded_labwares`](#opentrons.protocol_api.ProtocolContext.loaded_labwares 'opentrons.protocol_api.ProtocolContext.loaded_labwares') to get a dict of labwares +and [`loaded_modules`](#opentrons.protocol_api.ProtocolContext.loaded_modules 'opentrons.protocol_api.ProtocolContext.loaded_modules') to get a dict of modules. + +For [Advanced Control](index.html#advanced-control) _only_, you can delete an element of the `deck` dict. +This only works for deck slots that contain labware objects. For example, if slot +1 contains a labware, `del protocol.deck["1"]` will free the slot so you can +load another labware there. + +Warning + +Deleting labware from a deck slot does not pause the protocol. Subsequent +commands continue immediately. If you need to physically move the labware to +reflect the new deck state, add a [`pause()`](#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause') or use +[`move_labware()`](#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') instead. + +Changed in version 2\.14: Includes the Thermocycler in all of the slots it occupies. + +Changed in version 2\.15: `del` sets the corresponding labware’s location to `OFF_DECK`. + +New in version 2\.0\. + +define*liquid(\_self*, _name: 'str'_, _description: 'Optional\[str]'_, _display_color: 'Optional\[str]'_) → 'Liquid' +Define a liquid within a protocol. + +Parameters: + +- **name** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – A human\-readable name for the liquid. +- **description** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional description of the liquid. +- **display_color** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional hex color code, with hash included, to represent the specified liquid. Standard three\-value, four\-value, six\-value, and eight\-value syntax are all acceptable. + +Returns: +A [`Liquid`](#opentrons.protocol_api.Liquid 'opentrons.protocol_api.Liquid') object representing the specified liquid. + +New in version 2\.14\. + +delay(_self_, _seconds: 'float' \= 0_, _minutes: 'float' \= 0_, _msg: 'Optional\[str]' \= None_) → 'None' +Delay protocol execution for a specific amount of time. + +Parameters: + +- **seconds** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The time to delay in seconds. +- **minutes** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The time to delay in minutes. + +If both `seconds` and `minutes` are specified, they will be added together. + +New in version 2\.0\. + +_property_ door_closed*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Returns `True` if the front door of the robot is closed. + +New in version 2\.5\. + +_property_ fixed_trash*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware'), [TrashBin](index.html#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.disposal_locations.TrashBin')]* +The trash fixed to slot 12 of an OT\-2’s deck. + +In API version 2\.15 and earlier, the fixed trash is a [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') object with one well. Access it like labware in your protocol. For example, `protocol.fixed_trash["A1"]`. + +In API version 2\.15 only, Flex protocols have a fixed trash in slot A3\. + +In API version 2\.16 and later, the fixed trash only exists in OT\-2 protocols. It is a [`TrashBin`](#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin') object, which doesn’t have any wells. Trying to access `fixed_trash` in a Flex protocol will raise an error. See [Trash Bin](index.html#configure-trash-bin) for details on using the movable trash in Flex protocols. + +Changed in version 2\.16: Returns a `TrashBin` object. + +New in version 2\.0\. + +home(_self_) → 'None' +Home the movement system of the robot. + +New in version 2\.0\. + +is*simulating(\_self*) → 'bool' +Returns `True` if the protocol is running in simulation. + +Returns `False` if the protocol is running on actual hardware. + +You can evaluate the result of this method in an `if` statement to make your +protocol behave differently in different environments. For example, you could +refer to a data file on your computer when simulating and refer to a data file +stored on the robot when not simulating. + +You can also use it to skip time\-consuming aspects of your protocol. Most Python +Protocol API methods, like [`delay()`](#opentrons.protocol_api.ProtocolContext.delay 'opentrons.protocol_api.ProtocolContext.delay'), are designed to evaluate +instantaneously in simulation. But external methods, like those from the +[`time`](https://docs.python.org/3/library/time.html#module-time '(in Python v3.12)') module, will run at normal speed if not skipped. + +New in version 2\.0\. + +load*adapter(\_self*, _load_name: 'str'_, _location: 'Union\[DeckLocation, OffDeckType]'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto a location. + +For adapters already defined by Opentrons, this is a convenient way +to collapse the two stages of adapter initialization (creating +the adapter and adding it to the protocol) into one. + +This function returns the created and initialized adapter for use +later in the protocol. + +Parameters: + +- **load_name** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – A string to use for looking up a labware definition for the adapter. + You can find the `load_name` for any standard adapter on the Opentrons + [Labware Library](https://labware.opentrons.com). +- **location** (int or str or [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK')) – Either a [deck slot](index.html#deck-slots), + like `1`, `"1"`, or `"D1"`, or the special value [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK'). +- **namespace** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – The namespace that the labware definition belongs to. + If unspecified, the API will automatically search two namespaces: + +> - `"opentrons"`, to load standard Opentrons labware definitions. +> - `"custom_beta"`, to load custom labware definitions created with the +> [Custom Labware Creator](https://labware.opentrons.com/create). + +You might need to specify an explicit `namespace` if you have a custom +definition whose `load_name` is the same as an Opentrons standard +definition, and you want to explicitly choose one or the other. + +- **version** – The version of the labware definition. You should normally + leave this unspecified to let `load_adapter()` choose a version automatically. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _adapter_def: "'LabwareDefinition'"_, _location: 'Union\[DeckLocation, OffDeckType]'_) → 'Labware' +Specify the presence of an adapter on the deck. + +This function loads the adapter definition specified by `adapter_def` +to the location specified by `location`. + +Parameters: + +- **adapter_def** – The adapter’s labware definition. +- **location** (int or str or [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK')) – The slot into which to load the labware, + such as `1`, `"1"`, or `"D1"`. See [Deck Slots](index.html#deck-slots). + +New in version 2\.15\. + +load*instrument(\_self*, _instrument_name: 'str'_, _mount: 'Union\[Mount, str, None]' \= None_, _tip_racks: 'Optional\[List\[Labware]]' \= None_, _replace: 'bool' \= False_, _liquid_presence_detection: 'Optional\[bool]' \= None_) → 'InstrumentContext' +Load a specific instrument for use in the protocol. + +When analyzing the protocol on the robot, instruments loaded with this method +are compared against the instruments attached to the robot. You won’t be able to +start the protocol until the correct instruments are attached and calibrated. + +Currently, this method only loads pipettes. You do not need to load the Flex +Gripper to use it in protocols. See [Automatic vs Manual Moves](index.html#automatic-manual-moves). + +Parameters: + +- **instrument_name** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – The instrument to load. See [API Load Names](index.html#new-pipette-models) + for the valid values. +- **mount** (types.Mount or str or `None`) – The mount where the instrument should be attached. + This can either be an instance of [`types.Mount`](#opentrons.types.Mount 'opentrons.types.Mount') or one + of the strings `"left"` or `"right"`. When loading a Flex + 96\-Channel Pipette (`instrument_name="flex_96channel_1000"`), + you can leave this unspecified, since it always occupies both + mounts; if you do specify a value, it will be ignored. +- **tip_racks** (List\[[`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware')]) – A list of tip racks from which to pick tips when calling + [`InstrumentContext.pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') without arguments. +- **replace** ([_bool_](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')) – If `True`, replace the currently loaded instrument in + `mount`, if any. This is intended for [advanced + control](index.html#advanced-control) applications. You cannot + replace an instrument in the middle of a protocol being run + from the Opentrons App or touchscreen. +- **liquid_presence_detection** ([_bool_](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')) – If `True`, enable liquid presence detection for instrument. Only available on Flex robots in API Version 2\.20 and above. + +New in version 2\.0\. + +load*labware(\_self*, _load_name: 'str'_, _location: 'Union\[DeckLocation, OffDeckType]'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto a location. + +For Opentrons\-verified labware, this is a convenient way +to collapse the two stages of labware initialization (creating +the labware and adding it to the protocol) into one. + +This function returns the created and initialized labware for use +later in the protocol. + +Parameters: + +- **load_name** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – A string to use for looking up a labware definition. + You can find the `load_name` for any Opentrons\-verified labware on the + [Labware Library](https://labware.opentrons.com). +- **location** (int or str or [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK')) – Either a [deck slot](index.html#deck-slots), + like `1`, `"1"`, or `"D1"`, or the special value [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK'). + +Changed in version 2\.15: You can now specify a deck slot as a coordinate, like `"D1"`. + +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If specified, + this is how the labware will appear in the run log, Labware Position + Check, and elsewhere in the Opentrons App and on the touchscreen. +- **namespace** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – The namespace that the labware definition belongs to. + If unspecified, the API will automatically search two namespaces: + +> - `"opentrons"`, to load standard Opentrons labware definitions. +> - `"custom_beta"`, to load custom labware definitions created with the +> [Custom Labware Creator](https://labware.opentrons.com/create). + +You might need to specify an explicit `namespace` if you have a custom +definition whose `load_name` is the same as an Opentrons\-verified +definition, and you want to explicitly choose one or the other. + +- **version** – The version of the labware definition. You should normally + leave this unspecified to let `load_labware()` choose a version + automatically. +- **adapter** – An adapter to load the labware on top of. Accepts the same + values as the `load_name` parameter of [`load_adapter()`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter'). The + adapter will use the same namespace as the labware, and the API will + choose the adapter’s version automatically. + +> New in version 2\.15\. + +New in version 2\.0\. + +load*labware_by_name(\_self*, _load_name: 'str'_, _location: 'DeckLocation'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'int' \= 1_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') instead. + +New in version 2\.0\. + +load*labware_from_definition(\_self*, _labware_def: "'LabwareDefinition'"_, _location: 'Union\[DeckLocation, OffDeckType]'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Specify the presence of a labware on the deck. + +This function loads the labware definition specified by `labware_def` +to the location specified by `location`. + +Parameters: + +- **labware_def** – The labware’s definition. +- **location** (int or str or [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK')) – The slot into which to load the labware, + such as `1`, `"1"`, or `"D1"`. See [Deck Slots](index.html#deck-slots). +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If specified, + this is how the labware will appear in the run log, Labware Position + Check, and elsewhere in the Opentrons App and on the touchscreen. + +New in version 2\.0\. + +load*module(\_self*, _module_name: 'str'_, _location: 'Optional\[DeckLocation]' \= None_, _configuration: 'Optional\[str]' \= None_) → 'ModuleTypes' +Load a module onto the deck, given its name or model. + +This is the function to call to use a module in your protocol, like +[`load_instrument()`](#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument') is the method to call to use an instrument +in your protocol. It returns the created and initialized module +context, which will be a different class depending on the kind of +module loaded. + +After loading modules, you can access a map of deck positions to loaded modules +with [`loaded_modules`](#opentrons.protocol_api.ProtocolContext.loaded_modules 'opentrons.protocol_api.ProtocolContext.loaded_modules'). + +Parameters: + +- **module_name** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – The name or model of the module. + See [Available Modules](index.html#available-modules) for possible values. +- **location** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)') _or_ [_int_](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)') _or_ _None_) – The location of the module. + +This is usually the name or number of the slot on the deck where you +will be placing the module, like `1`, `"1"`, or `"D1"`. See [Deck Slots](index.html#deck-slots). + +The Thermocycler is only valid in one deck location. +You don’t have to specify a location when loading it, but if you do, +it must be `7`, `"7"`, or `"B1"`. See [Thermocycler Module](index.html#thermocycler-module). + +Changed in version 2\.15: You can now specify a deck slot as a coordinate, like `"D1"`. + +- **configuration** – Configure a Thermocycler to be in the `semi` position. + This parameter does not work. Do not use it. + +Changed in version 2\.14: This parameter dangerously modified the protocol’s geometry system, +and it didn’t function properly, so it was removed. + +Returns: +The loaded and initialized module—a +[`HeaterShakerContext`](#opentrons.protocol_api.HeaterShakerContext 'opentrons.protocol_api.HeaterShakerContext'), +[`MagneticBlockContext`](#opentrons.protocol_api.MagneticBlockContext 'opentrons.protocol_api.MagneticBlockContext'), +[`MagneticModuleContext`](#opentrons.protocol_api.MagneticModuleContext 'opentrons.protocol_api.MagneticModuleContext'), +[`TemperatureModuleContext`](#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.TemperatureModuleContext'), or +[`ThermocyclerContext`](#opentrons.protocol_api.ThermocyclerContext 'opentrons.protocol_api.ThermocyclerContext'), +depending on what you requested with `module_name`. + +Changed in version 2\.13: Added `HeaterShakerContext` return value. + +Changed in version 2\.15: Added `MagneticBlockContext` return value. + +New in version 2\.0\. + +load*trash_bin(\_self*, _location: 'DeckLocation'_) → 'TrashBin' +Load a trash bin on the deck of a Flex. + +See [Trash Bin](index.html#configure-trash-bin) for details. + +If you try to load a trash bin on an OT\-2, the API will raise an error. + +Parameters: +**location** – The [deck slot](index.html#deck-slots) where the trash bin is. The +location can be any unoccupied slot in column 1 or 3\. + +If you try to load a trash bin in column 2 or 4, the API will raise an error. + +New in version 2\.16\. + +load*waste_chute(\_self*) → 'WasteChute' +Load the waste chute on the deck of a Flex. + +See [Waste Chute](index.html#configure-waste-chute) for details, including the deck configuration +variants of the waste chute. + +The deck plate adapter for the waste chute can only go in slot D3\. If you try to +load another item in slot D3 after loading the waste chute, or vice versa, the +API will raise an error. + +New in version 2\.16\. + +_property_ loaded_instruments*: [Dict](https://docs.python.org/3/library/typing.html#typing.Dict '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), [InstrumentContext](index.html#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.instrument_context.InstrumentContext')]* +Get the instruments that have been loaded into the protocol. + +This is a map of mount name to instruments previously loaded with +[`load_instrument()`](#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument'). It does not reflect what instruments are actually +installed on the robot. For example, if the robot has instruments installed on +both mounts but your protocol has only loaded one of them with +[`load_instrument()`](#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument'), the unused one will not be included in +`loaded_instruments`. + +Returns: +A dict mapping mount name (`"left"` or `"right"`) to the +instrument in that mount. If a mount has no loaded instrument, that key +will be missing from the dict. + +New in version 2\.0\. + +_property_ loaded_labwares*: [Dict](https://docs.python.org/3/library/typing.html#typing.Dict '(in Python v3.12)')\[[int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)'), [Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +Get the labwares that have been loaded into the protocol context. + +Slots with nothing in them will not be present in the return value. + +Note + +If a module is present on the deck but no labware has been loaded +into it with `module.load_labware()`, there will +be no entry for that slot in this value. That means you should not +use `loaded_labwares` to determine if a slot is available or not, +only to get a list of labwares. If you want a data structure of all +objects on the deck regardless of type, use [`deck`](#opentrons.protocol_api.ProtocolContext.deck 'opentrons.protocol_api.ProtocolContext.deck'). + +Returns: +Dict mapping deck slot number to labware, sorted in order of +the locations. + +New in version 2\.0\. + +_property_ loaded_modules*: [Dict](https://docs.python.org/3/library/typing.html#typing.Dict '(in Python v3.12)')\[[int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)'), [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[TemperatureModuleContext](index.html#opentrons.protocol_api.TemperatureModuleContext 'opentrons.protocol_api.module_contexts.TemperatureModuleContext'), [MagneticModuleContext](index.html#opentrons.protocol_api.MagneticModuleContext 'opentrons.protocol_api.module_contexts.MagneticModuleContext'), [ThermocyclerContext](index.html#opentrons.protocol_api.ThermocyclerContext 'opentrons.protocol_api.module_contexts.ThermocyclerContext'), [HeaterShakerContext](index.html#opentrons.protocol_api.HeaterShakerContext 'opentrons.protocol_api.module_contexts.HeaterShakerContext'), [MagneticBlockContext](index.html#opentrons.protocol_api.MagneticBlockContext 'opentrons.protocol_api.module_contexts.MagneticBlockContext'), AbsorbanceReaderContext]]* +Get the modules loaded into the protocol context. + +This is a map of deck positions to modules loaded by previous calls to +[`load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). It does not reflect what modules are actually attached +to the robot. For example, if the robot has a Magnetic Module and a Temperature +Module attached, but the protocol has only loaded the Temperature Module with +[`load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'), only the Temperature Module will be included in +`loaded_modules`. + +Returns: +Dict mapping slot name to module contexts. The elements may not be +ordered by slot number. + +New in version 2\.0\. + +_property_ max_speeds*: AxisMaxSpeeds* +Per\-axis speed limits for moving instruments. + +Changing values within this property sets the speed limit for each non\-plunger +axis of the robot. Note that this property only sets upper limits and can’t +exceed the physical speed limits of the movement system. + +This property is a dict mapping string names of axes to float values +of maximum speeds in mm/s. To change a speed, set that axis’s value. To +reset an axis’s speed to default, delete the entry for that axis +or assign it to `None`. + +See [Axis Speed Limits](index.html#axis-speed-limits) for examples. + +Note + +This property is not yet supported in API version 2\.14 or higher. + +New in version 2\.0\. + +move*labware(\_self*, _labware: 'Labware'_, _new_location: 'Union\[DeckLocation, Labware, ModuleTypes, OffDeckType, WasteChute]'_, _use_gripper: 'bool' \= False_, _pick_up_offset: 'Optional\[Mapping\[str, float]]' \= None_, _drop_offset: 'Optional\[Mapping\[str, float]]' \= None_) → 'None' +Move a loaded labware to a new location. + +See [Moving Labware](index.html#moving-labware) for more details. + +Parameters: + +- **labware** – The labware to move. It should be a labware already loaded + using [`load_labware()`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). +- **new_location** – Where to move the labware to. This is either: + + - A deck slot like `1`, `"1"`, or `"D1"`. See [Deck Slots](index.html#deck-slots). + - A hardware module that’s already been loaded on the deck + with [`load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + - A labware or adapter that’s already been loaded on the deck + with [`load_labware()`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') or [`load_adapter()`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter'). + - The special constant [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK'). + +- **use_gripper** – Whether to use the Flex Gripper for this movement. + + - If `True`, use the gripper to perform an automatic + movement. This will raise an error in an OT\-2 protocol. + - If `False`, pause protocol execution until the user + performs the movement. Protocol execution remains paused until + the user presses **Confirm and resume**. + +Gripper\-only parameters: + +Parameters: + +- **pick_up_offset** – Optional x, y, z vector offset to use when picking up labware. +- **drop_offset** – Optional x, y, z vector offset to use when dropping off labware. + +Before moving a labware to or from a hardware module, make sure that the labware’s +current and new locations are accessible, i.e., open the Thermocycler lid or +open the Heater\-Shaker’s labware latch. + +New in version 2\.15\. + +_property_ params*: Parameters* +The values of runtime parameters, as set during run setup. + +Each attribute of this object corresponds to the `variable_name` of a parameter. +See [Using Parameters](index.html#using-rtp) for details. + +Parameter values can only be set during run setup. If you try to alter the value +of any attribute of `params`, the API will raise an error. + +New in version 2\.18\. + +pause(_self_, _msg: 'Optional\[str]' \= None_) → 'None' +Pause execution of the protocol until it’s resumed. + +A human can resume the protocol in the Opentrons App or on the touchscreen. + +Note + +In Python Protocol API version 2\.13 and earlier, the pause will only +take effect on the next function call that involves moving the robot. + +Parameters: +**msg** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional message to show in the run log entry for the pause step. + +New in version 2\.0\. + +_property_ rail_lights_on*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Returns `True` if the robot’s ambient lighting is on. + +New in version 2\.5\. + +resume(_self_) → 'None' +Resume the protocol after [`pause()`](#opentrons.protocol_api.ProtocolContext.pause 'opentrons.protocol_api.ProtocolContext.pause'). + +Deprecated since version 2\.12: The Python Protocol API supports no safe way for a protocol to resume itself. +If you’re looking for a way for your protocol to resume automatically +after a period of time, use [`delay()`](#opentrons.protocol_api.ProtocolContext.delay 'opentrons.protocol_api.ProtocolContext.delay'). + +New in version 2\.0\. + +set*rail_lights(\_self*, _on: 'bool'_) → 'None' +Controls the robot’s ambient lighting (rail lights). + +Parameters: +**on** ([_bool_](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')) – If `True`, turn on the lights; otherwise, turn them off. + +New in version 2\.5\. + +### Instruments + +_class_ opentrons.protocol*api.InstrumentContext(\_core: AbstractInstrument\[AbstractWellCore]*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _broker: LegacyBroker_, _api_version: APIVersion_, _tip_racks: [List](https://docs.python.org/3/library/typing.html#typing.List '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]_, _trash: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware'), [TrashBin](index.html#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.disposal_locations.TrashBin'), [WasteChute](index.html#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.disposal_locations.WasteChute')]]_, _requested_as: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')_) +A context for a specific pipette or instrument. + +The InstrumentContext class provides the objects, attributes, and methods that allow +you to use pipettes in your protocols. + +Methods generally fall into one of two categories. + +> - They can change the state of the InstrumentContext object, like how fast it +> moves liquid or where it disposes of used tips. +> - They can command the instrument to perform an action, like picking up tips, +> moving to certain locations, and aspirating or dispensing liquid. + +Objects in this class should not be instantiated directly. Instead, instances are +returned by [`ProtocolContext.load_instrument()`](#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument'). + +New in version 2\.0\. + +_property_ active_channels*: [int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')* +The number of channels the pipette will use to pick up tips. + +By default, all channels on the pipette. Use [`configure_nozzle_layout()`](#opentrons.protocol_api.InstrumentContext.configure_nozzle_layout 'opentrons.protocol_api.InstrumentContext.configure_nozzle_layout') +to set the pipette to use fewer channels. + +New in version 2\.16\. + +air*gap(\_self*, _volume: 'Optional\[float]' \= None_, _height: 'Optional\[float]' \= None_) → 'InstrumentContext' +Draw air into the pipette’s tip at the current well. + +See [Air Gap](index.html#air-gap). + +Parameters: + +- **volume** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The amount of air, measured in µL. Calling `air_gap()` with no + arguments uses the entire remaining volume in the pipette. +- **height** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The height, in mm, to move above the current well before creating + the air gap. The default is 5 mm above the current well. + +Raises: +`UnexpectedTipRemovalError` – If no tip is attached to the pipette. + +Raises: +[**RuntimeError**](https://docs.python.org/3/library/exceptions.html#RuntimeError '(in Python v3.12)') – If location cache is `None`. This should happen if +`air_gap()` is called without first calling a method +that takes a location (e.g., [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'), +[`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense')) + +Returns: +This instance. + +Note + +Both `volume` and `height` are optional, but if you want to specify only +`height` you must do it as a keyword argument: +`pipette.air_gap(height=2)`. If you call `air_gap` with a single, +unnamed argument, it will always be interpreted as a volume. + +New in version 2\.0\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +aspirate(_self_, _volume: 'Optional\[float]' \= None_, _location: 'Optional\[Union\[types.Location, labware.Well]]' \= None_, _rate: 'float' \= 1\.0_) → 'InstrumentContext' +Draw liquid into a pipette tip. + +See [Aspirate](index.html#new-aspirate) for more details and examples. + +Parameters: + +- **volume** ([_int_](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)') _or_ [_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The volume to aspirate, measured in µL. If unspecified, + defaults to the maximum volume for the pipette and its currently + attached tip. + +If `aspirate` is called with a volume of precisely 0, its behavior +depends on the API level of the protocol. On API levels below 2\.16, +it will behave the same as a volume of `None`/unspecified: aspirate +until the pipette is full. On API levels at or above 2\.16, no liquid +will be aspirated. + +- **location** – Tells the robot where to aspirate from. The location can be + a [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or a [`Location`](#opentrons.types.Location 'opentrons.types.Location'). + +> - If the location is a `Well`, the robot will aspirate at +> or above the bottom center of the well. The distance (in mm) +> from the well bottom is specified by +> [`well_bottom_clearance.aspirate`](#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance'). +> - If the location is a `Location` (e.g., the result of +> [`Well.top()`](#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') or [`Well.bottom()`](#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom')), the robot +> will aspirate from that specified position. +> - If the `location` is unspecified, the robot will +> aspirate from its current position. + +- **rate** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – A multiplier for the default flow rate of the pipette. Calculated + as `rate` multiplied by [`flow_rate.aspirate`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate'). If not specified, defaults to 1\.0\. See + [Pipette Flow Rates](index.html#new-plunger-flow-rates). + +Returns: +This instance. + +Note + +If `aspirate` is called with a single, unnamed argument, it will treat +that argument as `volume`. If you want to call `aspirate` with only +`location`, specify it as a keyword argument: +`pipette.aspirate(location=plate['A1'])` + +New in version 2\.0\. + +blow*out(\_self*, _location: 'Optional\[Union\[types.Location, labware.Well, TrashBin, WasteChute]]' \= None_) → 'InstrumentContext' +Blow an extra amount of air through a pipette’s tip to clear it. + +If [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') is used to empty a pipette, usually a small amount of +liquid remains in the tip. During a blowout, the pipette moves the plunger +beyond its normal limits to help remove all liquid from the pipette tip. See +[Blow Out](index.html#blow-out). + +Parameters: +**location** ([`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or [`Location`](#opentrons.types.Location 'opentrons.types.Location') or `None`) – The blowout location. If no location is specified, the pipette +will blow out from its current position. + +Changed in version 2\.16: Accepts `TrashBin` and `WasteChute` values. + +Raises: +[**RuntimeError**](https://docs.python.org/3/library/exceptions.html#RuntimeError '(in Python v3.12)') – If no location is specified and the location cache is +`None`. This should happen if `blow_out()` is called +without first calling a method that takes a location, like +[`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') or [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'). + +Returns: +This instance. + +New in version 2\.0\. + +_property_ channels*: [int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')* +The number of channels on the pipette. + +Possible values are 1, 8, or 96\. + +See also [`type`](#opentrons.protocol_api.InstrumentContext.type 'opentrons.protocol_api.InstrumentContext.type'). + +New in version 2\.0\. + +configure*for_volume(\_self*, _volume: 'float'_) → 'None' +Configure a pipette to handle a specific volume of liquid, measured in µL. +The pipette enters a volume mode depending on the volume provided. Changing +pipette modes alters properties of the instance of +[`InstrumentContext`](#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.InstrumentContext'), such as default flow rate, minimum volume, and +maximum volume. The pipette remains in the mode set by this function until it is +called again. + +The Flex 1\-Channel 50 µL and Flex 8\-Channel 50 µL pipettes must operate in a +low\-volume mode to accurately dispense very small volumes of liquid. Low\-volume +mode can only be set by calling `configure_for_volume()`. See +[Volume Modes](index.html#pipette-volume-modes). + +Note + +Changing a pipette’s mode will reset its [flow rates](index.html#new-plunger-flow-rates). + +This function will raise an error if called when the pipette’s tip contains +liquid. It won’t raise an error if a tip is not attached, but changing modes may +affect which tips the pipette can subsequently pick up without raising an error. + +This function will also raise an error if `volume` is outside of the +[minimum and maximum capacities](index.html#new-pipette-models) of the pipette (e.g., +setting `volume=1` for a Flex 1000 µL pipette). + +Parameters: +**volume** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The volume, in µL, that the pipette will prepare to handle. + +New in version 2\.15\. + +configure*nozzle_layout(\_self*, _style: 'NozzleLayout'_, _start: 'Optional\[str]' \= None_, _end: 'Optional\[str]' \= None_, _front_right: 'Optional\[str]' \= None_, _back_left: 'Optional\[str]' \= None_, _tip_racks: 'Optional\[List\[labware.Labware]]' \= None_) → 'None' +Configure how many tips the 8\-channel or 96\-channel pipette will pick up. + +Changing the nozzle layout will affect gantry movement for all subsequent +pipetting actions that the pipette performs. It also alters the pipette’s +behavior for picking up tips. The pipette will continue to use the specified +layout until this function is called again. + +Note + +When picking up fewer than 96 tips at once, the tip rack _must not_ be +placed in a tip rack adapter in the deck. If you try to pick up fewer than 96 +tips from a tip rack that is in an adapter, the API will raise an error. + +Parameters: + +- **style** (`NozzleLayout` or `None`) – The shape of the nozzle layout. + + - `SINGLE` sets the pipette to use 1 nozzle. This corresponds to a single of well on labware. + - `COLUMN` sets the pipette to use 8 nozzles, aligned from front to back + with respect to the deck. This corresponds to a column of wells on labware. + - `PARTIAL_COLUMN` sets the pipette to use 2\-7 nozzles, aligned from front to back + with respect to the deck. + - `ROW` sets the pipette to use 12 nozzles, aligned from left to right + with respect to the deck. This corresponds to a row of wells on labware. + - `ALL` resets the pipette to use all of its nozzles. Calling + `configure_nozzle_layout` with no arguments also resets the pipette. + +- **start** (str or `None`) – The primary nozzle of the layout, which the robot uses + to determine how it will move to different locations on the deck. The string + should be of the same format used when identifying wells by name. + Required unless setting `style=ALL`. + +Note + +If possible, don’t use both `start="A1"` and `start="A12"` to pick up +tips _from the same rack_. Doing so can affect positional accuracy. + +- **end** (str or `None`) – The nozzle at the end of a linear layout, which is used + to determine how many tips will be picked up by a pipette. The string + should be of the same format used when identifying wells by name. + Required when setting `style=PARTIAL_COLUMN`. + +Note + +Nozzle layouts numbering between 2\-7 nozzles, account for the distance from +`start`. For example, 4 nozzles would require `start="H1"` and `end="E1"`. + +- **tip_racks** (List\[[`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware')]) – Behaves the same as setting the `tip_racks` parameter of + [`load_instrument()`](#opentrons.protocol_api.ProtocolContext.load_instrument 'opentrons.protocol_api.ProtocolContext.load_instrument'). If not specified, the new configuration resets + [`InstrumentContext.tip_racks`](#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks') and you must specify the location + every time you call [`pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'). + +New in version 2\.16\. + +consolidate(_self_, _volume: 'Union\[float_, _Sequence\[float]]'_, _source: 'List\[labware.Well]'_, _dest: 'labware.Well'_, _\\\*args: 'Any'_, _\\\*\\\*kwargs: 'Any'_) → 'InstrumentContext' +Move liquid from multiple source wells to a single destination well. + +Parameters: + +- **volume** – The amount, in µL, to aspirate from each source well. +- **source** – A list of wells to aspirate liquid from. +- **dest** – A single well to dispense liquid into. +- **kwargs** – See [`transfer()`](#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') and the [Complex Liquid Handling Parameters](index.html#complex-params) page. + Some parameters behave differently than when transferring. + `disposal_volume` and `mix_before` are ignored. + +Returns: +This instance. + +New in version 2\.0\. + +_property_ current_volume*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The current amount of liquid held in the pipette, measured in µL. + +New in version 2\.0\. + +_property_ default_speed*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The speed at which the robot’s gantry moves in mm/s. + +The default speed for Flex varies between 300 and 350 mm/s. The OT\-2 default is +400 mm/s. In addition to changing the default, the speed of individual motions +can be changed with the `speed` argument of the +[`InstrumentContext.move_to()`](#opentrons.protocol_api.InstrumentContext.move_to 'opentrons.protocol_api.InstrumentContext.move_to') method. See [Gantry Speed](index.html#gantry-speed). + +New in version 2\.0\. + +detect*liquid_presence(\_self*, _well: 'labware.Well'_) → 'bool' +Check if there is liquid in a well. + +Returns: +A boolean. + +New in version 2\.20\. + +dispense(_self_, _volume: 'Optional\[float]' \= None_, _location: 'Optional\[Union\[types.Location, labware.Well, TrashBin, WasteChute]]' \= None_, _rate: 'float' \= 1\.0_, _push_out: 'Optional\[float]' \= None_) → 'InstrumentContext' +Dispense liquid from a pipette tip. + +See [Dispense](index.html#new-dispense) for more details and examples. + +Parameters: + +- **volume** ([_int_](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)') _or_ [_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The volume to dispense, measured in µL. + + - If unspecified or `None`, dispense the [`current_volume`](#opentrons.protocol_api.InstrumentContext.current_volume 'opentrons.protocol_api.InstrumentContext.current_volume'). + - If 0, the behavior of `dispense()` depends on the API level + of the protocol. In API version 2\.16 and earlier, dispense all + liquid in the pipette (same as unspecified or `None`). In API + version 2\.17 and later, dispense no liquid. + - If greater than [`current_volume`](#opentrons.protocol_api.InstrumentContext.current_volume 'opentrons.protocol_api.InstrumentContext.current_volume'), the behavior of + `dispense()` depends on the API level of the protocol. In API + version 2\.16 and earlier, dispense all liquid in the pipette. + In API version 2\.17 and later, raise an error. + +- **location** – Tells the robot where to dispense liquid held in the pipette. + The location can be a [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well'), [`Location`](#opentrons.types.Location 'opentrons.types.Location'), + [`TrashBin`](#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin'), or [`WasteChute`](#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.WasteChute'). + +> - If a `Well`, the pipette will dispense +> at or above the bottom center of the well. The distance (in +> mm) from the well bottom is specified by +> [`well_bottom_clearance.dispense`](#opentrons.protocol_api.InstrumentContext.well_bottom_clearance 'opentrons.protocol_api.InstrumentContext.well_bottom_clearance'). + If a `Location` (e.g., the result of +> [`Well.top()`](#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') or [`Well.bottom()`](#opentrons.protocol_api.Well.bottom 'opentrons.protocol_api.Well.bottom')), the pipette +> will dispense at that specified position. + If a trash container, the pipette will dispense at a location +> relative to its center and the trash container’s top center. +> See [Position Relative to Trash Containers](index.html#position-relative-trash) for details. + If unspecified, the pipette will +> dispense at its current position. +> If only a `location` is passed (e.g., +> `pipette.dispense(location=plate['A1'])`), all of the +> liquid aspirated into the pipette will be dispensed (the +> amount is accessible through [`current_volume`](#opentrons.protocol_api.InstrumentContext.current_volume 'opentrons.protocol_api.InstrumentContext.current_volume')). + +Changed in version 2\.16: Accepts `TrashBin` and `WasteChute` values. + +- **rate** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – How quickly a pipette dispenses liquid. The speed in µL/s is + calculated as `rate` multiplied by [`flow_rate.dispense`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate'). If not specified, defaults to 1\.0\. See + [Pipette Flow Rates](index.html#new-plunger-flow-rates). +- **push_out** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – Continue past the plunger bottom to help ensure all liquid + leaves the tip. Measured in µL. The default value is `None`. + +See [Push Out After Dispense](index.html#push-out-dispense) for details. + +Returns: +This instance. + +Note + +If `dispense` is called with a single, unnamed argument, it will treat +that argument as `volume`. If you want to call `dispense` with only +`location`, specify it as a keyword argument: +`pipette.dispense(location=plate['A1'])`. + +Changed in version 2\.15: Added the `push_out` parameter. + +Changed in version 2\.17: Behavior of the `volume` parameter. + +New in version 2\.0\. + +distribute(_self_, _volume: 'Union\[float_, _Sequence\[float]]'_, _source: 'labware.Well'_, _dest: 'List\[labware.Well]'_, _\\\*args: 'Any'_, _\\\*\\\*kwargs: 'Any'_) → 'InstrumentContext' +Move a volume of liquid from one source to multiple destinations. + +Parameters: + +- **volume** – The amount, in µL, to dispense into each destination well. +- **source** – A single well to aspirate liquid from. +- **dest** – A list of wells to dispense liquid into. +- **kwargs** – See [`transfer()`](#opentrons.protocol_api.InstrumentContext.transfer 'opentrons.protocol_api.InstrumentContext.transfer') and the [Complex Liquid Handling Parameters](index.html#complex-params) page. + Some parameters behave differently than when transferring. + +> - `disposal_volume` aspirates additional liquid to improve the accuracy +> of each dispense. Defaults to the minimum volume of the pipette. See +> [Disposal Volume](index.html#param-disposal-volume) for details. +> - `mix_after` is ignored. + +Returns: +This instance. + +New in version 2\.0\. + +drop*tip(\_self*, _location: 'Optional\[Union\[types.Location, labware.Well, TrashBin, WasteChute]]' \= None_, _home_after: 'Optional\[bool]' \= None_) → 'InstrumentContext' +Drop the current tip. + +See [Dropping a Tip](index.html#pipette-drop-tip) for examples. + +If no location is passed (e.g. `pipette.drop_tip()`), the pipette will drop +the attached tip into its [`trash_container`](#opentrons.protocol_api.InstrumentContext.trash_container 'opentrons.protocol_api.InstrumentContext.trash_container'). + +The location in which to drop the tip can be manually specified with the +`location` argument. The `location` argument can be specified in several +ways: + +> - As a [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well'). This uses a default location relative to the well. +> This style of call can be used to make the robot drop a tip into labware +> like a well plate or a reservoir. For example, +> `pipette.drop_tip(location=reservoir["A1"])`. +> - As a [`Location`](#opentrons.types.Location 'opentrons.types.Location'). For example, to drop a tip from an +> unusually large height above the tip rack, you could call +> `pipette.drop_tip(tip_rack["A1"].top(z=10))`. +> - As a [`TrashBin`](#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin'). This uses a default location relative to the +> `TrashBin` object. For example, +> `pipette.drop_tip(location=trash_bin)`. +> - As a [`WasteChute`](#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.WasteChute'). This uses a default location relative to +> the `WasteChute` object. For example, +> `pipette.drop_tip(location=waste_chute)`. + +In API versions 2\.15 to 2\.17, if `location` is a `TrashBin` or not +specified, the API will instruct the pipette to drop tips in different locations +within the bin. Varying the tip drop location helps prevent tips +from piling up in a single location. + +Starting with API version 2\.18, the API will only vary the tip drop location if +`location` is not specified. Specifying a `TrashBin` as the `location` +behaves the same as specifying [`TrashBin.top()`](#opentrons.protocol_api.TrashBin.top 'opentrons.protocol_api.TrashBin.top'), which is a fixed position. + +Parameters: + +- **location** ([`Location`](#opentrons.types.Location 'opentrons.types.Location') or [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or `None`) – Where to drop the tip. + +Changed in version 2\.16: Accepts `TrashBin` and `WasteChute` values. + +- **home_after** – Whether to home the pipette’s plunger after dropping the tip. If not + specified, defaults to `True` on an OT\-2\. + +When `False`, the pipette does not home its plunger. This can save a few +seconds, but is not recommended. Homing helps the robot track the pipette’s +position. + +Returns: +This instance. + +New in version 2\.0\. + +_property_ flow_rate*: FlowRates* +The speeds, in µL/s, configured for the pipette. + +See [Pipette Flow Rates](index.html#new-plunger-flow-rates). + +This is an object with attributes `aspirate`, `dispense`, and `blow_out` +holding the flow rate for the corresponding operation. + +Note + +Setting values of [`speed`](#opentrons.protocol_api.InstrumentContext.speed 'opentrons.protocol_api.InstrumentContext.speed'), which is deprecated, will override the +values in [`flow_rate`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate'). + +New in version 2\.0\. + +_property_ has_tip*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Whether this instrument has a tip attached or not. + +The value of this property is determined logically by the API, not by detecting +the physical presence of a tip. This is the case even on Flex, which has sensors +to detect tip attachment. + +New in version 2\.7\. + +home(_self_) → 'InstrumentContext' +Home the robot. + +See [Homing](index.html#utility-homing). + +Returns: +This instance. + +New in version 2\.0\. + +home*plunger(\_self*) → 'InstrumentContext' +Home the plunger associated with this mount. + +Returns: +This instance. + +New in version 2\.0\. + +_property_ hw_pipette*: PipetteDict* +View the information returned by the hardware API directly. + +Raises: +[`types.PipetteNotAttachedError`](#opentrons.types.PipetteNotAttachedError 'opentrons.types.PipetteNotAttachedError') if the pipette is +no longer attached (should not happen). + +New in version 2\.0\. + +_property_ liquid_presence_detection*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Gets the global setting for liquid level detection. + +When True, liquid_probe will be called before +aspirates and dispenses to bring the tip to the liquid level. + +The default value is False. + +New in version 2\.20\. + +_property_ max_volume*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The maximum volume, in µL, that the pipette can hold. + +The maximum volume that you can actually aspirate might be lower than this, +depending on what kind of tip is attached to this pipette. For example, a P300 +Single\-Channel pipette always has a `max_volume` of 300 µL, but if it’s using +a 200 µL filter tip, its usable volume would be limited to 200 µL. + +New in version 2\.0\. + +_property_ min_volume*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The minimum volume, in µL, that the pipette can hold. This value may change +based on the [volume mode](index.html#pipette-volume-modes) that the pipette is +currently configured for. + +New in version 2\.0\. + +mix(_self_, _repetitions: 'int' \= 1_, _volume: 'Optional\[float]' \= None_, _location: 'Optional\[Union\[types.Location, labware.Well]]' \= None_, _rate: 'float' \= 1\.0_) → 'InstrumentContext' +Mix a volume of liquid by repeatedly aspirating and dispensing it in a single location. + +See [Mix](index.html#mix) for examples. + +Parameters: + +- **repetitions** – Number of times to mix (default is 1\). +- **volume** – The volume to mix, measured in µL. If unspecified, defaults + to the maximum volume for the pipette and its attached tip. + +If `mix` is called with a volume of precisely 0, its behavior +depends on the API level of the protocol. On API levels below 2\.16, +it will behave the same as a volume of `None`/unspecified: mix +the full working volume of the pipette. On API levels at or above 2\.16, +no liquid will be mixed. + +- **location** – The [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or [`Location`](#opentrons.types.Location 'opentrons.types.Location') where the + pipette will mix. If unspecified, the pipette will mix at its + current position. +- **rate** – How quickly the pipette aspirates and dispenses liquid while + mixing. The aspiration flow rate is calculated as `rate` + multiplied by [`flow_rate.aspirate`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate'). The + dispensing flow rate is calculated as `rate` multiplied by + [`flow_rate.dispense`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate'). See + [Pipette Flow Rates](index.html#new-plunger-flow-rates). + +Raises: +`UnexpectedTipRemovalError` – If no tip is attached to the pipette. + +Returns: +This instance. + +Note + +All the arguments of `mix` are optional. However, if you omit one of them, +all subsequent arguments must be passed as keyword arguments. For instance, +`pipette.mix(1, location=wellplate['A1'])` is a valid call, but +`pipette.mix(1, wellplate['A1'])` is not. + +New in version 2\.0\. + +_property_ model*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The model string for the pipette (e.g., `'p300_single_v1.3'`) + +New in version 2\.0\. + +_property_ mount*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +Return the name of the mount the pipette is attached to. + +The possible names are `"left"` and `"right"`. + +New in version 2\.0\. + +move*to(\_self*, _location: 'Union\[types.Location, TrashBin, WasteChute]'_, _force_direct: 'bool' \= False_, _minimum_z_height: 'Optional\[float]' \= None_, _speed: 'Optional\[float]' \= None_, _publish: 'bool' \= True_) → 'InstrumentContext' +Move the instrument. + +See [Move To](index.html#move-to) for examples. + +Parameters: + +- **location** ([`Location`](#opentrons.types.Location 'opentrons.types.Location')) – Where to move to. + +Changed in version 2\.16: Accepts `TrashBin` and `WasteChute` values. + +- **force_direct** – If `True`, move directly to the destination without arc + motion. + +Warning + +Forcing direct motion can cause the pipette to crash +into labware, modules, or other objects on the deck. + +- **minimum_z_height** – An amount, measured in mm, to raise the mid\-arc height. + The mid\-arc height can’t be lowered. +- **speed** – The speed at which to move. By default, + [`InstrumentContext.default_speed`](#opentrons.protocol_api.InstrumentContext.default_speed 'opentrons.protocol_api.InstrumentContext.default_speed'). This controls the + straight linear speed of the motion. To limit individual axis + speeds, use [`ProtocolContext.max_speeds`](#opentrons.protocol_api.ProtocolContext.max_speeds 'opentrons.protocol_api.ProtocolContext.max_speeds'). +- **publish** – Whether to list this function call in the run preview. + Default is `True`. + +New in version 2\.0\. + +_property_ name*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name string for the pipette (e.g., `"p300_single"`). + +New in version 2\.0\. + +pick*up_tip(\_self*, _location: 'Union\[types.Location, labware.Well, labware.Labware, None]' \= None_, _presses: 'Optional\[int]' \= None_, _increment: 'Optional\[float]' \= None_, _prep_after: 'Optional\[bool]' \= None_) → 'InstrumentContext' +Pick up a tip for the pipette to run liquid\-handling commands. + +See [Picking Up a Tip](index.html#basic-tip-pickup). + +If no location is passed, the pipette will pick up the next available tip in its +[`tip_racks`](#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks') list. Within each tip rack, tips will +be picked up in the order specified by the labware definition and +[`Labware.wells()`](#opentrons.protocol_api.Labware.wells 'opentrons.protocol_api.Labware.wells'). To adjust where the sequence starts, use +[`starting_tip`](#opentrons.protocol_api.InstrumentContext.starting_tip 'opentrons.protocol_api.InstrumentContext.starting_tip'). + +The exact position for tip pickup accounts for the length of the tip and how +much the tip overlaps with the pipette nozzle. These measurements are fixed +values on Flex, and are based on the results of tip length calibration on OT\-2\. + +Note + +API version 2\.19 updates the tip overlap values for Flex. When updating a +protocol from 2\.18 (or lower) to 2\.19 (or higher), pipette performance +should improve without additional changes to your protocol. Nevertheless, it +is good practice after updating to do the following: + +- Run Labware Position Check. +- Perform a dry run of your protocol. +- If tip position is slightly higher than expected, adjust the `location` + parameter of pipetting actions to achieve the desired result. + +Parameters: + +- **location** ([`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') or [`types.Location`](#opentrons.types.Location 'opentrons.types.Location')) – The location from which to pick up a tip. The `location` + argument can be specified in several ways: + +> - As a [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well'). For example, +> `pipette.pick_up_tip(tiprack.wells()[0])` will always pick +> up the first tip in `tiprack`, even if the rack is not a +> member of [`InstrumentContext.tip_racks`](#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks'). +> - As a labware. `pipette.pick_up_tip(tiprack)` will pick up +> the next available tip in `tiprack`, even if the rack is +> not a member of [`InstrumentContext.tip_racks`](#opentrons.protocol_api.InstrumentContext.tip_racks 'opentrons.protocol_api.InstrumentContext.tip_racks'). +> - As a [`Location`](#opentrons.types.Location 'opentrons.types.Location'). Use this to make fine +> adjustments to the pickup location. For example, to tell +> the robot to start its pick up tip routine 1 mm closer to +> the top of the well in the tip rack, call +> `pipette.pick_up_tip(tiprack["A1"].top(z=-1))`. + +- **presses** ([_int_](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')) – The number of times to lower and then raise the pipette when + picking up a tip, to ensure a good seal. Zero (`0`) will + result in the pipette hovering over the tip but not picking it + up (generally not desirable, but could be used for a dry run). + +> Deprecated since version 2\.14: Use the Opentrons App to change pipette pick\-up settings. + +- **increment** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The additional distance to travel on each successive press. + For example, if `presses=3` and `increment=1.0`, then the + first press will travel down into the tip by 3\.5 mm, the + second by 4\.5 mm, and the third by 5\.5 mm). + +> Deprecated since version 2\.14: Use the Opentrons App to change pipette pick\-up settings. + +- **prep_after** ([_bool_](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')) – Whether the pipette plunger should prepare itself to aspirate + immediately after picking up a tip. + +If `True`, the pipette will move its plunger position to +bottom in preparation for any following calls to +[`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'). + +If `False`, the pipette will prepare its plunger later, +during the next call to [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'). This is +accomplished by moving the tip to the top of the well, and +positioning the plunger outside any potential liquids. + +Warning + +This is provided for compatibility with older Python +Protocol API behavior. You should normally leave this +unset. + +Setting `prep_after=False` may create an unintended +pipette movement, when the pipette automatically moves +the tip to the top of the well to prepare the plunger. + +Changed in version 2\.13: Adds the `prep_after` argument. In version 2\.12 and earlier, the plunger +can’t prepare itself for aspiration during [`pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'), and will +instead always prepare during [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'). Version 2\.12 and earlier +will raise an `APIVersionError` if a value is set for `prep_after`. + +Changed in version 2\.19: Uses new values for how much a tip overlaps with the pipette nozzle. + +Returns: +This instance. + +New in version 2\.0\. + +prepare*to_aspirate(\_self*) → 'None' +Prepare a pipette for aspiration. + +Before a pipette can aspirate into an empty tip, the plunger must be in its +bottom position. After dropping a tip or blowing out, the plunger will be in a +different position. This function moves the plunger to the bottom position, +regardless of its current position, to make sure that the pipette is ready to +aspirate. + +You rarely need to call this function. The API automatically prepares the +pipette for aspiration as part of other commands: + +> - After picking up a tip with [`pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'). +> - When calling [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate'), if the pipette isn’t already prepared. +> If the pipette is in a well, it will move out of the well, move the plunger, +> and then move back. + +Use `prepare_to_aspirate` when you need to control exactly when the plunger +motion will happen. A common use case is a pre\-wetting routine, which requires +preparing for aspiration, moving into a well, and then aspirating _without +leaving the well_: + +``` +pipette.move_to(well.bottom(z=2)) +pipette.delay(5) +pipette.mix(10, 10) +pipette.move_to(well.top(z=5)) +pipette.blow_out() +pipette.prepare_to_aspirate() +pipette.move_to(well.bottom(z=2)) +pipette.delay(5) +pipette.aspirate(10, well.bottom(z=2)) + +``` + +The call to `prepare_to_aspirate()` means that the plunger will be in the +bottom position before the call to `aspirate()`. Since it doesn’t need to +prepare again, it will not move up out of the well to move the plunger. It will +aspirate in place. + +New in version 2\.16\. + +require*liquid_presence(\_self*, _well: 'labware.Well'_) → 'None' +If there is no liquid in a well, raise an error. + +Returns: +None. + +New in version 2\.20\. + +reset*tipracks(\_self*) → 'None' +Reload all tips in each tip rack and reset the starting tip. + +New in version 2\.0\. + +_property_ return_height*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The height to return a tip to its tip rack. + +Returns: +A scaling factor to apply to the tip length. +During [`drop_tip()`](#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip'), this factor is multiplied by the tip +length to get the distance from the top of the well to drop the tip. + +New in version 2\.2\. + +return*tip(\_self*, _home_after: 'Optional\[bool]' \= None_) → 'InstrumentContext' +Drop the currently attached tip in its original location in the tip rack. + +Returning a tip does not reset tip tracking, so [`Well.has_tip`](#opentrons.protocol_api.Well.has_tip 'opentrons.protocol_api.Well.has_tip') will +remain `False` for the destination. + +Returns: +This instance. + +Parameters: +**home_after** – See the `home_after` parameter of [`drop_tip()`](#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip'). + +New in version 2\.0\. + +_property_ speed*: PlungerSpeeds* +The speeds (in mm/s) configured for the pipette plunger. + +This is an object with attributes `aspirate`, `dispense`, and `blow_out` +holding the plunger speeds for the corresponding operation. + +Note + +Setting values of [`flow_rate`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate') will override the values in +[`speed`](#opentrons.protocol_api.InstrumentContext.speed 'opentrons.protocol_api.InstrumentContext.speed'). + +Changed in version 2\.14: This property has been removed because it’s fundamentally misaligned with +the step\-wise nature of a pipette’s plunger speed configuration. Use +[`flow_rate`](#opentrons.protocol_api.InstrumentContext.flow_rate 'opentrons.protocol_api.InstrumentContext.flow_rate') instead. + +New in version 2\.0\. + +_property_ starting_tip*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Well](index.html#opentrons.protocol_api.Well 'opentrons.protocol_api.labware.Well')]* +Which well of a tip rack the pipette should start at when automatically choosing tips to pick up. + +See [`pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip'). + +Note + +In robot software versions 6\.3\.0 and 6\.3\.1, protocols specifying API level +2\.14 ignored `starting_tip` on the second and subsequent calls to +[`InstrumentContext.pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') with no argument. This is fixed +for all API levels as of robot software version 7\.0\.0\. + +New in version 2\.0\. + +_property_ tip_racks*: [List](https://docs.python.org/3/library/typing.html#typing.List '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The tip racks that have been linked to this pipette. + +This is the property used to determine which tips to pick up next when calling +[`pick_up_tip()`](#opentrons.protocol_api.InstrumentContext.pick_up_tip 'opentrons.protocol_api.InstrumentContext.pick_up_tip') without arguments. See [Picking Up a Tip](index.html#basic-tip-pickup). + +New in version 2\.0\. + +touch*tip(\_self*, _location: 'Optional\[labware.Well]' \= None_, _radius: 'float' \= 1\.0_, _v_offset: 'float' \= \- 1\.0_, _speed: 'float' \= 60\.0_) → 'InstrumentContext' +Touch the pipette tip to the sides of a well, with the intent of removing leftover droplets. + +See [Touch Tip](index.html#touch-tip) for more details and examples. + +Parameters: + +- **location** ([`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') or `None`) – If no location is passed, the pipette will touch its tip at the + edges of the current well. +- **radius** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – How far to move, as a proportion of the target well’s radius. + When `radius=1.0`, the pipette tip will move all the way to the + edge of the target well. When `radius=0.5`, it will move to 50% + of the well’s radius. Default is 1\.0 (100%) +- **v_offset** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – How far above or below the well to touch the tip, measured in mm. + A positive offset moves the tip higher above the well. + A negative offset moves the tip lower into the well. + Default is \-1\.0 mm. +- **speed** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The speed for touch tip motion, in mm/s. + + - Default: 60\.0 mm/s + - Maximum: 80\.0 mm/s + - Minimum: 1\.0 mm/s + +Raises: +`UnexpectedTipRemovalError` – If no tip is attached to the pipette. + +Raises: +[**RuntimeError**](https://docs.python.org/3/library/exceptions.html#RuntimeError '(in Python v3.12)') – If no location is specified and the location cache is +`None`. This should happen if `touch_tip` is called +without first calling a method that takes a location, like +[`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') or [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'). + +Returns: +This instance. + +New in version 2\.0\. + +transfer(_self_, _volume: 'Union\[float_, _Sequence\[float]]'_, _source: 'AdvancedLiquidHandling'_, _dest: 'AdvancedLiquidHandling'_, _trash: 'bool' \= True_, _\\\*\\\*kwargs: 'Any'_) → 'InstrumentContext' +Move liquid from one well or group of wells to another. + +Transfer is a higher\-level command, incorporating other +[`InstrumentContext`](#opentrons.protocol_api.InstrumentContext 'opentrons.protocol_api.InstrumentContext') commands, like [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') and +[`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'). It makes writing a protocol easier at the cost of +specificity. See [Complex Commands](index.html#v2-complex-commands) for details on how transfer and +other complex commands perform their component steps. + +Parameters: + +- **volume** – The amount, in µL, to aspirate from each source and dispense to + each destination. If `volume` is a list, each amount will be + used for the source and destination at the matching index. A list + item of `0` will skip the corresponding wells entirely. See + [List of Volumes](index.html#complex-list-volumes) for details and examples. +- **source** – A single well or a list of wells to aspirate liquid from. +- **dest** – A single well or a list of wells to dispense liquid into. + +Keyword Arguments: +Transfer accepts a number of optional parameters that give +you greater control over the exact steps it performs. See +[Complex Liquid Handling Parameters](index.html#complex-params) or the links under each argument’s entry below for +additional details and examples. + +- **new_tip** (_string_) – + When to pick up and drop tips during the command. Defaults to `"once"`. + +> - `"once"`: Use one tip for the entire command. +> - `"always"`: Use a new tip for each set of aspirate and dispense steps. +> - `"never"`: Do not pick up or drop tips at all. + +See [Tip Handling](index.html#param-tip-handling) for details. + +- **trash** (_boolean_) – + If `True` (default), the pipette will drop tips in its + [`trash_container()`](#opentrons.protocol_api.InstrumentContext.trash_container 'opentrons.protocol_api.InstrumentContext.trash_container'). + If `False`, the pipette will return tips to their tip rack. + +See [Trash Tips](index.html#param-trash) for details. + +- **touch_tip** (_boolean_) – + If `True`, perform a [`touch_tip()`](#opentrons.protocol_api.InstrumentContext.touch_tip 'opentrons.protocol_api.InstrumentContext.touch_tip') following each + [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') and [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'). Defaults to `False`. + +See [Touch Tip](index.html#param-touch-tip) for details. + +- **blow_out** (_boolean_) – + If `True`, a [`blow_out()`](#opentrons.protocol_api.InstrumentContext.blow_out 'opentrons.protocol_api.InstrumentContext.blow_out') will occur following each + [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense'), but only if the pipette has no liquid left + in it. If `False` (default), the pipette will not blow out liquid. + +See [Blow Out](index.html#param-blow-out) for details. + +- **blowout_location** (_string_) – + Accepts one of three string values: `"trash"`, `"source well"`, or + `"destination well"`. + +If `blow_out` is `False` (its default), this parameter is ignored. + +If `blow_out` is `True` and this parameter is not set: + +> - Blow out into the trash, if the pipette is empty or only contains the +> disposal volume. +> - Blow out into the source well, if the pipette otherwise contains liquid. + +- **mix_before** (_tuple_) – + Perform a [`mix()`](#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix') before each [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') during the + transfer. The first value of the tuple is the number of repetitions, and + the second value is the amount of liquid to mix in µL. + +See [Mix Before](index.html#param-mix-before) for details. + +- **mix_after** (_tuple_) – + Perform a [`mix()`](#opentrons.protocol_api.InstrumentContext.mix 'opentrons.protocol_api.InstrumentContext.mix') after each [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') during the + transfer. The first value of the tuple is the number of repetitions, and + the second value is the amount of liquid to mix in µL. + +See [Mix After](index.html#param-mix-after) for details. + +- **disposal_volume** (_float_) – + Transfer ignores the numeric value of this parameter. If set, the pipette + will not aspirate additional liquid, but it will perform a very small blow + out after each dispense. + +See [Disposal Volume](index.html#param-disposal-volume) for details. + +Returns: +This instance. + +New in version 2\.0\. + +_property_ trash_container*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware'), [TrashBin](index.html#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.disposal_locations.TrashBin'), [WasteChute](index.html#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.disposal_locations.WasteChute')]* +The trash container associated with this pipette. + +This is the property used to determine where to drop tips and blow out liquids +when calling [`drop_tip()`](#opentrons.protocol_api.InstrumentContext.drop_tip 'opentrons.protocol_api.InstrumentContext.drop_tip') or [`blow_out()`](#opentrons.protocol_api.InstrumentContext.blow_out 'opentrons.protocol_api.InstrumentContext.blow_out') without arguments. + +You can set this to a [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware'), [`TrashBin`](#opentrons.protocol_api.TrashBin 'opentrons.protocol_api.TrashBin'), or [`WasteChute`](#opentrons.protocol_api.WasteChute 'opentrons.protocol_api.WasteChute'). + +The default value depends on the robot type and API version: + +- [`ProtocolContext.fixed_trash`](#opentrons.protocol_api.ProtocolContext.fixed_trash 'opentrons.protocol_api.ProtocolContext.fixed_trash'), if it exists. +- Otherwise, the first item previously loaded with + [`ProtocolContext.load_trash_bin()`](#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin') or + [`ProtocolContext.load_waste_chute()`](#opentrons.protocol_api.ProtocolContext.load_waste_chute 'opentrons.protocol_api.ProtocolContext.load_waste_chute'). + +Changed in version 2\.16: Added support for `TrashBin` and `WasteChute` objects. + +New in version 2\.0\. + +_property_ type*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +`'single'` if this is a 1\-channel pipette, or `'multi'` otherwise. + +See also [`channels`](#opentrons.protocol_api.InstrumentContext.channels 'opentrons.protocol_api.InstrumentContext.channels'), which can distinguish between 8\-channel and 96\-channel +pipettes. + +New in version 2\.0\. + +_property_ well_bottom_clearance*: Clearances* +The distance above the bottom of a well to aspirate or dispense. + +This is an object with attributes `aspirate` and `dispense`, describing the +default height of the corresponding operation. The default is 1\.0 mm for both +aspirate and dispense. + +When [`aspirate()`](#opentrons.protocol_api.InstrumentContext.aspirate 'opentrons.protocol_api.InstrumentContext.aspirate') or [`dispense()`](#opentrons.protocol_api.InstrumentContext.dispense 'opentrons.protocol_api.InstrumentContext.dispense') is given a [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') +rather than a full [`Location`](#opentrons.types.Location 'opentrons.types.Location'), the robot will move this distance +above the bottom of the well to aspirate or dispense. + +To change, set the corresponding attribute: + +``` +pipette.well_bottom_clearance.aspirate = 2 + +``` + +New in version 2\.0\. + +### Labware + +_class_ opentrons.protocol*api.Labware(\_core: AbstractLabware\[Any]*, _api_version: APIVersion_, _protocol_core: ProtocolCore_, _core_map: LoadedCoreMap_) +This class represents a piece of labware. + +Labware available in the API generally fall under two categories. + +> - Consumable labware: well plates, tubes in racks, reservoirs, tip racks, etc. +> - Adapters: durable items that hold other labware, either on modules or directly +> on the deck. + +The `Labware` class defines the physical geometry of the labware +and provides methods for [accessing wells](index.html#new-well-access) within the labware. + +Create `Labware` objects by calling the appropriate `load_labware()` method, +depending on where you are loading the labware. For example, to load labware on a +Thermocycler Module, use [`ThermocyclerContext.load_labware()`](#opentrons.protocol_api.ThermocyclerContext.load_labware 'opentrons.protocol_api.ThermocyclerContext.load_labware'). To load +labware directly on the deck, use [`ProtocolContext.load_labware()`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware'). See +[Loading Labware](index.html#loading-labware). + +_property_ api_version*: APIVersion* +See [`ProtocolContext.api_version`](#opentrons.protocol_api.ProtocolContext.api_version 'opentrons.protocol_api.ProtocolContext.api_version'). + +New in version 2\.0\. + +_property_ calibrated_offset*: [Point](index.html#opentrons.types.Point 'opentrons.types.Point')* +The front\-left\-bottom corner of the labware, including its labware offset. + +When running a protocol in the Opentrons App or on the touchscreen, Labware +Position Check sets the labware offset. + +New in version 2\.0\. + +_property_ child*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this labware. + +New in version 2\.15\. + +columns(_self_, _\\\*args: 'Union\[int_, _str]'_) → 'List\[List\[Well]]' +Accessor function to navigate through a labware by column. + +Use indexing to access individual columns or wells contained in the nested list. +For example, access column 1 with `labware.columns()[0]`. +On a standard 96\-well plate, this will output a list of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') +objects containing A1 through H1\. + +Note + +Using args with this method is deprecated. Use indexing instead. + +If your code uses args, they can be either strings or integers, but not a +mix of the two. For example, `.columns(1, 4)` or `.columns("1", "4")` is +valid, but `.columns("1", 4)` is not. + +Returns: +A list of column lists. + +New in version 2\.0\. + +columns*by_index(\_self*) → 'Dict\[str, List\[Well]]' + +Deprecated since version 2\.0: Use [`columns_by_name()`](#opentrons.protocol_api.Labware.columns_by_name 'opentrons.protocol_api.Labware.columns_by_name') instead. + +New in version 2\.0\. + +columns*by_name(\_self*) → 'Dict\[str, List\[Well]]' +Accessor function to navigate through a labware by column name. + +Use indexing to access individual columns or wells contained in the dictionary. +For example, access column 1 with `labware.columns_by_name()["1"]`. +On a standard 96\-well plate, this will output a list of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') +objects containing A1 through H1\. + +Returns: +Dictionary of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') lists keyed by column name. + +New in version 2\.0\. + +_property_ highest_z*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The z\-coordinate of the highest single point anywhere on the labware. + +This is taken from the `zDimension` property of the `dimensions` object in the +labware definition and takes into account the labware offset. + +New in version 2\.0\. + +_property_ is_adapter*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Whether the labware behaves as an adapter. + +Returns `True` if the labware definition specifies `adapter` as one of the +labware’s `allowedRoles`. + +New in version 2\.15\. + +_property_ is_tiprack*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Whether the labware behaves as a tip rack. + +Returns `True` if the labware definition specifies `isTiprack` as `True`. + +New in version 2\.0\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load a compatible labware onto the labware using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a compatible labware onto the labware using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If specified, + this is how the labware will appear in the run log, Labware Position + Check, and elsewhere in the Opentrons App and on the touchscreen. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +_property_ load_name*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The API load name of the labware definition. + +New in version 2\.0\. + +_property_ magdeck_engage_height*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +Return the default magnet engage height that +[`MagneticModuleContext.engage()`](#opentrons.protocol_api.MagneticModuleContext.engage 'opentrons.protocol_api.MagneticModuleContext.engage') will use for this labware. + +Warning + +This currently returns confusing and unpredictable results that do not +necessarily match what [`MagneticModuleContext.engage()`](#opentrons.protocol_api.MagneticModuleContext.engage 'opentrons.protocol_api.MagneticModuleContext.engage') will +actually choose for its default height. + +The confusion is related to how this height’s units and origin point are +defined, and differences between Magnetic Module generations. + +For now, we recommend you avoid accessing this property directly. + +New in version 2\.0\. + +_property_ name*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The display name of the labware. + +If you specified a value for `label` when loading the labware, `name` is +that value. + +Otherwise, it is the [`load_name`](#opentrons.protocol_api.Labware.load_name 'opentrons.protocol_api.Labware.load_name') of the labware. + +New in version 2\.0\. + +_property_ parameters*: LabwareParameters* +Internal properties of a labware including type and quirks. + +New in version 2\.0\. + +_property_ parent*: Union\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), [Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware'), ModuleTypes, OffDeckType]* +Where the labware is loaded. + +This corresponds to the physical object that the labware _directly_ rests upon. + +Returns: +If the labware is directly on the robot’s deck, the `str` name of the deck slot, +like `"D1"` (Flex) or `"1"` (OT\-2\). See [Deck Slots](index.html#deck-slots). + +If the labware is on a module, a module context. + +If the labware is on a labware or adapter, a [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware'). + +If the labware is off\-deck, [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK'). + +Changed in version 2\.14: Return type for module parent changed. +Formerly, the API returned an internal geometry interface. + +Changed in version 2\.15: Returns a [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') if the labware is loaded onto a labware/adapter. +Returns [`OFF_DECK`](#opentrons.protocol_api.OFF_DECK 'opentrons.protocol_api.OFF_DECK') if the labware is off\-deck. +Formerly, if the labware was removed by using `del` on [`deck`](#opentrons.protocol_api.ProtocolContext.deck 'opentrons.protocol_api.ProtocolContext.deck'), +this would return where it was before its removal. + +New in version 2\.0\. + +_property_ quirks*: [List](https://docs.python.org/3/library/typing.html#typing.List '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]* +Quirks specific to this labware. + +New in version 2\.0\. + +reset(_self_) → 'None' +Reset tip tracking for a tip rack. + +After resetting, the API treats all wells on the rack as if they contain unused tips. +This is useful if you want to reuse tips after calling [`return_tip()`](#opentrons.protocol_api.InstrumentContext.return_tip 'opentrons.protocol_api.InstrumentContext.return_tip'). + +If you need to physically replace an empty tip rack in the middle of your protocol, +use [`move_labware()`](#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware') instead. See [The Off\-Deck Location](index.html#off-deck-location) for an example. + +Changed in version 2\.14: This method will raise an exception if you call it on a labware that isn’t +a tip rack. Formerly, it would do nothing. + +New in version 2\.0\. + +rows(_self_, _\\\*args: 'Union\[int_, _str]'_) → 'List\[List\[Well]]' +Accessor function to navigate through a labware by row. + +Use indexing to access individual rows or wells contained in the nested list. +On a standard 96\-well plate, this will output a list of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') +objects containing A1 through A12\. + +Note + +Using args with this method is deprecated. Use indexing instead. + +If your code uses args, they can be either strings or integers, but not a +mix of the two. For example, `.rows(1, 4)` or `.rows("1", "4")` is +valid, but `.rows("1", 4)` is not. + +Returns: +A list of row lists. + +New in version 2\.0\. + +rows*by_index(\_self*) → 'Dict\[str, List\[Well]]' + +Deprecated since version 2\.0: Use [`rows_by_name()`](#opentrons.protocol_api.Labware.rows_by_name 'opentrons.protocol_api.Labware.rows_by_name') instead. + +New in version 2\.0\. + +rows*by_name(\_self*) → 'Dict\[str, List\[Well]]' +Accessor function to navigate through a labware by row name. + +Use indexing to access individual rows or wells contained in the dictionary. +For example, access row A with `labware.rows_by_name()["A"]`. +On a standard 96\-well plate, this will output a list of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') +objects containing A1 through A12\. + +Returns: +Dictionary of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') lists keyed by row name. + +New in version 2\.0\. + +set*calibration(\_self*, _delta: 'Point'_) → 'None' +An internal, deprecated method used for updating the labware offset. + +Deprecated since version 2\.14\. + +set*offset(\_self*, _x: 'float'_, _y: 'float'_, _z: 'float'_) → 'None' +Set the labware’s position offset. + +The offset is an x, y, z vector in deck coordinates +(see [Position Relative to the Deck](index.html#protocol-api-deck-coords)). + +How the motion system applies the offset depends on the API level of the protocol. + +| API level | Offset behavior | +| --------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| 2\.12–2\.13 | Offsets only apply to the exact [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') instance. | +| 2\.14–2\.17 | `set_offset()` is not available, and the API raises an error. | +| 2\.18 and newer | _ Offsets apply to any labware of the same type, in the same on\-deck location. _ Offsets can’t be set on labware that is currently off\-deck. \* Offsets do not follow a labware instance when using [`move_labware()`](#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware'). | + +Note + +Setting offsets with this method will override any labware offsets set +by running Labware Position Check in the Opentrons App. + +This method is designed for use with mechanisms like +[`opentrons.execute.get_protocol_api`](#opentrons.execute.get_protocol_api 'opentrons.execute.get_protocol_api'), which lack an interactive way +to adjust labware offsets. (See [Advanced Control](index.html#advanced-control).) + +Changed in version 2\.14: Temporarily removed. + +Changed in version 2\.18: Restored, and now applies to labware type–location pairs. + +New in version 2\.12\. + +_property_ tip_length*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +For a tip rack labware, the length of the tips it holds, in mm. + +This is taken from the `tipLength` property of the `parameters` object in the labware definition. + +This method will raise an exception if you call it on a labware that isn’t a tip rack. + +New in version 2\.0\. + +_property_ uri*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +A string fully identifying the labware. + +The URI has three parts and follows the pattern `"namespace/load_name/version"`. +For example, `opentrons/corning_96_wellplate_360ul_flat/2`. + +New in version 2\.0\. + +well(_self_, _idx: 'Union\[int, str]'_) → 'Well' +Deprecated. Use result of [`wells()`](#opentrons.protocol_api.Labware.wells 'opentrons.protocol_api.Labware.wells') or [`wells_by_name()`](#opentrons.protocol_api.Labware.wells_by_name 'opentrons.protocol_api.Labware.wells_by_name'). + +New in version 2\.0\. + +wells(_self_, _\\\*args: 'Union\[str_, _int]'_) → 'List\[Well]' +Accessor function to navigate a labware top to bottom, left to right. + +i.e., this method returns a list ordered A1, B1, C1…A2, B2, C2…. + +Use indexing to access individual wells contained in the list. +For example, access well A1 with `labware.wells()[0]`. + +Note + +Using args with this method is deprecated. Use indexing instead. + +If your code uses args, they can be either strings or integers, but not a +mix of the two. For example, `.wells(1, 4)` or `.wells("1", "4")` is +valid, but `.wells("1", 4)` is not. + +Returns: +Ordered list of all wells in a labware. + +New in version 2\.0\. + +wells*by_index(\_self*) → 'Dict\[str, Well]' + +Deprecated since version 2\.0: Use [`wells_by_name()`](#opentrons.protocol_api.Labware.wells_by_name 'opentrons.protocol_api.Labware.wells_by_name') or dict access instead. + +New in version 2\.0\. + +wells*by_name(\_self*) → 'Dict\[str, Well]' +Accessor function used to navigate through a labware by well name. + +Use indexing to access individual wells contained in the dictionary. +For example, access well A1 with `labware.wells_by_name()["A1"]`. + +Returns: +Dictionary of [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') objects keyed by well name. + +New in version 2\.0\. + +_class_ opentrons.protocol_api.TrashBin +Represents a Flex or OT\-2 trash bin. + +See [`ProtocolContext.load_trash_bin()`](#opentrons.protocol_api.ProtocolContext.load_trash_bin 'opentrons.protocol_api.ProtocolContext.load_trash_bin'). + +top(_self_, _x: 'float' \= 0_, _y: 'float' \= 0_, _z: 'float' \= 0_) → 'TrashBin' +Add a location offset to a trash bin. + +The default location (`x`, `y`, and `z` all set to `0`) is the center of +the bin on the x\- and y\-axes, and slightly below its physical top on the z\-axis. + +Offsets can be positive or negative and are measured in mm. +See [Position Relative to the Deck](index.html#protocol-api-deck-coords). + +New in version 2\.18\. + +_class_ opentrons.protocol_api.WasteChute +Represents a Flex waste chute. + +See [`ProtocolContext.load_waste_chute()`](#opentrons.protocol_api.ProtocolContext.load_waste_chute 'opentrons.protocol_api.ProtocolContext.load_waste_chute'). + +top(_self_, _x: 'float' \= 0_, _y: 'float' \= 0_, _z: 'float' \= 0_) → 'WasteChute' +Add a location offset to a waste chute. + +The default location (`x`, `y`, and `z` all set to `0`) is the center of +the chute’s opening on the x\- and y\-axes, and slightly below its physical top +on the z\-axis. See [Waste Chute](index.html#configure-waste-chute) for more information on possible +configurations of the chute. + +Offsets can be positive or negative and are measured in mm. +See [Position Relative to the Deck](index.html#protocol-api-deck-coords). + +New in version 2\.18\. + +### Wells and Liquids + +_class_ opentrons.protocol*api.Well(\_parent: [Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware')*, _core: WellCore_, _api_version: APIVersion_) +The Well class represents a single well in a [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware'). It provides parameters and functions for three major uses: + +> - Calculating positions relative to the well. See [Position Relative to Labware](index.html#position-relative-labware) for details. +> - Returning well measurements. See [Well Dimensions](index.html#new-labware-well-properties) for details. +> - Specifying what liquid should be in the well at the beginning of a protocol. See [Labeling Liquids in Wells](index.html#labeling-liquids) for details. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +bottom(_self_, _z: 'float' \= 0\.0_) → 'Location' + +Parameters: +**z** – An offset on the z\-axis, in mm. Positive offsets are higher and +negative offsets are lower. + +Returns: +A [`Location`](#opentrons.types.Location 'opentrons.types.Location') corresponding to the +absolute position of the bottom\-center of the well, plus the `z` offset +(if specified). + +New in version 2\.0\. + +center(_self_) → 'Location' + +Returns: +A [`Location`](#opentrons.types.Location 'opentrons.types.Location') corresponding to the +absolute position of the center of the well (in all three dimensions). + +New in version 2\.0\. + +_property_ depth*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The depth, in mm, of a well along the z\-axis, from the very top of the well to +the very bottom. + +New in version 2\.9\. + +_property_ diameter*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The diameter, in mm, of a circular well. Returns `None` +if the well is not circular. + +New in version 2\.0\. + +_property_ display_name*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +A human\-readable name for the well, including labware and deck location. + +For example, “A1 of Corning 96 Well Plate 360 µL Flat on slot D1”. Run log +entries use this format for identifying wells. See +[`ProtocolContext.commands()`](#opentrons.protocol_api.ProtocolContext.commands 'opentrons.protocol_api.ProtocolContext.commands'). + +from*center_cartesian(\_self*, _x: 'float'_, _y: 'float'_, _z: 'float'_) → 'Point' +Specifies a [`Point`](#opentrons.types.Point 'opentrons.types.Point') based on fractions of the +distance from the center of the well to the edge along each axis. + +For example, `from_center_cartesian(0, 0, 0.5)` specifies a point at the +well’s center on the x\- and y\-axis, and half of the distance from the center of +the well to its top along the z\-axis. To move the pipette to that location, +construct a [`Location`](#opentrons.types.Location 'opentrons.types.Location') relative to the same well: + +``` +location = types.Location( + plate["A1"].from_center_cartesian(0, 0, 0.5), plate["A1"] +) +pipette.move_to(location) + +``` + +See [Points and Locations](index.html#points-locations) for more information. + +Parameters: + +- **x** – The fraction of the distance from the well’s center to its edge + along the x\-axis. Negative values are to the left, and positive values + are to the right. +- **y** – The fraction of the distance from the well’s center to its edge + along the y\-axis. Negative values are to the front, and positive values + are to the back. +- **z** – The fraction of the distance from the well’s center to its edge + along the x\-axis. Negative values are down, and positive values are up. + +Returns: +A [`Point`](#opentrons.types.Point 'opentrons.types.Point') representing the specified +position in absolute deck coordinates. + +Note + +Even if the absolute values of `x`, `y`, and `z` are all less +than 1, a location constructed from the well and the result of +`from_center_cartesian` may be outside of the physical well. For example, +`from_center_cartesian(0.9, 0.9, 0)` would be outside of a cylindrical +well, but inside a square well. + +New in version 2\.8\. + +_property_ has_tip*: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)')* +Whether this well contains a tip. Always `False` if the parent labware +isn’t a tip rack. + +New in version 2\.0\. + +_property_ length*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The length, in mm, of a rectangular well along the x\-axis (left to right). +Returns `None` if the well is not rectangular. + +New in version 2\.9\. + +load*liquid(\_self*, _liquid: 'Liquid'_, _volume: 'float'_) → 'None' +Load a liquid into a well. + +Parameters: + +- **liquid** ([_Liquid_](index.html#opentrons.protocol_api.Liquid 'opentrons.protocol_api.Liquid')) – The liquid to load into the well. +- **volume** ([_float_](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')) – The volume of liquid to load, in µL. + +New in version 2\.14\. + +_property_ max_volume*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The maximum volume, in µL, that the well can hold. + +This amount is set by the JSON labware definition, specifically the `totalLiquidVolume` property of the particular well. + +_property_ parent*: [Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')* +The [`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') object that the well is a part of. + +New in version 2\.0\. + +top(_self_, _z: 'float' \= 0\.0_) → 'Location' + +Parameters: +**z** – An offset on the z\-axis, in mm. Positive offsets are higher and +negative offsets are lower. + +Returns: +A [`Location`](#opentrons.types.Location 'opentrons.types.Location') corresponding to the +absolute position of the top\-center of the well, plus the `z` offset +(if specified). + +New in version 2\.0\. + +_property_ well_name*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +A string representing the well’s coordinates. + +For example, “A1” or “H12”. + +The format of strings that this property returns is the same format as the key +for [accessing wells in a dictionary](index.html#well-dictionary-access). + +New in version 2\.7\. + +_property_ width*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The width, in mm, of a rectangular well along the y\-axis (front to back). +Returns `None` if the well is not rectangular. + +New in version 2\.9\. + +_class_ opentrons.protocol*api.Liquid(*\_id: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')_, \_name: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')_, _description: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]_, _display_color: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]_) +A liquid to load into a well. + +name +A human\-readable name for the liquid. + +Type: +[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)') + +description +An optional description. + +Type: +Optional\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')] + +display_color +An optional display color for the liquid. + +Type: +Optional\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')] + +New in version 2\.14\. + +### Modules + +_class_ opentrons.protocol*api.HeaterShakerContext(\_core: AbstractModuleCore*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _core_map: LoadedCoreMap_, _api_version: APIVersion_, _broker: LegacyBroker_) +An object representing a connected Heater\-Shaker Module. + +It should not be instantiated directly; instead, it should be +created through [`ProtocolContext.load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + +New in version 2\.13\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +close*labware_latch(\_self*) → 'None' +Closes the labware latch. + +The labware latch needs to be closed using this method before sending a shake command, +even if the latch was manually closed before starting the protocol. + +New in version 2\.13\. + +_property_ current_speed*: [int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')* +The current speed of the Heater\-Shaker’s plate in rpm. + +New in version 2\.13\. + +_property_ current_temperature*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The current temperature of the Heater\-Shaker’s plate in °C. + +Returns `23` in simulation if no target temperature has been set. + +New in version 2\.13\. + +deactivate*heater(\_self*) → 'None' +Stops heating. + +The module will passively cool to room temperature. +The Heater\-Shaker does not have active cooling. + +New in version 2\.13\. + +deactivate*shaker(\_self*) → 'None' +Stops shaking. + +Decelerating to 0 rpm typically only takes a few seconds. + +New in version 2\.13\. + +_property_ labware*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this module. + +New in version 2\.0\. + +_property_ labware_latch_status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +One of six possible latch statuses: + +- `opening` – The latch is currently opening (in motion). +- `idle_open` – The latch is open and not moving. +- `closing` – The latch is currently closing (in motion). +- `idle_closed` – The latch is closed and not moving. +- `idle_unknown` – The default status upon reset, regardless of physical latch position. + Use [`close_labware_latch()`](#opentrons.protocol_api.HeaterShakerContext.close_labware_latch 'opentrons.protocol_api.HeaterShakerContext.close_labware_latch') before other commands + requiring confirmation that the latch is closed. +- `unknown` – The latch status can’t be determined. + +New in version 2\.13\. + +load*adapter(\_self*, _name: 'str'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_adapter`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') (which loads adapters directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded adapter object. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _definition: 'LabwareDefinition'_) → 'Labware' +Load an adapter onto the module using an inline definition. + +Parameters: +**definition** – The labware definition. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.1: The _label,_ _namespace,_ and _version_ parameters. + +load*labware_by_name(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.HeaterShakerContext.load_labware 'opentrons.protocol_api.HeaterShakerContext.load_labware') instead. + +New in version 2\.1\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If + specified, this is the name the labware will appear + as in the run log and the calibration view in the + Opentrons app. + +Returns: +The initialized and loaded labware object. + +New in version 2\.0\. + +_property_ model*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleV1', 'magneticModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleV1', 'temperatureModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleV1', 'thermocyclerModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderV1']]* +Get the module’s model identifier. + +New in version 2\.14\. + +open*labware_latch(\_self*) → 'None' +Open the Heater\-Shaker’s labware latch. + +The labware latch needs to be closed before:\* Shaking + +- Pipetting to or from the labware on the Heater\-Shaker +- Pipetting to or from labware to the left or right of the Heater\-Shaker + +Attempting to open the latch while the Heater\-Shaker is shaking will raise an error. + +Note + +Before opening the latch, this command will retract the pipettes upward +if they are parked adjacent to the left or right of the Heater\-Shaker. + +New in version 2\.13\. + +_property_ parent*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name of the slot the module is on. + +On a Flex, this will be like `"D1"`. On an OT\-2, this will be like `"1"`. +See [Deck Slots](index.html#deck-slots). + +New in version 2\.14\. + +_property_ serial_number*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +Get the module’s unique hardware serial number. + +New in version 2\.14\. + +set*and_wait_for_shake_speed(\_self*, _rpm: 'int'_) → 'None' +Set a shake speed in rpm and block execution of further commands until the module reaches the target. + +Reaching a target shake speed typically only takes a few seconds. + +Note + +Before shaking, this command will retract the pipettes upward if they are parked adjacent to the Heater\-Shaker. + +Parameters: +**rpm** – A value between 200 and 3000, representing the target shake speed in revolutions per minute. + +New in version 2\.13\. + +set*and_wait_for_temperature(\_self*, _celsius: 'float'_) → 'None' +Set a target temperature and wait until the module reaches the target. + +No other protocol commands will execute while waiting for the temperature. + +Parameters: +**celsius** – A value between 27 and 95, representing the target temperature in °C. +Values are automatically truncated to two decimal places, +and the Heater\-Shaker module has a temperature accuracy of ±0\.5 °C. + +New in version 2\.13\. + +set*target_temperature(\_self*, _celsius: 'float'_) → 'None' +Set target temperature and return immediately. + +Sets the Heater\-Shaker’s target temperature and returns immediately without +waiting for the target to be reached. Does not delay the protocol until +target temperature has reached. +Use [`wait_for_temperature()`](#opentrons.protocol_api.HeaterShakerContext.wait_for_temperature 'opentrons.protocol_api.HeaterShakerContext.wait_for_temperature') to delay +protocol execution. + +Parameters: +**celsius** – A value between 27 and 95, representing the target temperature in °C. +Values are automatically truncated to two decimal places, +and the Heater\-Shaker module has a temperature accuracy of ±0\.5 °C. + +New in version 2\.13\. + +_property_ speed_status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +One of five possible shaking statuses: + +- `holding at target` – The module has reached its target shake speed + and is actively maintaining that speed. +- `speeding up` – The module is increasing its shake speed towards a target. +- `slowing down` – The module was previously shaking at a faster speed + and is currently reducing its speed to a lower target or to deactivate. +- `idle` – The module is not shaking. +- `error` – The shaking status can’t be determined. + +New in version 2\.13\. + +_property_ target_speed*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)')]* +Target speed of the Heater\-Shaker’s plate in rpm. + +New in version 2\.13\. + +_property_ target_temperature*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The target temperature of the Heater\-Shaker’s plate in °C. + +Returns `None` if no target has been set. + +New in version 2\.13\. + +_property_ temperature_status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +One of five possible temperature statuses: + +- `holding at target` – The module has reached its target temperature + and is actively maintaining that temperature. +- `cooling` – The module has previously heated and is now passively cooling. + The Heater\-Shaker does not have active cooling. +- `heating` – The module is heating to a target temperature. +- `idle` – The module has not heated since the beginning of the protocol. +- `error` – The temperature status can’t be determined. + +New in version 2\.13\. + +_property_ type*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderType']]* +Get the module’s general type identifier. + +New in version 2\.14\. + +wait*for_temperature(\_self*) → 'None' +Delays protocol execution until the Heater\-Shaker has reached its target +temperature. + +Raises an error if no target temperature was previously set. + +New in version 2\.13\. + +_class_ opentrons.protocol*api.MagneticBlockContext(\_core: AbstractModuleCore*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _core_map: LoadedCoreMap_, _api_version: APIVersion_, _broker: LegacyBroker_) +An object representing a Magnetic Block. + +It should not be instantiated directly; instead, it should be +created through [`ProtocolContext.load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + +New in version 2\.15\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +_property_ labware*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this module. + +New in version 2\.0\. + +load*adapter(\_self*, _name: 'str'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_adapter`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') (which loads adapters directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded adapter object. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _definition: 'LabwareDefinition'_) → 'Labware' +Load an adapter onto the module using an inline definition. + +Parameters: +**definition** – The labware definition. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.1: The _label,_ _namespace,_ and _version_ parameters. + +load*labware_by_name(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.MagneticBlockContext.load_labware 'opentrons.protocol_api.MagneticBlockContext.load_labware') instead. + +New in version 2\.1\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If + specified, this is the name the labware will appear + as in the run log and the calibration view in the + Opentrons app. + +Returns: +The initialized and loaded labware object. + +New in version 2\.0\. + +_property_ model*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleV1', 'magneticModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleV1', 'temperatureModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleV1', 'thermocyclerModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderV1']]* +Get the module’s model identifier. + +New in version 2\.14\. + +_property_ parent*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name of the slot the module is on. + +On a Flex, this will be like `"D1"`. On an OT\-2, this will be like `"1"`. +See [Deck Slots](index.html#deck-slots). + +New in version 2\.14\. + +_property_ type*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderType']]* +Get the module’s general type identifier. + +New in version 2\.14\. + +_class_ opentrons.protocol*api.MagneticModuleContext(\_core: AbstractModuleCore*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _core_map: LoadedCoreMap_, _api_version: APIVersion_, _broker: LegacyBroker_) +An object representing a connected Magnetic Module. + +It should not be instantiated directly; instead, it should be +created through [`ProtocolContext.load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + +New in version 2\.0\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +disengage(_self_) → 'None' +Lower the magnets back into the Magnetic Module. + +New in version 2\.0\. + +engage(_self_, _height: 'Optional\[float]' \= None_, _offset: 'Optional\[float]' \= None_, _height_from_base: 'Optional\[float]' \= None_) → 'None' +Raise the Magnetic Module’s magnets. You can specify how high the magnets +should move: + +> - No parameter: Move to the default height for the loaded labware. If +> the loaded labware has no default, or if no labware is loaded, this will +> raise an error. +> - `height_from_base` – Move this many millimeters above the bottom +> of the labware. Acceptable values are between `0` and `25`. +> +> This is the recommended way to adjust the magnets’ height. +> +> New in version 2\.2\. +> +> - `offset` – Move this many millimeters above (positive value) or below +> (negative value) the default height for the loaded labware. The sum of +> the default height and `offset` must be between 0 and 25\. +> - `height` – Intended to move this many millimeters above the magnets’ +> home position. However, depending on the generation of module and the loaded +> labware, this may produce unpredictable results. You should normally use +> `height_from_base` instead. +> +> Changed in version 2\.14: This parameter has been removed. + +You shouldn’t specify more than one of these parameters. However, if you do, +their order of precedence is `height`, then `height_from_base`, then `offset`. + +New in version 2\.0\. + +_property_ labware*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this module. + +New in version 2\.0\. + +load*adapter(\_self*, _name: 'str'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_adapter`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') (which loads adapters directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded adapter object. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _definition: 'LabwareDefinition'_) → 'Labware' +Load an adapter onto the module using an inline definition. + +Parameters: +**definition** – The labware definition. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.1: The _label,_ _namespace,_ and _version_ parameters. + +load*labware_by_name(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.MagneticModuleContext.load_labware 'opentrons.protocol_api.MagneticModuleContext.load_labware') instead. + +New in version 2\.1\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If + specified, this is the name the labware will appear + as in the run log and the calibration view in the + Opentrons app. + +Returns: +The initialized and loaded labware object. + +New in version 2\.0\. + +_property_ model*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleV1', 'magneticModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleV1', 'temperatureModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleV1', 'thermocyclerModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderV1']]* +Get the module’s model identifier. + +New in version 2\.14\. + +_property_ parent*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name of the slot the module is on. + +On a Flex, this will be like `"D1"`. On an OT\-2, this will be like `"1"`. +See [Deck Slots](index.html#deck-slots). + +New in version 2\.14\. + +_property_ serial_number*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +Get the module’s unique hardware serial number. + +New in version 2\.14\. + +_property_ status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The status of the module, either `engaged` or `disengaged`. + +New in version 2\.0\. + +_property_ type*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderType']]* +Get the module’s general type identifier. + +New in version 2\.14\. + +_class_ opentrons.protocol*api.TemperatureModuleContext(\_core: AbstractModuleCore*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _core_map: LoadedCoreMap_, _api_version: APIVersion_, _broker: LegacyBroker_) +An object representing a connected Temperature Module. + +It should not be instantiated directly; instead, it should be +created through [`ProtocolContext.load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + +New in version 2\.0\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +deactivate(_self_) → 'None' +Stop heating or cooling, and turn off the fan. + +New in version 2\.0\. + +_property_ labware*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this module. + +New in version 2\.0\. + +load*adapter(\_self*, _name: 'str'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_adapter`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') (which loads adapters directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded adapter object. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _definition: 'LabwareDefinition'_) → 'Labware' +Load an adapter onto the module using an inline definition. + +Parameters: +**definition** – The labware definition. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.1: The _label,_ _namespace,_ and _version_ parameters. + +load*labware_by_name(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.TemperatureModuleContext.load_labware 'opentrons.protocol_api.TemperatureModuleContext.load_labware') instead. + +New in version 2\.1\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If + specified, this is the name the labware will appear + as in the run log and the calibration view in the + Opentrons app. + +Returns: +The initialized and loaded labware object. + +New in version 2\.0\. + +_property_ model*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleV1', 'magneticModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleV1', 'temperatureModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleV1', 'thermocyclerModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderV1']]* +Get the module’s model identifier. + +New in version 2\.14\. + +_property_ parent*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name of the slot the module is on. + +On a Flex, this will be like `"D1"`. On an OT\-2, this will be like `"1"`. +See [Deck Slots](index.html#deck-slots). + +New in version 2\.14\. + +_property_ serial_number*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +Get the module’s unique hardware serial number. + +New in version 2\.14\. + +set*temperature(\_self*, _celsius: 'float'_) → 'None' +Set a target temperature and wait until the module reaches the target. + +No other protocol commands will execute while waiting for the temperature. + +Parameters: +**celsius** – A value between 4 and 95, representing the target temperature in °C. + +New in version 2\.0\. + +_property_ status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +One of four possible temperature statuses: + +- `holding at target` – The module has reached its target temperature + and is actively maintaining that temperature. +- `cooling` – The module is cooling to a target temperature. +- `heating` – The module is heating to a target temperature. +- `idle` – The module has been deactivated. + +New in version 2\.3\. + +_property_ target*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The target temperature of the Temperature Module’s deck in °C. + +Returns `None` if no target has been set. + +New in version 2\.0\. + +_property_ temperature*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +The current temperature of the Temperature Module’s deck in °C. + +Returns `0` in simulation if no target temperature has been set. + +New in version 2\.0\. + +_property_ type*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderType']]* +Get the module’s general type identifier. + +New in version 2\.14\. + +_class_ opentrons.protocol*api.ThermocyclerContext(\_core: AbstractModuleCore*, _protocol_core: AbstractProtocol\[AbstractInstrument\[AbstractWellCore], AbstractLabware\[AbstractWellCore], AbstractModuleCore]_, _core_map: LoadedCoreMap_, _api_version: APIVersion_, _broker: LegacyBroker_) +An object representing a connected Thermocycler Module. + +It should not be instantiated directly; instead, it should be +created through [`ProtocolContext.load_module()`](#opentrons.protocol_api.ProtocolContext.load_module 'opentrons.protocol_api.ProtocolContext.load_module'). + +New in version 2\.0\. + +_property_ api_version*: APIVersion* + +New in version 2\.0\. + +_property_ block_target_temperature*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The target temperature of the well block in °C. + +New in version 2\.0\. + +_property_ block_temperature*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The current temperature of the well block in °C. + +New in version 2\.0\. + +_property_ block_temperature_status*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +One of five possible temperature statuses: + +- `holding at target` – The block has reached its target temperature + and is actively maintaining that temperature. +- `cooling` – The block is cooling to a target temperature. +- `heating` – The block is heating to a target temperature. +- `idle` – The block is not currently heating or cooling. +- `error` – The temperature status can’t be determined. + +New in version 2\.0\. + +close*lid(\_self*) → 'str' +Close the lid. + +New in version 2\.0\. + +deactivate(_self_) → 'None' +Turn off both the well block temperature controller and the lid heater. + +New in version 2\.0\. + +deactivate*block(\_self*) → 'None' +Turn off the well block temperature controller. + +New in version 2\.0\. + +deactivate*lid(\_self*) → 'None' +Turn off the lid heater. + +New in version 2\.0\. + +execute*profile(\_self*, _steps: 'List\[ThermocyclerStep]'_, _repetitions: 'int'_, _block_max_volume: 'Optional\[float]' \= None_) → 'None' +Execute a Thermocycler profile, defined as a cycle of +`steps`, for a given number of `repetitions`. + +Parameters: + +- **steps** – List of unique steps that make up a single cycle. + Each list item should be a dictionary that maps to + the parameters of the [`set_block_temperature()`](#opentrons.protocol_api.ThermocyclerContext.set_block_temperature 'opentrons.protocol_api.ThermocyclerContext.set_block_temperature') + method with a `temperature` key, and either or both of + `hold_time_seconds` and `hold_time_minutes`. +- **repetitions** – The number of times to repeat the cycled steps. +- **block_max_volume** – The greatest volume of liquid contained in any + individual well of the loaded labware, in µL. + If not specified, the default is 25 µL. + +New in version 2\.0\. + +_property_ labware*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[Labware](index.html#opentrons.protocol_api.Labware 'opentrons.protocol_api.labware.Labware')]* +The labware (if any) present on this module. + +New in version 2\.0\. + +_property_ lid_position*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]* +One of these possible lid statuses: + +- `closed` – The lid is closed. +- `in_between` – The lid is neither open nor closed. +- `open` – The lid is open. +- `unknown` – The lid position can’t be determined. + +New in version 2\.0\. + +_property_ lid_target_temperature*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The target temperature of the lid in °C. + +New in version 2\.0\. + +_property_ lid_temperature*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')]* +The current temperature of the lid in °C. + +New in version 2\.0\. + +_property_ lid_temperature_status*: [Optional](https://docs.python.org/3/library/typing.html#typing.Optional '(in Python v3.12)')\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]* +One of five possible temperature statuses: + +- `holding at target` – The lid has reached its target temperature + and is actively maintaining that temperature. +- `cooling` – The lid has previously heated and is now passively cooling.The Thermocycler lid does not have active cooling. +- `heating` – The lid is heating to a target temperature. +- `idle` – The lid has not heated since the beginning of the protocol. +- `error` – The temperature status can’t be determined. + +New in version 2\.0\. + +load*adapter(\_self*, _name: 'str'_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' +Load an adapter onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_adapter`](#opentrons.protocol_api.ProtocolContext.load_adapter 'opentrons.protocol_api.ProtocolContext.load_adapter') (which loads adapters directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded adapter object. + +New in version 2\.15\. + +load*adapter_from_definition(\_self*, _definition: 'LabwareDefinition'_) → 'Labware' +Load an adapter onto the module using an inline definition. + +Parameters: +**definition** – The labware definition. + +Returns: +The initialized and loaded labware object. + +New in version 2\.15\. + +load*labware(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_, _adapter: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using its load parameters. + +The parameters of this function behave like those of +[`ProtocolContext.load_labware`](#opentrons.protocol_api.ProtocolContext.load_labware 'opentrons.protocol_api.ProtocolContext.load_labware') (which loads labware directly +onto the deck). Note that the parameter `name` here corresponds to +`load_name` on the `ProtocolContext` function. + +Returns: +The initialized and loaded labware object. + +New in version 2\.1: The _label,_ _namespace,_ and _version_ parameters. + +load*labware_by_name(\_self*, _name: 'str'_, _label: 'Optional\[str]' \= None_, _namespace: 'Optional\[str]' \= None_, _version: 'Optional\[int]' \= None_) → 'Labware' + +Deprecated since version 2\.0: Use [`load_labware()`](#opentrons.protocol_api.ThermocyclerContext.load_labware 'opentrons.protocol_api.ThermocyclerContext.load_labware') instead. + +New in version 2\.1\. + +load*labware_from_definition(\_self*, _definition: 'LabwareDefinition'_, _label: 'Optional\[str]' \= None_) → 'Labware' +Load a labware onto the module using an inline definition. + +Parameters: + +- **definition** – The labware definition. +- **label** ([_str_](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')) – An optional special name to give the labware. If + specified, this is the name the labware will appear + as in the run log and the calibration view in the + Opentrons app. + +Returns: +The initialized and loaded labware object. + +New in version 2\.0\. + +_property_ model*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleV1', 'magneticModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleV1', 'temperatureModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleV1', 'thermocyclerModuleV2'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockV1'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderV1']]* +Get the module’s model identifier. + +New in version 2\.14\. + +open*lid(\_self*) → 'str' +Open the lid. + +New in version 2\.0\. + +_property_ parent*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +The name of the slot the module is on. + +On a Flex, this will be like `"D1"`. On an OT\-2, this will be like `"1"`. +See [Deck Slots](index.html#deck-slots). + +New in version 2\.14\. + +_property_ serial_number*: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')* +Get the module’s unique hardware serial number. + +New in version 2\.14\. + +set*block_temperature(\_self*, _temperature: 'float'_, _hold_time_seconds: 'Optional\[float]' \= None_, _hold_time_minutes: 'Optional\[float]' \= None_, _ramp_rate: 'Optional\[float]' \= None_, _block_max_volume: 'Optional\[float]' \= None_) → 'None' +Set the target temperature for the well block, in °C. + +Parameters: + +- **temperature** – A value between 4 and 99, representing the target + temperature in °C. +- **hold_time_minutes** – The number of minutes to hold, after reaching + `temperature`, before proceeding to the + next command. If `hold_time_seconds` is also + specified, the times are added together. +- **hold_time_seconds** – The number of seconds to hold, after reaching + `temperature`, before proceeding to the + next command. If `hold_time_minutes` is also + specified, the times are added together. +- **block_max_volume** – The greatest volume of liquid contained in any + individual well of the loaded labware, in µL. + If not specified, the default is 25 µL. + +New in version 2\.0\. + +set*lid_temperature(\_self*, _temperature: 'float'_) → 'None' +Set the target temperature for the heated lid, in °C. + +Parameters: +**temperature** – A value between 37 and 110, representing the target +temperature in °C. + +New in version 2\.0\. + +_property_ type*: [Union](https://docs.python.org/3/library/typing.html#typing.Union '(in Python v3.12)')\[[Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['temperatureModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['thermocyclerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['heaterShakerModuleType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['magneticBlockType'], [Literal](https://docs.python.org/3/library/typing.html#typing.Literal '(in Python v3.12)')\['absorbanceReaderType']]* +Get the module’s general type identifier. + +New in version 2\.14\. + +### Useful Types + +_class_ opentrons.types.Location(_point: [Point](index.html#opentrons.types.Point 'opentrons.types.Point')_, _labware: Union\['Labware', 'Well', [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), 'ModuleGeometry', LabwareLike, [None](https://docs.python.org/3/library/constants.html#None '(in Python v3.12)'), 'ModuleContext']_) +A location to target as a motion. + +The location contains a [`Point`](#opentrons.types.Point 'opentrons.types.Point') (in +[Position Relative to the Deck](index.html#protocol-api-deck-coords)) and possibly an associated +[`Labware`](#opentrons.protocol_api.Labware 'opentrons.protocol_api.Labware') or [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') instance. + +It should rarely be constructed directly by the user; rather, it is the +return type of most [`Well`](#opentrons.protocol_api.Well 'opentrons.protocol_api.Well') accessors like [`Well.top()`](#opentrons.protocol_api.Well.top 'opentrons.protocol_api.Well.top') +and is passed directly into a method like `InstrumentContext.aspirate()`. + +Warning + +The `.labware` attribute of this class is used by the protocol +API internals to, among other things, determine safe heights to retract +the instruments to when moving between locations. If constructing an +instance of this class manually, be sure to either specify `None` as the +labware (so the robot does its worst case retraction) or specify the +correct labware for the `.point` attribute. + +Warning + +The `==` operation compares both the position and associated labware. +If you only need to compare locations, compare the `.point` +of each item. + +move(_self_, _point: 'Point'_) → "'Location'" +Alter the point stored in the location while preserving the labware. + +This returns a new Location and does not alter the current one. It +should be used like + +``` +>>> loc = Location(Point(1, 1, 1), None) +>>> new_loc = loc.move(Point(1, 1, 1)) +>>> +>>> # The new point is the old one plus the given offset. +>>> assert new_loc.point == Point(2, 2, 2) # True +>>> +>>> # The old point hasn't changed. +>>> assert loc.point == Point(1, 1, 1) # True + +``` + +_class_ opentrons.types.Mount(_value_) +An enumeration. + +_exception_ opentrons.types.PipetteNotAttachedError +An error raised if a pipette is accessed that is not attached + +_class_ opentrons.types.Point(_x_, _y_, _z_) + +x*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +Alias for field number 0 + +y*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +Alias for field number 1 + +z*: [float](https://docs.python.org/3/library/functions.html#float '(in Python v3.12)')* +Alias for field number 2 + +opentrons.protocol_api.OFF_DECK +A special location value, indicating that a labware is not currently on the robot’s deck. + +See [The Off\-Deck Location](index.html#off-deck-location) for details on using `OFF_DECK` with [`ProtocolContext.move_labware()`](#opentrons.protocol_api.ProtocolContext.move_labware 'opentrons.protocol_api.ProtocolContext.move_labware'). + +### Executing and Simulating Protocols + +opentrons.execute: functions and entrypoint for running protocols + +This module has functions that can be imported to provide protocol +contexts for running protocols during interactive sessions like Jupyter or just +regular python shells. It also provides a console entrypoint for running a +protocol from the command line. + +opentrons.execute.execute(_protocol_file: Union\[BinaryIO, TextIO]_, _protocol_name: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')_, _propagate_logs: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)') \= False_, _log_level: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)') \= 'warning'_, _emit_runlog: Optional\[Callable\[\[Union\[opentrons.legacy_commands.types.DropTipMessage, opentrons.legacy_commands.types.DropTipInDisposalLocationMessage, opentrons.legacy_commands.types.PickUpTipMessage, opentrons.legacy_commands.types.ReturnTipMessage, opentrons.legacy_commands.types.AirGapMessage, opentrons.legacy_commands.types.TouchTipMessage, opentrons.legacy_commands.types.BlowOutMessage, opentrons.legacy_commands.types.BlowOutInDisposalLocationMessage, opentrons.legacy_commands.types.MixMessage, opentrons.legacy_commands.types.TransferMessage, opentrons.legacy_commands.types.DistributeMessage, opentrons.legacy_commands.types.ConsolidateMessage, opentrons.legacy_commands.types.DispenseMessage, opentrons.legacy_commands.types.DispenseInDisposalLocationMessage, opentrons.legacy_commands.types.AspirateMessage, opentrons.legacy_commands.types.HomeMessage, opentrons.legacy_commands.types.HeaterShakerSetTargetTemperatureMessage, opentrons.legacy_commands.types.HeaterShakerWaitForTemperatureMessage, opentrons.legacy_commands.types.HeaterShakerSetAndWaitForShakeSpeedMessage, opentrons.legacy_commands.types.HeaterShakerOpenLabwareLatchMessage, opentrons.legacy_commands.types.HeaterShakerCloseLabwareLatchMessage, opentrons.legacy_commands.types.HeaterShakerDeactivateShakerMessage, opentrons.legacy_commands.types.HeaterShakerDeactivateHeaterMessage, opentrons.legacy_commands.types.ThermocyclerCloseMessage, opentrons.legacy_commands.types.ThermocyclerWaitForLidTempMessage, opentrons.legacy_commands.types.ThermocyclerDeactivateMessage, opentrons.legacy_commands.types.ThermocyclerDeactivateBlockMessage, opentrons.legacy_commands.types.ThermocyclerDeactivateLidMessage, opentrons.legacy_commands.types.ThermocyclerSetLidTempMessage, opentrons.legacy_commands.types.ThermocyclerWaitForTempMessage, opentrons.legacy_commands.types.ThermocyclerWaitForHoldMessage, opentrons.legacy_commands.types.ThermocyclerExecuteProfileMessage, opentrons.legacy_commands.types.ThermocyclerSetBlockTempMessage, opentrons.legacy_commands.types.ThermocyclerOpenMessage, opentrons.legacy_commands.types.TempdeckSetTempMessage, opentrons.legacy_commands.types.TempdeckDeactivateMessage, opentrons.legacy_commands.types.MagdeckEngageMessage, opentrons.legacy_commands.types.MagdeckDisengageMessage, opentrons.legacy_commands.types.MagdeckCalibrateMessage, opentrons.legacy_commands.types.CommentMessage, opentrons.legacy_commands.types.DelayMessage, opentrons.legacy_commands.types.PauseMessage, opentrons.legacy_commands.types.ResumeMessage, opentrons.legacy_commands.types.MoveToMessage, opentrons.legacy_commands.types.MoveToDisposalLocationMessage, opentrons.legacy_commands.types.MoveLabwareMessage]], NoneType]] \= None_, _custom_labware_paths: Optional\[List\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]] \= None_, _custom_data_paths: Optional\[List\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]] \= None_) → [None](https://docs.python.org/3/library/constants.html#None '(in Python v3.12)') +Run the protocol itself. + +This is a one\-stop function to run a protocol, whether python or json, +no matter the api version, from external (i.e. not bound up in other +internal server infrastructure) sources. + +To run an opentrons protocol from other places, pass in a file like +object as protocol_file; this function either returns (if the run has no +problems) or raises an exception. + +To call from the command line use either the autogenerated entrypoint +`opentrons_execute` or `python -m opentrons.execute`. + +Parameters: + +- **protocol_file** – The protocol file to execute +- **protocol_name** – The name of the protocol file. This is required + internally, but it may not be a thing we can get + from the protocol_file argument. +- **propagate_logs** – Whether this function should allow logs from the + Opentrons stack to propagate up to the root handler. + This can be useful if you’re integrating this + function in a larger application, but most logs that + occur during protocol simulation are best associated + with the actions in the protocol that cause them. + Default: `False` +- **log_level** – The level of logs to emit on the command line: + `"debug"`, `"info"`, `"warning"`, or `"error"`. + Defaults to `"warning"`. +- **emit_runlog** – A callback for printing the run log. If specified, this + will be called whenever a command adds an entry to the + run log, which can be used for display and progress + estimation. If specified, the callback should take a + single argument (the name doesn’t matter) which will + be a dictionary: + +``` +{ + 'name': command_name, + 'payload': { + 'text': string_command_text, + # The rest of this struct is + # command-dependent; see + # opentrons.legacy_commands.commands. + } +} + +``` + +Note + +In older software versions, `payload["text"]` was a +[format string](https://docs.python.org/3/library/string.html#formatstrings). +To get human\-readable text, you had to do `payload["text"].format(**payload)`. +Don’t do that anymore. If `payload["text"]` happens to contain any +`{` or `}` characters, it can confuse `.format()` and cause it to raise a +`KeyError`. + +- **custom_labware_paths** – A list of directories to search for custom labware. + Loads valid labware from these paths and makes them available + to the protocol context. If this is `None` (the default), and + this function is called on a robot, it will look in the `labware` + subdirectory of the Jupyter data directory. +- **custom_data_paths** – A list of directories or files to load custom + data files from. Ignored if the apiv2 feature + flag if not set. Entries may be either files or + directories. Specified files and the + non\-recursive contents of specified directories + are presented by the protocol context in + `ProtocolContext.bundled_data`. + +opentrons.execute.get*arguments(\_parser: [argparse.ArgumentParser](https://docs.python.org/3/library/argparse.html#argparse.ArgumentParser '(in Python v3.12)')*) → [argparse.ArgumentParser](https://docs.python.org/3/library/argparse.html#argparse.ArgumentParser '(in Python v3.12)') +Get the argument parser for this module + +Useful if you want to use this module as a component of another CLI program +and want to add its arguments. + +Parameters: +**parser** – A parser to add arguments to. + +Returns argparse.ArgumentParser: +The parser with arguments added. + +opentrons.execute.get*protocol_api(\_version: Union\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), opentrons.protocols.api_support.types.APIVersion]*, _bundled_labware: Optional\[Dict\[str, ForwardRef('LabwareDefinitionDict')]] \= None_, _bundled_data: Optional\[Dict\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), [bytes](https://docs.python.org/3/library/stdtypes.html#bytes '(in Python v3.12)')]] \= None_, _extra_labware: Optional\[Dict\[str, ForwardRef('LabwareDefinitionDict')]] \= None_) → [opentrons.protocol_api.protocol_context.ProtocolContext](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.protocol_context.ProtocolContext') +Build and return a `protocol_api.ProtocolContext` +connected to the robot. + +This can be used to run protocols from interactive Python sessions +such as Jupyter or an interpreter on the command line: + +``` +>>> from opentrons.execute import get_protocol_api +>>> protocol = get_protocol_api('2.0') +>>> instr = protocol.load_instrument('p300_single', 'right') +>>> instr.home() + +``` + +When this function is called, modules and instruments will be recached. + +Parameters: + +- **version** – The API version to use. This must be lower than + `opentrons.protocol_api.MAX_SUPPORTED_VERSION`. + It may be specified either as a string (`'2.0'`) or + as a `protocols.types.APIVersion` + (`APIVersion(2, 0)`). +- **bundled_labware** – If specified, a mapping from labware names to + labware definitions for labware to consider in the + protocol. Note that if you specify this, \_only\_ + labware in this argument will be allowed in the + protocol. This is preparation for a beta feature + and is best not used. +- **bundled_data** – If specified, a mapping from filenames to contents + for data to be available in the protocol from + [`opentrons.protocol_api.ProtocolContext.bundled_data`](#opentrons.protocol_api.ProtocolContext.bundled_data 'opentrons.protocol_api.ProtocolContext.bundled_data'). +- **extra_labware** – A mapping from labware load names to custom labware definitions. + If this is `None` (the default), and this function is called on a robot, + it will look for labware in the `labware` subdirectory of the Jupyter + data directory. + +Returns: +The protocol context. + +opentrons.execute.main() → [int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)') +Handler for command line invocation to run a protocol. + +Parameters: +**argv** – The arguments the program was invoked with; this is usually +[`sys.argv`](https://docs.python.org/3/library/sys.html#sys.argv '(in Python v3.12)') but if you want to override that you can. + +Returns int: +A success or failure value suitable for use as a shell +return code passed to [`sys.exit`](https://docs.python.org/3/library/sys.html#sys.exit '(in Python v3.12)') (0 means success, +anything else is a kind of failure). + +opentrons.simulate: functions and entrypoints for simulating protocols + +This module has functions that provide a console entrypoint for simulating +a protocol from the command line. + +opentrons.simulate.allow_bundle() → [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)') +Check if bundling is allowed with a special not\-exposed\-to\-the\-app flag. + +Returns `True` if the environment variable +`OT_API_FF_allowBundleCreation` is `"1"` + +opentrons.simulate.bundle*from_sim(\_protocol: opentrons.protocols.types.PythonProtocol*, _context: [opentrons.protocol_api.protocol_context.ProtocolContext](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.protocol_context.ProtocolContext')_) → opentrons.protocols.types.BundleContents +From a protocol, and the context that has finished simulating that +protocol, determine what needs to go in a bundle for the protocol. + +opentrons.simulate.format*runlog(\_runlog: List\[Mapping\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), Any]]*) → [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)') +Format a run log (return value of [`simulate`](#opentrons.simulate.simulate 'opentrons.simulate.simulate')) into a +human\-readable string + +Parameters: +**runlog** – The output of a call to [`simulate`](#opentrons.simulate.simulate 'opentrons.simulate.simulate') + +opentrons.simulate.get*arguments(\_parser: [argparse.ArgumentParser](https://docs.python.org/3/library/argparse.html#argparse.ArgumentParser '(in Python v3.12)')*) → [argparse.ArgumentParser](https://docs.python.org/3/library/argparse.html#argparse.ArgumentParser '(in Python v3.12)') +Get the argument parser for this module + +Useful if you want to use this module as a component of another CLI program +and want to add its arguments. + +Parameters: +**parser** – A parser to add arguments to. If not specified, one will be +created. + +Returns argparse.ArgumentParser: +The parser with arguments added. + +opentrons.simulate.get*protocol_api(\_version: Union\[str, opentrons.protocols.api_support.types.APIVersion], bundled_labware: Optional\[Dict\[str, ForwardRef('LabwareDefinitionDict')]] \= None, bundled_data: Optional\[Dict\[str, bytes]] \= None, extra_labware: Optional\[Dict\[str, ForwardRef('LabwareDefinitionDict')]] \= None, hardware_simulator: Optional\[opentrons.hardware_control.thread_manager.ThreadManager\[Union\[opentrons.hardware_control.protocols.HardwareControlInterface\[opentrons.hardware_control.robot_calibration.RobotCalibration, opentrons.types.Mount, opentrons.config.types.RobotConfig], opentrons.hardware_control.protocols.FlexHardwareControlInterface\[opentrons.hardware_control.ot3_calibration.OT3Transforms, Union\[opentrons.types.Mount, opentrons.hardware_control.types.OT3Mount], opentrons.config.types.OT3Config]]]] \= None, \\\*, robot_type: Optional\[Literal\['OT\-2', 'Flex']] \= None, use_virtual_hardware: bool \= True*) → [opentrons.protocol_api.protocol_context.ProtocolContext](index.html#opentrons.protocol_api.ProtocolContext 'opentrons.protocol_api.protocol_context.ProtocolContext') +Build and return a `protocol_api.ProtocolContext` +connected to Virtual Smoothie. + +This can be used to run protocols from interactive Python sessions +such as Jupyter or an interpreter on the command line: + +``` +>>> from opentrons.simulate import get_protocol_api +>>> protocol = get_protocol_api('2.0') +>>> instr = protocol.load_instrument('p300_single', 'right') +>>> instr.home() + +``` + +Parameters: + +- **version** – The API version to use. This must be lower than + `opentrons.protocol_api.MAX_SUPPORTED_VERSION`. + It may be specified either as a string (`'2.0'`) or + as a `protocols.types.APIVersion` + (`APIVersion(2, 0)`). +- **bundled_labware** – If specified, a mapping from labware names to + labware definitions for labware to consider in the + protocol. Note that if you specify this, \_only\_ + labware in this argument will be allowed in the + protocol. This is preparation for a beta feature + and is best not used. +- **bundled_data** – If specified, a mapping from filenames to contents + for data to be available in the protocol from + [`opentrons.protocol_api.ProtocolContext.bundled_data`](#opentrons.protocol_api.ProtocolContext.bundled_data 'opentrons.protocol_api.ProtocolContext.bundled_data'). +- **extra_labware** – A mapping from labware load names to custom labware definitions. + If this is `None` (the default), and this function is called on a robot, + it will look for labware in the `labware` subdirectory of the Jupyter + data directory. +- **hardware_simulator** – If specified, a hardware simulator instance. +- **robot_type** – The type of robot to simulate: either `"Flex"` or `"OT-2"`. + If you’re running this function on a robot, the default is the type of that + robot. Otherwise, the default is `"OT-2"`, for backwards compatibility. +- **use_virtual_hardware** – If true, use the protocol engines virtual hardware, if false use the lower level hardware simulator. + +Returns: +The protocol context. + +opentrons.simulate.main() → [int](https://docs.python.org/3/library/functions.html#int '(in Python v3.12)') +Run the simulation + +opentrons.simulate.simulate(_protocol_file: Union\[BinaryIO, TextIO]_, _file_name: Optional\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')] \= None_, _custom_labware_paths: Optional\[List\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]] \= None_, _custom_data_paths: Optional\[List\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')]] \= None_, _propagate_logs: [bool](https://docs.python.org/3/library/functions.html#bool '(in Python v3.12)') \= False_, _hardware_simulator_file_path: Optional\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)')] \= None_, _duration_estimator: Optional\[opentrons.protocols.duration.estimator.DurationEstimator] \= None_, _log_level: [str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)') \= 'warning'_) → Tuple\[List\[Mapping\[[str](https://docs.python.org/3/library/stdtypes.html#str '(in Python v3.12)'), Any]], Optional\[opentrons.protocols.types.BundleContents]] +Simulate the protocol itself. + +This is a one\-stop function to simulate a protocol, whether python or json, +no matter the api version, from external (i.e. not bound up in other +internal server infrastructure) sources. + +To simulate an opentrons protocol from other places, pass in a file like +object as protocol_file; this function either returns (if the simulation +has no problems) or raises an exception. + +To call from the command line use either the autogenerated entrypoint +`opentrons_simulate` (`opentrons_simulate.exe`, on windows) or +`python -m opentrons.simulate`. + +The return value is the run log, a list of dicts that represent the +commands executed by the robot; and either the contents of the protocol +that would be required to bundle, or `None`. + +Each dict element in the run log has the following keys: + +> - `level`: The depth at which this command is nested. If this an +> aspirate inside a mix inside a transfer, for instance, it would be 3\. +> - `payload`: The command. The human\-readable run log text is available at +> `payload["text"]`. The other keys of `payload` are command\-dependent; +> see `opentrons.legacy_commands`. +> +> Note +> +> In older software versions, `payload["text"]` was a +> [format string](https://docs.python.org/3/library/string.html#formatstrings). +> To get human\-readable text, you had to do `payload["text"].format(**payload)`. +> Don’t do that anymore. If `payload["text"]` happens to contain any +> `{` or `}` characters, it can confuse `.format()` and cause it to raise a +> `KeyError`. +> +> - `logs`: Any log messages that occurred during execution of this +> command, as a standard Python [`LogRecord`](https://docs.python.org/3/library/logging.html#logging.LogRecord '(in Python v3.12)'). + +Parameters: + +- **protocol_file** – The protocol file to simulate. +- **file_name** – The name of the file +- **custom_labware_paths** – A list of directories to search for custom labware. + Loads valid labware from these paths and makes them available + to the protocol context. If this is `None` (the default), and + this function is called on a robot, it will look in the `labware` + subdirectory of the Jupyter data directory. +- **custom_data_paths** – A list of directories or files to load custom + data files from. Ignored if the apiv2 feature + flag if not set. Entries may be either files or + directories. Specified files and the + non\-recursive contents of specified directories + are presented by the protocol context in + `protocol_api.ProtocolContext.bundled_data`. +- **hardware_simulator_file_path** – A path to a JSON file defining a + hardware simulator. +- **duration_estimator** – For internal use only. + Optional duration estimator object. +- **propagate_logs** – Whether this function should allow logs from the + Opentrons stack to propagate up to the root handler. + This can be useful if you’re integrating this + function in a larger application, but most logs that + occur during protocol simulation are best associated + with the actions in the protocol that cause them. + Default: `False` +- **log_level** – The level of logs to capture in the run log: + `"debug"`, `"info"`, `"warning"`, or `"error"`. + Defaults to `"warning"`. + +Returns: +A tuple of a run log for user output, and possibly the required +data to write to a bundle to bundle this protocol. The bundle is +only emitted if bundling is allowed +and this is an unbundled Protocol API +v2 python protocol. In other cases it is None. diff --git a/opentrons-ai-server/api/utils/convert_to_markdown.py b/opentrons-ai-server/api/utils/convert_to_markdown.py new file mode 100644 index 00000000000..cacd23d2558 --- /dev/null +++ b/opentrons-ai-server/api/utils/convert_to_markdown.py @@ -0,0 +1,214 @@ +import os.path +import subprocess +import uuid + +from bs4 import BeautifulSoup +from bs4.element import Tag +from markdownify import markdownify # type: ignore + + +def run_sphinx_build(command: str) -> None: + """Run the sphinx command to convert rst files to a single HTML file.""" + try: + subprocess.run(command, check=True, shell=True) + except subprocess.CalledProcessError as e: + print(f"An error occurred while running Sphinx build: {e}") + + +def remove_specific_logos(soup: BeautifulSoup) -> BeautifulSoup: + """Remove specific logos from the HTML.""" + logos = soup.find_all("img", src=lambda x: x and ("opentrons-images/website" in x)) + for logo in logos: + logo.decompose() + return soup + + +def remove_all_images(soup: BeautifulSoup) -> BeautifulSoup: + """Remove all images from the HTML.""" + all_images = soup.find_all("img") + for img in all_images: + img.decompose() + return soup + + +def remove_pilcrow_symbols(soup: BeautifulSoup) -> BeautifulSoup: + """Remove all pilcrow symbols from the HTML.""" + pilcrow_symbols = soup.find_all("a", string="¶") + for symbol in pilcrow_symbols: + symbol.decompose() + return soup + + +def remove_list_items_containing_ot1(soup: BeautifulSoup) -> BeautifulSoup: + """Remove all
  • elements containing 'OT-1'.""" + list_items = soup.find_all("li") + for li in list_items: + if "OT-1" in li.get_text(): + li.decompose() + return soup + + +def remove_top_section(soup: BeautifulSoup) -> BeautifulSoup: + """Remove everything before a Python API docs header section.""" + # Remove everything before the
    element + start_section = soup.find("div", class_="document") + + # Check if the section was found + if not start_section: + print("Start section not found in the HTML content.") + return soup + + # Find the head tag and remove it + head_tag = soup.find("head") + if isinstance(head_tag, Tag): + head_tag.decompose() + + # Remove all previous siblings of the start_section + for previous in list(start_section.previous_siblings): + previous.extract() + + # Remove the parent elements if they are no longer needed + for parent in list(start_section.parents): + if parent.name == "body": + break + if not parent.find_previous_siblings() and not parent.find_next_siblings(): + parent.extract() + + return soup + + +def remove_footer_content(soup: BeautifulSoup) -> BeautifulSoup: + """Remove the footer content from the HTML.""" + footer_section = soup.find("footer") + if isinstance(footer_section, Tag): + footer_section.decompose() + return soup + + +def clean_html(soup: BeautifulSoup) -> BeautifulSoup: + """Clean up the unused features in the HTML file.""" + soup = remove_specific_logos(soup) + soup = remove_all_images(soup) + soup = remove_pilcrow_symbols(soup) + soup = remove_list_items_containing_ot1(soup) + soup = remove_top_section(soup) + soup = remove_footer_content(soup) + return soup + + +def extract_and_remove_api_reference(html_file_path: str, output_file_path: str) -> BeautifulSoup: + """Extract and remove the API Version 2 Reference section and write it to a Markdown file.""" + + with open(html_file_path, "r", encoding="utf-8") as file: + html_content = file.read() + + soup = BeautifulSoup(html_content, "html.parser") + soup = clean_html(soup) + + # Find the start and end points + start_span = soup.find("span", id="document-new_protocol_api") + if start_span is None: + print("Start span not found.") + return soup + + # Get the section to keep + api_section = start_span.find_next_sibling("section", id="api-version-2-reference") + if api_section is None: + print("API section not found.") + return soup + + # Create a BeautifulSoup object for the extracted section + extracted_html = str(start_span) + str(api_section) + reference_markdown = markdownify(extracted_html) + + # Write the extracted content to a Markdown file + with open(output_file_path, "w", encoding="utf-8") as file: + file.write(reference_markdown) + + # Remove it from the main markdown file + if isinstance(start_span, Tag) and isinstance(api_section, Tag): + start_span.decompose() + api_section.decompose() + + return soup + + +def extract_tab_content(soup: BeautifulSoup) -> tuple[BeautifulSoup, dict[str, str]]: + """Find all tabbed content sections and convert each tabbed section to markdown format.""" + tab_sections = soup.find_all(class_="sphinx-tabs docutils container") + tab_markdown = {} + + for _idx, tab_section in enumerate(tab_sections): + tab_buttons = tab_section.find_all(class_="sphinx-tabs-tab") + tab_panels = tab_section.find_all(class_="sphinx-tabs-panel") + + section_markdown = [] + for button, panel in zip(tab_buttons, tab_panels, strict=False): + section_markdown.append(f"### {button.text.strip()}\n") + panel_content = markdownify(str(panel), strip=["div"]) + section_markdown.append(panel_content) + combined_section_markdown = "\n".join(section_markdown) + "\n\n" + # Replace the original tab section with an unique placeholder in the soup + placeholder = f"tabSectionIs{uuid.uuid4().hex}" + tab_markdown[placeholder] = combined_section_markdown + placeholder_tag = soup.new_tag("div") + placeholder_tag.string = placeholder + tab_section.replace_with(placeholder_tag) + + return soup, tab_markdown + + +def convert_html_to_markdown(html_file_path: str, markdown_file_path: str, reference_file_path: str) -> None: + """Converts an HTML file to a Markdown file with specific modifications.""" + + soup = extract_and_remove_api_reference(html_file_path, reference_file_path) + soup, tab_markdown = extract_tab_content(soup) + + modified_html_content = str(soup) + full_markdown = markdownify(modified_html_content) + + for placeholder, section_md in tab_markdown.items(): + if placeholder in full_markdown: + full_markdown = full_markdown.replace(placeholder, section_md) + + with open(markdown_file_path, "w", encoding="utf-8") as file: + file.write(full_markdown) + + +def get_latest_version() -> str: + """Get the lastest docs version number.""" + try: + # Run the git command to get the latest tag + command = "git tag -l 'docs@2*' --sort=-taggerdate | head -n 1" + result = subprocess.run(command, capture_output=True, text=True, shell=True) + # Extract the tag from the output and remove '.' + tag = "".join(result.stdout.strip().split(".")) + + version = tag.split("@")[1] + version = version.split("_")[0] + return version + except subprocess.CalledProcessError as e: + print(f"An error occurred while getting the version: {e}") + return "" + + +def get_markdown_format() -> None: + """Generates a version-aware Markdown file from HTML documentation.""" + current_version = get_latest_version() + current_dir = os.path.dirname(__file__) + + docs_src_path = os.path.join("..", "api", "docs", "v2") + build_html_path = os.path.join(current_dir, "build", "docs", "html", "v2") + html_file_path = os.path.join(build_html_path, "index.html") + markdown_file_path = os.path.join(current_dir, "..", "data", f"python_api_{current_version}_docs.md") + reference_file_path = os.path.join(current_dir, "..", "data", f"python_api_{current_version}_reference.md") + + command = f"pipenv run sphinx-build -b singlehtml {docs_src_path} {build_html_path}" + + run_sphinx_build(command) + + convert_html_to_markdown(html_file_path, markdown_file_path, reference_file_path) + + +if __name__ == "__main__": + get_markdown_format() diff --git a/opentrons-ai-server/tests/test_convert_to_markdown.py b/opentrons-ai-server/tests/test_convert_to_markdown.py new file mode 100644 index 00000000000..8210ccf4f3f --- /dev/null +++ b/opentrons-ai-server/tests/test_convert_to_markdown.py @@ -0,0 +1,145 @@ +from unittest.mock import MagicMock, mock_open, patch + +import pytest +from api.utils.convert_to_markdown import ( + clean_html, + convert_html_to_markdown, + extract_and_remove_api_reference, + extract_tab_content, + get_latest_version, + get_markdown_format, + remove_all_images, + remove_footer_content, + remove_list_items_containing_ot1, + remove_pilcrow_symbols, + remove_specific_logos, + remove_top_section, + run_sphinx_build, +) +from bs4 import BeautifulSoup + +# Sample HTML content for testing +sample_html = """ + + + +
    +
    + + +
  • OT-1
  • + +
    +
    +
    Tab 1
    +
    Content 1
    +
    Tab 2
    +
    Content 2
    +
    + + +""" + + +@pytest.fixture +def soup() -> BeautifulSoup: + return BeautifulSoup(sample_html, "html.parser") + + +@pytest.mark.unit +def test_run_sphinx_build() -> None: + with patch("subprocess.run") as mock_run: + run_sphinx_build("echo test") + mock_run.assert_called_once_with("echo test", check=True, shell=True) + + +@pytest.mark.unit +def test_remove_specific_logos(soup: BeautifulSoup) -> None: + soup = remove_specific_logos(soup) + assert not soup.find_all("img", src="opentrons-images/website/logo.png") + + +@pytest.mark.unit +def test_remove_all_images(soup: BeautifulSoup) -> None: + soup = remove_all_images(soup) + assert not soup.find_all("img") + + +@pytest.mark.unit +def test_remove_pilcrow_symbols(soup: BeautifulSoup) -> None: + soup = remove_pilcrow_symbols(soup) + assert not soup.find_all("a", string="¶") + + +@pytest.mark.unit +def test_remove_list_items_containing_ot1(soup: BeautifulSoup) -> None: + soup = remove_list_items_containing_ot1(soup) + assert not soup.find_all("li", string="OT-1") + + +@pytest.mark.unit +def test_remove_top_section(soup: BeautifulSoup) -> None: + soup = remove_top_section(soup) + assert not soup.find("head") + + +@pytest.mark.unit +def test_remove_footer_content(soup: BeautifulSoup) -> None: + soup = remove_footer_content(soup) + assert not soup.find("footer") + + +@pytest.mark.unit +def test_clean_html(soup: BeautifulSoup) -> None: + soup = clean_html(soup) + assert not soup.find_all("img", src="opentrons-images/website/logo.png") + assert not soup.find_all("img") + assert not soup.find_all("a", string="¶") + assert not soup.find_all("li", string="OT-1") + assert not soup.find("head") + assert not soup.find("footer") + + +@pytest.mark.unit +@patch("builtins.open", new_callable=mock_open, read_data=sample_html) +def test_extract_and_remove_api_reference(mock_file: MagicMock, soup: BeautifulSoup) -> None: + output_file_path = "output.md" + html_file_path = "index.html" + soup = extract_and_remove_api_reference(html_file_path, output_file_path) + assert not soup.find("span", id="document-new_protocol_api") + assert not soup.find("section", id="api-version-2-reference") + + +@pytest.mark.unit +def test_extract_tab_content(soup: BeautifulSoup) -> None: + soup, tab_markdown = extract_tab_content(soup) + assert len(tab_markdown) == 1 + + +@pytest.mark.unit +@patch("builtins.open", new_callable=mock_open) +def test_convert_html_to_markdown(mock_file: MagicMock, soup: BeautifulSoup) -> None: + html_file_path = "index.html" + markdown_file_path = "output.md" + reference_file_path = "reference.md" + convert_html_to_markdown(html_file_path, markdown_file_path, reference_file_path) + mock_file.assert_called() + + +@pytest.mark.unit +@patch("subprocess.run") +def test_get_latest_version(mock_run: MagicMock) -> None: + mock_run.return_value.stdout = "docs@2.19_2\n" + version = get_latest_version() + assert version == "219" + + +@pytest.mark.unit +@patch("api.utils.convert_to_markdown.get_latest_version") +@patch("api.utils.convert_to_markdown.run_sphinx_build") +@patch("api.utils.convert_to_markdown.convert_html_to_markdown") +def test_get_markdown_format(mock_convert: MagicMock, mock_build: MagicMock, mock_version: MagicMock) -> None: + mock_version.return_value = "200" + get_markdown_format() + mock_build.assert_called() + mock_convert.assert_called() diff --git a/performance-metrics/README.md b/performance-metrics/README.md index 008ff0f6db2..637bcea0b8e 100644 --- a/performance-metrics/README.md +++ b/performance-metrics/README.md @@ -13,29 +13,7 @@ It is assumed that you already have the other projects in the monorepo setup cor make -C performance-metrics setup ``` -### Testing against OT-2 Dev Server - -```bash -make -C robot-server dev-ot2 -``` - -### Testing against real OT-2 - -To push development packages to OT-2 run the following commands from the root directory of this repo: - -```bash -make -C performance-metrics push-no-restart host= -make -C api push-no-restart host= -make -C robot-server push host= -``` - -### Testing against Flex Dev Server - -```bash -make -C robot-server dev-flex -``` - -### Testing against real Flex +### Pushing performance-metrics package to Flex ```bash make -C performance-metrics push-no-restart-ot3 host= @@ -69,3 +47,34 @@ To disable it run: ```bash make unset-performance-metrics-ff host= ``` + +## Available features + +### Robot activity tracking + +#### Description + +Developers are able to track when the robot is in a block of code they choose to monitor. Looking at +`api/src/opentrons/util/performance_helpers.py` you will see a class called `TrackingFunctions`. This class +defines static methods which are decorators that can be used wrap arbitrary functions. + +As of 2024-07-31, the following tracking functions are available: + +- `track_analysis` +- `track_getting_cached_protocol_analysis` + +Looking at `TrackingFunctions.track_analysis` we see that the underlying call to \_track_a_function specifies a string `"ANALYZING_PROTOCOL"`. Whenever a function that is wrapped with `TrackingFunctions.track_analysis` executes, the tracking function will label the underlying function as `"ANALYZING_PROTOCOL"`. + +To see where tracking function is used look at `robot_server/robot-server/protocols/protocol_analyzer.py`. You will see that the `ProtocolAnalyzer.analyze` function is wrapped with `TrackingFunctions.track_analysis`. Whenever `ProtocolAnalyzer.analyze` is called, the tracking function will start a timer. When the `ProtocolAnalyzer.analyze` function completes, the tracking function will stop the timer. It will then store the function start time and duration to the csv file, /data/performance_metrics_data/robot_activity_data + +#### Adding new tracking decorator + +To add a new tracking decorator, go to `performance-metrics/src/performance_metrics/_types.py`, and look at RobotActivityState literal and add a new state. +Go to `api/src/opentrons/util/performance_helpers.py` and add a static method to the `TrackingFunctions` class that uses the new state. + +You can now wrap your functions with your new tracking decorator. + +### System resource tracking + +performance-metrics also exposes a tracking application called `SystemResourceTracker`. The application is implemented as a systemd service on the robot and records system resource usage by process. See the `oe-core` repo for more details. +You can configure the system resource tracker by modifying the environment variables set for the service. The service file lives at `/lib/systemd/system/system-resource-tracker.service`. You can change the defined environment variables or remove them and define them in the robot's environment variables. See `performance-metrics/src/performance_metrics/system_resource_tracker/_config.py` to see what environment variables are available. diff --git a/performance-metrics/src/performance_metrics/__init__.py b/performance-metrics/src/performance_metrics/__init__.py index 998e9181bf5..c87532d9449 100644 --- a/performance-metrics/src/performance_metrics/__init__.py +++ b/performance-metrics/src/performance_metrics/__init__.py @@ -1,11 +1,11 @@ """Opentrons performance metrics library.""" -from ._robot_context_tracker import RobotContextTracker -from ._types import RobotContextState, SupportsTracking +from ._robot_activity_tracker import RobotActivityTracker +from ._types import RobotActivityState, SupportsTracking __all__ = [ - "RobotContextTracker", - "RobotContextState", + "RobotActivityTracker", + "RobotActivityState", "SupportsTracking", ] diff --git a/performance-metrics/src/performance_metrics/_data_shapes.py b/performance-metrics/src/performance_metrics/_data_shapes.py index d07a1da71fd..237a2e1b066 100644 --- a/performance-metrics/src/performance_metrics/_data_shapes.py +++ b/performance-metrics/src/performance_metrics/_data_shapes.py @@ -4,52 +4,48 @@ import typing from pathlib import Path -from ._types import SupportsCSVStorage, StorableData, RobotContextState +from ._types import StorableData, RobotActivityState from ._util import get_timing_function _timing_function = get_timing_function() @dataclasses.dataclass(frozen=True) -class RawContextData(SupportsCSVStorage): - """Represents raw duration data with context state information. +class CSVStorageBase: + """Base class for all data classes.""" + + @classmethod + def headers(cls) -> typing.Tuple[str, ...]: + """Returns the headers for the BaseData class.""" + return tuple([field.name for field in dataclasses.fields(cls)]) + + def csv_row(self) -> typing.Tuple[StorableData, ...]: + """Returns the object as a CSV row.""" + return dataclasses.astuple(self) + + @classmethod + def from_csv_row(cls, row: typing.Sequence[StorableData]) -> "CSVStorageBase": + """Returns an object from a CSV row.""" + return cls(*row) + + +@dataclasses.dataclass(frozen=True) +class RawActivityData(CSVStorageBase): + """Represents raw duration data with activity state information. Attributes: - - function_start_time (int): The start time of the function. - - duration_measurement_start_time (int): The start time for duration measurement. - - duration_measurement_end_time (int): The end time for duration measurement. - - state (RobotContextStates): The current state of the context. + - state (RobotActivityStates): The current state of the activity. + - func_start (int): The start time of the function. + - duration (int): The start time for duration measurement. """ - state: RobotContextState + state: RobotActivityState func_start: int duration: int - @classmethod - def headers(self) -> typing.Tuple[str, str, str]: - """Returns the headers for the raw context data.""" - return ("state_name", "function_start_time", "duration") - - def csv_row(self) -> typing.Tuple[str, int, int]: - """Returns the raw context data as a string.""" - return ( - self.state, - self.func_start, - self.duration, - ) - - @classmethod - def from_csv_row(cls, row: typing.Sequence[StorableData]) -> SupportsCSVStorage: - """Returns a RawContextData object from a CSV row.""" - return cls( - state=typing.cast(RobotContextState, row[0]), - func_start=int(row[1]), - duration=int(row[2]), - ) - @dataclasses.dataclass(frozen=True) -class ProcessResourceUsageSnapshot(SupportsCSVStorage): +class ProcessResourceUsageSnapshot(CSVStorageBase): """Represents process resource usage data. Attributes: @@ -68,41 +64,6 @@ class ProcessResourceUsageSnapshot(SupportsCSVStorage): system_cpu_time: float # seconds memory_percent: float - @classmethod - def headers(self) -> typing.Tuple[str, str, str, str, str, str]: - """Returns the headers for the process resource usage data.""" - return ( - "query_time", - "command", - "running_since", - "user_cpu_time", - "system_cpu_time", - "memory_percent", - ) - - def csv_row(self) -> typing.Tuple[int, str, float, float, float, float]: - """Returns the process resource usage data as a string.""" - return ( - self.query_time, - self.command, - self.running_since, - self.user_cpu_time, - self.system_cpu_time, - self.memory_percent, - ) - - @classmethod - def from_csv_row(cls, row: typing.Sequence[StorableData]) -> SupportsCSVStorage: - """Returns a ProcessResourceUsageData object from a CSV row.""" - return cls( - query_time=int(row[0]), - command=str(row[1]), - running_since=float(row[2]), - user_cpu_time=float(row[3]), - system_cpu_time=float(row[4]), - memory_percent=float(row[4]), - ) - @dataclasses.dataclass(frozen=True) class MetricsMetadata: diff --git a/performance-metrics/src/performance_metrics/_metrics_store.py b/performance-metrics/src/performance_metrics/_metrics_store.py index 09bcce50e29..8d790c67a07 100644 --- a/performance-metrics/src/performance_metrics/_metrics_store.py +++ b/performance-metrics/src/performance_metrics/_metrics_store.py @@ -3,30 +3,29 @@ import csv import typing import logging -from ._data_shapes import MetricsMetadata -from ._types import SupportsCSVStorage +from ._data_shapes import MetricsMetadata, CSVStorageBase from ._logging_config import LOGGER_NAME logger = logging.getLogger(LOGGER_NAME) -T = typing.TypeVar("T", bound=SupportsCSVStorage) +T = typing.TypeVar("T", bound=CSVStorageBase) class MetricsStore(typing.Generic[T]): - """Dataclass to store data for tracking robot context.""" + """Dataclass to store data for tracking robot activity.""" def __init__(self, metadata: MetricsMetadata) -> None: """Initialize the metrics store.""" self.metadata = metadata - self._data: typing.List[T] = [] + self._data_store: typing.List[T] = [] - def add(self, context_data: T) -> None: + def add(self, data: T) -> None: """Add data to the store.""" - self._data.append(context_data) + self._data_store.append(data) - def add_all(self, context_data: typing.Iterable[T]) -> None: + def add_all(self, data: typing.Iterable[T]) -> None: """Add data to the store.""" - self._data.extend(context_data) + self._data_store.extend(data) def setup(self) -> None: """Set up the data store.""" @@ -40,9 +39,9 @@ def setup(self) -> None: def store(self) -> None: """Clear the stored data and write it to the storage file.""" - stored_data = self._data.copy() - self._data.clear() - rows_to_write = [context_data.csv_row() for context_data in stored_data] + stored_data = self._data_store.copy() + self._data_store.clear() + rows_to_write = [activity_data.csv_row() for activity_data in stored_data] with open(self.metadata.data_file_location, "a") as storage_file: logger.debug( f"Writing {len(rows_to_write)} rows to {self.metadata.data_file_location}" diff --git a/performance-metrics/src/performance_metrics/_robot_context_tracker.py b/performance-metrics/src/performance_metrics/_robot_activity_tracker.py similarity index 82% rename from performance-metrics/src/performance_metrics/_robot_context_tracker.py rename to performance-metrics/src/performance_metrics/_robot_activity_tracker.py index 61f29573681..7e599104a3d 100644 --- a/performance-metrics/src/performance_metrics/_robot_context_tracker.py +++ b/performance-metrics/src/performance_metrics/_robot_activity_tracker.py @@ -1,4 +1,4 @@ -"""Module for tracking robot context and execution duration for different operations.""" +"""Module for tracking robot activity and execution duration for different operations.""" import inspect from pathlib import Path @@ -8,8 +8,8 @@ import typing from ._metrics_store import MetricsStore -from ._data_shapes import RawContextData, MetricsMetadata -from ._types import SupportsTracking, RobotContextState +from ._data_shapes import RawActivityData, MetricsMetadata +from ._types import SupportsTracking, RobotActivityState from ._util import get_timing_function _UnderlyingFunctionParameters = typing.ParamSpec("_UnderlyingFunctionParameters") @@ -22,20 +22,20 @@ _timing_function = get_timing_function() -class RobotContextTracker(SupportsTracking): - """Tracks and stores robot context and execution duration for different operations.""" +class RobotActivityTracker(SupportsTracking): + """Tracks and stores robot activity and execution duration for different operations.""" METADATA_NAME: typing.Final[ - typing.Literal["robot_context_data"] - ] = "robot_context_data" + typing.Literal["robot_activity_data"] + ] = "robot_activity_data" def __init__(self, storage_location: Path, should_track: bool) -> None: - """Initializes the RobotContextTracker with an empty storage list.""" - self._store = MetricsStore[RawContextData]( + """Initializes the RobotActivityTracker with an empty storage list.""" + self._store = MetricsStore[RawActivityData]( MetricsMetadata( name=self.METADATA_NAME, storage_dir=storage_location, - headers=RawContextData.headers(), + headers=RawActivityData.headers(), ) ) self._should_track = should_track @@ -45,7 +45,7 @@ def __init__(self, storage_location: Path, should_track: bool) -> None: def track( self, - state: RobotContextState, + state: RobotActivityState, ) -> typing.Callable[ [_UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn]], _UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn], @@ -56,7 +56,7 @@ def track( Args: func_to_track: The function to track. - state: The state of the robot context during the function execution. + state: The state of the robot activity during the function execution. *args: The arguments to pass to the function. **kwargs: The keyword arguments to pass to the function. @@ -90,7 +90,7 @@ async def async_wrapper( duration_end_time = perf_counter_ns() self._store.add( - RawContextData( + RawActivityData( func_start=function_start_time, duration=duration_end_time - duration_start_time, state=state, @@ -116,7 +116,7 @@ def wrapper( duration_end_time = perf_counter_ns() self._store.add( - RawContextData( + RawActivityData( func_start=function_start_time, duration=duration_end_time - duration_start_time, state=state, @@ -130,7 +130,7 @@ def wrapper( return inner_decorator def store(self) -> None: - """Returns the stored context data and clears the storage list.""" + """Returns the stored activity data and clears the storage list.""" if not self._should_track: return self._store.store() diff --git a/performance-metrics/src/performance_metrics/_types.py b/performance-metrics/src/performance_metrics/_types.py index 4e79123d016..353917d8feb 100644 --- a/performance-metrics/src/performance_metrics/_types.py +++ b/performance-metrics/src/performance_metrics/_types.py @@ -10,7 +10,7 @@ ] -RobotContextState = typing.Literal[ +RobotActivityState = typing.Literal[ "ANALYZING_PROTOCOL", "GETTING_CACHED_ANALYSIS", "RUNNING_PROTOCOL", @@ -21,7 +21,7 @@ class SupportsTracking(typing.Protocol): - """Protocol for classes that support tracking of robot context.""" + """Protocol for classes that support tracking of robot activity.""" def __init__(self, storage_location: Path, should_track: bool) -> None: """Initialize the tracker.""" @@ -29,7 +29,7 @@ def __init__(self, storage_location: Path, should_track: bool) -> None: def track( self, - state: "RobotContextState", + state: "RobotActivityState", ) -> typing.Callable[ [_UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn]], _UnderlyingFunction[_UnderlyingFunctionParameters, _UnderlyingFunctionReturn], @@ -43,21 +43,3 @@ def store(self) -> None: StorableData = typing.Union[int, float, str] - - -class SupportsCSVStorage(typing.Protocol): - """A protocol for classes that support CSV storage.""" - - @classmethod - def headers(self) -> typing.Tuple[str, ...]: - """Returns the headers for the CSV data.""" - ... - - def csv_row(self) -> typing.Tuple[StorableData, ...]: - """Returns the object as a CSV row.""" - ... - - @classmethod - def from_csv_row(cls, row: typing.Tuple[StorableData, ...]) -> "SupportsCSVStorage": - """Returns an object from a CSV row.""" - ... diff --git a/performance-metrics/tests/performance_metrics/test_data_shapes.py b/performance-metrics/tests/performance_metrics/test_data_shapes.py new file mode 100644 index 00000000000..c417b5ba6a3 --- /dev/null +++ b/performance-metrics/tests/performance_metrics/test_data_shapes.py @@ -0,0 +1,65 @@ +"""Tests for the data shapes.""" + +from performance_metrics._data_shapes import ProcessResourceUsageSnapshot + + +def test_headers_ordering() -> None: + """Tests that the headers are in the correct order.""" + assert ProcessResourceUsageSnapshot.headers() == ( + "query_time", + "command", + "running_since", + "user_cpu_time", + "system_cpu_time", + "memory_percent", + ) + + +def test_csv_row_method_ordering() -> None: + """Tests that the CSV row method returns the correct order.""" + expected = ( + 1, + "test", + 2, + 3, + 4, + 5, + ) + + assert ( + ProcessResourceUsageSnapshot( + query_time=1, + command="test", + running_since=2, + user_cpu_time=3, + system_cpu_time=4, + memory_percent=5, + ).csv_row() + == expected + ) + + assert ( + ProcessResourceUsageSnapshot( + command="test", + query_time=1, + user_cpu_time=3, + system_cpu_time=4, + running_since=2, + memory_percent=5, + ).csv_row() + == expected + ) + + assert ( + ProcessResourceUsageSnapshot.from_csv_row( + ( + 1, + "test", + 2, + 3, + 4, + 5, + ) + ).csv_row() + == expected + ) diff --git a/performance-metrics/tests/performance_metrics/test_metrics_store.py b/performance-metrics/tests/performance_metrics/test_metrics_store.py index 9e750a3820e..4adc42fba3d 100644 --- a/performance-metrics/tests/performance_metrics/test_metrics_store.py +++ b/performance-metrics/tests/performance_metrics/test_metrics_store.py @@ -3,8 +3,8 @@ from pathlib import Path from time import sleep -from performance_metrics._robot_context_tracker import RobotContextTracker -from performance_metrics._data_shapes import RawContextData +from performance_metrics._robot_activity_tracker import RobotActivityTracker +from performance_metrics._data_shapes import RawActivityData # Corrected times in seconds STARTING_TIME = 0.001 @@ -16,17 +16,17 @@ async def test_storing_to_file(tmp_path: Path) -> None: """Tests storing the tracked data to a file.""" - robot_context_tracker = RobotContextTracker(tmp_path, should_track=True) + robot_activity_tracker = RobotActivityTracker(tmp_path, should_track=True) - @robot_context_tracker.track("ROBOT_STARTING_UP") + @robot_activity_tracker.track("ROBOT_STARTING_UP") def starting_robot() -> None: sleep(STARTING_TIME) - @robot_context_tracker.track("CALIBRATING") + @robot_activity_tracker.track("CALIBRATING") async def calibrating_robot() -> None: sleep(CALIBRATING_TIME) - @robot_context_tracker.track("ANALYZING_PROTOCOL") + @robot_activity_tracker.track("ANALYZING_PROTOCOL") def analyzing_protocol() -> None: sleep(ANALYZING_TIME) @@ -34,9 +34,9 @@ def analyzing_protocol() -> None: await calibrating_robot() analyzing_protocol() - robot_context_tracker.store() + robot_activity_tracker.store() - with open(robot_context_tracker._store.metadata.data_file_location, "r") as file: + with open(robot_activity_tracker._store.metadata.data_file_location, "r") as file: lines = file.readlines() assert len(lines) == 3, "All stored data should be written to the file." @@ -44,10 +44,12 @@ def analyzing_protocol() -> None: line.replace('"', "").strip().split(",") for line in lines ] assert all( - RawContextData.from_csv_row(line) for line in split_lines - ), "All lines should be valid RawContextData instances." + RawActivityData.from_csv_row(line) for line in split_lines + ), "All lines should be valid RawActivityData instances." - with open(robot_context_tracker._store.metadata.headers_file_location, "r") as file: + with open( + robot_activity_tracker._store.metadata.headers_file_location, "r" + ) as file: headers = file.readlines() assert len(headers) == 1, "Header should be written to the headers file." - assert tuple(headers[0].strip().split(",")) == RawContextData.headers() + assert tuple(headers[0].strip().split(",")) == RawActivityData.headers() diff --git a/performance-metrics/tests/performance_metrics/test_robot_context_tracker.py b/performance-metrics/tests/performance_metrics/test_robot_context_tracker.py index 6ba81600263..c08439203bd 100644 --- a/performance-metrics/tests/performance_metrics/test_robot_context_tracker.py +++ b/performance-metrics/tests/performance_metrics/test_robot_context_tracker.py @@ -1,9 +1,9 @@ -"""Tests for the RobotContextTracker class in performance_metrics._robot_context_tracker.""" +"""Tests for the RobotActivityTracker class in performance_metrics._robot_activity_tracker.""" import asyncio from pathlib import Path import pytest -from performance_metrics._robot_context_tracker import RobotContextTracker +from performance_metrics._robot_activity_tracker import RobotActivityTracker from time import sleep, time_ns from unittest.mock import patch @@ -16,43 +16,43 @@ @pytest.fixture -def robot_context_tracker(tmp_path: Path) -> RobotContextTracker: - """Fixture to provide a fresh instance of RobotContextTracker for each test.""" - return RobotContextTracker(storage_location=tmp_path, should_track=True) +def robot_activity_tracker(tmp_path: Path) -> RobotActivityTracker: + """Fixture to provide a fresh instance of RobotActivityTracker for each test.""" + return RobotActivityTracker(storage_location=tmp_path, should_track=True) -async def test_robot_context_tracker( - robot_context_tracker: RobotContextTracker, +async def test_robot_activity_tracker( + robot_activity_tracker: RobotActivityTracker, ) -> None: - """Tests the tracking of various robot context states through RobotContextTracker.""" + """Tests the tracking of various robot activity states through RobotActivityTracker.""" - @robot_context_tracker.track(state="ROBOT_STARTING_UP") + @robot_activity_tracker.track(state="ROBOT_STARTING_UP") async def starting_robot() -> str: sleep(STARTING_TIME) return "Robot is starting up." - @robot_context_tracker.track(state="CALIBRATING") + @robot_activity_tracker.track(state="CALIBRATING") def calibrating_robot() -> None: sleep(CALIBRATING_TIME) - @robot_context_tracker.track(state="ANALYZING_PROTOCOL") + @robot_activity_tracker.track(state="ANALYZING_PROTOCOL") def analyzing_protocol() -> None: sleep(ANALYZING_TIME) - @robot_context_tracker.track(state="RUNNING_PROTOCOL") + @robot_activity_tracker.track(state="RUNNING_PROTOCOL") async def running_protocol(run_time: int) -> int: sleep(RUNNING_TIME) return run_time - @robot_context_tracker.track(state="ROBOT_SHUTTING_DOWN") + @robot_activity_tracker.track(state="ROBOT_SHUTTING_DOWN") def shutting_down_robot() -> str: sleep(SHUTTING_DOWN_TIME) return "Robot is shutting down." # Ensure storage is initially empty assert ( - len(robot_context_tracker._store._data) == 0 + len(robot_activity_tracker._store._data_store) == 0 ), "Storage should be initially empty." assert await starting_robot() == "Robot is starting up.", "Operation should return." @@ -64,7 +64,9 @@ def shutting_down_robot() -> str: ), "Operation should return." # Verify that all states were tracked - assert len(robot_context_tracker._store._data) == 5, "All states should be tracked." + assert ( + len(robot_activity_tracker._store._data_store) == 5 + ), "All states should be tracked." # Validate the sequence and accuracy of tracked states expected_states = [ @@ -76,44 +78,44 @@ def shutting_down_robot() -> str: ] for i, state in enumerate(expected_states): assert ( - robot_context_tracker._store._data[i].state == state + robot_activity_tracker._store._data_store[i].state == state ), f"State at index {i} should be {state}." async def test_multiple_operations_single_state( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: - """Tests tracking multiple operations within a single robot context state.""" + """Tests tracking multiple operations within a single robot activity state.""" async def first_operation() -> None: sleep(RUNNING_TIME) - @robot_context_tracker.track(state="RUNNING_PROTOCOL") + @robot_activity_tracker.track(state="RUNNING_PROTOCOL") def second_operation() -> None: sleep(RUNNING_TIME) - wrapped_first_operation = robot_context_tracker.track(state="RUNNING_PROTOCOL")( + wrapped_first_operation = robot_activity_tracker.track(state="RUNNING_PROTOCOL")( first_operation ) await wrapped_first_operation() second_operation() assert ( - len(robot_context_tracker._store._data) == 2 + len(robot_activity_tracker._store._data_store) == 2 ), "Both operations should be tracked." assert ( - robot_context_tracker._store._data[0].state - == robot_context_tracker._store._data[1].state + robot_activity_tracker._store._data_store[0].state + == robot_activity_tracker._store._data_store[1].state == "RUNNING_PROTOCOL" ), "Both operations should have the same state." async def test_exception_handling_in_tracked_function( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Ensures exceptions in tracked operations are handled correctly.""" - @robot_context_tracker.track(state="ROBOT_SHUTTING_DOWN") + @robot_activity_tracker.track(state="ROBOT_SHUTTING_DOWN") async def error_prone_operation() -> None: sleep(SHUTTING_DOWN_TIME) raise RuntimeError("Simulated operation failure") @@ -122,45 +124,45 @@ async def error_prone_operation() -> None: await error_prone_operation() assert ( - len(robot_context_tracker._store._data) == 1 + len(robot_activity_tracker._store._data_store) == 1 ), "Failed operation should still be tracked." assert ( - robot_context_tracker._store._data[0].state == "ROBOT_SHUTTING_DOWN" + robot_activity_tracker._store._data_store[0].state == "ROBOT_SHUTTING_DOWN" ), "State should be correctly logged despite the exception." @pytest.mark.asyncio async def test_async_operation_tracking( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Tests tracking of an asynchronous operation.""" - @robot_context_tracker.track(state="ANALYZING_PROTOCOL") + @robot_activity_tracker.track(state="ANALYZING_PROTOCOL") async def async_analyzing_operation() -> None: await asyncio.sleep(ANALYZING_TIME) await async_analyzing_operation() assert ( - len(robot_context_tracker._store._data) == 1 + len(robot_activity_tracker._store._data_store) == 1 ), "Async operation should be tracked." assert ( - robot_context_tracker._store._data[0].state == "ANALYZING_PROTOCOL" + robot_activity_tracker._store._data_store[0].state == "ANALYZING_PROTOCOL" ), "State should be ANALYZING_PROTOCOL." def test_sync_operation_timing_accuracy( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Tests the timing accuracy of a synchronous operation tracking.""" - @robot_context_tracker.track(state="RUNNING_PROTOCOL") + @robot_activity_tracker.track(state="RUNNING_PROTOCOL") def running_operation() -> None: sleep(RUNNING_TIME) running_operation() - duration_data = robot_context_tracker._store._data[0] + duration_data = robot_activity_tracker._store._data_store[0] assert ( abs(duration_data.duration - RUNNING_TIME * 1e9) < 1e7 ), "Measured duration for sync operation should closely match the expected duration." @@ -168,17 +170,17 @@ def running_operation() -> None: @pytest.mark.asyncio async def test_async_operation_timing_accuracy( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Tests the timing accuracy of an async operation tracking.""" - @robot_context_tracker.track(state="RUNNING_PROTOCOL") + @robot_activity_tracker.track(state="RUNNING_PROTOCOL") async def async_running_operation() -> None: await asyncio.sleep(RUNNING_TIME) await async_running_operation() - duration_data = robot_context_tracker._store._data[0] + duration_data = robot_activity_tracker._store._data_store[0] assert ( abs(duration_data.duration - RUNNING_TIME * 1e9) < 1e7 ), "Measured duration for async operation should closely match the expected duration." @@ -186,11 +188,11 @@ async def async_running_operation() -> None: @pytest.mark.asyncio async def test_exception_in_async_operation( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Ensures exceptions in tracked async operations are correctly handled.""" - @robot_context_tracker.track(state="ROBOT_SHUTTING_DOWN") + @robot_activity_tracker.track(state="ROBOT_SHUTTING_DOWN") async def async_error_prone_operation() -> None: await asyncio.sleep(SHUTTING_DOWN_TIME) raise RuntimeError("Simulated async operation failure") @@ -199,58 +201,59 @@ async def async_error_prone_operation() -> None: await async_error_prone_operation() assert ( - len(robot_context_tracker._store._data) == 1 + len(robot_activity_tracker._store._data_store) == 1 ), "Failed async operation should still be tracked." assert ( - robot_context_tracker._store._data[0].state == "ROBOT_SHUTTING_DOWN" + robot_activity_tracker._store._data_store[0].state == "ROBOT_SHUTTING_DOWN" ), "State should be ROBOT_SHUTTING_DOWN despite the exception." @pytest.mark.asyncio async def test_concurrent_async_operations( - robot_context_tracker: RobotContextTracker, + robot_activity_tracker: RobotActivityTracker, ) -> None: """Tests tracking of concurrent async operations.""" - @robot_context_tracker.track(state="CALIBRATING") + @robot_activity_tracker.track(state="CALIBRATING") async def first_async_calibrating() -> None: await asyncio.sleep(CALIBRATING_TIME) - @robot_context_tracker.track(state="CALIBRATING") + @robot_activity_tracker.track(state="CALIBRATING") async def second_async_calibrating() -> None: await asyncio.sleep(CALIBRATING_TIME) await asyncio.gather(first_async_calibrating(), second_async_calibrating()) assert ( - len(robot_context_tracker._store._data) == 2 + len(robot_activity_tracker._store._data_store) == 2 ), "Both concurrent async operations should be tracked." assert all( - data.state == "CALIBRATING" for data in robot_context_tracker._store._data + data.state == "CALIBRATING" + for data in robot_activity_tracker._store._data_store ), "All tracked operations should be in CALIBRATING state." def test_no_tracking(tmp_path: Path) -> None: """Tests that operations are not tracked when tracking is disabled.""" - robot_context_tracker = RobotContextTracker(tmp_path, should_track=False) + robot_activity_tracker = RobotActivityTracker(tmp_path, should_track=False) - @robot_context_tracker.track(state="ROBOT_STARTING_UP") + @robot_activity_tracker.track(state="ROBOT_STARTING_UP") def operation_without_tracking() -> None: sleep(STARTING_TIME) operation_without_tracking() assert ( - len(robot_context_tracker._store._data) == 0 + len(robot_activity_tracker._store._data_store) == 0 ), "Operation should not be tracked when tracking is disabled." @pytest.mark.asyncio async def test_async_exception_handling_when_not_tracking(tmp_path: Path) -> None: """Ensures exceptions in operations are still raised when tracking is disabled.""" - robot_context_tracker = RobotContextTracker(tmp_path, should_track=False) + robot_activity_tracker = RobotActivityTracker(tmp_path, should_track=False) - @robot_context_tracker.track(state="ROBOT_SHUTTING_DOWN") + @robot_activity_tracker.track(state="ROBOT_SHUTTING_DOWN") async def error_prone_operation() -> None: sleep(SHUTTING_DOWN_TIME) raise RuntimeError("Simulated operation failure") @@ -261,9 +264,9 @@ async def error_prone_operation() -> None: def test_sync_exception_handling_when_not_tracking(tmp_path: Path) -> None: """Ensures exceptions in operations are still raised when tracking is disabled.""" - robot_context_tracker = RobotContextTracker(tmp_path, should_track=False) + robot_activity_tracker = RobotActivityTracker(tmp_path, should_track=False) - @robot_context_tracker.track(state="ROBOT_SHUTTING_DOWN") + @robot_activity_tracker.track(state="ROBOT_SHUTTING_DOWN") def error_prone_operation() -> None: sleep(SHUTTING_DOWN_TIME) raise RuntimeError("Simulated operation failure") @@ -279,20 +282,20 @@ def error_prone_operation() -> None: def test_using_non_linux_time_functions(tmp_path: Path) -> None: """Tests tracking operations using non-Linux time functions.""" file_path = tmp_path / "test_file.csv" - robot_context_tracker = RobotContextTracker(file_path, should_track=True) + robot_activity_tracker = RobotActivityTracker(file_path, should_track=True) - @robot_context_tracker.track(state="ROBOT_STARTING_UP") + @robot_activity_tracker.track(state="ROBOT_STARTING_UP") def starting_robot() -> None: sleep(STARTING_TIME) - @robot_context_tracker.track(state="CALIBRATING") + @robot_activity_tracker.track(state="CALIBRATING") def calibrating_robot() -> None: sleep(CALIBRATING_TIME) starting_robot() calibrating_robot() - storage = robot_context_tracker._store._data + storage = robot_activity_tracker._store._data_store assert all( data.func_start > 0 for data in storage ), "All function start times should be greater than 0." diff --git a/protocol-designer/src/components/StepEditForm/fields/BlowoutZOffsetField.tsx b/protocol-designer/src/components/StepEditForm/fields/BlowoutZOffsetField.tsx index 631876ef442..453e86cee26 100644 --- a/protocol-designer/src/components/StepEditForm/fields/BlowoutZOffsetField.tsx +++ b/protocol-designer/src/components/StepEditForm/fields/BlowoutZOffsetField.tsx @@ -4,6 +4,7 @@ import { DEST_WELL_BLOWOUT_DESTINATION, SOURCE_WELL_BLOWOUT_DESTINATION, } from '@opentrons/step-generation' +import { getWellDepth } from '@opentrons/shared-data' import { COLORS, Flex, @@ -45,9 +46,9 @@ export function BlowoutZOffsetField( labwareId = destLabwareId } - const labwareZDimension = - labwareId != null - ? labwareEntities[String(labwareId)]?.def.dimensions.zDimension + const labwareWellDepth = + labwareId != null && labwareEntities[String(labwareId)]?.def != null + ? getWellDepth(labwareEntities[String(labwareId)].def, 'A1') : 0 return ( @@ -61,7 +62,7 @@ export function BlowoutZOffsetField( name={name} zValue={Number(value)} updateValue={updateValue} - wellDepthMm={labwareZDimension} + wellDepthMm={labwareWellDepth} /> ) : null} = { comment: 'comment', moveLabware: 'move-xy', @@ -103,7 +148,6 @@ export const stepIconsByType: Record = { mix: 'ot-mix', pause: 'pause', manualIntervention: 'pause', - // TODO Ian 2018-12-13 pause icon for this is a placeholder magnet: 'ot-magnet-v2', temperature: 'ot-temperature-v2', thermocycler: 'ot-thermocycler', @@ -124,14 +168,14 @@ export interface ChangeTipFields { export type MixForm = AnnotationFields & BlowoutFields & ChangeTipFields & { - stepType: 'mix' id: StepIdType + stepType: 'mix' labware?: string pipette?: string times?: string + touchTip?: boolean volume?: string wells?: string[] - touchTip?: boolean } export type PauseForm = AnnotationFields & { stepType: 'pause' @@ -176,153 +220,151 @@ export type BlankForm = AnnotationFields & { stepType: StepType id: StepIdType } -// TODO: Ian 2019-01-15 these types are a placeholder. Should be used in form hydration. -// TODO: this is the type we are aiming for + export interface HydratedMoveLiquidFormData { id: string stepType: 'moveLiquid' stepName: string - description: string | null | undefined fields: { - tipRack: string - pipette: PipetteEntity - volume: number - path: PathOption - changeTip: ChangeTipOptions - aspirate_wells_grouped: boolean | null | undefined - preWetTip: boolean | null | undefined + aspirate_airGap_checkbox: boolean + aspirate_delay_checkbox: boolean aspirate_labware: LabwareEntity - aspirate_wells: string[] + aspirate_mix_checkbox: boolean + aspirate_touchTip_checkbox: boolean aspirate_wellOrder_first: WellOrderOption aspirate_wellOrder_second: WellOrderOption - aspirate_flowRate: number | null | undefined - aspirate_mmFromBottom: number | null | undefined - aspirate_touchTip_checkbox: boolean - aspirate_touchTip_mmFromBottom: number | null | undefined - aspirate_mix_checkbox: boolean - aspirate_mix_volume: number | null | undefined - aspirate_mix_times: number | null | undefined - aspirate_airGap_checkbox: boolean - aspirate_airGap_volume: number | null | undefined - aspirate_delay_checkbox: boolean - aspirate_delay_seconds: number | null | undefined - aspirate_delay_mmFromBottom: number | null | undefined - // TODO(IL, 2020-09-30): when FF is removed, change to `dispense_airGap_checkbox: boolean` (no longer Maybe-typed) + aspirate_wells: string[] + blowout_checkbox: boolean + changeTip: ChangeTipOptions dispense_airGap_checkbox: boolean - dispense_airGap_volume: number | null | undefined dispense_delay_checkbox: boolean - dispense_delay_seconds: number | null | undefined - dispense_delay_mmFromBottom: number | null | undefined dispense_labware: LabwareEntity | AdditionalEquipmentEntity - dispense_wells: string[] + dispense_mix_checkbox: boolean + dispense_touchTip_checkbox: boolean dispense_wellOrder_first: WellOrderOption dispense_wellOrder_second: WellOrderOption - dispense_flowRate: number | null | undefined - dispense_mmFromBottom: number | null | undefined - dispense_touchTip_checkbox: boolean - dispense_touchTip_mmFromBottom: number | null | undefined - dispense_mix_checkbox: boolean - dispense_mix_volume: number | null | undefined - dispense_mix_times: number | null | undefined + dispense_wells: string[] disposalVolume_checkbox: boolean - disposalVolume_volume: number | null | undefined - blowout_checkbox: boolean - blowout_location: string | null | undefined // labwareId or 'SOURCE_WELL' or 'DEST_WELL' dropTip_location: string nozzles: NozzleConfigurationStyle | null + path: PathOption + pipette: PipetteEntity + tipRack: string + volume: number + aspirate_airGap_volume?: number | null + aspirate_delay_mmFromBottom?: number | null + aspirate_delay_seconds?: number | null + aspirate_flowRate?: number | null + aspirate_mix_times?: number | null + aspirate_mix_volume?: number | null + aspirate_mmFromBottom?: number | null + aspirate_touchTip_mmFromBottom?: number | null + aspirate_wells_grouped?: boolean | null aspirate_x_position?: number | null aspirate_y_position?: number | null + blowout_flowRate?: number | null + blowout_location?: string | null + blowout_z_offset?: number | null + dispense_airGap_volume?: number | null + dispense_delay_mmFromBottom?: number | null + dispense_delay_seconds?: number | null + dispense_flowRate?: number | null + dispense_mix_times?: number | null + dispense_mix_volume?: number | null + dispense_mmFromBottom?: number | null + dispense_touchTip_mmFromBottom?: number | null dispense_x_position?: number | null dispense_y_position?: number | null - blowout_z_offset?: number | null - blowout_flowRate?: number | null + disposalVolume_volume?: number | null + preWetTip?: boolean | null } + description?: string | null } export interface HydratedMoveLabwareFormData { id: string stepType: 'moveLabware' stepName: string - description: string | null | undefined fields: { labware: LabwareEntity newLocation: LabwareLocation useGripper: boolean } + description?: string | null } export interface HydratedCommentFormData { id: string stepType: 'comment' stepName: string - stepDetails?: string | null fields: { message: string } + stepDetails?: string | null } export interface HydratedMixFormDataLegacy { - id: string - stepType: 'mix' - stepName: string - tipRack: string - stepDetails: string | null | undefined - pipette: PipetteEntity - volume: number + aspirate_delay_checkbox: boolean + blowout_checkbox: boolean changeTip: ChangeTipOptions + dispense_delay_checkbox: boolean + dropTip_location: string + id: string labware: LabwareEntity - wells: string[] + mix_touchTip_checkbox: boolean mix_wellOrder_first: WellOrderOption mix_wellOrder_second: WellOrderOption - aspirate_flowRate: number | null | undefined - mix_mmFromBottom: number | null | undefined - mix_touchTip_checkbox: boolean - mix_touchTip_mmFromBottom: number | null | undefined - times: number | null | undefined - dispense_flowRate: number | null | undefined - blowout_checkbox: boolean - blowout_location: string | null | undefined // labwareId or 'SOURCE_WELL' or 'DEST_WELL' - aspirate_delay_checkbox: boolean - aspirate_delay_seconds: number | null | undefined - dispense_delay_checkbox: boolean - dispense_delay_seconds: number | null | undefined - dropTip_location: string nozzles: NozzleConfigurationStyle | null + pipette: PipetteEntity + stepName: string + stepType: 'mix' + tipRack: string + volume: number + wells: string[] + aspirate_delay_seconds?: number | null + aspirate_flowRate?: number | null + blowout_flowRate?: number | null + blowout_location?: string | null + blowout_z_offset?: number | null + dispense_delay_seconds?: number | null + dispense_flowRate?: number | null + mix_mmFromBottom?: number | null + mix_touchTip_mmFromBottom?: number | null mix_x_position?: number | null mix_y_position?: number | null - blowout_z_offset?: number | null - blowout_flowRate?: number | null + stepDetails?: string | null + times?: number | null } export type MagnetAction = 'engage' | 'disengage' export type HydratedMagnetFormData = AnnotationFields & { + engageHeight: string | null id: string - stepType: 'magnet' - stepDetails: string | null - moduleId: string | null magnetAction: MagnetAction - engageHeight: string | null + moduleId: string | null + stepDetails: string | null + stepType: 'magnet' } export interface HydratedTemperatureFormData { id: string - stepType: 'temperature' - stepDetails: string | null moduleId: string | null setTemperature: 'true' | 'false' + stepDetails: string | null + stepType: 'temperature' targetTemperature: string | null } export interface HydratedHeaterShakerFormData { + heaterShakerSetTimer: 'true' | 'false' | null + heaterShakerTimerMinutes: string | null + heaterShakerTimerSeconds: string | null id: string - stepType: 'heaterShaker' - stepDetails: string | null + latchOpen: boolean moduleId: string - heaterShakerSetTimer: 'true' | 'false' | null setHeaterShakerTemperature: boolean setShake: boolean - latchOpen: boolean + stepDetails: string | null + stepType: 'heaterShaker' targetHeaterShakerTemperature: string | null targetSpeed: string | null - heaterShakerTimerMinutes: string | null - heaterShakerTimerSeconds: string | null } // TODO: Ian 2019-01-17 Moving away from this and towards nesting all form fields // inside `fields` key, but deprecating transfer/consolidate/distribute is a pre-req @@ -355,9 +397,11 @@ export type TipXOffsetFields = export type DelayCheckboxFields = | 'aspirate_delay_checkbox' | 'dispense_delay_checkbox' + export type DelaySecondFields = | 'aspirate_delay_seconds' | 'dispense_delay_seconds' + export function getIsTouchTipField(fieldName: StepFieldName): boolean { const touchTipFields = [ 'aspirate_touchTip_mmFromBottom', @@ -366,6 +410,7 @@ export function getIsTouchTipField(fieldName: StepFieldName): boolean { ] return touchTipFields.includes(fieldName) } + export function getIsDelayPositionField(fieldName: string): boolean { const delayPositionFields = [ 'aspirate_delay_mmFromBottom', diff --git a/protocol-designer/src/steplist/formLevel/getDefaultsForStepType.ts b/protocol-designer/src/steplist/formLevel/getDefaultsForStepType.ts index 915757e48cd..e175cea4fc7 100644 --- a/protocol-designer/src/steplist/formLevel/getDefaultsForStepType.ts +++ b/protocol-designer/src/steplist/formLevel/getDefaultsForStepType.ts @@ -13,90 +13,90 @@ export function getDefaultsForStepType( switch (stepType) { case 'mix': return { - times: null, - changeTip: DEFAULT_CHANGE_TIP_OPTION, - labware: null, - mix_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, - mix_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, - blowout_checkbox: false, - blowout_location: null, - // NOTE(IL, 2021-03-12): mix uses dispense for both asp + disp, unless its falsey. // For now, unlike the other mmFromBottom fields, it's initializing to a constant instead of + // NOTE(IL, 2021-03-12): mix uses dispense for both asp + disp, unless its falsey. // using null to represent default (because null becomes 1mm asp, 0.5mm dispense -- see #7470.) - mix_mmFromBottom: DEFAULT_MM_FROM_BOTTOM_DISPENSE, - pipette: null, - volume: undefined, - wells: [], - aspirate_flowRate: null, - dispense_flowRate: null, aspirate_delay_checkbox: false, aspirate_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, + aspirate_flowRate: null, + blowout_checkbox: false, + blowout_flowRate: null, + blowout_location: null, + blowout_z_offset: DEFAULT_MM_BLOWOUT_OFFSET_FROM_TOP, + changeTip: DEFAULT_CHANGE_TIP_OPTION, dispense_delay_checkbox: false, dispense_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, + dispense_flowRate: null, + dropTip_location: null, + labware: null, + mix_mmFromBottom: DEFAULT_MM_FROM_BOTTOM_DISPENSE, mix_touchTip_checkbox: false, mix_touchTip_mmFromBottom: null, - dropTip_location: null, - nozzles: null, - tipRack: null, + mix_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, + mix_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, mix_x_position: 0, mix_y_position: 0, - blowout_z_offset: DEFAULT_MM_BLOWOUT_OFFSET_FROM_TOP, - blowout_flowRate: null, + nozzles: null, + pipette: null, + times: null, + tipRack: null, + volume: undefined, + wells: [], } case 'moveLiquid': return { - pipette: null, - volume: null, - tipRack: null, - changeTip: DEFAULT_CHANGE_TIP_OPTION, - path: 'single', - aspirate_wells_grouped: false, + aspirate_airGap_checkbox: false, + aspirate_airGap_volume: null, + aspirate_delay_checkbox: false, + aspirate_delay_mmFromBottom: null, + aspirate_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, aspirate_flowRate: null, aspirate_labware: null, - aspirate_wells: [], - aspirate_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, - aspirate_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, aspirate_mix_checkbox: false, aspirate_mix_times: null, aspirate_mix_volume: null, aspirate_mmFromBottom: null, aspirate_touchTip_checkbox: false, aspirate_touchTip_mmFromBottom: null, + aspirate_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, + aspirate_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, + aspirate_wells_grouped: false, + aspirate_wells: [], + aspirate_x_position: 0, + aspirate_y_position: 0, + blowout_checkbox: false, + blowout_flowRate: null, + blowout_location: null, + blowout_z_offset: DEFAULT_MM_BLOWOUT_OFFSET_FROM_TOP, + changeTip: DEFAULT_CHANGE_TIP_OPTION, + dispense_airGap_checkbox: false, + dispense_airGap_volume: null, + dispense_delay_checkbox: false, + dispense_delay_mmFromBottom: null, + dispense_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, dispense_flowRate: null, dispense_labware: null, - dispense_wells: [], - dispense_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, - dispense_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, dispense_mix_checkbox: false, dispense_mix_times: null, dispense_mix_volume: null, dispense_mmFromBottom: null, dispense_touchTip_checkbox: false, dispense_touchTip_mmFromBottom: null, + dispense_wellOrder_first: DEFAULT_WELL_ORDER_FIRST_OPTION, + dispense_wellOrder_second: DEFAULT_WELL_ORDER_SECOND_OPTION, + dispense_wells: [], + dispense_x_position: 0, + dispense_y_position: 0, disposalVolume_checkbox: false, disposalVolume_volume: null, - blowout_checkbox: false, - blowout_location: null, - preWetTip: false, - aspirate_airGap_checkbox: false, - aspirate_airGap_volume: null, - aspirate_delay_checkbox: false, - aspirate_delay_mmFromBottom: null, - aspirate_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, - dispense_airGap_checkbox: false, - dispense_airGap_volume: null, - dispense_delay_checkbox: false, - dispense_delay_seconds: `${DEFAULT_DELAY_SECONDS}`, - dispense_delay_mmFromBottom: null, dropTip_location: null, nozzles: null, - dispense_x_position: 0, - dispense_y_position: 0, - aspirate_x_position: 0, - aspirate_y_position: 0, - blowout_z_offset: DEFAULT_MM_BLOWOUT_OFFSET_FROM_TOP, - blowout_flowRate: null, + path: 'single', + pipette: null, + preWetTip: false, + tipRack: null, + volume: null, } case 'comment': @@ -106,33 +106,33 @@ export function getDefaultsForStepType( case 'moveLabware': return { labware: null, - useGripper: false, newLocation: null, + useGripper: false, } case 'pause': return { + moduleId: null, pauseAction: null, pauseHour: null, + pauseMessage: '', pauseMinute: null, pauseSecond: null, - pauseMessage: '', - moduleId: null, pauseTemperature: null, } case 'manualIntervention': return { labwareLocationUpdate: {}, - pipetteLocationUpdate: {}, moduleLocationUpdate: {}, + pipetteLocationUpdate: {}, } case 'magnet': return { - moduleId: null, - magnetAction: null, engageHeight: null, + magnetAction: null, + moduleId: null, } case 'temperature': @@ -143,34 +143,34 @@ export function getDefaultsForStepType( } case 'heaterShaker': return { + heaterShakerSetTimer: null, + heaterShakerTimerMinutes: null, + heaterShakerTimerSeconds: null, + latchOpen: false, moduleId: null, setHeaterShakerTemperature: null, + setShake: null, targetHeaterShakerTemperature: null, targetSpeed: null, - setShake: null, - latchOpen: false, - heaterShakerSetTimer: null, - heaterShakerTimerMinutes: null, - heaterShakerTimerSeconds: null, } case 'thermocycler': return { - thermocyclerFormType: null, - moduleId: null, blockIsActive: false, + blockIsActiveHold: false, blockTargetTemp: null, + blockTargetTempHold: null, lidIsActive: false, - lidTargetTemp: null, + lidIsActiveHold: false, lidOpen: false, - profileVolume: null, - profileTargetLidTemp: null, + lidOpenHold: null, + lidTargetTemp: null, + lidTargetTempHold: null, + moduleId: null, orderedProfileItems: [], profileItemsById: {}, - blockIsActiveHold: false, - blockTargetTempHold: null, - lidIsActiveHold: false, - lidTargetTempHold: null, - lidOpenHold: null, + profileTargetLidTemp: null, + profileVolume: null, + thermocyclerFormType: null, } default: diff --git a/react-api-client/src/client_data/index.ts b/react-api-client/src/client_data/index.ts new file mode 100644 index 00000000000..c41aa391dd0 --- /dev/null +++ b/react-api-client/src/client_data/index.ts @@ -0,0 +1,4 @@ +export { useClientData } from './useClientData' +export { useUpdateClientData } from './useUpdateClientData' + +export type * from './useUpdateClientData' diff --git a/react-api-client/src/client_data/useClientData.ts b/react-api-client/src/client_data/useClientData.ts new file mode 100644 index 00000000000..bf8c72105ad --- /dev/null +++ b/react-api-client/src/client_data/useClientData.ts @@ -0,0 +1,27 @@ +import { useQuery } from 'react-query' +import { getClientData } from '@opentrons/api-client' + +import { useHost } from '../api' + +import type { UseQueryOptions, UseQueryResult } from 'react-query' +import type { AxiosError } from 'axios' +import type { + ClientDataResponse, + HostConfig, + DefaultClientData, +} from '@opentrons/api-client' + +export function useClientData( + key: string, + options: UseQueryOptions, AxiosError> = {} +): UseQueryResult, AxiosError> { + const host = useHost() + const query = useQuery, AxiosError>( + [host, 'client_data', key], + () => + getClientData(host as HostConfig, key).then(response => response.data), + { enabled: host !== null, ...options } + ) + + return query +} diff --git a/react-api-client/src/client_data/useUpdateClientData.ts b/react-api-client/src/client_data/useUpdateClientData.ts new file mode 100644 index 00000000000..e1e9695b4cb --- /dev/null +++ b/react-api-client/src/client_data/useUpdateClientData.ts @@ -0,0 +1,48 @@ +import { useMutation } from 'react-query' +import { updateClientData } from '@opentrons/api-client' +import { useHost } from '../api' + +import type { AxiosError } from 'axios' +import type { + UseMutationResult, + UseMutationOptions, + UseMutateFunction, +} from 'react-query' +import type { + ClientDataResponse, + DefaultClientData, + HostConfig, +} from '@opentrons/api-client' + +export type UseUpdateClientDataMutationResult< + T = DefaultClientData +> = UseMutationResult, AxiosError, T> & { + updateClientData: UseMutateFunction, AxiosError, T> +} + +export type UseUpdateClientDataMutationOptions< + T = DefaultClientData +> = UseMutationOptions, AxiosError, T> + +export function useUpdateClientData( + key: string, + options: UseUpdateClientDataMutationOptions = {} +): UseUpdateClientDataMutationResult { + const host = useHost() + + const mutation = useMutation, AxiosError, T>( + [host, 'client_data', key], + (clientData: T) => + updateClientData(host as HostConfig, key, clientData) + .then(response => response.data) + .catch(e => { + throw e + }), + options + ) + + return { + ...mutation, + updateClientData: mutation.mutate, + } +} diff --git a/react-api-client/src/dataFiles/__tests__/useCsvFileRawQuery.test.tsx b/react-api-client/src/dataFiles/__tests__/useCsvFileRawQuery.test.tsx new file mode 100644 index 00000000000..e24ff3d6a82 --- /dev/null +++ b/react-api-client/src/dataFiles/__tests__/useCsvFileRawQuery.test.tsx @@ -0,0 +1,70 @@ +import * as React from 'react' +import { describe, it, expect, beforeEach, vi } from 'vitest' +import { QueryClient, QueryClientProvider } from 'react-query' +import { renderHook, waitFor } from '@testing-library/react' +import { getCsvFileRaw } from '@opentrons/api-client' +import { useHost } from '../../api' +import { useCsvFileRawQuery } from '..' + +import type { + HostConfig, + Response, + DownloadedCsvFileResponse, +} from '@opentrons/api-client' + +vi.mock('@opentrons/api-client') +vi.mock('../../api/useHost') + +const HOST_CONFIG: HostConfig = { hostname: 'localhost' } +const FILE_ID = 'file123' +const FILE_CONTENT_RESPONSE = 'content,of,my,csv\nfile,' as DownloadedCsvFileResponse + +describe('useCsvFileRawQuery hook', () => { + let wrapper: React.FunctionComponent<{ children: React.ReactNode }> + + beforeEach(() => { + const queryClient = new QueryClient() + const clientProvider: React.FunctionComponent<{ + children: React.ReactNode + }> = ({ children }) => ( + {children} + ) + + wrapper = clientProvider + }) + + it('should return no data if no host', () => { + vi.mocked(useHost).mockReturnValue(null) + + const { result } = renderHook(() => useCsvFileRawQuery(FILE_ID), { + wrapper, + }) + + expect(result.current.data).toBeUndefined() + }) + + it('should return no data if the get file request fails', () => { + vi.mocked(useHost).mockReturnValue(HOST_CONFIG) + vi.mocked(getCsvFileRaw).mockRejectedValue('oh no') + + const { result } = renderHook(() => useCsvFileRawQuery(FILE_ID), { + wrapper, + }) + expect(result.current.data).toBeUndefined() + }) + + it('should return file data if successful request', async () => { + vi.mocked(useHost).mockReturnValue(HOST_CONFIG) + vi.mocked(getCsvFileRaw).mockResolvedValue({ + data: FILE_CONTENT_RESPONSE, + } as Response) + + const { result } = renderHook(() => useCsvFileRawQuery(FILE_ID), { + wrapper, + }) + + await waitFor(() => { + expect(result.current.data).toEqual(FILE_CONTENT_RESPONSE) + }) + }) +}) diff --git a/react-api-client/src/dataFiles/index.ts b/react-api-client/src/dataFiles/index.ts index 3ff92db8497..cd6fe47daf0 100644 --- a/react-api-client/src/dataFiles/index.ts +++ b/react-api-client/src/dataFiles/index.ts @@ -1 +1,2 @@ +export { useCsvFileRawQuery } from './useCsvFileRawQuery' export { useUploadCsvFileMutation } from './useUploadCsvFileMutation' diff --git a/react-api-client/src/dataFiles/useCsvFileRawQuery.ts b/react-api-client/src/dataFiles/useCsvFileRawQuery.ts new file mode 100644 index 00000000000..22cae3ad920 --- /dev/null +++ b/react-api-client/src/dataFiles/useCsvFileRawQuery.ts @@ -0,0 +1,28 @@ +import { useQuery } from 'react-query' +import { getCsvFileRaw } from '@opentrons/api-client' +import { useHost } from '../api' + +import type { UseQueryOptions, UseQueryResult } from 'react-query' +import type { + HostConfig, + DownloadedCsvFileResponse, +} from '@opentrons/api-client' + +export function useCsvFileRawQuery( + fileId: string, + options?: UseQueryOptions +): UseQueryResult { + const host = useHost() + const allOptions: UseQueryOptions = { + ...options, + enabled: host !== null && fileId !== null, + } + + const query = useQuery( + [host, `/dataFiles/${fileId}/download`], + () => + getCsvFileRaw(host as HostConfig, fileId).then(response => response.data), + allOptions + ) + return query +} diff --git a/react-api-client/src/index.ts b/react-api-client/src/index.ts index 6bfb45c7eb9..fbfd11ad355 100644 --- a/react-api-client/src/index.ts +++ b/react-api-client/src/index.ts @@ -16,3 +16,4 @@ export * from './server' export * from './sessions' export * from './subsystems' export * from './system' +export * from './client_data' diff --git a/react-api-client/src/runs/index.ts b/react-api-client/src/runs/index.ts index 207950738e1..72a087d1529 100644 --- a/react-api-client/src/runs/index.ts +++ b/react-api-client/src/runs/index.ts @@ -15,6 +15,7 @@ export { useAllCommandsAsPreSerializedList } from './useAllCommandsAsPreSerializ export { useCommandQuery } from './useCommandQuery' export * from './useCreateLabwareOffsetMutation' export * from './useCreateLabwareDefinitionMutation' +export * from './useUpdateErrorRecoveryPolicy' export type { UsePlayRunMutationResult } from './usePlayRunMutation' export type { UsePauseRunMutationResult } from './usePauseRunMutation' diff --git a/react-api-client/src/runs/useDismissCurrentRunMutation.ts b/react-api-client/src/runs/useDismissCurrentRunMutation.ts index 1212317d563..5fdcc56fd3d 100644 --- a/react-api-client/src/runs/useDismissCurrentRunMutation.ts +++ b/react-api-client/src/runs/useDismissCurrentRunMutation.ts @@ -22,7 +22,9 @@ export type UseDismissCurrentRunMutationOptions = UseMutationOptions< string > -export function useDismissCurrentRunMutation(): UseDismissCurrentRunMutationResult { +export function useDismissCurrentRunMutation( + options: UseDismissCurrentRunMutationOptions = {} +): UseDismissCurrentRunMutationResult { const host = useHost() const queryClient = useQueryClient() @@ -34,7 +36,8 @@ export function useDismissCurrentRunMutation(): UseDismissCurrentRunMutationResu console.error(`error invalidating runs query: ${e.message}`) }) return response.data - }) + }), + options ) return { diff --git a/react-api-client/src/runs/useRunQuery.ts b/react-api-client/src/runs/useRunQuery.ts index 9cf74cb2429..4cb231eb5df 100644 --- a/react-api-client/src/runs/useRunQuery.ts +++ b/react-api-client/src/runs/useRunQuery.ts @@ -7,9 +7,12 @@ import type { HostConfig, Run } from '@opentrons/api-client' export function useRunQuery( runId: string | null, - options: UseQueryOptions = {} + options: UseQueryOptions = {}, + hostOverride?: HostConfig | null ): UseQueryResult { - const host = useHost() + const contextHost = useHost() + const host = + hostOverride != null ? { ...contextHost, ...hostOverride } : contextHost const query = useQuery( [host, 'runs', runId, 'details'], () => diff --git a/react-api-client/src/runs/useUpdateErrorRecoveryPolicy.ts b/react-api-client/src/runs/useUpdateErrorRecoveryPolicy.ts new file mode 100644 index 00000000000..1fa379b1bc5 --- /dev/null +++ b/react-api-client/src/runs/useUpdateErrorRecoveryPolicy.ts @@ -0,0 +1,62 @@ +import { useMutation } from 'react-query' + +import { updateErrorRecoveryPolicy } from '@opentrons/api-client' + +import { useHost } from '../api' + +import type { + UseMutationOptions, + UseMutationResult, + UseMutateFunction, +} from 'react-query' +import type { AxiosError } from 'axios' +import type { + RecoveryPolicyRulesParams, + UpdateErrorRecoveryPolicyResponse, + HostConfig, +} from '@opentrons/api-client' + +export type UseUpdateErrorRecoveryPolicyResponse = UseMutationResult< + UpdateErrorRecoveryPolicyResponse, + AxiosError, + RecoveryPolicyRulesParams +> & { + updateErrorRecoveryPolicy: UseMutateFunction< + UpdateErrorRecoveryPolicyResponse, + AxiosError, + RecoveryPolicyRulesParams + > +} + +export type UseUpdateErrorRecoveryPolicyOptions = UseMutationOptions< + UpdateErrorRecoveryPolicyResponse, + AxiosError, + RecoveryPolicyRulesParams +> + +export function useUpdateErrorRecoveryPolicy( + runId: string, + options: UseUpdateErrorRecoveryPolicyOptions = {} +): UseUpdateErrorRecoveryPolicyResponse { + const host = useHost() + + const mutation = useMutation< + UpdateErrorRecoveryPolicyResponse, + AxiosError, + RecoveryPolicyRulesParams + >( + [host, 'runs', runId, 'errorRecoveryPolicy'], + (policyRules: RecoveryPolicyRulesParams) => + updateErrorRecoveryPolicy(host as HostConfig, runId, policyRules) + .then(response => response.data) + .catch(e => { + throw e + }), + options + ) + + return { + ...mutation, + updateErrorRecoveryPolicy: mutation.mutate, + } +} diff --git a/robot-server/robot_server/client_data/__init__.py b/robot-server/robot_server/client_data/__init__.py new file mode 100644 index 00000000000..aa0df22b844 --- /dev/null +++ b/robot-server/robot_server/client_data/__init__.py @@ -0,0 +1 @@ +"""Support for the `/clientData` endpoints.""" diff --git a/robot-server/robot_server/client_data/router.py b/robot-server/robot_server/client_data/router.py new file mode 100644 index 00000000000..d04eaeea9ad --- /dev/null +++ b/robot-server/robot_server/client_data/router.py @@ -0,0 +1,146 @@ +"""Endpoint functions for the `/clientData` endpoints.""" + +import textwrap +from typing import Annotated, Literal + +import fastapi + +from robot_server.client_data.store import ( + ClientData, + ClientDataStore, + get_client_data_store, +) +from robot_server.errors.error_responses import ErrorBody, ErrorDetails +from robot_server.service.json_api.request import RequestModel +from robot_server.service.json_api.response import SimpleBody, SimpleEmptyBody +from robot_server.service.notifications.publishers.client_data_publisher import ( + ClientDataPublisher, + get_client_data_publisher, +) + +router = fastapi.APIRouter() + + +Key = Annotated[ + str, + fastapi.Path( + regex="^[a-zA-Z0-9-_]*$", + description=( + "A key for storing and retrieving the piece of data." + " This should be chosen to avoid colliding with other clients," + " and to unambiguously identify the data stored inside." + " The allowed characters are restricted to avoid any that" + " are special in URLs or MQTT topics." + ), + examples=["exampleOrganization-userNotes-v2"], + ), +] + + +class ClientDataKeyDoesNotExist(ErrorDetails): + """An error returned if trying to access a client data key that doesn't exist.""" + + id: Literal["ClientDataKeyDoesNotExist"] = "ClientDataKeyDoesNotExist" + title: str = "Client Data Key Does Not Exist" + + +@router.put( + path="/clientData/{key}", + summary="Store client-defined data", + description=textwrap.dedent( + """\ + Store a small amount of arbitrary client-defined data. + + This endpoint is experimental and may be changed or removed without warning. + + This is intended to help coordinate between multiple clients accessing the same + robot, and to help clients pick up from where they left off if they're closed + and reopened. For example, suppose your client shows a user interface for + physically setting up the deck with labware, step by step. You could use this + to store which step the user is currently on. + + The data is cleared when the robot reboots. + """ + ), +) +async def put_client_data( # noqa: D103 + key: Key, + request_body: RequestModel[ClientData], + store: Annotated[ClientDataStore, fastapi.Depends(get_client_data_store)], + client_data_publisher: Annotated[ + ClientDataPublisher, fastapi.Depends(get_client_data_publisher) + ], +) -> SimpleBody[ClientData]: + store.put(key, request_body.data) + await client_data_publisher.publish_client_data(key) + return SimpleBody.construct(data=store.get(key)) + + +@router.get( + path="/clientData/{key}", + summary="Get client-defined data", + description="Return the currently-stored client data at the given key. See `PUT /clientData` for background.", + responses={ + fastapi.status.HTTP_200_OK: {"model": SimpleBody[ClientData]}, + fastapi.status.HTTP_404_NOT_FOUND: { + "model": ErrorBody[ClientDataKeyDoesNotExist] + }, + }, +) +async def get_client_data( # noqa: D103 + key: Key, + store: ClientDataStore = fastapi.Depends(get_client_data_store), +) -> SimpleBody[ClientData]: + try: + return SimpleBody.construct(data=store.get(key)) + except KeyError as e: + raise ClientDataKeyDoesNotExist.from_exc(e).as_error( + fastapi.status.HTTP_404_NOT_FOUND + ) from e + + +@router.delete( + path="/clientData/{key}", + summary="Delete client-defined data", + description="Delete the client-defined data at the given key. See `PUT /clientData` for background.", + responses={ + fastapi.status.HTTP_200_OK: {"model": SimpleBody[ClientData]}, + fastapi.status.HTTP_404_NOT_FOUND: { + "model": ErrorBody[ClientDataKeyDoesNotExist] + }, + }, +) +async def delete_client_data( # noqa: D103 + key: Key, + store: Annotated[ClientDataStore, fastapi.Depends(get_client_data_store)], + client_data_publisher: Annotated[ + ClientDataPublisher, fastapi.Depends(get_client_data_publisher) + ], +) -> SimpleEmptyBody: + try: + store.delete(key) + except KeyError as e: + raise ClientDataKeyDoesNotExist.from_exc(e).as_error( + fastapi.status.HTTP_404_NOT_FOUND + ) from e + else: + await client_data_publisher.publish_client_data(key) + return SimpleEmptyBody.construct() + + +@router.delete( + path="/clientData", + summary="Delete all client-defined data", + description="Delete all client-defined data. See `PUT /clientData` for background.", +) +async def delete_all_client_data( # noqa: D103 + store: Annotated[ClientDataStore, fastapi.Depends(get_client_data_store)], + client_data_publisher: Annotated[ + ClientDataPublisher, fastapi.Depends(get_client_data_publisher) + ], +) -> SimpleEmptyBody: + keys_that_will_be_deleted = store.get_keys() + store.delete_all() + for deleted_key in keys_that_will_be_deleted: + await client_data_publisher.publish_client_data(deleted_key) + return SimpleEmptyBody.construct() diff --git a/robot-server/robot_server/client_data/store.py b/robot-server/robot_server/client_data/store.py new file mode 100644 index 00000000000..2fcd90e8feb --- /dev/null +++ b/robot-server/robot_server/client_data/store.py @@ -0,0 +1,59 @@ +"""An in-memory store for arbitrary client-defined JSON objects.""" + +import fastapi + +from server_utils.fastapi_utils.app_state import ( + AppState, + AppStateAccessor, + get_app_state, +) + + +ClientData = dict[str, object] + + +class ClientDataStore: + """An in-memory store for client-defined JSON objects.""" + + def __init__(self) -> None: + self._current_data: dict[str, ClientData] = {} + + def put(self, key: str, new_data: ClientData) -> None: + """Store new data at the given key, replacing any data that already exists.""" + self._current_data[key] = new_data + + def get(self, key: str) -> ClientData: + """Return the currently-stored data. + + If the given key has no data, raise `KeyError`. + """ + return self._current_data[key] + + def get_keys(self) -> list[str]: + """Return the keys that currently have data stored.""" + return list(self._current_data.keys()) + + def delete(self, key: str) -> None: + """Delete the data at the given key. + + If the given key has no data, raise `KeyError`. + """ + del self._current_data[key] + + def delete_all(self) -> None: + """Delete all data from the store.""" + self._current_data.clear() + + +_app_state_accessor = AppStateAccessor[ClientDataStore]("client_data_store") + + +async def get_client_data_store( + app_state: AppState = fastapi.Depends(get_app_state), +) -> ClientDataStore: + """A FastAPI dependency to return the server's singleton `ClientDataStore`.""" + store = _app_state_accessor.get_from(app_state) + if store is None: + store = ClientDataStore() + _app_state_accessor.set_on(app_state, store) + return store diff --git a/robot-server/robot_server/deck_configuration/router.py b/robot-server/robot_server/deck_configuration/router.py index 6e9d68d9f1b..f458d1af194 100644 --- a/robot-server/robot_server/deck_configuration/router.py +++ b/robot-server/robot_server/deck_configuration/router.py @@ -7,7 +7,7 @@ import fastapi from starlette.status import HTTP_422_UNPROCESSABLE_ENTITY -from opentrons_shared_data.deck.dev_types import DeckDefinitionV5 +from opentrons_shared_data.deck.types import DeckDefinitionV5 from robot_server.errors.error_responses import ErrorBody from robot_server.hardware import get_deck_definition diff --git a/robot-server/robot_server/deck_configuration/validation.py b/robot-server/robot_server/deck_configuration/validation.py index a3c043f8f51..51e4a50dc45 100644 --- a/robot-server/robot_server/deck_configuration/validation.py +++ b/robot-server/robot_server/deck_configuration/validation.py @@ -5,7 +5,7 @@ from dataclasses import dataclass from typing import DefaultDict, FrozenSet, List, Set, Tuple, Union, Optional -from opentrons_shared_data.deck import dev_types as deck_types +from opentrons_shared_data.deck import types as deck_types @dataclass(frozen=True) diff --git a/robot-server/robot_server/hardware.py b/robot-server/robot_server/hardware.py index 2994248a302..039f727ce87 100644 --- a/robot-server/robot_server/hardware.py +++ b/robot-server/robot_server/hardware.py @@ -18,7 +18,7 @@ from contextlib import contextmanager, suppress from opentrons_shared_data import deck -from opentrons_shared_data.robot.dev_types import RobotType, RobotTypeEnum +from opentrons_shared_data.robot.types import RobotType, RobotTypeEnum from opentrons import initialize as initialize_api, should_use_ot3 from opentrons.config import ( @@ -381,7 +381,7 @@ async def get_deck_type() -> DeckType: async def get_deck_definition( deck_type: DeckType = Depends(get_deck_type), -) -> deck.dev_types.DeckDefinitionV5: +) -> deck.types.DeckDefinitionV5: """Return this robot's deck definition.""" return deck.load(deck_type, version=5) diff --git a/robot-server/robot_server/health/models.py b/robot-server/robot_server/health/models.py index 9f886ca8f2f..ce8a0c2a56f 100644 --- a/robot-server/robot_server/health/models.py +++ b/robot-server/robot_server/health/models.py @@ -1,7 +1,7 @@ """HTTP request and response models for /health endpoints.""" import typing from pydantic import BaseModel, Field -from opentrons_shared_data.deck.dev_types import RobotModel +from opentrons_shared_data.deck.types import RobotModel from robot_server.service.json_api import BaseResponseBody diff --git a/robot-server/robot_server/health/router.py b/robot-server/robot_server/health/router.py index 9d9572bfc9b..92cdfd7cd63 100644 --- a/robot-server/robot_server/health/router.py +++ b/robot-server/robot_server/health/router.py @@ -16,7 +16,7 @@ ) from robot_server.service.legacy.models import V1BasicResponse -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from .models import Health, HealthLinks diff --git a/robot-server/robot_server/instruments/instrument_models.py b/robot-server/robot_server/instruments/instrument_models.py index 78bdd918938..3bd9885d26d 100644 --- a/robot-server/robot_server/instruments/instrument_models.py +++ b/robot-server/robot_server/instruments/instrument_models.py @@ -10,7 +10,7 @@ from opentrons.calibration_storage.types import SourceType from opentrons.protocol_engine.types import Vec3f -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( PipetteName, PipetteModel, ChannelCount, diff --git a/robot-server/robot_server/maintenance_runs/dependencies.py b/robot-server/robot_server/maintenance_runs/dependencies.py index 313fe71f1a1..dda7db0d0e0 100644 --- a/robot-server/robot_server/maintenance_runs/dependencies.py +++ b/robot-server/robot_server/maintenance_runs/dependencies.py @@ -1,7 +1,7 @@ """Maintenance Run router dependency-injection wire-up.""" from fastapi import Depends -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.hardware_control import HardwareControlAPI from opentrons.protocol_engine import DeckType diff --git a/robot-server/robot_server/maintenance_runs/maintenance_run_orchestrator_store.py b/robot-server/robot_server/maintenance_runs/maintenance_run_orchestrator_store.py index 4f63e67418b..169875d4b7d 100644 --- a/robot-server/robot_server/maintenance_runs/maintenance_run_orchestrator_store.py +++ b/robot-server/robot_server/maintenance_runs/maintenance_run_orchestrator_store.py @@ -31,8 +31,8 @@ HardwareEventHandler, ) -from opentrons_shared_data.robot.dev_types import RobotType, RobotTypeEnum -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.robot.types import RobotType, RobotTypeEnum +from opentrons_shared_data.labware.types import LabwareUri from opentrons_shared_data.labware.labware_definition import LabwareDefinition _log = logging.getLogger(__name__) diff --git a/robot-server/robot_server/persistence/__init__.py b/robot-server/robot_server/persistence/__init__.py index 7e1794d35fc..531ac268bf0 100644 --- a/robot-server/robot_server/persistence/__init__.py +++ b/robot-server/robot_server/persistence/__init__.py @@ -1 +1,5 @@ """Support for persisting data across device reboots.""" + +from ._files_and_directories import LATEST_VERSION_DIRECTORY + +__all__ = ["LATEST_VERSION_DIRECTORY"] diff --git a/robot-server/robot_server/persistence/_files_and_directories.py b/robot-server/robot_server/persistence/_files_and_directories.py new file mode 100644 index 00000000000..4e61fd7d4f6 --- /dev/null +++ b/robot-server/robot_server/persistence/_files_and_directories.py @@ -0,0 +1,8 @@ +"""Files and directories in persistent storage.""" +from typing import Final + +DECK_CONFIGURATION_FILE: Final = "deck_configuration.json" +PROTOCOLS_DIRECTORY: Final = "protocols" +DATA_FILES_DIRECTORY: Final = "data_files" +DB_FILE: Final = "robot_server.db" +LATEST_VERSION_DIRECTORY: Final = "6" diff --git a/robot-server/robot_server/persistence/_legacy_pickle.py b/robot-server/robot_server/persistence/_legacy_pickle.py index 36d68a1968a..6b60347b16a 100644 --- a/robot-server/robot_server/persistence/_legacy_pickle.py +++ b/robot-server/robot_server/persistence/_legacy_pickle.py @@ -188,7 +188,7 @@ def _get_legacy_ot_types() -> List[_LegacyTypeInfo]: _LegacyTypeInfo(original_name="MovementAxis", current_type=MovementAxis) ) - from opentrons_shared_data.pipette.dev_types import PipetteNameType + from opentrons_shared_data.pipette.types import PipetteNameType _legacy_ot_types.append( _LegacyTypeInfo(original_name="PipetteName", current_type=PipetteNameType) ) diff --git a/robot-server/robot_server/persistence/_migrations/up_to_3.py b/robot-server/robot_server/persistence/_migrations/up_to_3.py index a91f31930e9..c06c851bd2d 100644 --- a/robot-server/robot_server/persistence/_migrations/up_to_3.py +++ b/robot-server/robot_server/persistence/_migrations/up_to_3.py @@ -33,18 +33,16 @@ ) from ..tables import schema_2, schema_3 from .._folder_migrator import Migration +from .._files_and_directories import ( + DECK_CONFIGURATION_FILE, + PROTOCOLS_DIRECTORY, + DB_FILE, +) from ._util import copy_rows_unmodified, copy_if_exists, copytree_if_exists from . import up_to_2 - from . import _up_to_3_worker -# TODO: Define a single source of truth somewhere for these paths. -_DECK_CONFIGURATION_FILE = "deck_configuration.json" -_PROTOCOLS_DIRECTORY = "protocols" -_DB_FILE = "robot_server.db" - - _log = getLogger(__name__) @@ -52,14 +50,14 @@ class MigrationUpTo3(Migration): # noqa: D101 def migrate(self, source_dir: Path, dest_dir: Path) -> None: """Migrate the persistence directory from schema 2 to 3.""" copy_if_exists( - source_dir / _DECK_CONFIGURATION_FILE, dest_dir / _DECK_CONFIGURATION_FILE + source_dir / DECK_CONFIGURATION_FILE, dest_dir / DECK_CONFIGURATION_FILE ) copytree_if_exists( - source_dir / _PROTOCOLS_DIRECTORY, dest_dir / _PROTOCOLS_DIRECTORY + source_dir / PROTOCOLS_DIRECTORY, dest_dir / PROTOCOLS_DIRECTORY ) - source_db_file = source_dir / _DB_FILE - dest_db_file = dest_dir / _DB_FILE + source_db_file = source_dir / DB_FILE + dest_db_file = dest_dir / DB_FILE with ExitStack() as exit_stack: source_engine = exit_stack.enter_context(sql_engine_ctx(source_db_file)) diff --git a/robot-server/robot_server/persistence/_migrations/v5_to_v6.py b/robot-server/robot_server/persistence/_migrations/v5_to_v6.py new file mode 100644 index 00000000000..00cd0169d6e --- /dev/null +++ b/robot-server/robot_server/persistence/_migrations/v5_to_v6.py @@ -0,0 +1,152 @@ +"""Migrate the persistence directory from schema 5 to 6. + +Summary of changes from schema 5: + +- Removes the "run_time_parameter_values_and_defaults" column of analysis_table +- Adds a separate analysis_primitive_type_rtp_table to store fully validated primitive + run time parameters. + - NOTE: V5 to V6 migration does not port the data from run_time_parameter_values_and_defaults + into analysis_primitive_type_rtp_table. The consequence of which is that + any checks for previous matching analysis (for protocols with RTPs only) + will fail and a new analysis will be triggered. This new analysis will then + save its RTP data to the new table. RTP data belonging to previous analyses + will still be available as part of the completed analysis blob. +- Adds a new analysis_csv_rtp_table to store the CSV parameters' file IDs used in analysis +- Adds a new run_csv_rtp_table to store the CSV parameters' file IDs used in runs +- Converts protocol.protocol_kind to a constrained string (a SQL "enum"), makes it + non-nullable (NULL was semantically equivalent to "standard"), and adds an index. +""" + +from pathlib import Path +from contextlib import ExitStack + +import sqlalchemy + +from ..database import sql_engine_ctx, sqlite_rowid +from ..tables import schema_5, schema_6 +from .._folder_migrator import Migration + +from ._util import copy_rows_unmodified, copy_if_exists, copytree_if_exists +from .._files_and_directories import ( + DECK_CONFIGURATION_FILE, + PROTOCOLS_DIRECTORY, + DATA_FILES_DIRECTORY, + DB_FILE, +) + + +class Migration5to6(Migration): # noqa: D101 + def migrate(self, source_dir: Path, dest_dir: Path) -> None: + """Migrate the persistence directory from schema 5 to 6.""" + # Copy over unmodified directories and files to new version + copy_if_exists( + source_dir / DECK_CONFIGURATION_FILE, dest_dir / DECK_CONFIGURATION_FILE + ) + copytree_if_exists( + source_dir / PROTOCOLS_DIRECTORY, dest_dir / PROTOCOLS_DIRECTORY + ) + copytree_if_exists( + source_dir / DATA_FILES_DIRECTORY, dest_dir / DATA_FILES_DIRECTORY + ) + + source_db_file = source_dir / DB_FILE + dest_db_file = dest_dir / DB_FILE + + # Append the new column to existing protocols in v4 database + with ExitStack() as exit_stack: + source_engine = exit_stack.enter_context(sql_engine_ctx(source_db_file)) + + dest_engine = exit_stack.enter_context(sql_engine_ctx(dest_db_file)) + schema_6.metadata.create_all(dest_engine) + + source_transaction = exit_stack.enter_context(source_engine.begin()) + dest_transaction = exit_stack.enter_context(dest_engine.begin()) + + _migrate_db_with_changes(source_transaction, dest_transaction) + + +def _migrate_db_with_changes( + source_transaction: sqlalchemy.engine.Connection, + dest_transaction: sqlalchemy.engine.Connection, +) -> None: + copy_rows_unmodified( + schema_5.data_files_table, + schema_6.data_files_table, + source_transaction, + dest_transaction, + order_by_rowid=True, + ) + _migrate_protocol_table_with_new_protocol_kind_col( + source_transaction, + dest_transaction, + ) + _migrate_analysis_table_excluding_rtp_defaults_and_vals( + source_transaction, + dest_transaction, + ) + copy_rows_unmodified( + schema_5.run_table, + schema_6.run_table, + source_transaction, + dest_transaction, + order_by_rowid=True, + ) + copy_rows_unmodified( + schema_5.action_table, + schema_6.action_table, + source_transaction, + dest_transaction, + order_by_rowid=True, + ) + copy_rows_unmodified( + schema_5.run_command_table, + schema_6.run_command_table, + source_transaction, + dest_transaction, + order_by_rowid=True, + ) + + +def _migrate_protocol_table_with_new_protocol_kind_col( + source_transaction: sqlalchemy.engine.Connection, + dest_transaction: sqlalchemy.engine.Connection, +) -> None: + """Add a new 'protocol_kind' column to protocols table.""" + select_old_protocols = sqlalchemy.select(schema_5.protocol_table).order_by( + sqlite_rowid + ) + insert_new_protocol = sqlalchemy.insert(schema_6.protocol_table) + for old_row in source_transaction.execute(select_old_protocols).all(): + new_protocol_kind = ( + # Account for old_row.protocol_kind being NULL. + schema_6.ProtocolKindSQLEnum.QUICK_TRANSFER + if old_row.protocol_kind == "quick-transfer" + else schema_6.ProtocolKindSQLEnum.STANDARD + ) + dest_transaction.execute( + insert_new_protocol, + id=old_row.id, + created_at=old_row.created_at, + protocol_key=old_row.protocol_key, + protocol_kind=new_protocol_kind, + ) + + +def _migrate_analysis_table_excluding_rtp_defaults_and_vals( + source_transaction: sqlalchemy.engine.Connection, + dest_transaction: sqlalchemy.engine.Connection, +) -> None: + """Remove run_time_parameter_values_and_defaults column from analysis_table.""" + select_old_analyses = sqlalchemy.select(schema_5.analysis_table).order_by( + sqlite_rowid + ) + insert_new_analyses = sqlalchemy.insert(schema_6.analysis_table) + for old_row in source_transaction.execute(select_old_analyses).all(): + dest_transaction.execute( + insert_new_analyses, + id=old_row.id, + protocol_id=old_row.protocol_id, + analyzer_version=old_row.analyzer_version, + completed_analysis=old_row.completed_analysis, + # run_time_parameter_values_and_defaults column is omitted + ) diff --git a/robot-server/robot_server/persistence/persistence_directory.py b/robot-server/robot_server/persistence/persistence_directory.py index 800cbd5b6f6..403f0acca71 100644 --- a/robot-server/robot_server/persistence/persistence_directory.py +++ b/robot-server/robot_server/persistence/persistence_directory.py @@ -11,8 +11,8 @@ from anyio import Path as AsyncPath, to_thread from ._folder_migrator import MigrationOrchestrator -from ._migrations import up_to_3, v3_to_v4, v4_to_v5 - +from ._migrations import up_to_3, v3_to_v4, v4_to_v5, v5_to_v6 +from . import LATEST_VERSION_DIRECTORY _TEMP_PERSISTENCE_DIR_PREFIX: Final = "opentrons-robot-server-" _RESET_MARKER_FILE_NAME: Final = "_TO_BE_DELETED_ON_REBOOT" @@ -22,7 +22,6 @@ after which it will delete this file. """ - _log = getLogger(__name__) @@ -52,6 +51,7 @@ async def prepare_active_subdirectory(prepared_root: Path) -> Path: up_to_3.MigrationUpTo3(subdirectory="3"), v3_to_v4.Migration3to4(subdirectory="4"), v4_to_v5.Migration4to5(subdirectory="5"), + v5_to_v6.Migration5to6(subdirectory=LATEST_VERSION_DIRECTORY), ], temp_file_prefix="temp-", ) diff --git a/robot-server/robot_server/persistence/tables/__init__.py b/robot-server/robot_server/persistence/tables/__init__.py index f43c261cec5..a4bcf487e2a 100644 --- a/robot-server/robot_server/persistence/tables/__init__.py +++ b/robot-server/robot_server/persistence/tables/__init__.py @@ -1,14 +1,18 @@ """SQL database schemas.""" # Re-export the latest schema. -from .schema_5 import ( +from .schema_6 import ( metadata, protocol_table, analysis_table, + analysis_primitive_type_rtp_table, + analysis_csv_rtp_table, run_table, run_command_table, action_table, data_files_table, + PrimitiveParamSQLEnum, + ProtocolKindSQLEnum, ) @@ -16,8 +20,12 @@ "metadata", "protocol_table", "analysis_table", + "analysis_primitive_type_rtp_table", + "analysis_csv_rtp_table", "run_table", "run_command_table", "action_table", "data_files_table", + "PrimitiveParamSQLEnum", + "ProtocolKindSQLEnum", ] diff --git a/robot-server/robot_server/persistence/tables/schema_6.py b/robot-server/robot_server/persistence/tables/schema_6.py new file mode 100644 index 00000000000..d875ac4ab78 --- /dev/null +++ b/robot-server/robot_server/persistence/tables/schema_6.py @@ -0,0 +1,263 @@ +"""v6 of our SQLite schema.""" +import enum +import sqlalchemy + +from robot_server.persistence._utc_datetime import UTCDateTime + +metadata = sqlalchemy.MetaData() + + +class PrimitiveParamSQLEnum(enum.Enum): + """Enum type to store primitive param type.""" + + INT = "int" + FLOAT = "float" + BOOL = "bool" + STR = "str" + + +class ProtocolKindSQLEnum(enum.Enum): + """What kind a stored protocol is.""" + + STANDARD = "standard" + QUICK_TRANSFER = "quick-transfer" + + +protocol_table = sqlalchemy.Table( + "protocol", + metadata, + sqlalchemy.Column( + "id", + sqlalchemy.String, + primary_key=True, + ), + sqlalchemy.Column( + "created_at", + UTCDateTime, + nullable=False, + ), + sqlalchemy.Column("protocol_key", sqlalchemy.String, nullable=True), + sqlalchemy.Column( + "protocol_kind", + sqlalchemy.Enum( + ProtocolKindSQLEnum, + values_callable=lambda obj: [e.value for e in obj], + create_constraint=True, + ), + index=True, + nullable=False, + ), +) + +analysis_table = sqlalchemy.Table( + "analysis", + metadata, + sqlalchemy.Column( + "id", + sqlalchemy.String, + primary_key=True, + ), + sqlalchemy.Column( + "protocol_id", + sqlalchemy.String, + sqlalchemy.ForeignKey("protocol.id"), + index=True, + nullable=False, + ), + sqlalchemy.Column( + "analyzer_version", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "completed_analysis", + # Stores a JSON string. See CompletedAnalysisStore. + sqlalchemy.String, + nullable=False, + ), +) + +analysis_primitive_type_rtp_table = sqlalchemy.Table( + "analysis_primitive_rtp_table", + metadata, + sqlalchemy.Column( + "row_id", + sqlalchemy.Integer, + primary_key=True, + ), + sqlalchemy.Column( + "analysis_id", + sqlalchemy.ForeignKey("analysis.id"), + nullable=False, + ), + sqlalchemy.Column( + "parameter_variable_name", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "parameter_type", + sqlalchemy.Enum( + PrimitiveParamSQLEnum, + values_callable=lambda obj: [e.value for e in obj], + create_constraint=True, + ), + nullable=False, + ), + sqlalchemy.Column( + "parameter_value", + sqlalchemy.String, + nullable=False, + ), +) + +analysis_csv_rtp_table = sqlalchemy.Table( + "analysis_csv_rtp_table", + metadata, + sqlalchemy.Column( + "row_id", + sqlalchemy.Integer, + primary_key=True, + ), + sqlalchemy.Column( + "analysis_id", + sqlalchemy.ForeignKey("analysis.id"), + nullable=False, + ), + sqlalchemy.Column( + "parameter_variable_name", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "file_id", + sqlalchemy.ForeignKey("data_files.id"), + nullable=True, + ), +) + +run_table = sqlalchemy.Table( + "run", + metadata, + sqlalchemy.Column( + "id", + sqlalchemy.String, + primary_key=True, + ), + sqlalchemy.Column( + "created_at", + UTCDateTime, + nullable=False, + ), + sqlalchemy.Column( + "protocol_id", + sqlalchemy.String, + sqlalchemy.ForeignKey("protocol.id"), + nullable=True, + ), + sqlalchemy.Column( + "state_summary", + sqlalchemy.String, + nullable=True, + ), + sqlalchemy.Column("engine_status", sqlalchemy.String, nullable=True), + sqlalchemy.Column("_updated_at", UTCDateTime, nullable=True), + sqlalchemy.Column( + "run_time_parameters", + # Stores a JSON string. See RunStore. + sqlalchemy.String, + nullable=True, + ), +) + +action_table = sqlalchemy.Table( + "action", + metadata, + sqlalchemy.Column( + "id", + sqlalchemy.String, + primary_key=True, + ), + sqlalchemy.Column("created_at", UTCDateTime, nullable=False), + sqlalchemy.Column("action_type", sqlalchemy.String, nullable=False), + sqlalchemy.Column( + "run_id", + sqlalchemy.String, + sqlalchemy.ForeignKey("run.id"), + nullable=False, + ), +) + +run_command_table = sqlalchemy.Table( + "run_command", + metadata, + sqlalchemy.Column("row_id", sqlalchemy.Integer, primary_key=True), + sqlalchemy.Column( + "run_id", sqlalchemy.String, sqlalchemy.ForeignKey("run.id"), nullable=False + ), + sqlalchemy.Column("index_in_run", sqlalchemy.Integer, nullable=False), + sqlalchemy.Column("command_id", sqlalchemy.String, nullable=False), + sqlalchemy.Column("command", sqlalchemy.String, nullable=False), + sqlalchemy.Index( + "ix_run_run_id_command_id", # An arbitrary name for the index. + "run_id", + "command_id", + unique=True, + ), + sqlalchemy.Index( + "ix_run_run_id_index_in_run", # An arbitrary name for the index. + "run_id", + "index_in_run", + unique=True, + ), +) + +data_files_table = sqlalchemy.Table( + "data_files", + metadata, + sqlalchemy.Column( + "id", + sqlalchemy.String, + primary_key=True, + ), + sqlalchemy.Column( + "name", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "file_hash", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "created_at", + UTCDateTime, + nullable=False, + ), +) + +run_csv_rtp_table = sqlalchemy.Table( + "run_csv_rtp_table", + metadata, + sqlalchemy.Column( + "row_id", + sqlalchemy.Integer, + primary_key=True, + ), + sqlalchemy.Column( + "run_id", + sqlalchemy.ForeignKey("run.id"), + nullable=False, + ), + sqlalchemy.Column( + "parameter_variable_name", + sqlalchemy.String, + nullable=False, + ), + sqlalchemy.Column( + "file_id", + sqlalchemy.ForeignKey("data_files.id"), + nullable=True, + ), +) diff --git a/robot-server/robot_server/protocols/analyses_manager.py b/robot-server/robot_server/protocols/analyses_manager.py index 21d9d4687a5..4485dce3b60 100644 --- a/robot-server/robot_server/protocols/analyses_manager.py +++ b/robot-server/robot_server/protocols/analyses_manager.py @@ -1,7 +1,13 @@ """A collaborator for managing protocol analyses.""" from typing import Optional -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.util import helpers as datetime_helper + +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, +) +from opentrons.protocol_engine.errors import ErrorOccurrence from robot_server.protocols.analysis_models import ( AnalysisStatus, @@ -11,6 +17,15 @@ from robot_server.protocols import protocol_analyzer from robot_server.protocols.protocol_store import ProtocolResource from robot_server.service.task_runner import TaskRunner +import robot_server.errors.error_mappers as em + + +class FailedToInitializeAnalyzer(Exception): + """Error raised when analyzer initialization failed.""" + + def __init__(self) -> None: + """Initialize the error's message.""" + super().__init__("Failure while initializing analyzer.") class AnalysesManager: @@ -20,45 +35,70 @@ def __init__(self, analysis_store: AnalysisStore, task_runner: TaskRunner) -> No self._analysis_store = analysis_store self._task_runner = task_runner - async def start_analysis( + async def initialize_analyzer( self, analysis_id: str, protocol_resource: ProtocolResource, - run_time_param_values: Optional[RunTimeParamValuesType], - ) -> AnalysisSummary: - """Start an analysis of the given protocol resource with run time param values.""" + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_files: Optional[CSVRunTimeParamFilesType], + ) -> protocol_analyzer.ProtocolAnalyzer: + """Initialize the protocol analyzer with protocol resource and run time parameter values & fileIds. + + If an error is raised during initialization, then we abandon the analysis process + and save the failed analysis, along with the error message, to the database. + See `RunOrchestrator.get_run_time_parameters()` for details of which RTPs get + saved in the analysis when such a failure occurs. + + Returns: the successfully initialized analyzer that is ready to start analyzing. + Raises: FailedToInitializeAnalyzer if initialization failed due to error in creating + the protocol runner or loading the protocol resource or + validating the run time parameters. + """ analyzer = protocol_analyzer.create_protocol_analyzer( - analysis_store=self._analysis_store, protocol_resource=protocol_resource - ) - pending = self._analysis_store.add_pending( - protocol_id=protocol_resource.protocol_id, - analysis_id=analysis_id, + analysis_store=self._analysis_store, + protocol_resource=protocol_resource, ) try: - orchestrator = await analyzer.load_runner( - run_time_param_values=run_time_param_values + await analyzer.load_orchestrator( + run_time_param_values=run_time_param_values, + run_time_param_files=run_time_param_files, ) - pending.runTimeParameters = orchestrator.get_run_time_parameters() - except BaseException as error: - await analyzer.update_to_failed_analysis( + except Exception as error: + internal_error = em.map_unexpected_error(error) + await self._analysis_store.save_initialization_failed_analysis( + protocol_id=protocol_resource.protocol_id, analysis_id=analysis_id, - protocol_robot_type=protocol_resource.source.robot_type, - error=error, - run_time_parameters=[], - ) - return AnalysisSummary( - id=analysis_id, - status=AnalysisStatus.COMPLETED, + robot_type=protocol_resource.source.robot_type, + run_time_parameters=analyzer.get_verified_run_time_parameters(), + errors=[ + ErrorOccurrence.from_failed( + id="internal-error", + createdAt=datetime_helper.utc_now(), + error=internal_error, + ) + ], ) + raise FailedToInitializeAnalyzer() from error + return analyzer + async def start_analysis( + self, + analysis_id: str, + analyzer: protocol_analyzer.ProtocolAnalyzer, + ) -> AnalysisSummary: + """Start an analysis of the given protocol resource with verified run time parameters.""" + run_time_parameters = analyzer.get_verified_run_time_parameters() + self._analysis_store.add_pending( + protocol_id=analyzer.protocol_resource.protocol_id, + analysis_id=analysis_id, + run_time_parameters=run_time_parameters, + ) self._task_runner.run( analyzer.analyze, - orchestrator=orchestrator, analysis_id=analysis_id, - run_time_parameters=orchestrator.get_run_time_parameters(), ) return AnalysisSummary( id=analysis_id, status=AnalysisStatus.PENDING, - runTimeParameters=pending.runTimeParameters, + runTimeParameters=run_time_parameters, ) diff --git a/robot-server/robot_server/protocols/analysis_models.py b/robot-server/robot_server/protocols/analysis_models.py index 8eeeb7fad76..1e377aec3dd 100644 --- a/robot-server/robot_server/protocols/analysis_models.py +++ b/robot-server/robot_server/protocols/analysis_models.py @@ -2,8 +2,12 @@ # TODO(mc, 2021-08-25): add modules to simulation result from enum import Enum -from opentrons.protocol_engine.types import RunTimeParameter, RunTimeParamValuesType -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons.protocol_engine.types import ( + RunTimeParameter, + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, +) +from opentrons_shared_data.robot.types import RobotType from pydantic import BaseModel, Field from typing import List, Optional, Union, NamedTuple from typing_extensions import Literal @@ -48,9 +52,13 @@ class AnalysisResult(str, Enum): class AnalysisRequest(BaseModel): """Model for analysis request body.""" - runTimeParameterValues: RunTimeParamValuesType = Field( + runTimeParameterValues: PrimitiveRunTimeParamValuesType = Field( + default={}, + description="Key-value pairs of primitive run-time parameters defined in a protocol.", + ) + runTimeParameterFiles: CSVRunTimeParamFilesType = Field( default={}, - description="Key-value pairs of run-time parameters defined in a protocol.", + description="Key-fileId pairs of CSV run-time parameters defined in a protocol.", ) forceReAnalyze: bool = Field( False, description="Whether to force start a new analysis." diff --git a/robot-server/robot_server/protocols/analysis_store.py b/robot-server/robot_server/protocols/analysis_store.py index 4480cf0adaa..4458cecf025 100644 --- a/robot-server/robot_server/protocols/analysis_store.py +++ b/robot-server/robot_server/protocols/analysis_store.py @@ -6,11 +6,10 @@ from typing import Dict, List, Optional from typing_extensions import Final -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons_shared_data.errors import ErrorCodes from opentrons.protocol_engine.types import ( RunTimeParameter, - RunTimeParamValuesType, CSVParameter, ) from opentrons.protocol_engine import ( @@ -30,12 +29,11 @@ CompletedAnalysis, AnalysisResult, AnalysisStatus, - RunTimeParameterAnalysisData, - AnalysisParameterType, ) from .completed_analysis_store import CompletedAnalysisStore, CompletedAnalysisResource from .analysis_memcache import MemoryCache +from .rtp_resources import PrimitiveParameterResource, CSVParameterResource _log = getLogger(__name__) @@ -123,7 +121,8 @@ def add_pending( self, protocol_id: str, analysis_id: str, - ) -> PendingAnalysis: + run_time_parameters: Optional[List[RunTimeParameter]], + ) -> None: """Add a new pending analysis to the store. Args: @@ -136,9 +135,10 @@ def add_pending( Returns: A summary of the just-added analysis. """ - return self._pending_store.add( + self._pending_store.add( protocol_id=protocol_id, analysis_id=analysis_id, + run_time_parameters=run_time_parameters or [], ) async def update( @@ -207,16 +207,53 @@ async def update( protocol_id=protocol_id, analyzer_version=_CURRENT_ANALYZER_VERSION, completed_analysis=completed_analysis, - run_time_parameter_values_and_defaults=self._extract_run_time_param_values_and_defaults( - completed_analysis - ), ) + primitive_rtp_resources = self._extract_primitive_run_time_params( + completed_analysis + ) + csv_rtp_resources = self._extract_csv_run_time_params(completed_analysis) await self._completed_store.make_room_and_add( - completed_analysis_resource=completed_analysis_resource + completed_analysis_resource=completed_analysis_resource, + primitive_rtp_resources=primitive_rtp_resources, + csv_rtp_resources=csv_rtp_resources, ) self._pending_store.remove(analysis_id=analysis_id) + async def save_initialization_failed_analysis( + self, + protocol_id: str, + analysis_id: str, + robot_type: RobotType, + run_time_parameters: List[RunTimeParameter], + errors: List[ErrorOccurrence], + ) -> None: + """Commit the failed analysis to store.""" + completed_analysis = CompletedAnalysis.construct( + id=analysis_id, + result=AnalysisResult.NOT_OK, + robotType=robot_type, + status=AnalysisStatus.COMPLETED, + runTimeParameters=run_time_parameters, + commands=[], + labware=[], + modules=[], + pipettes=[], + errors=errors, + liquids=[], + ) + completed_analysis_resource = CompletedAnalysisResource( + id=completed_analysis.id, + protocol_id=protocol_id, + analyzer_version=_CURRENT_ANALYZER_VERSION, + completed_analysis=completed_analysis, + ) + await self._completed_store.make_room_and_add( + completed_analysis_resource=completed_analysis_resource, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) + async def get(self, analysis_id: str) -> ProtocolAnalysis: """Get a single protocol analysis by its ID. @@ -258,9 +295,6 @@ def get_summaries_by_protocol(self, protocol_id: str) -> List[AnalysisSummary]: completed_analysis_ids = self._completed_store.get_ids_by_protocol( protocol_id=protocol_id ) - # TODO (spp, 2024-06-05): populate runTimeParameters in the completed analysis summaries once - # we start saving RTPs to their own table. Currently, fetching RTPs from a - # completed analysis requires de-serializing the full analysis resource. completed_analysis_summaries = [ AnalysisSummary.construct(id=analysis_id, status=AnalysisStatus.COMPLETED) for analysis_id in completed_analysis_ids @@ -292,39 +326,42 @@ async def get_by_protocol(self, protocol_id: str) -> List[ProtocolAnalysis]: return completed_analyses + [pending_analysis] @staticmethod - def _extract_run_time_param_values_and_defaults( + def _extract_primitive_run_time_params( completed_analysis: CompletedAnalysis, - ) -> Dict[str, RunTimeParameterAnalysisData]: - """Extract the Run Time Parameters with current value and default value of each. - - We do this in order to save the RTP data separately, outside the analysis - in the database. This saves us from having to de-serialize the entire analysis - to read just the RTP values. - """ + ) -> List[PrimitiveParameterResource]: + """Extract the Primitive Run Time Parameters from analysis for saving in DB.""" rtp_list = completed_analysis.runTimeParameters + return [ + PrimitiveParameterResource( + analysis_id=completed_analysis.id, + parameter_variable_name=param.variableName, + parameter_type=param.type, + parameter_value=param.value, + ) + for param in rtp_list + if not isinstance(param, CSVParameter) + ] - rtp_values_and_defaults = {} - for param_spec in rtp_list: - value: AnalysisParameterType - if isinstance(param_spec, CSVParameter): - default = None - value = param_spec.file.id if param_spec.file is not None else None - else: - default = param_spec.default - value = param_spec.value - # TODO(jbl 2024-06-04) we might want to add type here, since CSV files value is a str and right now the only - # thing disambiguating that is that default for that will be None, if we ever want to discern type. - rtp_values_and_defaults.update( - { - param_spec.variableName: RunTimeParameterAnalysisData( - value=value, default=default - ) - } + @staticmethod + def _extract_csv_run_time_params( + completed_analysis: CompletedAnalysis, + ) -> List[CSVParameterResource]: + """Extract the Primitive Run Time Parameters from analysis for saving in DB.""" + csv_rtp_list = completed_analysis.runTimeParameters + return [ + CSVParameterResource( + analysis_id=completed_analysis.id, + parameter_variable_name=param.variableName, + file_id=param.file.id if param.file else None, ) - return rtp_values_and_defaults + for param in csv_rtp_list + if isinstance(param, CSVParameter) + ] async def matching_rtp_values_in_analysis( - self, analysis_summary: AnalysisSummary, new_rtp_values: RunTimeParamValuesType + self, + last_analysis_summary: AnalysisSummary, + new_parameters: List[RunTimeParameter], ) -> bool: """Return whether the last analysis of the given protocol used the mentioned RTP values. @@ -342,48 +379,39 @@ async def matching_rtp_values_in_analysis( with the values provided in the current request, and also verify that rest of the parameters in the analysis use default values. """ - if analysis_summary.status == AnalysisStatus.PENDING: + if last_analysis_summary.status == AnalysisStatus.PENDING: # TODO: extract defaults and values from pending analysis now that they're available # If the pending analysis RTPs match the current RTPs, do nothing(?). # If the pending analysis RTPs DO NOT match the current RTPs, raise the # AnalysisIsPending error. Eventually, we might allow either canceling the # pending analysis or starting another analysis when there's already a pending one. - raise AnalysisIsPendingError(analysis_summary.id) + raise AnalysisIsPendingError(last_analysis_summary.id) - rtp_values_and_defaults_in_last_analysis = ( - await self._completed_store.get_rtp_values_and_defaults_by_analysis_id( - analysis_summary.id + primitive_rtps_in_last_analysis = ( + self._completed_store.get_primitive_rtps_by_analysis_id( + last_analysis_summary.id ) ) - # We already make sure that the protocol has an analysis associated with before - # checking the RTP values so this assert should never raise. - # It is only added for type checking. - assert ( - rtp_values_and_defaults_in_last_analysis is not None - ), "This protocol has no analysis associated with it." - - if not set(new_rtp_values.keys()).issubset( - set(rtp_values_and_defaults_in_last_analysis.keys()) - ): - # Since the RTP keys in analysis represent all params defined in the protocol, - # if the client passes a parameter that's not present in the analysis, - # it means that the client is sending incorrect parameters. - # We will let this request trigger an analysis using the incorrect params - # and have the analysis raise an appropriate error instead of giving an - # error response to the protocols request. - # This makes the behavior of robot server consistent regardless of whether - # the client is sending a protocol for the first time or for the nth time. + if len(primitive_rtps_in_last_analysis) == 0: + # Protocols migrated from v4 will not have any entries in RTP table, + # this is fine and we should just trigger a new analysis and have + # the new values be stored in the RTP table. return False - for ( - parameter, - prev_value_and_default, - ) in rtp_values_and_defaults_in_last_analysis.items(): - if ( - new_rtp_values.get(parameter, prev_value_and_default.default) - == prev_value_and_default.value - ): - continue - else: + csv_rtps_in_last_analysis = self._completed_store.get_csv_rtps_by_analysis_id( + last_analysis_summary.id + ) + total_params_in_last_analysis = list( + primitive_rtps_in_last_analysis.keys() + ) + list(csv_rtps_in_last_analysis.keys()) + assert set(param.variableName for param in new_parameters) == set( + total_params_in_last_analysis + ), "Mismatch in parameters found in the current request vs. last saved parameters." # Indicates internal bug + for param in new_parameters: + if isinstance(param, CSVParameter): + new_file_id = param.file.id if param.file else None + if csv_rtps_in_last_analysis[param.variableName] != new_file_id: + return False + elif primitive_rtps_in_last_analysis[param.variableName] != param.value: return False return True @@ -407,7 +435,8 @@ def add( self, protocol_id: str, analysis_id: str, - ) -> PendingAnalysis: + run_time_parameters: List[RunTimeParameter], + ) -> None: """Add a new pending analysis and associate it with the given protocol.""" assert ( protocol_id not in self._analysis_ids_by_protocol_id @@ -415,14 +444,13 @@ def add( new_pending_analysis = PendingAnalysis.construct( id=analysis_id, + runTimeParameters=run_time_parameters, ) self._analyses_by_id[analysis_id] = new_pending_analysis self._analysis_ids_by_protocol_id[protocol_id] = analysis_id self._protocol_ids_by_analysis_id[analysis_id] = protocol_id - return new_pending_analysis - def remove(self, analysis_id: str) -> None: """Remove the pending analysis with the given ID. diff --git a/robot-server/robot_server/protocols/completed_analysis_store.py b/robot-server/robot_server/protocols/completed_analysis_store.py index 5f72357050b..bf8cca74871 100644 --- a/robot-server/robot_server/protocols/completed_analysis_store.py +++ b/robot-server/robot_server/protocols/completed_analysis_store.py @@ -2,22 +2,25 @@ from __future__ import annotations import asyncio -import json -from typing import Dict, List, Optional +from typing import Dict, List, Optional, Union, Mapping from logging import getLogger from dataclasses import dataclass import sqlalchemy import anyio -from pydantic import parse_raw_as +from opentrons.protocols.parameters.types import PrimitiveAllowedTypes from robot_server.persistence.database import sqlite_rowid -from robot_server.persistence.tables import analysis_table +from robot_server.persistence.tables import ( + analysis_table, + analysis_primitive_type_rtp_table, + analysis_csv_rtp_table, +) from robot_server.persistence.pydantic import json_to_pydantic, pydantic_to_json -from .analysis_models import CompletedAnalysis, RunTimeParameterAnalysisData +from .analysis_models import CompletedAnalysis from .analysis_memcache import MemoryCache - +from .rtp_resources import PrimitiveParameterResource, CSVParameterResource _log = getLogger(__name__) @@ -35,7 +38,6 @@ class CompletedAnalysisResource: protocol_id: str analyzer_version: str completed_analysis: CompletedAnalysis - run_time_parameter_values_and_defaults: Dict[str, RunTimeParameterAnalysisData] async def to_sql_values(self) -> Dict[str, object]: """Return this data as a dict that can be passed to a SQLALchemy insert. @@ -51,25 +53,17 @@ async def to_sql_values(self) -> Dict[str, object]: def serialize_completed_analysis() -> str: return pydantic_to_json(self.completed_analysis) - def serialize_rtp_dict() -> str: - return json.dumps(self.run_time_parameter_values_and_defaults) - serialized_analysis = await anyio.to_thread.run_sync( serialize_completed_analysis, # Cancellation may orphan the worker thread, # but that should be harmless in this case. cancellable=True, ) - serialized_rtp_dict = await anyio.to_thread.run_sync( - serialize_rtp_dict, - cancellable=True, - ) return { "id": self.id, "protocol_id": self.protocol_id, "analyzer_version": self.analyzer_version, "completed_analysis": serialized_analysis, - "run_time_parameter_values_and_defaults": serialized_rtp_dict, } @classmethod @@ -106,40 +100,12 @@ def parse_completed_analysis() -> CompletedAnalysis: # but that should be harmless in this case. cancellable=True, ) - rtp_values_and_defaults = await cls.get_run_time_parameter_values_and_defaults( - sql_row - ) + return cls( id=id, protocol_id=protocol_id, analyzer_version=analyzer_version, completed_analysis=completed_analysis, - run_time_parameter_values_and_defaults=rtp_values_and_defaults, - ) - - @classmethod - async def get_run_time_parameter_values_and_defaults( - cls, sql_row: sqlalchemy.engine.Row - ) -> Dict[str, RunTimeParameterAnalysisData]: - """Get the run-time parameters used in the analysis with their values & defaults.""" - - def parse_rtp_dict() -> Dict[str, RunTimeParameterAnalysisData]: - rtp_contents = sql_row.run_time_parameter_values_and_defaults - return ( - parse_raw_as( - Dict[str, RunTimeParameterAnalysisData], - sql_row.run_time_parameter_values_and_defaults, - ) - if rtp_contents - else {} - ) - - # In most cases, this parsing should be quite quick but theoretically - # there could be an unexpectedly large number of run time params. - # So we delegate the parsing of this to a cancellable thread as well. - return await anyio.to_thread.run_sync( - parse_rtp_dict, - cancellable=True, ) @@ -225,40 +191,6 @@ async def get_by_id_as_document(self, analysis_id: str) -> Optional[str]: return document - async def get_rtp_values_and_defaults_by_analysis_id( - self, analysis_id: str - ) -> Optional[Dict[str, RunTimeParameterAnalysisData]]: - """Return the dictionary of run time parameter values & defaults used in the given analysis. - - If the analysis ID doesn't exist, return None. - These RTP values are not cached in memory by themselves since we don't anticipate - that fetching the values from the database to be a time-consuming operation. - """ - async with self._memcache_lock: - try: - analysis = self._memcache.get(analysis_id) - except KeyError: - pass - else: - return analysis.run_time_parameter_values_and_defaults - - statement = sqlalchemy.select(analysis_table).where( - analysis_table.c.id == analysis_id - ) - with self._sql_engine.begin() as transaction: - try: - result = transaction.execute(statement).one() - except sqlalchemy.exc.NoResultFound: - # Since we just no-op when fetching non-existent analysis, - # do the same for non-existent RTP data - return None - - rtp_values_and_defaults = await CompletedAnalysisResource.get_run_time_parameter_values_and_defaults( - result - ) - - return rtp_values_and_defaults - async def get_by_protocol( self, protocol_id: str ) -> List[CompletedAnalysisResource]: @@ -336,8 +268,48 @@ def get_ids_by_protocol(self, protocol_id: str) -> List[str]: return result_ids + def get_primitive_rtps_by_analysis_id( + self, analysis_id: str + ) -> Dict[str, PrimitiveAllowedTypes]: + """Get the saved primitive RTP values from database.""" + statement = ( + sqlalchemy.select(analysis_primitive_type_rtp_table) + .where(analysis_primitive_type_rtp_table.c.analysis_id == analysis_id) + .order_by(sqlite_rowid) + ) + with self._sql_engine.begin() as transaction: + results = transaction.execute(statement).all() + + rtps: Dict[str, PrimitiveAllowedTypes] = {} + for row in results: + param = PrimitiveParameterResource.from_sql_row(row) + rtps.update({param.parameter_variable_name: param.parameter_value}) + return rtps + + def get_csv_rtps_by_analysis_id( + self, + analysis_id: str, + ) -> Mapping[str, Union[str, None]]: + """Get the saved CSV RTP file IDs from database.""" + statement = ( + sqlalchemy.select(analysis_csv_rtp_table) + .where(analysis_csv_rtp_table.c.analysis_id == analysis_id) + .order_by(sqlite_rowid) + ) + with self._sql_engine.begin() as transaction: + results = transaction.execute(statement).all() + + csv_rtps: Dict[str, Optional[str]] = {} + for row in results: + param = CSVParameterResource.from_sql_row(row) + csv_rtps.update({param.parameter_variable_name: param.file_id}) + return csv_rtps + async def make_room_and_add( - self, completed_analysis_resource: CompletedAnalysisResource + self, + completed_analysis_resource: CompletedAnalysisResource, + primitive_rtp_resources: List[PrimitiveParameterResource], + csv_rtp_resources: List[CSVParameterResource], ) -> None: """Make room and add a resource to the store. @@ -354,6 +326,16 @@ async def make_room_and_add( analyses_to_delete = analyses_ids[: -MAX_ANALYSES_TO_STORE + 1] for analysis_id in analyses_to_delete: self._memcache.remove(analysis_id) + + # Delete the RTP table rows that reference the analyses being deleted + delete_primitive_rtp_statement = ( + analysis_primitive_type_rtp_table.delete().where( + analysis_primitive_type_rtp_table.c.analysis_id.in_(analyses_to_delete) + ) + ) + delete_csv_rtp_statement = analysis_csv_rtp_table.delete().where( + analysis_csv_rtp_table.c.analysis_id.in_(analyses_to_delete) + ) delete_statement = analysis_table.delete().where( analysis_table.c.id.in_(analyses_to_delete) ) @@ -361,9 +343,24 @@ async def make_room_and_add( insert_statement = analysis_table.insert().values( await completed_analysis_resource.to_sql_values() ) + insert_rtp_statement = analysis_primitive_type_rtp_table.insert() + insert_csv_rtp_statement = analysis_csv_rtp_table.insert() + with self._sql_engine.begin() as transaction: + transaction.execute(delete_primitive_rtp_statement) + transaction.execute(delete_csv_rtp_statement) transaction.execute(delete_statement) transaction.execute(insert_statement) + for param in primitive_rtp_resources: + transaction.execute( + insert_rtp_statement, + param.to_sql_values(), + ) + for csv_param in csv_rtp_resources: + transaction.execute( + insert_csv_rtp_statement, + csv_param.to_sql_values(), + ) self._memcache.insert( completed_analysis_resource.id, completed_analysis_resource ) diff --git a/robot-server/robot_server/protocols/protocol_analyzer.py b/robot-server/robot_server/protocols/protocol_analyzer.py index 19b32140cd0..5ab628e42b0 100644 --- a/robot-server/robot_server/protocols/protocol_analyzer.py +++ b/robot-server/robot_server/protocols/protocol_analyzer.py @@ -2,12 +2,16 @@ import logging from typing import Optional, List -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType import opentrons.protocol_runner.create_simulating_orchestrator as simulating_runner from opentrons.protocol_engine.errors import ErrorOccurrence from opentrons.util.performance_helpers import TrackingFunctions -from opentrons.protocol_engine.types import RunTimeParamValuesType, RunTimeParameter +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + RunTimeParameter, + CSVRunTimeParamFilesType, +) import opentrons.util.helpers as datetime_helper from opentrons.protocol_runner import ( RunOrchestrator, @@ -34,37 +38,51 @@ def __init__( """Initialize the analyzer and its dependencies.""" self._analysis_store = analysis_store self._protocol_resource = protocol_resource + self._orchestrator: Optional[RunOrchestrator] = None + + @property + def protocol_resource(self) -> ProtocolResource: + """Return the protocol resource.""" + return self._protocol_resource + + def get_verified_run_time_parameters(self) -> List[RunTimeParameter]: + """Get the validated RTPs with values set by the client.""" + assert self._orchestrator is not None + return self._orchestrator.get_run_time_parameters() - async def load_runner( + async def load_orchestrator( self, - run_time_param_values: Optional[RunTimeParamValuesType], - ) -> RunOrchestrator: + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType], + run_time_param_files: Optional[CSVRunTimeParamFilesType], + ) -> None: """Load runner with the protocol and run time parameter values. Returns: The RunOrchestrator instance. """ - orchestrator = await simulating_runner.create_simulating_orchestrator( + self._orchestrator = await simulating_runner.create_simulating_orchestrator( robot_type=self._protocol_resource.source.robot_type, protocol_config=self._protocol_resource.source.config, ) - await orchestrator.load( + await self._orchestrator.load( protocol_source=self._protocol_resource.source, parse_mode=ParseMode.NORMAL, run_time_param_values=run_time_param_values, + run_time_param_files=run_time_param_files, ) - return orchestrator @TrackingFunctions.track_analysis async def analyze( self, - orchestrator: RunOrchestrator, analysis_id: str, - run_time_parameters: Optional[List[RunTimeParameter]] = None, ) -> None: - """Analyze a given protocol, storing the analysis when complete.""" + """Analyze a given protocol, storing the analysis when complete. + + This method should only be called once the run orchestrator is loaded. + """ assert self._protocol_resource is not None + assert self._orchestrator is not None try: - result = await orchestrator.run( + result = await self._orchestrator.run( deck_configuration=[], ) except BaseException as error: @@ -72,7 +90,7 @@ async def analyze( analysis_id=analysis_id, protocol_robot_type=self._protocol_resource.source.robot_type, error=error, - run_time_parameters=run_time_parameters or [], + run_time_parameters=self._orchestrator.get_run_time_parameters(), ) return diff --git a/robot-server/robot_server/protocols/protocol_auto_deleter.py b/robot-server/robot_server/protocols/protocol_auto_deleter.py index b872cab7379..c028a0c82db 100644 --- a/robot-server/robot_server/protocols/protocol_auto_deleter.py +++ b/robot-server/robot_server/protocols/protocol_auto_deleter.py @@ -27,7 +27,7 @@ def make_room_for_new_protocol(self) -> None: protocols = { p.protocol_id for p in self._protocol_store.get_all() - if p.protocol_kind == self._protocol_kind.value + if p.protocol_kind == self._protocol_kind } protocol_run_usage_info = [ diff --git a/robot-server/robot_server/protocols/protocol_models.py b/robot-server/robot_server/protocols/protocol_models.py index b0c28e5e4eb..c7841f25776 100644 --- a/robot-server/robot_server/protocols/protocol_models.py +++ b/robot-server/robot_server/protocols/protocol_models.py @@ -10,7 +10,7 @@ ProtocolFileRole as ProtocolFileRole, ) -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from robot_server.service.json_api import ResourceModel from .analysis_models import AnalysisSummary @@ -22,14 +22,6 @@ class ProtocolKind(str, Enum): STANDARD = "standard" QUICK_TRANSFER = "quick-transfer" - @staticmethod - def from_string(name: Optional[str]) -> Optional["ProtocolKind"]: - """Get the ProtocolKind from a string.""" - for item in ProtocolKind: - if name == item.value: - return item - return None - class ProtocolFile(BaseModel): """A file in a protocol.""" diff --git a/robot-server/robot_server/protocols/protocol_store.py b/robot-server/robot_server/protocols/protocol_store.py index 1a8b207f950..0488a958a12 100644 --- a/robot-server/robot_server/protocols/protocol_store.py +++ b/robot-server/robot_server/protocols/protocol_store.py @@ -14,12 +14,19 @@ from opentrons.protocols.parse import PythonParseMode from opentrons.protocol_reader import ProtocolReader, ProtocolSource + +from robot_server.data_files.models import DataFile from robot_server.persistence.database import sqlite_rowid from robot_server.persistence.tables import ( analysis_table, protocol_table, run_table, + analysis_primitive_type_rtp_table, + analysis_csv_rtp_table, + data_files_table, + ProtocolKindSQLEnum, ) +from robot_server.protocols.protocol_models import ProtocolKind _CACHE_ENTRIES = 32 @@ -36,7 +43,7 @@ class ProtocolResource: created_at: datetime source: ProtocolSource protocol_key: Optional[str] - protocol_kind: Optional[str] + protocol_kind: ProtocolKind @dataclass(frozen=True) @@ -168,7 +175,7 @@ def insert(self, resource: ProtocolResource) -> None: protocol_id=resource.protocol_id, created_at=resource.created_at, protocol_key=resource.protocol_key, - protocol_kind=resource.protocol_kind, + protocol_kind=_http_protocol_kind_to_sql(resource.protocol_kind), ) ) self._sources_by_id[resource.protocol_id] = resource.source @@ -186,7 +193,7 @@ def get(self, protocol_id: str) -> ProtocolResource: protocol_id=sql_resource.protocol_id, created_at=sql_resource.created_at, protocol_key=sql_resource.protocol_key, - protocol_kind=sql_resource.protocol_kind, + protocol_kind=_sql_protocol_kind_to_http(sql_resource.protocol_kind), source=self._sources_by_id[sql_resource.protocol_id], ) @@ -202,7 +209,7 @@ def get_all(self) -> List[ProtocolResource]: protocol_id=r.protocol_id, created_at=r.created_at, protocol_key=r.protocol_key, - protocol_kind=r.protocol_kind, + protocol_kind=_sql_protocol_kind_to_http(r.protocol_kind), source=self._sources_by_id[r.protocol_id], ) for r in all_sql_resources @@ -300,6 +307,33 @@ def get_usage_info(self) -> List[ProtocolUsageInfo]: return usage_info + # TODO (spp, 2024-07-22): get files referenced in runs as well + async def get_referenced_data_files(self, protocol_id: str) -> List[DataFile]: + """Get a list of data files referenced in specified protocol's analyses and runs.""" + # Get analyses of protocol_id + select_referencing_analysis_ids = sqlalchemy.select(analysis_table.c.id).where( + analysis_table.c.protocol_id == protocol_id + ) + # Get all entries in csv table that match the analyses + csv_file_ids = sqlalchemy.select(analysis_csv_rtp_table.c.file_id).where( + analysis_csv_rtp_table.c.analysis_id.in_(select_referencing_analysis_ids) + ) + # Get list of data file IDs from the entries + select_data_file_rows_statement = data_files_table.select().where( + data_files_table.c.id.in_(csv_file_ids) + ) + with self._sql_engine.begin() as transaction: + data_files_rows = transaction.execute(select_data_file_rows_statement).all() + + return [ + DataFile( + id=sql_row.id, + name=sql_row.name, + createdAt=sql_row.created_at, + ) + for sql_row in data_files_rows + ] + def get_referencing_run_ids(self, protocol_id: str) -> List[str]: """Return a list of run ids that reference a particular protocol. @@ -347,6 +381,21 @@ def _sql_get_all_from_engine( return [_convert_sql_row_to_dataclass(sql_row=row) for row in all_rows] def _sql_remove(self, protocol_id: str) -> None: + select_referencing_analysis_ids = sqlalchemy.select(analysis_table.c.id).where( + analysis_table.c.protocol_id == protocol_id + ) + delete_analysis_rtps_statement = sqlalchemy.delete( + analysis_primitive_type_rtp_table + ).where( + analysis_primitive_type_rtp_table.c.analysis_id.in_( + select_referencing_analysis_ids + ) + ) + delete_analysis_csv_rtps_statement = sqlalchemy.delete( + analysis_csv_rtp_table + ).where( + analysis_csv_rtp_table.c.analysis_id.in_(select_referencing_analysis_ids) + ) delete_analyses_statement = sqlalchemy.delete(analysis_table).where( analysis_table.c.protocol_id == protocol_id ) @@ -355,16 +404,18 @@ def _sql_remove(self, protocol_id: str) -> None: ) with self._sql_engine.begin() as transaction: - # TODO(mm, 2022-04-28): Deleting analyses from the table is enough to - # avoid a SQL foreign key conflict. But, if this protocol had any *pending* - # analyses, they'll be left behind in the AnalysisStore, orphaned, - # since they're stored independently of this SQL table. + # TODO(mm, 2022-04-28): Deleting analyses, and any RTP tables that reference + # those analyses, from the table is enough to avoid a SQL foreign key conflict. + # But, if this protocol had any *pending* analyses, they'll be left behind + # in the AnalysisStore, orphaned, since they're stored independently of this SQL table. # - # To fix this, we'll need to either: + # To fix this, we'll need to either: # - # * Merge the Store classes or otherwise give them access to each other. - # * Switch from SQLAlchemy Core to ORM and use cascade deletes. + # * Merge the Store classes or otherwise give them access to each other. + # * Switch from SQLAlchemy Core to ORM and use cascade deletes. try: + transaction.execute(delete_analysis_rtps_statement) + transaction.execute(delete_analysis_csv_rtps_statement) transaction.execute(delete_analyses_statement) result = transaction.execute(delete_protocol_statement) except sqlalchemy.exc.IntegrityError as e: @@ -476,7 +527,7 @@ class _DBProtocolResource: protocol_id: str created_at: datetime protocol_key: Optional[str] - protocol_kind: Optional[str] + protocol_kind: ProtocolKindSQLEnum def _convert_sql_row_to_dataclass( @@ -491,9 +542,9 @@ def _convert_sql_row_to_dataclass( assert protocol_key is None or isinstance( protocol_key, str ), f"Protocol Key {protocol_key} not a string or None" - assert protocol_kind is None or isinstance( - protocol_kind, str - ), f"Protocol Kind {protocol_kind} not a string or None" + assert isinstance( + protocol_kind, ProtocolKindSQLEnum + ), f"Protocol Kind {protocol_kind} not the expected enum" return _DBProtocolResource( protocol_id=protocol_id, @@ -512,3 +563,19 @@ def _convert_dataclass_to_sql_values( "protocol_key": resource.protocol_key, "protocol_kind": resource.protocol_kind, } + + +def _http_protocol_kind_to_sql(http_enum: ProtocolKind) -> ProtocolKindSQLEnum: + match http_enum: + case ProtocolKind.STANDARD: + return ProtocolKindSQLEnum.STANDARD + case ProtocolKind.QUICK_TRANSFER: + return ProtocolKindSQLEnum.QUICK_TRANSFER + + +def _sql_protocol_kind_to_http(sql_enum: ProtocolKindSQLEnum) -> ProtocolKind: + match sql_enum: + case ProtocolKindSQLEnum.STANDARD: + return ProtocolKind.STANDARD + case ProtocolKindSQLEnum.QUICK_TRANSFER: + return ProtocolKind.QUICK_TRANSFER diff --git a/robot-server/robot_server/protocols/router.py b/robot-server/robot_server/protocols/router.py index 706f4f41ece..1e50b6a13a9 100644 --- a/robot-server/robot_server/protocols/router.py +++ b/robot-server/robot_server/protocols/router.py @@ -7,7 +7,10 @@ from pathlib import Path from typing import List, Optional, Union, Tuple -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + CSVRunTimeParamFilesType, +) from opentrons_shared_data.robot import user_facing_robot_type from opentrons.util.performance_helpers import TrackingFunctions from typing_extensions import Literal @@ -31,7 +34,7 @@ FileReaderWriter, FileHasher, ) -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from robot_server.errors.error_responses import ErrorDetails, ErrorBody from robot_server.hardware import get_robot_type @@ -45,12 +48,19 @@ PydanticResponse, RequestModel, ) -from .analyses_manager import AnalysesManager +from robot_server.data_files.models import DataFile + +from .analyses_manager import AnalysesManager, FailedToInitializeAnalyzer from .protocol_auto_deleter import ProtocolAutoDeleter from .protocol_models import Protocol, ProtocolFile, Metadata, ProtocolKind from .analysis_store import AnalysisStore, AnalysisNotFoundError, AnalysisIsPendingError -from .analysis_models import ProtocolAnalysis, AnalysisRequest, AnalysisSummary +from .analysis_models import ( + ProtocolAnalysis, + AnalysisRequest, + AnalysisSummary, + AnalysisStatus, +) from .protocol_store import ( ProtocolStore, ProtocolResource, @@ -70,7 +80,6 @@ get_maximum_quick_transfer_protocols, ) - log = logging.getLogger(__name__) @@ -210,14 +219,21 @@ async def create_protocol( # noqa: C901 " always trigger an analysis (for now).", alias="runTimeParameterValues", ), - protocol_kind: Optional[ProtocolKind] = Form( - default=None, + protocol_kind: ProtocolKind = Form( + # This default needs to be kept in sync with the function body. + # See todo comments. + default=ProtocolKind.STANDARD, description=( "Whether this is a `standard` protocol or a `quick-transfer` protocol." "if omitted, the protocol will be `standard` by default." ), alias="protocolKind", ), + run_time_parameter_files: Optional[str] = Form( + default=None, + description="Param-file pairs of CSV run-time parameters defined in the protocol.", + alias="runTimeParameterFiles", + ), protocol_directory: Path = Depends(get_protocol_directory), protocol_store: ProtocolStore = Depends(get_protocol_store), analysis_store: AnalysisStore = Depends(get_analysis_store), @@ -242,7 +258,8 @@ async def create_protocol( # noqa: C901 Arguments: files: List of uploaded files, from form-data. key: Optional key for cli-side tracking - run_time_parameter_values: Key value pairs of run-time parameters defined in a protocol. + run_time_parameter_values: Key value pairs of primitive run-time parameters defined in a protocol. + run_time_parameter_files: Stringified dictionary of CSV parameters and their file IDs protocol_kind: Optional key representing the kind of protocol. protocol_directory: Location to store uploaded files. protocol_store: In-memory database of protocol resources. @@ -262,12 +279,30 @@ async def create_protocol( # noqa: C901 created_at: Timestamp to attach to the new resource. maximum_quick_transfer_protocols: Robot setting value limiting stored quick transfers protocols. """ - kind = ProtocolKind.from_string(protocol_kind) or ProtocolKind.STANDARD - if kind == ProtocolKind.QUICK_TRANSFER: + # We have to do these isinstance checks because if `runTimeParameterValues` or + # `protocolKind` are not specified in the request, then they get assigned a + # Form(default) value instead of just the default value. \(O.o)/ + # TODO: check if we can make our own "RTP multipart-form field" Pydantic type + # so we can validate the data contents and return a better error response. + # TODO: check if this is still necessary after converting FastAPI args to Annotated. + parsed_rtp_values = ( + json.loads(run_time_parameter_values) + if isinstance(run_time_parameter_values, str) + else {} + ) + parsed_rtp_files = ( + json.loads(run_time_parameter_files) + if isinstance(run_time_parameter_files, str) + else {} + ) + if not isinstance(protocol_kind, ProtocolKind): + protocol_kind = ProtocolKind.STANDARD + + if protocol_kind == ProtocolKind.QUICK_TRANSFER: quick_transfer_protocols = [ protocol for protocol in protocol_store.get_all() - if protocol.protocol_kind == ProtocolKind.QUICK_TRANSFER.value + if protocol.protocol_kind == ProtocolKind.QUICK_TRANSFER ] if len(quick_transfer_protocols) >= maximum_quick_transfer_protocols: raise HTTPException( @@ -278,15 +313,7 @@ async def create_protocol( # noqa: C901 # TODO(mm, 2024-02-07): Investigate whether the filename can actually be None. assert file.filename is not None buffered_files = await file_reader_writer.read(files=files) # type: ignore[arg-type] - if isinstance(run_time_parameter_values, str): - # We have to do this isinstance check because if `runTimeParameterValues` is - # not specified in the request, then it gets assigned a Form(None) value - # instead of just a None. \(O.o)/ - # TODO: check if we can make our own "RTP multipart-form field" Pydantic type - # so we can validate the data contents and return a better error response. - parsed_rtp = json.loads(run_time_parameter_values) - else: - parsed_rtp = {} + content_hash = await file_hasher.hash(buffered_files) cached_protocol_id = protocol_store.get_id_by_hash(content_hash) @@ -301,9 +328,12 @@ async def _get_cached_protocol_analysis() -> PydanticResponse[ analysis_summaries, _ = await _start_new_analysis_if_necessary( protocol_id=cached_protocol_id, analysis_id=analysis_id, - rtp_values=parsed_rtp, - force_reanalyze=False, - protocol_store=protocol_store, + force_analyze=False, + rtp_values=parsed_rtp_values, + rtp_files=parsed_rtp_files, + protocol_resource=protocol_store.get( + protocol_id=cached_protocol_id + ), analysis_store=analysis_store, analyses_manager=analyses_manager, ) @@ -315,7 +345,7 @@ async def _get_cached_protocol_analysis() -> PydanticResponse[ data = Protocol.construct( id=cached_protocol_id, createdAt=resource.created_at, - protocolKind=ProtocolKind.from_string(resource.protocol_kind), + protocolKind=resource.protocol_kind, protocolType=resource.source.config.protocol_type, robotType=resource.source.robot_type, metadata=Metadata.parse_obj(resource.source.metadata), @@ -365,29 +395,34 @@ async def _get_cached_protocol_analysis() -> PydanticResponse[ created_at=created_at, source=source, protocol_key=key, - protocol_kind=kind.value, + protocol_kind=protocol_kind, ) protocol_deleter: ProtocolAutoDeleter = protocol_auto_deleter - if kind == ProtocolKind.QUICK_TRANSFER: + if protocol_kind == ProtocolKind.QUICK_TRANSFER: protocol_deleter = quick_transfer_protocol_auto_deleter protocol_deleter.make_room_for_new_protocol() protocol_store.insert(protocol_resource) - new_analysis_summary = await analyses_manager.start_analysis( + analysis_summaries, _ = await _start_new_analysis_if_necessary( + protocol_id=protocol_id, analysis_id=analysis_id, + force_analyze=True, + rtp_values=parsed_rtp_values, + rtp_files=parsed_rtp_files, protocol_resource=protocol_resource, - run_time_param_values=parsed_rtp, + analysis_store=analysis_store, + analyses_manager=analyses_manager, ) data = Protocol( id=protocol_id, createdAt=created_at, - protocolKind=kind, + protocolKind=protocol_kind, protocolType=source.config.protocol_type, robotType=source.robot_type, metadata=Metadata.parse_obj(source.metadata), - analysisSummaries=[new_analysis_summary], + analysisSummaries=analysis_summaries, key=key, files=[ProtocolFile(name=f.path.name, role=f.role) for f in source.files], ) @@ -403,9 +438,10 @@ async def _get_cached_protocol_analysis() -> PydanticResponse[ async def _start_new_analysis_if_necessary( protocol_id: str, analysis_id: str, - force_reanalyze: bool, - rtp_values: RunTimeParamValuesType, - protocol_store: ProtocolStore, + force_analyze: bool, + rtp_values: PrimitiveRunTimeParamValuesType, + rtp_files: CSVRunTimeParamFilesType, + protocol_resource: ProtocolResource, analysis_store: AnalysisStore, analyses_manager: AnalysesManager, ) -> Tuple[List[AnalysisSummary], bool]: @@ -414,30 +450,45 @@ async def _start_new_analysis_if_necessary( Returns a tuple of the latest list of analysis summaries (including any newly started analysis) and whether a new analysis was started. """ - resource = protocol_store.get(protocol_id=protocol_id) analyses = analysis_store.get_summaries_by_protocol(protocol_id=protocol_id) started_new_analysis = False - if ( - force_reanalyze - or - # Unexpected situations, like powering off the robot after a protocol upload - # but before the analysis is complete, can leave the protocol resource - # without an associated analysis. - len(analyses) == 0 - or - # The most recent analysis was done using different RTP values - not await analysis_store.matching_rtp_values_in_analysis( - analysis_summary=analyses[-1], new_rtp_values=rtp_values + + try: + analyzer = await analyses_manager.initialize_analyzer( + analysis_id=analysis_id, + protocol_resource=protocol_resource, + run_time_param_values=rtp_values, + run_time_param_files=rtp_files, ) - ): - started_new_analysis = True + except FailedToInitializeAnalyzer: analyses.append( - await analyses_manager.start_analysis( - analysis_id=analysis_id, - protocol_resource=resource, - run_time_param_values=rtp_values, + AnalysisSummary( + id=analysis_id, + status=AnalysisStatus.COMPLETED, ) ) + else: + if ( + force_analyze + or + # Unexpected situations, like powering off the robot after a protocol upload + # but before the analysis is complete, can leave the protocol resource + # without an associated analysis. + len(analyses) == 0 + or + # The most recent analysis was done using different RTP values + not await analysis_store.matching_rtp_values_in_analysis( + last_analysis_summary=analyses[-1], + new_parameters=analyzer.get_verified_run_time_parameters(), + ) + ): + started_new_analysis = True + analyses.append( + await analyses_manager.start_analysis( + analysis_id=analysis_id, + analyzer=analyzer, + ) + ) return analyses, started_new_analysis @@ -477,7 +528,7 @@ async def get_protocols( Protocol.construct( id=r.protocol_id, createdAt=r.created_at, - protocolKind=ProtocolKind.from_string(r.protocol_kind), + protocolKind=r.protocol_kind, protocolType=r.source.config.protocol_type, robotType=r.source.robot_type, metadata=Metadata.parse_obj(r.source.metadata), @@ -558,7 +609,7 @@ async def get_protocol_by_id( data = Protocol.construct( id=protocolId, createdAt=resource.created_at, - protocolKind=ProtocolKind.from_string(resource.protocol_kind), + protocolKind=resource.protocol_kind, protocolType=resource.source.config.protocol_type, robotType=resource.source.robot_type, metadata=Metadata.parse_obj(resource.source.metadata), @@ -648,7 +699,7 @@ async def create_protocol_analysis( """Start a new analysis for the given existing protocol. Starts a new analysis for the protocol along with the provided run-time parameter - values (if any), and appends it to the existing analyses. + values (if any) and file IDs (if any), and appends it to the existing analyses. If the last analysis in the existing analyses used the same RTP values, then a new analysis is not created. @@ -669,9 +720,10 @@ async def create_protocol_analysis( ) = await _start_new_analysis_if_necessary( protocol_id=protocolId, analysis_id=analysis_id, + force_analyze=request_body.data.forceReAnalyze if request_body else False, rtp_values=request_body.data.runTimeParameterValues if request_body else {}, - force_reanalyze=request_body.data.forceReAnalyze if request_body else False, - protocol_store=protocol_store, + rtp_files=request_body.data.runTimeParameterFiles if request_body else {}, + protocol_resource=protocol_store.get(protocol_id=protocolId), analysis_store=analysis_store, analyses_manager=analyses_manager, ) @@ -823,3 +875,43 @@ async def get_protocol_analysis_as_document( ) from error return PlainTextResponse(content=analysis, media_type="application/json") + + +@PydanticResponse.wrap_route( + protocols_router.get, + path="/protocols/{protocolId}/dataFiles", + summary="Get all the data files used with the specified protocol.", + description=( + "Returns a list of all data files used in analyses and runs associated with" + " the specified protocol." + ), + responses={ + status.HTTP_200_OK: {"model": SimpleMultiBody[DataFile]}, + status.HTTP_404_NOT_FOUND: {"model": ErrorBody[ProtocolNotFound]}, + }, +) +async def get_protocol_data_files( + protocolId: str, + protocol_store: ProtocolStore = Depends(get_protocol_store), +) -> PydanticResponse[SimpleMultiBody[DataFile]]: + """Get the list of all data files associated with a protocol. + + The list includes all files used in analysis and runs stored on the robot + that are associated with the protocol. + + Arguments: + protocolId: ID of the protocol whose files are to be fetched. + protocol_store: Database of protocol resources. + """ + if not protocol_store.has(protocolId): + raise ProtocolNotFound(detail=f"Protocol {protocolId} not found").as_error( + status.HTTP_404_NOT_FOUND + ) + + data_files = await protocol_store.get_referenced_data_files(protocolId) + + return await PydanticResponse.create( + content=SimpleMultiBody.construct( + data=data_files, meta=MultiBodyMeta(cursor=0, totalLength=len(data_files)) + ) + ) diff --git a/robot-server/robot_server/protocols/rtp_resources.py b/robot-server/robot_server/protocols/rtp_resources.py new file mode 100644 index 00000000000..b0d9a0eaf8e --- /dev/null +++ b/robot-server/robot_server/protocols/rtp_resources.py @@ -0,0 +1,95 @@ +"""Primitive and CSV run time parameter resources.""" +from __future__ import annotations + +import sqlalchemy +import json + +from dataclasses import dataclass +from typing import Dict, Optional + +from robot_server.persistence.tables import PrimitiveParamSQLEnum +from opentrons.protocols.parameters.types import PrimitiveAllowedTypes + + +@dataclass +class PrimitiveParameterResource: + """A primitive runtime parameter from a completed analysis, storable in a SQL database.""" + + analysis_id: str + parameter_variable_name: str + parameter_type: str + parameter_value: PrimitiveAllowedTypes + + def to_sql_values(self) -> Dict[str, object]: + """Return this data as a dict that can be passed to an SQLAlchemy insert.""" + return { + "analysis_id": self.analysis_id, + "parameter_variable_name": self.parameter_variable_name, + "parameter_type": PrimitiveParamSQLEnum(self.parameter_type), + "parameter_value": json.dumps(self.parameter_value), + } + + @classmethod + def from_sql_row( + cls, + sql_row: sqlalchemy.engine.Row, + ) -> PrimitiveParameterResource: + """Extract the data from an SQLAlchemy row object.""" + analysis_id = sql_row.analysis_id + assert isinstance(analysis_id, str) + + parameter_variable_name = sql_row.parameter_variable_name + assert isinstance(parameter_variable_name, str) + + parameter_type = sql_row.parameter_type + assert isinstance(parameter_type, PrimitiveParamSQLEnum) + + parameter_val_str = sql_row.parameter_value + assert isinstance(parameter_val_str, str) + + parameter_value = json.loads(sql_row.parameter_value) + + return cls( + analysis_id=analysis_id, + parameter_variable_name=parameter_variable_name, + parameter_type=parameter_type.value, + parameter_value=parameter_value, + ) + + +@dataclass +class CSVParameterResource: + """A CSV runtime parameter from a completed analysis, storable in a SQL database.""" + + analysis_id: str + parameter_variable_name: str + file_id: Optional[str] + + def to_sql_values(self) -> Dict[str, object]: + """Return this data as a dict that can be passed to an SQLAlchemy insert.""" + return { + "analysis_id": self.analysis_id, + "parameter_variable_name": self.parameter_variable_name, + "file_id": self.file_id, + } + + @classmethod + def from_sql_row( + cls, + sql_row: sqlalchemy.engine.Row, + ) -> CSVParameterResource: + """Extract CSV resource data from SQLAlchemy row object.""" + analysis_id = sql_row.analysis_id + assert isinstance(analysis_id, str) + + parameter_variable_name = sql_row.parameter_variable_name + assert isinstance(parameter_variable_name, str) + + csv_file_id = sql_row.file_id + assert isinstance(csv_file_id, str) or csv_file_id is None + + return cls( + analysis_id=analysis_id, + parameter_variable_name=parameter_variable_name, + file_id=csv_file_id, + ) diff --git a/robot-server/robot_server/robot/calibration/check/user_flow.py b/robot-server/robot_server/robot/calibration/check/user_flow.py index bdd343b8c7c..a6ae99dbf2f 100644 --- a/robot-server/robot_server/robot/calibration/check/user_flow.py +++ b/robot-server/robot_server/robot/calibration/check/user_flow.py @@ -35,7 +35,7 @@ guess_from_global_config as guess_deck_type_from_global_config, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from robot_server.robot.calibration.constants import ( MOVE_TO_DECK_SAFETY_BUFFER, diff --git a/robot-server/robot_server/robot/calibration/deck/user_flow.py b/robot-server/robot_server/robot/calibration/deck/user_flow.py index a857e593820..5d41a1b5c18 100644 --- a/robot-server/robot_server/robot/calibration/deck/user_flow.py +++ b/robot-server/robot_server/robot/calibration/deck/user_flow.py @@ -41,8 +41,8 @@ from opentrons.types import Mount, Point, Location from opentrons.util import linal -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.pipette.types import LabwareUri from robot_server.robot.calibration.constants import TIP_RACK_LOOKUP_BY_MAX_VOL from robot_server.service.errors import RobotServerError diff --git a/robot-server/robot_server/robot/calibration/helper_classes.py b/robot-server/robot_server/robot/calibration/helper_classes.py index 68da8509222..5e7eeabeb57 100644 --- a/robot-server/robot_server/robot/calibration/helper_classes.py +++ b/robot-server/robot_server/robot/calibration/helper_classes.py @@ -5,7 +5,7 @@ from dataclasses import dataclass, fields from pydantic import BaseModel, Field -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from opentrons.protocol_api import labware from opentrons.types import DeckLocation diff --git a/robot-server/robot_server/robot/calibration/pipette_offset/user_flow.py b/robot-server/robot_server/robot/calibration/pipette_offset/user_flow.py index c68255ce2f3..3c56e58fd19 100644 --- a/robot-server/robot_server/robot/calibration/pipette_offset/user_flow.py +++ b/robot-server/robot_server/robot/calibration/pipette_offset/user_flow.py @@ -29,7 +29,7 @@ from opentrons.protocols.api_support.deck_type import ( guess_from_global_config as guess_deck_type_from_global_config, ) -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri from opentrons.protocol_api import labware from opentrons.protocol_api.core.legacy.deck import Deck from opentrons.types import Mount, Point, Location @@ -56,7 +56,7 @@ PipetteOffsetWithTipLengthStateMachine, ) -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition MODULE_LOG = logging.getLogger(__name__) diff --git a/robot-server/robot_server/robot/calibration/tip_length/user_flow.py b/robot-server/robot_server/robot/calibration/tip_length/user_flow.py index 282c1feb6f0..6a1a151d604 100644 --- a/robot-server/robot_server/robot/calibration/tip_length/user_flow.py +++ b/robot-server/robot_server/robot/calibration/tip_length/user_flow.py @@ -9,8 +9,8 @@ from opentrons.protocol_api import labware from opentrons.protocol_api.core.legacy.deck import Deck -from opentrons_shared_data.labware.dev_types import LabwareDefinition -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareDefinition +from opentrons_shared_data.pipette.types import LabwareUri from robot_server.robot.calibration import util from robot_server.service.errors import RobotServerError diff --git a/robot-server/robot_server/robot/calibration/util.py b/robot-server/robot_server/robot/calibration/util.py index f07f6852e68..60192aaa74b 100644 --- a/robot-server/robot_server/robot/calibration/util.py +++ b/robot-server/robot_server/robot/calibration/util.py @@ -34,8 +34,8 @@ from .tip_length.user_flow import TipCalibrationUserFlow from .pipette_offset.user_flow import PipetteOffsetCalibrationUserFlow from .check.user_flow import CheckCalibrationUserFlow - from opentrons_shared_data.pipette.dev_types import LabwareUri - from opentrons_shared_data.labware.dev_types import LabwareDefinition + from opentrons_shared_data.pipette.types import LabwareUri + from opentrons_shared_data.labware.types import LabwareDefinition ValidState = Union[ TipCalibrationState, diff --git a/robot-server/robot_server/robot/control/router.py b/robot-server/robot_server/robot/control/router.py index 012d9d63997..8bada478caf 100644 --- a/robot-server/robot_server/robot/control/router.py +++ b/robot-server/robot_server/robot/control/router.py @@ -2,8 +2,8 @@ from fastapi import APIRouter, status, Depends from typing import TYPE_CHECKING -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.robot.types import RobotTypeEnum from robot_server.hardware import get_robot_type from robot_server.errors.error_responses import ErrorBody diff --git a/robot-server/robot_server/router.py b/robot-server/robot_server/router.py index c5c80ae777e..ff60286b4f9 100644 --- a/robot-server/robot_server/router.py +++ b/robot-server/robot_server/router.py @@ -5,6 +5,7 @@ from .errors.error_responses import LegacyErrorResponse from .versioning import check_version_header +from .client_data.router import router as client_data_router from .commands.router import commands_router from .deck_configuration.router import router as deck_configuration_router from .health.router import health_router @@ -47,6 +48,12 @@ }, ) +router.include_router( + router=client_data_router, + tags=["Client Data"], + dependencies=[Depends(check_version_header)], +) + router.include_router( router=runs_router, tags=["Run Management"], diff --git a/robot-server/robot_server/runs/dependencies.py b/robot-server/robot_server/runs/dependencies.py index e9b39d7aa77..3fbef3a7e30 100644 --- a/robot-server/robot_server/runs/dependencies.py +++ b/robot-server/robot_server/runs/dependencies.py @@ -5,7 +5,7 @@ from robot_server.protocols.protocol_store import ProtocolStore from sqlalchemy.engine import Engine as SQLEngine -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.hardware_control import HardwareControlAPI from opentrons.protocol_engine import DeckType diff --git a/robot-server/robot_server/runs/error_recovery_mapping.py b/robot-server/robot_server/runs/error_recovery_mapping.py new file mode 100644 index 00000000000..fcc5c2f2f54 --- /dev/null +++ b/robot-server/robot_server/runs/error_recovery_mapping.py @@ -0,0 +1,69 @@ +"""Functions used for managing error recovery policy.""" +from typing import Optional +from opentrons.protocol_engine.state.config import Config +from robot_server.runs.error_recovery_models import ErrorRecoveryRule, ReactionIfMatch +from opentrons.protocol_engine.commands.command_unions import ( + Command, + CommandDefinedErrorData, +) +from opentrons.protocol_engine.error_recovery_policy import ( + ErrorRecoveryPolicy, + ErrorRecoveryType, +) + + +def create_error_recovery_policy_from_rules( + rules: list[ErrorRecoveryRule], +) -> ErrorRecoveryPolicy: + """Given a list of error recovery rules return an error recovery policy.""" + + def _policy( + config: Config, + failed_command: Command, + defined_error_data: Optional[CommandDefinedErrorData], + ) -> ErrorRecoveryType: + for rule in rules: + command_type_matches = ( + failed_command.commandType == rule.matchCriteria.command.commandType + ) + error_type_matches = ( + defined_error_data is not None + and defined_error_data.public.errorType + == rule.matchCriteria.command.error.errorType + ) + + if command_type_matches and error_type_matches: + if rule.ifMatch == ReactionIfMatch.IGNORE_AND_CONTINUE: + return ErrorRecoveryType.IGNORE_AND_CONTINUE + elif rule.ifMatch == ReactionIfMatch.FAIL_RUN: + return ErrorRecoveryType.FAIL_RUN + elif rule.ifMatch == ReactionIfMatch.WAIT_FOR_RECOVERY: + return ErrorRecoveryType.WAIT_FOR_RECOVERY + + return default_error_recovery_policy(config, failed_command, defined_error_data) + + return _policy + + +def default_error_recovery_policy( + config: Config, + failed_command: Command, + defined_error_data: Optional[CommandDefinedErrorData], +) -> ErrorRecoveryType: + """The `ErrorRecoveryPolicy` to use when none has been set on a run. + + This is only appropriate for normal protocol runs, not maintenance runs, + since it assumes + """ + # Although error recovery can theoretically work on OT-2s, we haven't tested it, + # and it's generally scarier because the OT-2 has much less hardware feedback. + robot_is_flex = config.robot_type == "OT-3 Standard" + # If the error is defined, we're taking that to mean that we should + # WAIT_FOR_RECOVERY. This is not necessarily the right long-term logic--we might + # want to FAIL_RUN on certain defined errors and WAIT_FOR_RECOVERY on certain + # undefined errors--but this is convenient for now. + error_is_defined = defined_error_data is not None + if robot_is_flex and error_is_defined: + return ErrorRecoveryType.WAIT_FOR_RECOVERY + else: + return ErrorRecoveryType.FAIL_RUN diff --git a/robot-server/robot_server/runs/error_recovery_models.py b/robot-server/robot_server/runs/error_recovery_models.py new file mode 100644 index 00000000000..5558c65a8ac --- /dev/null +++ b/robot-server/robot_server/runs/error_recovery_models.py @@ -0,0 +1,81 @@ +"""Request and response models for dealing with error recovery policies.""" +from enum import Enum +from typing import List + +from pydantic import BaseModel, Field + + +class ReactionIfMatch(Enum): + """The type of the error recovery setting. + + * `"ignoreAndContinue"`: Ignore this error and future errors of the same type. + * `"failRun"`: Errors of this type should fail the run. + * `"waitForRecovery"`: Instances of this error should initiate a recover operation. + + """ + + IGNORE_AND_CONTINUE = "ignoreAndContinue" + FAIL_RUN = "failRun" + WAIT_FOR_RECOVERY = "waitForRecovery" + + +# There's a lot of nested classes here. This is the JSON schema this code models. +# "ErrorRecoveryRule": { +# "matchCriteria": { +# "command": { +# "commandType": "foo", +# "error": { +# "errorType": "bar" +# } +# } +# }, +# "ifMatch": "baz" +# } + + +class ErrorMatcher(BaseModel): + """The error type that this rule applies to.""" + + errorType: str = Field(..., description="The error type that this rule applies to.") + + +class CommandMatcher(BaseModel): + """Command/error data used for matching rules.""" + + commandType: str = Field( + ..., description="The command type that this rule applies to." + ) + error: ErrorMatcher = Field( + ..., description="The error details that this rule applies to." + ) + + +class MatchCriteria(BaseModel): + """The criteria that this rule will attempt to match.""" + + command: CommandMatcher = Field( + ..., description="The command and error types that this rule applies to." + ) + + +class ErrorRecoveryRule(BaseModel): + """Model for new error recovery rule.""" + + matchCriteria: MatchCriteria = Field( + ..., + description="The criteria that must be met for this rule to be applied.", + ) + ifMatch: ReactionIfMatch = Field( + ..., + description="The specific recovery setting that will be in use if the type parameters match.", + ) + + +class ErrorRecoveryPolicy(BaseModel): + """Request/Response model for new error recovery policy rules creation.""" + + policyRules: List[ErrorRecoveryRule] = Field( + ..., + description="A list or error recovery rules to apply for a run's recovery management." + "The rules are evaluated first-to-last. The first exact match will dectate recovery management.", + ) diff --git a/robot-server/robot_server/runs/router/base_router.py b/robot-server/robot_server/runs/router/base_router.py index a3cab892c7f..14c5b822fda 100644 --- a/robot-server/robot_server/runs/router/base_router.py +++ b/robot-server/robot_server/runs/router/base_router.py @@ -45,6 +45,7 @@ get_run_auto_deleter, get_quick_transfer_run_auto_deleter, ) +from ..error_recovery_models import ErrorRecoveryPolicy from robot_server.deck_configuration.fastapi_dependencies import ( get_deck_configuration_store, @@ -362,3 +363,44 @@ async def update_run( content=SimpleBody.construct(data=run_data), status_code=status.HTTP_200_OK, ) + + +@PydanticResponse.wrap_route( + base_router.put, + path="/runs/{runId}/errorRecoveryPolicy", + summary="Set run policies", + description=dedent( + """ + Update how to handle different kinds of command failures. + The following rules will persist during the run. + """ + ), + status_code=status.HTTP_201_CREATED, + responses={ + status.HTTP_200_OK: {"model": SimpleEmptyBody}, + status.HTTP_409_CONFLICT: {"model": ErrorBody[RunStopped]}, + }, +) +async def put_error_recovery_policy( + runId: str, + request_body: RequestModel[ErrorRecoveryPolicy], + run_data_manager: RunDataManager = Depends(get_run_data_manager), +) -> PydanticResponse[SimpleEmptyBody]: + """Create run polices. + + Arguments: + runId: Run ID pulled from URL. + request_body: Request body with run policies data. + run_data_manager: Current and historical run data management. + """ + policies = request_body.data.policyRules + if policies: + try: + run_data_manager.set_policies(run_id=runId, policies=policies) + except RunNotCurrentError as e: + raise RunStopped(detail=str(e)).as_error(status.HTTP_409_CONFLICT) from e + + return await PydanticResponse.create( + content=SimpleEmptyBody.construct(), + status_code=status.HTTP_200_OK, + ) diff --git a/robot-server/robot_server/runs/run_auto_deleter.py b/robot-server/robot_server/runs/run_auto_deleter.py index f4e9485f691..6760e715923 100644 --- a/robot-server/robot_server/runs/run_auto_deleter.py +++ b/robot-server/robot_server/runs/run_auto_deleter.py @@ -29,9 +29,7 @@ def make_room_for_new_run(self) -> None: # noqa: D102 protocols = self._protocol_store.get_all() protocol_ids = [p.protocol_id for p in protocols] filtered_protocol_ids = [ - p.protocol_id - for p in protocols - if p.protocol_kind == self._protocol_kind.value + p.protocol_id for p in protocols if p.protocol_kind == self._protocol_kind ] # runs with no protocols first, then oldest to newest. diff --git a/robot-server/robot_server/runs/run_data_manager.py b/robot-server/robot_server/runs/run_data_manager.py index adfc657c9f1..5ae7581d952 100644 --- a/robot-server/robot_server/runs/run_data_manager.py +++ b/robot-server/robot_server/runs/run_data_manager.py @@ -12,11 +12,13 @@ CommandPointer, Command, ) -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.protocol_engine.types import PrimitiveRunTimeParamValuesType from robot_server.protocols.protocol_store import ProtocolResource from robot_server.service.task_runner import TaskRunner from robot_server.service.notifications import RunsPublisher +from . import error_recovery_mapping +from .error_recovery_models import ErrorRecoveryRule from .run_orchestrator_store import RunOrchestratorStore from .run_store import RunResource, RunStore, BadRunResource, BadStateSummary @@ -150,7 +152,7 @@ async def create( created_at: datetime, labware_offsets: List[LabwareOffsetCreate], deck_configuration: DeckConfigurationType, - run_time_param_values: Optional[RunTimeParamValuesType], + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType], notify_publishers: Callable[[], None], protocol: Optional[ProtocolResource], ) -> Union[Run, BadRun]: @@ -433,6 +435,17 @@ def get_all_commands_as_preserialized_list(self, run_id: str) -> List[str]: ) return self._run_store.get_all_commands_as_preserialized_list(run_id) + def set_policies(self, run_id: str, policies: List[ErrorRecoveryRule]) -> None: + """Create run policy rules for error recovery.""" + if run_id != self._run_orchestrator_store.current_run_id: + raise RunNotCurrentError( + f"Cannot update {run_id} because it is not the current run." + ) + policy = error_recovery_mapping.create_error_recovery_policy_from_rules( + policies + ) + self._run_orchestrator_store.set_error_recovery_policy(policy=policy) + def _get_state_summary(self, run_id: str) -> Union[StateSummary, BadStateSummary]: if run_id == self._run_orchestrator_store.current_run_id: return self._run_orchestrator_store.get_state_summary() diff --git a/robot-server/robot_server/runs/run_models.py b/robot-server/robot_server/runs/run_models.py index 8b61754e959..5d2fb0a41ca 100644 --- a/robot-server/robot_server/runs/run_models.py +++ b/robot-server/robot_server/runs/run_models.py @@ -18,7 +18,10 @@ Liquid, CommandNote, ) -from opentrons.protocol_engine.types import RunTimeParameter, RunTimeParamValuesType +from opentrons.protocol_engine.types import ( + RunTimeParameter, + PrimitiveRunTimeParamValuesType, +) from opentrons_shared_data.errors import GeneralError from robot_server.service.json_api import ResourceModel from robot_server.errors.error_responses import ErrorDetails @@ -237,7 +240,7 @@ class RunCreate(BaseModel): default_factory=list, description="Labware offsets to apply as labware are loaded.", ) - runTimeParameterValues: Optional[RunTimeParamValuesType] = Field( + runTimeParameterValues: Optional[PrimitiveRunTimeParamValuesType] = Field( None, description="Key-value pairs of run-time parameters defined in a protocol.", ) diff --git a/robot-server/robot_server/runs/run_orchestrator_store.py b/robot-server/robot_server/runs/run_orchestrator_store.py index de346a72968..11448a81d0c 100644 --- a/robot-server/robot_server/runs/run_orchestrator_store.py +++ b/robot-server/robot_server/runs/run_orchestrator_store.py @@ -7,8 +7,8 @@ from opentrons.protocol_engine.types import PostRunHardwareState, RunTimeParameter from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.robot.types import RobotTypeEnum from opentrons.config import feature_flags from opentrons.hardware_control import HardwareControlAPI @@ -41,11 +41,12 @@ from robot_server.protocols.protocol_store import ProtocolResource from opentrons.protocol_engine.types import ( DeckConfigurationType, - RunTimeParamValuesType, + PrimitiveRunTimeParamValuesType, EngineStatus, ) -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri +from .error_recovery_mapping import default_error_recovery_policy _log = logging.getLogger(__name__) @@ -186,7 +187,7 @@ async def create( deck_configuration: DeckConfigurationType, notify_publishers: Callable[[], None], protocol: Optional[ProtocolResource], - run_time_param_values: Optional[RunTimeParamValuesType] = None, + run_time_param_values: Optional[PrimitiveRunTimeParamValuesType] = None, ) -> StateSummary: """Create and store a ProtocolRunner and ProtocolEngine for a given Run. @@ -221,7 +222,7 @@ async def create( RobotTypeEnum.robot_literal_to_enum(self._robot_type) ), ), - error_recovery_policy=error_recovery_policy.standard_run_policy, + error_recovery_policy=default_error_recovery_policy, load_fixed_trash=load_fixed_trash, deck_configuration=deck_configuration, notify_publishers=notify_publishers, @@ -242,6 +243,8 @@ async def create( await self.run_orchestrator.load( protocol.source, run_time_param_values=run_time_param_values, + # TODO (spp, 2024-07-16): update this once runs accept csv params + run_time_param_files={}, parse_mode=ParseMode.ALLOW_LEGACY_METADATA_AND_REQUIREMENTS, ) else: @@ -359,6 +362,12 @@ def add_labware_definition(self, definition: LabwareDefinition) -> LabwareUri: """Add a new labware definition to state.""" return self.run_orchestrator.add_labware_definition(definition) + def set_error_recovery_policy( + self, policy: error_recovery_policy.ErrorRecoveryPolicy + ) -> None: + """Create run policy rules for error recovery.""" + self.run_orchestrator.set_error_recovery_policy(policy) + async def add_command_and_wait_for_interval( self, request: CommandCreate, diff --git a/robot-server/robot_server/service/errors.py b/robot-server/robot_server/service/errors.py index f9bd269b965..94a8d758563 100644 --- a/robot-server/robot_server/service/errors.py +++ b/robot-server/robot_server/service/errors.py @@ -1,5 +1,6 @@ # TODO(mc, 2021-05-10): delete this file; these models have been moved to # robot_server/errors/error_responses.py and robot_server/errors/global_errors.py +# Note: (2024-07-18): this file does not actually seem to be safe to delete from dataclasses import dataclass, asdict from enum import Enum from typing import Any, Dict, Optional, Sequence, Tuple diff --git a/robot-server/robot_server/service/legacy/routers/settings.py b/robot-server/robot_server/service/legacy/routers/settings.py index 5d79053d696..65e2d0c63d4 100644 --- a/robot-server/robot_server/service/legacy/routers/settings.py +++ b/robot-server/robot_server/service/legacy/routers/settings.py @@ -56,7 +56,7 @@ get_persistence_resetter, ) from robot_server.persistence.persistence_directory import PersistenceResetter -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum log = logging.getLogger(__name__) diff --git a/robot-server/robot_server/service/notifications/__init__.py b/robot-server/robot_server/service/notifications/__init__.py index e62402f06f5..3b40ee17379 100644 --- a/robot-server/robot_server/service/notifications/__init__.py +++ b/robot-server/robot_server/service/notifications/__init__.py @@ -15,7 +15,6 @@ get_runs_publisher, get_deck_configuration_publisher, ) -from .topics import Topics __all__ = [ # main export @@ -35,5 +34,4 @@ "get_deck_configuration_publisher", # for testing "PublisherNotifier", - "Topics", ] diff --git a/robot-server/robot_server/service/notifications/notification_client.py b/robot-server/robot_server/service/notifications/notification_client.py index f53de3bbe39..81ec2b69a1d 100644 --- a/robot-server/robot_server/service/notifications/notification_client.py +++ b/robot-server/robot_server/service/notifications/notification_client.py @@ -7,6 +7,8 @@ from typing import Any, Dict, Optional from enum import Enum + +from .topics import TopicName from ..json_api import NotifyRefetchBody, NotifyUnsubscribeBody from server_utils.fastapi_utils.app_state import ( AppState, @@ -80,7 +82,7 @@ async def disconnect(self) -> None: self._client.loop_stop() await to_thread.run_sync(self._client.disconnect) - async def publish_advise_refetch_async(self, topic: str) -> None: + async def publish_advise_refetch_async(self, topic: TopicName) -> None: """Asynchronously publish a refetch message on a specific topic to the MQTT broker. Args: @@ -88,7 +90,7 @@ async def publish_advise_refetch_async(self, topic: str) -> None: """ await to_thread.run_sync(self.publish_advise_refetch, topic) - async def publish_advise_unsubscribe_async(self, topic: str) -> None: + async def publish_advise_unsubscribe_async(self, topic: TopicName) -> None: """Asynchronously publish an unsubscribe message on a specific topic to the MQTT broker. Args: diff --git a/robot-server/robot_server/service/notifications/publishers/client_data_publisher.py b/robot-server/robot_server/service/notifications/publishers/client_data_publisher.py new file mode 100644 index 00000000000..77225368a35 --- /dev/null +++ b/robot-server/robot_server/service/notifications/publishers/client_data_publisher.py @@ -0,0 +1,29 @@ +from typing import Annotated +import fastapi +from robot_server.service.notifications import topics +from robot_server.service.notifications.notification_client import ( + NotificationClient, + get_notification_client, +) + + +class ClientDataPublisher: + """Publishes clientData topics.""" + + def __init__(self, client: NotificationClient) -> None: + self._client = client + + async def publish_client_data(self, client_data_key: str) -> None: + """Publish the equivalent of `GET /clientData/{key}`.""" + await self._client.publish_advise_refetch_async( + topics.client_data(client_data_key) + ) + + +async def get_client_data_publisher( + notification_client: Annotated[ + NotificationClient, fastapi.Depends(get_notification_client) + ], +) -> ClientDataPublisher: + """Return a ClientDataPublisher for use by FastAPI endpoints.""" + return ClientDataPublisher(notification_client) diff --git a/robot-server/robot_server/service/notifications/publishers/deck_configuration_publisher.py b/robot-server/robot_server/service/notifications/publishers/deck_configuration_publisher.py index a1c0bc1e9a5..14361873a7b 100644 --- a/robot-server/robot_server/service/notifications/publishers/deck_configuration_publisher.py +++ b/robot-server/robot_server/service/notifications/publishers/deck_configuration_publisher.py @@ -6,7 +6,7 @@ get_app_state, ) from ..notification_client import NotificationClient, get_notification_client -from ..topics import Topics +from .. import topics class DeckConfigurationPublisher: @@ -20,7 +20,7 @@ async def publish_deck_configuration( self, ) -> None: """Publishes the equivalent of GET /deck_configuration""" - await self._client.publish_advise_refetch_async(topic=Topics.DECK_CONFIGURATION) + await self._client.publish_advise_refetch_async(topic=topics.DECK_CONFIGURATION) _deck_configuration_publisher_accessor: AppStateAccessor[ diff --git a/robot-server/robot_server/service/notifications/publishers/maintenance_runs_publisher.py b/robot-server/robot_server/service/notifications/publishers/maintenance_runs_publisher.py index 8ef07fd7eac..1c382d37102 100644 --- a/robot-server/robot_server/service/notifications/publishers/maintenance_runs_publisher.py +++ b/robot-server/robot_server/service/notifications/publishers/maintenance_runs_publisher.py @@ -6,7 +6,7 @@ get_app_state, ) from ..notification_client import NotificationClient, get_notification_client -from ..topics import Topics +from .. import topics class MaintenanceRunsPublisher: @@ -21,7 +21,7 @@ async def publish_current_maintenance_run( ) -> None: """Publishes the equivalent of GET /maintenance_run/current_run""" await self._client.publish_advise_refetch_async( - topic=Topics.MAINTENANCE_RUNS_CURRENT_RUN + topic=topics.MAINTENANCE_RUNS_CURRENT_RUN ) diff --git a/robot-server/robot_server/service/notifications/publishers/runs_publisher.py b/robot-server/robot_server/service/notifications/publishers/runs_publisher.py index bcb54133b3e..9de9e2c7c51 100644 --- a/robot-server/robot_server/service/notifications/publishers/runs_publisher.py +++ b/robot-server/robot_server/service/notifications/publishers/runs_publisher.py @@ -11,7 +11,7 @@ ) from ..notification_client import NotificationClient, get_notification_client from ..publisher_notifier import PublisherNotifier, get_pe_publisher_notifier -from ..topics import Topics +from .. import topics @dataclass @@ -89,36 +89,40 @@ async def _publish_command_links(self) -> None: (regardless of query parameters). """ await self._client.publish_advise_refetch_async( - topic=Topics.RUNS_COMMANDS_LINKS + topic=topics.RUNS_COMMANDS_LINKS ) async def _publish_runs_advise_refetch_async(self, run_id: str) -> None: """Publish a refetch flag for relevant runs topics.""" - await self._client.publish_advise_refetch_async(topic=Topics.RUNS) + await self._client.publish_advise_refetch_async(topic=topics.RUNS) if self._run_hooks is not None: await self._client.publish_advise_refetch_async( - topic=f"{Topics.RUNS}/{run_id}" + topic=topics.TopicName(f"{topics.RUNS}/{run_id}") ) async def _publish_runs_advise_unsubscribe_async(self, run_id: str) -> None: """Publish an unsubscribe flag for relevant runs topics.""" if self._run_hooks is not None: await self._client.publish_advise_unsubscribe_async( - topic=f"{Topics.RUNS}/{run_id}" + topic=topics.TopicName(f"{topics.RUNS}/{run_id}") ) await self._client.publish_advise_unsubscribe_async( - topic=Topics.RUNS_COMMANDS_LINKS + topic=topics.RUNS_COMMANDS_LINKS ) await self._client.publish_advise_unsubscribe_async( - topic=f"{Topics.RUNS_PRE_SERIALIZED_COMMANDS}/{run_id}" + topic=topics.TopicName( + f"{topics.RUNS_PRE_SERIALIZED_COMMANDS}/{run_id}" + ) ) async def publish_pre_serialized_commands_notification(self, run_id: str) -> None: """Publishes notification for GET /runs/:runId/commandsAsPreSerializedList.""" if self._run_hooks is not None: await self._client.publish_advise_refetch_async( - topic=f"{Topics.RUNS_PRE_SERIALIZED_COMMANDS}/{run_id}" + topic=topics.TopicName( + f"{topics.RUNS_PRE_SERIALIZED_COMMANDS}/{run_id}" + ) ) async def _handle_current_command_change(self) -> None: diff --git a/robot-server/robot_server/service/notifications/topics.py b/robot-server/robot_server/service/notifications/topics.py index bb21d7e6760..8065b4d38fd 100644 --- a/robot-server/robot_server/service/notifications/topics.py +++ b/robot-server/robot_server/service/notifications/topics.py @@ -1,18 +1,43 @@ -"""Notification topics.""" -from enum import Enum +"""MQTT topics for server-emitted notifications. +These are the MQTT functional equivalent of HTTP endpoints. +Each topic should generally be named after the HTTP endpoint whose data it's reflecting. -_TOPIC_BASE = "robot-server" +It's helpful to have these centralized in this one file so we can see all the topics +that we currently support. +""" +from typing import NewType +import re -class Topics(str, Enum): - """Notification Topics - MQTT functional equivalent of endpoints. - """ +TopicName = NewType("TopicName", str) +"""A string suitable for the server to use as an MQTT topic to publish on.""" - MAINTENANCE_RUNS_CURRENT_RUN = f"{_TOPIC_BASE}/maintenance_runs/current_run" - RUNS_COMMANDS_LINKS = f"{_TOPIC_BASE}/runs/commands_links" - RUNS = f"{_TOPIC_BASE}/runs" - DECK_CONFIGURATION = f"{_TOPIC_BASE}/deck_configuration" - RUNS_PRE_SERIALIZED_COMMANDS = f"{_TOPIC_BASE}/runs/pre_serialized_commands" + +_TOPIC_BASE = TopicName("robot-server") + + +def _is_valid_topic_name_level(level: str) -> bool: + """Return whether a string is valid as a level (segment) in an MQTT topic name.""" + return not re.match("[/#+]", level) + + +MAINTENANCE_RUNS_CURRENT_RUN = TopicName(f"{_TOPIC_BASE}/maintenance_runs/current_run") +RUNS_COMMANDS_LINKS = TopicName(f"{_TOPIC_BASE}/runs/commands_links") +# todo(mm, 2024-07-24): We actually publish on subtopics of /runs. Convert this to a +# function like we do for clientData. +RUNS = TopicName(f"{_TOPIC_BASE}/runs") +DECK_CONFIGURATION = TopicName(f"{_TOPIC_BASE}/deck_configuration") +RUNS_PRE_SERIALIZED_COMMANDS = TopicName(f"{_TOPIC_BASE}/runs/pre_serialized_commands") + + +def client_data(key: str) -> TopicName: + """Return the dynamic MQTT topic name for the given clientData key.""" + base = f"{_TOPIC_BASE}/clientData" + if _is_valid_topic_name_level(key): + return TopicName(f"{base}/{key}") + else: + raise ValueError( + f"{repr(key)} is not valid as a segment in an MQTT topic name." + ) diff --git a/robot-server/robot_server/service/session/session_types/pipette_offset_calibration.py b/robot-server/robot_server/service/session/session_types/pipette_offset_calibration.py index 61804c06352..4392f037978 100644 --- a/robot-server/robot_server/service/session/session_types/pipette_offset_calibration.py +++ b/robot-server/robot_server/service/session/session_types/pipette_offset_calibration.py @@ -2,7 +2,7 @@ from typing import Awaitable, Optional, cast from opentrons.types import Mount from opentrons.protocol_api import labware -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from robot_server.robot.calibration.pipette_offset.user_flow import ( PipetteOffsetCalibrationUserFlow, ) diff --git a/robot-server/robot_server/service/session/session_types/tip_length_calibration.py b/robot-server/robot_server/service/session/session_types/tip_length_calibration.py index 61325dece32..a2e7c06b6e3 100644 --- a/robot-server/robot_server/service/session/session_types/tip_length_calibration.py +++ b/robot-server/robot_server/service/session/session_types/tip_length_calibration.py @@ -1,6 +1,6 @@ from typing import cast, Awaitable, Optional, Any, Union from opentrons.types import Mount -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from robot_server.robot.calibration.tip_length.user_flow import TipCalibrationUserFlow from robot_server.robot.calibration.models import SessionCreateParams from robot_server.robot.calibration.tip_length.models import TipCalibrationSessionStatus diff --git a/robot-server/robot_server/service/tip_length/router.py b/robot-server/robot_server/service/tip_length/router.py index 7758a96e8b8..e4658f59dc7 100644 --- a/robot-server/robot_server/service/tip_length/router.py +++ b/robot-server/robot_server/service/tip_length/router.py @@ -12,7 +12,7 @@ from robot_server.service.shared_models import calibration as cal_model from opentrons.hardware_control import API -from opentrons_shared_data.pipette.dev_types import LabwareUri +from opentrons_shared_data.pipette.types import LabwareUri router = APIRouter() diff --git a/robot-server/tests/conftest.py b/robot-server/tests/conftest.py index 8f13c278e9a..aed43626a2b 100644 --- a/robot-server/tests/conftest.py +++ b/robot-server/tests/conftest.py @@ -15,7 +15,7 @@ from fastapi import routing from starlette.testclient import TestClient -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from opentrons import config from opentrons.hardware_control import API, HardwareControlAPI, ThreadedAsyncLock diff --git a/robot-server/tests/instruments/test_router.py b/robot-server/tests/instruments/test_router.py index 8d45c10c5d8..01ba0cdf7c6 100644 --- a/robot-server/tests/instruments/test_router.py +++ b/robot-server/tests/instruments/test_router.py @@ -25,7 +25,7 @@ GripperModelStr, GripperModel, ) -from opentrons_shared_data.pipette.dev_types import PipetteName, PipetteModel +from opentrons_shared_data.pipette.types import PipetteName, PipetteModel from opentrons.hardware_control.protocols.types import FlexRobotType, OT2RobotType from robot_server.instruments.instrument_models import ( diff --git a/robot-server/tests/integration/conftest.py b/robot-server/tests/integration/conftest.py index 2c8d66853a1..ebea66e92ba 100644 --- a/robot-server/tests/integration/conftest.py +++ b/robot-server/tests/integration/conftest.py @@ -129,6 +129,8 @@ async def _clean_server_state_async() -> None: await _reset_deck_configuration(robot_client) + await _delete_client_data(robot_client) + asyncio.run(_clean_server_state_async()) @@ -141,7 +143,7 @@ async def _delete_all_runs(robot_client: RobotClient) -> None: async def _delete_all_protocols(robot_client: RobotClient) -> None: - """Delete all protocols on the robot server""" + """Delete all protocols on the robot server.""" response = await robot_client.get_protocols() protocol_ids = [p["id"] for p in response.json()["data"]] for protocol_id in protocol_ids: @@ -155,5 +157,9 @@ async def _delete_all_sessions(robot_client: RobotClient) -> None: await robot_client.delete_session(session_id) +async def _delete_client_data(robot_client: RobotClient) -> None: + await robot_client.delete_all_client_data() + + async def _reset_deck_configuration(robot_client: RobotClient) -> None: await robot_client.post_setting_reset_options({"deckConfiguration": True}) diff --git a/robot-server/tests/integration/fixtures.py b/robot-server/tests/integration/fixtures.py index 08175cc6b99..910eb9256dd 100644 --- a/robot-server/tests/integration/fixtures.py +++ b/robot-server/tests/integration/fixtures.py @@ -9,7 +9,7 @@ MIN_SUPPORTED_VERSION_FOR_FLEX, ) from opentrons import __version__, config -from opentrons_shared_data.module.dev_types import ModuleModel +from opentrons_shared_data.module.types import ModuleModel def check_health_response(response: Response) -> None: diff --git a/robot-server/tests/integration/http_api/data_files/test_upload_data_file.tavern.yaml b/robot-server/tests/integration/http_api/data_files/test_upload_data_file.tavern.yaml index 14540f3c502..8ec1e815e4e 100644 --- a/robot-server/tests/integration/http_api/data_files/test_upload_data_file.tavern.yaml +++ b/robot-server/tests/integration/http_api/data_files/test_upload_data_file.tavern.yaml @@ -33,7 +33,7 @@ stages: json: data: !force_original_structure '{file_info}' - - name: Upload sample_records.csv file using file path + - name: Upload sample_record.csv file using file path request: url: '{ot2_server_base_url}/dataFiles' method: POST diff --git a/robot-server/tests/integration/http_api/persistence/test_reset.py b/robot-server/tests/integration/http_api/persistence/test_reset.py index 455385e95c5..937d17c81f0 100644 --- a/robot-server/tests/integration/http_api/persistence/test_reset.py +++ b/robot-server/tests/integration/http_api/persistence/test_reset.py @@ -40,9 +40,9 @@ async def _assert_reset_was_successful( all_files_and_directories = set(persistence_directory.glob("**/*")) expected_files_and_directories = { persistence_directory / "robot_server.db", - persistence_directory / "5", - persistence_directory / "5" / "protocols", - persistence_directory / "5" / "robot_server.db", + persistence_directory / "6", + persistence_directory / "6" / "protocols", + persistence_directory / "6" / "robot_server.db", } assert all_files_and_directories == expected_files_and_directories diff --git a/robot-server/tests/integration/http_api/protocols/test_analyses_with_csv_file_parameters.tavern.yaml b/robot-server/tests/integration/http_api/protocols/test_analyses_with_csv_file_parameters.tavern.yaml new file mode 100644 index 00000000000..d915d599ba5 --- /dev/null +++ b/robot-server/tests/integration/http_api/protocols/test_analyses_with_csv_file_parameters.tavern.yaml @@ -0,0 +1,136 @@ +test_name: Test the protocol analysis endpoints with a CSV file + +marks: + - usefixtures: + - ot2_server_base_url + +stages: + - name: Upload a CSV file + request: + url: '{ot2_server_base_url}/dataFiles' + method: POST + data: + filePath: 'tests/integration/data_files/sample_record.csv' + response: + save: + json: + csv_file_id: data.id + csv_file_name: data.name + file_created_at: data.createdAt + status_code: + # If the file exists on the test server then accept 200 else 201. + # We have to do this because the persistent storage is scoped to a test session and + # there are tests elsewhere in the suite that upload the same file. + # We currently do not allow deleting data files over http, so this situation is not easy to change. + - 200 + - 201 + json: + data: + id: !anystr + name: "sample_record.csv" + createdAt: !re_fullmatch "\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d+(Z|([+-]\\d{2}:\\d{2}))" + + - name: Upload a protocol + request: + url: '{ot2_server_base_url}/protocols' + method: POST + data: + runTimeParameterValues: '{{"sample_count": 10}}' + runTimeParameterFiles: '{{"liq_handling_csv_file": "{csv_file_id}"}}' + files: + files: 'tests/integration/protocols/basic_transfer_with_run_time_parameters.py' + response: + save: + json: + protocol_id: data.id + analysis_id: data.analysisSummaries[0].id + run_time_parameters_data1: data.analysisSummaries[0].runTimeParameters + strict: + - json:off + status_code: 201 + json: + data: + analyses: [] + analysisSummaries: + - id: !anystr + status: pending + runTimeParameters: + - displayName: Sample count + variableName: sample_count + type: int + default: 6.0 + min: 1.0 + max: 12.0 + value: 10.0 + description: How many samples to process. + - displayName: Pipette volume + variableName: volume + type: float + default: 20.1 + choices: + - displayName: Low Volume + value: 10.23 + - displayName: Medium Volume + value: 20.1 + - displayName: High Volume + value: 50.5 + value: 20.1 + description: How many microliters to pipette of each sample. + - displayName: Dry Run + variableName: dry_run + type: bool + default: false + value: false + description: Skip aspirate and dispense steps. + - displayName: Pipette Name + variableName: pipette + type: str + choices: + - displayName: Single channel 50µL + value: flex_1channel_50 + - displayName: Eight Channel 50µL + value: flex_8channel_50 + default: flex_1channel_50 + value: flex_1channel_50 + description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting + file: + id: '{csv_file_id}' + name: '' + + - name: Wait until analysis is completed + max_retries: 5 + delay_after: 1 + request: + url: '{ot2_server_base_url}/protocols/{protocol_id}' + response: + status_code: 200 + json: + data: + analyses: [] + analysisSummaries: + - id: '{analysis_id}' + status: completed + id: !anything + protocolType: !anything + files: !anything + createdAt: !anything + robotType: !anything + protocolKind: !anything + metadata: !anything + links: !anything + + - name: Get the data file used by protocol ID + request: + url: '{ot2_server_base_url}/protocols/{protocol_id}/dataFiles' + response: + strict: + - json:off + status_code: 200 + json: + data: + - id: '{csv_file_id}' + name: '{csv_file_name}' + createdAt: '{file_created_at}' diff --git a/robot-server/tests/integration/http_api/protocols/test_analyses_with_run_time_parameters.tavern.yaml b/robot-server/tests/integration/http_api/protocols/test_analyses_with_run_time_parameters.tavern.yaml index c6562f68cc5..333f7ebc827 100644 --- a/robot-server/tests/integration/http_api/protocols/test_analyses_with_run_time_parameters.tavern.yaml +++ b/robot-server/tests/integration/http_api/protocols/test_analyses_with_run_time_parameters.tavern.yaml @@ -65,6 +65,9 @@ stages: default: flex_1channel_50 value: flex_1channel_50 description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting - name: Check that the analysis summary is present in /protocols/:id; retry until it says it's completed max_retries: 5 @@ -167,6 +170,9 @@ stages: default: flex_1channel_50 value: flex_1channel_50 description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting - name: Check that the new analysis uses run time parameter values from client; retry until analysis is completed @@ -244,3 +250,6 @@ stages: default: flex_1channel_50 value: flex_1channel_50 description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting \ No newline at end of file diff --git a/robot-server/tests/integration/http_api/protocols/test_persistence.py b/robot-server/tests/integration/http_api/protocols/test_persistence.py index f18107b94c1..c8f4ae4d84f 100644 --- a/robot-server/tests/integration/http_api/protocols/test_persistence.py +++ b/robot-server/tests/integration/http_api/protocols/test_persistence.py @@ -5,6 +5,8 @@ import pytest +from robot_server.persistence import LATEST_VERSION_DIRECTORY + from tests.integration.dev_server import DevServer from tests.integration.robot_client import RobotClient from tests.integration.protocol_files import get_py_protocol, get_json_protocol @@ -120,10 +122,10 @@ async def test_protocol_labware_files_persist() -> None: assert restarted_protocol_detail == protocol_detail four_tuberack = Path( - f"{server.persistence_directory}/5/protocols/{protocol_id}/cpx_4_tuberack_100ul.json" + f"{server.persistence_directory}/{LATEST_VERSION_DIRECTORY}/protocols/{protocol_id}/cpx_4_tuberack_100ul.json" ) six_tuberack = Path( - f"{server.persistence_directory}/5/protocols/{protocol_id}/cpx_6_tuberack_100ul.json" + f"{server.persistence_directory}/{LATEST_VERSION_DIRECTORY}/protocols/{protocol_id}/cpx_6_tuberack_100ul.json" ) assert four_tuberack.is_file() assert six_tuberack.is_file() diff --git a/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml index de0966ac42f..2a71179b37d 100644 --- a/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml @@ -134,6 +134,9 @@ stages: default: flex_1channel_50 value: flex_8channel_50 description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting protocolId: '{protocol_id}' - name: Mark the run as not-current @@ -200,4 +203,7 @@ stages: default: flex_1channel_50 value: flex_8channel_50 description: What pipette to use during the protocol. + - displayName: Liquid handling CSV file + variableName: liq_handling_csv_file + description: A CSV file that contains wells to use for pipetting protocolId: '{protocol_id}' diff --git a/robot-server/tests/integration/http_api/test_client_data.tavern.yaml b/robot-server/tests/integration/http_api/test_client_data.tavern.yaml new file mode 100644 index 00000000000..781475e63de --- /dev/null +++ b/robot-server/tests/integration/http_api/test_client_data.tavern.yaml @@ -0,0 +1,112 @@ +test_name: Test getting and setting client data + +marks: + - usefixtures: + - ot3_server_base_url + +stages: + - name: Check the initial client data + request: + method: GET + url: '{ot3_server_base_url}/clientData/foo' + response: + status_code: 404 + + - name: Set client data + request: + method: PUT + url: '{ot3_server_base_url}/clientData/foo' + json: + data: &put_data + stringField: string value + numberField: 123.456 + boolField: true + nullField: null + objectField: + key: value + response: + status_code: 200 + json: + data: *put_data + + - name: Check that PUT rejects non-object data + request: + method: PUT + url: '{ot3_server_base_url}/clientData/foo' + json: + data: This is a string and not an object. + response: + status_code: 422 + + - name: Retrieve client data + request: + method: GET + url: '{ot3_server_base_url}/clientData/foo' + response: + status_code: 200 + json: + data: *put_data + + - name: Check that trailing slashes are ignored + request: + method: GET + url: '{ot3_server_base_url}/clientData/foo/' + follow_redirects: true # FastAPI redirects when given a trailing slash. + response: + status_code: 200 + json: + data: *put_data + + - name: Delete client data + request: + method: DELETE + url: '{ot3_server_base_url}/clientData/foo' + response: + status_code: 200 + + - name: Check that it was deleted + request: + method: GET + url: '{ot3_server_base_url}/clientData/foo' + response: + status_code: 404 + +--- +test_name: Test client data key validation + +marks: + - usefixtures: + - ot3_server_base_url + - parametrize: + key: + - bad_key + - expected_status_code + vals: + - ['foo/bar', 404] + - ['foo*', 422] + - ['+', 422] # "+" is a wildcard in MQTT, but HTTP-brained code might decode it as a space. + - ['foo+bar', 422] + +stages: + - name: Check that PUT rejects bad keys + request: + method: PUT + url: '{ot3_server_base_url}/clientData/{bad_key}' + json: + data: {} + response: + status_code: !int '{expected_status_code}' + + - name: Check that GET rejects bad keys + request: + method: GET + url: '{ot3_server_base_url}/clientData/{bad_key}' + response: + status_code: !int '{expected_status_code}' + + - name: Check that DELETE rejects bad keys + request: + method: DELETE + url: '{ot3_server_base_url}/clientData/{bad_key}' + response: + status_code: !int '{expected_status_code}' diff --git a/robot-server/tests/integration/protocols/basic_transfer_with_run_time_parameters.py b/robot-server/tests/integration/protocols/basic_transfer_with_run_time_parameters.py index 7fe90c65d8c..0e038331101 100644 --- a/robot-server/tests/integration/protocols/basic_transfer_with_run_time_parameters.py +++ b/robot-server/tests/integration/protocols/basic_transfer_with_run_time_parameters.py @@ -44,6 +44,11 @@ def add_parameters(parameters: ParameterContext): default="flex_1channel_50", description="What pipette to use during the protocol.", ) + parameters.add_csv_file( + display_name="Liquid handling CSV file", + variable_name="liq_handling_csv_file", + description="A CSV file that contains wells to use for pipetting", + ) def run(protocol: ProtocolContext) -> None: diff --git a/robot-server/tests/integration/robot_client.py b/robot-server/tests/integration/robot_client.py index b6d9614d7b2..e531eb16ea0 100644 --- a/robot-server/tests/integration/robot_client.py +++ b/robot-server/tests/integration/robot_client.py @@ -349,6 +349,11 @@ async def get_data_files_download(self, data_file_id: str) -> Response: response.raise_for_status() return response + async def delete_all_client_data(self) -> Response: + response = await self.httpx_client.delete(url=f"{self.base_url}/clientData") + response.raise_for_status() + return response + async def poll_until_run_completes( robot_client: RobotClient, run_id: str, poll_interval: float = _RUN_POLL_INTERVAL diff --git a/robot-server/tests/maintenance_runs/router/test_labware_router.py b/robot-server/tests/maintenance_runs/router/test_labware_router.py index 4d63c69eaae..cb88688731e 100644 --- a/robot-server/tests/maintenance_runs/router/test_labware_router.py +++ b/robot-server/tests/maintenance_runs/router/test_labware_router.py @@ -3,7 +3,7 @@ from datetime import datetime from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.types import DeckSlotName from opentrons.protocol_engine import EngineStatus, types as pe_types diff --git a/robot-server/tests/maintenance_runs/test_engine_store.py b/robot-server/tests/maintenance_runs/test_engine_store.py index 994b55c06b5..bf01c653df1 100644 --- a/robot-server/tests/maintenance_runs/test_engine_store.py +++ b/robot-server/tests/maintenance_runs/test_engine_store.py @@ -4,7 +4,7 @@ import pytest from decoy import Decoy, matchers -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocol_engine.errors.exceptions import EStopActivatedError from opentrons.types import DeckSlotName diff --git a/robot-server/tests/persistence/test_tables.py b/robot-server/tests/persistence/test_tables.py index f741533f1da..5b38cf1c616 100644 --- a/robot-server/tests/persistence/test_tables.py +++ b/robot-server/tests/persistence/test_tables.py @@ -12,6 +12,7 @@ schema_2, schema_4, schema_5, + schema_6, ) # The statements that we expect to emit when we create a fresh database. @@ -27,6 +28,120 @@ # # Whitespace and formatting changes, on the other hand, are allowed. EXPECTED_STATEMENTS_LATEST = [ + """ + CREATE TABLE protocol ( + id VARCHAR NOT NULL, + created_at DATETIME NOT NULL, + protocol_key VARCHAR, + protocol_kind VARCHAR(14) NOT NULL, + PRIMARY KEY (id), + CONSTRAINT protocolkindsqlenum CHECK (protocol_kind IN ('standard', 'quick-transfer')) + ) + """, + """ + CREATE TABLE analysis ( + id VARCHAR NOT NULL, + protocol_id VARCHAR NOT NULL, + analyzer_version VARCHAR NOT NULL, + completed_analysis VARCHAR NOT NULL, + PRIMARY KEY (id), + FOREIGN KEY(protocol_id) REFERENCES protocol (id) + ) + """, + """ + CREATE TABLE analysis_primitive_rtp_table ( + row_id INTEGER NOT NULL, + analysis_id VARCHAR NOT NULL, + parameter_variable_name VARCHAR NOT NULL, + parameter_type VARCHAR(5) NOT NULL, + parameter_value VARCHAR NOT NULL, + PRIMARY KEY (row_id), + FOREIGN KEY(analysis_id) REFERENCES analysis (id), + CONSTRAINT primitiveparamsqlenum CHECK (parameter_type IN ('int', 'float', 'bool', 'str')) + ) + """, + """ + CREATE TABLE analysis_csv_rtp_table ( + row_id INTEGER NOT NULL, + analysis_id VARCHAR NOT NULL, + parameter_variable_name VARCHAR NOT NULL, + file_id VARCHAR, + PRIMARY KEY (row_id), + FOREIGN KEY(analysis_id) REFERENCES analysis (id), + FOREIGN KEY(file_id) REFERENCES data_files (id) + ) + """, + """ + CREATE INDEX ix_analysis_protocol_id ON analysis (protocol_id) + """, + """ + CREATE TABLE run ( + id VARCHAR NOT NULL, + created_at DATETIME NOT NULL, + protocol_id VARCHAR, + state_summary VARCHAR, + engine_status VARCHAR, + _updated_at DATETIME, + run_time_parameters VARCHAR, + PRIMARY KEY (id), + FOREIGN KEY(protocol_id) REFERENCES protocol (id) + ) + """, + """ + CREATE TABLE action ( + id VARCHAR NOT NULL, + created_at DATETIME NOT NULL, + action_type VARCHAR NOT NULL, + run_id VARCHAR NOT NULL, + PRIMARY KEY (id), + FOREIGN KEY(run_id) REFERENCES run (id) + ) + """, + """ + CREATE TABLE run_command ( + row_id INTEGER NOT NULL, + run_id VARCHAR NOT NULL, + index_in_run INTEGER NOT NULL, + command_id VARCHAR NOT NULL, + command VARCHAR NOT NULL, + PRIMARY KEY (row_id), + FOREIGN KEY(run_id) REFERENCES run (id) + ) + """, + """ + CREATE UNIQUE INDEX ix_run_run_id_command_id ON run_command (run_id, command_id) + """, + """ + CREATE UNIQUE INDEX ix_run_run_id_index_in_run ON run_command (run_id, index_in_run) + """, + """ + CREATE INDEX ix_protocol_protocol_kind ON protocol (protocol_kind) + """, + """ + CREATE TABLE data_files ( + id VARCHAR NOT NULL, + name VARCHAR NOT NULL, + file_hash VARCHAR NOT NULL, + created_at DATETIME NOT NULL, + PRIMARY KEY (id) + ) + """, + """ + CREATE TABLE run_csv_rtp_table ( + row_id INTEGER NOT NULL, + run_id VARCHAR NOT NULL, + parameter_variable_name VARCHAR NOT NULL, + file_id VARCHAR, + PRIMARY KEY (row_id), + FOREIGN KEY(run_id) REFERENCES run (id), + FOREIGN KEY(file_id) REFERENCES data_files (id) + ) + """, +] + +EXPECTED_STATEMENTS_V6 = EXPECTED_STATEMENTS_LATEST + +EXPECTED_STATEMENTS_V5 = [ """ CREATE TABLE protocol ( id VARCHAR NOT NULL, @@ -101,8 +216,6 @@ """, ] -EXPECTED_STATEMENTS_V5 = EXPECTED_STATEMENTS_LATEST - EXPECTED_STATEMENTS_V4 = [ """ CREATE TABLE protocol ( @@ -307,6 +420,7 @@ def _normalize_statement(statement: str) -> str: ("metadata", "expected_statements"), [ (latest_metadata, EXPECTED_STATEMENTS_LATEST), + (schema_6.metadata, EXPECTED_STATEMENTS_V6), (schema_5.metadata, EXPECTED_STATEMENTS_V5), (schema_4.metadata, EXPECTED_STATEMENTS_V4), (schema_3.metadata, EXPECTED_STATEMENTS_V3), diff --git a/robot-server/tests/protocols/test_analyses_manager.py b/robot-server/tests/protocols/test_analyses_manager.py index 766908adce0..46c46a5243f 100644 --- a/robot-server/tests/protocols/test_analyses_manager.py +++ b/robot-server/tests/protocols/test_analyses_manager.py @@ -1,26 +1,29 @@ """Tests for the Analyses Manager interface.""" import pytest -from decoy import Decoy +from decoy import Decoy, matchers from datetime import datetime from pathlib import Path -from opentrons.protocol_runner import RunOrchestrator -import opentrons.protocol_runner.create_simulating_orchestrator as simulating_runner +from opentrons.protocol_engine import ErrorOccurrence from opentrons.protocol_engine.types import BooleanParameter from opentrons.protocol_reader import ProtocolSource, JsonProtocolConfig -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.errors import EnumeratedError, ErrorCodes +from opentrons_shared_data.robot.types import RobotType from robot_server.protocols import protocol_analyzer from robot_server.protocols.protocol_models import ProtocolKind -from robot_server.protocols.analyses_manager import AnalysesManager +from robot_server.protocols.analyses_manager import ( + AnalysesManager, + FailedToInitializeAnalyzer, +) from robot_server.protocols.analysis_models import ( AnalysisSummary, AnalysisStatus, - PendingAnalysis, ) from robot_server.protocols.analysis_store import AnalysisStore from robot_server.protocols.protocol_store import ProtocolResource from robot_server.service.task_runner import TaskRunner +import robot_server.errors.error_mappers as em @pytest.fixture @@ -45,12 +48,12 @@ def patch_mock_create_protocol_analyzer( @pytest.fixture(autouse=True) -def patch_mock_create_simulating_orchestrator( +def patch_mock_map_unexpected_error( decoy: Decoy, monkeypatch: pytest.MonkeyPatch ) -> None: - """Replace protocol_runner.check() with a mock.""" - mock = decoy.mock(func=simulating_runner.create_simulating_orchestrator) - monkeypatch.setattr(simulating_runner, "create_simulating_orchestrator", mock) + """Replace map_unexpected_error with a mock.""" + mock_map_unexpected_error = decoy.mock(func=em.map_unexpected_error) + monkeypatch.setattr(em, "map_unexpected_error", mock_map_unexpected_error) @pytest.fixture @@ -59,13 +62,13 @@ def subject(analysis_store: AnalysisStore, task_runner: TaskRunner) -> AnalysesM return AnalysesManager(analysis_store=analysis_store, task_runner=task_runner) -async def test_start_analysis( +async def test_initialize_analyzer( decoy: Decoy, analysis_store: AnalysisStore, task_runner: TaskRunner, subject: AnalysesManager, ) -> None: - """It should start protocol analysis and return summary with run time parameters.""" + """It should create analyzer and load its orchestrator.""" robot_type: RobotType = "OT-3 Standard" protocol_resource = ProtocolResource( protocol_id="protocol-id", @@ -80,62 +83,41 @@ async def test_start_analysis( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind=ProtocolKind.STANDARD.value, - ) - bool_parameter = BooleanParameter( - displayName="Foo", variableName="Bar", default=True, value=False + protocol_kind=ProtocolKind.STANDARD, ) - pending_analysis = PendingAnalysis(id="analysis-id") analyzer = decoy.mock(cls=protocol_analyzer.ProtocolAnalyzer) - orchestrator = decoy.mock(cls=RunOrchestrator) decoy.when( protocol_analyzer.create_protocol_analyzer( analysis_store=analysis_store, protocol_resource=protocol_resource, ) ).then_return(analyzer) - decoy.when( - analysis_store.add_pending( - protocol_id="protocol-id", - analysis_id="analysis-id", - ) - ).then_return(pending_analysis) - decoy.when( - await analyzer.load_runner(run_time_param_values={"baz": True}) - ).then_return(orchestrator) - decoy.when(orchestrator.get_run_time_parameters()).then_return([bool_parameter]) - analysis_summary_result = await subject.start_analysis( + + await subject.initialize_analyzer( analysis_id="analysis-id", protocol_resource=protocol_resource, - run_time_param_values={"baz": True}, - ) - - assert analysis_summary_result == AnalysisSummary( - id="analysis-id", - status=AnalysisStatus.PENDING, - runTimeParameters=[bool_parameter], + run_time_param_values={"sample_count": 123}, + run_time_param_files={"my_file": "file-id"}, ) decoy.verify( - task_runner.run( - analyzer.analyze, - orchestrator=orchestrator, - analysis_id="analysis-id", - run_time_parameters=[bool_parameter], + await analyzer.load_orchestrator( + run_time_param_values={"sample_count": 123}, + run_time_param_files={"my_file": "file-id"}, ) ) -async def test_rtp_validation_error_in_start_analysis( +async def test_raises_error_and_saves_result_if_initialization_errors( decoy: Decoy, analysis_store: AnalysisStore, task_runner: TaskRunner, subject: AnalysesManager, ) -> None: - """It should catch RTP validation error early and cleanup the analysis process.""" + """It should save the result to analysis store and re-raise error when analyzer initialization errors out.""" robot_type: RobotType = "OT-3 Standard" protocol_resource = ProtocolResource( protocol_id="protocol-id", - created_at=datetime(year=2024, month=6, day=6), + created_at=datetime(year=2021, month=1, day=1), source=ProtocolSource( directory=Path("/dev/null"), main_file=Path("/dev/null/abc.json"), @@ -146,11 +128,13 @@ async def test_rtp_validation_error_in_start_analysis( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, + ) + raised_exception = Exception("Oh noooo!") + enumerated_error = EnumeratedError( + code=ErrorCodes.GENERAL_ERROR, + message="You got me!!", ) - - runner_load_exception = Exception("Uh oh!") - pending_analysis = PendingAnalysis(id="analysis-id") analyzer = decoy.mock(cls=protocol_analyzer.ProtocolAnalyzer) decoy.when( protocol_analyzer.create_protocol_analyzer( @@ -159,28 +143,88 @@ async def test_rtp_validation_error_in_start_analysis( ) ).then_return(analyzer) decoy.when( - analysis_store.add_pending( + await analyzer.load_orchestrator( + run_time_param_values={"sample_count": 123}, + run_time_param_files={}, + ) + ).then_raise(raised_exception) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return([]) + decoy.when(em.map_unexpected_error(error=raised_exception)).then_return( + enumerated_error + ) + with pytest.raises(FailedToInitializeAnalyzer): + await subject.initialize_analyzer( + analysis_id="analysis-id", + protocol_resource=protocol_resource, + run_time_param_values={"sample_count": 123}, + run_time_param_files={}, + ) + decoy.verify( + await analysis_store.save_initialization_failed_analysis( protocol_id="protocol-id", analysis_id="analysis-id", + robot_type=robot_type, + run_time_parameters=[], + errors=[ + ErrorOccurrence.from_failed( + id="internal-error", + createdAt=matchers.IsA(datetime), + error=enumerated_error, + ) + ], ) - ).then_return(pending_analysis) - decoy.when( - await analyzer.load_runner(run_time_param_values={"baz": True}) - ).then_raise(runner_load_exception) + ) + + +async def test_start_analysis( + decoy: Decoy, + analysis_store: AnalysisStore, + task_runner: TaskRunner, + subject: AnalysesManager, +) -> None: + """It should start protocol analysis and return summary with run time parameters.""" + robot_type: RobotType = "OT-3 Standard" + protocol_resource = ProtocolResource( + protocol_id="protocol-id", + created_at=datetime(year=2021, month=1, day=1), + source=ProtocolSource( + directory=Path("/dev/null"), + main_file=Path("/dev/null/abc.json"), + config=JsonProtocolConfig(schema_version=123), + files=[], + metadata={}, + robot_type=robot_type, + content_hash="abc123", + ), + protocol_key="dummy-data-111", + protocol_kind=ProtocolKind.STANDARD, + ) + bool_parameter = BooleanParameter( + displayName="Foo", variableName="Bar", default=True, value=False + ) + analyzer = decoy.mock(cls=protocol_analyzer.ProtocolAnalyzer) + decoy.when(analyzer.protocol_resource).then_return(protocol_resource) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [bool_parameter] + ) analysis_summary_result = await subject.start_analysis( analysis_id="analysis-id", - protocol_resource=protocol_resource, - run_time_param_values={"baz": True}, + analyzer=analyzer, ) assert analysis_summary_result == AnalysisSummary( - id="analysis-id", status=AnalysisStatus.COMPLETED + id="analysis-id", + status=AnalysisStatus.PENDING, + runTimeParameters=[bool_parameter], ) decoy.verify( - await analyzer.update_to_failed_analysis( + analysis_store.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id", - protocol_robot_type=protocol_resource.source.robot_type, - error=runner_load_exception, - run_time_parameters=[], - ) + run_time_parameters=[bool_parameter], + ), + task_runner.run( + analyzer.analyze, + analysis_id="analysis-id", + ), ) diff --git a/robot-server/tests/protocols/test_analysis_store.py b/robot-server/tests/protocols/test_analysis_store.py index 606d6f13f10..7a3d979be44 100644 --- a/robot-server/tests/protocols/test_analysis_store.py +++ b/robot-server/tests/protocols/test_analysis_store.py @@ -3,15 +3,23 @@ from datetime import datetime, timezone from pathlib import Path -from typing import List, NamedTuple +from typing import List, NamedTuple, Optional import pytest from decoy import Decoy -from opentrons.protocol_engine.types import RunTimeParamValuesType +from opentrons.protocol_engine.types import ( + RunTimeParameter, + NumberParameter, + EnumParameter, + EnumChoice, + BooleanParameter, + CSVParameter, + FileInfo, +) from sqlalchemy.engine import Engine as SQLEngine -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.errors import ErrorCodes from opentrons.types import MountType, DeckSlotName @@ -31,7 +39,6 @@ AnalysisSummary, PendingAnalysis, CompletedAnalysis, - RunTimeParameterAnalysisData, ) from robot_server.protocols.analysis_store import ( AnalysisStore, @@ -43,10 +50,15 @@ CompletedAnalysisStore, CompletedAnalysisResource, ) +from robot_server.protocols.protocol_models import ProtocolKind from robot_server.protocols.protocol_store import ( ProtocolStore, ProtocolResource, ) +from robot_server.protocols.rtp_resources import ( + PrimitiveParameterResource, + CSVParameterResource, +) @pytest.fixture @@ -85,7 +97,52 @@ def make_dummy_protocol_resource(protocol_id: str) -> ProtocolResource: content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, + ) + + +def mock_number_param(name: str, value: float) -> NumberParameter: + """Return a NumberParameter.""" + return NumberParameter( + variableName=name, + displayName="num param", + value=value, + default=3.0, + max=10, + min=0, + type="float", + ) + + +def mock_csv_param(name: str, file: Optional[FileInfo]) -> CSVParameter: + """Return a CSVParameter.""" + return CSVParameter( + variableName=name, + displayName="csv param", + file=file, + ) + + +def mock_enum_param(name: str, value: str) -> EnumParameter: + """Return a EnumParameter.""" + return EnumParameter( + variableName=name, + displayName="enum param", + type="str", + value=value, + default="blah", + choices=[EnumChoice(displayName="floo", value="barr")], + ) + + +def mock_bool_param(name: str, value: bool) -> BooleanParameter: + """Return a BooleanParameter.""" + return BooleanParameter( + variableName=name, + displayName="enum param", + type="bool", + value=value, + default=False, ) @@ -115,11 +172,10 @@ async def test_add_pending( status=AnalysisStatus.PENDING, ) - result = subject.add_pending(protocol_id="protocol-id", analysis_id="analysis-id") - assert result == expected_analysis + subject.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id", run_time_parameters=[] + ) - analysis_result = await subject.get("analysis-id") - assert analysis_result == expected_analysis assert await subject.get_by_protocol("protocol-id") == [expected_analysis] assert subject.get_summaries_by_protocol("protocol-id") == [expected_summary] with pytest.raises(AnalysisNotFoundError, match="analysis-id"): @@ -134,7 +190,9 @@ async def test_returned_in_order_added( protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) for analysis_id in ["analysis-id-1", "analysis-id-2", "analysis-id-3"]: - subject.add_pending(protocol_id="protocol-id", analysis_id=analysis_id) + subject.add_pending( + protocol_id="protocol-id", analysis_id=analysis_id, run_time_parameters=[] + ) await subject.update( analysis_id=analysis_id, robot_type="OT-2 Standard", @@ -147,7 +205,9 @@ async def test_returned_in_order_added( liquids=[], ) - subject.add_pending(protocol_id="protocol-id", analysis_id="analysis-id-4") + subject.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id-4", run_time_parameters=[] + ) # Leave as pending, to test that we interleave completed & pending analyses # in the correct order. @@ -191,7 +251,9 @@ async def test_update_adds_details_and_completes_analysis( value=2.0, default=3.0, ) - subject.add_pending(protocol_id="protocol-id", analysis_id="analysis-id") + subject.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id", run_time_parameters=[] + ) await subject.update( analysis_id="analysis-id", robot_type="OT-2 Standard", @@ -257,7 +319,7 @@ async def test_update_adds_details_and_completes_analysis( } -async def test_update_adds_rtp_values_and_defaults_to_completed_store( +async def test_update_adds_rtp_values_to_completed_store( decoy: Decoy, sql_engine: SQLEngine, protocol_store: ProtocolStore ) -> None: """It should add RTP values and defaults to completed analysis store.""" @@ -281,6 +343,11 @@ async def test_update_adds_rtp_values_and_defaults_to_completed_store( value="baz", default="blah", ) + csv_param = pe_types.CSVParameter( + displayName="A CSV param", + variableName="coolest_param", + file=FileInfo(id="file-id", name="file-name"), + ) expected_completed_analysis_resource = CompletedAnalysisResource( id="analysis-id", protocol_id="protocol-id", @@ -290,7 +357,7 @@ async def test_update_adds_rtp_values_and_defaults_to_completed_store( status=AnalysisStatus.COMPLETED, result=AnalysisResult.OK, robotType="OT-2 Standard", - runTimeParameters=[number_param, string_param], + runTimeParameters=[number_param, string_param, csv_param], labware=[], pipettes=[], modules=[], @@ -298,21 +365,19 @@ async def test_update_adds_rtp_values_and_defaults_to_completed_store( errors=[], liquids=[], ), - run_time_parameter_values_and_defaults={ - "cool_param": RunTimeParameterAnalysisData(value=2.0, default=3.0), - "cooler_param": RunTimeParameterAnalysisData(value="baz", default="blah"), - }, ) mock_completed_store = decoy.mock(cls=CompletedAnalysisStore) subject = AnalysisStore(sql_engine=sql_engine, completed_store=mock_completed_store) protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) - subject.add_pending(protocol_id="protocol-id", analysis_id="analysis-id") + subject.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id", run_time_parameters=[] + ) await subject.update( analysis_id="analysis-id", robot_type="OT-2 Standard", - run_time_parameters=[number_param, string_param], + run_time_parameters=[number_param, string_param, csv_param], labware=[], pipettes=[], modules=[], @@ -322,7 +387,28 @@ async def test_update_adds_rtp_values_and_defaults_to_completed_store( ) decoy.verify( await mock_completed_store.make_room_and_add( - completed_analysis_resource=expected_completed_analysis_resource + completed_analysis_resource=expected_completed_analysis_resource, + primitive_rtp_resources=[ + PrimitiveParameterResource( + analysis_id="analysis-id", + parameter_variable_name="cool_param", + parameter_type="int", + parameter_value=2.0, + ), + PrimitiveParameterResource( + analysis_id="analysis-id", + parameter_variable_name="cooler_param", + parameter_type="str", + parameter_value="baz", + ), + ], + csv_rtp_resources=[ + CSVParameterResource( + analysis_id="analysis-id", + parameter_variable_name="coolest_param", + file_id="file-id", + ), + ], ) ) @@ -388,7 +474,9 @@ async def test_update_infers_status_from_errors( ) -> None: """It should decide the analysis result based on whether there are errors.""" protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) - subject.add_pending(protocol_id="protocol-id", analysis_id="analysis-id") + subject.add_pending( + protocol_id="protocol-id", analysis_id="analysis-id", run_time_parameters=[] + ) await subject.update( analysis_id="analysis-id", robot_type="OT-2 Standard", @@ -405,30 +493,117 @@ async def test_update_infers_status_from_errors( assert analysis.result == expected_result +async def test_save_initialization_failed_analysis( + decoy: Decoy, sql_engine: SQLEngine, protocol_store: ProtocolStore +) -> None: + """It should save the analysis that failed during analyzer initialization.""" + validated_rtp = NumberParameter( + displayName="My parameter", + variableName="cool_param", + type="int", + min=1, + max=5, + value=2.0, + default=3.0, + ) + error_occurence = pe_errors.ErrorOccurrence( + id="error-id", + createdAt=datetime(year=2021, month=1, day=1, tzinfo=timezone.utc), + errorType="BadError", + detail="oh no", + ) + expected_completed_analysis_resource = CompletedAnalysisResource( + id="analysis-id", + protocol_id="protocol-id", + analyzer_version=_CURRENT_ANALYZER_VERSION, + completed_analysis=CompletedAnalysis( + id="analysis-id", + status=AnalysisStatus.COMPLETED, + result=AnalysisResult.NOT_OK, + robotType="OT-2 Standard", + runTimeParameters=[validated_rtp], + labware=[], + pipettes=[], + modules=[], + commands=[], + errors=[error_occurence], + liquids=[], + ), + ) + + mock_completed_store = decoy.mock(cls=CompletedAnalysisStore) + subject = AnalysisStore(sql_engine=sql_engine, completed_store=mock_completed_store) + protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) + + await subject.save_initialization_failed_analysis( + protocol_id="protocol-id", + analysis_id="analysis-id", + robot_type="OT-2 Standard", + run_time_parameters=[validated_rtp], + errors=[error_occurence], + ) + decoy.verify( + await mock_completed_store.make_room_and_add( + completed_analysis_resource=expected_completed_analysis_resource, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) + ) + + @pytest.mark.parametrize( - argnames=["rtp_values_from_client", "expected_match"], + argnames=["parameters_from_client", "expected_match"], argvalues=[ - ({"cool_param": 2.0, "cooler_param": "baz", "uncool_param": 5}, True), ( - {"cool_param": 2, "cooler_param": "baz"}, + [ + mock_number_param("cool_param", 2.0), + mock_enum_param("cooler_param", "baz"), + mock_bool_param("uncool_param", True), + mock_csv_param( + "coolest_param", FileInfo(id="file-id", name="file-name") + ), + ], True, ), ( - {"cool_param": 2, "cooler_param": "buzzzzzzz"}, + [ # Params have non-matching values + mock_number_param("cool_param", 2), + mock_enum_param("cooler_param", "buzzzzz"), + mock_bool_param("uncool_param", False), + mock_csv_param( + "coolest_param", FileInfo(id="file-id", name="file-name") + ), + ], False, ), ( - {"cool_param": 2.0, "cooler_param": "baz", "weird_param": 5}, + [ + # params in different order, cool param is '2' instead of '2.0' + mock_enum_param("cooler_param", "baz"), + mock_bool_param("uncool_param", True), + mock_csv_param( + "coolest_param", FileInfo(id="file-id", name="file-name") + ), + mock_number_param("cool_param", 2), + ], + True, + ), + ( + [ # CSV param file ID is None + mock_number_param("cool_param", 2.0), + mock_enum_param("cooler_param", "baz"), + mock_bool_param("uncool_param", True), + mock_csv_param("coolest_param", None), + ], False, ), - ({}, False), ], ) async def test_matching_rtp_values_in_analysis( decoy: Decoy, sql_engine: SQLEngine, protocol_store: ProtocolStore, - rtp_values_from_client: RunTimeParamValuesType, + parameters_from_client: List[RunTimeParameter], expected_match: bool, ) -> None: """It should return whether the client's RTP values match with those in the last analysis of protocol.""" @@ -437,59 +612,25 @@ async def test_matching_rtp_values_in_analysis( protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) decoy.when( - await mock_completed_store.get_rtp_values_and_defaults_by_analysis_id( - "analysis-2" - ) + mock_completed_store.get_primitive_rtps_by_analysis_id("analysis-2") ).then_return( { - "cool_param": RunTimeParameterAnalysisData(value=2.0, default=3.0), - "cooler_param": RunTimeParameterAnalysisData( - value="baz", default="very cool" - ), - "uncool_param": RunTimeParameterAnalysisData(value=5, default=5), + "cool_param": 2.0, + "cooler_param": "baz", + "uncool_param": True, } ) - assert ( - await subject.matching_rtp_values_in_analysis( - analysis_summary=AnalysisSummary( - id="analysis-2", status=AnalysisStatus.COMPLETED - ), - new_rtp_values=rtp_values_from_client, - ) - == expected_match - ) - - -async def test_matching_default_rtp_values_in_analysis_with_no_client_rtp_values( - decoy: Decoy, - sql_engine: SQLEngine, - protocol_store: ProtocolStore, -) -> None: - """It should return a match when client sends no RTP values and last analysis used all default values.""" - params_with_only_default_values = { - "cool_param": RunTimeParameterAnalysisData(value=2.0, default=2.0), - "cooler_param": RunTimeParameterAnalysisData( - value="very cool", default="very cool" - ), - "uncool_param": RunTimeParameterAnalysisData(value=True, default=True), - } - mock_completed_store = decoy.mock(cls=CompletedAnalysisStore) - subject = AnalysisStore(sql_engine=sql_engine, completed_store=mock_completed_store) - protocol_store.insert(make_dummy_protocol_resource(protocol_id="protocol-id")) - decoy.when( - await mock_completed_store.get_rtp_values_and_defaults_by_analysis_id( - "analysis-2" - ) - ).then_return(params_with_only_default_values) + mock_completed_store.get_csv_rtps_by_analysis_id("analysis-2") + ).then_return({"coolest_param": "file-id"}) assert ( await subject.matching_rtp_values_in_analysis( - analysis_summary=AnalysisSummary( + last_analysis_summary=AnalysisSummary( id="analysis-2", status=AnalysisStatus.COMPLETED ), - new_rtp_values={}, + new_parameters=parameters_from_client, ) - is True + == expected_match ) @@ -499,5 +640,5 @@ async def test_matching_default_rtp_values_in_analysis_with_pending_analysis( """It should raise an error if analysis is pending.""" with pytest.raises(AnalysisIsPendingError): await subject.matching_rtp_values_in_analysis( - AnalysisSummary(id="analysis-id", status=AnalysisStatus.PENDING), {} + AnalysisSummary(id="analysis-id", status=AnalysisStatus.PENDING), [] ) diff --git a/robot-server/tests/protocols/test_completed_analysis_store.py b/robot-server/tests/protocols/test_completed_analysis_store.py index 9dac72db211..3f1e5302bdf 100644 --- a/robot-server/tests/protocols/test_completed_analysis_store.py +++ b/robot-server/tests/protocols/test_completed_analysis_store.py @@ -8,7 +8,11 @@ from sqlalchemy.engine import Engine from decoy import Decoy -from robot_server.persistence.tables import analysis_table +from robot_server.persistence.tables import ( + analysis_table, + analysis_primitive_type_rtp_table, + analysis_csv_rtp_table, +) from robot_server.protocols.completed_analysis_store import ( CompletedAnalysisResource, CompletedAnalysisStore, @@ -17,6 +21,7 @@ ProtocolSource, JsonProtocolConfig, ) +from robot_server.data_files.data_files_store import DataFilesStore, DataFileInfo from robot_server.protocols.analysis_memcache import MemoryCache from robot_server.protocols.analysis_models import ( CompletedAnalysis, @@ -24,10 +29,15 @@ AnalysisStatus, RunTimeParameterAnalysisData, ) +from robot_server.protocols.protocol_models import ProtocolKind from robot_server.protocols.protocol_store import ( ProtocolStore, ProtocolResource, ) +from robot_server.protocols.rtp_resources import ( + PrimitiveParameterResource, + CSVParameterResource, +) @pytest.fixture @@ -56,6 +66,17 @@ def protocol_store(sql_engine: Engine) -> ProtocolStore: return ProtocolStore.create_empty(sql_engine=sql_engine) +@pytest.fixture +def data_files_store(sql_engine: Engine) -> DataFilesStore: + """Return a `DataFilesStore` linked to the same database as the subject under test. + + `DataFilesStore` is tested elsewhere. + We only need it here to prepare the database for the analysis store tests. + The CSV parameters table always needs a data file to link to. + """ + return DataFilesStore(sql_engine=sql_engine) + + def make_dummy_protocol_resource(protocol_id: str) -> ProtocolResource: """Return a placeholder `ProtocolResource` to insert into a `ProtocolStore`. @@ -75,7 +96,7 @@ def make_dummy_protocol_resource(protocol_id: str) -> ProtocolResource: content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) @@ -99,7 +120,6 @@ def _completed_analysis_resource( errors=[], liquids=[], ), - run_time_parameter_values_and_defaults=rtp_values_and_defaults or {}, ) @@ -127,7 +147,11 @@ async def test_get_by_analysis_id_falls_back_to_sql( """It should return analyses from sql if they are not cached.""" resource = _completed_analysis_resource("analysis-id", "protocol-id") protocol_store.insert(make_dummy_protocol_resource("protocol-id")) - await subject.make_room_and_add(resource) + await subject.make_room_and_add( + completed_analysis_resource=resource, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) # the analysis is not cached decoy.when(memcache.get("analysis-id")).then_raise(KeyError()) analysis_from_sql = await subject.get_by_id("analysis-id") @@ -144,7 +168,11 @@ async def test_get_by_analysis_id_stores_results_in_cache( """It should cache successful fetches from sql.""" resource = _completed_analysis_resource("analysis-id", "protocol-id") protocol_store.insert(make_dummy_protocol_resource("protocol-id")) - await subject.make_room_and_add(resource) + await subject.make_room_and_add( + completed_analysis_resource=resource, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) # the analysis is not cached decoy.when(memcache.get("analysis-id")).then_raise(KeyError()) from_sql = await subject.get_by_id("analysis-id") @@ -159,7 +187,11 @@ async def test_get_by_analysis_id_as_document( """It should return the analysis serialized as a JSON string.""" resource = _completed_analysis_resource("analysis-id", "protocol-id") protocol_store.insert(make_dummy_protocol_resource("protocol-id")) - await subject.make_room_and_add(resource) + await subject.make_room_and_add( + completed_analysis_resource=resource, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) result = await subject.get_by_id_as_document("analysis-id") assert result is not None assert json.loads(result) == { @@ -185,9 +217,9 @@ async def test_get_ids_by_protocol( resource_3 = _completed_analysis_resource("analysis-id-3", "protocol-id-2") protocol_store.insert(make_dummy_protocol_resource("protocol-id-1")) protocol_store.insert(make_dummy_protocol_resource("protocol-id-2")) - await subject.make_room_and_add(resource_1) - await subject.make_room_and_add(resource_2) - await subject.make_room_and_add(resource_3) + await subject.make_room_and_add(resource_1, [], []) + await subject.make_room_and_add(resource_2, [], []) + await subject.make_room_and_add(resource_3, [], []) assert subject.get_ids_by_protocol("protocol-id-1") == [ "analysis-id-1", "analysis-id-2", @@ -209,9 +241,9 @@ async def test_get_by_protocol( decoy.when(memcache.insert("analysis-id-1", resource_1)).then_return(None) decoy.when(memcache.insert("analysis-id-2", resource_2)).then_return(None) decoy.when(memcache.insert("analysis-id-3", resource_3)).then_return(None) - await subject.make_room_and_add(resource_1) - await subject.make_room_and_add(resource_2) - await subject.make_room_and_add(resource_3) + await subject.make_room_and_add(resource_1, [], []) + await subject.make_room_and_add(resource_2, [], []) + await subject.make_room_and_add(resource_3, [], []) decoy.when(memcache.get("analysis-id-1")).then_raise(KeyError()) decoy.when(memcache.get("analysis-id-2")).then_return(resource_2) decoy.when(memcache.contains("analysis-id-1")).then_return(False) @@ -221,48 +253,89 @@ async def test_get_by_protocol( assert resources == [resource_1, resource_2] -async def test_get_rtp_values_and_defaults_by_analysis_id_prefers_memcache( +async def test_store_and_get_primitive_rtps_by_analysis( subject: CompletedAnalysisStore, - memcache: MemoryCache[str, CompletedAnalysisResource], protocol_store: ProtocolStore, - decoy: Decoy, ) -> None: - """It should return RTP values and defaults dict from memcache.""" - resource = _completed_analysis_resource( + """It should store the primitive run time parameters & fetch them using analysis ID.""" + analysis_resource = _completed_analysis_resource( analysis_id="analysis-id", protocol_id="protocol-id", - rtp_values_and_defaults={ - "abc": RunTimeParameterAnalysisData(value=123, default=234) - }, ) + rtp_resources = [ + PrimitiveParameterResource( + analysis_id="analysis-id", + parameter_variable_name="foo", + parameter_type="int", + parameter_value=10, + ), + PrimitiveParameterResource( + analysis_id="analysis-id", + parameter_variable_name="bar", + parameter_type="bool", + parameter_value=True, + ), + PrimitiveParameterResource( + analysis_id="analysis-id", + parameter_variable_name="baz", + parameter_type="str", + parameter_value="10", + ), + ] protocol_store.insert(make_dummy_protocol_resource("protocol-id")) - # When we retrieve a resource via its id we should see it query the cache, and it should - # return the identity-same resource - decoy.when(memcache.get("analysis-id")).then_return(resource) - result = await subject.get_rtp_values_and_defaults_by_analysis_id("analysis-id") - assert result == resource.run_time_parameter_values_and_defaults + + await subject.make_room_and_add( + completed_analysis_resource=analysis_resource, + primitive_rtp_resources=rtp_resources, + csv_rtp_resources=[], + ) + assert subject.get_primitive_rtps_by_analysis_id("analysis-id") == { + "foo": 10, + "bar": True, + "baz": "10", + } -async def test_get_rtp_values_and_defaults_by_analysis_from_db( +async def test_store_and_get_csv_rtps_by_analysis_id( subject: CompletedAnalysisStore, - memcache: MemoryCache[str, CompletedAnalysisResource], protocol_store: ProtocolStore, - decoy: Decoy, + data_files_store: DataFilesStore, ) -> None: - """It should fetch the RTP values and defaults dict from database if not present in cache.""" - resource = _completed_analysis_resource( + """It should store the CSV run time parameters & fetch them using analysis ID.""" + analysis_resource = _completed_analysis_resource( analysis_id="analysis-id", protocol_id="protocol-id", - rtp_values_and_defaults={ - "xyz": RunTimeParameterAnalysisData(value=123, default=234) - }, ) + csv_rtp_resources = [ + CSVParameterResource( + analysis_id="analysis-id", + parameter_variable_name="baz", + file_id="file-id", + ), + CSVParameterResource( + analysis_id="analysis-id", + parameter_variable_name="bar", + file_id=None, + ), + ] protocol_store.insert(make_dummy_protocol_resource("protocol-id")) - await subject.make_room_and_add(resource) - # Not in memcache - decoy.when(memcache.get("analysis-id")).then_raise(KeyError()) - result = await subject.get_rtp_values_and_defaults_by_analysis_id("analysis-id") - assert result == resource.run_time_parameter_values_and_defaults + await data_files_store.insert( + DataFileInfo( + id="file-id", + name="my_csv_file.csv", + file_hash="file-hash", + created_at=datetime(year=2024, month=1, day=1, tzinfo=timezone.utc), + ) + ) + await subject.make_room_and_add( + completed_analysis_resource=analysis_resource, + primitive_rtp_resources=[], + csv_rtp_resources=csv_rtp_resources, + ) + assert subject.get_csv_rtps_by_analysis_id("analysis-id") == { + "baz": "file-id", + "bar": None, + } @pytest.mark.parametrize( @@ -342,10 +415,12 @@ async def test_add_makes_room_for_new_analysis( assert subject.get_ids_by_protocol("protocol-id") == existing_analysis_ids await subject.make_room_and_add( - _completed_analysis_resource( + completed_analysis_resource=_completed_analysis_resource( analysis_id="new-analysis-id", protocol_id="protocol-id", - ) + ), + primitive_rtp_resources=[], + csv_rtp_resources=[], ) assert ( subject.get_ids_by_protocol("protocol-id") @@ -359,3 +434,117 @@ async def test_add_makes_room_for_new_analysis( ] for analysis_id in removed_ids: decoy.verify(memcache.remove(analysis_id)) + + +async def test_make_room_and_add_handles_rtp_tables_correctly( + subject: CompletedAnalysisStore, + memcache: MemoryCache[str, CompletedAnalysisResource], + protocol_store: ProtocolStore, + data_files_store: DataFilesStore, + sql_engine: Engine, +) -> None: + """It should delete any RTP table entries that reference the analyses being deleted, and then insert new RTP entries.""" + existing_analysis_ids = [ + "analysis-id-0", + "analysis-id-1", + "analysis-id-2", + "analysis-id-3", + "analysis-id-4", + ] + + protocol_store.insert(make_dummy_protocol_resource("protocol-id")) + await data_files_store.insert( + DataFileInfo( + id="file-id", + name="my_csv_file.csv", + file_hash="file-hash", + created_at=datetime(year=2024, month=1, day=1, tzinfo=timezone.utc), + ) + ) + # Set up the database with existing analyses + resources = [ + _completed_analysis_resource( + analysis_id=analysis_id, + protocol_id="protocol-id", + ) + for analysis_id in existing_analysis_ids + ] + existing_primitive_rtp_resources = [ + PrimitiveParameterResource( + analysis_id="analysis-id-0", + parameter_variable_name="foo", + parameter_type="int", + parameter_value=10, + ), + PrimitiveParameterResource( + analysis_id="analysis-id-2", + parameter_variable_name="bar", + parameter_type="bool", + parameter_value=True, + ), + ] + existing_csv_rtp_resources = [ + CSVParameterResource( + analysis_id="analysis-id-0", + parameter_variable_name="baz", + file_id="file-id", + ), + CSVParameterResource( + analysis_id="analysis-id-1", + parameter_variable_name="bar", + file_id=None, + ), + ] + for resource in resources: + statement = analysis_table.insert().values(await resource.to_sql_values()) + with sql_engine.begin() as transaction: + transaction.execute(statement) + for primitive_rtp_resource in existing_primitive_rtp_resources: + statement = analysis_primitive_type_rtp_table.insert().values( + primitive_rtp_resource.to_sql_values() + ) + with sql_engine.begin() as transaction: + transaction.execute(statement) + for csv_resource in existing_csv_rtp_resources: + statement = analysis_csv_rtp_table.insert().values(csv_resource.to_sql_values()) + with sql_engine.begin() as transaction: + transaction.execute(statement) + + assert subject.get_ids_by_protocol("protocol-id") == existing_analysis_ids + await subject.make_room_and_add( + completed_analysis_resource=_completed_analysis_resource( + analysis_id="new-analysis-id", + protocol_id="protocol-id", + ), + primitive_rtp_resources=[ + PrimitiveParameterResource( + analysis_id="new-analysis-id", + parameter_variable_name="baz", + parameter_type="str", + parameter_value="10", + ) + ], + csv_rtp_resources=[ + CSVParameterResource( + analysis_id="new-analysis-id", + parameter_variable_name="bar", + file_id="file-id", + ) + ], + ) + + assert subject.get_ids_by_protocol("protocol-id") == [ + "analysis-id-1", + "analysis-id-2", + "analysis-id-3", + "analysis-id-4", + "new-analysis-id", + ] + + assert subject.get_primitive_rtps_by_analysis_id("analysis-id-2") == {"bar": True} + assert subject.get_primitive_rtps_by_analysis_id("analysis-id-0") == {} + assert subject.get_primitive_rtps_by_analysis_id("new-analysis-id") == {"baz": "10"} + + assert subject.get_csv_rtps_by_analysis_id("analysis-id-0") == {} + assert subject.get_csv_rtps_by_analysis_id("analysis-id-1") == {"bar": None} + assert subject.get_csv_rtps_by_analysis_id("new-analysis-id") == {"bar": "file-id"} diff --git a/robot-server/tests/protocols/test_protocol_analyzer.py b/robot-server/tests/protocols/test_protocol_analyzer.py index 44fde5bd6c2..1e9c004999a 100644 --- a/robot-server/tests/protocols/test_protocol_analyzer.py +++ b/robot-server/tests/protocols/test_protocol_analyzer.py @@ -5,8 +5,8 @@ from pathlib import Path from opentrons.protocols.api_support.types import APIVersion -from opentrons_shared_data.robot.dev_types import RobotType -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.robot.types import RobotType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons.types import MountType, DeckSlotName from opentrons.protocol_engine import ( @@ -67,11 +67,11 @@ def analysis_store(decoy: Decoy) -> AnalysisStore: return decoy.mock(cls=AnalysisStore) -async def test_load_runner( +async def test_load_orchestrator( decoy: Decoy, analysis_store: AnalysisStore, ) -> None: - """It should load the appropriate runner.""" + """It should load the appropriate run orchestrator.""" robot_type: RobotType = "OT-3 Standard" protocol_source = ProtocolSource( directory=Path("/dev/null"), @@ -87,29 +87,30 @@ async def test_load_runner( created_at=datetime(year=2021, month=1, day=1), source=protocol_source, protocol_key="dummy-data-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) - subject = ProtocolAnalyzer( analysis_store=analysis_store, protocol_resource=protocol_resource ) run_orchestrator = decoy.mock(cls=protocol_runner.RunOrchestrator) - python_runner = decoy.mock(cls=protocol_runner.PythonAndLegacyRunner) - decoy.when(run_orchestrator.get_protocol_runner()).then_return(python_runner) decoy.when( await simulating_runner.create_simulating_orchestrator( robot_type=robot_type, protocol_config=PythonProtocolConfig(api_version=APIVersion(100, 200)), ) ).then_return(run_orchestrator) - runner = await subject.load_runner(run_time_param_values={"rtp_var": 123}) - assert runner.get_protocol_runner() == run_orchestrator.get_protocol_runner() + await subject.load_orchestrator( + run_time_param_values={"rtp_var": 123}, + run_time_param_files={"csv_param": "file-id"}, + ) + decoy.verify( await run_orchestrator.load( protocol_source=protocol_source, parse_mode=ParseMode.NORMAL, run_time_param_values={"rtp_var": 123}, + run_time_param_files={"csv_param": "file-id"}, ), times=1, ) @@ -119,7 +120,7 @@ async def test_analyze( decoy: Decoy, analysis_store: AnalysisStore, ) -> None: - """It should be able to start a protocol analysis and return the analysis summary.""" + """It should be able to start a protocol analysis and update the analysis store when completed.""" robot_type: RobotType = "OT-3 Standard" protocol_resource = ProtocolResource( @@ -135,7 +136,7 @@ async def test_analyze( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) analysis_command = pe_commands.WaitForResume( @@ -165,10 +166,18 @@ async def test_analyze( ) orchestrator = decoy.mock(cls=protocol_runner.RunOrchestrator) + decoy.when( + await simulating_runner.create_simulating_orchestrator( + robot_type=robot_type, + protocol_config=JsonProtocolConfig(schema_version=123), + ) + ).then_return(orchestrator) subject = ProtocolAnalyzer( analysis_store=analysis_store, protocol_resource=protocol_resource ) - + await subject.load_orchestrator( + run_time_param_values={"rtp_var": 123}, run_time_param_files={} + ) decoy.when(await orchestrator.run(deck_configuration=[],)).then_return( protocol_runner.RunResult( commands=[analysis_command], @@ -187,8 +196,6 @@ async def test_analyze( await subject.analyze( analysis_id="analysis-id", - orchestrator=orchestrator, - run_time_parameters=[bool_parameter], ) decoy.verify( await analysis_store.update( @@ -225,7 +232,7 @@ async def test_analyze_updates_pending_on_error( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) raised_exception = Exception("You got me!!") @@ -242,18 +249,23 @@ async def test_analyze_updates_pending_on_error( message="You got me!!", ) - # json_runner = decoy.mock(cls=protocol_runner.JsonRunner) orchestrator = decoy.mock(cls=protocol_runner.RunOrchestrator) + decoy.when( + await simulating_runner.create_simulating_orchestrator( + robot_type=robot_type, + protocol_config=JsonProtocolConfig(schema_version=123), + ) + ).then_return(orchestrator) + subject = ProtocolAnalyzer( analysis_store=analysis_store, protocol_resource=protocol_resource ) - decoy.when( await orchestrator.run( deck_configuration=[], ) ).then_raise(raised_exception) - + decoy.when(orchestrator.get_run_time_parameters()).then_return([]) decoy.when(em.map_unexpected_error(error=raised_exception)).then_return( enumerated_error ) @@ -261,9 +273,10 @@ async def test_analyze_updates_pending_on_error( decoy.when(datetime_helper.utc_now()).then_return( datetime(year=2023, month=3, day=3) ) - + await subject.load_orchestrator( + run_time_param_values={"rtp_var": 123}, run_time_param_files={} + ) await subject.analyze( - orchestrator=orchestrator, analysis_id="analysis-id", ) diff --git a/robot-server/tests/protocols/test_protocol_auto_deleter.py b/robot-server/tests/protocols/test_protocol_auto_deleter.py index 1969f64cf61..840e9e8e31a 100644 --- a/robot-server/tests/protocols/test_protocol_auto_deleter.py +++ b/robot-server/tests/protocols/test_protocol_auto_deleter.py @@ -46,7 +46,7 @@ def test_make_room_for_new_protocol( created_at=datetime(year=2020, month=1, day=1), source=mock_protocol_source, protocol_key=f"{p.protocol_id}{idx}", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) for idx, p in enumerate(usage_info) ] @@ -96,9 +96,9 @@ def test_make_room_for_new_quick_transfer_protocol( created_at=datetime(year=2020, month=1, day=1), source=mock_protocol_source, protocol_key=f"{p.protocol_id}{idx}", - protocol_kind=ProtocolKind.STANDARD.value + protocol_kind=ProtocolKind.STANDARD if idx not in [3, 4] - else ProtocolKind.QUICK_TRANSFER.value, + else ProtocolKind.QUICK_TRANSFER, ) for idx, p in enumerate(usage_info_all) ] diff --git a/robot-server/tests/protocols/test_protocol_store.py b/robot-server/tests/protocols/test_protocol_store.py index e97a921ff01..6550db63371 100644 --- a/robot-server/tests/protocols/test_protocol_store.py +++ b/robot-server/tests/protocols/test_protocol_store.py @@ -13,6 +13,19 @@ PythonProtocolConfig, ) +from robot_server.data_files.data_files_store import DataFilesStore, DataFileInfo +from robot_server.data_files.models import DataFile +from robot_server.protocols.analysis_memcache import MemoryCache +from robot_server.protocols.analysis_models import ( + CompletedAnalysis, + AnalysisStatus, + AnalysisResult, +) +from robot_server.protocols.completed_analysis_store import ( + CompletedAnalysisResource, + CompletedAnalysisStore, +) +from robot_server.protocols.protocol_models import ProtocolKind from robot_server.protocols.protocol_store import ( ProtocolStore, ProtocolResource, @@ -20,6 +33,7 @@ ProtocolNotFoundError, ProtocolUsedByRunError, ) +from robot_server.protocols.rtp_resources import CSVParameterResource from robot_server.runs.run_store import RunStore @@ -53,6 +67,21 @@ def run_store(sql_engine: SQLEngine, mock_runs_publisher: RunsPublisher) -> RunS return RunStore(sql_engine=sql_engine) +@pytest.fixture +def data_files_store(sql_engine: SQLEngine) -> DataFilesStore: + """Get a mocked out DataFilesStore.""" + return DataFilesStore(sql_engine=sql_engine) + + +@pytest.fixture +def completed_analysis_store( + decoy: Decoy, + sql_engine: SQLEngine, +) -> CompletedAnalysisStore: + """Get a subject.""" + return CompletedAnalysisStore(sql_engine, decoy.mock(cls=MemoryCache), "2") + + async def test_insert_and_get_protocol( protocol_file_directory: Path, subject: ProtocolStore ) -> None: @@ -70,7 +99,7 @@ async def test_insert_and_get_protocol( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) assert subject.has("protocol-id") is False @@ -99,7 +128,7 @@ async def test_insert_with_duplicate_key_raises( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) protocol_resource_2 = ProtocolResource( protocol_id="protocol-id", @@ -114,7 +143,7 @@ async def test_insert_with_duplicate_key_raises( content_hash="abc123", ), protocol_key="dummy-data-222", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(protocol_resource_1) @@ -152,7 +181,7 @@ async def test_get_all_protocols( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) resource_2 = ProtocolResource( protocol_id="123", @@ -167,7 +196,7 @@ async def test_get_all_protocols( content_hash="abc123", ), protocol_key="dummy-data-222", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(resource_1) @@ -204,7 +233,7 @@ async def test_remove_protocol( content_hash="abc123", ), protocol_key="dummy-data-111", - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(protocol_resource) @@ -244,7 +273,7 @@ def test_remove_protocol_conflict( content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(protocol_resource) @@ -279,7 +308,7 @@ def test_get_usage_info( content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) protocol_resource_2 = ProtocolResource( protocol_id="protocol-id-2", @@ -294,7 +323,7 @@ def test_get_usage_info( content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(protocol_resource_1) @@ -364,7 +393,7 @@ def test_get_referencing_run_ids( content_hash="abc123", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) subject.insert(protocol_resource_1) @@ -408,7 +437,7 @@ def test_get_protocol_ids( content_hash="abc1", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) protocol_resource_2 = ProtocolResource( @@ -424,7 +453,7 @@ def test_get_protocol_ids( content_hash="abc2", ), protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) assert subject.get_all_ids() == [] @@ -459,7 +488,7 @@ async def test_insert_and_get_quick_transfer_protocol( content_hash="abc123", ), protocol_key="dummy-key-111", - protocol_kind="quick-transfer", + protocol_kind=ProtocolKind.QUICK_TRANSFER, ) assert subject.has("protocol-id") is False @@ -468,5 +497,134 @@ async def test_insert_and_get_quick_transfer_protocol( result = subject.get("protocol-id") assert result == protocol_resource - assert result.protocol_kind == "quick-transfer" + assert result.protocol_kind == ProtocolKind.QUICK_TRANSFER assert subject.has("protocol-id") is True + + +def get_completed_analysis_resource( + analysis_id: str, + protocol_id: str, +) -> CompletedAnalysisResource: + """Get a CompletedAnalysisResource.""" + return CompletedAnalysisResource( + analysis_id, + protocol_id, + "2", + CompletedAnalysis( + id=analysis_id, + status=AnalysisStatus.COMPLETED, + result=AnalysisResult.OK, + pipettes=[], + labware=[], + modules=[], + commands=[], + errors=[], + liquids=[], + ), + ) + + +async def test_get_referenced_data_files( + subject: ProtocolStore, + data_files_store: DataFilesStore, + completed_analysis_store: CompletedAnalysisStore, +) -> None: + """It should fetch a list of data files referenced in protocol's analyses and runs.""" + protocol_resource_1 = ProtocolResource( + protocol_id="protocol-id", + created_at=datetime(year=2024, month=1, day=1, tzinfo=timezone.utc), + source=ProtocolSource( + directory=None, + main_file=Path("/dev/null"), + config=JsonProtocolConfig(schema_version=123), + files=[], + metadata={}, + robot_type="OT-2 Standard", + content_hash="abc1", + ), + protocol_key=None, + protocol_kind=ProtocolKind.STANDARD, + ) + analysis_resource1 = CompletedAnalysisResource( + "analysis-id-1", + "protocol-id", + "2", + CompletedAnalysis( + id="analysis-id-1", + status=AnalysisStatus.COMPLETED, + result=AnalysisResult.OK, + pipettes=[], + labware=[], + modules=[], + commands=[], + errors=[], + liquids=[], + ), + ) + analysis_resource2 = CompletedAnalysisResource( + "analysis-id-2", + "protocol-id", + "2", + CompletedAnalysis( + id="analysis-id-2", + status=AnalysisStatus.COMPLETED, + result=AnalysisResult.OK, + pipettes=[], + labware=[], + modules=[], + commands=[], + errors=[], + liquids=[], + ), + ) + subject.insert(protocol_resource_1) + await data_files_store.insert( + DataFileInfo( + id="data-file-id", + name="file-name", + file_hash="abc123", + created_at=datetime(year=2021, month=1, day=1, tzinfo=timezone.utc), + ) + ) + await data_files_store.insert( + DataFileInfo( + id="data-file-id-2", + name="file-name", + file_hash="abc123", + created_at=datetime(year=2021, month=1, day=1, tzinfo=timezone.utc), + ) + ) + await completed_analysis_store.make_room_and_add( + completed_analysis_resource=analysis_resource1, + primitive_rtp_resources=[], + csv_rtp_resources=[ + CSVParameterResource( + analysis_id="analysis-id-1", + parameter_variable_name="csv-var", + file_id="data-file-id", + ), + CSVParameterResource( + analysis_id="analysis-id-1", + parameter_variable_name="csv-var", + file_id="data-file-id-2", + ), + ], + ) + await completed_analysis_store.make_room_and_add( + completed_analysis_resource=analysis_resource2, + primitive_rtp_resources=[], + csv_rtp_resources=[], + ) + result = await subject.get_referenced_data_files("protocol-id") + assert result == [ + DataFile( + id="data-file-id", + name="file-name", + createdAt=datetime(year=2021, month=1, day=1, tzinfo=timezone.utc), + ), + DataFile( + id="data-file-id-2", + name="file-name", + createdAt=datetime(year=2021, month=1, day=1, tzinfo=timezone.utc), + ), + ] diff --git a/robot-server/tests/protocols/test_protocols_router.py b/robot-server/tests/protocols/test_protocols_router.py index a28d57e3745..e1e9968b232 100644 --- a/robot-server/tests/protocols/test_protocols_router.py +++ b/robot-server/tests/protocols/test_protocols_router.py @@ -8,7 +8,13 @@ from fastapi import HTTPException, UploadFile from pathlib import Path -from opentrons.protocol_engine.types import RunTimeParamValuesType, NumberParameter +from opentrons.protocol_engine.types import ( + PrimitiveRunTimeParamValuesType, + NumberParameter, + CSVParameter, + CSVRunTimeParamFilesType, + FileInfo, +) from opentrons.protocols.api_support.types import APIVersion from opentrons.protocol_reader import ( @@ -24,8 +30,10 @@ BufferedFile, ) +from robot_server.data_files.models import DataFile from robot_server.errors.error_responses import ApiError from robot_server.protocols.analyses_manager import AnalysesManager +from robot_server.protocols.protocol_analyzer import ProtocolAnalyzer from robot_server.service.json_api import SimpleEmptyBody, MultiBodyMeta, RequestModel from robot_server.protocols.analysis_store import ( AnalysisStore, @@ -67,6 +75,7 @@ get_protocol_analyses, get_protocol_analysis_by_id, get_protocol_analysis_as_document, + get_protocol_data_files, ) @@ -148,7 +157,7 @@ async def test_get_protocols( content_hash="a_b_c", ), protocol_key="dummy-key-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) resource_2 = ProtocolResource( protocol_id="123", @@ -163,7 +172,7 @@ async def test_get_protocols( content_hash="1_2_3", ), protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) resource_3 = ProtocolResource( protocol_id="333", @@ -178,7 +187,7 @@ async def test_get_protocols( content_hash="3_3_3", ), protocol_key="dummy-key-333", - protocol_kind=ProtocolKind.QUICK_TRANSFER.value, + protocol_kind=ProtocolKind.QUICK_TRANSFER, ) analysis_1 = AnalysisSummary(id="analysis-id-abc", status=AnalysisStatus.PENDING) @@ -321,7 +330,7 @@ async def test_get_protocol_by_id( content_hash="a_b_c", ), protocol_key="dummy-key-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) analysis_summary = AnalysisSummary( @@ -417,14 +426,14 @@ async def test_create_existing_protocol( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) completed_analysis = AnalysisSummary( id="analysis-id", status=AnalysisStatus.COMPLETED, ) - + analyzer = decoy.mock(cls=ProtocolAnalyzer) decoy.when( await file_reader_writer.read( # TODO(mm, 2024-02-07): Recent FastAPI upgrades mean protocol_file.filename @@ -442,9 +451,18 @@ async def test_create_existing_protocol( decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="the-og-proto-id") ).then_return([completed_analysis]) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id", + protocol_resource=stored_protocol_resource, + run_time_param_values={}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return([]) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - analysis_summary=completed_analysis, new_rtp_values={} + last_analysis_summary=completed_analysis, new_parameters=[] ) ).then_return(True) decoy.when(protocol_store.get_all()).then_return([stored_protocol_resource]) @@ -519,7 +537,7 @@ async def test_create_protocol( created_at=datetime(year=2021, month=1, day=1), source=protocol_source, protocol_key="dummy-key-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) pending_analysis = AnalysisSummary( @@ -545,12 +563,23 @@ async def test_create_protocol( ) ).then_return(protocol_source) decoy.when(protocol_store.get_all()).then_return([]) - decoy.when( - await analyses_manager.start_analysis( + analysis_store.get_summaries_by_protocol(protocol_id="protocol-id") + ).then_return([]) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( analysis_id="analysis-id", protocol_resource=protocol_resource, run_time_param_values={}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return([]) + decoy.when( + await analyses_manager.start_analysis( + analysis_id="analysis-id", + analyzer=analyzer, ) ).then_return(pending_analysis) @@ -630,7 +659,7 @@ async def test_create_new_protocol_with_run_time_params( created_at=datetime(year=2021, month=1, day=1), source=protocol_source, protocol_key="dummy-key-111", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) run_time_parameter = NumberParameter( displayName="My parameter", @@ -665,10 +694,21 @@ async def test_create_new_protocol_with_run_time_params( ) ).then_return(protocol_source) decoy.when( - await analyses_manager.start_analysis( + analysis_store.get_summaries_by_protocol(protocol_id="protocol-id") + ).then_return([]) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( analysis_id="analysis-id", protocol_resource=protocol_resource, run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + run_time_param_files={"my_csv_file": "file-id"}, + ) + ).then_return(analyzer) + decoy.when( + await analyses_manager.start_analysis( + analysis_id="analysis-id", + analyzer=analyzer, ) ).then_return(pending_analysis) decoy.when(protocol_store.get_all()).then_return([]) @@ -677,6 +717,7 @@ async def test_create_new_protocol_with_run_time_params( files=[protocol_file], key="dummy-key-111", run_time_parameter_values='{"vol": 123, "dry_run": true, "mount": "left"}', + run_time_parameter_files='{"my_csv_file": "file-id"}', protocol_directory=protocol_directory, protocol_store=protocol_store, analysis_store=analysis_store, @@ -736,7 +777,7 @@ async def test_create_existing_protocol_with_no_previous_analysis( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) run_time_parameter = NumberParameter( displayName="My parameter", @@ -770,12 +811,20 @@ async def test_create_existing_protocol_with_no_previous_analysis( decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="the-og-proto-id") ).then_return([]) - + analyzer = decoy.mock(cls=ProtocolAnalyzer) decoy.when( - await analyses_manager.start_analysis( + await analyses_manager.initialize_analyzer( analysis_id="analysis-id", protocol_resource=stored_protocol_resource, run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + run_time_param_files={}, + ) + ).then_return(analyzer) + + decoy.when( + await analyses_manager.start_analysis( + analysis_id="analysis-id", + analyzer=analyzer, ) ).then_return(pending_analysis) @@ -850,7 +899,7 @@ async def test_create_existing_protocol_with_different_run_time_params( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) completed_summary = AnalysisSummary( @@ -889,16 +938,27 @@ async def test_create_existing_protocol_with_different_run_time_params( decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="the-og-proto-id") ).then_return([completed_summary]) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id", + protocol_resource=stored_protocol_resource, + run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + run_time_param_files={"my_csv_file": "csv-file-id"}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [run_time_parameter] + ) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - completed_summary, {"vol": 123, "dry_run": True, "mount": "left"} + completed_summary, [run_time_parameter] ) ).then_return(False) decoy.when( await analyses_manager.start_analysis( analysis_id="analysis-id", - protocol_resource=stored_protocol_resource, - run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + analyzer=analyzer, ) ).then_return(pending_summary) @@ -906,6 +966,7 @@ async def test_create_existing_protocol_with_different_run_time_params( files=[protocol_file], key="dummy-key-111", run_time_parameter_values='{"vol": 123, "dry_run": true, "mount": "left"}', + run_time_parameter_files='{"my_csv_file": "csv-file-id"}', protocol_directory=protocol_directory, protocol_store=protocol_store, analysis_store=analysis_store, @@ -967,15 +1028,22 @@ async def test_create_existing_protocol_with_same_run_time_params( config=JsonProtocolConfig(schema_version=123), content_hash="a_b_c", ) - stored_protocol_resource = ProtocolResource( protocol_id="protocol-id", created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, + ) + run_time_parameter = NumberParameter( + displayName="My parameter", + variableName="cool_param", + type="int", + min=1, + max=5, + value=2.0, + default=3.0, ) - analysis_summaries = [ AnalysisSummary( id="analysis-id", @@ -1001,9 +1069,21 @@ async def test_create_existing_protocol_with_same_run_time_params( decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="the-og-proto-id") ).then_return(analysis_summaries) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id", + protocol_resource=stored_protocol_resource, + run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [run_time_parameter] + ) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - analysis_summaries[-1], {"vol": 123, "dry_run": True, "mount": "left"} + analysis_summaries[-1], [run_time_parameter] ) ).then_return(True) @@ -1078,9 +1158,17 @@ async def test_create_existing_protocol_with_pending_analysis_raises( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, + ) + run_time_parameter = NumberParameter( + displayName="My parameter", + variableName="cool_param", + type="int", + min=1, + max=5, + value=2.0, + default=3.0, ) - analysis_summaries = [ AnalysisSummary( id="analysis-id", @@ -1106,9 +1194,21 @@ async def test_create_existing_protocol_with_pending_analysis_raises( decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="the-og-proto-id") ).then_return(analysis_summaries) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id", + protocol_resource=stored_protocol_resource, + run_time_param_values={"vol": 123, "dry_run": True, "mount": "left"}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [run_time_parameter] + ) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - analysis_summaries[-1], {"vol": 123, "dry_run": True, "mount": "left"} + analysis_summaries[-1], [run_time_parameter] ) ).then_raise(AnalysisIsPendingError("a-id")) @@ -1463,20 +1563,70 @@ async def test_create_protocol_analyses_with_same_rtp_values( analyses_manager: AnalysesManager, ) -> None: """It should not start a new analysis for the new rtp values.""" - rtp_values: RunTimeParamValuesType = {"vol": 123, "dry_run": True, "mount": "left"} + rtp_values: PrimitiveRunTimeParamValuesType = { + "vol": 123, + "dry_run": True, + "mount": "left", + } analysis_summaries = [ AnalysisSummary( id="analysis-id", status=AnalysisStatus.COMPLETED, ), ] + run_time_parameter = NumberParameter( + displayName="My parameter", + variableName="cool_param", + type="int", + min=1, + max=5, + value=2.0, + default=3.0, + ) + protocol_source = ProtocolSource( + directory=Path("/dev/null"), + main_file=Path("/dev/null/foo.json"), + files=[ + ProtocolSourceFile( + path=Path("/dev/null/foo.json"), + role=ProtocolFileRole.MAIN, + ) + ], + metadata={"this_is_fake_metadata": True}, + robot_type="OT-2 Standard", + config=JsonProtocolConfig(schema_version=123), + content_hash="a_b_c", + ) + + stored_protocol_resource = ProtocolResource( + protocol_id="protocol-id", + created_at=datetime(year=2020, month=1, day=1), + source=protocol_source, + protocol_key="dummy-key-222", + protocol_kind=ProtocolKind.STANDARD, + ) decoy.when(protocol_store.has(protocol_id="protocol-id")).then_return(True) + decoy.when(protocol_store.get(protocol_id="protocol-id")).then_return( + stored_protocol_resource + ) decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="protocol-id") ).then_return(analysis_summaries) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id-2", + protocol_resource=stored_protocol_resource, + run_time_param_values=rtp_values, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [run_time_parameter] + ) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - analysis_summaries[-1], rtp_values + analysis_summaries[-1], [run_time_parameter] ) ).then_return(True) @@ -1501,7 +1651,14 @@ async def test_update_protocol_analyses_with_new_rtp_values( analyses_manager: AnalysesManager, ) -> None: """It should start a new analysis for the new rtp values.""" - rtp_values: RunTimeParamValuesType = {"vol": 123, "dry_run": True, "mount": "left"} + rtp_values: PrimitiveRunTimeParamValuesType = { + "vol": 123, + "dry_run": True, + "mount": "left", + } + rtp_files: CSVRunTimeParamFilesType = { + "csv_param": "file-id", + } protocol_source = ProtocolSource( directory=Path("/dev/null"), main_file=Path("/dev/null/foo.json"), @@ -1522,7 +1679,7 @@ async def test_update_protocol_analyses_with_new_rtp_values( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) analysis_summaries = [ AnalysisSummary( @@ -1539,36 +1696,54 @@ async def test_update_protocol_analyses_with_new_rtp_values( value=2.0, default=3.0, ) + csv_parameter = CSVParameter( + displayName="CSV parameter", + variableName="csv_param", + file=FileInfo(id="file-id", name=""), + ) decoy.when(protocol_store.has(protocol_id="protocol-id")).then_return(True) + decoy.when(protocol_store.get(protocol_id="protocol-id")).then_return( + stored_protocol_resource + ) decoy.when( analysis_store.get_summaries_by_protocol(protocol_id="protocol-id") ).then_return(analysis_summaries) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( + analysis_id="analysis-id-2", + protocol_resource=stored_protocol_resource, + run_time_param_values=rtp_values, + run_time_param_files=rtp_files, + ) + ).then_return(analyzer) + decoy.when(analyzer.get_verified_run_time_parameters()).then_return( + [run_time_parameter, csv_parameter] + ) decoy.when( await analysis_store.matching_rtp_values_in_analysis( - analysis_summaries[-1], rtp_values + analysis_summaries[-1], [run_time_parameter, csv_parameter] ) ).then_return(False) - decoy.when(protocol_store.get(protocol_id="protocol-id")).then_return( - stored_protocol_resource - ) decoy.when( await analyses_manager.start_analysis( analysis_id="analysis-id-2", - protocol_resource=stored_protocol_resource, - run_time_param_values=rtp_values, + analyzer=analyzer, ) ).then_return( AnalysisSummary( id="analysis-id-2", status=AnalysisStatus.PENDING, - runTimeParameters=[run_time_parameter], + runTimeParameters=[run_time_parameter, csv_parameter], ) ) result = await create_protocol_analysis( protocolId="protocol-id", request_body=RequestModel( - data=AnalysisRequest(runTimeParameterValues=rtp_values) + data=AnalysisRequest( + runTimeParameterValues=rtp_values, runTimeParameterFiles=rtp_files + ) ), protocol_store=protocol_store, analysis_store=analysis_store, @@ -1580,7 +1755,7 @@ async def test_update_protocol_analyses_with_new_rtp_values( AnalysisSummary( id="analysis-id-2", status=AnalysisStatus.PENDING, - runTimeParameters=[run_time_parameter], + runTimeParameters=[run_time_parameter, csv_parameter], ), ] assert result.status_code == 201 @@ -1619,7 +1794,7 @@ async def test_update_protocol_analyses_with_forced_reanalysis( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.STANDARD.value, + protocol_kind=ProtocolKind.STANDARD, ) decoy.when(protocol_store.has(protocol_id="protocol-id")).then_return(True) decoy.when( @@ -1628,11 +1803,19 @@ async def test_update_protocol_analyses_with_forced_reanalysis( decoy.when(protocol_store.get(protocol_id="protocol-id")).then_return( stored_protocol_resource ) + analyzer = decoy.mock(cls=ProtocolAnalyzer) decoy.when( - await analyses_manager.start_analysis( + await analyses_manager.initialize_analyzer( analysis_id="analysis-id-2", protocol_resource=stored_protocol_resource, run_time_param_values={}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when( + await analyses_manager.start_analysis( + analysis_id="analysis-id-2", + analyzer=analyzer, ) ).then_return(AnalysisSummary(id="analysis-id-2", status=AnalysisStatus.PENDING)) @@ -1689,7 +1872,7 @@ async def test_create_protocol_kind_quick_transfer( created_at=datetime(year=2021, month=1, day=1), source=protocol_source, protocol_key="dummy-key-111", - protocol_kind=ProtocolKind.QUICK_TRANSFER.value, + protocol_kind=ProtocolKind.QUICK_TRANSFER, ) run_time_parameter = NumberParameter( displayName="My parameter", @@ -1724,10 +1907,21 @@ async def test_create_protocol_kind_quick_transfer( ) ).then_return(protocol_source) decoy.when( - await analyses_manager.start_analysis( + analysis_store.get_summaries_by_protocol(protocol_id="protocol-id") + ).then_return([]) + analyzer = decoy.mock(cls=ProtocolAnalyzer) + decoy.when( + await analyses_manager.initialize_analyzer( analysis_id="analysis-id", protocol_resource=protocol_resource, run_time_param_values={}, + run_time_param_files={}, + ) + ).then_return(analyzer) + decoy.when( + await analyses_manager.start_analysis( + analysis_id="analysis-id", + analyzer=analyzer, ) ).then_return(pending_analysis) decoy.when(protocol_store.get_all()).then_return([]) @@ -1806,7 +2000,7 @@ async def test_create_protocol_maximum_quick_transfer_protocols_exceeded( created_at=datetime(year=2020, month=1, day=1), source=protocol_source, protocol_key="dummy-key-222", - protocol_kind=ProtocolKind.QUICK_TRANSFER.value, + protocol_kind=ProtocolKind.QUICK_TRANSFER, ) decoy.when(protocol_store.get_all()).then_return([stored_protocol_resource]) @@ -1831,3 +2025,48 @@ async def test_create_protocol_maximum_quick_transfer_protocols_exceeded( ) assert exc_info.value.status_code == 409 + + +async def test_get_data_files( + decoy: Decoy, + protocol_store: ProtocolStore, +) -> None: + """It should get all the data files associated with the protocol.""" + data_files = [ + DataFile( + id="id1", + name="csv-file1.csv", + createdAt=datetime(year=2024, month=1, day=1), + ), + DataFile( + id="id2", + name="csv-file2.csv", + createdAt=datetime(year=2024, month=1, day=1), + ), + ] + decoy.when(protocol_store.has(protocol_id="protocol-id")).then_return(True) + decoy.when( + await protocol_store.get_referenced_data_files("protocol-id") + ).then_return(data_files) + result = await get_protocol_data_files( + protocolId="protocol-id", + protocol_store=protocol_store, + ) + assert result.status_code == 200 + assert result.content.data == data_files + + +async def test_get_non_existent_protocol_data_files( + decoy: Decoy, + protocol_store: ProtocolStore, +) -> None: + """It should 404 if a protocol does not exist.""" + decoy.when(protocol_store.has("protocol-id")).then_return(False) + + with pytest.raises(ApiError) as exc_info: + await get_protocol_data_files( + protocolId="protocol-id", + protocol_store=protocol_store, + ) + assert exc_info.value.status_code == 404 + assert exc_info.value.content["errors"][0]["id"] == "ProtocolNotFound" diff --git a/robot-server/tests/runs/router/test_base_router.py b/robot-server/tests/runs/router/test_base_router.py index 716a54c9bd2..8ee37d64b9d 100644 --- a/robot-server/tests/runs/router/test_base_router.py +++ b/robot-server/tests/runs/router/test_base_router.py @@ -9,6 +9,7 @@ from opentrons.protocol_reader import ProtocolSource, JsonProtocolConfig from robot_server.errors.error_responses import ApiError +from robot_server.runs.error_recovery_models import ErrorRecoveryPolicy from robot_server.service.json_api import ( RequestModel, SimpleBody, @@ -17,6 +18,7 @@ ResourceLink, ) +from robot_server.protocols.protocol_models import ProtocolKind from robot_server.protocols.protocol_store import ( ProtocolNotFoundError, ProtocolResource, @@ -37,6 +39,7 @@ get_runs, remove_run, update_run, + put_error_recovery_policy, ) from robot_server.deck_configuration.store import DeckConfigurationStore @@ -130,7 +133,7 @@ async def test_create_protocol_run( protocol_resource = ProtocolResource( protocol_id=protocol_id, protocol_key=None, - protocol_kind=None, + protocol_kind=ProtocolKind.STANDARD, created_at=datetime(year=2022, month=2, day=2), source=ProtocolSource( directory=Path("/dev/null"), @@ -569,3 +572,41 @@ async def test_update_to_current_missing( assert exc_info.value.status_code == 404 assert exc_info.value.content["errors"][0]["id"] == "RunNotFound" + + +async def test_create_policies( + decoy: Decoy, mock_run_data_manager: RunDataManager +) -> None: + """It should call RunDataManager create run policies.""" + policies = decoy.mock(cls=ErrorRecoveryPolicy) + await put_error_recovery_policy( + runId="rud-id", + request_body=RequestModel(data=policies), + run_data_manager=mock_run_data_manager, + ) + decoy.verify( + mock_run_data_manager.set_policies( + run_id="rud-id", policies=policies.policyRules + ) + ) + + +async def test_create_policies_raises_not_active_run( + decoy: Decoy, mock_run_data_manager: RunDataManager +) -> None: + """It should raise that the run is not current.""" + policies = decoy.mock(cls=ErrorRecoveryPolicy) + decoy.when( + mock_run_data_manager.set_policies( + run_id="rud-id", policies=policies.policyRules + ) + ).then_raise(RunNotCurrentError()) + with pytest.raises(ApiError) as exc_info: + await put_error_recovery_policy( + runId="rud-id", + request_body=RequestModel(data=policies), + run_data_manager=mock_run_data_manager, + ) + + assert exc_info.value.status_code == 409 + assert exc_info.value.content["errors"][0]["id"] == "RunStopped" diff --git a/robot-server/tests/runs/router/test_labware_router.py b/robot-server/tests/runs/router/test_labware_router.py index 824971abade..09811f20a38 100644 --- a/robot-server/tests/runs/router/test_labware_router.py +++ b/robot-server/tests/runs/router/test_labware_router.py @@ -3,7 +3,7 @@ from datetime import datetime from decoy import Decoy -from opentrons_shared_data.labware.dev_types import LabwareDefinition as LabwareDefDict +from opentrons_shared_data.labware.types import LabwareDefinition as LabwareDefDict from opentrons.types import DeckSlotName from opentrons.protocol_engine import EngineStatus, types as pe_types diff --git a/robot-server/tests/runs/test_engine_store.py b/robot-server/tests/runs/test_engine_store.py index e2d316592dd..46f25f3edb4 100644 --- a/robot-server/tests/runs/test_engine_store.py +++ b/robot-server/tests/runs/test_engine_store.py @@ -4,7 +4,7 @@ from decoy import Decoy, matchers from opentrons_shared_data import get_shared_data_root -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType from opentrons.protocol_engine.errors.exceptions import EStopActivatedError from opentrons.types import DeckSlotName diff --git a/robot-server/tests/runs/test_error_recovery_mapping.py b/robot-server/tests/runs/test_error_recovery_mapping.py new file mode 100644 index 00000000000..21195872d39 --- /dev/null +++ b/robot-server/tests/runs/test_error_recovery_mapping.py @@ -0,0 +1,116 @@ +"""Unit tests for `error_recovery_mapping`.""" +import pytest +from decoy import Decoy + + +from opentrons.protocol_engine.commands.pipetting_common import ( + LiquidNotFoundError, + LiquidNotFoundErrorInternalData, +) +from opentrons.protocol_engine.commands.command import ( + DefinedErrorData, +) +from opentrons.protocol_engine.commands.command_unions import CommandDefinedErrorData +from opentrons.protocol_engine.commands.liquid_probe import LiquidProbe +from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryType +from opentrons.protocol_engine.state.config import Config +from opentrons.protocol_engine.types import DeckType +from robot_server.runs.error_recovery_mapping import ( + create_error_recovery_policy_from_rules, +) +from robot_server.runs.error_recovery_models import ( + ErrorRecoveryRule, + MatchCriteria, + CommandMatcher, + ErrorMatcher, + ReactionIfMatch, +) + + +@pytest.fixture +def mock_command(decoy: Decoy) -> LiquidProbe: + """Get a mock PickUpTip command.""" + mock = decoy.mock(cls=LiquidProbe) + decoy.when(mock.commandType).then_return("liquidProbe") + return mock + + +@pytest.fixture +def mock_error_data(decoy: Decoy) -> CommandDefinedErrorData: + """Get a mock TipPhysicallyMissingError.""" + mock = decoy.mock( + cls=DefinedErrorData[LiquidNotFoundError, LiquidNotFoundErrorInternalData] + ) + mock_lnfe = decoy.mock(cls=LiquidNotFoundError) + decoy.when(mock.public).then_return(mock_lnfe) + decoy.when(mock_lnfe.errorType).then_return("liquidNotFound") + return mock + + +@pytest.fixture +def mock_criteria(decoy: Decoy) -> MatchCriteria: + """Get a mock Match Criteria.""" + mock = decoy.mock(cls=MatchCriteria) + mock_command = decoy.mock(cls=CommandMatcher) + decoy.when(mock_command.commandType).then_return("liquidProbe") + mock_error_matcher = decoy.mock(cls=ErrorMatcher) + decoy.when(mock_error_matcher.errorType).then_return("liquidNotFound") + decoy.when(mock.command).then_return(mock_command) + decoy.when(mock_command.error).then_return(mock_error_matcher) + return mock + + +@pytest.fixture +def mock_rule(decoy: Decoy, mock_criteria: MatchCriteria) -> ErrorRecoveryRule: + """Get a mock ErrorRecoveryRule.""" + mock = decoy.mock(cls=ErrorRecoveryRule) + decoy.when(mock.ifMatch).then_return(ReactionIfMatch.IGNORE_AND_CONTINUE) + decoy.when(mock.matchCriteria).then_return(mock_criteria) + return mock + + +def test_create_error_recovery_policy_with_rules( + decoy: Decoy, + mock_command: LiquidProbe, + mock_error_data: CommandDefinedErrorData, + mock_rule: ErrorRecoveryRule, +) -> None: + """Should return IGNORE_AND_CONTINUE if that's what we specify as the rule.""" + policy = create_error_recovery_policy_from_rules([mock_rule]) + exampleConfig = Config( + robot_type="OT-3 Standard", + deck_type=DeckType.OT3_STANDARD, + ) + assert ( + policy(exampleConfig, mock_command, mock_error_data) + == ErrorRecoveryType.IGNORE_AND_CONTINUE + ) + + +def test_create_error_recovery_policy_undefined_error( + decoy: Decoy, mock_command: LiquidProbe +) -> None: + """Should return a FAIL_RUN policy when error is not defined.""" + policy = create_error_recovery_policy_from_rules(rules=[]) + exampleConfig = Config( + robot_type="OT-3 Standard", + deck_type=DeckType.OT3_STANDARD, + ) + + assert policy(exampleConfig, mock_command, None) == ErrorRecoveryType.FAIL_RUN + + +def test_create_error_recovery_policy_defined_error( + decoy: Decoy, mock_command: LiquidProbe, mock_error_data: CommandDefinedErrorData +) -> None: + """Should return a WAIT_FOR_RECOVERY policy when error is defined.""" + policy = create_error_recovery_policy_from_rules(rules=[]) + exampleConfig = Config( + robot_type="OT-3 Standard", + deck_type=DeckType.OT3_STANDARD, + ) + + assert ( + policy(exampleConfig, mock_command, mock_error_data) + == ErrorRecoveryType.WAIT_FOR_RECOVERY + ) diff --git a/robot-server/tests/runs/test_run_data_manager.py b/robot-server/tests/runs/test_run_data_manager.py index f7165f3ec52..63716a8ebd5 100644 --- a/robot-server/tests/runs/test_run_data_manager.py +++ b/robot-server/tests/runs/test_run_data_manager.py @@ -1,12 +1,9 @@ """Tests for RunDataManager.""" +from datetime import datetime from typing import Optional, List import pytest -from datetime import datetime from decoy import Decoy, matchers - -from opentrons.types import DeckSlotName -from opentrons.protocol_runner import RunResult from opentrons.protocol_engine import ( EngineStatus, StateSummary, @@ -20,31 +17,36 @@ LoadedModule, LabwareOffset, ) +from opentrons.protocol_engine import Liquid +from opentrons.protocol_engine.error_recovery_policy import ErrorRecoveryPolicy +from opentrons.protocol_runner import RunResult +from opentrons.types import DeckSlotName + +from opentrons_shared_data.errors.exceptions import InvalidStoredData +from opentrons_shared_data.labware.labware_definition import LabwareDefinition +from robot_server.protocols.protocol_models import ProtocolKind from robot_server.protocols.protocol_store import ProtocolResource -from robot_server.runs.run_orchestrator_store import ( - RunOrchestratorStore, - RunConflictError, -) +from robot_server.runs import error_recovery_mapping +from robot_server.runs.error_recovery_models import ErrorRecoveryRule from robot_server.runs.run_data_manager import ( RunDataManager, RunNotCurrentError, PreSerializedCommandsNotAvailableError, ) from robot_server.runs.run_models import Run, BadRun, RunNotFoundError, RunDataError +from robot_server.runs.run_orchestrator_store import ( + RunOrchestratorStore, + RunConflictError, +) from robot_server.runs.run_store import ( RunStore, RunResource, CommandNotFoundError, BadStateSummary, ) -from robot_server.service.task_runner import TaskRunner from robot_server.service.notifications import RunsPublisher - -from opentrons.protocol_engine import Liquid - -from opentrons_shared_data.labware.labware_definition import LabwareDefinition -from opentrons_shared_data.errors.exceptions import InvalidStoredData +from robot_server.service.task_runner import TaskRunner def mock_notify_publishers() -> None: @@ -218,7 +220,7 @@ async def test_create_with_options( created_at=datetime(year=2022, month=2, day=2), source=None, # type: ignore[arg-type] protocol_key=None, - protocol_kind="standard", + protocol_kind=ProtocolKind.STANDARD, ) labware_offset = pe_types.LabwareOffsetCreate( @@ -1001,3 +1003,44 @@ async def test_get_current_run_labware_definition( LabwareDefinition.construct(namespace="test_1"), # type: ignore[call-arg] LabwareDefinition.construct(namespace="test_2"), # type: ignore[call-arg] ] + + +async def test_create_policies_raises_run_not_current( + decoy: Decoy, + mock_run_orchestrator_store: RunOrchestratorStore, + subject: RunDataManager, +) -> None: + """Should raise run not current.""" + decoy.when(mock_run_orchestrator_store.current_run_id).then_return( + "not-current-run-id" + ) + with pytest.raises(RunNotCurrentError): + subject.set_policies( + run_id="run-id", policies=decoy.mock(cls=List[ErrorRecoveryRule]) + ) + + +async def test_create_policies_translates_and_calls_orchestrator( + decoy: Decoy, + monkeypatch: pytest.MonkeyPatch, + mock_run_orchestrator_store: RunOrchestratorStore, + subject: RunDataManager, +) -> None: + """Should translate rules into policy and call orchestrator.""" + monkeypatch.setattr( + error_recovery_mapping, + "create_error_recovery_policy_from_rules", + decoy.mock( + func=decoy.mock( + func=error_recovery_mapping.create_error_recovery_policy_from_rules + ) + ), + ) + input_rules = decoy.mock(cls=List[ErrorRecoveryRule]) + expected_output = decoy.mock(cls=ErrorRecoveryPolicy) + decoy.when( + error_recovery_mapping.create_error_recovery_policy_from_rules(input_rules) + ).then_return(expected_output) + decoy.when(mock_run_orchestrator_store.current_run_id).then_return("run-id") + subject.set_policies(run_id="run-id", policies=input_rules) + decoy.verify(mock_run_orchestrator_store.set_error_recovery_policy(expected_output)) diff --git a/robot-server/tests/runs/test_run_store.py b/robot-server/tests/runs/test_run_store.py index ee7697107f6..94899c5c20e 100644 --- a/robot-server/tests/runs/test_run_store.py +++ b/robot-server/tests/runs/test_run_store.py @@ -7,7 +7,7 @@ from sqlalchemy.engine import Engine from unittest import mock -from opentrons_shared_data.pipette.dev_types import PipetteNameType +from opentrons_shared_data.pipette.types import PipetteNameType from opentrons_shared_data.errors.codes import ErrorCodes from robot_server.protocols.protocol_store import ProtocolNotFoundError @@ -152,6 +152,11 @@ def run_time_parameters() -> List[pe_types.RunTimeParameter]: default="cooler choice", value="coolest choice", ), + pe_types.CSVParameter( + displayName="Display Name 4", + variableName="variable_name_4", + description="a csv parameter without file id", + ), ] diff --git a/robot-server/tests/service/legacy/routers/test_settings.py b/robot-server/tests/service/legacy/routers/test_settings.py index 630adc3a546..6c9ae8adb56 100644 --- a/robot-server/tests/service/legacy/routers/test_settings.py +++ b/robot-server/tests/service/legacy/routers/test_settings.py @@ -15,7 +15,7 @@ pipette_definition as pip_def, ) from opentrons.types import Mount -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum from robot_server.app import app diff --git a/robot-server/tests/service/notifications/publishers/test_deck_configuration_publisher.py b/robot-server/tests/service/notifications/publishers/test_deck_configuration_publisher.py index 3f2b8481967..b31f8f5bcaf 100644 --- a/robot-server/tests/service/notifications/publishers/test_deck_configuration_publisher.py +++ b/robot-server/tests/service/notifications/publishers/test_deck_configuration_publisher.py @@ -2,7 +2,7 @@ import pytest from unittest.mock import AsyncMock -from robot_server.service.notifications import DeckConfigurationPublisher, Topics +from robot_server.service.notifications import DeckConfigurationPublisher, topics @pytest.fixture @@ -27,5 +27,5 @@ async def test_publish_current_maintenance_run( """It should publish a notify flag for deck configuration updates.""" await deck_configuration_publisher.publish_deck_configuration() notification_client.publish_advise_refetch_async.assert_awaited_once_with( - topic=Topics.DECK_CONFIGURATION + topic=topics.DECK_CONFIGURATION ) diff --git a/robot-server/tests/service/notifications/publishers/test_maintenance_runs_publisher.py b/robot-server/tests/service/notifications/publishers/test_maintenance_runs_publisher.py index 8a0cb6a1832..fcc4cac5aac 100644 --- a/robot-server/tests/service/notifications/publishers/test_maintenance_runs_publisher.py +++ b/robot-server/tests/service/notifications/publishers/test_maintenance_runs_publisher.py @@ -2,7 +2,7 @@ import pytest from unittest.mock import AsyncMock -from robot_server.service.notifications import MaintenanceRunsPublisher, Topics +from robot_server.service.notifications import MaintenanceRunsPublisher, topics @pytest.fixture @@ -26,5 +26,5 @@ async def test_publish_current_maintenance_run( """It should publish a notify flag for maintenance runs.""" await maintenance_runs_publisher.publish_current_maintenance_run() notification_client.publish_advise_refetch_async.assert_awaited_once_with( - topic=Topics.MAINTENANCE_RUNS_CURRENT_RUN + topic=topics.MAINTENANCE_RUNS_CURRENT_RUN ) diff --git a/robot-server/tests/service/notifications/publishers/test_runs_publisher.py b/robot-server/tests/service/notifications/publishers/test_runs_publisher.py index fe71f322f59..f3148927dca 100644 --- a/robot-server/tests/service/notifications/publishers/test_runs_publisher.py +++ b/robot-server/tests/service/notifications/publishers/test_runs_publisher.py @@ -5,7 +5,7 @@ from opentrons.protocol_engine import CommandPointer, EngineStatus -from robot_server.service.notifications import RunsPublisher, Topics +from robot_server.service.notifications import RunsPublisher, topics from robot_server.service.notifications.notification_client import NotificationClient from robot_server.service.notifications.publisher_notifier import PublisherNotifier @@ -70,9 +70,9 @@ async def test_initialize( assert runs_publisher._engine_state_slice.recovery_target_command is None assert runs_publisher._engine_state_slice.state_summary_status is None - notification_client.publish_advise_refetch_async.assert_any_await(topic=Topics.RUNS) + notification_client.publish_advise_refetch_async.assert_any_await(topic=topics.RUNS) notification_client.publish_advise_refetch_async.assert_any_await( - topic=f"{Topics.RUNS}/1234" + topic=f"{topics.RUNS}/1234" ) @@ -86,18 +86,18 @@ async def test_clean_up_current_run( await runs_publisher.clean_up_run(run_id="1234") - notification_client.publish_advise_refetch_async.assert_any_await(topic=Topics.RUNS) + notification_client.publish_advise_refetch_async.assert_any_await(topic=topics.RUNS) notification_client.publish_advise_refetch_async.assert_any_await( - topic=f"{Topics.RUNS}/1234" + topic=f"{topics.RUNS}/1234" ) notification_client.publish_advise_unsubscribe_async.assert_any_await( - topic=f"{Topics.RUNS}/1234" + topic=f"{topics.RUNS}/1234" ) notification_client.publish_advise_unsubscribe_async.assert_any_await( - topic=Topics.RUNS_COMMANDS_LINKS + topic=topics.RUNS_COMMANDS_LINKS ) notification_client.publish_advise_unsubscribe_async.assert_any_await( - topic=f"{Topics.RUNS_PRE_SERIALIZED_COMMANDS}/1234" + topic=f"{topics.RUNS_PRE_SERIALIZED_COMMANDS}/1234" ) @@ -132,7 +132,7 @@ async def test_handle_current_command_change( await runs_publisher._handle_current_command_change() notification_client.publish_advise_refetch_async.assert_any_await( - topic=Topics.RUNS_COMMANDS_LINKS + topic=topics.RUNS_COMMANDS_LINKS ) @@ -167,7 +167,7 @@ async def test_handle_recovery_target_command_change( await runs_publisher._handle_recovery_target_command_change() notification_client.publish_advise_refetch_async.assert_any_await( - topic=Topics.RUNS_COMMANDS_LINKS + topic=topics.RUNS_COMMANDS_LINKS ) @@ -203,9 +203,9 @@ async def test_handle_engine_status_change( await runs_publisher._handle_engine_status_change() - notification_client.publish_advise_refetch_async.assert_any_await(topic=Topics.RUNS) + notification_client.publish_advise_refetch_async.assert_any_await(topic=topics.RUNS) notification_client.publish_advise_refetch_async.assert_any_await( - topic=f"{Topics.RUNS}/1234" + topic=f"{topics.RUNS}/1234" ) @@ -231,5 +231,5 @@ async def test_publish_pre_serialized_commannds_notif( assert notification_client.publish_advise_refetch_async.call_count == 3 notification_client.publish_advise_refetch_async.assert_any_await( - topic=f"{Topics.RUNS_PRE_SERIALIZED_COMMANDS}/1234" + topic=f"{topics.RUNS_PRE_SERIALIZED_COMMANDS}/1234" ) diff --git a/shared-data/command/index.ts b/shared-data/command/index.ts index 4f7c3c0ba85..b0e207055c2 100644 --- a/shared-data/command/index.ts +++ b/shared-data/command/index.ts @@ -1,5 +1,6 @@ import commandSchemaV7 from './schemas/7.json' import commandSchemaV8 from './schemas/8.json' +import commandSchemaV9 from './schemas/9.json' export * from './types/index' -export { commandSchemaV7, commandSchemaV8 } +export { commandSchemaV7, commandSchemaV8, commandSchemaV9 } diff --git a/shared-data/command/schemas/9.json b/shared-data/command/schemas/9.json new file mode 100644 index 00000000000..1cb30c99d69 --- /dev/null +++ b/shared-data/command/schemas/9.json @@ -0,0 +1,4381 @@ +{ + "title": "CreateCommandUnion", + "description": "Model that validates a union of all CommandCreate models.", + "discriminator": { + "propertyName": "commandType", + "mapping": { + "aspirate": "#/definitions/AspirateCreate", + "aspirateInPlace": "#/definitions/AspirateInPlaceCreate", + "comment": "#/definitions/CommentCreate", + "configureForVolume": "#/definitions/ConfigureForVolumeCreate", + "configureNozzleLayout": "#/definitions/ConfigureNozzleLayoutCreate", + "custom": "#/definitions/CustomCreate", + "dispense": "#/definitions/DispenseCreate", + "dispenseInPlace": "#/definitions/DispenseInPlaceCreate", + "blowout": "#/definitions/BlowOutCreate", + "blowOutInPlace": "#/definitions/BlowOutInPlaceCreate", + "dropTip": "#/definitions/DropTipCreate", + "dropTipInPlace": "#/definitions/DropTipInPlaceCreate", + "home": "#/definitions/HomeCreate", + "retractAxis": "#/definitions/RetractAxisCreate", + "loadLabware": "#/definitions/LoadLabwareCreate", + "reloadLabware": "#/definitions/ReloadLabwareCreate", + "loadLiquid": "#/definitions/LoadLiquidCreate", + "loadModule": "#/definitions/LoadModuleCreate", + "loadPipette": "#/definitions/LoadPipetteCreate", + "moveLabware": "#/definitions/MoveLabwareCreate", + "moveRelative": "#/definitions/MoveRelativeCreate", + "moveToCoordinates": "#/definitions/MoveToCoordinatesCreate", + "moveToWell": "#/definitions/MoveToWellCreate", + "moveToAddressableArea": "#/definitions/MoveToAddressableAreaCreate", + "moveToAddressableAreaForDropTip": "#/definitions/MoveToAddressableAreaForDropTipCreate", + "prepareToAspirate": "#/definitions/PrepareToAspirateCreate", + "waitForResume": "#/definitions/WaitForResumeCreate", + "pause": "#/definitions/WaitForResumeCreate", + "waitForDuration": "#/definitions/WaitForDurationCreate", + "pickUpTip": "#/definitions/PickUpTipCreate", + "savePosition": "#/definitions/SavePositionCreate", + "setRailLights": "#/definitions/SetRailLightsCreate", + "touchTip": "#/definitions/TouchTipCreate", + "setStatusBar": "#/definitions/SetStatusBarCreate", + "verifyTipPresence": "#/definitions/VerifyTipPresenceCreate", + "getTipPresence": "#/definitions/GetTipPresenceCreate", + "liquidProbe": "#/definitions/LiquidProbeCreate", + "tryLiquidProbe": "#/definitions/TryLiquidProbeCreate", + "heaterShaker/waitForTemperature": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__wait_for_temperature__WaitForTemperatureCreate", + "heaterShaker/setTargetTemperature": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__set_target_temperature__SetTargetTemperatureCreate", + "heaterShaker/deactivateHeater": "#/definitions/DeactivateHeaterCreate", + "heaterShaker/setAndWaitForShakeSpeed": "#/definitions/SetAndWaitForShakeSpeedCreate", + "heaterShaker/deactivateShaker": "#/definitions/DeactivateShakerCreate", + "heaterShaker/openLabwareLatch": "#/definitions/OpenLabwareLatchCreate", + "heaterShaker/closeLabwareLatch": "#/definitions/CloseLabwareLatchCreate", + "magneticModule/disengage": "#/definitions/DisengageCreate", + "magneticModule/engage": "#/definitions/EngageCreate", + "temperatureModule/setTargetTemperature": "#/definitions/opentrons__protocol_engine__commands__temperature_module__set_target_temperature__SetTargetTemperatureCreate", + "temperatureModule/waitForTemperature": "#/definitions/opentrons__protocol_engine__commands__temperature_module__wait_for_temperature__WaitForTemperatureCreate", + "temperatureModule/deactivate": "#/definitions/DeactivateTemperatureCreate", + "thermocycler/setTargetBlockTemperature": "#/definitions/SetTargetBlockTemperatureCreate", + "thermocycler/waitForBlockTemperature": "#/definitions/WaitForBlockTemperatureCreate", + "thermocycler/setTargetLidTemperature": "#/definitions/SetTargetLidTemperatureCreate", + "thermocycler/waitForLidTemperature": "#/definitions/WaitForLidTemperatureCreate", + "thermocycler/deactivateBlock": "#/definitions/DeactivateBlockCreate", + "thermocycler/deactivateLid": "#/definitions/DeactivateLidCreate", + "thermocycler/openLid": "#/definitions/OpenLidCreate", + "thermocycler/closeLid": "#/definitions/CloseLidCreate", + "thermocycler/runProfile": "#/definitions/RunProfileCreate", + "absorbanceReader/initialize": "#/definitions/InitializeCreate", + "absorbanceReader/measure": "#/definitions/MeasureAbsorbanceCreate", + "calibration/calibrateGripper": "#/definitions/CalibrateGripperCreate", + "calibration/calibratePipette": "#/definitions/CalibratePipetteCreate", + "calibration/calibrateModule": "#/definitions/CalibrateModuleCreate", + "calibration/moveToMaintenancePosition": "#/definitions/MoveToMaintenancePositionCreate", + "unsafe/blowOutInPlace": "#/definitions/UnsafeBlowOutInPlaceCreate", + "unsafe/dropTipInPlace": "#/definitions/UnsafeDropTipInPlaceCreate", + "unsafe/updatePositionEstimators": "#/definitions/UpdatePositionEstimatorsCreate" + } + }, + "oneOf": [ + { + "$ref": "#/definitions/AspirateCreate" + }, + { + "$ref": "#/definitions/AspirateInPlaceCreate" + }, + { + "$ref": "#/definitions/CommentCreate" + }, + { + "$ref": "#/definitions/ConfigureForVolumeCreate" + }, + { + "$ref": "#/definitions/ConfigureNozzleLayoutCreate" + }, + { + "$ref": "#/definitions/CustomCreate" + }, + { + "$ref": "#/definitions/DispenseCreate" + }, + { + "$ref": "#/definitions/DispenseInPlaceCreate" + }, + { + "$ref": "#/definitions/BlowOutCreate" + }, + { + "$ref": "#/definitions/BlowOutInPlaceCreate" + }, + { + "$ref": "#/definitions/DropTipCreate" + }, + { + "$ref": "#/definitions/DropTipInPlaceCreate" + }, + { + "$ref": "#/definitions/HomeCreate" + }, + { + "$ref": "#/definitions/RetractAxisCreate" + }, + { + "$ref": "#/definitions/LoadLabwareCreate" + }, + { + "$ref": "#/definitions/ReloadLabwareCreate" + }, + { + "$ref": "#/definitions/LoadLiquidCreate" + }, + { + "$ref": "#/definitions/LoadModuleCreate" + }, + { + "$ref": "#/definitions/LoadPipetteCreate" + }, + { + "$ref": "#/definitions/MoveLabwareCreate" + }, + { + "$ref": "#/definitions/MoveRelativeCreate" + }, + { + "$ref": "#/definitions/MoveToCoordinatesCreate" + }, + { + "$ref": "#/definitions/MoveToWellCreate" + }, + { + "$ref": "#/definitions/MoveToAddressableAreaCreate" + }, + { + "$ref": "#/definitions/MoveToAddressableAreaForDropTipCreate" + }, + { + "$ref": "#/definitions/PrepareToAspirateCreate" + }, + { + "$ref": "#/definitions/WaitForResumeCreate" + }, + { + "$ref": "#/definitions/WaitForDurationCreate" + }, + { + "$ref": "#/definitions/PickUpTipCreate" + }, + { + "$ref": "#/definitions/SavePositionCreate" + }, + { + "$ref": "#/definitions/SetRailLightsCreate" + }, + { + "$ref": "#/definitions/TouchTipCreate" + }, + { + "$ref": "#/definitions/SetStatusBarCreate" + }, + { + "$ref": "#/definitions/VerifyTipPresenceCreate" + }, + { + "$ref": "#/definitions/GetTipPresenceCreate" + }, + { + "$ref": "#/definitions/LiquidProbeCreate" + }, + { + "$ref": "#/definitions/TryLiquidProbeCreate" + }, + { + "$ref": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__wait_for_temperature__WaitForTemperatureCreate" + }, + { + "$ref": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__set_target_temperature__SetTargetTemperatureCreate" + }, + { + "$ref": "#/definitions/DeactivateHeaterCreate" + }, + { + "$ref": "#/definitions/SetAndWaitForShakeSpeedCreate" + }, + { + "$ref": "#/definitions/DeactivateShakerCreate" + }, + { + "$ref": "#/definitions/OpenLabwareLatchCreate" + }, + { + "$ref": "#/definitions/CloseLabwareLatchCreate" + }, + { + "$ref": "#/definitions/DisengageCreate" + }, + { + "$ref": "#/definitions/EngageCreate" + }, + { + "$ref": "#/definitions/opentrons__protocol_engine__commands__temperature_module__set_target_temperature__SetTargetTemperatureCreate" + }, + { + "$ref": "#/definitions/opentrons__protocol_engine__commands__temperature_module__wait_for_temperature__WaitForTemperatureCreate" + }, + { + "$ref": "#/definitions/DeactivateTemperatureCreate" + }, + { + "$ref": "#/definitions/SetTargetBlockTemperatureCreate" + }, + { + "$ref": "#/definitions/WaitForBlockTemperatureCreate" + }, + { + "$ref": "#/definitions/SetTargetLidTemperatureCreate" + }, + { + "$ref": "#/definitions/WaitForLidTemperatureCreate" + }, + { + "$ref": "#/definitions/DeactivateBlockCreate" + }, + { + "$ref": "#/definitions/DeactivateLidCreate" + }, + { + "$ref": "#/definitions/OpenLidCreate" + }, + { + "$ref": "#/definitions/CloseLidCreate" + }, + { + "$ref": "#/definitions/RunProfileCreate" + }, + { + "$ref": "#/definitions/InitializeCreate" + }, + { + "$ref": "#/definitions/MeasureAbsorbanceCreate" + }, + { + "$ref": "#/definitions/CalibrateGripperCreate" + }, + { + "$ref": "#/definitions/CalibratePipetteCreate" + }, + { + "$ref": "#/definitions/CalibrateModuleCreate" + }, + { + "$ref": "#/definitions/MoveToMaintenancePositionCreate" + }, + { + "$ref": "#/definitions/UnsafeBlowOutInPlaceCreate" + }, + { + "$ref": "#/definitions/UnsafeDropTipInPlaceCreate" + }, + { + "$ref": "#/definitions/UpdatePositionEstimatorsCreate" + } + ], + "definitions": { + "WellOrigin": { + "title": "WellOrigin", + "description": "Origin of WellLocation offset.\n\nProps:\n TOP: the top-center of the well\n BOTTOM: the bottom-center of the well\n CENTER: the middle-center of the well", + "enum": ["top", "bottom", "center"], + "type": "string" + }, + "WellOffset": { + "title": "WellOffset", + "description": "An offset vector in (x, y, z).", + "type": "object", + "properties": { + "x": { + "title": "X", + "default": 0, + "type": "number" + }, + "y": { + "title": "Y", + "default": 0, + "type": "number" + }, + "z": { + "title": "Z", + "default": 0, + "type": "number" + } + } + }, + "WellLocation": { + "title": "WellLocation", + "description": "A relative location in reference to a well's location.", + "type": "object", + "properties": { + "origin": { + "default": "top", + "allOf": [ + { + "$ref": "#/definitions/WellOrigin" + } + ] + }, + "offset": { + "$ref": "#/definitions/WellOffset" + } + } + }, + "AspirateParams": { + "title": "AspirateParams", + "description": "Parameters required to aspirate from a specific well.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "volume": { + "title": "Volume", + "description": "The amount of liquid to aspirate, in \u00b5L. Must not be greater than the remaining available amount, which depends on the pipette (see `loadPipette`), its configuration (see `configureForVolume`), the tip (see `pickUpTip`), and the amount you've aspirated so far. There is some tolerance for floating point rounding errors.", + "minimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "flowRate", "volume", "pipetteId"] + }, + "CommandIntent": { + "title": "CommandIntent", + "description": "Run intent for a given command.\n\nProps:\n PROTOCOL: the command is part of the protocol run itself.\n SETUP: the command is part of the setup phase of a run.", + "enum": ["protocol", "setup", "fixit"], + "type": "string" + }, + "AspirateCreate": { + "title": "AspirateCreate", + "description": "Create aspirate command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "aspirate", + "enum": ["aspirate"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/AspirateParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "AspirateInPlaceParams": { + "title": "AspirateInPlaceParams", + "description": "Payload required to aspirate in place.", + "type": "object", + "properties": { + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "volume": { + "title": "Volume", + "description": "The amount of liquid to aspirate, in \u00b5L. Must not be greater than the remaining available amount, which depends on the pipette (see `loadPipette`), its configuration (see `configureForVolume`), the tip (see `pickUpTip`), and the amount you've aspirated so far. There is some tolerance for floating point rounding errors.", + "minimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["flowRate", "volume", "pipetteId"] + }, + "AspirateInPlaceCreate": { + "title": "AspirateInPlaceCreate", + "description": "AspirateInPlace command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "aspirateInPlace", + "enum": ["aspirateInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/AspirateInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CommentParams": { + "title": "CommentParams", + "description": "Payload required to annotate execution with a comment.", + "type": "object", + "properties": { + "message": { + "title": "Message", + "description": "A user-facing message", + "type": "string" + } + }, + "required": ["message"] + }, + "CommentCreate": { + "title": "CommentCreate", + "description": "Comment command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "comment", + "enum": ["comment"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CommentParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "ConfigureForVolumeParams": { + "title": "ConfigureForVolumeParams", + "description": "Parameters required to configure volume for a specific pipette.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "volume": { + "title": "Volume", + "description": "Amount of liquid in uL. Must be at least 0 and no greater than a pipette-specific maximum volume.", + "minimum": 0, + "type": "number" + }, + "tipOverlapNotAfterVersion": { + "title": "Tipoverlapnotafterversion", + "description": "A version of tip overlap data to not exceed. The highest-versioned tip overlap data that does not exceed this version will be used. Versions are expressed as vN where N is an integer, counting up from v0. If None, the current highest version will be used.", + "type": "string" + } + }, + "required": ["pipetteId", "volume"] + }, + "ConfigureForVolumeCreate": { + "title": "ConfigureForVolumeCreate", + "description": "Configure for volume command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "configureForVolume", + "enum": ["configureForVolume"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/ConfigureForVolumeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "AllNozzleLayoutConfiguration": { + "title": "AllNozzleLayoutConfiguration", + "description": "All basemodel to represent a reset to the nozzle configuration. Sending no parameters resets to default.", + "type": "object", + "properties": { + "style": { + "title": "Style", + "default": "ALL", + "enum": ["ALL"], + "type": "string" + } + } + }, + "SingleNozzleLayoutConfiguration": { + "title": "SingleNozzleLayoutConfiguration", + "description": "Minimum information required for a new nozzle configuration.", + "type": "object", + "properties": { + "style": { + "title": "Style", + "default": "SINGLE", + "enum": ["SINGLE"], + "type": "string" + }, + "primaryNozzle": { + "title": "Primarynozzle", + "description": "The primary nozzle to use in the layout configuration. This nozzle will update the critical point of the current pipette. For now, this is also the back left corner of your rectangle.", + "enum": ["A1", "H1", "A12", "H12"], + "type": "string" + } + }, + "required": ["primaryNozzle"] + }, + "RowNozzleLayoutConfiguration": { + "title": "RowNozzleLayoutConfiguration", + "description": "Minimum information required for a new nozzle configuration.", + "type": "object", + "properties": { + "style": { + "title": "Style", + "default": "ROW", + "enum": ["ROW"], + "type": "string" + }, + "primaryNozzle": { + "title": "Primarynozzle", + "description": "The primary nozzle to use in the layout configuration. This nozzle will update the critical point of the current pipette. For now, this is also the back left corner of your rectangle.", + "enum": ["A1", "H1", "A12", "H12"], + "type": "string" + } + }, + "required": ["primaryNozzle"] + }, + "ColumnNozzleLayoutConfiguration": { + "title": "ColumnNozzleLayoutConfiguration", + "description": "Information required for nozzle configurations of type ROW and COLUMN.", + "type": "object", + "properties": { + "style": { + "title": "Style", + "default": "COLUMN", + "enum": ["COLUMN"], + "type": "string" + }, + "primaryNozzle": { + "title": "Primarynozzle", + "description": "The primary nozzle to use in the layout configuration. This nozzle will update the critical point of the current pipette. For now, this is also the back left corner of your rectangle.", + "enum": ["A1", "H1", "A12", "H12"], + "type": "string" + } + }, + "required": ["primaryNozzle"] + }, + "QuadrantNozzleLayoutConfiguration": { + "title": "QuadrantNozzleLayoutConfiguration", + "description": "Information required for nozzle configurations of type QUADRANT.", + "type": "object", + "properties": { + "style": { + "title": "Style", + "default": "QUADRANT", + "enum": ["QUADRANT"], + "type": "string" + }, + "primaryNozzle": { + "title": "Primarynozzle", + "description": "The primary nozzle to use in the layout configuration. This nozzle will update the critical point of the current pipette. For now, this is also the back left corner of your rectangle.", + "enum": ["A1", "H1", "A12", "H12"], + "type": "string" + }, + "frontRightNozzle": { + "title": "Frontrightnozzle", + "description": "The front right nozzle in your configuration.", + "pattern": "[A-Z]\\d{1,2}", + "type": "string" + }, + "backLeftNozzle": { + "title": "Backleftnozzle", + "description": "The back left nozzle in your configuration.", + "pattern": "[A-Z]\\d{1,2}", + "type": "string" + } + }, + "required": ["primaryNozzle", "frontRightNozzle", "backLeftNozzle"] + }, + "ConfigureNozzleLayoutParams": { + "title": "ConfigureNozzleLayoutParams", + "description": "Parameters required to configure the nozzle layout for a specific pipette.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "configurationParams": { + "title": "Configurationparams", + "anyOf": [ + { + "$ref": "#/definitions/AllNozzleLayoutConfiguration" + }, + { + "$ref": "#/definitions/SingleNozzleLayoutConfiguration" + }, + { + "$ref": "#/definitions/RowNozzleLayoutConfiguration" + }, + { + "$ref": "#/definitions/ColumnNozzleLayoutConfiguration" + }, + { + "$ref": "#/definitions/QuadrantNozzleLayoutConfiguration" + } + ] + } + }, + "required": ["pipetteId", "configurationParams"] + }, + "ConfigureNozzleLayoutCreate": { + "title": "ConfigureNozzleLayoutCreate", + "description": "Configure nozzle layout creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "configureNozzleLayout", + "enum": ["configureNozzleLayout"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/ConfigureNozzleLayoutParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CustomParams": { + "title": "CustomParams", + "description": "Payload used by a custom command.", + "type": "object", + "properties": {} + }, + "CustomCreate": { + "title": "CustomCreate", + "description": "A request to create a custom command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "custom", + "enum": ["custom"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CustomParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DispenseParams": { + "title": "DispenseParams", + "description": "Payload required to dispense to a specific well.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "volume": { + "title": "Volume", + "description": "The amount of liquid to dispense, in \u00b5L. Must not be greater than the currently aspirated volume. There is some tolerance for floating point rounding errors.", + "minimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "pushOut": { + "title": "Pushout", + "description": "push the plunger a small amount farther than necessary for accurate low-volume dispensing", + "type": "number" + } + }, + "required": ["labwareId", "wellName", "flowRate", "volume", "pipetteId"] + }, + "DispenseCreate": { + "title": "DispenseCreate", + "description": "Create dispense command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "dispense", + "enum": ["dispense"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DispenseParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DispenseInPlaceParams": { + "title": "DispenseInPlaceParams", + "description": "Payload required to dispense in place.", + "type": "object", + "properties": { + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "volume": { + "title": "Volume", + "description": "The amount of liquid to dispense, in \u00b5L. Must not be greater than the currently aspirated volume. There is some tolerance for floating point rounding errors.", + "minimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "pushOut": { + "title": "Pushout", + "description": "push the plunger a small amount farther than necessary for accurate low-volume dispensing", + "type": "number" + } + }, + "required": ["flowRate", "volume", "pipetteId"] + }, + "DispenseInPlaceCreate": { + "title": "DispenseInPlaceCreate", + "description": "DispenseInPlace command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "dispenseInPlace", + "enum": ["dispenseInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DispenseInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "BlowOutParams": { + "title": "BlowOutParams", + "description": "Payload required to blow-out a specific well.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "flowRate", "pipetteId"] + }, + "BlowOutCreate": { + "title": "BlowOutCreate", + "description": "Create blow-out command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "blowout", + "enum": ["blowout"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/BlowOutParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "BlowOutInPlaceParams": { + "title": "BlowOutInPlaceParams", + "description": "Payload required to blow-out in place.", + "type": "object", + "properties": { + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["flowRate", "pipetteId"] + }, + "BlowOutInPlaceCreate": { + "title": "BlowOutInPlaceCreate", + "description": "BlowOutInPlace command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "blowOutInPlace", + "enum": ["blowOutInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/BlowOutInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DropTipWellOrigin": { + "title": "DropTipWellOrigin", + "description": "The origin of a DropTipWellLocation offset.\n\nProps:\n TOP: the top-center of the well\n BOTTOM: the bottom-center of the well\n CENTER: the middle-center of the well\n DEFAULT: the default drop-tip location of the well,\n based on pipette configuration and length of the tip.", + "enum": ["top", "bottom", "center", "default"], + "type": "string" + }, + "DropTipWellLocation": { + "title": "DropTipWellLocation", + "description": "Like WellLocation, but for dropping tips.\n\nUnlike a typical WellLocation, the location for a drop tip\ndefaults to location based on the tip length rather than the well's top.", + "type": "object", + "properties": { + "origin": { + "default": "default", + "allOf": [ + { + "$ref": "#/definitions/DropTipWellOrigin" + } + ] + }, + "offset": { + "$ref": "#/definitions/WellOffset" + } + } + }, + "DropTipParams": { + "title": "DropTipParams", + "description": "Payload required to drop a tip in a specific well.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to drop the tip.", + "allOf": [ + { + "$ref": "#/definitions/DropTipWellLocation" + } + ] + }, + "homeAfter": { + "title": "Homeafter", + "description": "Whether to home this pipette's plunger after dropping the tip. You should normally leave this unspecified to let the robot choose a safe default depending on its hardware.", + "type": "boolean" + }, + "alternateDropLocation": { + "title": "Alternatedroplocation", + "description": "Whether to alternate location where tip is dropped within the labware. If True, this command will ignore the wellLocation provided and alternate between dropping tips at two predetermined locations inside the specified labware well. If False, the tip will be dropped at the top center of the well.", + "default": false, + "type": "boolean" + } + }, + "required": ["pipetteId", "labwareId", "wellName"] + }, + "DropTipCreate": { + "title": "DropTipCreate", + "description": "Drop tip command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "dropTip", + "enum": ["dropTip"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DropTipParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DropTipInPlaceParams": { + "title": "DropTipInPlaceParams", + "description": "Payload required to drop a tip in place.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "homeAfter": { + "title": "Homeafter", + "description": "Whether to home this pipette's plunger after dropping the tip. You should normally leave this unspecified to let the robot choose a safe default depending on its hardware.", + "type": "boolean" + } + }, + "required": ["pipetteId"] + }, + "DropTipInPlaceCreate": { + "title": "DropTipInPlaceCreate", + "description": "Drop tip in place command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "dropTipInPlace", + "enum": ["dropTipInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DropTipInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MotorAxis": { + "title": "MotorAxis", + "description": "Motor axis on which to issue a home command.", + "enum": [ + "x", + "y", + "leftZ", + "rightZ", + "leftPlunger", + "rightPlunger", + "extensionZ", + "extensionJaw" + ], + "type": "string" + }, + "MountType": { + "title": "MountType", + "description": "An enumeration.", + "enum": ["left", "right", "extension"], + "type": "string" + }, + "HomeParams": { + "title": "HomeParams", + "description": "Payload required for a Home command.", + "type": "object", + "properties": { + "axes": { + "description": "Axes to return to their home positions. If omitted, will home all motors. Extra axes may be implicitly homed to ensure accurate homing of the explicitly specified axes.", + "type": "array", + "items": { + "$ref": "#/definitions/MotorAxis" + } + }, + "skipIfMountPositionOk": { + "description": "If this parameter is provided, the gantry will only be homed if the specified mount has an invalid position. If omitted, the homing action will be executed unconditionally.", + "allOf": [ + { + "$ref": "#/definitions/MountType" + } + ] + } + } + }, + "HomeCreate": { + "title": "HomeCreate", + "description": "Data to create a Home command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "home", + "enum": ["home"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/HomeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "RetractAxisParams": { + "title": "RetractAxisParams", + "description": "Payload required for a Retract Axis command.", + "type": "object", + "properties": { + "axis": { + "description": "Axis to retract to its home position as quickly as safely possible. The difference between retracting an axis and homing an axis using the home command is that a home will always probe the limit switch and will work as the first motion command a robot will need to execute; On the other hand, retraction will rely on this previously determined home position to move to it as fast as safely possible. So on the Flex, it will move (fast) the axis to the previously recorded home position and on the OT2, it will move (fast) the axis a safe distance from the previously recorded home position, and then slowly approach the limit switch.", + "allOf": [ + { + "$ref": "#/definitions/MotorAxis" + } + ] + } + }, + "required": ["axis"] + }, + "RetractAxisCreate": { + "title": "RetractAxisCreate", + "description": "Data to create a Retract Axis command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "retractAxis", + "enum": ["retractAxis"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/RetractAxisParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeckSlotName": { + "title": "DeckSlotName", + "description": "Deck slot identifiers.", + "enum": [ + "1", + "2", + "3", + "4", + "5", + "6", + "7", + "8", + "9", + "10", + "11", + "12", + "A1", + "A2", + "A3", + "B1", + "B2", + "B3", + "C1", + "C2", + "C3", + "D1", + "D2", + "D3" + ] + }, + "DeckSlotLocation": { + "title": "DeckSlotLocation", + "description": "The location of something placed in a single deck slot.", + "type": "object", + "properties": { + "slotName": { + "description": "A slot on the robot's deck.\n\nThe plain numbers like `\"5\"` are for the OT-2, and the coordinates like `\"C2\"` are for the Flex.\n\nWhen you provide one of these values, you can use either style. It will automatically be converted to match the robot.\n\nWhen one of these values is returned, it will always match the robot.", + "allOf": [ + { + "$ref": "#/definitions/DeckSlotName" + } + ] + } + }, + "required": ["slotName"] + }, + "ModuleLocation": { + "title": "ModuleLocation", + "description": "The location of something placed atop a hardware module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "The ID of a loaded module from a prior `loadModule` command.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "OnLabwareLocation": { + "title": "OnLabwareLocation", + "description": "The location of something placed atop another labware.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "The ID of a loaded Labware from a prior `loadLabware` command.", + "type": "string" + } + }, + "required": ["labwareId"] + }, + "AddressableAreaLocation": { + "title": "AddressableAreaLocation", + "description": "The location of something place in an addressable area. This is a superset of deck slots.", + "type": "object", + "properties": { + "addressableAreaName": { + "title": "Addressableareaname", + "description": "The name of the addressable area that you want to use. Valid values are the `id`s of `addressableArea`s in the [deck definition](https://github.com/Opentrons/opentrons/tree/edge/shared-data/deck).", + "type": "string" + } + }, + "required": ["addressableAreaName"] + }, + "LoadLabwareParams": { + "title": "LoadLabwareParams", + "description": "Payload required to load a labware into a slot.", + "type": "object", + "properties": { + "location": { + "title": "Location", + "description": "Location the labware should be loaded into.", + "anyOf": [ + { + "$ref": "#/definitions/DeckSlotLocation" + }, + { + "$ref": "#/definitions/ModuleLocation" + }, + { + "$ref": "#/definitions/OnLabwareLocation" + }, + { + "enum": ["offDeck"], + "type": "string" + }, + { + "$ref": "#/definitions/AddressableAreaLocation" + } + ] + }, + "loadName": { + "title": "Loadname", + "description": "Name used to reference a labware definition.", + "type": "string" + }, + "namespace": { + "title": "Namespace", + "description": "The namespace the labware definition belongs to.", + "type": "string" + }, + "version": { + "title": "Version", + "description": "The labware definition version.", + "type": "integer" + }, + "labwareId": { + "title": "Labwareid", + "description": "An optional ID to assign to this labware. If None, an ID will be generated.", + "type": "string" + }, + "displayName": { + "title": "Displayname", + "description": "An optional user-specified display name or label for this labware.", + "type": "string" + } + }, + "required": ["location", "loadName", "namespace", "version"] + }, + "LoadLabwareCreate": { + "title": "LoadLabwareCreate", + "description": "Load labware command creation request.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "loadLabware", + "enum": ["loadLabware"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/LoadLabwareParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "ReloadLabwareParams": { + "title": "ReloadLabwareParams", + "description": "Payload required to load a labware into a slot.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "The already-loaded labware instance to update.", + "type": "string" + } + }, + "required": ["labwareId"] + }, + "ReloadLabwareCreate": { + "title": "ReloadLabwareCreate", + "description": "Reload labware command creation request.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "reloadLabware", + "enum": ["reloadLabware"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/ReloadLabwareParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "LoadLiquidParams": { + "title": "LoadLiquidParams", + "description": "Payload required to load a liquid into a well.", + "type": "object", + "properties": { + "liquidId": { + "title": "Liquidid", + "description": "Unique identifier of the liquid to load.", + "type": "string" + }, + "labwareId": { + "title": "Labwareid", + "description": "Unique identifier of labware to load liquid into.", + "type": "string" + }, + "volumeByWell": { + "title": "Volumebywell", + "description": "Volume of liquid, in \u00b5L, loaded into each well by name, in this labware.", + "type": "object", + "additionalProperties": { + "type": "number" + } + } + }, + "required": ["liquidId", "labwareId", "volumeByWell"] + }, + "LoadLiquidCreate": { + "title": "LoadLiquidCreate", + "description": "Load liquid command creation request.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "loadLiquid", + "enum": ["loadLiquid"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/LoadLiquidParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "ModuleModel": { + "title": "ModuleModel", + "description": "All available modules' models.", + "enum": [ + "temperatureModuleV1", + "temperatureModuleV2", + "magneticModuleV1", + "magneticModuleV2", + "thermocyclerModuleV1", + "thermocyclerModuleV2", + "heaterShakerModuleV1", + "magneticBlockV1", + "absorbanceReaderV1" + ], + "type": "string" + }, + "LoadModuleParams": { + "title": "LoadModuleParams", + "description": "Payload required to load a module.", + "type": "object", + "properties": { + "model": { + "description": "The model name of the module to load.\n\nProtocol Engine will look for a connected module that either exactly matches this one, or is compatible.\n\n For example, if you request a `temperatureModuleV1` here, Protocol Engine might load a `temperatureModuleV1` or a `temperatureModuleV2`.\n\n The model that it finds connected will be available through `result.model`.", + "allOf": [ + { + "$ref": "#/definitions/ModuleModel" + } + ] + }, + "location": { + "title": "Location", + "description": "The location into which this module should be loaded.\n\nFor the Thermocycler Module, which occupies multiple deck slots, this should be the front-most occupied slot (normally slot 7).", + "allOf": [ + { + "$ref": "#/definitions/DeckSlotLocation" + } + ] + }, + "moduleId": { + "title": "Moduleid", + "description": "An optional ID to assign to this module. If None, an ID will be generated.", + "type": "string" + } + }, + "required": ["model", "location"] + }, + "LoadModuleCreate": { + "title": "LoadModuleCreate", + "description": "The model for a creation request for a load module command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "loadModule", + "enum": ["loadModule"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/LoadModuleParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "PipetteNameType": { + "title": "PipetteNameType", + "description": "Pipette load name values.", + "enum": [ + "p10_single", + "p10_multi", + "p20_single_gen2", + "p20_multi_gen2", + "p50_single", + "p50_multi", + "p50_single_flex", + "p50_multi_flex", + "p300_single", + "p300_multi", + "p300_single_gen2", + "p300_multi_gen2", + "p1000_single", + "p1000_single_gen2", + "p1000_single_flex", + "p1000_multi_flex", + "p1000_96" + ], + "type": "string" + }, + "LoadPipetteParams": { + "title": "LoadPipetteParams", + "description": "Payload needed to load a pipette on to a mount.", + "type": "object", + "properties": { + "pipetteName": { + "description": "The load name of the pipette to be required.", + "allOf": [ + { + "$ref": "#/definitions/PipetteNameType" + } + ] + }, + "mount": { + "description": "The mount the pipette should be present on.", + "allOf": [ + { + "$ref": "#/definitions/MountType" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "An optional ID to assign to this pipette. If None, an ID will be generated.", + "type": "string" + }, + "tipOverlapNotAfterVersion": { + "title": "Tipoverlapnotafterversion", + "description": "A version of tip overlap data to not exceed. The highest-versioned tip overlap data that does not exceed this version will be used. Versions are expressed as vN where N is an integer, counting up from v0. If None, the current highest version will be used.", + "type": "string" + }, + "liquidPresenceDetection": { + "title": "Liquidpresencedetection", + "description": "Enable liquid presence detection for this pipette. Defaults to False.", + "type": "boolean" + } + }, + "required": ["pipetteName", "mount"] + }, + "LoadPipetteCreate": { + "title": "LoadPipetteCreate", + "description": "Load pipette command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "loadPipette", + "enum": ["loadPipette"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/LoadPipetteParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "LabwareMovementStrategy": { + "title": "LabwareMovementStrategy", + "description": "Strategy to use for labware movement.", + "enum": ["usingGripper", "manualMoveWithPause", "manualMoveWithoutPause"], + "type": "string" + }, + "LabwareOffsetVector": { + "title": "LabwareOffsetVector", + "description": "Offset, in deck coordinates from nominal to actual position.", + "type": "object", + "properties": { + "x": { + "title": "X", + "type": "number" + }, + "y": { + "title": "Y", + "type": "number" + }, + "z": { + "title": "Z", + "type": "number" + } + }, + "required": ["x", "y", "z"] + }, + "MoveLabwareParams": { + "title": "MoveLabwareParams", + "description": "Input parameters for a ``moveLabware`` command.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "The ID of the labware to move.", + "type": "string" + }, + "newLocation": { + "title": "Newlocation", + "description": "Where to move the labware.", + "anyOf": [ + { + "$ref": "#/definitions/DeckSlotLocation" + }, + { + "$ref": "#/definitions/ModuleLocation" + }, + { + "$ref": "#/definitions/OnLabwareLocation" + }, + { + "enum": ["offDeck"], + "type": "string" + }, + { + "$ref": "#/definitions/AddressableAreaLocation" + } + ] + }, + "strategy": { + "description": "Whether to use the gripper to perform the labware movement or to perform a manual movement with an option to pause.", + "allOf": [ + { + "$ref": "#/definitions/LabwareMovementStrategy" + } + ] + }, + "pickUpOffset": { + "title": "Pickupoffset", + "description": "Offset to use when picking up labware. Experimental param, subject to change", + "allOf": [ + { + "$ref": "#/definitions/LabwareOffsetVector" + } + ] + }, + "dropOffset": { + "title": "Dropoffset", + "description": "Offset to use when dropping off labware. Experimental param, subject to change", + "allOf": [ + { + "$ref": "#/definitions/LabwareOffsetVector" + } + ] + } + }, + "required": ["labwareId", "newLocation", "strategy"] + }, + "MoveLabwareCreate": { + "title": "MoveLabwareCreate", + "description": "A request to create a ``moveLabware`` command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveLabware", + "enum": ["moveLabware"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveLabwareParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MovementAxis": { + "title": "MovementAxis", + "description": "Axis on which to issue a relative movement.", + "enum": ["x", "y", "z"], + "type": "string" + }, + "MoveRelativeParams": { + "title": "MoveRelativeParams", + "description": "Payload required for a MoveRelative command.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Pipette to move.", + "type": "string" + }, + "axis": { + "description": "Axis along which to move.", + "allOf": [ + { + "$ref": "#/definitions/MovementAxis" + } + ] + }, + "distance": { + "title": "Distance", + "description": "Distance to move in millimeters. A positive number will move towards the right (x), back (y), top (z) of the deck.", + "type": "number" + } + }, + "required": ["pipetteId", "axis", "distance"] + }, + "MoveRelativeCreate": { + "title": "MoveRelativeCreate", + "description": "Data to create a MoveRelative command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveRelative", + "enum": ["moveRelative"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveRelativeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeckPoint": { + "title": "DeckPoint", + "description": "Coordinates of a point in deck space.", + "type": "object", + "properties": { + "x": { + "title": "X", + "type": "number" + }, + "y": { + "title": "Y", + "type": "number" + }, + "z": { + "title": "Z", + "type": "number" + } + }, + "required": ["x", "y", "z"] + }, + "MoveToCoordinatesParams": { + "title": "MoveToCoordinatesParams", + "description": "Payload required to move a pipette to coordinates.", + "type": "object", + "properties": { + "minimumZHeight": { + "title": "Minimumzheight", + "description": "Optional minimal Z margin in mm. If this is larger than the API's default safe Z margin, it will make the arc higher. If it's smaller, it will have no effect.", + "type": "number" + }, + "forceDirect": { + "title": "Forcedirect", + "description": "If true, moving from one labware/well to another will not arc to the default safe z, but instead will move directly to the specified location. This will also force the `minimumZHeight` param to be ignored. A 'direct' movement is in X/Y/Z simultaneously.", + "default": false, + "type": "boolean" + }, + "speed": { + "title": "Speed", + "description": "Override the travel speed in mm/s. This controls the straight linear speed of motion.", + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "coordinates": { + "title": "Coordinates", + "description": "X, Y and Z coordinates in mm from deck's origin location (left-front-bottom corner of work space)", + "allOf": [ + { + "$ref": "#/definitions/DeckPoint" + } + ] + } + }, + "required": ["pipetteId", "coordinates"] + }, + "MoveToCoordinatesCreate": { + "title": "MoveToCoordinatesCreate", + "description": "Move to coordinates command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveToCoordinates", + "enum": ["moveToCoordinates"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveToCoordinatesParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MoveToWellParams": { + "title": "MoveToWellParams", + "description": "Payload required to move a pipette to a specific well.", + "type": "object", + "properties": { + "minimumZHeight": { + "title": "Minimumzheight", + "description": "Optional minimal Z margin in mm. If this is larger than the API's default safe Z margin, it will make the arc higher. If it's smaller, it will have no effect.", + "type": "number" + }, + "forceDirect": { + "title": "Forcedirect", + "description": "If true, moving from one labware/well to another will not arc to the default safe z, but instead will move directly to the specified location. This will also force the `minimumZHeight` param to be ignored. A 'direct' movement is in X/Y/Z simultaneously.", + "default": false, + "type": "boolean" + }, + "speed": { + "title": "Speed", + "description": "Override the travel speed in mm/s. This controls the straight linear speed of motion.", + "type": "number" + }, + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "pipetteId"] + }, + "MoveToWellCreate": { + "title": "MoveToWellCreate", + "description": "Move to well command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveToWell", + "enum": ["moveToWell"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveToWellParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "AddressableOffsetVector": { + "title": "AddressableOffsetVector", + "description": "Offset, in deck coordinates, from nominal to actual position of an addressable area.", + "type": "object", + "properties": { + "x": { + "title": "X", + "type": "number" + }, + "y": { + "title": "Y", + "type": "number" + }, + "z": { + "title": "Z", + "type": "number" + } + }, + "required": ["x", "y", "z"] + }, + "MoveToAddressableAreaParams": { + "title": "MoveToAddressableAreaParams", + "description": "Payload required to move a pipette to a specific addressable area.\n\nAn *addressable area* is a space in the robot that may or may not be usable depending on how\nthe robot's deck is configured. For example, if a Flex is configured with a waste chute, it will\nhave additional addressable areas representing the opening of the waste chute, where tips and\nlabware can be dropped.\n\nThis moves the pipette so all of its nozzles are centered over the addressable area.\nIf the pipette is currently configured with a partial tip layout, this centering is over all\nthe pipette's physical nozzles, not just the nozzles that are active.\n\nThe z-position will be chosen to put the bottom of the tips---or the bottom of the nozzles,\nif there are no tips---level with the top of the addressable area.\n\nWhen this command is executed, Protocol Engine will make sure the robot's deck is configured\nsuch that the requested addressable area actually exists. For example, if you request\nthe addressable area B4, it will make sure the robot is set up with a B3/B4 staging area slot.\nIf that's not the case, the command will fail.", + "type": "object", + "properties": { + "minimumZHeight": { + "title": "Minimumzheight", + "description": "Optional minimal Z margin in mm. If this is larger than the API's default safe Z margin, it will make the arc higher. If it's smaller, it will have no effect.", + "type": "number" + }, + "forceDirect": { + "title": "Forcedirect", + "description": "If true, moving from one labware/well to another will not arc to the default safe z, but instead will move directly to the specified location. This will also force the `minimumZHeight` param to be ignored. A 'direct' movement is in X/Y/Z simultaneously.", + "default": false, + "type": "boolean" + }, + "speed": { + "title": "Speed", + "description": "Override the travel speed in mm/s. This controls the straight linear speed of motion.", + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "addressableAreaName": { + "title": "Addressableareaname", + "description": "The name of the addressable area that you want to use. Valid values are the `id`s of `addressableArea`s in the [deck definition](https://github.com/Opentrons/opentrons/tree/edge/shared-data/deck).", + "type": "string" + }, + "offset": { + "title": "Offset", + "description": "Relative offset of addressable area to move pipette's critical point.", + "default": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "allOf": [ + { + "$ref": "#/definitions/AddressableOffsetVector" + } + ] + }, + "stayAtHighestPossibleZ": { + "title": "Stayathighestpossiblez", + "description": "If `true`, the pipette will retract to its highest possible height and stay there instead of descending to the destination. `minimumZHeight` will be ignored.", + "default": false, + "type": "boolean" + } + }, + "required": ["pipetteId", "addressableAreaName"] + }, + "MoveToAddressableAreaCreate": { + "title": "MoveToAddressableAreaCreate", + "description": "Move to addressable area command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveToAddressableArea", + "enum": ["moveToAddressableArea"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveToAddressableAreaParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MoveToAddressableAreaForDropTipParams": { + "title": "MoveToAddressableAreaForDropTipParams", + "description": "Payload required to move a pipette to a specific addressable area.\n\nAn *addressable area* is a space in the robot that may or may not be usable depending on how\nthe robot's deck is configured. For example, if a Flex is configured with a waste chute, it will\nhave additional addressable areas representing the opening of the waste chute, where tips and\nlabware can be dropped.\n\nThis moves the pipette so all of its nozzles are centered over the addressable area.\nIf the pipette is currently configured with a partial tip layout, this centering is over all\nthe pipette's physical nozzles, not just the nozzles that are active.\n\nThe z-position will be chosen to put the bottom of the tips---or the bottom of the nozzles,\nif there are no tips---level with the top of the addressable area.\n\nWhen this command is executed, Protocol Engine will make sure the robot's deck is configured\nsuch that the requested addressable area actually exists. For example, if you request\nthe addressable area B4, it will make sure the robot is set up with a B3/B4 staging area slot.\nIf that's not the case, the command will fail.", + "type": "object", + "properties": { + "minimumZHeight": { + "title": "Minimumzheight", + "description": "Optional minimal Z margin in mm. If this is larger than the API's default safe Z margin, it will make the arc higher. If it's smaller, it will have no effect.", + "type": "number" + }, + "forceDirect": { + "title": "Forcedirect", + "description": "If true, moving from one labware/well to another will not arc to the default safe z, but instead will move directly to the specified location. This will also force the `minimumZHeight` param to be ignored. A 'direct' movement is in X/Y/Z simultaneously.", + "default": false, + "type": "boolean" + }, + "speed": { + "title": "Speed", + "description": "Override the travel speed in mm/s. This controls the straight linear speed of motion.", + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "addressableAreaName": { + "title": "Addressableareaname", + "description": "The name of the addressable area that you want to use. Valid values are the `id`s of `addressableArea`s in the [deck definition](https://github.com/Opentrons/opentrons/tree/edge/shared-data/deck).", + "type": "string" + }, + "offset": { + "title": "Offset", + "description": "Relative offset of addressable area to move pipette's critical point.", + "default": { + "x": 0.0, + "y": 0.0, + "z": 0.0 + }, + "allOf": [ + { + "$ref": "#/definitions/AddressableOffsetVector" + } + ] + }, + "alternateDropLocation": { + "title": "Alternatedroplocation", + "description": "Whether to alternate location where tip is dropped within the addressable area. If True, this command will ignore the offset provided and alternate between dropping tips at two predetermined locations inside the specified labware well. If False, the tip will be dropped at the top center of the area.", + "default": false, + "type": "boolean" + }, + "ignoreTipConfiguration": { + "title": "Ignoretipconfiguration", + "description": "Whether to utilize the critical point of the tip configuraiton when moving to an addressable area. If True, this command will ignore the tip configuration and use the center of the entire instrument as the critical point for movement. If False, this command will use the critical point provided by the current tip configuration.", + "default": true, + "type": "boolean" + } + }, + "required": ["pipetteId", "addressableAreaName"] + }, + "MoveToAddressableAreaForDropTipCreate": { + "title": "MoveToAddressableAreaForDropTipCreate", + "description": "Move to addressable area for drop tip command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "moveToAddressableAreaForDropTip", + "enum": ["moveToAddressableAreaForDropTip"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveToAddressableAreaForDropTipParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "PrepareToAspirateParams": { + "title": "PrepareToAspirateParams", + "description": "Parameters required to prepare a specific pipette for aspiration.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["pipetteId"] + }, + "PrepareToAspirateCreate": { + "title": "PrepareToAspirateCreate", + "description": "Prepare for aspirate command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "prepareToAspirate", + "enum": ["prepareToAspirate"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/PrepareToAspirateParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "WaitForResumeParams": { + "title": "WaitForResumeParams", + "description": "Payload required to pause the protocol.", + "type": "object", + "properties": { + "message": { + "title": "Message", + "description": "A user-facing message associated with the pause", + "type": "string" + } + } + }, + "WaitForResumeCreate": { + "title": "WaitForResumeCreate", + "description": "Wait for resume command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "waitForResume", + "enum": ["waitForResume", "pause"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/WaitForResumeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "WaitForDurationParams": { + "title": "WaitForDurationParams", + "description": "Payload required to pause the protocol.", + "type": "object", + "properties": { + "seconds": { + "title": "Seconds", + "description": "Duration, in seconds, to wait for.", + "type": "number" + }, + "message": { + "title": "Message", + "description": "A user-facing message associated with the pause", + "type": "string" + } + }, + "required": ["seconds"] + }, + "WaitForDurationCreate": { + "title": "WaitForDurationCreate", + "description": "Wait for duration command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "waitForDuration", + "enum": ["waitForDuration"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/WaitForDurationParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "PickUpTipParams": { + "title": "PickUpTipParams", + "description": "Payload needed to move a pipette to a specific well.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "pipetteId"] + }, + "PickUpTipCreate": { + "title": "PickUpTipCreate", + "description": "Pick up tip command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "pickUpTip", + "enum": ["pickUpTip"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/PickUpTipParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "SavePositionParams": { + "title": "SavePositionParams", + "description": "Payload needed to save a pipette's current position.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Unique identifier of the pipette in question.", + "type": "string" + }, + "positionId": { + "title": "Positionid", + "description": "An optional ID to assign to this command instance. Auto-assigned if not defined.", + "type": "string" + }, + "failOnNotHomed": { + "title": "Failonnothomed", + "default": true, + "descrption": "Require all axes to be homed before saving position.", + "type": "boolean" + } + }, + "required": ["pipetteId"] + }, + "SavePositionCreate": { + "title": "SavePositionCreate", + "description": "Save position command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "savePosition", + "enum": ["savePosition"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SavePositionParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "SetRailLightsParams": { + "title": "SetRailLightsParams", + "description": "Payload required to set the rail lights on or off.", + "type": "object", + "properties": { + "on": { + "title": "On", + "description": "The field that determines if the light is turned off or on.", + "type": "boolean" + } + }, + "required": ["on"] + }, + "SetRailLightsCreate": { + "title": "SetRailLightsCreate", + "description": "setRailLights command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "setRailLights", + "enum": ["setRailLights"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SetRailLightsParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "TouchTipParams": { + "title": "TouchTipParams", + "description": "Payload needed to touch a pipette tip the sides of a specific well.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "radius": { + "title": "Radius", + "description": "The proportion of the target well's radius the pipette tip will move towards.", + "default": 1.0, + "type": "number" + }, + "speed": { + "title": "Speed", + "description": "Override the travel speed in mm/s. This controls the straight linear speed of motion.", + "type": "number" + } + }, + "required": ["labwareId", "wellName", "pipetteId"] + }, + "TouchTipCreate": { + "title": "TouchTipCreate", + "description": "Touch tip command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "touchTip", + "enum": ["touchTip"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/TouchTipParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "StatusBarAnimation": { + "title": "StatusBarAnimation", + "description": "Status Bar animation options.", + "enum": ["idle", "confirm", "updating", "disco", "off"] + }, + "SetStatusBarParams": { + "title": "SetStatusBarParams", + "description": "Payload required to set the status bar to run an animation.", + "type": "object", + "properties": { + "animation": { + "description": "The animation that should be executed on the status bar.", + "allOf": [ + { + "$ref": "#/definitions/StatusBarAnimation" + } + ] + } + }, + "required": ["animation"] + }, + "SetStatusBarCreate": { + "title": "SetStatusBarCreate", + "description": "setStatusBar command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "setStatusBar", + "enum": ["setStatusBar"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SetStatusBarParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "TipPresenceStatus": { + "title": "TipPresenceStatus", + "description": "Tip presence status reported by a pipette.", + "enum": ["present", "absent", "unknown"], + "type": "string" + }, + "InstrumentSensorId": { + "title": "InstrumentSensorId", + "description": "Primary and secondary sensor ids.", + "enum": ["primary", "secondary", "both"], + "type": "string" + }, + "VerifyTipPresenceParams": { + "title": "VerifyTipPresenceParams", + "description": "Payload required for a VerifyTipPresence command.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "expectedState": { + "description": "The expected tip presence status on the pipette.", + "allOf": [ + { + "$ref": "#/definitions/TipPresenceStatus" + } + ] + }, + "followSingularSensor": { + "description": "The sensor id to follow if the other can be ignored.", + "allOf": [ + { + "$ref": "#/definitions/InstrumentSensorId" + } + ] + } + }, + "required": ["pipetteId", "expectedState"] + }, + "VerifyTipPresenceCreate": { + "title": "VerifyTipPresenceCreate", + "description": "VerifyTipPresence command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "verifyTipPresence", + "enum": ["verifyTipPresence"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/VerifyTipPresenceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "GetTipPresenceParams": { + "title": "GetTipPresenceParams", + "description": "Payload required for a GetTipPresence command.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["pipetteId"] + }, + "GetTipPresenceCreate": { + "title": "GetTipPresenceCreate", + "description": "GetTipPresence command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "getTipPresence", + "enum": ["getTipPresence"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/GetTipPresenceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "LiquidProbeParams": { + "title": "LiquidProbeParams", + "description": "Parameters required for a `liquidProbe` command.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "pipetteId"] + }, + "LiquidProbeCreate": { + "title": "LiquidProbeCreate", + "description": "The request model for a `liquidProbe` command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "liquidProbe", + "enum": ["liquidProbe"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/LiquidProbeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "TryLiquidProbeParams": { + "title": "TryLiquidProbeParams", + "description": "Parameters required for a `tryLiquidProbe` command.", + "type": "object", + "properties": { + "labwareId": { + "title": "Labwareid", + "description": "Identifier of labware to use.", + "type": "string" + }, + "wellName": { + "title": "Wellname", + "description": "Name of well to use in labware.", + "type": "string" + }, + "wellLocation": { + "title": "Welllocation", + "description": "Relative well location at which to perform the operation", + "allOf": [ + { + "$ref": "#/definitions/WellLocation" + } + ] + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["labwareId", "wellName", "pipetteId"] + }, + "TryLiquidProbeCreate": { + "title": "TryLiquidProbeCreate", + "description": "The request model for a `tryLiquidProbe` command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "tryLiquidProbe", + "enum": ["tryLiquidProbe"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/TryLiquidProbeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "opentrons__protocol_engine__commands__heater_shaker__wait_for_temperature__WaitForTemperatureParams": { + "title": "WaitForTemperatureParams", + "description": "Input parameters to wait for a Heater-Shaker's target temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C. If not specified, will default to the module's target temperature. Specifying a celsius parameter other than the target temperature could lead to unpredictable behavior and hence is not recommended for use. This parameter can be removed in a future version without prior notice.", + "type": "number" + } + }, + "required": ["moduleId"] + }, + "opentrons__protocol_engine__commands__heater_shaker__wait_for_temperature__WaitForTemperatureCreate": { + "title": "WaitForTemperatureCreate", + "description": "A request to create a Heater-Shaker's wait for temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/waitForTemperature", + "enum": ["heaterShaker/waitForTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__wait_for_temperature__WaitForTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "opentrons__protocol_engine__commands__heater_shaker__set_target_temperature__SetTargetTemperatureParams": { + "title": "SetTargetTemperatureParams", + "description": "Input parameters to set a Heater-Shaker's target temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C.", + "type": "number" + } + }, + "required": ["moduleId", "celsius"] + }, + "opentrons__protocol_engine__commands__heater_shaker__set_target_temperature__SetTargetTemperatureCreate": { + "title": "SetTargetTemperatureCreate", + "description": "A request to create a Heater-Shaker's set temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/setTargetTemperature", + "enum": ["heaterShaker/setTargetTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/opentrons__protocol_engine__commands__heater_shaker__set_target_temperature__SetTargetTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeactivateHeaterParams": { + "title": "DeactivateHeaterParams", + "description": "Input parameters to unset a Heater-Shaker's target temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DeactivateHeaterCreate": { + "title": "DeactivateHeaterCreate", + "description": "A request to create a Heater-Shaker's deactivate heater command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/deactivateHeater", + "enum": ["heaterShaker/deactivateHeater"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DeactivateHeaterParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "SetAndWaitForShakeSpeedParams": { + "title": "SetAndWaitForShakeSpeedParams", + "description": "Input parameters to set and wait for a shake speed for a Heater-Shaker Module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + }, + "rpm": { + "title": "Rpm", + "description": "Target speed in rotations per minute.", + "type": "number" + } + }, + "required": ["moduleId", "rpm"] + }, + "SetAndWaitForShakeSpeedCreate": { + "title": "SetAndWaitForShakeSpeedCreate", + "description": "A request to create a Heater-Shaker's set and wait for shake speed command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/setAndWaitForShakeSpeed", + "enum": ["heaterShaker/setAndWaitForShakeSpeed"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SetAndWaitForShakeSpeedParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeactivateShakerParams": { + "title": "DeactivateShakerParams", + "description": "Input parameters to deactivate shaker for a Heater-Shaker Module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DeactivateShakerCreate": { + "title": "DeactivateShakerCreate", + "description": "A request to create a Heater-Shaker's deactivate shaker command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/deactivateShaker", + "enum": ["heaterShaker/deactivateShaker"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DeactivateShakerParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "OpenLabwareLatchParams": { + "title": "OpenLabwareLatchParams", + "description": "Input parameters to open a Heater-Shaker Module's labware latch.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "OpenLabwareLatchCreate": { + "title": "OpenLabwareLatchCreate", + "description": "A request to create a Heater-Shaker's open labware latch command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/openLabwareLatch", + "enum": ["heaterShaker/openLabwareLatch"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/OpenLabwareLatchParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CloseLabwareLatchParams": { + "title": "CloseLabwareLatchParams", + "description": "Input parameters to close a Heater-Shaker Module's labware latch.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Heater-Shaker Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "CloseLabwareLatchCreate": { + "title": "CloseLabwareLatchCreate", + "description": "A request to create a Heater-Shaker's close latch command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "heaterShaker/closeLabwareLatch", + "enum": ["heaterShaker/closeLabwareLatch"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CloseLabwareLatchParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DisengageParams": { + "title": "DisengageParams", + "description": "Input data to disengage a Magnetic Module's magnets.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "The ID of the Magnetic Module whose magnets you want to disengage, from a prior `loadModule` command.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DisengageCreate": { + "title": "DisengageCreate", + "description": "A request to create a Magnetic Module disengage command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "magneticModule/disengage", + "enum": ["magneticModule/disengage"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DisengageParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "EngageParams": { + "title": "EngageParams", + "description": "Input data to engage a Magnetic Module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "The ID of the Magnetic Module whose magnets you want to raise, from a prior `loadModule` command.", + "type": "string" + }, + "height": { + "title": "Height", + "description": "How high, in millimeters, to raise the magnets.\n\nZero means the tops of the magnets are level with the ledge that the labware rests on. This will be slightly above the magnets' minimum height, the hardware home position. Negative values are allowed, to put the magnets below the ledge.\n\nUnits are always true millimeters. This is unlike certain labware definitions, engage commands in the Python Protocol API, and engage commands in older versions of the JSON protocol schema. Take care to convert properly.", + "type": "number" + } + }, + "required": ["moduleId", "height"] + }, + "EngageCreate": { + "title": "EngageCreate", + "description": "A request to create a Magnetic Module engage command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "magneticModule/engage", + "enum": ["magneticModule/engage"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/EngageParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "opentrons__protocol_engine__commands__temperature_module__set_target_temperature__SetTargetTemperatureParams": { + "title": "SetTargetTemperatureParams", + "description": "Input parameters to set a Temperature Module's target temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Temperature Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C.", + "type": "number" + } + }, + "required": ["moduleId", "celsius"] + }, + "opentrons__protocol_engine__commands__temperature_module__set_target_temperature__SetTargetTemperatureCreate": { + "title": "SetTargetTemperatureCreate", + "description": "A request to create a Temperature Module's set temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "temperatureModule/setTargetTemperature", + "enum": ["temperatureModule/setTargetTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/opentrons__protocol_engine__commands__temperature_module__set_target_temperature__SetTargetTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "opentrons__protocol_engine__commands__temperature_module__wait_for_temperature__WaitForTemperatureParams": { + "title": "WaitForTemperatureParams", + "description": "Input parameters to wait for a Temperature Module's target temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Temperature Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C. If not specified, will default to the module's target temperature. Specifying a celsius parameter other than the target temperature could lead to unpredictable behavior and hence is not recommended for use. This parameter can be removed in a future version without prior notice.", + "type": "number" + } + }, + "required": ["moduleId"] + }, + "opentrons__protocol_engine__commands__temperature_module__wait_for_temperature__WaitForTemperatureCreate": { + "title": "WaitForTemperatureCreate", + "description": "A request to create a Temperature Module's wait for temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "temperatureModule/waitForTemperature", + "enum": ["temperatureModule/waitForTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/opentrons__protocol_engine__commands__temperature_module__wait_for_temperature__WaitForTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeactivateTemperatureParams": { + "title": "DeactivateTemperatureParams", + "description": "Input parameters to deactivate a Temperature Module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Temperature Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DeactivateTemperatureCreate": { + "title": "DeactivateTemperatureCreate", + "description": "A request to deactivate a Temperature Module.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "temperatureModule/deactivate", + "enum": ["temperatureModule/deactivate"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DeactivateTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "SetTargetBlockTemperatureParams": { + "title": "SetTargetBlockTemperatureParams", + "description": "Input parameters to set a Thermocycler's target block temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C.", + "type": "number" + }, + "blockMaxVolumeUl": { + "title": "Blockmaxvolumeul", + "description": "Amount of liquid in uL of the most-full well in labware loaded onto the thermocycler.", + "type": "number" + }, + "holdTimeSeconds": { + "title": "Holdtimeseconds", + "description": "Amount of time, in seconds, to hold the temperature for. If specified, a waitForBlockTemperature command will block until the given hold time has elapsed.", + "type": "number" + } + }, + "required": ["moduleId", "celsius"] + }, + "SetTargetBlockTemperatureCreate": { + "title": "SetTargetBlockTemperatureCreate", + "description": "A request to create a Thermocycler's set block temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/setTargetBlockTemperature", + "enum": ["thermocycler/setTargetBlockTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SetTargetBlockTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "WaitForBlockTemperatureParams": { + "title": "WaitForBlockTemperatureParams", + "description": "Input parameters to wait for Thermocycler's target block temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "WaitForBlockTemperatureCreate": { + "title": "WaitForBlockTemperatureCreate", + "description": "A request to create Thermocycler's wait for block temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/waitForBlockTemperature", + "enum": ["thermocycler/waitForBlockTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/WaitForBlockTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "SetTargetLidTemperatureParams": { + "title": "SetTargetLidTemperatureParams", + "description": "Input parameters to set a Thermocycler's target lid temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler Module.", + "type": "string" + }, + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C.", + "type": "number" + } + }, + "required": ["moduleId", "celsius"] + }, + "SetTargetLidTemperatureCreate": { + "title": "SetTargetLidTemperatureCreate", + "description": "A request to create a Thermocycler's set lid temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/setTargetLidTemperature", + "enum": ["thermocycler/setTargetLidTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/SetTargetLidTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "WaitForLidTemperatureParams": { + "title": "WaitForLidTemperatureParams", + "description": "Input parameters to wait for Thermocycler's lid temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler Module.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "WaitForLidTemperatureCreate": { + "title": "WaitForLidTemperatureCreate", + "description": "A request to create Thermocycler's wait for lid temperature command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/waitForLidTemperature", + "enum": ["thermocycler/waitForLidTemperature"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/WaitForLidTemperatureParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeactivateBlockParams": { + "title": "DeactivateBlockParams", + "description": "Input parameters to unset a Thermocycler's target block temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DeactivateBlockCreate": { + "title": "DeactivateBlockCreate", + "description": "A request to create a Thermocycler's deactivate block command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/deactivateBlock", + "enum": ["thermocycler/deactivateBlock"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DeactivateBlockParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "DeactivateLidParams": { + "title": "DeactivateLidParams", + "description": "Input parameters to unset a Thermocycler's target lid temperature.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "DeactivateLidCreate": { + "title": "DeactivateLidCreate", + "description": "A request to create a Thermocycler's deactivate lid command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/deactivateLid", + "enum": ["thermocycler/deactivateLid"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/DeactivateLidParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "OpenLidParams": { + "title": "OpenLidParams", + "description": "Input parameters to open a Thermocycler's lid.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "OpenLidCreate": { + "title": "OpenLidCreate", + "description": "A request to open a Thermocycler's lid.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/openLid", + "enum": ["thermocycler/openLid"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/OpenLidParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CloseLidParams": { + "title": "CloseLidParams", + "description": "Input parameters to close a Thermocycler's lid.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler.", + "type": "string" + } + }, + "required": ["moduleId"] + }, + "CloseLidCreate": { + "title": "CloseLidCreate", + "description": "A request to close a Thermocycler's lid.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/closeLid", + "enum": ["thermocycler/closeLid"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CloseLidParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "RunProfileStepParams": { + "title": "RunProfileStepParams", + "description": "Input parameters for an individual Thermocycler profile step.", + "type": "object", + "properties": { + "celsius": { + "title": "Celsius", + "description": "Target temperature in \u00b0C.", + "type": "number" + }, + "holdSeconds": { + "title": "Holdseconds", + "description": "Time to hold target temperature at in seconds.", + "type": "number" + } + }, + "required": ["celsius", "holdSeconds"] + }, + "RunProfileParams": { + "title": "RunProfileParams", + "description": "Input parameters to run a Thermocycler profile.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Thermocycler.", + "type": "string" + }, + "profile": { + "title": "Profile", + "description": "Array of profile steps with target temperature and temperature hold time.", + "type": "array", + "items": { + "$ref": "#/definitions/RunProfileStepParams" + } + }, + "blockMaxVolumeUl": { + "title": "Blockmaxvolumeul", + "description": "Amount of liquid in uL of the most-full well in labware loaded onto the thermocycler.", + "type": "number" + } + }, + "required": ["moduleId", "profile"] + }, + "RunProfileCreate": { + "title": "RunProfileCreate", + "description": "A request to execute a Thermocycler profile run.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "thermocycler/runProfile", + "enum": ["thermocycler/runProfile"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/RunProfileParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "InitializeParams": { + "title": "InitializeParams", + "description": "Input parameters to initialize an absorbance reading.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the absorbance reader.", + "type": "string" + }, + "sampleWavelength": { + "title": "Samplewavelength", + "description": "Sample wavelength in nm.", + "type": "integer" + } + }, + "required": ["moduleId", "sampleWavelength"] + }, + "InitializeCreate": { + "title": "InitializeCreate", + "description": "A request to execute an Absorbance Reader measurement.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "absorbanceReader/initialize", + "enum": ["absorbanceReader/initialize"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/InitializeParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MeasureAbsorbanceParams": { + "title": "MeasureAbsorbanceParams", + "description": "Input parameters for a single absorbance reading.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "Unique ID of the Absorbance Reader.", + "type": "string" + }, + "sampleWavelength": { + "title": "Samplewavelength", + "description": "Sample wavelength in nm.", + "type": "integer" + } + }, + "required": ["moduleId", "sampleWavelength"] + }, + "MeasureAbsorbanceCreate": { + "title": "MeasureAbsorbanceCreate", + "description": "A request to execute an Absorbance Reader measurement.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "absorbanceReader/measure", + "enum": ["absorbanceReader/measure"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MeasureAbsorbanceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CalibrateGripperParamsJaw": { + "title": "CalibrateGripperParamsJaw", + "description": "An enumeration.", + "enum": ["front", "rear"] + }, + "Vec3f": { + "title": "Vec3f", + "description": "A 3D vector of floats.", + "type": "object", + "properties": { + "x": { + "title": "X", + "type": "number" + }, + "y": { + "title": "Y", + "type": "number" + }, + "z": { + "title": "Z", + "type": "number" + } + }, + "required": ["x", "y", "z"] + }, + "CalibrateGripperParams": { + "title": "CalibrateGripperParams", + "description": "Parameters for a `calibrateGripper` command.", + "type": "object", + "properties": { + "jaw": { + "description": "Which of the gripper's jaws to use to measure its offset. The robot will assume that a human operator has already attached the capacitive probe to the jaw and none is attached to the other jaw.", + "allOf": [ + { + "$ref": "#/definitions/CalibrateGripperParamsJaw" + } + ] + }, + "otherJawOffset": { + "title": "Otherjawoffset", + "description": "If an offset for the other probe is already found, then specifying it here will enable the CalibrateGripper command to complete the calibration process by calculating the total offset and saving it to disk. If this param is not specified then the command will only find and return the offset for the specified probe.", + "allOf": [ + { + "$ref": "#/definitions/Vec3f" + } + ] + } + }, + "required": ["jaw"] + }, + "CalibrateGripperCreate": { + "title": "CalibrateGripperCreate", + "description": "A request to create a `calibrateGripper` command.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "calibration/calibrateGripper", + "enum": ["calibration/calibrateGripper"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CalibrateGripperParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CalibratePipetteParams": { + "title": "CalibratePipetteParams", + "description": "Payload required to calibrate-pipette.", + "type": "object", + "properties": { + "mount": { + "description": "Instrument mount to calibrate.", + "allOf": [ + { + "$ref": "#/definitions/MountType" + } + ] + } + }, + "required": ["mount"] + }, + "CalibratePipetteCreate": { + "title": "CalibratePipetteCreate", + "description": "Create calibrate-pipette command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "calibration/calibratePipette", + "enum": ["calibration/calibratePipette"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CalibratePipetteParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "CalibrateModuleParams": { + "title": "CalibrateModuleParams", + "description": "Payload required to calibrate-module.", + "type": "object", + "properties": { + "moduleId": { + "title": "Moduleid", + "description": "The unique id of module to calibrate.", + "type": "string" + }, + "labwareId": { + "title": "Labwareid", + "description": "The unique id of module calibration adapter labware.", + "type": "string" + }, + "mount": { + "description": "The instrument mount used to calibrate the module.", + "allOf": [ + { + "$ref": "#/definitions/MountType" + } + ] + } + }, + "required": ["moduleId", "labwareId", "mount"] + }, + "CalibrateModuleCreate": { + "title": "CalibrateModuleCreate", + "description": "Create calibrate-module command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "calibration/calibrateModule", + "enum": ["calibration/calibrateModule"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/CalibrateModuleParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "MaintenancePosition": { + "title": "MaintenancePosition", + "description": "Maintenance position options.", + "enum": ["attachPlate", "attachInstrument"] + }, + "MoveToMaintenancePositionParams": { + "title": "MoveToMaintenancePositionParams", + "description": "Calibration set up position command parameters.", + "type": "object", + "properties": { + "mount": { + "description": "Gantry mount to move maintenance position.", + "allOf": [ + { + "$ref": "#/definitions/MountType" + } + ] + }, + "maintenancePosition": { + "description": "The position the gantry mount needs to move to.", + "default": "attachInstrument", + "allOf": [ + { + "$ref": "#/definitions/MaintenancePosition" + } + ] + } + }, + "required": ["mount"] + }, + "MoveToMaintenancePositionCreate": { + "title": "MoveToMaintenancePositionCreate", + "description": "Calibration set up position command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "calibration/moveToMaintenancePosition", + "enum": ["calibration/moveToMaintenancePosition"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/MoveToMaintenancePositionParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "UnsafeBlowOutInPlaceParams": { + "title": "UnsafeBlowOutInPlaceParams", + "description": "Payload required to blow-out in place while position is unknown.", + "type": "object", + "properties": { + "flowRate": { + "title": "Flowrate", + "description": "Speed in \u00b5L/s configured for the pipette", + "exclusiveMinimum": 0, + "type": "number" + }, + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + } + }, + "required": ["flowRate", "pipetteId"] + }, + "UnsafeBlowOutInPlaceCreate": { + "title": "UnsafeBlowOutInPlaceCreate", + "description": "UnsafeBlowOutInPlace command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "unsafe/blowOutInPlace", + "enum": ["unsafe/blowOutInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/UnsafeBlowOutInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "UnsafeDropTipInPlaceParams": { + "title": "UnsafeDropTipInPlaceParams", + "description": "Payload required to drop a tip in place even if the plunger position is not known.", + "type": "object", + "properties": { + "pipetteId": { + "title": "Pipetteid", + "description": "Identifier of pipette to use for liquid handling.", + "type": "string" + }, + "homeAfter": { + "title": "Homeafter", + "description": "Whether to home this pipette's plunger after dropping the tip. You should normally leave this unspecified to let the robot choose a safe default depending on its hardware.", + "type": "boolean" + } + }, + "required": ["pipetteId"] + }, + "UnsafeDropTipInPlaceCreate": { + "title": "UnsafeDropTipInPlaceCreate", + "description": "Drop tip in place command creation request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "unsafe/dropTipInPlace", + "enum": ["unsafe/dropTipInPlace"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/UnsafeDropTipInPlaceParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + }, + "UpdatePositionEstimatorsParams": { + "title": "UpdatePositionEstimatorsParams", + "description": "Payload required for an UpdatePositionEstimators command.", + "type": "object", + "properties": { + "axes": { + "description": "The axes for which to update the position estimators.", + "type": "array", + "items": { + "$ref": "#/definitions/MotorAxis" + } + } + }, + "required": ["axes"] + }, + "UpdatePositionEstimatorsCreate": { + "title": "UpdatePositionEstimatorsCreate", + "description": "UpdatePositionEstimators command request model.", + "type": "object", + "properties": { + "commandType": { + "title": "Commandtype", + "default": "unsafe/updatePositionEstimators", + "enum": ["unsafe/updatePositionEstimators"], + "type": "string" + }, + "params": { + "$ref": "#/definitions/UpdatePositionEstimatorsParams" + }, + "intent": { + "description": "The reason the command was added. If not specified or `protocol`, the command will be treated as part of the protocol run itself, and added to the end of the existing command queue.\n\nIf `setup`, the command will be treated as part of run setup. A setup command may only be enqueued if the run has not started.\n\nUse setup commands for activities like pre-run calibration checks and module setup, like pre-heating.", + "allOf": [ + { + "$ref": "#/definitions/CommandIntent" + } + ] + }, + "key": { + "title": "Key", + "description": "A key value, unique in this run, that can be used to track the same logical command across multiple runs of the same protocol. If a value is not provided, one will be generated.", + "type": "string" + } + }, + "required": ["params"] + } + }, + "$id": "opentronsCommandSchemaV9", + "$schema": "http://json-schema.org/draft-07/schema#" +} diff --git a/shared-data/command/types/index.ts b/shared-data/command/types/index.ts index 5d6503761a3..f668c602f35 100644 --- a/shared-data/command/types/index.ts +++ b/shared-data/command/types/index.ts @@ -1,3 +1,4 @@ +import type { ErrorCodes } from '../../errors' import type { PipettingRunTimeCommand, PipettingCreateCommand, @@ -18,6 +19,7 @@ import type { CalibrationRunTimeCommand, CalibrationCreateCommand, } from './calibration' +import type { UnsafeRunTimeCommand, UnsafeCreateCommand } from './unsafe' export * from './annotation' export * from './calibration' @@ -27,6 +29,7 @@ export * from './module' export * from './pipetting' export * from './setup' export * from './timing' +export * from './unsafe' // NOTE: these key/value pairs will only be present on commands at analysis/run time // they pertain only to the actual execution status of a command on hardware, as opposed to @@ -66,6 +69,7 @@ export type CreateCommand = | CalibrationCreateCommand // for automatic pipette calibration | AnnotationCreateCommand // annotating command execution | IncidentalCreateCommand // command with only incidental effects (status bar animations) + | UnsafeCreateCommand // command providing capabilities that are not safe for scientific uses // commands will be required to have a key, but will not be created with one export type RunTimeCommand = @@ -77,15 +81,30 @@ export type RunTimeCommand = | CalibrationRunTimeCommand // for automatic pipette calibration | AnnotationRunTimeCommand // annotating command execution | IncidentalRunTimeCommand // command with only incidental effects (status bar animations) + | UnsafeRunTimeCommand // command providing capabilities that are not safe for scientific uses + +export type RunCommandError = + | RunCommandErrorUndefined + | RunCommandErrorOverpressure // TODO(jh, 05-24-24): Update when some of these newer properties become more finalized. -export interface RunCommandError { +export interface RunCommandErrorBase { createdAt: string detail: string - errorCode: string - errorType: string id: string - isDefined: boolean - errorInfo?: Record wrappedErrors?: RunCommandError[] } + +export interface RunCommandErrorUndefined extends RunCommandErrorBase { + errorCode: ErrorCodes + errorType: string + isDefined: false + errorInfo?: Record +} + +export interface RunCommandErrorOverpressure extends RunCommandErrorBase { + errorCode: '3006' + errorType: 'overpressure' + isDefined: true + errorInfo: { retryLocation: [number, number, number] } +} diff --git a/shared-data/command/types/unsafe.ts b/shared-data/command/types/unsafe.ts new file mode 100644 index 00000000000..8ff4d7e74aa --- /dev/null +++ b/shared-data/command/types/unsafe.ts @@ -0,0 +1,58 @@ +import type { CommonCommandRunTimeInfo, CommonCommandCreateInfo } from '.' +import type { MotorAxes } from '../../js/types' + +export type UnsafeRunTimeCommand = + | UnsafeBlowoutInPlaceRunTimeCommand + | UnsafeDropTipInPlaceRunTimeCommand + | UnsafeUpdatePositionEstimatorsRunTimeCommand + +export type UnsafeCreateCommand = + | UnsafeBlowoutInPlaceCreateCommand + | UnsafeDropTipInPlaceCreateCommand + | UnsafeUpdatePositionEstimatorsCreateCommand + +export interface UnsafeBlowoutInPlaceParams { + pipetteId: string + flowRate: number // µL/s +} + +export interface UnsafeBlowoutInPlaceCreateCommand + extends CommonCommandCreateInfo { + commandType: 'unsafe/blowOutInPlace' + params: UnsafeBlowoutInPlaceParams +} +export interface UnsafeBlowoutInPlaceRunTimeCommand + extends CommonCommandRunTimeInfo, + UnsafeBlowoutInPlaceCreateCommand { + result?: {} +} + +export interface UnsafeDropTipInPlaceParams { + pipetteId: string +} + +export interface UnsafeDropTipInPlaceCreateCommand + extends CommonCommandCreateInfo { + commandType: 'unsafe/dropTipInPlace' + params: UnsafeDropTipInPlaceParams +} +export interface UnsafeDropTipInPlaceRunTimeCommand + extends CommonCommandRunTimeInfo, + UnsafeDropTipInPlaceCreateCommand { + result?: any +} + +export interface UnsafeUpdatePositionEstimatorsParams { + axes: MotorAxes +} + +export interface UnsafeUpdatePositionEstimatorsCreateCommand + extends CommonCommandCreateInfo { + commandType: 'unsafe/updatePositionEstimators' + params: UnsafeUpdatePositionEstimatorsParams +} +export interface UnsafeUpdatePositionEstimatorsRunTimeCommand + extends CommonCommandRunTimeInfo, + UnsafeUpdatePositionEstimatorsCreateCommand { + result?: any +} diff --git a/shared-data/errors/index.ts b/shared-data/errors/index.ts new file mode 100644 index 00000000000..8f12e888d7e --- /dev/null +++ b/shared-data/errors/index.ts @@ -0,0 +1,4 @@ +import errorSchemaV1 from './schemas/1.json' +import errorData from './definitions/1/errors.json' +export type * from './types' +export { errorSchemaV1, errorData } diff --git a/shared-data/errors/types/index.ts b/shared-data/errors/types/index.ts new file mode 100644 index 00000000000..d10699c0672 --- /dev/null +++ b/shared-data/errors/types/index.ts @@ -0,0 +1,15 @@ +// eslint-disable-next-line @typescript-eslint/consistent-type-imports +import ERROR_DATA from '../definitions/1/errors.json' + +export type ErrorCategories = keyof typeof ERROR_DATA['categories'] +export interface CategorySpec { + detail: string + codePrefix: string +} +export type ErrorCodes = keyof typeof ERROR_DATA['codes'] +export interface ErrorSpec { + detail: string + category: ErrorCategories +} +export type ErrorSpecs = Record +export type CategorySpecs = Record diff --git a/shared-data/js/protocols.ts b/shared-data/js/protocols.ts index fdd9425b4f4..e8effe36237 100644 --- a/shared-data/js/protocols.ts +++ b/shared-data/js/protocols.ts @@ -4,6 +4,7 @@ import Ajv from 'ajv' +import commandSchema9 from '../command/schemas/9.json' import commandSchema8 from '../command/schemas/8.json' import commandSchema7 from '../command/schemas/7.json' import commandAnnotationSchema1 from '../commandAnnotation/schemas/1.json' @@ -29,6 +30,9 @@ const validateCommands8 = ( new Promise((resolve, reject) => { const requestedSchema = toValidate.commandSchemaId switch (requestedSchema) { + case 'opentronsCommandSchemaV9': + resolve(commandSchema9) + break case 'opentronsCommandSchemaV8': resolve(commandSchema8) break diff --git a/shared-data/protocol/README.md b/shared-data/protocol/README.md index 7df080a73c2..92c9c27dc84 100644 --- a/shared-data/protocol/README.md +++ b/shared-data/protocol/README.md @@ -2,7 +2,7 @@ 1. Create new JSON schema in `shared-data/protocol/schemas/${schemaVersion}.json` 2. Create TS types for new schema in `shared-data/protocol/types` -3. Create or modify Python types as necessary in both `python/opentrons_shared_data/protocol/dev_types` and in the python api `dev_types` in `api/src/opentrons/protocol_api/dev_types.py` +3. Create or modify Python types as necessary in both `python/opentrons_shared_data/protocol/types` and in the python api `types` in `api/src/opentrons/protocol_api/types.py` 4. Create new executor in api to handle new schema. Be sure to include unit tests 5. Add new test in `api/tests/opentrons/test_execute.py` for the schema 6. Add new test in `shared-data/js/__tests__/protocolSchemaV${schemaVersion}.test.js` to ensure new schema definition functions as intended diff --git a/shared-data/protocol/fixtures/8/simpleFlexV9CommandsV8.json b/shared-data/protocol/fixtures/8/simpleFlexV9CommandsV8.json new file mode 100644 index 00000000000..d350d8cd288 --- /dev/null +++ b/shared-data/protocol/fixtures/8/simpleFlexV9CommandsV8.json @@ -0,0 +1,1475 @@ +{ + "$otSharedSchema": "#/protocol/schemas/8", + "schemaVersion": 8, + "metadata": { + "protocolName": "Simple test protocol", + "author": "engineering ", + "description": "A short test protocol", + "created": 1223131231, + "tags": ["unitTest"] + }, + "robot": { + "model": "OT-3 Standard", + "deckId": "ot3_standard" + }, + "liquidSchemaId": "opentronsLiquidSchemaV1", + "liquids": { + "waterId": { + "displayName": "Water", + "description": "Liquid H2O", + "displayColor": "#7332a8" + } + }, + "labwareDefinitionSchemaId": "opentronsLabwareSchemaV2", + "labwareDefinitions": { + "opentrons/opentrons_1_trash_1100ml_fixed/1": { + "ordering": [["A1"]], + "metadata": { + "displayCategory": "trash", + "displayVolumeUnits": "mL", + "displayName": "Opentrons Fixed Trash", + "tags": [] + }, + "schemaVersion": 2, + "version": 1, + "namespace": "opentrons", + "dimensions": { + "xDimension": 172.86, + "yDimension": 165.86, + "zDimension": 82 + }, + "parameters": { + "format": "trash", + "isTiprack": false, + "loadName": "opentrons_1_trash_1100ml_fixed", + "isMagneticModuleCompatible": false, + "quirks": ["fixedTrash", "centerMultichannelOnWells"] + }, + "wells": { + "A1": { + "shape": "rectangular", + "yDimension": 165.67, + "xDimension": 107.11, + "totalLiquidVolume": 1100000, + "depth": 77, + "x": 82.84, + "y": 53.56, + "z": 5 + } + }, + "brand": { + "brand": "Opentrons" + }, + "groups": [ + { + "wells": ["A1"], + "metadata": {} + } + ], + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + }, + "opentrons/opentrons_96_tiprack_10ul/1": { + "ordering": [ + ["A1", "B1", "C1", "D1", "E1", "F1", "G1", "H1"], + ["A2", "B2", "C2", "D2", "E2", "F2", "G2", "H2"], + ["A3", "B3", "C3", "D3", "E3", "F3", "G3", "H3"], + ["A4", "B4", "C4", "D4", "E4", "F4", "G4", "H4"], + ["A5", "B5", "C5", "D5", "E5", "F5", "G5", "H5"], + ["A6", "B6", "C6", "D6", "E6", "F6", "G6", "H6"], + ["A7", "B7", "C7", "D7", "E7", "F7", "G7", "H7"], + ["A8", "B8", "C8", "D8", "E8", "F8", "G8", "H8"], + ["A9", "B9", "C9", "D9", "E9", "F9", "G9", "H9"], + ["A10", "B10", "C10", "D10", "E10", "F10", "G10", "H10"], + ["A11", "B11", "C11", "D11", "E11", "F11", "G11", "H11"], + ["A12", "B12", "C12", "D12", "E12", "F12", "G12", "H12"] + ], + "brand": { + "brand": "Opentrons", + "brandId": [], + "links": [ + "https://shop.opentrons.com/collections/opentrons-tips/products/opentrons-10ul-tips" + ] + }, + "metadata": { + "displayName": "Opentrons 96 Tip Rack 10 µL", + "displayCategory": "tipRack", + "displayVolumeUnits": "µL", + "tags": [] + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 64.69 + }, + "wells": { + "A1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 74.24, + "z": 25.49 + }, + "B1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 65.24, + "z": 25.49 + }, + "C1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 56.24, + "z": 25.49 + }, + "D1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 47.24, + "z": 25.49 + }, + "E1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 38.24, + "z": 25.49 + }, + "F1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 29.24, + "z": 25.49 + }, + "G1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 20.24, + "z": 25.49 + }, + "H1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 11.24, + "z": 25.49 + }, + "A2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 74.24, + "z": 25.49 + }, + "B2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 65.24, + "z": 25.49 + }, + "C2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 56.24, + "z": 25.49 + }, + "D2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 47.24, + "z": 25.49 + }, + "E2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 38.24, + "z": 25.49 + }, + "F2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 29.24, + "z": 25.49 + }, + "G2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 20.24, + "z": 25.49 + }, + "H2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 11.24, + "z": 25.49 + }, + "A3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 74.24, + "z": 25.49 + }, + "B3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 65.24, + "z": 25.49 + }, + "C3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 56.24, + "z": 25.49 + }, + "D3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 47.24, + "z": 25.49 + }, + "E3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 38.24, + "z": 25.49 + }, + "F3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 29.24, + "z": 25.49 + }, + "G3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 20.24, + "z": 25.49 + }, + "H3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 11.24, + "z": 25.49 + }, + "A4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 74.24, + "z": 25.49 + }, + "B4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 65.24, + "z": 25.49 + }, + "C4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 56.24, + "z": 25.49 + }, + "D4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 47.24, + "z": 25.49 + }, + "E4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 38.24, + "z": 25.49 + }, + "F4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 29.24, + "z": 25.49 + }, + "G4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 20.24, + "z": 25.49 + }, + "H4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 11.24, + "z": 25.49 + }, + "A5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 74.24, + "z": 25.49 + }, + "B5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 65.24, + "z": 25.49 + }, + "C5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 56.24, + "z": 25.49 + }, + "D5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 47.24, + "z": 25.49 + }, + "E5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 38.24, + "z": 25.49 + }, + "F5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 29.24, + "z": 25.49 + }, + "G5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 20.24, + "z": 25.49 + }, + "H5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 11.24, + "z": 25.49 + }, + "A6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 74.24, + "z": 25.49 + }, + "B6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 65.24, + "z": 25.49 + }, + "C6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 56.24, + "z": 25.49 + }, + "D6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 47.24, + "z": 25.49 + }, + "E6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 38.24, + "z": 25.49 + }, + "F6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 29.24, + "z": 25.49 + }, + "G6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 20.24, + "z": 25.49 + }, + "H6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 11.24, + "z": 25.49 + }, + "A7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 74.24, + "z": 25.49 + }, + "B7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 65.24, + "z": 25.49 + }, + "C7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 56.24, + "z": 25.49 + }, + "D7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 47.24, + "z": 25.49 + }, + "E7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 38.24, + "z": 25.49 + }, + "F7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 29.24, + "z": 25.49 + }, + "G7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 20.24, + "z": 25.49 + }, + "H7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 11.24, + "z": 25.49 + }, + "A8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 74.24, + "z": 25.49 + }, + "B8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 65.24, + "z": 25.49 + }, + "C8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 56.24, + "z": 25.49 + }, + "D8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 47.24, + "z": 25.49 + }, + "E8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 38.24, + "z": 25.49 + }, + "F8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 29.24, + "z": 25.49 + }, + "G8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 20.24, + "z": 25.49 + }, + "H8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 11.24, + "z": 25.49 + }, + "A9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 74.24, + "z": 25.49 + }, + "B9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 65.24, + "z": 25.49 + }, + "C9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 56.24, + "z": 25.49 + }, + "D9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 47.24, + "z": 25.49 + }, + "E9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 38.24, + "z": 25.49 + }, + "F9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 29.24, + "z": 25.49 + }, + "G9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 20.24, + "z": 25.49 + }, + "H9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 11.24, + "z": 25.49 + }, + "A10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 74.24, + "z": 25.49 + }, + "B10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 65.24, + "z": 25.49 + }, + "C10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 56.24, + "z": 25.49 + }, + "D10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 47.24, + "z": 25.49 + }, + "E10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 38.24, + "z": 25.49 + }, + "F10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 29.24, + "z": 25.49 + }, + "G10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 20.24, + "z": 25.49 + }, + "H10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 11.24, + "z": 25.49 + }, + "A11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 74.24, + "z": 25.49 + }, + "B11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 65.24, + "z": 25.49 + }, + "C11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 56.24, + "z": 25.49 + }, + "D11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 47.24, + "z": 25.49 + }, + "E11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 38.24, + "z": 25.49 + }, + "F11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 29.24, + "z": 25.49 + }, + "G11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 20.24, + "z": 25.49 + }, + "H11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 11.24, + "z": 25.49 + }, + "A12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 74.24, + "z": 25.49 + }, + "B12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 65.24, + "z": 25.49 + }, + "C12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 56.24, + "z": 25.49 + }, + "D12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 47.24, + "z": 25.49 + }, + "E12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 38.24, + "z": 25.49 + }, + "F12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 29.24, + "z": 25.49 + }, + "G12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 20.24, + "z": 25.49 + }, + "H12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 11.24, + "z": 25.49 + } + }, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1", + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2", + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3", + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4", + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5", + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6", + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7", + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8", + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9", + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10", + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11", + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ] + } + ], + "parameters": { + "format": "96Standard", + "isTiprack": true, + "tipLength": 39.2, + "tipOverlap": 3.29, + "isMagneticModuleCompatible": false, + "loadName": "opentrons_96_tiprack_10ul" + }, + "namespace": "opentrons", + "version": 1, + "schemaVersion": 2, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + }, + "example/plate/1": { + "ordering": [ + ["A1", "B1", "C1", "D1"], + ["A2", "B2", "C2", "D2"] + ], + "brand": { + "brand": "foo", + "brandId": [] + }, + "metadata": { + "displayName": "Foo 8 Well Plate 33uL", + "displayCategory": "wellPlate", + "displayVolumeUnits": "µL" + }, + "dimensions": { + "xDimension": 127.75, + "yDimension": 85.5, + "zDimension": 100 + }, + "wells": { + "A1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 75.43, + "z": 75 + }, + "B1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 56.15, + "z": 75 + }, + "C1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 36.87, + "z": 75 + }, + "D1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 17.59, + "z": 75 + }, + "A2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 75.43, + "z": 75 + }, + "B2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 56.15, + "z": 75 + }, + "C2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 36.87, + "z": 75 + }, + "D2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 17.59, + "z": 75 + } + }, + "groups": [ + { + "metadata": {}, + "wells": ["A1", "B1", "C1", "A2", "B2", "C2"] + } + ], + "parameters": { + "format": "irregular", + "quirks": [], + "isTiprack": false, + "isMagneticModuleCompatible": false, + "loadName": "foo_8_plate_33ul" + }, + "namespace": "example", + "version": 1, + "schemaVersion": 2, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "commandSchemaId": "opentronsCommandSchemaV9", + "commands": [ + { + "commandType": "loadPipette", + "params": { + "pipetteId": "pipetteId", + "pipetteName": "p1000_96", + "mount": "left" + } + }, + { + "commandType": "loadModule", + "params": { + "moduleId": "magneticBlockId", + "model": "magneticBlockV1", + "location": { "slotName": "3" } + } + }, + { + "commandType": "loadModule", + "params": { + "moduleId": "temperatureModuleId", + "model": "temperatureModuleV2", + "location": { "slotName": "1" } + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "sourcePlateId", + "loadName": "armadillo_96_wellplate_200ul_pcr_full_skirt", + "namespace": "opentrons", + "version": 1, + "location": { + "moduleId": "temperatureModuleId" + }, + "displayName": "Source Plate" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "destPlateId", + "loadName": "armadillo_96_wellplate_200ul_pcr_full_skirt", + "namespace": "opentrons", + "version": 1, + "location": { + "moduleId": "magneticBlockId" + }, + "displayName": "Sample Collection Plate" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "tipRackId", + "location": { "slotName": "8" }, + "loadName": "opentrons_96_tiprack_1000ul", + "namespace": "opentrons", + "version": 1, + "displayName": "Opentrons 96 Tip Rack 1000 µL" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "fixedTrash", + "location": { + "slotName": "12" + }, + "loadName": "opentrons_1_trash_1100ml_fixed", + "namespace": "opentrons", + "version": 1, + "displayName": "Trash" + } + }, + { + "commandType": "loadLiquid", + "params": { + "liquidId": "waterId", + "labwareId": "sourcePlateId", + "volumeByWell": { + "A1": 100, + "B1": 100 + } + } + }, + { + "commandType": "home", + "params": {} + }, + { + "commandType": "pickUpTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "tipRackId", + "wellName": "B1" + } + }, + { + "commandType": "aspirate", + "params": { + "pipetteId": "pipetteId", + "labwareId": "sourcePlateId", + "wellName": "A1", + "volume": 5, + "flowRate": 3, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 2 } + } + } + }, + { + "commandType": "waitForDuration", + "params": { + "seconds": 42 + } + }, + { + "commandType": "dispense", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "volume": 4.5, + "flowRate": 2.5, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 1 } + } + } + }, + { + "commandType": "touchTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "speed": 42.0, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 11 } + } + } + }, + { + "commandType": "blowout", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "flowRate": 2, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 12 } + } + } + }, + { + "commandType": "moveToCoordinates", + "params": { + "pipetteId": "pipetteId", + "coordinates": { "x": 100, "y": 100, "z": 100 } + } + }, + { + "commandType": "moveToWell", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B2", + "speed": 12.3 + } + }, + { + "commandType": "moveToWell", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B2", + "wellLocation": { + "origin": "bottom", + "offset": { "x": 2, "y": 3, "z": 10 } + }, + "minimumZHeight": 35, + "forceDirect": true + } + }, + { + "commandType": "dropTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "fixedTrash", + "wellName": "A1" + } + }, + { + "commandType": "waitForResume", + "params": { + "message": "pause command" + } + }, + { + "commandType": "moveToCoordinates", + "params": { + "pipetteId": "pipetteId", + "coordinates": { "x": 0, "y": 0, "z": 0 }, + "minimumZHeight": 35, + "forceDirect": true + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "x", + "distance": 1 + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "y", + "distance": 0.1 + } + }, + { + "commandType": "savePosition", + "params": { + "pipetteId": "pipetteId" + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "z", + "distance": 10 + } + }, + { + "commandType": "savePosition", + "params": { + "pipetteId": "pipetteId", + "positionId": "positionId" + } + } + ], + "commandAnnotationSchemaId": "opentronsCommandAnnotationSchemaV1", + "commandAnnotations": [ + { + "commandKeys": [ + "1abc123", + "2abc123", + "3abc123", + "4abc123", + "5abc123", + "6abc123", + "7abc123" + ], + "annotationType": "custom" + } + ] +} diff --git a/shared-data/protocol/fixtures/8/simpleV9CommandsV8.json b/shared-data/protocol/fixtures/8/simpleV9CommandsV8.json new file mode 100644 index 00000000000..ff500d341f4 --- /dev/null +++ b/shared-data/protocol/fixtures/8/simpleV9CommandsV8.json @@ -0,0 +1,1473 @@ +{ + "$otSharedSchema": "#/protocol/schemas/8", + "schemaVersion": 8, + "metadata": { + "protocolName": "Simple test protocol", + "author": "engineering ", + "description": "A short test protocol", + "created": 1223131231, + "tags": ["unitTest"] + }, + "robot": { + "model": "OT-2 Standard", + "deckId": "ot2_standard" + }, + "liquidSchemaId": "opentronsLiquidSchemaV1", + "liquids": { + "waterId": { + "displayName": "Water", + "description": "Liquid H2O", + "displayColor": "#7332a8" + } + }, + "labwareDefinitionSchemaId": "opentronsLabwareSchemaV2", + "labwareDefinitions": { + "opentrons/opentrons_1_trash_1100ml_fixed/1": { + "ordering": [["A1"]], + "metadata": { + "displayCategory": "trash", + "displayVolumeUnits": "mL", + "displayName": "Opentrons Fixed Trash", + "tags": [] + }, + "schemaVersion": 2, + "version": 1, + "namespace": "opentrons", + "dimensions": { + "xDimension": 172.86, + "yDimension": 165.86, + "zDimension": 82 + }, + "parameters": { + "format": "trash", + "isTiprack": false, + "loadName": "opentrons_1_trash_1100ml_fixed", + "isMagneticModuleCompatible": false, + "quirks": ["fixedTrash", "centerMultichannelOnWells"] + }, + "wells": { + "A1": { + "shape": "rectangular", + "yDimension": 165.67, + "xDimension": 107.11, + "totalLiquidVolume": 1100000, + "depth": 77, + "x": 82.84, + "y": 53.56, + "z": 5 + } + }, + "brand": { + "brand": "Opentrons" + }, + "groups": [ + { + "wells": ["A1"], + "metadata": {} + } + ], + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + }, + "opentrons/opentrons_96_tiprack_10ul/1": { + "ordering": [ + ["A1", "B1", "C1", "D1", "E1", "F1", "G1", "H1"], + ["A2", "B2", "C2", "D2", "E2", "F2", "G2", "H2"], + ["A3", "B3", "C3", "D3", "E3", "F3", "G3", "H3"], + ["A4", "B4", "C4", "D4", "E4", "F4", "G4", "H4"], + ["A5", "B5", "C5", "D5", "E5", "F5", "G5", "H5"], + ["A6", "B6", "C6", "D6", "E6", "F6", "G6", "H6"], + ["A7", "B7", "C7", "D7", "E7", "F7", "G7", "H7"], + ["A8", "B8", "C8", "D8", "E8", "F8", "G8", "H8"], + ["A9", "B9", "C9", "D9", "E9", "F9", "G9", "H9"], + ["A10", "B10", "C10", "D10", "E10", "F10", "G10", "H10"], + ["A11", "B11", "C11", "D11", "E11", "F11", "G11", "H11"], + ["A12", "B12", "C12", "D12", "E12", "F12", "G12", "H12"] + ], + "brand": { + "brand": "Opentrons", + "brandId": [], + "links": [ + "https://shop.opentrons.com/collections/opentrons-tips/products/opentrons-10ul-tips" + ] + }, + "metadata": { + "displayName": "Opentrons 96 Tip Rack 10 µL", + "displayCategory": "tipRack", + "displayVolumeUnits": "µL", + "tags": [] + }, + "dimensions": { + "xDimension": 127.76, + "yDimension": 85.48, + "zDimension": 64.69 + }, + "wells": { + "A1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 74.24, + "z": 25.49 + }, + "B1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 65.24, + "z": 25.49 + }, + "C1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 56.24, + "z": 25.49 + }, + "D1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 47.24, + "z": 25.49 + }, + "E1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 38.24, + "z": 25.49 + }, + "F1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 29.24, + "z": 25.49 + }, + "G1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 20.24, + "z": 25.49 + }, + "H1": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 14.38, + "y": 11.24, + "z": 25.49 + }, + "A2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 74.24, + "z": 25.49 + }, + "B2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 65.24, + "z": 25.49 + }, + "C2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 56.24, + "z": 25.49 + }, + "D2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 47.24, + "z": 25.49 + }, + "E2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 38.24, + "z": 25.49 + }, + "F2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 29.24, + "z": 25.49 + }, + "G2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 20.24, + "z": 25.49 + }, + "H2": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 23.38, + "y": 11.24, + "z": 25.49 + }, + "A3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 74.24, + "z": 25.49 + }, + "B3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 65.24, + "z": 25.49 + }, + "C3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 56.24, + "z": 25.49 + }, + "D3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 47.24, + "z": 25.49 + }, + "E3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 38.24, + "z": 25.49 + }, + "F3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 29.24, + "z": 25.49 + }, + "G3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 20.24, + "z": 25.49 + }, + "H3": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 32.38, + "y": 11.24, + "z": 25.49 + }, + "A4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 74.24, + "z": 25.49 + }, + "B4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 65.24, + "z": 25.49 + }, + "C4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 56.24, + "z": 25.49 + }, + "D4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 47.24, + "z": 25.49 + }, + "E4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 38.24, + "z": 25.49 + }, + "F4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 29.24, + "z": 25.49 + }, + "G4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 20.24, + "z": 25.49 + }, + "H4": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 41.38, + "y": 11.24, + "z": 25.49 + }, + "A5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 74.24, + "z": 25.49 + }, + "B5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 65.24, + "z": 25.49 + }, + "C5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 56.24, + "z": 25.49 + }, + "D5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 47.24, + "z": 25.49 + }, + "E5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 38.24, + "z": 25.49 + }, + "F5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 29.24, + "z": 25.49 + }, + "G5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 20.24, + "z": 25.49 + }, + "H5": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 50.38, + "y": 11.24, + "z": 25.49 + }, + "A6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 74.24, + "z": 25.49 + }, + "B6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 65.24, + "z": 25.49 + }, + "C6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 56.24, + "z": 25.49 + }, + "D6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 47.24, + "z": 25.49 + }, + "E6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 38.24, + "z": 25.49 + }, + "F6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 29.24, + "z": 25.49 + }, + "G6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 20.24, + "z": 25.49 + }, + "H6": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 59.38, + "y": 11.24, + "z": 25.49 + }, + "A7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 74.24, + "z": 25.49 + }, + "B7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 65.24, + "z": 25.49 + }, + "C7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 56.24, + "z": 25.49 + }, + "D7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 47.24, + "z": 25.49 + }, + "E7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 38.24, + "z": 25.49 + }, + "F7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 29.24, + "z": 25.49 + }, + "G7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 20.24, + "z": 25.49 + }, + "H7": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 68.38, + "y": 11.24, + "z": 25.49 + }, + "A8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 74.24, + "z": 25.49 + }, + "B8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 65.24, + "z": 25.49 + }, + "C8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 56.24, + "z": 25.49 + }, + "D8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 47.24, + "z": 25.49 + }, + "E8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 38.24, + "z": 25.49 + }, + "F8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 29.24, + "z": 25.49 + }, + "G8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 20.24, + "z": 25.49 + }, + "H8": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 77.38, + "y": 11.24, + "z": 25.49 + }, + "A9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 74.24, + "z": 25.49 + }, + "B9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 65.24, + "z": 25.49 + }, + "C9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 56.24, + "z": 25.49 + }, + "D9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 47.24, + "z": 25.49 + }, + "E9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 38.24, + "z": 25.49 + }, + "F9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 29.24, + "z": 25.49 + }, + "G9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 20.24, + "z": 25.49 + }, + "H9": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 86.38, + "y": 11.24, + "z": 25.49 + }, + "A10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 74.24, + "z": 25.49 + }, + "B10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 65.24, + "z": 25.49 + }, + "C10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 56.24, + "z": 25.49 + }, + "D10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 47.24, + "z": 25.49 + }, + "E10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 38.24, + "z": 25.49 + }, + "F10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 29.24, + "z": 25.49 + }, + "G10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 20.24, + "z": 25.49 + }, + "H10": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 95.38, + "y": 11.24, + "z": 25.49 + }, + "A11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 74.24, + "z": 25.49 + }, + "B11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 65.24, + "z": 25.49 + }, + "C11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 56.24, + "z": 25.49 + }, + "D11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 47.24, + "z": 25.49 + }, + "E11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 38.24, + "z": 25.49 + }, + "F11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 29.24, + "z": 25.49 + }, + "G11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 20.24, + "z": 25.49 + }, + "H11": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 104.38, + "y": 11.24, + "z": 25.49 + }, + "A12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 74.24, + "z": 25.49 + }, + "B12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 65.24, + "z": 25.49 + }, + "C12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 56.24, + "z": 25.49 + }, + "D12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 47.24, + "z": 25.49 + }, + "E12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 38.24, + "z": 25.49 + }, + "F12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 29.24, + "z": 25.49 + }, + "G12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 20.24, + "z": 25.49 + }, + "H12": { + "depth": 39.2, + "shape": "circular", + "diameter": 3.27, + "totalLiquidVolume": 10, + "x": 113.38, + "y": 11.24, + "z": 25.49 + } + }, + "groups": [ + { + "metadata": {}, + "wells": [ + "A1", + "B1", + "C1", + "D1", + "E1", + "F1", + "G1", + "H1", + "A2", + "B2", + "C2", + "D2", + "E2", + "F2", + "G2", + "H2", + "A3", + "B3", + "C3", + "D3", + "E3", + "F3", + "G3", + "H3", + "A4", + "B4", + "C4", + "D4", + "E4", + "F4", + "G4", + "H4", + "A5", + "B5", + "C5", + "D5", + "E5", + "F5", + "G5", + "H5", + "A6", + "B6", + "C6", + "D6", + "E6", + "F6", + "G6", + "H6", + "A7", + "B7", + "C7", + "D7", + "E7", + "F7", + "G7", + "H7", + "A8", + "B8", + "C8", + "D8", + "E8", + "F8", + "G8", + "H8", + "A9", + "B9", + "C9", + "D9", + "E9", + "F9", + "G9", + "H9", + "A10", + "B10", + "C10", + "D10", + "E10", + "F10", + "G10", + "H10", + "A11", + "B11", + "C11", + "D11", + "E11", + "F11", + "G11", + "H11", + "A12", + "B12", + "C12", + "D12", + "E12", + "F12", + "G12", + "H12" + ] + } + ], + "parameters": { + "format": "96Standard", + "isTiprack": true, + "tipLength": 39.2, + "tipOverlap": 3.29, + "isMagneticModuleCompatible": false, + "loadName": "opentrons_96_tiprack_10ul" + }, + "namespace": "opentrons", + "version": 1, + "schemaVersion": 2, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + }, + "example/plate/1": { + "ordering": [ + ["A1", "B1", "C1", "D1"], + ["A2", "B2", "C2", "D2"] + ], + "brand": { + "brand": "foo", + "brandId": [] + }, + "metadata": { + "displayName": "Foo 8 Well Plate 33uL", + "displayCategory": "wellPlate", + "displayVolumeUnits": "µL" + }, + "dimensions": { + "xDimension": 127.75, + "yDimension": 85.5, + "zDimension": 100 + }, + "wells": { + "A1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 75.43, + "z": 75 + }, + "B1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 56.15, + "z": 75 + }, + "C1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 36.87, + "z": 75 + }, + "D1": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 18.21, + "y": 17.59, + "z": 75 + }, + "A2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 75.43, + "z": 75 + }, + "B2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 56.15, + "z": 75 + }, + "C2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 36.87, + "z": 75 + }, + "D2": { + "depth": 25, + "totalLiquidVolume": 33, + "shape": "circular", + "diameter": 10, + "x": 38.1, + "y": 17.59, + "z": 75 + } + }, + "groups": [ + { + "metadata": {}, + "wells": ["A1", "B1", "C1", "A2", "B2", "C2"] + } + ], + "parameters": { + "format": "irregular", + "quirks": [], + "isTiprack": false, + "isMagneticModuleCompatible": false, + "loadName": "foo_8_plate_33ul" + }, + "namespace": "example", + "version": 1, + "schemaVersion": 2, + "cornerOffsetFromSlot": { + "x": 0, + "y": 0, + "z": 0 + } + } + }, + "commandSchemaId": "opentronsCommandSchemaV9", + "commands": [ + { + "commandType": "loadPipette", + "params": { + "pipetteId": "pipetteId", + "pipetteName": "p10_single", + "mount": "left" + } + }, + { + "commandType": "loadModule", + "params": { + "moduleId": "magneticModuleId", + "model": "magneticModuleV2", + "location": { "slotName": "3" } + } + }, + { + "commandType": "loadModule", + "params": { + "moduleId": "temperatureModuleId", + "model": "temperatureModuleV2", + "location": { "slotName": "1" } + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "sourcePlateId", + "loadName": "armadillo_96_wellplate_200ul_pcr_full_skirt", + "namespace": "opentrons", + "version": 1, + "location": { + "moduleId": "temperatureModuleId" + }, + "displayName": "Source Plate" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "destPlateId", + "loadName": "armadillo_96_wellplate_200ul_pcr_full_skirt", + "namespace": "opentrons", + "version": 1, + "location": { + "moduleId": "magneticModuleId" + }, + "displayName": "Sample Collection Plate" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "tipRackId", + "location": { "slotName": "8" }, + "loadName": "opentrons_96_tiprack_10ul", + "namespace": "opentrons", + "version": 1, + "displayName": "Opentrons 96 Tip Rack 10 µL" + } + }, + { + "commandType": "loadLabware", + "params": { + "labwareId": "fixedTrash", + "location": { + "slotName": "12" + }, + "loadName": "opentrons_1_trash_1100ml_fixed", + "namespace": "opentrons", + "version": 1, + "displayName": "Trash" + } + }, + { + "commandType": "loadLiquid", + "params": { + "liquidId": "waterId", + "labwareId": "sourcePlateId", + "volumeByWell": { + "A1": 100, + "B1": 100 + } + } + }, + { + "commandType": "home", + "params": {} + }, + { + "commandType": "pickUpTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "tipRackId", + "wellName": "B1" + } + }, + { + "commandType": "aspirate", + "params": { + "pipetteId": "pipetteId", + "labwareId": "sourcePlateId", + "wellName": "A1", + "volume": 5, + "flowRate": 3, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 2 } + } + } + }, + { + "commandType": "waitForDuration", + "params": { + "seconds": 42 + } + }, + { + "commandType": "dispense", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "volume": 4.5, + "flowRate": 2.5, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 1 } + } + } + }, + { + "commandType": "touchTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 11 } + } + } + }, + { + "commandType": "blowout", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B1", + "flowRate": 2, + "wellLocation": { + "origin": "bottom", + "offset": { "x": 0, "y": 0, "z": 12 } + } + } + }, + { + "commandType": "moveToCoordinates", + "params": { + "pipetteId": "pipetteId", + "coordinates": { "x": 100, "y": 100, "z": 100 } + } + }, + { + "commandType": "moveToWell", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B2" + } + }, + { + "commandType": "moveToWell", + "params": { + "pipetteId": "pipetteId", + "labwareId": "destPlateId", + "wellName": "B2", + "wellLocation": { + "origin": "bottom", + "offset": { "x": 2, "y": 3, "z": 10 } + }, + "minimumZHeight": 35, + "forceDirect": true + } + }, + { + "commandType": "dropTip", + "params": { + "pipetteId": "pipetteId", + "labwareId": "fixedTrash", + "wellName": "A1" + } + }, + { + "commandType": "waitForResume", + "params": { + "message": "pause command" + } + }, + { + "commandType": "moveToCoordinates", + "params": { + "pipetteId": "pipetteId", + "coordinates": { "x": 0, "y": 0, "z": 0 }, + "minimumZHeight": 35, + "forceDirect": true + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "x", + "distance": 1 + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "y", + "distance": 0.1 + } + }, + { + "commandType": "savePosition", + "params": { + "pipetteId": "pipetteId" + } + }, + { + "commandType": "moveRelative", + "params": { + "pipetteId": "pipetteId", + "axis": "z", + "distance": 10 + } + }, + { + "commandType": "savePosition", + "params": { + "pipetteId": "pipetteId", + "positionId": "positionId" + } + } + ], + "commandAnnotationSchemaId": "opentronsCommandAnnotationSchemaV1", + "commandAnnotations": [ + { + "commandKeys": [ + "1abc123", + "2abc123", + "3abc123", + "4abc123", + "5abc123", + "6abc123", + "7abc123" + ], + "annotationType": "custom" + } + ] +} diff --git a/shared-data/protocol/types/schemaV8/index.ts b/shared-data/protocol/types/schemaV8/index.ts index c8df6ae95a9..7755837b14f 100644 --- a/shared-data/protocol/types/schemaV8/index.ts +++ b/shared-data/protocol/types/schemaV8/index.ts @@ -22,6 +22,11 @@ export interface CommandV8Mixin { commands: CreateCommand[] } +export interface CommandV9Mixin { + commandSchemaId: 'opentronsCommandSchemaV9' + commands: CreateCommand[] +} + export interface CommandAnnotationsStructure { commandAnnotationSchemaId: string commandAnnotations: any[] @@ -110,7 +115,7 @@ export type ProtocolFile< (OT2RobotMixin | OT3RobotMixin) & LabwareV2Mixin & LiquidV1Mixin & - CommandV8Mixin & + (CommandV8Mixin | CommandV9Mixin) & CommandAnnotationV1Mixin export type ProtocolStructure = ProtocolBase<{}> & diff --git a/shared-data/python/opentrons_shared_data/deck/__init__.py b/shared-data/python/opentrons_shared_data/deck/__init__.py index 24d56ad730e..38607263418 100644 --- a/shared-data/python/opentrons_shared_data/deck/__init__.py +++ b/shared-data/python/opentrons_shared_data/deck/__init__.py @@ -8,7 +8,7 @@ from .. import get_shared_data_root, load_shared_data if TYPE_CHECKING: - from .dev_types import ( + from .types import ( DeckSchema, DeckDefinition, DeckDefinitionV3, diff --git a/shared-data/python/opentrons_shared_data/deck/dev_types.py b/shared-data/python/opentrons_shared_data/deck/types.py similarity index 97% rename from shared-data/python/opentrons_shared_data/deck/dev_types.py rename to shared-data/python/opentrons_shared_data/deck/types.py index 4563ff10953..4e905c99529 100644 --- a/shared-data/python/opentrons_shared_data/deck/dev_types.py +++ b/shared-data/python/opentrons_shared_data/deck/types.py @@ -1,5 +1,5 @@ """ -opentrons_shared_data.deck.dev_types: types for deck defs +opentrons_shared_data.deck.types: types for deck defs This should only be imported if typing.TYPE_CHECKING is True """ @@ -7,7 +7,7 @@ from typing import Any, Dict, List, NewType, Union from typing_extensions import Literal, TypedDict -from ..module.dev_types import ModuleType +from ..module.types import ModuleType DeckSchemaVersion5 = Literal[5] diff --git a/shared-data/python/opentrons_shared_data/labware/__init__.py b/shared-data/python/opentrons_shared_data/labware/__init__.py index 2f881ee18c9..8ffd7cbdf55 100644 --- a/shared-data/python/opentrons_shared_data/labware/__init__.py +++ b/shared-data/python/opentrons_shared_data/labware/__init__.py @@ -7,7 +7,7 @@ from .. import load_shared_data if TYPE_CHECKING: - from .dev_types import LabwareDefinition + from .types import LabwareDefinition Schema = NewType("Schema", Dict[str, Any]) diff --git a/shared-data/python/opentrons_shared_data/labware/dev_types.py b/shared-data/python/opentrons_shared_data/labware/types.py similarity index 97% rename from shared-data/python/opentrons_shared_data/labware/dev_types.py rename to shared-data/python/opentrons_shared_data/labware/types.py index 75714a363d0..a938f337c0f 100644 --- a/shared-data/python/opentrons_shared_data/labware/dev_types.py +++ b/shared-data/python/opentrons_shared_data/labware/types.py @@ -1,4 +1,4 @@ -""" opentrons_shared_data.labware.dev_types: types for labware defs +""" opentrons_shared_data.labware.types: types for labware defs types in this file by and large require the use of typing_extensions. this module shouldn't be imported unless typing.TYPE_CHECKING is true. diff --git a/shared-data/python/opentrons_shared_data/module/__init__.py b/shared-data/python/opentrons_shared_data/module/__init__.py index 7f3dd0a602f..bb3f0d6072c 100644 --- a/shared-data/python/opentrons_shared_data/module/__init__.py +++ b/shared-data/python/opentrons_shared_data/module/__init__.py @@ -4,7 +4,7 @@ from typing import Union, cast, overload from ..load import load_shared_data -from .dev_types import ( +from .types import ( SchemaVersions, ModuleSchema, SchemaV1, diff --git a/shared-data/python/opentrons_shared_data/module/dev_types.py b/shared-data/python/opentrons_shared_data/module/types.py similarity index 98% rename from shared-data/python/opentrons_shared_data/module/dev_types.py rename to shared-data/python/opentrons_shared_data/module/types.py index 827905d8a31..ab9465b04f6 100644 --- a/shared-data/python/opentrons_shared_data/module/dev_types.py +++ b/shared-data/python/opentrons_shared_data/module/types.py @@ -1,5 +1,5 @@ """ -opentrons_shared_data.module.dev_types: types requiring typing_extensions +opentrons_shared_data.module.types: types requiring typing_extensions for modules """ diff --git a/shared-data/python/opentrons_shared_data/pipette/__init__.py b/shared-data/python/opentrons_shared_data/pipette/__init__.py index 3d3c392b677..1f3fce2b6d5 100644 --- a/shared-data/python/opentrons_shared_data/pipette/__init__.py +++ b/shared-data/python/opentrons_shared_data/pipette/__init__.py @@ -11,7 +11,7 @@ from .. import load_shared_data if TYPE_CHECKING: - from .dev_types import ( + from .types import ( PipetteNameSpecs, PipetteModelSpecs, PipetteName, diff --git a/shared-data/python/opentrons_shared_data/pipette/dev_types.py b/shared-data/python/opentrons_shared_data/pipette/dev_types.py index 6497ab3b784..b633f8c4315 100644 --- a/shared-data/python/opentrons_shared_data/pipette/dev_types.py +++ b/shared-data/python/opentrons_shared_data/pipette/dev_types.py @@ -1,5 +1,5 @@ """ -opentrons_shared_data.pipette.dev_types: types for pipette config that +opentrons_shared_data.pipette.types: types for pipette config that require typing_extensions. This module should only be imported if typing.TYPE_CHECKING is True. @@ -11,7 +11,7 @@ # TODO(mc, 2022-06-16): remove type alias when able # and when certain removal will not break any pickling -from ..labware.dev_types import LabwareUri as LabwareUri +from ..labware.types import LabwareUri as LabwareUri PipetteName = Literal[ diff --git a/shared-data/python/opentrons_shared_data/pipette/load_data.py b/shared-data/python/opentrons_shared_data/pipette/load_data.py index 7e0a13de3b7..f8c361cca0c 100644 --- a/shared-data/python/opentrons_shared_data/pipette/load_data.py +++ b/shared-data/python/opentrons_shared_data/pipette/load_data.py @@ -1,7 +1,8 @@ import json -import os +from pathlib import Path +from logging import getLogger -from typing import Dict, Any, Union, Optional, List +from typing import Dict, Any, Union, Optional, List, Iterator from typing_extensions import Literal from functools import lru_cache @@ -26,6 +27,8 @@ LoadedConfiguration = Dict[str, Union[str, Dict[str, Any]]] +LOG = getLogger(__name__) + def _get_configuration_dictionary( config_type: Literal["general", "geometry", "liquid"], @@ -96,6 +99,12 @@ def _physical( return _get_configuration_dictionary("general", channels, model, version) +def _dirs_in(path: Path) -> Iterator[Path]: + for child in path.iterdir(): + if child.is_dir(): + yield child + + @lru_cache(maxsize=None) def load_serial_lookup_table() -> Dict[str, str]: """Load a serial abbreviation lookup table mapped to model name.""" @@ -112,23 +121,27 @@ def load_serial_lookup_table() -> Dict[str, str]: "eight_channel": "multi", } _model_shorthand = {"p1000": "p1k", "p300": "p3h"} - for channel_dir in os.listdir(config_path): - for model_dir in os.listdir(config_path / channel_dir): - for version_file in os.listdir(config_path / channel_dir / model_dir): - version_list = version_file.split(".json")[0].split("_") - built_model = f"{model_dir}_{_channel_model_str[channel_dir]}_v{version_list[0]}.{version_list[1]}" - - model_shorthand = _model_shorthand.get(model_dir, model_dir) - + for channel_dir in _dirs_in(config_path): + for model_dir in _dirs_in(channel_dir): + for version_file in model_dir.iterdir(): + if version_file.suffix != ".json": + continue + try: + version_list = version_file.stem.split("_") + built_model = f"{model_dir.stem}_{_channel_model_str[channel_dir.stem]}_v{version_list[0]}.{version_list[1]}" + except IndexError: + LOG.warning(f"Pipette def with bad name {version_file} ignored") + continue + model_shorthand = _model_shorthand.get(model_dir.stem, model_dir.stem) if ( - model_dir == "p300" + model_dir.stem == "p300" and int(version_list[0]) == 1 and int(version_list[1]) == 0 ): # Well apparently, we decided to switch the shorthand of the p300 depending # on whether it's a "V1" model or not...so...here is the lovely workaround. - model_shorthand = model_dir - serial_shorthand = f"{model_shorthand.upper()}{_channel_shorthand[channel_dir]}V{version_list[0]}{version_list[1]}" + model_shorthand = model_dir.stem + serial_shorthand = f"{model_shorthand.upper()}{_channel_shorthand[channel_dir.stem]}V{version_list[0]}{version_list[1]}" _lookup_table[serial_shorthand] = built_model return _lookup_table diff --git a/shared-data/python/opentrons_shared_data/pipette/mutable_configurations.py b/shared-data/python/opentrons_shared_data/pipette/mutable_configurations.py index 53882dd8f11..06b31215b65 100644 --- a/shared-data/python/opentrons_shared_data/pipette/mutable_configurations.py +++ b/shared-data/python/opentrons_shared_data/pipette/mutable_configurations.py @@ -31,7 +31,7 @@ MutableConfigurationEncoder, MutableConfigurationDecoder, ) -from .dev_types import PipetteModel, PipetteName +from .types import PipetteModel, PipetteName log = logging.getLogger(__name__) diff --git a/shared-data/python/opentrons_shared_data/pipette/pipette_definition.py b/shared-data/python/opentrons_shared_data/pipette/pipette_definition.py index f37051b69a2..84566c4ea92 100644 --- a/shared-data/python/opentrons_shared_data/pipette/pipette_definition.py +++ b/shared-data/python/opentrons_shared_data/pipette/pipette_definition.py @@ -4,7 +4,7 @@ from typing_extensions import Literal from dataclasses import dataclass -from . import types as pip_types, dev_types +from . import types as pip_types, types # The highest and lowest existing overlap version values. TIP_OVERLAP_VERSION_MINIMUM = 0 @@ -302,7 +302,7 @@ class PipettePhysicalPropertiesDefinition(BaseModel): description="The display or full product name of the pipette.", alias="displayName", ) - pipette_backcompat_names: List[dev_types.PipetteName] = Field( + pipette_backcompat_names: List[types.PipetteName] = Field( ..., description="A list of pipette names that are compatible with this pipette.", alias="backCompatNames", diff --git a/shared-data/python/opentrons_shared_data/pipette/pipette_load_name_conversions.py b/shared-data/python/opentrons_shared_data/pipette/pipette_load_name_conversions.py index 9853d58b4ae..7569c736332 100644 --- a/shared-data/python/opentrons_shared_data/pipette/pipette_load_name_conversions.py +++ b/shared-data/python/opentrons_shared_data/pipette/pipette_load_name_conversions.py @@ -2,7 +2,7 @@ from functools import lru_cache from typing import List, Optional, Union, cast, Literal, Tuple from opentrons_shared_data import get_shared_data_root -from .dev_types import PipetteModel, PipetteName +from .types import PipetteModel, PipetteName from .types import ( PipetteChannelType, diff --git a/shared-data/python/opentrons_shared_data/pipette/scripts/build_json_script.py b/shared-data/python/opentrons_shared_data/pipette/scripts/build_json_script.py index 594e7738aea..510d0ae5251 100644 --- a/shared-data/python/opentrons_shared_data/pipette/scripts/build_json_script.py +++ b/shared-data/python/opentrons_shared_data/pipette/scripts/build_json_script.py @@ -23,7 +23,7 @@ PlungerEjectDropTipConfiguration, ) -from ..dev_types import PipetteModelSpec +from ..types import PipetteModelSpec PIPETTE_DEFINITION_ROOT = Path("pipette") / "definitions" / "2" diff --git a/shared-data/python/opentrons_shared_data/pipette/scripts/update_configuration_files.py b/shared-data/python/opentrons_shared_data/pipette/scripts/update_configuration_files.py index 34df66bca48..e787ac2a1cf 100644 --- a/shared-data/python/opentrons_shared_data/pipette/scripts/update_configuration_files.py +++ b/shared-data/python/opentrons_shared_data/pipette/scripts/update_configuration_files.py @@ -31,7 +31,7 @@ ) from ..load_data import _geometry, _physical, _liquid from ..pipette_load_name_conversions import convert_pipette_model -from ..dev_types import PipetteModel +from ..types import PipetteModel """ Instructions: diff --git a/shared-data/python/opentrons_shared_data/pipette/types.py b/shared-data/python/opentrons_shared_data/pipette/types.py index 9ad68132948..7e6fd382dc0 100644 --- a/shared-data/python/opentrons_shared_data/pipette/types.py +++ b/shared-data/python/opentrons_shared_data/pipette/types.py @@ -1,7 +1,12 @@ import enum from dataclasses import dataclass -from typing import Union, Dict, Mapping, Tuple, cast -from typing_extensions import Literal +from typing_extensions import Literal, TypedDict +from typing import Dict, List, Mapping, NewType, Union, Tuple, cast + + +# TODO(mc, 2022-06-16): remove type alias when able +# and when certain removal will not break any pickling +from ..labware.types import LabwareUri as LabwareUri """Pipette Definition V2 Types""" @@ -191,3 +196,166 @@ def dict_for_encode(self) -> bool: TypeOverrides = Mapping[str, Union[float, bool, None]] OverrideType = Dict[str, Union[Dict[str, QuirkConfig], MutableConfig, str]] + + +PipetteName = Literal[ + "p10_single", + "p10_multi", + "p20_single_gen2", + "p20_multi_gen2", + "p50_single", + "p50_multi", + "p50_single_flex", + "p50_multi_flex", + "p300_single", + "p300_multi", + "p300_single_gen2", + "p300_multi_gen2", + "p1000_single", + "p1000_single_gen2", + "p1000_single_flex", + "p1000_multi_flex", + "p1000_96", +] + + +class PipetteNameType(str, enum.Enum): + """Pipette load name values.""" + + value: PipetteName + + P10_SINGLE = "p10_single" + P10_MULTI = "p10_multi" + P20_SINGLE_GEN2 = "p20_single_gen2" + P20_MULTI_GEN2 = "p20_multi_gen2" + P50_SINGLE = "p50_single" + P50_MULTI = "p50_multi" + P50_SINGLE_FLEX = "p50_single_flex" + P50_MULTI_FLEX = "p50_multi_flex" + P300_SINGLE = "p300_single" + P300_MULTI = "p300_multi" + P300_SINGLE_GEN2 = "p300_single_gen2" + P300_MULTI_GEN2 = "p300_multi_gen2" + P1000_SINGLE = "p1000_single" + P1000_SINGLE_GEN2 = "p1000_single_gen2" + P1000_SINGLE_FLEX = "p1000_single_flex" + P1000_MULTI_FLEX = "p1000_multi_flex" + P1000_96 = "p1000_96" + + +# Generic NewType for models because we get new ones frequently and theres +# a huge number of them +PipetteModel = NewType("PipetteModel", str) + +DisplayCategory = Literal["GEN1", "GEN2", "FLEX"] + +# todo(mm, 2022-03-18): +# The JSON schema defines this as any string, not as an enum of string literals. +# Check if it's safe to simplify this to just str. +ConfigUnit = Literal[ + "mm", + "amps", + "mm/sec", + "mm/s", # todo(mm, 2022-03-18): Standardize specs to mm/sec or mm/s. + "presses", +] + +Quirk = NewType("Quirk", str) + +ChannelCount = Literal[1, 8, 96] + +UlPerMmAction = Literal["aspirate", "dispense", "blowout"] + + +class PipetteConfigElement(TypedDict): + value: float + min: float + max: float + + +class PipetteConfigElementWithPerApiLevelValue(TypedDict): + value: float + min: float + max: float + valuesByApiLevel: Dict[str, float] + + +# TypedDicts can't be generic sadly +class PipetteCustomizableConfigElementFloat(TypedDict): + value: float + min: float + max: float + units: ConfigUnit + type: Literal["float"] + + +class PipetteCustomizableConfigElementInt(TypedDict): + value: int + min: int + max: int + units: ConfigUnit + type: Literal["int"] + + +PipetteCustomizableConfigElement = Union[ + PipetteCustomizableConfigElementFloat, PipetteCustomizableConfigElementInt +] + +SmoothieConfigs = TypedDict( + "SmoothieConfigs", + {"stepsPerMM": float, "homePosition": float, "travelDistance": float}, +) + + +class PipetteNameSpec(TypedDict): + displayName: str + displayCategory: DisplayCategory + minVolume: Union[float, int] + maxVolume: Union[float, int] + channels: ChannelCount + defaultAspirateFlowRate: PipetteConfigElementWithPerApiLevelValue + defaultDispenseFlowRate: PipetteConfigElementWithPerApiLevelValue + defaultBlowOutFlowRate: PipetteConfigElementWithPerApiLevelValue + smoothieConfigs: SmoothieConfigs + defaultTipracks: List[LabwareUri] + + +PipetteNameSpecs = Dict[PipetteName, PipetteNameSpec] + +UlPerMm = Dict[UlPerMmAction, List[List[float]]] + + +class PipetteModelSpec(TypedDict, total=False): + name: PipetteName + top: PipetteCustomizableConfigElementFloat + bottom: PipetteCustomizableConfigElementFloat + blowout: PipetteCustomizableConfigElementFloat + dropTip: PipetteCustomizableConfigElementFloat + pickUpCurrent: PipetteCustomizableConfigElementFloat + pickUpDistance: PipetteCustomizableConfigElementFloat + pickUpIncrement: PipetteCustomizableConfigElementFloat + pickUpPresses: PipetteCustomizableConfigElementInt + pickUpSpeed: PipetteCustomizableConfigElementFloat + plungerCurrent: PipetteCustomizableConfigElementFloat + dropTipCurrent: PipetteCustomizableConfigElementFloat + dropTipSpeed: PipetteCustomizableConfigElementFloat + modelOffset: List[float] + nozzleOffset: List[float] + ulPerMm: List[UlPerMm] + tipOverlap: Dict[str, float] + tipLength: PipetteCustomizableConfigElementFloat + quirks: List[Quirk] + # these keys are not present in some pipette definitions + backCompatNames: List[PipetteName] + idleCurrent: float + returnTipHeight: float + + +class PipetteFusedSpec(PipetteNameSpec, PipetteModelSpec, total=False): + pass + + +class PipetteModelSpecs(TypedDict): + config: Dict[PipetteModel, PipetteModelSpec] + mutableConfigs: List[str] + validQuirks: List[str] diff --git a/shared-data/python/opentrons_shared_data/protocol/constants.py b/shared-data/python/opentrons_shared_data/protocol/constants.py index f3ce8462b37..cd838273dd3 100644 --- a/shared-data/python/opentrons_shared_data/protocol/constants.py +++ b/shared-data/python/opentrons_shared_data/protocol/constants.py @@ -3,7 +3,7 @@ if TYPE_CHECKING: - from .dev_types import ( + from .types import ( DelayCommandId, BlowoutCommandId, PickUpTipCommandId, diff --git a/shared-data/python/opentrons_shared_data/protocol/models/protocol_schema_v8.py b/shared-data/python/opentrons_shared_data/protocol/models/protocol_schema_v8.py index 5ec17baa46c..ce5b814d73f 100644 --- a/shared-data/python/opentrons_shared_data/protocol/models/protocol_schema_v8.py +++ b/shared-data/python/opentrons_shared_data/protocol/models/protocol_schema_v8.py @@ -100,7 +100,7 @@ class ProtocolSchemaV8(BaseModel): liquids: Dict[str, Liquid] labwareDefinitionSchemaId: Literal["opentronsLabwareSchemaV2"] labwareDefinitions: Dict[str, LabwareDefinition] - commandSchemaId: Literal["opentronsCommandSchemaV8"] + commandSchemaId: Literal["opentronsCommandSchemaV8", "opentronsCommandSchemaV9"] commands: List[Command] commandAnnotationSchemaId: Literal["opentronsCommandAnnotationSchemaV1"] commandAnnotations: List[CommandAnnotation] diff --git a/shared-data/python/opentrons_shared_data/protocol/dev_types.py b/shared-data/python/opentrons_shared_data/protocol/types.py similarity index 98% rename from shared-data/python/opentrons_shared_data/protocol/dev_types.py rename to shared-data/python/opentrons_shared_data/protocol/types.py index 43bf8ab7b6d..962969659fc 100644 --- a/shared-data/python/opentrons_shared_data/protocol/dev_types.py +++ b/shared-data/python/opentrons_shared_data/protocol/types.py @@ -1,14 +1,14 @@ """ -opentrons_shared_data.protocol.dev_types: types for json protocols +opentrons_shared_data.protocol.types: types for json protocols """ from typing import Any, Dict, List, Optional, Union from enum import Enum from typing_extensions import TypedDict, Literal -from ..pipette.dev_types import PipetteName -from ..labware.dev_types import LabwareDefinition -from ..module.dev_types import ModuleModel +from ..pipette.types import PipetteName +from ..labware.types import LabwareDefinition +from ..module.types import ModuleModel SlotSpan = Literal["span7_8_10_11"] diff --git a/shared-data/python/opentrons_shared_data/robot/__init__.py b/shared-data/python/opentrons_shared_data/robot/__init__.py index 74f44994415..4dfdc542d52 100644 --- a/shared-data/python/opentrons_shared_data/robot/__init__.py +++ b/shared-data/python/opentrons_shared_data/robot/__init__.py @@ -7,7 +7,7 @@ from .. import get_shared_data_root -from .dev_types import RobotDefinition, RobotType +from .types import RobotDefinition, RobotType DEFAULT_ROBOT_DEFINITION_VERSION: Final = 1 diff --git a/shared-data/python/opentrons_shared_data/robot/dev_types.py b/shared-data/python/opentrons_shared_data/robot/types.py similarity index 95% rename from shared-data/python/opentrons_shared_data/robot/dev_types.py rename to shared-data/python/opentrons_shared_data/robot/types.py index 90f0f19c1c4..e478957bc29 100644 --- a/shared-data/python/opentrons_shared_data/robot/dev_types.py +++ b/shared-data/python/opentrons_shared_data/robot/types.py @@ -1,4 +1,4 @@ -"""opentrons_shared_data.robot.dev_types: types for robot def.""" +"""opentrons_shared_data.robot.types: types for robot def.""" import enum from typing import NewType, List, Dict, Any from typing_extensions import Literal, TypedDict, NotRequired diff --git a/shared-data/python/tests/deck/test_position.py b/shared-data/python/tests/deck/test_position.py index 1b10d556444..719b5c0dc4a 100644 --- a/shared-data/python/tests/deck/test_position.py +++ b/shared-data/python/tests/deck/test_position.py @@ -6,7 +6,7 @@ list_names as list_deck_definition_names, load as load_deck_definition, ) -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.deck.types import ( AddressableArea, Cutout, CutoutFixture, diff --git a/shared-data/python/tests/deck/test_typechecks.py b/shared-data/python/tests/deck/test_typechecks.py index 4e2406df0fa..a5fa3747e99 100644 --- a/shared-data/python/tests/deck/test_typechecks.py +++ b/shared-data/python/tests/deck/test_typechecks.py @@ -5,7 +5,7 @@ list_names as list_deck_definition_names, load as load_deck_definition, ) -from opentrons_shared_data.deck.dev_types import ( +from opentrons_shared_data.deck.types import ( DeckDefinitionV3, DeckDefinitionV5, ) diff --git a/shared-data/python/tests/labware/test_typechecks.py b/shared-data/python/tests/labware/test_typechecks.py index 64f76679cf9..b62dd27cc70 100644 --- a/shared-data/python/tests/labware/test_typechecks.py +++ b/shared-data/python/tests/labware/test_typechecks.py @@ -2,7 +2,7 @@ import typeguard from opentrons_shared_data.labware import load_definition -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from . import get_ot_defs diff --git a/shared-data/python/tests/module/test_typechecks.py b/shared-data/python/tests/module/test_typechecks.py index 9aa5ded743e..bc6885d94fb 100644 --- a/shared-data/python/tests/module/test_typechecks.py +++ b/shared-data/python/tests/module/test_typechecks.py @@ -2,7 +2,7 @@ import typeguard from opentrons_shared_data import module -from opentrons_shared_data.module import dev_types +from opentrons_shared_data.module import types from . import list_v2_defs, list_v3_defs @@ -11,16 +11,16 @@ def test_v3_definitions_match_types(defname: str) -> None: """Test that V3 module definitions match ModuleDefinitionV3.""" def_dict = module.load_definition("3", defname) - typeguard.check_type(def_dict, dev_types.ModuleDefinitionV3) + typeguard.check_type(def_dict, types.ModuleDefinitionV3) @pytest.mark.parametrize("defname", list_v2_defs()) def test_v2_definitions_match_types(defname: str) -> None: defdict = module.load_definition("2", defname) # type: ignore [call-overload] - typeguard.check_type(defdict, dev_types.ModuleDefinitionV2) + typeguard.check_type(defdict, types.ModuleDefinitionV2) @pytest.mark.parametrize("defname", ["magdeck", "tempdeck", "thermocycler"]) def test_v1_definitions_match_types(defname: str) -> None: defdict = module.load_definition("1", defname) - typeguard.check_type(defdict, dev_types.ModuleDefinitionV1) + typeguard.check_type(defdict, types.ModuleDefinitionV1) diff --git a/shared-data/python/tests/pipette/test_load_data.py b/shared-data/python/tests/pipette/test_load_data.py index 1b9e9775c16..012aed7baca 100644 --- a/shared-data/python/tests/pipette/test_load_data.py +++ b/shared-data/python/tests/pipette/test_load_data.py @@ -3,7 +3,7 @@ from opentrons_shared_data.pipette import ( load_data, pipette_load_name_conversions, - dev_types, + types, ) from opentrons_shared_data.pipette.types import ( PipetteChannelType, @@ -80,7 +80,7 @@ def test_update_pipette_configuration( liquid_class = LiquidClasses.default model_name = pipette_load_name_conversions.convert_pipette_model( - cast(dev_types.PipetteModel, pipette_model) + cast(types.PipetteModel, pipette_model) ) base_configurations = load_data.load_definition( model_name.pipette_type, model_name.pipette_channels, model_name.pipette_version diff --git a/shared-data/python/tests/pipette/test_max_flow_rates_per_volume.py b/shared-data/python/tests/pipette/test_max_flow_rates_per_volume.py index 45ce013d9d8..b64f0a0b5c4 100644 --- a/shared-data/python/tests/pipette/test_max_flow_rates_per_volume.py +++ b/shared-data/python/tests/pipette/test_max_flow_rates_per_volume.py @@ -8,7 +8,7 @@ from opentrons_shared_data.pipette.load_data import load_definition from opentrons_shared_data.pipette.ul_per_mm import piecewise_volume_conversion -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel from opentrons_shared_data.pipette.pipette_definition import ( ulPerMMDefinition, ) diff --git a/shared-data/python/tests/pipette/test_mutable_configurations.py b/shared-data/python/tests/pipette/test_mutable_configurations.py index db851a15042..40059f6bfe6 100644 --- a/shared-data/python/tests/pipette/test_mutable_configurations.py +++ b/shared-data/python/tests/pipette/test_mutable_configurations.py @@ -11,7 +11,6 @@ pipette_definition, pipette_load_name_conversions as pip_conversions, load_data, - dev_types, ) @@ -243,13 +242,13 @@ def test_save_invalid_overrides( argvalues=[ [ pip_conversions.convert_pipette_model( - cast(dev_types.PipetteModel, "p1000_96_v3.3") + cast(types.PipetteModel, "p1000_96_v3.3") ), "P1KHV3320230629", ], [ pip_conversions.convert_pipette_model( - cast(dev_types.PipetteModel, "p50_multi_v1.5") + cast(types.PipetteModel, "p50_multi_v1.5") ), TEST_SERIAL_NUMBER, ], diff --git a/shared-data/python/tests/pipette/test_pipette_load_name_conversions.py b/shared-data/python/tests/pipette/test_pipette_load_name_conversions.py index a62880429e1..6e792560e9c 100644 --- a/shared-data/python/tests/pipette/test_pipette_load_name_conversions.py +++ b/shared-data/python/tests/pipette/test_pipette_load_name_conversions.py @@ -7,7 +7,7 @@ PipetteVersionType, PipetteGenerationType, ) -from opentrons_shared_data.pipette.dev_types import PipetteModel, PipetteName +from opentrons_shared_data.pipette.types import PipetteModel, PipetteName from opentrons_shared_data.pipette import ( pipette_definition as pc, pipette_load_name_conversions as ps, diff --git a/shared-data/python/tests/pipette/test_typechecks.py b/shared-data/python/tests/pipette/test_typechecks.py index f4d1ec91af9..a44b673d3ed 100644 --- a/shared-data/python/tests/pipette/test_typechecks.py +++ b/shared-data/python/tests/pipette/test_typechecks.py @@ -7,7 +7,7 @@ fuse_specs, dummy_model_for_name, ) -from opentrons_shared_data.pipette.dev_types import ( +from opentrons_shared_data.pipette.types import ( PipetteModelSpecs, PipetteNameSpecs, PipetteFusedSpec, diff --git a/shared-data/python/tests/pipette/test_validate_schema.py b/shared-data/python/tests/pipette/test_validate_schema.py index f9dc4209298..0b703504957 100644 --- a/shared-data/python/tests/pipette/test_validate_schema.py +++ b/shared-data/python/tests/pipette/test_validate_schema.py @@ -13,7 +13,7 @@ from opentrons_shared_data.pipette.pipette_load_name_conversions import ( convert_pipette_model, ) -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel def iterate_models() -> Iterator[PipetteModel]: @@ -68,6 +68,7 @@ def test_pick_up_configs_configuration_by_nozzle_map_keys() -> None: for channel_dir in os.listdir(paths_to_validate): for model_dir in os.listdir(paths_to_validate / channel_dir): for version_file in os.listdir(paths_to_validate / channel_dir / model_dir): + print(version_file) version_list = version_file.split(".json")[0].split("_") built_model: PipetteModel = PipetteModel( f"{model_dir}_{_channel_model_str[channel_dir]}_v{version_list[0]}.{version_list[1]}" diff --git a/shared-data/python/tests/protocol/test_typechecks.py b/shared-data/python/tests/protocol/test_typechecks.py index 934944828e2..0b52f15c24e 100644 --- a/shared-data/python/tests/protocol/test_typechecks.py +++ b/shared-data/python/tests/protocol/test_typechecks.py @@ -3,7 +3,7 @@ import typeguard from opentrons_shared_data import load_shared_data -from opentrons_shared_data.protocol.dev_types import ( +from opentrons_shared_data.protocol.types import ( JsonProtocolV3, JsonProtocolV4, JsonProtocolV5, diff --git a/shared-data/python/tests/robot/test_typechecks.py b/shared-data/python/tests/robot/test_typechecks.py index 5d838901178..66a7e559051 100644 --- a/shared-data/python/tests/robot/test_typechecks.py +++ b/shared-data/python/tests/robot/test_typechecks.py @@ -3,7 +3,7 @@ from opentrons_shared_data.robot import load -from opentrons_shared_data.robot.dev_types import RobotDefinition, RobotType +from opentrons_shared_data.robot.types import RobotDefinition, RobotType @pytest.mark.parametrize("defname", ["OT-2 Standard", "OT-3 Standard"]) diff --git a/shared-data/tsconfig.json b/shared-data/tsconfig.json index cb960e927cb..85db0e9c2bc 100644 --- a/shared-data/tsconfig.json +++ b/shared-data/tsconfig.json @@ -5,7 +5,7 @@ "composite": true, "rootDir": ".", "outDir": "lib", - "moduleResolution": "node", + "moduleResolution": "node" }, "module": "ESNext", "include": [ @@ -15,8 +15,9 @@ "labware", "deck", "command", + "errors", "liquid/types", "commandAnnotation/types", - "vite.config.ts", + "vite.config.ts" ] }