From 29d7d79a75f4d616eeb5ecabb36bbfbf479ee8d7 Mon Sep 17 00:00:00 2001 From: Brayan Almonte Date: Mon, 17 Jun 2024 13:53:19 -0400 Subject: [PATCH 01/50] feat(hardware): added safety relay active state to HepaUVState and safety_relay_inactive error (#15311) --- .../firmware_bindings/constants.py | 2 ++ .../firmware_bindings/messages/payloads.py | 2 ++ .../hardware_control/hepa_uv_settings.py | 3 ++ .../hardware_control/test_hepauv_settings.py | 28 ++++++++++++++++--- 4 files changed, 31 insertions(+), 4 deletions(-) diff --git a/hardware/opentrons_hardware/firmware_bindings/constants.py b/hardware/opentrons_hardware/firmware_bindings/constants.py index cd91ced91b7..4dd7c759782 100644 --- a/hardware/opentrons_hardware/firmware_bindings/constants.py +++ b/hardware/opentrons_hardware/firmware_bindings/constants.py @@ -4,6 +4,7 @@ by default. Please do not unconditionally import things outside the python standard library. """ + from enum import Enum, unique from typing import Union, Dict, List @@ -294,6 +295,7 @@ class ErrorCode(int, Enum): door_open = 0x0E reed_open = 0x0F motor_driver_error_detected = 0x10 + safety_relay_inactive = 0x11 @unique diff --git a/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py b/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py index c351495ba5b..50890187fe8 100644 --- a/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py +++ b/hardware/opentrons_hardware/firmware_bindings/messages/payloads.py @@ -1,4 +1,5 @@ """Payloads of can bus messages.""" + # TODO (amit, 2022-01-26): Figure out why using annotations import ruins # dataclass fields interpretation. # from __future__ import annotations @@ -684,6 +685,7 @@ class GetHepaUVStatePayloadResponse(EmptyPayload): uv_light_on: utils.UInt8Field remaining_time_s: utils.UInt32Field uv_current_ma: utils.UInt16Field + safety_relay_active: utils.UInt8Field @dataclass(eq=False) diff --git a/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py b/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py index 2812cdf3f7d..6e87c466b7a 100644 --- a/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py +++ b/hardware/opentrons_hardware/hardware_control/hepa_uv_settings.py @@ -1,4 +1,5 @@ """Utilities for controlling the hepa/uv extension module.""" + import logging import asyncio from typing import Optional @@ -46,6 +47,7 @@ class HepaUVState: uv_duration_s: int remaining_time_s: int uv_current_ma: int + safety_relay_active: bool async def set_hepa_fan_state( @@ -136,6 +138,7 @@ def _listener(message: MessageDefinition, arb_id: ArbitrationId) -> None: uv_duration_s=int(message.payload.uv_duration_s.value), remaining_time_s=int(message.payload.remaining_time_s.value), uv_current_ma=int(message.payload.uv_current_ma.value), + safety_relay_active=bool(message.payload.safety_relay_active.value), ) def _filter(arb_id: ArbitrationId) -> bool: diff --git a/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py b/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py index dcaf85a8653..ad188bd153c 100644 --- a/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py +++ b/hardware/tests/opentrons_hardware/hardware_control/test_hepauv_settings.py @@ -55,6 +55,7 @@ def create_hepa_uv_state_response( duration: int, remaining_time: int, uv_current: int, + safety_relay_active: bool, ) -> MessageDefinition: """Create a GetHepaUVStateResponse.""" return md.GetHepaUVStateResponse( @@ -63,6 +64,7 @@ def create_hepa_uv_state_response( uv_duration_s=UInt32Field(duration), remaining_time_s=UInt32Field(remaining_time), uv_current_ma=UInt16Field(uv_current), + safety_relay_active=UInt8Field(safety_relay_active), ) ) @@ -162,12 +164,29 @@ def responder( [ ( NodeId.host, - create_hepa_uv_state_response(True, 900, 300, 3300), + create_hepa_uv_state_response(True, 900, 300, 3300, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(True, 0, 0, 33000, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(True, 0, 0, 33000, False), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(False, 0, 0, 0, True), + NodeId.hepa_uv, + ), + ( + NodeId.host, + create_hepa_uv_state_response(False, 900, 0, 0, False), NodeId.hepa_uv, ), - (NodeId.host, create_hepa_uv_state_response(True, 0, 0, 33000), NodeId.hepa_uv), - (NodeId.host, create_hepa_uv_state_response(False, 0, 0, 0), NodeId.hepa_uv), - (NodeId.host, create_hepa_uv_state_response(False, 900, 0, 0), NodeId.hepa_uv), ], ) async def test_get_hepa_uv_state( @@ -202,6 +221,7 @@ def responder( int(payload.uv_duration_s.value), int(payload.remaining_time_s.value), int(payload.uv_current_ma.value), + bool(payload.safety_relay_active.value), ) == res ) From 1193c2921666a46cb6a97b54400cdba9357fc556 Mon Sep 17 00:00:00 2001 From: Ed Cormany Date: Wed, 26 Jun 2024 16:15:58 -0400 Subject: [PATCH 02/50] chore: 7.4.0 release notes (#15531) # Overview Indicate that 7.4.0 adds support for HEPA/UV. # Test Plan Check in-app after alpha 0 tag is cut. # Changelog app-shell and api notes. # Risk assessment nil --- api/release-notes.md | 8 ++++++++ app-shell/build/release-notes.md | 8 ++++++++ 2 files changed, 16 insertions(+) diff --git a/api/release-notes.md b/api/release-notes.md index 4ba3074bf31..5c433e6cba8 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -6,6 +6,14 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- +## Opentrons Robot Software Changes in 7.4.0 + +Welcome to the v7.4.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). + +--- + ## Opentrons Robot Software Changes in 7.3.1 Welcome to the v7.3.1 release of the Opentrons robot software! diff --git a/app-shell/build/release-notes.md b/app-shell/build/release-notes.md index eadc8b068b2..60adacf1a4a 100644 --- a/app-shell/build/release-notes.md +++ b/app-shell/build/release-notes.md @@ -6,6 +6,14 @@ log][]. For a list of currently known issues, please see the [Opentrons issue tr --- +## Opentrons App Changes in 7.4.0 + +Welcome to the v7.4.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). + +--- + ## Opentrons App Changes in 7.3.1 Welcome to the v7.3.1 release of the Opentrons App! From 9b8eafc442abffb1cce857fecb1a5cdf1e878205 Mon Sep 17 00:00:00 2001 From: Shlok Amin Date: Thu, 27 Jun 2024 13:34:54 -0400 Subject: [PATCH 03/50] fix(api): Parse all RTP fields strictly to fix flakiness (#15187) (#15536) closes https://opentrons.atlassian.net/browse/RESC-287 --------- Co-authored-by: Max Marrone Co-authored-by: Edward Cormany Co-authored-by: Sanniti Pimpley --- api/release-notes.md | 3 ++ api/src/opentrons/protocol_engine/types.py | 44 ++++++++++++++-------- 2 files changed, 31 insertions(+), 16 deletions(-) diff --git a/api/release-notes.md b/api/release-notes.md index 5c433e6cba8..dbfbfc5bad4 100644 --- a/api/release-notes.md +++ b/api/release-notes.md @@ -12,6 +12,9 @@ Welcome to the v7.4.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). +### Bug Fixes + +- Fixed certain string runtime parameter values being misinterpreted as an incorrect type. --- ## Opentrons Robot Software Changes in 7.3.1 diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index 13e9515e447..77ab6231b71 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -4,7 +4,15 @@ from datetime import datetime from enum import Enum from dataclasses import dataclass -from pydantic import BaseModel, Field, validator +from pydantic import ( + BaseModel, + Field, + StrictBool, + StrictFloat, + StrictInt, + StrictStr, + validator, +) from typing import Optional, Union, List, Dict, Any, NamedTuple, Tuple, FrozenSet from typing_extensions import Literal, TypeGuard @@ -877,12 +885,14 @@ def from_hw_state(cls, state: HwTipStateType) -> "TipPresenceStatus": class RTPBase(BaseModel): """Parameters defined in a protocol.""" - displayName: str = Field(..., description="Display string for the parameter.") - variableName: str = Field(..., description="Python variable name of the parameter.") - description: Optional[str] = Field( + displayName: StrictStr = Field(..., description="Display string for the parameter.") + variableName: StrictStr = Field( + ..., description="Python variable name of the parameter." + ) + description: Optional[StrictStr] = Field( None, description="Detailed description of the parameter." ) - suffix: Optional[str] = Field( + suffix: Optional[StrictStr] = Field( None, description="Units (like mL, mm/sec, etc) or a custom suffix for the parameter.", ) @@ -894,17 +904,17 @@ class NumberParameter(RTPBase): type: Literal["int", "float"] = Field( ..., description="String specifying whether the number is an int or float type." ) - min: float = Field( + min: Union[StrictInt, StrictFloat] = Field( ..., description="Minimum value that the number param is allowed to have." ) - max: float = Field( + max: Union[StrictInt, StrictFloat] = Field( ..., description="Maximum value that the number param is allowed to have." ) - value: float = Field( + value: Union[StrictInt, StrictFloat] = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: float = Field( + default: Union[StrictInt, StrictFloat] = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -916,11 +926,11 @@ class BooleanParameter(RTPBase): type: Literal["bool"] = Field( default="bool", description="String specifying the type of this parameter" ) - value: bool = Field( + value: StrictBool = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: bool = Field( + default: StrictBool = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -929,8 +939,10 @@ class BooleanParameter(RTPBase): class EnumChoice(BaseModel): """Components of choices used in RTP Enum Parameters.""" - displayName: str = Field(..., description="Display string for the param's choice.") - value: Union[float, str] = Field( + displayName: StrictStr = Field( + ..., description="Display string for the param's choice." + ) + value: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="Enum value of the param's choice." ) @@ -945,11 +957,11 @@ class EnumParameter(RTPBase): choices: List[EnumChoice] = Field( ..., description="List of valid choices for this parameter." ) - value: Union[float, str] = Field( + value: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="The value assigned to the parameter; if not supplied by the client, will be assigned the default value.", ) - default: Union[float, str] = Field( + default: Union[StrictInt, StrictFloat, StrictStr] = Field( ..., description="Default value of the parameter, to be used when there is no client-specified value.", ) @@ -958,5 +970,5 @@ class EnumParameter(RTPBase): RunTimeParameter = Union[NumberParameter, EnumParameter, BooleanParameter] RunTimeParamValuesType = Dict[ - str, Union[float, bool, str] + StrictStr, Union[StrictInt, StrictFloat, StrictBool, StrictStr] ] # update value types as more RTP types are added From 50eeb06d6df9733f06b8852a711ca701a7cc15ee Mon Sep 17 00:00:00 2001 From: Edward Cormany Date: Fri, 12 Jul 2024 14:48:09 -0400 Subject: [PATCH 04/50] docs(api): fix typo calling `pick_up_tip` instead of `drop_tip` --- api/docs/v2/basic_commands/pipette_tips.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) 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:: From 6fe994e2e73232542d4cdadf0117ee680d1937a2 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Mon, 22 Jul 2024 12:05:44 -0500 Subject: [PATCH 05/50] build: Sign windows builds via remote key storage (#15718) (#15735) ## `cherry-pick` 246efcbb52 Must update `chore_release-7.4.0` so that windows builds work and we may test an update. To understand the changes see #15718 Co-authored-by: Seth Foster --- .github/workflows/app-test-build-deploy.yaml | 77 +++++++++++++++++--- app-shell/electron-builder.config.js | 6 ++ app-shell/scripts/windows-custom-sign.js | 62 ++++++++++++++++ 3 files changed, 134 insertions(+), 11 deletions(-) create mode 100644 app-shell/scripts/windows-custom-sign.js diff --git a/.github/workflows/app-test-build-deploy.yaml b/.github/workflows/app-test-build-deploy.yaml index 878a875bdfc..4cc3754ace6 100644 --- a/.github/workflows/app-test-build-deploy.yaml +++ b/.github/workflows/app-test-build-deploy.yaml @@ -184,24 +184,45 @@ jobs: echo "both develop builds for edge" echo 'variants=["release", "internal-release"]' >> $GITHUB_OUTPUT echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build-internal')) }}" = "true" ] ; then - echo "internal-release builds for app-build-internal suffixes" + elif [ "${{ format('{0}', contains(github.ref, 'app-build-internal')) }}" = "true" ] ; then + echo 'variants=["internal-release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build')) }}" = "true" ] ; then - echo "release develop builds for app-build suffixes" + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "internal-release as-release builds for app-build-internal + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "internal-release develop builds for app-build-internal suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi + elif [ "${{ format('{0}', contains(github.ref, 'app-build')) }}" = "true" ] ; then echo 'variants=["release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT - elif [ "${{ format('{0}', endsWith(github.ref, 'app-build-both')) }}" = "true" ] ; then - echo "Both develop builds for app-build-both suffixes" + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "release as-release builds for app-build + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "release develop builds for app-build suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi + elif [ "${{ format('{0}', contains(github.ref, 'app-build-both')) }}" = "true" ] ; then + echo 'variants=["release", "internal-release"]' >> $GITHUB_OUTPUT - echo 'type=develop' >> $GITHUB_OUTPUT + if [ "${{ format('{0}', contains(github.ref, 'as-release')) }}" = "true" ] ; then + echo "Both as-release builds for app-build-both + as-release suffixes" + echo 'type=as-release' >> $GITHUB_OUTPUT + else + echo "Both develop builds for app-build-both + as-release suffixes" + echo 'type=develop' >> $GITHUB_OUTPUT + fi else echo "No build for ref ${{github.ref}} and event ${{github.event_type}}" echo 'variants=[]' >> $GITHUB_OUTPUT echo 'type=develop' >> $GITHUB_OUTPUT fi + - name: set summary + run: | + echo 'Type: ${{steps.determine-build-type.outputs.type}} Variants: ${{steps.determine-build-type.outputs.variants}}' >> $GITHUB_STEP_SUMMARY + build-app: needs: [determine-build-type] if: needs.determine-build-type.outputs.variants != '[]' @@ -276,6 +297,34 @@ jobs: npm config set cache ${{ github.workspace }}/.npm-cache yarn config set cache-folder ${{ github.workspace }}/.yarn-cache make setup-js + + - name: 'Configure Windows code signing environment' + if: startsWith(matrix.os, 'windows') && contains(needs.determine-build-type.outputs.type, 'release') + shell: bash + run: | + echo "${{ secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 + echo "${{ secrets.WINDOWS_CSC_B64}}" | base64 --decode > /d/opentrons_labworks_inc.crt + echo "C:\Program Files (x86)\Windows Kits\10\App Certification Kit" >> $GITHUB_PATH + echo "C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools" >> $GITHUB_PATH + echo "C:\Program Files\DigiCert\DigiCert Keylocker Tools" >> $GITHUB_PATH + + - name: 'Setup Windows code signing helpers' + if: startsWith(matrix.os, 'windows') && contains(needs.determine-build-type.outputs.type, 'release') + shell: cmd + env: + SM_HOST: ${{ secrets.SM_HOST }} + SM_CLIENT_CERT_FILE: "D:\\Certificate_pkcs12.p12" + SM_CLIENT_CERT_PASSWORD: ${{secrets.SM_CLIENT_CERT_PASSWORD}} + SM_API_KEY: ${{secrets.SM_API_KEY}} + run: | + curl -X GET https://one.digicert.com/signingmanager/api-ui/v1/releases/Keylockertools-windows-x64.msi/download -H "x-api-key:${{secrets.SM_API_KEY}}" -o Keylockertools-windows-x64.msi + msiexec /i Keylockertools-windows-x64.msi /quiet /qn + smksp_registrar.exe list + smctl.exe keypair ls + C:\Windows\System32\certutil.exe -csp "DigiCert Signing Manager KSP" -key -user + smksp_cert_sync.exe + smctl.exe healthcheck --all + # build the desktop app and deploy it - name: 'build ${{matrix.variant}} app for ${{ matrix.os }}' if: matrix.target == 'desktop' @@ -283,8 +332,14 @@ jobs: env: OT_APP_MIXPANEL_ID: ${{ secrets.OT_APP_MIXPANEL_ID }} OT_APP_INTERCOM_ID: ${{ secrets.OT_APP_INTERCOM_ID }} - WIN_CSC_LINK: ${{ secrets.OT_APP_CSC_WINDOWS }} - WIN_CSC_KEY_PASSWORD: ${{ secrets.OT_APP_CSC_KEY_WINDOWS }} + WINDOWS_SIGN: ${{ format('{0}', contains(needs.determine-build-type.outputs.type, 'release')) }} + SM_HOST: ${{secrets.SM_HOST}} + SM_CLIENT_CERT_FILE: "D:\\Certificate_pkcs12.p12" + SM_CLIENT_CERT_PASSWORD: ${{secrets.SM_CLIENT_CERT_PASSWORD}} + SM_API_KEY: ${{secrets.SM_API_KEY}} + SM_CODE_SIGNING_CERT_SHA1_HASH: ${{secrets.SM_CODE_SIGNING_CERT_SHA1_HASH}} + SM_KEYPAIR_ALIAS: ${{secrets.SM_KEYPAIR_ALIAS}} + WINDOWS_CSC_FILEPATH: "D:\\opentrons_labworks_inc.crt" CSC_LINK: ${{ secrets.OT_APP_CSC_MACOS }} CSC_KEY_PASSWORD: ${{ secrets.OT_APP_CSC_KEY_MACOS }} APPLE_ID: ${{ secrets.OT_APP_APPLE_ID }} diff --git a/app-shell/electron-builder.config.js b/app-shell/electron-builder.config.js index aa61720338b..2e66f01690c 100644 --- a/app-shell/electron-builder.config.js +++ b/app-shell/electron-builder.config.js @@ -8,6 +8,7 @@ const { } = process.env const DEV_MODE = process.env.NODE_ENV !== 'production' const USE_PYTHON = process.env.NO_PYTHON !== 'true' +const WINDOWS_SIGN = process.env.WINDOWS_SIGN === 'true' const project = process.env.OPENTRONS_PROJECT ?? 'robot-stack' // this will generate either @@ -72,6 +73,11 @@ module.exports = async () => ({ target: ['nsis'], publisherName: 'Opentrons Labworks Inc.', icon: project === 'robot-stack' ? 'build/icon.ico' : 'build/three.ico', + forceCodeSigning: WINDOWS_SIGN, + rfc3161TimeStampServer: 'http://timestamp.digicert.com', + sign: 'scripts/windows-custom-sign.js', + signDlls: true, + signingHashAlgorithms: ['sha256'], }, nsis: { oneClick: false, diff --git a/app-shell/scripts/windows-custom-sign.js b/app-shell/scripts/windows-custom-sign.js new file mode 100644 index 00000000000..90d7927ab6a --- /dev/null +++ b/app-shell/scripts/windows-custom-sign.js @@ -0,0 +1,62 @@ +// from https://github.com/electron-userland/electron-builder/issues/7605 + +'use strict' + +const { execSync } = require('node:child_process') + +exports.default = async configuration => { + const signCmd = `smctl sign --keypair-alias="${String( + process.env.SM_KEYPAIR_ALIAS + )}" --input "${String(configuration.path)}" --certificate="${String( + process.env.WINDOWS_CSC_FILEPATH + )}" --exit-non-zero-on-fail --failfast --verbose` + console.log(signCmd) + try { + const signProcess = execSync(signCmd, { + stdio: 'pipe', + }) + console.log(`Sign success!`) + console.log( + `Sign stdout: ${signProcess?.stdout?.toString() ?? ''}` + ) + console.log( + `Sign stderr: ${signProcess?.stderr?.toString() ?? ''}` + ) + console.log(`Sign code: ${signProcess.code}`) + } catch (err) { + console.error(`Exception running sign: ${err.status}! +Process stdout: + ${err?.stdout?.toString() ?? ''} +------------- +Process stderr: +${err?.stdout?.toString() ?? ''} +------------- +`) + throw err + } + const verifyCmd = `smctl sign verify --fingerprint="${String( + process.env.SM_CODE_SIGNING_CERT_SHA1_HASH + )}" --input="${String(configuration.path)}" --verbose` + console.log(verifyCmd) + try { + const verifyProcess = execSync(verifyCmd, { stdio: 'pipe' }) + console.log(`Verify success!`) + console.log( + `Verify stdout: ${verifyProcess?.stdout?.toString() ?? ''}` + ) + console.log( + `Verify stderr: ${verifyProcess?.stderr?.toString() ?? ''}` + ) + } catch (err) { + console.error(` +Exception running verification: ${err.status}! +Process stdout: + ${err?.stdout?.toString() ?? ''} +-------------- +Process stderr: + ${err?.stderr?.toString() ?? ''} +-------------- +`) + throw err + } +} From aa35e2d59c2ece888d1c4bed663ade680d34dc7c Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Mon, 22 Jul 2024 14:41:19 -0400 Subject: [PATCH 06/50] feat(app): add the protocol run's current action count to the ODD (#15724) Closes EXEC-561 The current run action is displayed on the desktop app but is missing from the ODD. This adds it to the RunningProtocolCommandList. --- .../RunningProtocol/RunningProtocolCommandList.tsx | 10 +++++++++- .../__tests__/RunningProtocolCommandList.test.tsx | 5 +++++ 2 files changed, 14 insertions(+), 1 deletion(-) diff --git a/app/src/organisms/OnDeviceDisplay/RunningProtocol/RunningProtocolCommandList.tsx b/app/src/organisms/OnDeviceDisplay/RunningProtocol/RunningProtocolCommandList.tsx index f20cafd8875..9bfe2ca73fc 100644 --- a/app/src/organisms/OnDeviceDisplay/RunningProtocol/RunningProtocolCommandList.tsx +++ b/app/src/organisms/OnDeviceDisplay/RunningProtocol/RunningProtocolCommandList.tsx @@ -18,6 +18,7 @@ import { SPACING, LegacyStyledText, TYPOGRAPHY, + StyledText, } from '@opentrons/components' import { RUN_STATUS_RUNNING, RUN_STATUS_IDLE } from '@opentrons/api-client' @@ -221,8 +222,15 @@ export function RunningProtocolCommandList({ + + {index + 1} + { expect(mockShowModal).toHaveBeenCalled() }) + it("it displays the run's current action number", () => { + render({ ...props, currentRunCommandIndex: 11 }) + screen.getByText(12) + }) + // ToDo (kj:04/10/2023) once we fix the track event stuff, we can implement tests it.todo('when tapping play button, track event mock function is called') }) From 0a4b77488d4eef28ba8a7595791e770bfe7670c8 Mon Sep 17 00:00:00 2001 From: Ryan Howard Date: Mon, 22 Jul 2024 14:57:31 -0400 Subject: [PATCH 07/50] chore(hardware-testing): Add lpc protocols for lld testing (#15710) # Overview Adds protocols so we can run tests on other pipettes, and also fixes an issue with the 96 channel by slowing down the plunger enough so it doesn't stall # Test Plan # Changelog # Review requests # Risk assessment --- api/src/opentrons/hardware_control/ot3api.py | 8 ++++ .../hardware_testing/liquid_sense/__main__.py | 18 ++++---- .../liquid_sense_ot3_p1000_96_1well.py | 41 +++++++++++++++++++ .../liquid_sense_ot3_p1000_multi_12well.py | 32 +++++++++++++++ ...> liquid_sense_ot3_p1000_single_96well.py} | 2 +- ...y => liquid_sense_ot3_p50_multi_12well.py} | 3 +- ... => liquid_sense_ot3_p50_single_96well.py} | 4 +- 7 files changed, 96 insertions(+), 12 deletions(-) create mode 100644 hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_1well.py create mode 100644 hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_multi_12well.py rename hardware-testing/hardware_testing/protocols/liquid_sense_lpc/{liquid_sense_ot3_p1000_96_well.py => liquid_sense_ot3_p1000_single_96well.py} (93%) rename hardware-testing/hardware_testing/protocols/liquid_sense_lpc/{liquid_sense_ot3_p50_multi.py => liquid_sense_ot3_p50_multi_12well.py} (93%) rename hardware-testing/hardware_testing/protocols/liquid_sense_lpc/{liquid_sense_ot3_p50_single_vial.py => liquid_sense_ot3_p50_single_96well.py} (88%) diff --git a/api/src/opentrons/hardware_control/ot3api.py b/api/src/opentrons/hardware_control/ot3api.py index 6d567b7a667..ab3f94e232f 100644 --- a/api/src/opentrons/hardware_control/ot3api.py +++ b/api/src/opentrons/hardware_control/ot3api.py @@ -2631,6 +2631,14 @@ async def liquid_probe( ) max_speeds = self.config.motion_settings.default_max_speed p_prep_speed = max_speeds[self.gantry_load][OT3AxisKind.P] + # We need to significatly slow down the 96 channel liquid probe + if self.gantry_load == GantryLoad.HIGH_THROUGHPUT: + max_plunger_speed = self.config.motion_settings.max_speed_discontinuity[ + GantryLoad.HIGH_THROUGHPUT + ][OT3AxisKind.P] + probe_settings.plunger_speed = min( + max_plunger_speed, probe_settings.plunger_speed + ) error: Optional[PipetteLiquidNotFoundError] = None pos = await self.gantry_position(checked_mount, refresh=True) diff --git a/hardware-testing/hardware_testing/liquid_sense/__main__.py b/hardware-testing/hardware_testing/liquid_sense/__main__.py index ca0b632290c..f17c08677fd 100644 --- a/hardware-testing/hardware_testing/liquid_sense/__main__.py +++ b/hardware-testing/hardware_testing/liquid_sense/__main__.py @@ -37,9 +37,11 @@ from .post_process import process_csv_directory, process_google_sheet from hardware_testing.protocols.liquid_sense_lpc import ( - liquid_sense_ot3_p50_single_vial, - liquid_sense_ot3_p1000_96_well, - liquid_sense_ot3_p50_multi, + liquid_sense_ot3_p50_single_96well, + liquid_sense_ot3_p1000_96_1well, + liquid_sense_ot3_p1000_single_96well, + liquid_sense_ot3_p50_multi_12well, + liquid_sense_ot3_p1000_multi_12well, ) try: @@ -71,13 +73,13 @@ LIQUID_SENSE_CFG: Dict[int, Dict[int, Any]] = { 50: { - 1: liquid_sense_ot3_p50_single_vial, - 8: liquid_sense_ot3_p50_multi, + 1: liquid_sense_ot3_p50_single_96well, + 8: liquid_sense_ot3_p50_multi_12well, }, 1000: { - 1: liquid_sense_ot3_p1000_96_well, - 8: None, - 96: None, + 1: liquid_sense_ot3_p1000_single_96well, + 8: liquid_sense_ot3_p1000_multi_12well, + 96: liquid_sense_ot3_p1000_96_1well, }, } diff --git a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_1well.py b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_1well.py new file mode 100644 index 00000000000..ae89b4550a7 --- /dev/null +++ b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_1well.py @@ -0,0 +1,41 @@ +"""lld OT3 P1000.""" +from opentrons.protocol_api import ProtocolContext, OFF_DECK + +metadata = {"protocolName": "liquid-sense-ot3-p1000-96"} +requirements = {"robotType": "Flex", "apiLevel": "2.17"} + +SLOT_SCALE = 1 +SLOT_DIAL = 9 + +SLOTS_TIPRACK = { + 50: [2, 3, 4, 5, 6], + 200: [2, 3, 4, 5, 6], + 1000: [2, 3, 4, 5, 6], +} + +LABWARE_ON_SCALE = "nest_1_reservoir_195ml" + + +def run(ctx: ProtocolContext) -> None: + """Run.""" + trash = ctx.load_trash_bin("A3") + vial = ctx.load_labware(LABWARE_ON_SCALE, SLOT_SCALE) + dial = ctx.load_labware("dial_indicator", SLOT_DIAL) + pipette = ctx.load_instrument("flex_96channel_1000", "left") + adapters = [ + ctx.load_adapter("opentrons_flex_96_tiprack_adapter", slot) + for slot in SLOTS_TIPRACK[50] + ] + for size, slots in SLOTS_TIPRACK.items(): + tipracks = [ + adapter.load_labware(f"opentrons_flex_96_tiprack_{size}uL") + for adapter in adapters + ] + for rack in tipracks: + pipette.pick_up_tip(rack["A1"]) + pipette.aspirate(10, vial["A1"].top()) + pipette.dispense(10, vial["A1"].top()) + pipette.aspirate(10, dial["A1"].top()) + pipette.dispense(10, dial["A1"].top()) + pipette.drop_tip(trash) + ctx.move_labware(rack, OFF_DECK) diff --git a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_multi_12well.py b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_multi_12well.py new file mode 100644 index 00000000000..b77461598c8 --- /dev/null +++ b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_multi_12well.py @@ -0,0 +1,32 @@ +"""Liquid Sense OT3 P1000.""" +from opentrons.protocol_api import ProtocolContext, OFF_DECK + +metadata = {"protocolName": "liquid-sense-ot3-p50-multi"} +requirements = {"robotType": "Flex", "apiLevel": "2.15"} + +SLOT_SCALE = 1 +SLOT_DIAL = 9 +SLOTS_TIPRACK = { + 50: [3], + 200: [3], + 1000: [3], +} +LABWARE_ON_SCALE = "nest_12_reservoir_15ml" + + +def run(ctx: ProtocolContext) -> None: + """Run.""" + trash = ctx.load_trash_bin("A3") + vial = ctx.load_labware(LABWARE_ON_SCALE, SLOT_SCALE) + dial = ctx.load_labware("dial_indicator", SLOT_DIAL) + pipette = ctx.load_instrument("flex_8channel_1000", "left") + for size, slots in SLOTS_TIPRACK.items(): + for slot in slots: + rack = ctx.load_labware(f"opentrons_flex_96_tiprack_{size}uL", slot) + pipette.pick_up_tip(rack["A1"]) + pipette.aspirate(10, vial["A1"].top()) + pipette.dispense(10, vial["A1"].top()) + pipette.aspirate(10, dial["A1"].top()) + pipette.dispense(10, dial["A1"].top()) + pipette.drop_tip(trash) + ctx.move_labware(rack, OFF_DECK) diff --git a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_well.py b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_single_96well.py similarity index 93% rename from hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_well.py rename to hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_single_96well.py index 306abe2d48d..af96858af57 100644 --- a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_96_well.py +++ b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p1000_single_96well.py @@ -1,7 +1,7 @@ """Liquid Sense OT3.""" from opentrons.protocol_api import ProtocolContext, OFF_DECK -metadata = {"protocolName": "liquid-sense-ot3-p1000-single-vial"} +metadata = {"protocolName": "liquid-sense-ot3-p1000-single-96well"} requirements = {"robotType": "Flex", "apiLevel": "2.17"} SLOT_SCALE = 1 diff --git a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi.py b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi_12well.py similarity index 93% rename from hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi.py rename to hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi_12well.py index 69d571f0259..455565001cf 100644 --- a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi.py +++ b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_multi_12well.py @@ -12,6 +12,7 @@ def run(ctx: ProtocolContext) -> None: """Run.""" + trash = ctx.load_trash_bin("A3") tipracks = [ ctx.load_labware(f"opentrons_flex_96_tiprack_{size}uL", slot) for size, slots in SLOTS_TIPRACK.items() @@ -26,4 +27,4 @@ def run(ctx: ProtocolContext) -> None: pipette.dispense(10, vial["A1"].top()) pipette.aspirate(1, dial["A1"].top()) pipette.dispense(1, dial["A1"].top()) - pipette.drop_tip(home_after=False) + pipette.drop_tip(trash) diff --git a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_vial.py b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_96well.py similarity index 88% rename from hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_vial.py rename to hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_96well.py index 7a7e607d08e..9b597074a34 100644 --- a/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_vial.py +++ b/hardware-testing/hardware_testing/protocols/liquid_sense_lpc/liquid_sense_ot3_p50_single_96well.py @@ -1,7 +1,7 @@ """Liquid Sense OT3.""" from opentrons.protocol_api import ProtocolContext, OFF_DECK -metadata = {"protocolName": "liquid-sense-ot3-p50-single-vial"} +metadata = {"protocolName": "liquid-sense-ot3-p50-single-96well"} requirements = {"robotType": "Flex", "apiLevel": "2.17"} SLOT_SCALE = 1 @@ -9,7 +9,7 @@ SLOTS_TIPRACK = { 50: [3], } -LABWARE_ON_SCALE = "radwag_pipette_calibration_vial" +LABWARE_ON_SCALE = "corning_96_wellplate_360ul_flat" def run(ctx: ProtocolContext) -> None: From 305f00ec042a55aa55f7c3096d5766f87be1c011 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Mon, 22 Jul 2024 14:24:38 -0500 Subject: [PATCH 08/50] fix(scripts): setuptools v71 removed the need for extern (#15717) (#15741) ## `cherry-pick` afaa7177fe Must update `chore_release-7.4.0` so that Linux app builds work due to a change in setuptools. To understand the changes see #15717 Co-authored-by: Ryan Howard --- scripts/python_build_utils.py | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/scripts/python_build_utils.py b/scripts/python_build_utils.py index d55ece0e0c8..eff17678fda 100644 --- a/scripts/python_build_utils.py +++ b/scripts/python_build_utils.py @@ -54,9 +54,18 @@ def normalize_version(package, project, extra_tag='', git_dir=None): # the way they vendor dependencies, like the packaging module that # provides the way to normalize version numbers for wheel file names. So # we try all the possible ways to find it. + # Since 71.0.0 they have removed the need for extern + # So depending on the version of 3.10 you're building on you may or may not + # need to use the extern or import it directly try: - # new way - from setuptools.extern import packaging + import setuptools + major, minor, patch = [int(x, 10) for x in setuptools.__version__.split('.')] + if major < 71: + # new way + from setuptools.extern import packaging + else: + # new new way + import packaging except ImportError: # old way from pkg_resources.extern import packaging From 80aa14e9a6ae8d9a55c3584d26d2371121de5758 Mon Sep 17 00:00:00 2001 From: Ryan Howard Date: Mon, 22 Jul 2024 16:14:05 -0400 Subject: [PATCH 09/50] fix(api): Lld movement adjustments (#15732) # Overview In order to remove some magic numbers we do a more clear calculation about where certain numbers in ot3api.py come from additionally we added a more robust move_to_plunger_top that incorporates backlash values, mostly due to an expectation that the 96 channel will need this due to it's current physical requirements for backlash # Test Plan # Changelog # Review requests # Risk assessment --------- Co-authored-by: Andy Sigler --- .../backends/flex_protocol.py | 1 + .../backends/ot3controller.py | 2 + .../hardware_control/backends/ot3simulator.py | 1 + api/src/opentrons/hardware_control/ot3api.py | 103 ++++++++++++++---- .../backends/test_ot3_controller.py | 1 + .../hardware_control/test_ot3_api.py | 19 ++-- .../hardware_control/tool_sensors.py | 7 +- .../hardware_control/test_tool_sensors.py | 2 + 8 files changed, 101 insertions(+), 35 deletions(-) diff --git a/api/src/opentrons/hardware_control/backends/flex_protocol.py b/api/src/opentrons/hardware_control/backends/flex_protocol.py index 6e96f3f3485..9e7218099cc 100644 --- a/api/src/opentrons/hardware_control/backends/flex_protocol.py +++ b/api/src/opentrons/hardware_control/backends/flex_protocol.py @@ -146,6 +146,7 @@ async def liquid_probe( mount_speed: float, plunger_speed: float, threshold_pascals: float, + plunger_impulse_time: float, output_format: OutputOptions = OutputOptions.can_bus_only, data_files: Optional[Dict[InstrumentProbeType, str]] = None, probe: InstrumentProbeType = InstrumentProbeType.PRIMARY, diff --git a/api/src/opentrons/hardware_control/backends/ot3controller.py b/api/src/opentrons/hardware_control/backends/ot3controller.py index e23cbcdd8c3..cd6aa9e112a 100644 --- a/api/src/opentrons/hardware_control/backends/ot3controller.py +++ b/api/src/opentrons/hardware_control/backends/ot3controller.py @@ -1357,6 +1357,7 @@ async def liquid_probe( mount_speed: float, plunger_speed: float, threshold_pascals: float, + plunger_impulse_time: float, output_option: OutputOptions = OutputOptions.can_bus_only, data_files: Optional[Dict[InstrumentProbeType, str]] = None, probe: InstrumentProbeType = InstrumentProbeType.PRIMARY, @@ -1387,6 +1388,7 @@ async def liquid_probe( plunger_speed=plunger_speed, mount_speed=mount_speed, threshold_pascals=threshold_pascals, + plunger_impulse_time=plunger_impulse_time, csv_output=csv_output, sync_buffer_output=sync_buffer_output, can_bus_only_output=can_bus_only_output, diff --git a/api/src/opentrons/hardware_control/backends/ot3simulator.py b/api/src/opentrons/hardware_control/backends/ot3simulator.py index 8e3a7f8990c..34c8fe0df68 100644 --- a/api/src/opentrons/hardware_control/backends/ot3simulator.py +++ b/api/src/opentrons/hardware_control/backends/ot3simulator.py @@ -345,6 +345,7 @@ async def liquid_probe( mount_speed: float, plunger_speed: float, threshold_pascals: float, + plunger_impulse_time: float, output_format: OutputOptions = OutputOptions.can_bus_only, data_files: Optional[Dict[InstrumentProbeType, str]] = None, probe: InstrumentProbeType = InstrumentProbeType.PRIMARY, diff --git a/api/src/opentrons/hardware_control/ot3api.py b/api/src/opentrons/hardware_control/ot3api.py index ab3f94e232f..cdc95bdd7de 100644 --- a/api/src/opentrons/hardware_control/ot3api.py +++ b/api/src/opentrons/hardware_control/ot3api.py @@ -1932,6 +1932,37 @@ async def _move_to_plunger_bottom( acquire_lock=acquire_lock, ) + async def _move_to_plunger_top_for_liquid_probe( + self, + mount: OT3Mount, + rate: float, + acquire_lock: bool = True, + ) -> None: + """ + Move an instrument's plunger to the top, to prepare for a following + liquid probe action. + + The plunger backlash distance (mm) is used to ensure the plunger is pre-loaded + in the downward direction. This means that the final position will not be + the plunger's configured "top" position, but "top" plus the "backlashDistance". + """ + max_speeds = self.config.motion_settings.default_max_speed + speed = max_speeds[self.gantry_load][OT3AxisKind.P] + instrument = self._pipette_handler.get_pipette(mount) + top_plunger_pos = target_position_from_plunger( + OT3Mount.from_mount(mount), + instrument.plunger_positions.top, + self._current_position, + ) + target_pos = top_plunger_pos.copy() + target_pos[Axis.of_main_tool_actuator(mount)] += instrument.backlash_distance + await self._move(top_plunger_pos, speed=speed * rate, acquire_lock=acquire_lock) + # NOTE: This should ALWAYS be moving DOWN. + # There should never be a time that this function is called and + # the plunger doesn't physically move DOWN. + # This is to make sure we are always engaged at the beginning of liquid-probe. + await self._move(target_pos, speed=speed * rate, acquire_lock=acquire_lock) + async def configure_for_volume( self, mount: Union[top_types.Mount, OT3Mount], volume: float ) -> None: @@ -2568,6 +2599,21 @@ def add_gripper_probe(self, probe: GripperProbe) -> None: def remove_gripper_probe(self) -> None: self._gripper_handler.remove_probe() + @staticmethod + def liquid_probe_non_responsive_z_distance(z_speed: float) -> float: + """Calculate the Z distance travelled where the LLD pass will be unresponsive.""" + # NOTE: (sigler) Here lye some magic numbers. + # The Z axis probing motion uses the first 20 samples to calculate + # a baseline for all following samples, making the very beginning of + # that Z motion unable to detect liquid. The sensor is configured for + # 4ms sample readings, and so we then assume it takes ~80ms to complete. + # If the Z is moving at 5mm/sec, then ~80ms equates to ~0.4 + baseline_during_z_sample_num = 20 # FIXME: (sigler) shouldn't be defined here? + sample_time_sec = 0.004 # FIXME: (sigler) shouldn't be defined here? + baseline_duration_sec = baseline_during_z_sample_num * sample_time_sec + non_responsive_z_mm = baseline_duration_sec * z_speed + return non_responsive_z_mm + async def _liquid_probe_pass( self, mount: OT3Mount, @@ -2583,6 +2629,7 @@ async def _liquid_probe_pass( probe_settings.mount_speed, (probe_settings.plunger_speed * plunger_direction), probe_settings.sensor_threshold_pascals, + probe_settings.plunger_impulse_time, probe_settings.output_option, probe_settings.data_files, probe=probe, @@ -2626,11 +2673,15 @@ async def liquid_probe( probe_start_pos = await self.gantry_position(checked_mount, refresh=True) - p_travel = ( + # plunger travel distance is from TOP->BOTTOM (minus the backlash distance + impulse) + # FIXME: logic for how plunger moves is divided between here and tool_sensors.py + p_impulse_mm = ( + probe_settings.plunger_impulse_time * probe_settings.plunger_speed + ) + p_total_mm = ( instrument.plunger_positions.bottom - instrument.plunger_positions.top ) - max_speeds = self.config.motion_settings.default_max_speed - p_prep_speed = max_speeds[self.gantry_load][OT3AxisKind.P] + # We need to significatly slow down the 96 channel liquid probe if self.gantry_load == GantryLoad.HIGH_THROUGHPUT: max_plunger_speed = self.config.motion_settings.max_speed_discontinuity[ @@ -2640,24 +2691,44 @@ async def liquid_probe( max_plunger_speed, probe_settings.plunger_speed ) + p_working_mm = p_total_mm - (instrument.backlash_distance + p_impulse_mm) + + # height where probe action will begin + # TODO: (sigler) add this to pipette's liquid def (per tip) + probe_pass_overlap_mm = 0.1 + non_responsive_z_mm = OT3API.liquid_probe_non_responsive_z_distance( + probe_settings.mount_speed + ) + probe_pass_z_offset_mm = non_responsive_z_mm + probe_pass_overlap_mm + + # height that is considered safe to reset the plunger without disturbing liquid + # this usually needs to at least 1-2mm from liquid, to avoid splashes from air + # TODO: (sigler) add this to pipette's liquid def (per tip) + probe_safe_reset_mm = max(2.0, probe_pass_z_offset_mm) + error: Optional[PipetteLiquidNotFoundError] = None pos = await self.gantry_position(checked_mount, refresh=True) while (probe_start_pos.z - pos.z) < max_z_dist: # safe distance so we don't accidentally aspirate liquid if we're already close to liquid - safe_plunger_pos = top_types.Point(pos.x, pos.y, pos.z + 2) + safe_plunger_pos = top_types.Point( + pos.x, pos.y, pos.z + probe_safe_reset_mm + ) # overlap amount we want to use between passes - pass_start_pos = top_types.Point(pos.x, pos.y, pos.z + 0.5) + pass_start_pos = top_types.Point( + pos.x, pos.y, pos.z + probe_pass_z_offset_mm + ) max_z_time = ( max_z_dist - (probe_start_pos.z - safe_plunger_pos.z) ) / probe_settings.mount_speed - pass_travel = min(max_z_time * probe_settings.plunger_speed, p_travel) + p_travel_required_for_z = max_z_time * probe_settings.plunger_speed + p_pass_travel = min(p_travel_required_for_z, p_working_mm) # Prep the plunger await self.move_to(checked_mount, safe_plunger_pos) if probe_settings.aspirate_while_sensing: # TODO(cm, 7/8/24): remove p_prep_speed from the rate at some point - await self._move_to_plunger_bottom(checked_mount, rate=p_prep_speed) + await self._move_to_plunger_bottom(checked_mount, rate=1) else: - await self._move_to_plunger_top(checked_mount, rate=p_prep_speed) + await self._move_to_plunger_top_for_liquid_probe(checked_mount, rate=1) try: # move to where we want to start a pass and run a pass @@ -2666,7 +2737,7 @@ async def liquid_probe( checked_mount, probe_settings, probe if probe else InstrumentProbeType.PRIMARY, - pass_travel, + p_pass_travel + p_impulse_mm, ) # if we made it here without an error we found the liquid error = None @@ -2682,20 +2753,6 @@ async def liquid_probe( raise error return height - async def _move_to_plunger_top( - self, - mount: OT3Mount, - rate: float, - acquire_lock: bool = True, - ) -> None: - instrument = self._pipette_handler.get_pipette(mount) - target_pos = target_position_from_plunger( - OT3Mount.from_mount(mount), - instrument.plunger_positions.top, - self._current_position, - ) - await self._move(target_pos, speed=rate, acquire_lock=acquire_lock) - async def capacitive_probe( self, mount: OT3Mount, diff --git a/api/tests/opentrons/hardware_control/backends/test_ot3_controller.py b/api/tests/opentrons/hardware_control/backends/test_ot3_controller.py index ae5385ff1f9..fa57c4347ff 100644 --- a/api/tests/opentrons/hardware_control/backends/test_ot3_controller.py +++ b/api/tests/opentrons/hardware_control/backends/test_ot3_controller.py @@ -719,6 +719,7 @@ async def test_liquid_probe( mount_speed=fake_liquid_settings.mount_speed, plunger_speed=fake_liquid_settings.plunger_speed, threshold_pascals=fake_liquid_settings.sensor_threshold_pascals, + plunger_impulse_time=fake_liquid_settings.plunger_impulse_time, output_option=fake_liquid_settings.output_option, ) except PipetteLiquidNotFoundError: diff --git a/api/tests/opentrons/hardware_control/test_ot3_api.py b/api/tests/opentrons/hardware_control/test_ot3_api.py index 2b77ebdcd00..0c1fff849c0 100644 --- a/api/tests/opentrons/hardware_control/test_ot3_api.py +++ b/api/tests/opentrons/hardware_control/test_ot3_api.py @@ -116,8 +116,8 @@ def fake_settings() -> CapacitivePassSettings: @pytest.fixture def fake_liquid_settings() -> LiquidProbeSettings: return LiquidProbeSettings( - mount_speed=40, - plunger_speed=10, + mount_speed=5, + plunger_speed=20, plunger_impulse_time=0.2, sensor_threshold_pascals=15, output_option=OutputOptions.can_bus_only, @@ -825,8 +825,8 @@ async def test_liquid_probe( # make sure aspirate while sensing reverses direction mock_liquid_probe.return_value = return_dict fake_settings_aspirate = LiquidProbeSettings( - mount_speed=40, - plunger_speed=10, + mount_speed=5, + plunger_speed=20, plunger_impulse_time=0.2, sensor_threshold_pascals=15, output_option=OutputOptions.can_bus_only, @@ -838,10 +838,11 @@ async def test_liquid_probe( mock_move_to_plunger_bottom.call_count == 2 mock_liquid_probe.assert_called_once_with( mount, - 3.0, + 52, fake_settings_aspirate.mount_speed, (fake_settings_aspirate.plunger_speed * -1), fake_settings_aspirate.sensor_threshold_pascals, + fake_settings_aspirate.plunger_impulse_time, fake_settings_aspirate.output_option, fake_settings_aspirate.data_files, probe=InstrumentProbeType.PRIMARY, @@ -913,10 +914,11 @@ async def test_multi_liquid_probe( assert mock_move_to_plunger_bottom.call_count == 4 mock_liquid_probe.assert_called_with( OT3Mount.LEFT, - plunger_positions.bottom - plunger_positions.top, + plunger_positions.bottom - plunger_positions.top - 0.1, fake_settings_aspirate.mount_speed, (fake_settings_aspirate.plunger_speed * -1), fake_settings_aspirate.sensor_threshold_pascals, + fake_settings_aspirate.plunger_impulse_time, fake_settings_aspirate.output_option, fake_settings_aspirate.data_files, probe=InstrumentProbeType.PRIMARY, @@ -954,6 +956,7 @@ async def _fake_pos_update_and_raise( mount_speed: float, plunger_speed: float, threshold_pascals: float, + plunger_impulse_time: float, output_format: OutputOptions = OutputOptions.can_bus_only, data_files: Optional[Dict[InstrumentProbeType, str]] = None, probe: InstrumentProbeType = InstrumentProbeType.PRIMARY, @@ -986,8 +989,8 @@ async def _fake_pos_update_and_raise( await ot3_hardware.liquid_probe( OT3Mount.LEFT, fake_max_z_dist, fake_settings_aspirate ) - # assert that it went through 3 passes and then prepared to aspirate - assert mock_move_to_plunger_bottom.call_count == 4 + # assert that it went through 4 passes and then prepared to aspirate + assert mock_move_to_plunger_bottom.call_count == 5 @pytest.mark.parametrize( diff --git a/hardware/opentrons_hardware/hardware_control/tool_sensors.py b/hardware/opentrons_hardware/hardware_control/tool_sensors.py index 6b762ef7c30..eeb4736a6d9 100644 --- a/hardware/opentrons_hardware/hardware_control/tool_sensors.py +++ b/hardware/opentrons_hardware/hardware_control/tool_sensors.py @@ -82,8 +82,6 @@ # FIXME we should organize all of these functions to use the sensor drivers. # FIXME we should restrict some of these functions by instrument type. -PLUNGER_SOLO_MOVE_TIME = 0.2 - def _fix_pass_step_for_buffer( move_group: MoveGroupStep, @@ -393,6 +391,7 @@ async def liquid_probe( plunger_speed: float, mount_speed: float, threshold_pascals: float, + plunger_impulse_time: float, csv_output: bool = False, sync_buffer_output: bool = False, can_bus_only_output: bool = False, @@ -425,7 +424,7 @@ async def liquid_probe( sensor_driver, True, ) - p_prep_distance = float(PLUNGER_SOLO_MOVE_TIME * plunger_speed) + p_prep_distance = float(plunger_impulse_time * plunger_speed) p_pass_distance = float(max_p_distance - p_prep_distance) max_z_distance = (p_pass_distance / plunger_speed) * mount_speed @@ -433,7 +432,7 @@ async def liquid_probe( distance={tool: float64(p_prep_distance)}, velocity={tool: float64(plunger_speed)}, acceleration={}, - duration=float64(PLUNGER_SOLO_MOVE_TIME), + duration=float64(plunger_impulse_time), present_nodes=[tool], ) sensor_group = _build_pass_step( diff --git a/hardware/tests/opentrons_hardware/hardware_control/test_tool_sensors.py b/hardware/tests/opentrons_hardware/hardware_control/test_tool_sensors.py index 86a1d2d40a7..f4dddc8ca37 100644 --- a/hardware/tests/opentrons_hardware/hardware_control/test_tool_sensors.py +++ b/hardware/tests/opentrons_hardware/hardware_control/test_tool_sensors.py @@ -217,6 +217,7 @@ def move_responder( mount_speed=10, plunger_speed=8, threshold_pascals=threshold_pascals, + plunger_impulse_time=0.2, csv_output=False, sync_buffer_output=False, can_bus_only_output=False, @@ -348,6 +349,7 @@ def move_responder( mount_speed=10, plunger_speed=8, threshold_pascals=14, + plunger_impulse_time=0.2, csv_output=csv_output, sync_buffer_output=sync_buffer_output, can_bus_only_output=can_bus_only_output, From 043bca7aec2a0279d91b8e5b81273454e6dfbabb Mon Sep 17 00:00:00 2001 From: Shlok Amin Date: Mon, 1 Jul 2024 14:43:17 -0400 Subject: [PATCH 10/50] ci(app-shell): update macos github runner version (#15559) --- .github/workflows/app-test-build-deploy.yaml | 6 +++--- app-shell/Makefile | 3 --- 2 files changed, 3 insertions(+), 6 deletions(-) diff --git a/.github/workflows/app-test-build-deploy.yaml b/.github/workflows/app-test-build-deploy.yaml index 4cc3754ace6..79a234eba14 100644 --- a/.github/workflows/app-test-build-deploy.yaml +++ b/.github/workflows/app-test-build-deploy.yaml @@ -96,7 +96,7 @@ jobs: backend-unit-test: strategy: matrix: - os: ['windows-2022', 'ubuntu-22.04', 'macos-11'] + os: ['windows-2022', 'ubuntu-22.04', 'macos-latest'] name: 'opentrons app backend unit tests on ${{matrix.os}}' runs-on: ${{ matrix.os }} steps: @@ -228,13 +228,13 @@ jobs: if: needs.determine-build-type.outputs.variants != '[]' strategy: matrix: - os: ['windows-2022', 'ubuntu-22.04', 'macos-11'] + os: ['windows-2022', 'ubuntu-22.04', 'macos-latest'] variant: ${{fromJSON(needs.determine-build-type.outputs.variants)}} target: ['desktop', 'odd'] exclude: - os: 'windows-2022' target: 'odd' - - os: 'macos-11' + - os: 'macos-latest' target: 'odd' runs-on: ${{ matrix.os }} diff --git a/app-shell/Makefile b/app-shell/Makefile index ec86ee924ff..32c8b7ce01e 100644 --- a/app-shell/Makefile +++ b/app-shell/Makefile @@ -145,9 +145,6 @@ dist-ot3: package-deps .PHONY: dist-macos-latest dist-macos-latest: dist-osx -.PHONY: dist-macos-11 -dist-macos-11: dist-osx - .PHONY: dist-ubuntu-latest dist-ubuntu-latest: dist-linux From a77ffd7a091e3c1a603ea0bcf9c67ecda94bb9a5 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Thu, 11 Jul 2024 14:32:05 -0500 Subject: [PATCH 11/50] RQA-2830 fix mac dmg to be x64 binary and use x64 bundled python to pip install (#15624) ## Replaces #15618 with PR against branch name ending in `app-build-internal` to trigger a build ## Overview - - `macos-11` GitHub runner was intel and now we use macos-latest that is arm [doc](https://docs.github.com/en/actions/using-github-hosted-runners/about-github-hosted-runners/about-github-hosted-runners#standard-github-hosted-runners-for-public-repositories) - specify --x64 argument to electron-builder - ~~per [doc](https://github.com/actions/setup-python?tab=readme-ov-file#supported-architectures) x64 python should be installed so the pip install will be the right platform. I'm not confident in this.~~ - see comment in #15618 - we need to use the python we download on mac to do the pip install so the installed packages match the platform correctly ## Test Plan - [x] build locally and see that it works - [x] watch CI and install the built package - [x] validate that once the package dmg is x64 that analysis works (python has packages with the right arch) # Review requests - This the right approach? # Risk assessment High because our .dmg currently cannot analyze protocols 2.0.0-alpha.0 internal release build. --- app-shell/Makefile | 2 +- app-shell/scripts/before-pack.js | 11 ++++++++++- 2 files changed, 11 insertions(+), 2 deletions(-) diff --git a/app-shell/Makefile b/app-shell/Makefile index 32c8b7ce01e..9624aa1a54e 100644 --- a/app-shell/Makefile +++ b/app-shell/Makefile @@ -122,7 +122,7 @@ dist-posix: package-deps .PHONY: dist-osx dist-osx: package-deps - $(builder) --mac + $(builder) --mac --x64 $(MAKE) _dist-collect-artifacts .PHONY: dist-linux diff --git a/app-shell/scripts/before-pack.js b/app-shell/scripts/before-pack.js index 6e96609786f..bb5e2d45485 100644 --- a/app-shell/scripts/before-pack.js +++ b/app-shell/scripts/before-pack.js @@ -118,7 +118,16 @@ module.exports = function beforeBuild(context) { // TODO(mc, 2022-05-16): explore virtualenvs for a more reliable // implementation of this install - return execa(HOST_PYTHON, [ + console.log( + `Installing python native deps using ${path.join( + PYTHON_DESTINATION, + 'python3.10' + )}` + ) + const invokablePython = platformName.includes('darwin') + ? path.join(PYTHON_DESTINATION, 'python/bin/python3.10') + : HOST_PYTHON + return execa(invokablePython, [ '-m', 'pip', 'install', From 6b8f7fd121bca40e300fbad7da4226c30ae1f1c7 Mon Sep 17 00:00:00 2001 From: Seth Foster Date: Tue, 23 Jul 2024 09:54:48 -0400 Subject: [PATCH 12/50] fix(app): overpressure while aspirate styling (#15748) - Fix up styles for the overpressure while aspirating screens Closes EXEC-558 --- .../localization/en/error_recovery.json | 4 +- app/src/molecules/Command/Command.tsx | 39 ++++++++----------- app/src/molecules/Command/CommandText.tsx | 5 +-- .../ErrorRecoveryFlows/shared/ReplaceTips.tsx | 8 ++-- 4 files changed, 26 insertions(+), 30 deletions(-) diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index 53d2a07df43..eeaf8980cc8 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -41,8 +41,8 @@ "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.", "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}}", - "replace_with_new_tip_rack": "Replace with new tip rack", + "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}}", "resume": "Resume", "retry_now": "Retry now", "retry_step": "Retry step", diff --git a/app/src/molecules/Command/Command.tsx b/app/src/molecules/Command/Command.tsx index b1c2c935eb7..fb8452f2a92 100644 --- a/app/src/molecules/Command/Command.tsx +++ b/app/src/molecules/Command/Command.tsx @@ -164,15 +164,11 @@ export function CenteredCommand(
@@ -224,15 +220,11 @@ export function LeftAlignedCommand(
@@ -242,14 +234,17 @@ export function LeftAlignedCommand( const TEXT_CLIP_STYLE = ` display: -webkit-box; - -webkit-box-orient: vertical; - overflow: hidden; - text-overflow: ellipsis; - word-wrap: break-word; - -webkit-line-clamp: 2; -} + -webkit-box-orient: vertical; + overflow: hidden; + text-overflow: ellipsis; + word-wrap: break-word; + -webkit-line-clamp: 2; ` const ODD_ONLY_TEXT_CLIP_STYLE = ` + @media not (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { + max-height: 240px; + overflow: auto; + } @media (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { ${TEXT_CLIP_STYLE} } diff --git a/app/src/molecules/Command/CommandText.tsx b/app/src/molecules/Command/CommandText.tsx index f579c41ea8d..75b12733eca 100644 --- a/app/src/molecules/Command/CommandText.tsx +++ b/app/src/molecules/Command/CommandText.tsx @@ -9,7 +9,6 @@ import { LegacyStyledText, StyledText, RESPONSIVENESS, - styleProps, } from '@opentrons/components' import { useCommandTextString } from './hooks' @@ -78,14 +77,14 @@ function CommandStyledText( {props.children} ) } else { return ( - + {props.children} ) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx index f7513af14c8..9d7f8adfcd7 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/ReplaceTips.tsx @@ -7,6 +7,7 @@ import { RecoverySingleColumnContentWrapper } from './RecoveryContentWrapper' import { TwoColumn, DeckMapContent } from '../../../molecules/InterventionModal' import { RecoveryFooterButtons } from './RecoveryFooterButtons' import { LeftColumnLabwareInfo } from './LeftColumnLabwareInfo' +import { getSlotNameAndLwLocFrom } from '../hooks/useDeckMapUtils' import type { RecoveryContentProps } from '../types' @@ -17,20 +18,21 @@ export function ReplaceTips(props: RecoveryContentProps): JSX.Element | null { failedLabwareUtils, deckMapUtils, } = props - const { relevantWellName } = failedLabwareUtils + const { relevantWellName, failedLabware } = failedLabwareUtils const { proceedNextStep } = routeUpdateActions const { t } = useTranslation('error_recovery') const primaryOnClick = (): void => { void proceedNextStep() } - + const [slot] = getSlotNameAndLwLocFrom(failedLabware?.location ?? null, false) const buildTitle = (): string => { if (failedPipetteInfo?.data.channels === 96) { - return t('replace_with_new_tip_rack') + return t('replace_with_new_tip_rack', { slot }) } else { return t('replace_used_tips_in_rack_location', { location: relevantWellName, + slot, }) } } From 325c86781a780c95126f3227bef213334d46554f Mon Sep 17 00:00:00 2001 From: Seth Foster Date: Tue, 23 Jul 2024 11:00:12 -0400 Subject: [PATCH 13/50] build: disable sign without failing builds (#15743) Edge needs to be able to produce builds that aren't signed now. --- app-shell/scripts/windows-custom-sign.js | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/app-shell/scripts/windows-custom-sign.js b/app-shell/scripts/windows-custom-sign.js index 90d7927ab6a..f0735a50989 100644 --- a/app-shell/scripts/windows-custom-sign.js +++ b/app-shell/scripts/windows-custom-sign.js @@ -5,6 +5,10 @@ const { execSync } = require('node:child_process') exports.default = async configuration => { + const { WINDOWS_SIGN } = process.env + if (WINDOWS_SIGN !== 'true') { + return + } const signCmd = `smctl sign --keypair-alias="${String( process.env.SM_KEYPAIR_ALIAS )}" --input "${String(configuration.path)}" --certificate="${String( From 7fd60af430b2899494d6b59b5ecb259599df5ca1 Mon Sep 17 00:00:00 2001 From: Rhyann Clarke <146747548+rclarke0@users.noreply.github.com> Date: Tue, 23 Jul 2024 11:01:52 -0400 Subject: [PATCH 14/50] fix(abr-testing): error handling for mean calculation (#15751) # Overview Error Handling for statistics calculations. # Test Plan # Changelog # Review requests # Risk assessment --- .../data_collection/abr_robot_error.py | 17 ++++++++++++++--- 1 file changed, 14 insertions(+), 3 deletions(-) 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 a45e64cd86d..98af232304d 100644 --- a/abr-testing/abr_testing/data_collection/abr_robot_error.py +++ b/abr-testing/abr_testing/data_collection/abr_robot_error.py @@ -131,9 +131,20 @@ def compare_lpc_to_historical_data( current_x = round(labware_dict["X"], 2) current_y = round(labware_dict["Y"], 2) current_z = round(labware_dict["Z"], 2) - avg_x = round(mean(x_float), 2) - avg_y = round(mean(y_float), 2) - avg_z = round(mean(z_float), 2) + try: + avg_x = round(mean(x_float), 2) + avg_y = round(mean(y_float), 2) + avg_z = round(mean(z_float), 2) + except StatisticsError: + # If there is one value assign it as the average. + if len(x_float) == 1: + avg_x = x_float[0] + avg_y = y_float[0] + avg_z = z_float[0] + else: + avg_x = None + avg_y = None + avg_z = None # Formats LPC message for ticket. lpc_message = ( From 9319ae4cef7073ce9fd38c9bccda33b1228ab043 Mon Sep 17 00:00:00 2001 From: Brent Hagen Date: Tue, 23 Jul 2024 11:02:57 -0400 Subject: [PATCH 15/50] fix(app): fix ODD current run redirect whitescreen (#15753) wraps the ODD current run redirect Route in a Routes component to fix the whitescreen error "A is only ever to be used as the child of element, never rendered directly. Please wrap your in a ." Moves the TopLevelRedirects component inside of the ErrorBoundary to avoid whitescreens (though reload would still trigger the error boundary in this case) --- app/src/App/OnDeviceDisplayApp.tsx | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/app/src/App/OnDeviceDisplayApp.tsx b/app/src/App/OnDeviceDisplayApp.tsx index ae0e0a1d933..77aceabce20 100644 --- a/app/src/App/OnDeviceDisplayApp.tsx +++ b/app/src/App/OnDeviceDisplayApp.tsx @@ -201,8 +201,8 @@ export const OnDeviceDisplayApp = (): JSX.Element => { )} + - @@ -275,7 +275,9 @@ export function OnDeviceDisplayAppRoutes(): JSX.Element { function TopLevelRedirects(): JSX.Element | null { const currentRunRoute = useCurrentRunRoute() return currentRunRoute != null ? ( - } /> + + } /> + ) : null } From a2af480138d1cae609e0c9c17a879297cac677a0 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Tue, 23 Jul 2024 13:02:08 -0400 Subject: [PATCH 16/50] feat:(app) Add desktop support for Error Recovery drop tip flows (#15758) Closes EXEC-554 --- .../localization/en/drop_tip_wizard.json | 8 +- .../localization/en/error_recovery.json | 4 +- .../InProgressModal/InProgressModal.tsx | 2 +- app/src/molecules/SimpleWizardBody/index.tsx | 4 +- .../DropTipWizardFlows/BeforeBeginning.tsx | 103 +++++---- .../DropTipWizardFlows/ChooseLocation.tsx | 35 +-- .../DropTipWizardFlows/DropTipWizard.tsx | 20 +- .../DropTipWizardFlows/JogToPosition.tsx | 200 ++++++++++-------- .../organisms/DropTipWizardFlows/Success.tsx | 50 +++-- app/src/organisms/DropTipWizardFlows/types.ts | 5 + .../RecoveryOptions/CancelRun.tsx | 5 +- .../RecoveryOptions/ManageTips.tsx | 29 ++- .../__tests__/ManageTips.test.tsx | 16 +- .../shared/RecoveryInterventionModal.tsx | 15 +- 14 files changed, 288 insertions(+), 208 deletions(-) diff --git a/app/src/assets/localization/en/drop_tip_wizard.json b/app/src/assets/localization/en/drop_tip_wizard.json index 8092956f809..16a9fab24b6 100644 --- a/app/src/assets/localization/en/drop_tip_wizard.json +++ b/app/src/assets/localization/en/drop_tip_wizard.json @@ -16,6 +16,8 @@ "exit_screen_title": "Exit before completing drop tip?", "getting_ready": "Getting ready…", "go_back": "go back", + "jog_too_far": "Jog too far?", + "start_over": "Start over", "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.", @@ -25,9 +27,9 @@ "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", - "select_blowout_slot": "You can blow out liquid into a labware.Select the slot where you want to blow out the liquid on the deck map to the right. Once confirmed, the gantry will move to the chosen slot.", - "select_blowout_slot_odd": "You can blow out liquid into a labware.
After the gantry moves to the chosen slot, use the jog controls to move the pipette to the exact position for blowing out.", - "select_drop_tip_slot": "You can return tips to a tip rack or dispose of them.Select the slot where you want to drop the tips on the deck map to the right. Once confirmed, the gantry will move to the chosen slot.", + "select_blowout_slot": "Blowing out into a labware helps remove all liquid from the tip.
Select the slot where you want to blow out the liquid on the deck map to the right. Once confirmed, the gantry will move to the chosen slot.", + "select_blowout_slot_odd": "Blowing out into a labware helps remove all liquid from the tip.
After the gantry moves to the chosen slot, use the jog controls to move the pipette to the exact position for blowing out.", + "select_drop_tip_slot": "You can return tips to a tip rack or dispose of them.
Select the slot where you want to drop the tips on the deck map to the right. Once confirmed, the gantry will move to the chosen slot.", "select_drop_tip_slot_odd": "You can return tips to a tip rack or dispose of them.
After the gantry moves to the chosen slot, use the jog controls to move the pipette to the exact position for dropping tips.", "skip": "Skip", "stand_back_blowing_out": "Stand back, robot is blowing out liquid", diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index eeaf8980cc8..c139f21acd2 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -58,7 +58,7 @@ "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": "Skip", + "skip_removal": "Skip removal", "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", @@ -73,5 +73,5 @@ "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.", - "you_may_want_to_remove": "You may want to remove the tips from the {{mount}} pipette before using it again in a protocol" + "remove_tips_from_pipette": "Remove tips from {{mount}} pipette before canceling the run?" } diff --git a/app/src/molecules/InProgressModal/InProgressModal.tsx b/app/src/molecules/InProgressModal/InProgressModal.tsx index 63ed2e61365..c916f574472 100644 --- a/app/src/molecules/InProgressModal/InProgressModal.tsx +++ b/app/src/molecules/InProgressModal/InProgressModal.tsx @@ -60,7 +60,7 @@ const MODAL_STYLE = css` } ` const SPINNER_STYLE = css` - color: ${COLORS.grey50}; + color: ${COLORS.grey60}; opacity: 100%; @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { color: ${COLORS.black90}; diff --git a/app/src/molecules/SimpleWizardBody/index.tsx b/app/src/molecules/SimpleWizardBody/index.tsx index ec874876c23..4c941d73ba4 100644 --- a/app/src/molecules/SimpleWizardBody/index.tsx +++ b/app/src/molecules/SimpleWizardBody/index.tsx @@ -150,8 +150,8 @@ export function SimpleWizardBody(props: Props): JSX.Element { <> {isSuccess ? ( Success Icon diff --git a/app/src/organisms/DropTipWizardFlows/BeforeBeginning.tsx b/app/src/organisms/DropTipWizardFlows/BeforeBeginning.tsx index a71cb83f605..782b8bc6764 100644 --- a/app/src/organisms/DropTipWizardFlows/BeforeBeginning.tsx +++ b/app/src/organisms/DropTipWizardFlows/BeforeBeginning.tsx @@ -22,7 +22,7 @@ import { TYPOGRAPHY, } from '@opentrons/components' -import { SmallButton, MediumButton } from '../../atoms/buttons' +import { SmallButton, MediumButton, TextOnlyButton } from '../../atoms/buttons' import { DT_ROUTES } from './constants' import blowoutVideo from '../../assets/videos/droptip-wizard/Blowout-Liquid.webm' @@ -52,7 +52,7 @@ export const BeforeBeginning = ({ const buildTopText = (): string => { if (issuedCommandsType === 'fixit') { return fixitCommandTypeUtils?.copyOverrides - .tipDropCompleteBtnCopy as string + .beforeBeginningTopText as string } else { return t('before_you_begin_do_you_want_to_blowout') } @@ -60,52 +60,51 @@ export const BeforeBeginning = ({ if (isOnDevice) { return ( - <> - - {buildTopText()} - - { - setFlowType('blowout') - }} - buttonText={i18n.format(t('yes_blow_out_liquid'), 'capitalize')} - justifyContent={JUSTIFY_FLEX_START} - paddingLeft={SPACING.spacing24} - height="5.25rem" - /> - - - { - setFlowType('drop_tips') - }} - buttonText={i18n.format( - t('no_proceed_to_drop_tip'), - 'capitalize' - )} - justifyContent={JUSTIFY_FLEX_START} - paddingLeft={SPACING.spacing24} - height="5.25rem" - /> - - - - + + {buildTopText()} + + { + setFlowType('blowout') + }} + buttonText={i18n.format(t('yes_blow_out_liquid'), 'capitalize')} + justifyContent={JUSTIFY_FLEX_START} + paddingLeft={SPACING.spacing24} + height="5.25rem" + /> - + + { + setFlowType('drop_tips') + }} + buttonText={i18n.format(t('no_proceed_to_drop_tip'), 'capitalize')} + justifyContent={JUSTIFY_FLEX_START} + paddingLeft={SPACING.spacing24} + height="5.25rem" + /> + + + + + ) } else { return ( - {t('before_you_begin_do_you_want_to_blowout')} + {buildTopText()} - + {/* */} + {fixitCommandTypeUtils != null ? ( + + ) : null} {i18n.format(t('shared:continue'), 'capitalize')} @@ -242,6 +256,7 @@ const TILE_CONTAINER_STYLE = css` flex-direction: ${DIRECTION_COLUMN}; justify-content: ${JUSTIFY_SPACE_BETWEEN}; padding: ${SPACING.spacing32}; + height: 100%; @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { height: 29.5rem; } diff --git a/app/src/organisms/DropTipWizardFlows/ChooseLocation.tsx b/app/src/organisms/DropTipWizardFlows/ChooseLocation.tsx index e1c284efc4c..18766553999 100644 --- a/app/src/organisms/DropTipWizardFlows/ChooseLocation.tsx +++ b/app/src/organisms/DropTipWizardFlows/ChooseLocation.tsx @@ -6,7 +6,6 @@ import { ALIGN_CENTER, ALIGN_FLEX_END, Btn, - COLORS, DIRECTION_COLUMN, Flex, JUSTIFY_SPACE_BETWEEN, @@ -20,7 +19,7 @@ import { } from '@opentrons/components' import { getDeckDefFromRobotType } from '@opentrons/shared-data' -import { SmallButton } from '../../atoms/buttons' +import { SmallButton, TextOnlyButton } from '../../atoms/buttons' import { TwoColumn, DeckMapContent } from '../../molecules/InterventionModal' import type { @@ -59,6 +58,7 @@ export const ChooseLocation = ( body, robotType, moveToAddressableArea, + issuedCommandsType, } = props const { i18n, t } = useTranslation(['drop_tip_wizard', 'shared']) const [ @@ -79,7 +79,10 @@ export const ChooseLocation = ( } } return ( - + {title} @@ -102,9 +105,10 @@ export const ChooseLocation = ( handleGoBack() }} > - - {t('shared:go_back')} - + + return ( + <> + {issuedCommandsType === 'fixit' ? : null} + + + ) } function buildShowExitConfirmation(): JSX.Element { @@ -244,7 +252,7 @@ export const DropTipWizardContent = ( header={errorDetails?.header ?? t('error_dropping_tips')} subHeader={subHeader} justifyContentForOddButton={JUSTIFY_FLEX_END} - marginTop={`-${SPACING.spacing68}`} // See EXEC-520. This clearly isn't ideal. + css={ERROR_MODAL_FIXIT_STYLE} > {button} @@ -397,3 +405,9 @@ function useInitiateExit(): { return { isExitInitiated, toggleExitInitiated } } + +const ERROR_MODAL_FIXIT_STYLE = css` + @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { + margin-top: -${SPACING.spacing68}; // See EXEC-520. This clearly isn't ideal. + } +` diff --git a/app/src/organisms/DropTipWizardFlows/JogToPosition.tsx b/app/src/organisms/DropTipWizardFlows/JogToPosition.tsx index 72adeb1b9cc..7ecc9c4a1e1 100644 --- a/app/src/organisms/DropTipWizardFlows/JogToPosition.tsx +++ b/app/src/organisms/DropTipWizardFlows/JogToPosition.tsx @@ -23,7 +23,7 @@ import { } from '@opentrons/components' // import { NeedHelpLink } from '../CalibrationPanels' import { JogControls } from '../../molecules/JogControls' -import { SmallButton } from '../../atoms/buttons' +import { SmallButton, TextOnlyButton } from '../../atoms/buttons' import { InProgressModal } from '../../molecules/InProgressModal/InProgressModal' import { SimpleWizardBody } from '../../molecules/SimpleWizardBody' @@ -46,67 +46,76 @@ type ConfirmPositionProps = DropTipWizardContainerProps & { } const ConfirmPosition = (props: ConfirmPositionProps): JSX.Element | null => { - const { handlePipetteAction, handleGoBack, isOnDevice, currentStep } = props + const { + handlePipetteAction, + handleGoBack, + isOnDevice, + currentStep, + issuedCommandsType, + } = props const { i18n, t } = useTranslation(['drop_tip_wizard', 'shared']) const flowTitle = t('drop_tips') if (isOnDevice) { return ( - <> - - + + + + + - - - - - {currentStep === POSITION_AND_BLOWOUT - ? t('confirm_blowout_location', { flow: flowTitle }) - : t('confirm_drop_tip_location', { flow: flowTitle })} - + {currentStep === POSITION_AND_BLOWOUT + ? t('confirm_blowout_location', { flow: flowTitle }) + : t('confirm_drop_tip_location', { flow: flowTitle })} + + + + + - - - - - - - + + - + ) } else { return ( { {/* */} - - - {t('shared:go_back')} - + + {issuedCommandsType === 'setup' ? ( + + {t('shared:go_back')} + + ) : ( + + )} {currentStep === POSITION_AND_BLOWOUT ? i18n.format(t('blowout_liquid'), 'capitalize') @@ -158,6 +177,7 @@ export const JogToPosition = ( body, currentStep, isOnDevice, + issuedCommandsType, } = props const { i18n, t } = useTranslation(['drop_tip_wizard', 'shared']) const [ @@ -204,6 +224,7 @@ export const JogToPosition = ( flexDirection={DIRECTION_COLUMN} height="100%" gridGap={SPACING.spacing32} + padding={issuedCommandsType === 'fixit' ? SPACING.spacing32 : null} > @@ -231,7 +252,8 @@ export const JogToPosition = ( flexDirection={DIRECTION_COLUMN} justifyContent={JUSTIFY_SPACE_BETWEEN} padding={SPACING.spacing32} - minHeight="29.5rem" + gridGap={issuedCommandsType === 'fixit' ? SPACING.spacing24 : null} + height="100%" > {body}
{/* no animations */} - + {issuedCommandsType === 'setup' ? ( + + ) : null} - <> - - + + {/* */} + {issuedCommandsType === 'setup' ? ( + + {t('shared:go_back')} + + ) : ( + + )} + { + setShowPositionConfirmation(true) + }} > - {/* */} - - - {t('shared:go_back')} - - { - setShowPositionConfirmation(true) - }} - > - {t('shared:confirm_position')} - - - - + {t('shared:confirm_position')} + + ) } diff --git a/app/src/organisms/DropTipWizardFlows/Success.tsx b/app/src/organisms/DropTipWizardFlows/Success.tsx index 92618d36af0..7fb10ae9cc4 100644 --- a/app/src/organisms/DropTipWizardFlows/Success.tsx +++ b/app/src/organisms/DropTipWizardFlows/Success.tsx @@ -21,28 +21,40 @@ type SuccessProps = DropTipWizardContainerProps & { handleProceed: () => void } export const Success = (props: SuccessProps): JSX.Element => { - const { message, proceedText, handleProceed, isOnDevice } = props + const { + message, + proceedText, + handleProceed, + isOnDevice, + issuedCommandsType, + } = props const { i18n } = useTranslation(['drop_tip_wizard', 'shared']) return ( - - {isOnDevice ? ( - - - - ) : ( - {proceedText} - )} - + <> + {issuedCommandsType === 'fixit' ? : null} + + {isOnDevice ? ( + + + + ) : ( + {proceedText} + )} + + ) } diff --git a/app/src/organisms/DropTipWizardFlows/types.ts b/app/src/organisms/DropTipWizardFlows/types.ts index d40d2030488..d7a8309b60b 100644 --- a/app/src/organisms/DropTipWizardFlows/types.ts +++ b/app/src/organisms/DropTipWizardFlows/types.ts @@ -25,12 +25,17 @@ interface ErrorOverrides { generalFailure: () => void } +interface ButtonOverrides { + goBackBeforeBeginning: () => void +} + export interface FixitCommandTypeUtils { runId: string failedCommandId: string trackCurrentMap: ERUtilsResults['trackExternalMap'] copyOverrides: CopyOverrides errorOverrides: ErrorOverrides + buttonOverrides: ButtonOverrides routeOverride?: typeof DT_ROUTES[keyof typeof DT_ROUTES] } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx index 154e1600cb0..96de863c965 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/CancelRun.tsx @@ -68,7 +68,8 @@ function CancelRunConfirmation({ @@ -84,7 +85,7 @@ function CancelRunConfirmation({ {t('if_tips_are_attached')} diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx index a80b777d4ba..7295b663b89 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx @@ -5,6 +5,7 @@ import { css } from 'styled-components' import { DIRECTION_COLUMN, + COLORS, SPACING, Flex, StyledText, @@ -103,8 +104,9 @@ export function BeginRemoval({ } ` - const RADIO_GROUP_MARGIN = css` + const RADIO_GROUP_STYLE = css` @media not (${RESPONSIVENESS.touchscreenMediaQuerySpecs}) { + color: ${COLORS.black90}; margin-left: 0.5rem; } ` @@ -116,7 +118,7 @@ export function BeginRemoval({ oddStyle="level4HeaderSemiBold" desktopStyle="headingSmallSemiBold" > - {t('you_may_want_to_remove', { mount })} + {t('remove_tips_from_pipette', { mount })} { setSelected('skip') }} @@ -155,24 +157,24 @@ export function BeginRemoval({ }} options={[ { - value: t('begin_removal'), + value: 'begin-removal', children: ( {t('begin_removal')} ), }, { - value: t('skip'), + value: 'skip', children: ( - {t('skip')} + {t('skip_removal')} ), }, @@ -299,6 +301,14 @@ export function useDropTipFlowUtils({ } } + const buildButtonOverrides = (): FixitCommandTypeUtils['buttonOverrides'] => { + return { + goBackBeforeBeginning: () => { + return proceedToRouteAndStep(DROP_TIP_FLOWS.ROUTE) + }, + } + } + // If a specific step within the DROP_TIP_FLOWS route is selected, begin the Drop Tip Flows at its related route. const buildRouteOverride = (): FixitCommandTypeUtils['routeOverride'] => { switch (step) { @@ -315,6 +325,7 @@ export function useDropTipFlowUtils({ copyOverrides: buildCopyOverrides(), trackCurrentMap: trackExternalMap, errorOverrides: buildErrorOverrides(), + buttonOverrides: buildButtonOverrides(), routeOverride: buildRouteOverride(), } } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx index 8e9327dd45f..ed58d7e597a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx @@ -90,9 +90,7 @@ describe('ManageTips', () => { it(`renders BeginRemoval with correct copy when the step is ${DROP_TIP_FLOWS.STEPS.BEGIN_REMOVAL}`, () => { render(props) - screen.getByText( - 'You may want to remove the tips from the left pipette before using it again in a protocol' - ) + screen.getByText('Remove tips from left pipette before canceling the run?') screen.queryAllByText('Begin removal') screen.queryAllByText('Skip') expect(screen.getAllByText('Continue').length).toBe(2) @@ -102,7 +100,7 @@ describe('ManageTips', () => { render(props) const beginRemovalBtn = screen.queryAllByText('Begin removal')[0] - const skipBtn = screen.queryAllByText('Skip')[0] + const skipBtn = screen.queryAllByText('Skip removal')[0] fireEvent.click(beginRemovalBtn) clickButtonLabeled('Continue') @@ -124,7 +122,7 @@ describe('ManageTips', () => { } render(props) - const skipBtn = screen.queryAllByText('Skip')[0] + const skipBtn = screen.queryAllByText('Skip removal')[0] fireEvent.click(skipBtn) clickButtonLabeled('Continue') @@ -265,6 +263,14 @@ describe('useDropTipFlowUtils', () => { ) }) + it('should return the correct button overrides', () => { + const { result } = renderHook(() => useDropTipFlowUtils(mockProps)) + + result.current.buttonOverrides.goBackBeforeBeginning() + + expect(mockProceedToRouteAndStep).toHaveBeenCalledWith(DROP_TIP_FLOWS.ROUTE) + }) + it(`should return correct route overrides when the route is ${DROP_TIP_FLOWS.STEPS.CHOOSE_TIP_DROP}`, () => { const { result } = renderHook(() => useDropTipFlowUtils(mockProps)) diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx index 9332ab8766d..00a853ee99a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/RecoveryInterventionModal.tsx @@ -38,7 +38,6 @@ export function RecoveryInterventionModal({ ? SMALL_MODAL_STYLE : LARGE_MODAL_STYLE } - padding={SPACING.spacing32} > {children} @@ -47,17 +46,15 @@ export function RecoveryInterventionModal({ ) } -const ODD_STYLE = ` -@media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { - height: 100%; - } -` - const SMALL_MODAL_STYLE = css` height: 22rem; - ${ODD_STYLE} + padding: ${SPACING.spacing32}; + + @media ${RESPONSIVENESS.touchscreenMediaQuerySpecs} { + padding: ${SPACING.spacing32}; + height: 100%; + } ` const LARGE_MODAL_STYLE = css` height: 26.75rem; - ${ODD_STYLE} ` From d61e4bfbc6556bba119497556b03baa05f6c0035 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Tue, 23 Jul 2024 13:12:01 -0400 Subject: [PATCH 17/50] fix(app): add fallback text to recovery success toasts (#15763) Closes EXEC-624 If error recovery occurs for a step not in protocol analysis, or error recovery results in skipping to a command when the user is recovering from the last command of the run, there is no command text. As a fallback in these spots, let's just not show the full command text and just show the recovery action text. Note that this never occurs on the ODD, since we never show the full command text. --- .../__tests__/useRecoveryToasts.test.tsx | 49 +++++++++++++++++-- .../hooks/useRecoveryToasts.ts | 14 ++++-- 2 files changed, 56 insertions(+), 7 deletions(-) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx index 8766fc83590..82a14835958 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryToasts.test.tsx @@ -86,7 +86,7 @@ describe('useRecoveryToasts', () => { result.current.makeSuccessToast() expect(mockMakeToast).toHaveBeenCalledWith( - TEST_COMMAND, + 'Retrying step 1 succeeded.', 'success', expect.objectContaining({ closeButton: true, @@ -117,6 +117,32 @@ describe('useRecoveryToasts', () => { }) ) }) + + it('should use recoveryToastText when desktopFullCommandText is null', () => { + vi.mocked(useCommandTextString).mockReturnValue({ + commandText: '', + stepTexts: undefined, + }) + + const { result } = renderHook(() => + useRecoveryToasts({ + ...DEFAULT_PROPS, + commandTextData: { commands: [] } as any, + }) + ) + + result.current.makeSuccessToast() + expect(mockMakeToast).toHaveBeenCalledWith( + expect.any(String), + 'success', + expect.objectContaining({ + closeButton: true, + disableTimeout: true, + displayType: 'desktop', + heading: expect.any(String), + }) + ) + }) }) describe('useRecoveryToastText', () => { @@ -177,7 +203,7 @@ describe('useRecoveryFullCommandText', () => { const { result } = renderHook(() => useRecoveryFullCommandText({ robotType: FLEX_ROBOT_TYPE, - stepNumber: 1, + stepNumber: 0, commandTextData: { commands: [TEST_COMMAND] } as any, }) ) @@ -185,6 +211,23 @@ describe('useRecoveryFullCommandText', () => { expect(result.current).toBe(TEST_COMMAND) }) + it('should return null when relevantCmd is null', () => { + vi.mocked(useCommandTextString).mockReturnValue({ + commandText: '', + stepTexts: undefined, + }) + + const { result } = renderHook(() => + useRecoveryFullCommandText({ + robotType: FLEX_ROBOT_TYPE, + stepNumber: 1, + commandTextData: { commands: [] } as any, + }) + ) + + expect(result.current).toBeNull() + }) + it('should return stepNumber if it is a string', () => { const { result } = renderHook(() => useRecoveryFullCommandText({ @@ -206,7 +249,7 @@ describe('useRecoveryFullCommandText', () => { const { result } = renderHook(() => useRecoveryFullCommandText({ robotType: FLEX_ROBOT_TYPE, - stepNumber: 1, + stepNumber: 0, commandTextData: { commands: [TC_COMMAND], } as any, diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts index 632846329b5..00ccba0ea95 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryToasts.ts @@ -40,9 +40,11 @@ export function useRecoveryToasts({ selectedRecoveryOption, }) - // The "body" of the toast message. On ODD, this is the recovery-specific text. On desktop, this is the full command text. + // The "body" of the toast message. On desktop, this is the full command text, if present. Otherwise, this is the recovery-specific text. const bodyText = - displayType === 'desktop' ? desktopFullCommandText : recoveryToastText + displayType === 'desktop' && desktopFullCommandText != null + ? desktopFullCommandText + : recoveryToastText // The "heading" of the toast message. Currently, this text is only present on the desktop toasts. const headingText = displayType === 'desktop' ? recoveryToastText : undefined @@ -97,7 +99,7 @@ type UseRecoveryFullCommandTextParams = Omit< // Return the full command text of the recovery command that is "retried" or "skipped". export function useRecoveryFullCommandText( props: UseRecoveryFullCommandTextParams -): string { +): string | null { const { commandTextData, stepNumber } = props const relevantCmdIdx = typeof stepNumber === 'number' ? stepNumber : -1 @@ -110,12 +112,16 @@ export function useRecoveryFullCommandText( if (typeof stepNumber === 'string') { return stepNumber + } + // Occurs when the relevantCmd doesn't exist, ex, we "skip" the last command of a run. + else if (relevantCmd === null) { + return null } else { return truncateIfTCCommand(commandText, stepTexts != null) } } -// Return the user-facing step number. If the step number cannot be determined, return '?'. +// Return the user-facing step number, 0 indexed. If the step number cannot be determined, return '?'. export function getStepNumber( selectedRecoveryOption: BuildToast['selectedRecoveryOption'], currentStepCount: BuildToast['currentStepCount'] From 02acbe19fe7ce5359d3dd0619e0e1ab60232472d Mon Sep 17 00:00:00 2001 From: Max Marrone Date: Tue, 23 Jul 2024 13:37:06 -0400 Subject: [PATCH 18/50] chore(python): Revert jsonschema updates to fix snapshot tests (#15759) This reverts commit 8f2583f722a0d612f2e49172c218ef64e4bea302. --- Makefile | 2 +- api/setup.py | 2 +- hardware/Pipfile | 2 +- hardware/Pipfile.lock | 1006 ++++++++--------- .../drivers/can_bus/settings.py | 16 +- performance-metrics/Pipfile | 4 +- performance-metrics/Pipfile.lock | 15 +- shared-data/python/Pipfile | 2 +- shared-data/python/Pipfile.lock | 471 ++++---- shared-data/python/setup.py | 2 +- 10 files changed, 744 insertions(+), 778 deletions(-) diff --git a/Makefile b/Makefile index 47191131c7b..ffdbb8509c0 100755 --- a/Makefile +++ b/Makefile @@ -25,7 +25,7 @@ HARDWARE_DIR := hardware USB_BRIDGE_DIR := usb-bridge NODE_USB_BRIDGE_CLIENT_DIR := usb-bridge/node-client -PYTHON_DIRS := $(API_DIR) $(UPDATE_SERVER_DIR) $(ROBOT_SERVER_DIR) $(SERVER_UTILS_DIR) $(SHARED_DATA_DIR)/python $(SYSTEM_SERVER_DIR) $(G_CODE_TESTING_DIR) $(HARDWARE_DIR) $(USB_BRIDGE_DIR) +PYTHON_DIRS := $(API_DIR) $(UPDATE_SERVER_DIR) $(ROBOT_SERVER_DIR) $(SERVER_UTILS_DIR) $(SHARED_DATA_DIR)/python $(G_CODE_TESTING_DIR) $(HARDWARE_DIR) $(USB_BRIDGE_DIR) # This may be set as an environment variable (and is by CI tasks that upload # to test pypi) to add a .dev extension to the python package versions. If diff --git a/api/setup.py b/api/setup.py index 8c1dd7cfa63..1b2a7dde508 100755 --- a/api/setup.py +++ b/api/setup.py @@ -59,7 +59,7 @@ def get_version(): f"opentrons-shared-data=={VERSION}", "aionotify==0.3.1", "anyio>=3.6.1,<4.0.0", - "jsonschema>=3.0.1,<5", + "jsonschema>=3.0.1,<4.18.0", "numpy>=1.20.0,<2", "pydantic>=1.10.9,<2.0.0", "pyserial>=3.5", diff --git a/hardware/Pipfile b/hardware/Pipfile index eed4c866740..b02e50c7c51 100644 --- a/hardware/Pipfile +++ b/hardware/Pipfile @@ -8,7 +8,7 @@ python-can = "==4.2.2" pyserial = "==3.5" typing-extensions = ">=4.0.0,<5" numpy = "==1.22.3" -pydantic = "==1.10.12" +pydantic = "==1.9.2" [dev-packages] pytest = "==7.4.4" diff --git a/hardware/Pipfile.lock b/hardware/Pipfile.lock index faae814c9ef..ccab8884999 100644 --- a/hardware/Pipfile.lock +++ b/hardware/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "de5368468f6a396a6a3bd854b5e2035ce9c68ef958832198866c883b2b17c2e2" + "sha256": "04ae6d52e739cf67e21d13822316f7dc2030d272976c0a9cfd0f7d35db743301" }, "pipfile-spec": 6, "requires": { @@ -18,65 +18,65 @@ "default": { "msgpack": { "hashes": [ - "sha256:00e073efcba9ea99db5acef3959efa45b52bc67b61b00823d2a1a6944bf45982", - "sha256:0726c282d188e204281ebd8de31724b7d749adebc086873a59efb8cf7ae27df3", - "sha256:0ceea77719d45c839fd73abcb190b8390412a890df2f83fb8cf49b2a4b5c2f40", - "sha256:114be227f5213ef8b215c22dde19532f5da9652e56e8ce969bf0a26d7c419fee", - "sha256:13577ec9e247f8741c84d06b9ece5f654920d8365a4b636ce0e44f15e07ec693", - "sha256:1876b0b653a808fcd50123b953af170c535027bf1d053b59790eebb0aeb38950", - "sha256:1ab0bbcd4d1f7b6991ee7c753655b481c50084294218de69365f8f1970d4c151", - "sha256:1cce488457370ffd1f953846f82323cb6b2ad2190987cd4d70b2713e17268d24", - "sha256:26ee97a8261e6e35885c2ecd2fd4a6d38252246f94a2aec23665a4e66d066305", - "sha256:3528807cbbb7f315bb81959d5961855e7ba52aa60a3097151cb21956fbc7502b", - "sha256:374a8e88ddab84b9ada695d255679fb99c53513c0a51778796fcf0944d6c789c", - "sha256:376081f471a2ef24828b83a641a02c575d6103a3ad7fd7dade5486cad10ea659", - "sha256:3923a1778f7e5ef31865893fdca12a8d7dc03a44b33e2a5f3295416314c09f5d", - "sha256:4916727e31c28be8beaf11cf117d6f6f188dcc36daae4e851fee88646f5b6b18", - "sha256:493c5c5e44b06d6c9268ce21b302c9ca055c1fd3484c25ba41d34476c76ee746", - "sha256:505fe3d03856ac7d215dbe005414bc28505d26f0c128906037e66d98c4e95868", - "sha256:5845fdf5e5d5b78a49b826fcdc0eb2e2aa7191980e3d2cfd2a30303a74f212e2", - "sha256:5c330eace3dd100bdb54b5653b966de7f51c26ec4a7d4e87132d9b4f738220ba", - "sha256:5dbf059fb4b7c240c873c1245ee112505be27497e90f7c6591261c7d3c3a8228", - "sha256:5e390971d082dba073c05dbd56322427d3280b7cc8b53484c9377adfbae67dc2", - "sha256:5fbb160554e319f7b22ecf530a80a3ff496d38e8e07ae763b9e82fadfe96f273", - "sha256:64d0fcd436c5683fdd7c907eeae5e2cbb5eb872fafbc03a43609d7941840995c", - "sha256:69284049d07fce531c17404fcba2bb1df472bc2dcdac642ae71a2d079d950653", - "sha256:6a0e76621f6e1f908ae52860bdcb58e1ca85231a9b0545e64509c931dd34275a", - "sha256:73ee792784d48aa338bba28063e19a27e8d989344f34aad14ea6e1b9bd83f596", - "sha256:74398a4cf19de42e1498368c36eed45d9528f5fd0155241e82c4082b7e16cffd", - "sha256:7938111ed1358f536daf311be244f34df7bf3cdedb3ed883787aca97778b28d8", - "sha256:82d92c773fbc6942a7a8b520d22c11cfc8fd83bba86116bfcf962c2f5c2ecdaa", - "sha256:83b5c044f3eff2a6534768ccfd50425939e7a8b5cf9a7261c385de1e20dcfc85", - "sha256:8db8e423192303ed77cff4dce3a4b88dbfaf43979d280181558af5e2c3c71afc", - "sha256:9517004e21664f2b5a5fd6333b0731b9cf0817403a941b393d89a2f1dc2bd836", - "sha256:95c02b0e27e706e48d0e5426d1710ca78e0f0628d6e89d5b5a5b91a5f12274f3", - "sha256:99881222f4a8c2f641f25703963a5cefb076adffd959e0558dc9f803a52d6a58", - "sha256:9ee32dcb8e531adae1f1ca568822e9b3a738369b3b686d1477cbc643c4a9c128", - "sha256:a22e47578b30a3e199ab067a4d43d790249b3c0587d9a771921f86250c8435db", - "sha256:b5505774ea2a73a86ea176e8a9a4a7c8bf5d521050f0f6f8426afe798689243f", - "sha256:bd739c9251d01e0279ce729e37b39d49a08c0420d3fee7f2a4968c0576678f77", - "sha256:d16a786905034e7e34098634b184a7d81f91d4c3d246edc6bd7aefb2fd8ea6ad", - "sha256:d3420522057ebab1728b21ad473aa950026d07cb09da41103f8e597dfbfaeb13", - "sha256:d56fd9f1f1cdc8227d7b7918f55091349741904d9520c65f0139a9755952c9e8", - "sha256:d661dc4785affa9d0edfdd1e59ec056a58b3dbb9f196fa43587f3ddac654ac7b", - "sha256:dfe1f0f0ed5785c187144c46a292b8c34c1295c01da12e10ccddfc16def4448a", - "sha256:e1dd7839443592d00e96db831eddb4111a2a81a46b028f0facd60a09ebbdd543", - "sha256:e2872993e209f7ed04d963e4b4fbae72d034844ec66bc4ca403329db2074377b", - "sha256:e2f879ab92ce502a1e65fce390eab619774dda6a6ff719718069ac94084098ce", - "sha256:e3aa7e51d738e0ec0afbed661261513b38b3014754c9459508399baf14ae0c9d", - "sha256:e532dbd6ddfe13946de050d7474e3f5fb6ec774fbb1a188aaf469b08cf04189a", - "sha256:e6b7842518a63a9f17107eb176320960ec095a8ee3b4420b5f688e24bf50c53c", - "sha256:e75753aeda0ddc4c28dce4c32ba2f6ec30b1b02f6c0b14e547841ba5b24f753f", - "sha256:eadb9f826c138e6cf3c49d6f8de88225a3c0ab181a9b4ba792e006e5292d150e", - "sha256:ed59dd52075f8fc91da6053b12e8c89e37aa043f8986efd89e61fae69dc1b011", - "sha256:ef254a06bcea461e65ff0373d8a0dd1ed3aa004af48839f002a0c994a6f72d04", - "sha256:f3709997b228685fe53e8c433e2df9f0cdb5f4542bd5114ed17ac3c0129b0480", - "sha256:f51bab98d52739c50c56658cc303f190785f9a2cd97b823357e7aeae54c8f68a", - "sha256:f9904e24646570539a8950400602d66d2b2c492b9010ea7e965025cb71d0c86d", - "sha256:f9af38a89b6a5c04b7d18c492c8ccf2aee7048aff1ce8437c4683bb5a1df893d" + "sha256:04ad6069c86e531682f9e1e71b71c1c3937d6014a7c3e9edd2aa81ad58842862", + "sha256:0bfdd914e55e0d2c9e1526de210f6fe8ffe9705f2b1dfcc4aecc92a4cb4b533d", + "sha256:1dc93e8e4653bdb5910aed79f11e165c85732067614f180f70534f056da97db3", + "sha256:1e2d69948e4132813b8d1131f29f9101bc2c915f26089a6d632001a5c1349672", + "sha256:235a31ec7db685f5c82233bddf9858748b89b8119bf4538d514536c485c15fe0", + "sha256:27dcd6f46a21c18fa5e5deed92a43d4554e3df8d8ca5a47bf0615d6a5f39dbc9", + "sha256:28efb066cde83c479dfe5a48141a53bc7e5f13f785b92ddde336c716663039ee", + "sha256:3476fae43db72bd11f29a5147ae2f3cb22e2f1a91d575ef130d2bf49afd21c46", + "sha256:36e17c4592231a7dbd2ed09027823ab295d2791b3b1efb2aee874b10548b7524", + "sha256:384d779f0d6f1b110eae74cb0659d9aa6ff35aaf547b3955abf2ab4c901c4819", + "sha256:38949d30b11ae5f95c3c91917ee7a6b239f5ec276f271f28638dec9156f82cfc", + "sha256:3967e4ad1aa9da62fd53e346ed17d7b2e922cba5ab93bdd46febcac39be636fc", + "sha256:3e7bf4442b310ff154b7bb9d81eb2c016b7d597e364f97d72b1acc3817a0fdc1", + "sha256:3f0c8c6dfa6605ab8ff0611995ee30d4f9fcff89966cf562733b4008a3d60d82", + "sha256:484ae3240666ad34cfa31eea7b8c6cd2f1fdaae21d73ce2974211df099a95d81", + "sha256:4a7b4f35de6a304b5533c238bee86b670b75b03d31b7797929caa7a624b5dda6", + "sha256:4cb14ce54d9b857be9591ac364cb08dc2d6a5c4318c1182cb1d02274029d590d", + "sha256:4e71bc4416de195d6e9b4ee93ad3f2f6b2ce11d042b4d7a7ee00bbe0358bd0c2", + "sha256:52700dc63a4676669b341ba33520f4d6e43d3ca58d422e22ba66d1736b0a6e4c", + "sha256:572efc93db7a4d27e404501975ca6d2d9775705c2d922390d878fcf768d92c87", + "sha256:576eb384292b139821c41995523654ad82d1916da6a60cff129c715a6223ea84", + "sha256:5b0bf0effb196ed76b7ad883848143427a73c355ae8e569fa538365064188b8e", + "sha256:5b6ccc0c85916998d788b295765ea0e9cb9aac7e4a8ed71d12e7d8ac31c23c95", + "sha256:5ed82f5a7af3697b1c4786053736f24a0efd0a1b8a130d4c7bfee4b9ded0f08f", + "sha256:6d4c80667de2e36970ebf74f42d1088cc9ee7ef5f4e8c35eee1b40eafd33ca5b", + "sha256:730076207cb816138cf1af7f7237b208340a2c5e749707457d70705715c93b93", + "sha256:7687e22a31e976a0e7fc99c2f4d11ca45eff652a81eb8c8085e9609298916dcf", + "sha256:822ea70dc4018c7e6223f13affd1c5c30c0f5c12ac1f96cd8e9949acddb48a61", + "sha256:84b0daf226913133f899ea9b30618722d45feffa67e4fe867b0b5ae83a34060c", + "sha256:85765fdf4b27eb5086f05ac0491090fc76f4f2b28e09d9350c31aac25a5aaff8", + "sha256:8dd178c4c80706546702c59529ffc005681bd6dc2ea234c450661b205445a34d", + "sha256:8f5b234f567cf76ee489502ceb7165c2a5cecec081db2b37e35332b537f8157c", + "sha256:98bbd754a422a0b123c66a4c341de0474cad4a5c10c164ceed6ea090f3563db4", + "sha256:993584fc821c58d5993521bfdcd31a4adf025c7d745bbd4d12ccfecf695af5ba", + "sha256:a40821a89dc373d6427e2b44b572efc36a2778d3f543299e2f24eb1a5de65415", + "sha256:b291f0ee7961a597cbbcc77709374087fa2a9afe7bdb6a40dbbd9b127e79afee", + "sha256:b573a43ef7c368ba4ea06050a957c2a7550f729c31f11dd616d2ac4aba99888d", + "sha256:b610ff0f24e9f11c9ae653c67ff8cc03c075131401b3e5ef4b82570d1728f8a9", + "sha256:bdf38ba2d393c7911ae989c3bbba510ebbcdf4ecbdbfec36272abe350c454075", + "sha256:bfef2bb6ef068827bbd021017a107194956918ab43ce4d6dc945ffa13efbc25f", + "sha256:cab3db8bab4b7e635c1c97270d7a4b2a90c070b33cbc00c99ef3f9be03d3e1f7", + "sha256:cb70766519500281815dfd7a87d3a178acf7ce95390544b8c90587d76b227681", + "sha256:cca1b62fe70d761a282496b96a5e51c44c213e410a964bdffe0928e611368329", + "sha256:ccf9a39706b604d884d2cb1e27fe973bc55f2890c52f38df742bc1d79ab9f5e1", + "sha256:dc43f1ec66eb8440567186ae2f8c447d91e0372d793dfe8c222aec857b81a8cf", + "sha256:dd632777ff3beaaf629f1ab4396caf7ba0bdd075d948a69460d13d44357aca4c", + "sha256:e45ae4927759289c30ccba8d9fdce62bb414977ba158286b5ddaf8df2cddb5c5", + "sha256:e50ebce52f41370707f1e21a59514e3375e3edd6e1832f5e5235237db933c98b", + "sha256:ebbbba226f0a108a7366bf4b59bf0f30a12fd5e75100c630267d94d7f0ad20e5", + "sha256:ec79ff6159dffcc30853b2ad612ed572af86c92b5168aa3fc01a67b0fa40665e", + "sha256:f0936e08e0003f66bfd97e74ee530427707297b0d0361247e9b4f59ab78ddc8b", + "sha256:f26a07a6e877c76a88e3cecac8531908d980d3d5067ff69213653649ec0f60ad", + "sha256:f64e376cd20d3f030190e8c32e1c64582eba56ac6dc7d5b0b49a9d44021b52fd", + "sha256:f6ffbc252eb0d229aeb2f9ad051200668fc3a9aaa8994e49f0cb2ffe2b7867e7", + "sha256:f9a7c509542db4eceed3dcf21ee5267ab565a83555c9b88a8109dcecc4709002", + "sha256:ff1d0899f104f3921d94579a5638847f783c9b04f2d5f229392ca77fba5b82fc" ], "markers": "platform_system != 'Windows'", - "version": "==1.0.8" + "version": "==1.0.7" }, "numpy": { "hashes": [ @@ -107,54 +107,53 @@ }, "packaging": { "hashes": [ - "sha256:026ed72c8ed3fcce5bf8950572258698927fd1dbda10a5e981cdf0ac37f4f002", - "sha256:5b8f2217dbdbd2f7f384c41c628544e6d52f2d0f53c6d0c3ea61aa5d1d7ff124" + "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5", + "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7" ], - "markers": "python_version >= '3.8'", - "version": "==24.1" + "markers": "python_version >= '3.7'", + "version": "==23.2" }, "pydantic": { "hashes": [ - "sha256:0fe8a415cea8f340e7a9af9c54fc71a649b43e8ca3cc732986116b3cb135d303", - "sha256:1289c180abd4bd4555bb927c42ee42abc3aee02b0fb2d1223fb7c6e5bef87dbe", - "sha256:1eb2085c13bce1612da8537b2d90f549c8cbb05c67e8f22854e201bde5d98a47", - "sha256:2031de0967c279df0d8a1c72b4ffc411ecd06bac607a212892757db7462fc494", - "sha256:2a7bac939fa326db1ab741c9d7f44c565a1d1e80908b3797f7f81a4f86bc8d33", - "sha256:2d5a58feb9a39f481eda4d5ca220aa8b9d4f21a41274760b9bc66bfd72595b86", - "sha256:2f9a6fab5f82ada41d56b0602606a5506aab165ca54e52bc4545028382ef1c5d", - "sha256:2fcfb5296d7877af406ba1547dfde9943b1256d8928732267e2653c26938cd9c", - "sha256:549a8e3d81df0a85226963611950b12d2d334f214436a19537b2efed61b7639a", - "sha256:598da88dfa127b666852bef6d0d796573a8cf5009ffd62104094a4fe39599565", - "sha256:5d1197e462e0364906cbc19681605cb7c036f2475c899b6f296104ad42b9f5fb", - "sha256:69328e15cfda2c392da4e713443c7dbffa1505bc9d566e71e55abe14c97ddc62", - "sha256:6a9dfa722316f4acf4460afdf5d41d5246a80e249c7ff475c43a3a1e9d75cf62", - "sha256:6b30bcb8cbfccfcf02acb8f1a261143fab622831d9c0989707e0e659f77a18e0", - "sha256:6c076be61cd0177a8433c0adcb03475baf4ee91edf5a4e550161ad57fc90f523", - "sha256:771735dc43cf8383959dc9b90aa281f0b6092321ca98677c5fb6125a6f56d58d", - "sha256:795e34e6cc065f8f498c89b894a3c6da294a936ee71e644e4bd44de048af1405", - "sha256:87afda5539d5140cb8ba9e8b8c8865cb5b1463924d38490d73d3ccfd80896b3f", - "sha256:8fb2aa3ab3728d950bcc885a2e9eff6c8fc40bc0b7bb434e555c215491bcf48b", - "sha256:a1fcb59f2f355ec350073af41d927bf83a63b50e640f4dbaa01053a28b7a7718", - "sha256:a5e7add47a5b5a40c49b3036d464e3c7802f8ae0d1e66035ea16aa5b7a3923ed", - "sha256:a73f489aebd0c2121ed974054cb2759af8a9f747de120acd2c3394cf84176ccb", - "sha256:ab26038b8375581dc832a63c948f261ae0aa21f1d34c1293469f135fa92972a5", - "sha256:b0d191db0f92dfcb1dec210ca244fdae5cbe918c6050b342d619c09d31eea0cc", - "sha256:b749a43aa51e32839c9d71dc67eb1e4221bb04af1033a32e3923d46f9effa942", - "sha256:b7ccf02d7eb340b216ec33e53a3a629856afe1c6e0ef91d84a4e6f2fb2ca70fe", - "sha256:ba5b2e6fe6ca2b7e013398bc7d7b170e21cce322d266ffcd57cca313e54fb246", - "sha256:ba5c4a8552bff16c61882db58544116d021d0b31ee7c66958d14cf386a5b5350", - "sha256:c79e6a11a07da7374f46970410b41d5e266f7f38f6a17a9c4823db80dadf4303", - "sha256:ca48477862372ac3770969b9d75f1bf66131d386dba79506c46d75e6b48c1e09", - "sha256:dea7adcc33d5d105896401a1f37d56b47d443a2b2605ff8a969a0ed5543f7e33", - "sha256:e0a16d274b588767602b7646fa05af2782576a6cf1022f4ba74cbb4db66f6ca8", - "sha256:e4129b528c6baa99a429f97ce733fff478ec955513630e61b49804b6cf9b224a", - "sha256:e5f805d2d5d0a41633651a73fa4ecdd0b3d7a49de4ec3fadf062fe16501ddbf1", - "sha256:ef6c96b2baa2100ec91a4b428f80d8f28a3c9e53568219b6c298c1125572ebc6", - "sha256:fdbdd1d630195689f325c9ef1a12900524dceb503b00a987663ff4f58669b93d" + "sha256:1061c6ee6204f4f5a27133126854948e3b3d51fcc16ead2e5d04378c199b2f44", + "sha256:19b5686387ea0d1ea52ecc4cffb71abb21702c5e5b2ac626fd4dbaa0834aa49d", + "sha256:2bd446bdb7755c3a94e56d7bdfd3ee92396070efa8ef3a34fab9579fe6aa1d84", + "sha256:328558c9f2eed77bd8fffad3cef39dbbe3edc7044517f4625a769d45d4cf7555", + "sha256:32e0b4fb13ad4db4058a7c3c80e2569adbd810c25e6ca3bbd8b2a9cc2cc871d7", + "sha256:3ee0d69b2a5b341fc7927e92cae7ddcfd95e624dfc4870b32a85568bd65e6131", + "sha256:4aafd4e55e8ad5bd1b19572ea2df546ccace7945853832bb99422a79c70ce9b8", + "sha256:4b3946f87e5cef3ba2e7bd3a4eb5a20385fe36521d6cc1ebf3c08a6697c6cfb3", + "sha256:4de71c718c9756d679420c69f216776c2e977459f77e8f679a4a961dc7304a56", + "sha256:5565a49effe38d51882cb7bac18bda013cdb34d80ac336428e8908f0b72499b0", + "sha256:5803ad846cdd1ed0d97eb00292b870c29c1f03732a010e66908ff48a762f20e4", + "sha256:5da164119602212a3fe7e3bc08911a89db4710ae51444b4224c2382fd09ad453", + "sha256:615661bfc37e82ac677543704437ff737418e4ea04bef9cf11c6d27346606044", + "sha256:78a4d6bdfd116a559aeec9a4cfe77dda62acc6233f8b56a716edad2651023e5e", + "sha256:7d0f183b305629765910eaad707800d2f47c6ac5bcfb8c6397abdc30b69eeb15", + "sha256:7ead3cd020d526f75b4188e0a8d71c0dbbe1b4b6b5dc0ea775a93aca16256aeb", + "sha256:84d76ecc908d917f4684b354a39fd885d69dd0491be175f3465fe4b59811c001", + "sha256:8cb0bc509bfb71305d7a59d00163d5f9fc4530f0881ea32c74ff4f74c85f3d3d", + "sha256:91089b2e281713f3893cd01d8e576771cd5bfdfbff5d0ed95969f47ef6d676c3", + "sha256:9c9e04a6cdb7a363d7cb3ccf0efea51e0abb48e180c0d31dca8d247967d85c6e", + "sha256:a8c5360a0297a713b4123608a7909e6869e1b56d0e96eb0d792c27585d40757f", + "sha256:afacf6d2a41ed91fc631bade88b1d319c51ab5418870802cedb590b709c5ae3c", + "sha256:b34ba24f3e2d0b39b43f0ca62008f7ba962cff51efa56e64ee25c4af6eed987b", + "sha256:bd67cb2c2d9602ad159389c29e4ca964b86fa2f35c2faef54c3eb28b4efd36c8", + "sha256:c0f5e142ef8217019e3eef6ae1b6b55f09a7a15972958d44fbd228214cede567", + "sha256:cdb4272678db803ddf94caa4f94f8672e9a46bae4a44f167095e4d06fec12979", + "sha256:d70916235d478404a3fa8c997b003b5f33aeac4686ac1baa767234a0f8ac2326", + "sha256:d8ce3fb0841763a89322ea0432f1f59a2d3feae07a63ea2c958b2315e1ae8adb", + "sha256:e0b214e57623a535936005797567231a12d0da0c29711eb3514bc2b3cd008d0f", + "sha256:e631c70c9280e3129f071635b81207cad85e6c08e253539467e4ead0e5b219aa", + "sha256:e78578f0c7481c850d1c969aca9a65405887003484d24f6110458fb02cca7747", + "sha256:f0ca86b525264daa5f6b192f216a0d1e860b7383e3da1c65a1908f9c02f42801", + "sha256:f1a68f4f65a9ee64b6ccccb5bf7e17db07caebd2730109cb8a95863cfa9c4e55", + "sha256:fafe841be1103f340a24977f61dee76172e4ae5f647ab9e7fd1e1fca51524f08", + "sha256:ff68fc85355532ea77559ede81f35fff79a6a5543477e168ab3a381887caea76" ], "index": "pypi", - "markers": "python_version >= '3.7'", - "version": "==1.10.12" + "markers": "python_full_version >= '3.6.1'", + "version": "==1.9.2" }, "pyserial": { "hashes": [ @@ -175,20 +174,20 @@ }, "setuptools": { "hashes": [ - "sha256:3d8531791a27056f4a38cd3e54084d8b1c4228ff9cf3f2d7dd075ec99f9fd70d", - "sha256:f501b6e6db709818dc76882582d9c516bf3b67b948864c5fa1d1624c09a49207" + "sha256:385eb4edd9c9d5c17540511303e39a147ce2fc04bc55289c322b9e5904fe2c05", + "sha256:be1af57fc409f93647f2e8e4573a142ed38724b8cdd389706a867bb4efcf1e78" ], "markers": "python_version >= '3.8'", - "version": "==71.0.3" + "version": "==69.0.3" }, "typing-extensions": { "hashes": [ - "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d", - "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8" + "sha256:23478f88c37f27d76ac8aee6c905017a143b0b1b886c3c9f66bc2fd94f9f5783", + "sha256:af72aea155e91adfc61c3ae9e0e342dbc0cba726d6cba4b6c72c1f34e47291cd" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==4.12.2" + "version": "==4.9.0" }, "wrapt": { "hashes": [ @@ -316,114 +315,114 @@ }, "contourpy": { "hashes": [ - "sha256:00e5388f71c1a0610e6fe56b5c44ab7ba14165cdd6d695429c5cd94021e390b2", - "sha256:10a37ae557aabf2509c79715cd20b62e4c7c28b8cd62dd7d99e5ed3ce28c3fd9", - "sha256:11959f0ce4a6f7b76ec578576a0b61a28bdc0696194b6347ba3f1c53827178b9", - "sha256:187fa1d4c6acc06adb0fae5544c59898ad781409e61a926ac7e84b8f276dcef4", - "sha256:1a07fc092a4088ee952ddae19a2b2a85757b923217b7eed584fdf25f53a6e7ce", - "sha256:1cac0a8f71a041aa587410424ad46dfa6a11f6149ceb219ce7dd48f6b02b87a7", - "sha256:1d59e739ab0e3520e62a26c60707cc3ab0365d2f8fecea74bfe4de72dc56388f", - "sha256:2855c8b0b55958265e8b5888d6a615ba02883b225f2227461aa9127c578a4922", - "sha256:2e785e0f2ef0d567099b9ff92cbfb958d71c2d5b9259981cd9bee81bd194c9a4", - "sha256:309be79c0a354afff9ff7da4aaed7c3257e77edf6c1b448a779329431ee79d7e", - "sha256:39f3ecaf76cd98e802f094e0d4fbc6dc9c45a8d0c4d185f0f6c2234e14e5f75b", - "sha256:457499c79fa84593f22454bbd27670227874cd2ff5d6c84e60575c8b50a69619", - "sha256:49e70d111fee47284d9dd867c9bb9a7058a3c617274900780c43e38d90fe1205", - "sha256:4c75507d0a55378240f781599c30e7776674dbaf883a46d1c90f37e563453480", - "sha256:4c863140fafc615c14a4bf4efd0f4425c02230eb8ef02784c9a156461e62c965", - "sha256:4d8908b3bee1c889e547867ca4cdc54e5ab6be6d3e078556814a22457f49423c", - "sha256:5b9eb0ca724a241683c9685a484da9d35c872fd42756574a7cfbf58af26677fd", - "sha256:6022cecf8f44e36af10bd9118ca71f371078b4c168b6e0fab43d4a889985dbb5", - "sha256:6150ffa5c767bc6332df27157d95442c379b7dce3a38dff89c0f39b63275696f", - "sha256:62828cada4a2b850dbef89c81f5a33741898b305db244904de418cc957ff05dc", - "sha256:7b4182299f251060996af5249c286bae9361fa8c6a9cda5efc29fe8bfd6062ec", - "sha256:94b34f32646ca0414237168d68a9157cb3889f06b096612afdd296003fdd32fd", - "sha256:9ce6889abac9a42afd07a562c2d6d4b2b7134f83f18571d859b25624a331c90b", - "sha256:9cffe0f850e89d7c0012a1fb8730f75edd4320a0a731ed0c183904fe6ecfc3a9", - "sha256:a12a813949e5066148712a0626895c26b2578874e4cc63160bb007e6df3436fe", - "sha256:a1eea9aecf761c661d096d39ed9026574de8adb2ae1c5bd7b33558af884fb2ce", - "sha256:a31f94983fecbac95e58388210427d68cd30fe8a36927980fab9c20062645609", - "sha256:ac58bdee53cbeba2ecad824fa8159493f0bf3b8ea4e93feb06c9a465d6c87da8", - "sha256:af3f4485884750dddd9c25cb7e3915d83c2db92488b38ccb77dd594eac84c4a0", - "sha256:b33d2bc4f69caedcd0a275329eb2198f560b325605810895627be5d4b876bf7f", - "sha256:b59c0ffceff8d4d3996a45f2bb6f4c207f94684a96bf3d9728dbb77428dd8cb8", - "sha256:bb6834cbd983b19f06908b45bfc2dad6ac9479ae04abe923a275b5f48f1a186b", - "sha256:bd3db01f59fdcbce5b22afad19e390260d6d0222f35a1023d9adc5690a889364", - "sha256:bd7c23df857d488f418439686d3b10ae2fbf9bc256cd045b37a8c16575ea1040", - "sha256:c2528d60e398c7c4c799d56f907664673a807635b857df18f7ae64d3e6ce2d9f", - "sha256:d31a63bc6e6d87f77d71e1abbd7387ab817a66733734883d1fc0021ed9bfa083", - "sha256:d4492d82b3bc7fbb7e3610747b159869468079fe149ec5c4d771fa1f614a14df", - "sha256:ddcb8581510311e13421b1f544403c16e901c4e8f09083c881fab2be80ee31ba", - "sha256:e1d59258c3c67c865435d8fbeb35f8c59b8bef3d6f46c1f29f6123556af28445", - "sha256:eb3315a8a236ee19b6df481fc5f997436e8ade24a9f03dfdc6bd490fea20c6da", - "sha256:ef2b055471c0eb466033760a521efb9d8a32b99ab907fc8358481a1dd29e3bd3", - "sha256:ef5adb9a3b1d0c645ff694f9bca7702ec2c70f4d734f9922ea34de02294fdf72", - "sha256:f32c38afb74bd98ce26de7cc74a67b40afb7b05aae7b42924ea990d51e4dac02", - "sha256:fe0ccca550bb8e5abc22f530ec0466136379c01321fd94f30a22231e8a48d985" + "sha256:0274c1cb63625972c0c007ab14dd9ba9e199c36ae1a231ce45d725cbcbfd10a8", + "sha256:0d7e03c0f9a4f90dc18d4e77e9ef4ec7b7bbb437f7f675be8e530d65ae6ef956", + "sha256:11f8d2554e52f459918f7b8e6aa20ec2a3bce35ce95c1f0ef4ba36fbda306df5", + "sha256:139d8d2e1c1dd52d78682f505e980f592ba53c9f73bd6be102233e358b401063", + "sha256:16a7380e943a6d52472096cb7ad5264ecee36ed60888e2a3d3814991a0107286", + "sha256:171f311cb758de7da13fc53af221ae47a5877be5a0843a9fe150818c51ed276a", + "sha256:18fc2b4ed8e4a8fe849d18dce4bd3c7ea637758c6343a1f2bae1e9bd4c9f4686", + "sha256:1c203f617abc0dde5792beb586f827021069fb6d403d7f4d5c2b543d87edceb9", + "sha256:1c2559d6cffc94890b0529ea7eeecc20d6fadc1539273aa27faf503eb4656d8f", + "sha256:1c88dfb9e0c77612febebb6ac69d44a8d81e3dc60f993215425b62c1161353f4", + "sha256:1e9dc350fb4c58adc64df3e0703ab076f60aac06e67d48b3848c23647ae4310e", + "sha256:247b9d16535acaa766d03037d8e8fb20866d054d3c7fbf6fd1f993f11fc60ca0", + "sha256:266270c6f6608340f6c9836a0fb9b367be61dde0c9a9a18d5ece97774105ff3e", + "sha256:34b9071c040d6fe45d9826cbbe3727d20d83f1b6110d219b83eb0e2a01d79488", + "sha256:3d7d1f8871998cdff5d2ff6a087e5e1780139abe2838e85b0b46b7ae6cc25399", + "sha256:461e3ae84cd90b30f8d533f07d87c00379644205b1d33a5ea03381edc4b69431", + "sha256:464b423bc2a009088f19bdf1f232299e8b6917963e2b7e1d277da5041f33a779", + "sha256:491b1917afdd8638a05b611a56d46587d5a632cabead889a5440f7c638bc6ed9", + "sha256:4a1b1208102be6e851f20066bf0e7a96b7d48a07c9b0cfe6d0d4545c2f6cadab", + "sha256:575bcaf957a25d1194903a10bc9f316c136c19f24e0985a2b9b5608bdf5dbfe0", + "sha256:5c6b28956b7b232ae801406e529ad7b350d3f09a4fde958dfdf3c0520cdde0dd", + "sha256:5d16edfc3fc09968e09ddffada434b3bf989bf4911535e04eada58469873e28e", + "sha256:5fd1810973a375ca0e097dee059c407913ba35723b111df75671a1976efa04bc", + "sha256:67b7f17679fa62ec82b7e3e611c43a016b887bd64fb933b3ae8638583006c6d6", + "sha256:68ce4788b7d93e47f84edd3f1f95acdcd142ae60bc0e5493bfd120683d2d4316", + "sha256:6d3364b999c62f539cd403f8123ae426da946e142312a514162adb2addd8d808", + "sha256:6e739530c662a8d6d42c37c2ed52a6f0932c2d4a3e8c1f90692ad0ce1274abe0", + "sha256:6fdd887f17c2f4572ce548461e4f96396681212d858cae7bd52ba3310bc6f00f", + "sha256:78e6ad33cf2e2e80c5dfaaa0beec3d61face0fb650557100ee36db808bfa6843", + "sha256:884c3f9d42d7218304bc74a8a7693d172685c84bd7ab2bab1ee567b769696df9", + "sha256:8d8faf05be5ec8e02a4d86f616fc2a0322ff4a4ce26c0f09d9f7fb5330a35c95", + "sha256:999c71939aad2780f003979b25ac5b8f2df651dac7b38fb8ce6c46ba5abe6ae9", + "sha256:99ad97258985328b4f207a5e777c1b44a83bfe7cf1f87b99f9c11d4ee477c4de", + "sha256:9e6c93b5b2dbcedad20a2f18ec22cae47da0d705d454308063421a3b290d9ea4", + "sha256:ab459a1cbbf18e8698399c595a01f6dcc5c138220ca3ea9e7e6126232d102bb4", + "sha256:b69303ceb2e4d4f146bf82fda78891ef7bcd80c41bf16bfca3d0d7eb545448aa", + "sha256:b7caf9b241464c404613512d5594a6e2ff0cc9cb5615c9475cc1d9b514218ae8", + "sha256:b95a225d4948b26a28c08307a60ac00fb8671b14f2047fc5476613252a129776", + "sha256:bd2f1ae63998da104f16a8b788f685e55d65760cd1929518fd94cd682bf03e41", + "sha256:be16975d94c320432657ad2402f6760990cb640c161ae6da1363051805fa8108", + "sha256:ce96dd400486e80ac7d195b2d800b03e3e6a787e2a522bfb83755938465a819e", + "sha256:dbd50d0a0539ae2e96e537553aff6d02c10ed165ef40c65b0e27e744a0f10af8", + "sha256:dd10c26b4eadae44783c45ad6655220426f971c61d9b239e6f7b16d5cdaaa727", + "sha256:ebeac59e9e1eb4b84940d076d9f9a6cec0064e241818bcb6e32124cc5c3e377a" ], "markers": "python_version >= '3.9'", - "version": "==1.2.1" + "version": "==1.2.0" }, "coverage": { "extras": [ "toml" ], "hashes": [ - "sha256:0086cd4fc71b7d485ac93ca4239c8f75732c2ae3ba83f6be1c9be59d9e2c6382", - "sha256:01c322ef2bbe15057bc4bf132b525b7e3f7206f071799eb8aa6ad1940bcf5fb1", - "sha256:03cafe82c1b32b770a29fd6de923625ccac3185a54a5e66606da26d105f37dac", - "sha256:044a0985a4f25b335882b0966625270a8d9db3d3409ddc49a4eb00b0ef5e8cee", - "sha256:07ed352205574aad067482e53dd606926afebcb5590653121063fbf4e2175166", - "sha256:0d1b923fc4a40c5832be4f35a5dab0e5ff89cddf83bb4174499e02ea089daf57", - "sha256:0e7b27d04131c46e6894f23a4ae186a6a2207209a05df5b6ad4caee6d54a222c", - "sha256:1fad32ee9b27350687035cb5fdf9145bc9cf0a094a9577d43e909948ebcfa27b", - "sha256:289cc803fa1dc901f84701ac10c9ee873619320f2f9aff38794db4a4a0268d51", - "sha256:3c59105f8d58ce500f348c5b56163a4113a440dad6daa2294b5052a10db866da", - "sha256:46c3d091059ad0b9c59d1034de74a7f36dcfa7f6d3bde782c49deb42438f2450", - "sha256:482855914928c8175735a2a59c8dc5806cf7d8f032e4820d52e845d1f731dca2", - "sha256:49c76cdfa13015c4560702574bad67f0e15ca5a2872c6a125f6327ead2b731dd", - "sha256:4b03741e70fb811d1a9a1d75355cf391f274ed85847f4b78e35459899f57af4d", - "sha256:4bea27c4269234e06f621f3fac3925f56ff34bc14521484b8f66a580aacc2e7d", - "sha256:4d5fae0a22dc86259dee66f2cc6c1d3e490c4a1214d7daa2a93d07491c5c04b6", - "sha256:543ef9179bc55edfd895154a51792b01c017c87af0ebaae092720152e19e42ca", - "sha256:54dece71673b3187c86226c3ca793c5f891f9fc3d8aa183f2e3653da18566169", - "sha256:6379688fb4cfa921ae349c76eb1a9ab26b65f32b03d46bb0eed841fd4cb6afb1", - "sha256:65fa405b837060db569a61ec368b74688f429b32fa47a8929a7a2f9b47183713", - "sha256:6616d1c9bf1e3faea78711ee42a8b972367d82ceae233ec0ac61cc7fec09fa6b", - "sha256:6fe885135c8a479d3e37a7aae61cbd3a0fb2deccb4dda3c25f92a49189f766d6", - "sha256:7221f9ac9dad9492cecab6f676b3eaf9185141539d5c9689d13fd6b0d7de840c", - "sha256:76d5f82213aa78098b9b964ea89de4617e70e0d43e97900c2778a50856dac605", - "sha256:7792f0ab20df8071d669d929c75c97fecfa6bcab82c10ee4adb91c7a54055463", - "sha256:831b476d79408ab6ccfadaaf199906c833f02fdb32c9ab907b1d4aa0713cfa3b", - "sha256:9146579352d7b5f6412735d0f203bbd8d00113a680b66565e205bc605ef81bc6", - "sha256:9cc44bf0315268e253bf563f3560e6c004efe38f76db03a1558274a6e04bf5d5", - "sha256:a73d18625f6a8a1cbb11eadc1d03929f9510f4131879288e3f7922097a429f63", - "sha256:a8659fd33ee9e6ca03950cfdcdf271d645cf681609153f218826dd9805ab585c", - "sha256:a94925102c89247530ae1dab7dc02c690942566f22e189cbd53579b0693c0783", - "sha256:ad4567d6c334c46046d1c4c20024de2a1c3abc626817ae21ae3da600f5779b44", - "sha256:b2e16f4cd2bc4d88ba30ca2d3bbf2f21f00f382cf4e1ce3b1ddc96c634bc48ca", - "sha256:bbdf9a72403110a3bdae77948b8011f644571311c2fb35ee15f0f10a8fc082e8", - "sha256:beb08e8508e53a568811016e59f3234d29c2583f6b6e28572f0954a6b4f7e03d", - "sha256:c4cbe651f3904e28f3a55d6f371203049034b4ddbce65a54527a3f189ca3b390", - "sha256:c7b525ab52ce18c57ae232ba6f7010297a87ced82a2383b1afd238849c1ff933", - "sha256:ca5d79cfdae420a1d52bf177de4bc2289c321d6c961ae321503b2ca59c17ae67", - "sha256:cdab02a0a941af190df8782aafc591ef3ad08824f97850b015c8c6a8b3877b0b", - "sha256:d17c6a415d68cfe1091d3296ba5749d3d8696e42c37fca5d4860c5bf7b729f03", - "sha256:d39bd10f0ae453554798b125d2f39884290c480f56e8a02ba7a6ed552005243b", - "sha256:d4b3cd1ca7cd73d229487fa5caca9e4bc1f0bca96526b922d61053ea751fe791", - "sha256:d50a252b23b9b4dfeefc1f663c568a221092cbaded20a05a11665d0dbec9b8fb", - "sha256:da8549d17489cd52f85a9829d0e1d91059359b3c54a26f28bec2c5d369524807", - "sha256:dcd070b5b585b50e6617e8972f3fbbee786afca71b1936ac06257f7e178f00f6", - "sha256:ddaaa91bfc4477d2871442bbf30a125e8fe6b05da8a0015507bfbf4718228ab2", - "sha256:df423f351b162a702c053d5dddc0fc0ef9a9e27ea3f449781ace5f906b664428", - "sha256:dff044f661f59dace805eedb4a7404c573b6ff0cdba4a524141bc63d7be5c7fd", - "sha256:e7e128f85c0b419907d1f38e616c4f1e9f1d1b37a7949f44df9a73d5da5cd53c", - "sha256:ed8d1d1821ba5fc88d4a4f45387b65de52382fa3ef1f0115a4f7a20cdfab0e94", - "sha256:f2501d60d7497fd55e391f423f965bbe9e650e9ffc3c627d5f0ac516026000b8", - "sha256:f7db0b6ae1f96ae41afe626095149ecd1b212b424626175a6633c2999eaad45b" + "sha256:04387a4a6ecb330c1878907ce0dc04078ea72a869263e53c72a1ba5bbdf380ca", + "sha256:0676cd0ba581e514b7f726495ea75aba3eb20899d824636c6f59b0ed2f88c471", + "sha256:0e8d06778e8fbffccfe96331a3946237f87b1e1d359d7fbe8b06b96c95a5407a", + "sha256:0eb3c2f32dabe3a4aaf6441dde94f35687224dfd7eb2a7f47f3fd9428e421058", + "sha256:109f5985182b6b81fe33323ab4707011875198c41964f014579cf82cebf2bb85", + "sha256:13eaf476ec3e883fe3e5fe3707caeb88268a06284484a3daf8250259ef1ba143", + "sha256:164fdcc3246c69a6526a59b744b62e303039a81e42cfbbdc171c91a8cc2f9446", + "sha256:26776ff6c711d9d835557ee453082025d871e30b3fd6c27fcef14733f67f0590", + "sha256:26f66da8695719ccf90e794ed567a1549bb2644a706b41e9f6eae6816b398c4a", + "sha256:29f3abe810930311c0b5d1a7140f6395369c3db1be68345638c33eec07535105", + "sha256:316543f71025a6565677d84bc4df2114e9b6a615aa39fb165d697dba06a54af9", + "sha256:36b0ea8ab20d6a7564e89cb6135920bc9188fb5f1f7152e94e8300b7b189441a", + "sha256:3cc9d4bc55de8003663ec94c2f215d12d42ceea128da8f0f4036235a119c88ac", + "sha256:485e9f897cf4856a65a57c7f6ea3dc0d4e6c076c87311d4bc003f82cfe199d25", + "sha256:5040148f4ec43644702e7b16ca864c5314ccb8ee0751ef617d49aa0e2d6bf4f2", + "sha256:51456e6fa099a8d9d91497202d9563a320513fcf59f33991b0661a4a6f2ad450", + "sha256:53d7d9158ee03956e0eadac38dfa1ec8068431ef8058fe6447043db1fb40d932", + "sha256:5a10a4920def78bbfff4eff8a05c51be03e42f1c3735be42d851f199144897ba", + "sha256:5b14b4f8760006bfdb6e08667af7bc2d8d9bfdb648351915315ea17645347137", + "sha256:5b2ccb7548a0b65974860a78c9ffe1173cfb5877460e5a229238d985565574ae", + "sha256:697d1317e5290a313ef0d369650cfee1a114abb6021fa239ca12b4849ebbd614", + "sha256:6ae8c9d301207e6856865867d762a4b6fd379c714fcc0607a84b92ee63feff70", + "sha256:707c0f58cb1712b8809ece32b68996ee1e609f71bd14615bd8f87a1293cb610e", + "sha256:74775198b702868ec2d058cb92720a3c5a9177296f75bd97317c787daf711505", + "sha256:756ded44f47f330666843b5781be126ab57bb57c22adbb07d83f6b519783b870", + "sha256:76f03940f9973bfaee8cfba70ac991825611b9aac047e5c80d499a44079ec0bc", + "sha256:79287fd95585ed36e83182794a57a46aeae0b64ca53929d1176db56aacc83451", + "sha256:799c8f873794a08cdf216aa5d0531c6a3747793b70c53f70e98259720a6fe2d7", + "sha256:7d360587e64d006402b7116623cebf9d48893329ef035278969fa3bbf75b697e", + "sha256:80b5ee39b7f0131ebec7968baa9b2309eddb35b8403d1869e08f024efd883566", + "sha256:815ac2d0f3398a14286dc2cea223a6f338109f9ecf39a71160cd1628786bc6f5", + "sha256:83c2dda2666fe32332f8e87481eed056c8b4d163fe18ecc690b02802d36a4d26", + "sha256:846f52f46e212affb5bcf131c952fb4075b55aae6b61adc9856222df89cbe3e2", + "sha256:936d38794044b26c99d3dd004d8af0035ac535b92090f7f2bb5aa9c8e2f5cd42", + "sha256:9864463c1c2f9cb3b5db2cf1ff475eed2f0b4285c2aaf4d357b69959941aa555", + "sha256:995ea5c48c4ebfd898eacb098164b3cc826ba273b3049e4a889658548e321b43", + "sha256:a1526d265743fb49363974b7aa8d5899ff64ee07df47dd8d3e37dcc0818f09ed", + "sha256:a56de34db7b7ff77056a37aedded01b2b98b508227d2d0979d373a9b5d353daa", + "sha256:a7c97726520f784239f6c62506bc70e48d01ae71e9da128259d61ca5e9788516", + "sha256:b8e99f06160602bc64da35158bb76c73522a4010f0649be44a4e167ff8555952", + "sha256:bb1de682da0b824411e00a0d4da5a784ec6496b6850fdf8c865c1d68c0e318dd", + "sha256:bf477c355274a72435ceb140dc42de0dc1e1e0bf6e97195be30487d8eaaf1a09", + "sha256:bf635a52fc1ea401baf88843ae8708591aa4adff875e5c23220de43b1ccf575c", + "sha256:bfd5db349d15c08311702611f3dccbef4b4e2ec148fcc636cf8739519b4a5c0f", + "sha256:c530833afc4707fe48524a44844493f36d8727f04dcce91fb978c414a8556cc6", + "sha256:cc6d65b21c219ec2072c1293c505cf36e4e913a3f936d80028993dd73c7906b1", + "sha256:cd3c1e4cb2ff0083758f09be0f77402e1bdf704adb7f89108007300a6da587d0", + "sha256:cfd2a8b6b0d8e66e944d47cdec2f47c48fef2ba2f2dff5a9a75757f64172857e", + "sha256:d0ca5c71a5a1765a0f8f88022c52b6b8be740e512980362f7fdbb03725a0d6b9", + "sha256:e7defbb9737274023e2d7af02cac77043c86ce88a907c58f42b580a97d5bcca9", + "sha256:e9d1bf53c4c8de58d22e0e956a79a5b37f754ed1ffdbf1a260d9dcfa2d8a325e", + "sha256:ea81d8f9691bb53f4fb4db603203029643caffc82bf998ab5b59ca05560f4c06" ], "markers": "python_version >= '3.8'", - "version": "==7.6.0" + "version": "==7.4.0" }, "cycler": { "hashes": [ @@ -435,11 +434,11 @@ }, "exceptiongroup": { "hashes": [ - "sha256:3111b9d131c238bec2f8f516e123e14ba243563fb135d3fe885990585aa7795b", - "sha256:47c2edf7c6738fafb49fd34290706d1a1a2f4d1c6df275526b62cbb4aa5393cc" + "sha256:4bfd3996ac73b41e9b9628b04e079f193850720ea5945fc96a08633c66912f14", + "sha256:91f5c769735f051a4290d52edd0858999b57e5876e9f85937691bd4c9fa3ed68" ], "markers": "python_version < '3.11'", - "version": "==1.2.2" + "version": "==1.2.0" }, "flake8": { "hashes": [ @@ -479,60 +478,60 @@ }, "fonttools": { "hashes": [ - "sha256:02569e9a810f9d11f4ae82c391ebc6fb5730d95a0657d24d754ed7763fb2d122", - "sha256:0679a30b59d74b6242909945429dbddb08496935b82f91ea9bf6ad240ec23397", - "sha256:10f5e6c3510b79ea27bb1ebfcc67048cde9ec67afa87c7dd7efa5c700491ac7f", - "sha256:2af40ae9cdcb204fc1d8f26b190aa16534fcd4f0df756268df674a270eab575d", - "sha256:32f029c095ad66c425b0ee85553d0dc326d45d7059dbc227330fc29b43e8ba60", - "sha256:35250099b0cfb32d799fb5d6c651220a642fe2e3c7d2560490e6f1d3f9ae9169", - "sha256:3b3c8ebafbee8d9002bd8f1195d09ed2bd9ff134ddec37ee8f6a6375e6a4f0e8", - "sha256:4824c198f714ab5559c5be10fd1adf876712aa7989882a4ec887bf1ef3e00e31", - "sha256:5ff7e5e9bad94e3a70c5cd2fa27f20b9bb9385e10cddab567b85ce5d306ea923", - "sha256:651390c3b26b0c7d1f4407cad281ee7a5a85a31a110cbac5269de72a51551ba2", - "sha256:6e08f572625a1ee682115223eabebc4c6a2035a6917eac6f60350aba297ccadb", - "sha256:6ed170b5e17da0264b9f6fae86073be3db15fa1bd74061c8331022bca6d09bab", - "sha256:73379d3ffdeecb376640cd8ed03e9d2d0e568c9d1a4e9b16504a834ebadc2dfb", - "sha256:75a157d8d26c06e64ace9df037ee93a4938a4606a38cb7ffaf6635e60e253b7a", - "sha256:791b31ebbc05197d7aa096bbc7bd76d591f05905d2fd908bf103af4488e60670", - "sha256:7b6b35e52ddc8fb0db562133894e6ef5b4e54e1283dff606fda3eed938c36fc8", - "sha256:84ec3fb43befb54be490147b4a922b5314e16372a643004f182babee9f9c3407", - "sha256:8959a59de5af6d2bec27489e98ef25a397cfa1774b375d5787509c06659b3671", - "sha256:9dfdae43b7996af46ff9da520998a32b105c7f098aeea06b2226b30e74fbba88", - "sha256:9e6ceba2a01b448e36754983d376064730690401da1dd104ddb543519470a15f", - "sha256:9efd176f874cb6402e607e4cc9b4a9cd584d82fc34a4b0c811970b32ba62501f", - "sha256:a1c7c5aa18dd3b17995898b4a9b5929d69ef6ae2af5b96d585ff4005033d82f0", - "sha256:aae7bd54187e8bf7fd69f8ab87b2885253d3575163ad4d669a262fe97f0136cb", - "sha256:b21952c092ffd827504de7e66b62aba26fdb5f9d1e435c52477e6486e9d128b2", - "sha256:b96cd370a61f4d083c9c0053bf634279b094308d52fdc2dd9a22d8372fdd590d", - "sha256:becc5d7cb89c7b7afa8321b6bb3dbee0eec2b57855c90b3e9bf5fb816671fa7c", - "sha256:bee32ea8765e859670c4447b0817514ca79054463b6b79784b08a8df3a4d78e3", - "sha256:c6e7170d675d12eac12ad1a981d90f118c06cf680b42a2d74c6c931e54b50719", - "sha256:c818c058404eb2bba05e728d38049438afd649e3c409796723dfc17cd3f08749", - "sha256:c8696544c964500aa9439efb6761947393b70b17ef4e82d73277413f291260a4", - "sha256:c9cd19cf4fe0595ebdd1d4915882b9440c3a6d30b008f3cc7587c1da7b95be5f", - "sha256:d4d0096cb1ac7a77b3b41cd78c9b6bc4a400550e21dc7a92f2b5ab53ed74eb02", - "sha256:d92d3c2a1b39631a6131c2fa25b5406855f97969b068e7e08413325bc0afba58", - "sha256:da33440b1413bad53a8674393c5d29ce64d8c1a15ef8a77c642ffd900d07bfe1", - "sha256:e013aae589c1c12505da64a7d8d023e584987e51e62006e1bb30d72f26522c41", - "sha256:e128778a8e9bc11159ce5447f76766cefbd876f44bd79aff030287254e4752c4", - "sha256:e54f1bba2f655924c1138bbc7fa91abd61f45c68bd65ab5ed985942712864bbb", - "sha256:e5b708073ea3d684235648786f5f6153a48dc8762cdfe5563c57e80787c29fbb", - "sha256:e8bf06b94694251861ba7fdeea15c8ec0967f84c3d4143ae9daf42bbc7717fe3", - "sha256:f08df60fbd8d289152079a65da4e66a447efc1d5d5a4d3f299cdd39e3b2e4a7d", - "sha256:f1f8758a2ad110bd6432203a344269f445a2907dc24ef6bccfd0ac4e14e0d71d", - "sha256:f677ce218976496a587ab17140da141557beb91d2a5c1a14212c994093f2eae2" + "sha256:0255dbc128fee75fb9be364806b940ed450dd6838672a150d501ee86523ac61e", + "sha256:0a00bd0e68e88987dcc047ea31c26d40a3c61185153b03457956a87e39d43c37", + "sha256:0a1d313a415eaaba2b35d6cd33536560deeebd2ed758b9bfb89ab5d97dc5deac", + "sha256:0f750037e02beb8b3569fbff701a572e62a685d2a0e840d75816592280e5feae", + "sha256:13819db8445a0cec8c3ff5f243af6418ab19175072a9a92f6cc8ca7d1452754b", + "sha256:254d9a6f7be00212bf0c3159e0a420eb19c63793b2c05e049eb337f3023c5ecc", + "sha256:29495d6d109cdbabe73cfb6f419ce67080c3ef9ea1e08d5750240fd4b0c4763b", + "sha256:32ab2e9702dff0dd4510c7bb958f265a8d3dd5c0e2547e7b5f7a3df4979abb07", + "sha256:3480eeb52770ff75140fe7d9a2ec33fb67b07efea0ab5129c7e0c6a639c40c70", + "sha256:3a808f3c1d1df1f5bf39be869b6e0c263570cdafb5bdb2df66087733f566ea71", + "sha256:3b629108351d25512d4ea1a8393a2dba325b7b7d7308116b605ea3f8e1be88df", + "sha256:3d71606c9321f6701642bd4746f99b6089e53d7e9817fc6b964e90d9c5f0ecc6", + "sha256:3e2b95dce2ead58fb12524d0ca7d63a63459dd489e7e5838c3cd53557f8933e1", + "sha256:4a5a5318ba5365d992666ac4fe35365f93004109d18858a3e18ae46f67907670", + "sha256:4c811d3c73b6abac275babb8aa439206288f56fdb2c6f8835e3d7b70de8937a7", + "sha256:4e743935139aa485fe3253fc33fe467eab6ea42583fa681223ea3f1a93dd01e6", + "sha256:4ec558c543609e71b2275c4894e93493f65d2f41c15fe1d089080c1d0bb4d635", + "sha256:5465df494f20a7d01712b072ae3ee9ad2887004701b95cb2cc6dcb9c2c97a899", + "sha256:5b60e3afa9635e3dfd3ace2757039593e3bd3cf128be0ddb7a1ff4ac45fa5a50", + "sha256:63fbed184979f09a65aa9c88b395ca539c94287ba3a364517698462e13e457c9", + "sha256:69731e8bea0578b3c28fdb43dbf95b9386e2d49a399e9a4ad736b8e479b08085", + "sha256:6dd58cc03016b281bd2c74c84cdaa6bd3ce54c5a7f47478b7657b930ac3ed8eb", + "sha256:740947906590a878a4bde7dd748e85fefa4d470a268b964748403b3ab2aeed6c", + "sha256:7df26dd3650e98ca45f1e29883c96a0b9f5bb6af8d632a6a108bc744fa0bd9b3", + "sha256:7eb7ad665258fba68fd22228a09f347469d95a97fb88198e133595947a20a184", + "sha256:7ee48bd9d6b7e8f66866c9090807e3a4a56cf43ffad48962725a190e0dd774c8", + "sha256:86e0427864c6c91cf77f16d1fb9bf1bbf7453e824589e8fb8461b6ee1144f506", + "sha256:8f57ecd742545362a0f7186774b2d1c53423ed9ece67689c93a1055b236f638c", + "sha256:90f898cdd67f52f18049250a6474185ef6544c91f27a7bee70d87d77a8daf89c", + "sha256:94208ea750e3f96e267f394d5588579bb64cc628e321dbb1d4243ffbc291b18b", + "sha256:a1c154bb85dc9a4cf145250c88d112d88eb414bad81d4cb524d06258dea1bdc0", + "sha256:a5d77479fb885ef38a16a253a2f4096bc3d14e63a56d6246bfdb56365a12b20c", + "sha256:a86a5ab2873ed2575d0fcdf1828143cfc6b977ac448e3dc616bb1e3d20efbafa", + "sha256:ac71e2e201df041a2891067dc36256755b1229ae167edbdc419b16da78732c2f", + "sha256:b3e1304e5f19ca861d86a72218ecce68f391646d85c851742d265787f55457a4", + "sha256:b8be28c036b9f186e8c7eaf8a11b42373e7e4949f9e9f370202b9da4c4c3f56c", + "sha256:c19044256c44fe299d9a73456aabee4b4d06c6b930287be93b533b4737d70aa1", + "sha256:d49ce3ea7b7173faebc5664872243b40cf88814ca3eb135c4a3cdff66af71946", + "sha256:e040f905d542362e07e72e03612a6270c33d38281fd573160e1003e43718d68d", + "sha256:eabae77a07c41ae0b35184894202305c3ad211a93b2eb53837c2a1143c8bc952", + "sha256:f791446ff297fd5f1e2247c188de53c1bfb9dd7f0549eba55b73a3c2087a2703", + "sha256:f83a4daef6d2a202acb9bf572958f91cfde5b10c8ee7fb1d09a4c81e5d851fd8" ], "markers": "python_version >= '3.8'", - "version": "==4.53.1" + "version": "==4.47.2" }, "hypothesis": { "hashes": [ - "sha256:2beb7a148e95a2067563bcca017d71cc286805c792e43ec5cb155ed6d0a1990d", - "sha256:3b0d080bfd3b303e91388507ac7edebd7039ffcc96ac2cfcdc3c45806352c09f" + "sha256:848ea0952f0bdfd02eac59e41b03f1cbba8fa2cffeffa8db328bbd6cfe159974", + "sha256:955a57e56be4607c81c17ca53e594af54aadeed91e07b88bb7f84e8208ea7739" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==6.96.4" + "version": "==6.96.1" }, "iniconfig": { "hashes": [ @@ -544,19 +543,10 @@ }, "jsonschema": { "hashes": [ - "sha256:d71497fef26351a33265337fa77ffeb82423f3ea21283cd9467bb03999266bc4", - "sha256:fbadb6f8b144a8f8cf9f0b89ba94501d143e50411a1278633f56a7acf7fd5566" - ], - "markers": "python_version >= '3.8'", - "version": "==4.23.0" - }, - "jsonschema-specifications": { - "hashes": [ - "sha256:48a76787b3e70f5ed53f1160d2b81f586e4ca6d1548c5de7085d1682674764cc", - "sha256:87e4fdf3a94858b8a2ba2778d9ba57d8a9cafca7c7489c46ba0d30a8bc6a9c3c" + "sha256:5f9c0a719ca2ce14c5de2fd350a64fd2d13e8539db29836a86adc990bb1a068f", + "sha256:8d4a2b7b6c2237e0199c8ea1a6d3e05bf118e289ae2b9d7ba444182a2959560d" ], - "markers": "python_version >= '3.8'", - "version": "==2023.12.1" + "version": "==3.0.2" }, "kiwisolver": { "hashes": [ @@ -670,38 +660,38 @@ }, "matplotlib": { "hashes": [ - "sha256:1c13f041a7178f9780fb61cc3a2b10423d5e125480e4be51beaf62b172413b67", - "sha256:232ce322bfd020a434caaffbd9a95333f7c2491e59cfc014041d95e38ab90d1c", - "sha256:493e9f6aa5819156b58fce42b296ea31969f2aab71c5b680b4ea7a3cb5c07d94", - "sha256:50bac6e4d77e4262c4340d7a985c30912054745ec99756ce213bfbc3cb3808eb", - "sha256:606e3b90897554c989b1e38a258c626d46c873523de432b1462f295db13de6f9", - "sha256:6209e5c9aaccc056e63b547a8152661324404dd92340a6e479b3a7f24b42a5d0", - "sha256:6485ac1f2e84676cff22e693eaa4fbed50ef5dc37173ce1f023daef4687df616", - "sha256:6addbd5b488aedb7f9bc19f91cd87ea476206f45d7116fcfe3d31416702a82fa", - "sha256:72f9322712e4562e792b2961971891b9fbbb0e525011e09ea0d1f416c4645661", - "sha256:7a6769f58ce51791b4cb8b4d7642489df347697cd3e23d88266aaaee93b41d9a", - "sha256:8080d5081a86e690d7688ffa542532e87f224c38a6ed71f8fbed34dd1d9fedae", - "sha256:843cbde2f0946dadd8c5c11c6d91847abd18ec76859dc319362a0964493f0ba6", - "sha256:8aac397d5e9ec158960e31c381c5ffc52ddd52bd9a47717e2a694038167dffea", - "sha256:8f65c9f002d281a6e904976007b2d46a1ee2bcea3a68a8c12dda24709ddc9106", - "sha256:90df07db7b599fe7035d2f74ab7e438b656528c68ba6bb59b7dc46af39ee48ef", - "sha256:9bb0189011785ea794ee827b68777db3ca3f93f3e339ea4d920315a0e5a78d54", - "sha256:a0e47eda4eb2614300fc7bb4657fced3e83d6334d03da2173b09e447418d499f", - "sha256:abc9d838f93583650c35eca41cfcec65b2e7cb50fd486da6f0c49b5e1ed23014", - "sha256:ac24233e8f2939ac4fd2919eed1e9c0871eac8057666070e94cbf0b33dd9c338", - "sha256:b12ba985837e4899b762b81f5b2845bd1a28f4fdd1a126d9ace64e9c4eb2fb25", - "sha256:b7a2a253d3b36d90c8993b4620183b55665a429da8357a4f621e78cd48b2b30b", - "sha256:c7064120a59ce6f64103c9cefba8ffe6fba87f2c61d67c401186423c9a20fd35", - "sha256:c89ee9314ef48c72fe92ce55c4e95f2f39d70208f9f1d9db4e64079420d8d732", - "sha256:cc4ccdc64e3039fc303defd119658148f2349239871db72cd74e2eeaa9b80b71", - "sha256:ce1edd9f5383b504dbc26eeea404ed0a00656c526638129028b758fd43fc5f10", - "sha256:ecd79298550cba13a43c340581a3ec9c707bd895a6a061a78fa2524660482fc0", - "sha256:f51c4c869d4b60d769f7b4406eec39596648d9d70246428745a681c327a8ad30", - "sha256:fb44f53af0a62dc80bba4443d9b27f2fde6acfdac281d95bc872dc148a6509cc" + "sha256:01a978b871b881ee76017152f1f1a0cbf6bd5f7b8ff8c96df0df1bd57d8755a1", + "sha256:03f9d160a29e0b65c0790bb07f4f45d6a181b1ac33eb1bb0dd225986450148f0", + "sha256:091275d18d942cf1ee9609c830a1bc36610607d8223b1b981c37d5c9fc3e46a4", + "sha256:09796f89fb71a0c0e1e2f4bdaf63fb2cefc84446bb963ecdeb40dfee7dfa98c7", + "sha256:0f4fc5d72b75e2c18e55eb32292659cf731d9d5b312a6eb036506304f4675630", + "sha256:172f4d0fbac3383d39164c6caafd3255ce6fa58f08fc392513a0b1d3b89c4f89", + "sha256:1b0f3b8ea0e99e233a4bcc44590f01604840d833c280ebb8fe5554fd3e6cfe8d", + "sha256:3773002da767f0a9323ba1a9b9b5d00d6257dbd2a93107233167cfb581f64717", + "sha256:46a569130ff53798ea5f50afce7406e91fdc471ca1e0e26ba976a8c734c9427a", + "sha256:4c318c1e95e2f5926fba326f68177dee364aa791d6df022ceb91b8221bd0a627", + "sha256:4e208f46cf6576a7624195aa047cb344a7f802e113bb1a06cfd4bee431de5e31", + "sha256:533b0e3b0c6768eef8cbe4b583731ce25a91ab54a22f830db2b031e83cca9213", + "sha256:5864bdd7da445e4e5e011b199bb67168cdad10b501750367c496420f2ad00843", + "sha256:5ba9cbd8ac6cf422f3102622b20f8552d601bf8837e49a3afed188d560152788", + "sha256:6f9c6976748a25e8b9be51ea028df49b8e561eed7809146da7a47dbecebab367", + "sha256:7c48d9e221b637c017232e3760ed30b4e8d5dfd081daf327e829bf2a72c731b4", + "sha256:830f00640c965c5b7f6bc32f0d4ce0c36dfe0379f7dd65b07a00c801713ec40a", + "sha256:9a5430836811b7652991939012f43d2808a2db9b64ee240387e8c43e2e5578c8", + "sha256:aa11b3c6928a1e496c1a79917d51d4cd5d04f8a2e75f21df4949eeefdf697f4b", + "sha256:b78e4f2cedf303869b782071b55fdde5987fda3038e9d09e58c91cc261b5ad18", + "sha256:b9576723858a78751d5aacd2497b8aef29ffea6d1c95981505877f7ac28215c6", + "sha256:bddfb1db89bfaa855912261c805bd0e10218923cc262b9159a49c29a7a1c1afa", + "sha256:c7d36c2209d9136cd8e02fab1c0ddc185ce79bc914c45054a9f514e44c787917", + "sha256:d1095fecf99eeb7384dabad4bf44b965f929a5f6079654b681193edf7169ec20", + "sha256:d7b1704a530395aaf73912be741c04d181f82ca78084fbd80bc737be04848331", + "sha256:d86593ccf546223eb75a39b44c32788e6f6440d13cfc4750c1c15d0fcb850b63", + "sha256:deaed9ad4da0b1aea77fe0aa0cebb9ef611c70b3177be936a95e5d01fa05094f", + "sha256:ef8345b48e95cee45ff25192ed1f4857273117917a4dcd48e3905619bcd9c9b8" ], "index": "pypi", "markers": "python_version >= '3.9'", - "version": "==3.8.4" + "version": "==3.8.2" }, "mccabe": { "hashes": [ @@ -791,16 +781,16 @@ }, "opentrons-shared-data": { "editable": true, - "markers": "python_version >= '3.10'", + "markers": "python_version >= '3.7'", "path": "../shared-data/python" }, "packaging": { "hashes": [ - "sha256:026ed72c8ed3fcce5bf8950572258698927fd1dbda10a5e981cdf0ac37f4f002", - "sha256:5b8f2217dbdbd2f7f384c41c628544e6d52f2d0f53c6d0c3ea61aa5d1d7ff124" + "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5", + "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7" ], - "markers": "python_version >= '3.8'", - "version": "==24.1" + "markers": "python_version >= '3.7'", + "version": "==23.2" }, "pathspec": { "hashes": [ @@ -812,105 +802,93 @@ }, "pillow": { "hashes": [ - "sha256:02a2be69f9c9b8c1e97cf2713e789d4e398c751ecfd9967c18d0ce304efbf885", - "sha256:030abdbe43ee02e0de642aee345efa443740aa4d828bfe8e2eb11922ea6a21ea", - "sha256:06b2f7898047ae93fad74467ec3d28fe84f7831370e3c258afa533f81ef7f3df", - "sha256:0755ffd4a0c6f267cccbae2e9903d95477ca2f77c4fcf3a3a09570001856c8a5", - "sha256:0a9ec697746f268507404647e531e92889890a087e03681a3606d9b920fbee3c", - "sha256:0ae24a547e8b711ccaaf99c9ae3cd975470e1a30caa80a6aaee9a2f19c05701d", - "sha256:134ace6dc392116566980ee7436477d844520a26a4b1bd4053f6f47d096997fd", - "sha256:166c1cd4d24309b30d61f79f4a9114b7b2313d7450912277855ff5dfd7cd4a06", - "sha256:1b5dea9831a90e9d0721ec417a80d4cbd7022093ac38a568db2dd78363b00908", - "sha256:1d846aea995ad352d4bdcc847535bd56e0fd88d36829d2c90be880ef1ee4668a", - "sha256:1ef61f5dd14c300786318482456481463b9d6b91ebe5ef12f405afbba77ed0be", - "sha256:297e388da6e248c98bc4a02e018966af0c5f92dfacf5a5ca22fa01cb3179bca0", - "sha256:298478fe4f77a4408895605f3482b6cc6222c018b2ce565c2b6b9c354ac3229b", - "sha256:29dbdc4207642ea6aad70fbde1a9338753d33fb23ed6956e706936706f52dd80", - "sha256:2db98790afc70118bd0255c2eeb465e9767ecf1f3c25f9a1abb8ffc8cfd1fe0a", - "sha256:32cda9e3d601a52baccb2856b8ea1fc213c90b340c542dcef77140dfa3278a9e", - "sha256:37fb69d905be665f68f28a8bba3c6d3223c8efe1edf14cc4cfa06c241f8c81d9", - "sha256:416d3a5d0e8cfe4f27f574362435bc9bae57f679a7158e0096ad2beb427b8696", - "sha256:43efea75eb06b95d1631cb784aa40156177bf9dd5b4b03ff38979e048258bc6b", - "sha256:4b35b21b819ac1dbd1233317adeecd63495f6babf21b7b2512d244ff6c6ce309", - "sha256:4d9667937cfa347525b319ae34375c37b9ee6b525440f3ef48542fcf66f2731e", - "sha256:5161eef006d335e46895297f642341111945e2c1c899eb406882a6c61a4357ab", - "sha256:543f3dc61c18dafb755773efc89aae60d06b6596a63914107f75459cf984164d", - "sha256:551d3fd6e9dc15e4c1eb6fc4ba2b39c0c7933fa113b220057a34f4bb3268a060", - "sha256:59291fb29317122398786c2d44427bbd1a6d7ff54017075b22be9d21aa59bd8d", - "sha256:5b001114dd152cfd6b23befeb28d7aee43553e2402c9f159807bf55f33af8a8d", - "sha256:5b4815f2e65b30f5fbae9dfffa8636d992d49705723fe86a3661806e069352d4", - "sha256:5dc6761a6efc781e6a1544206f22c80c3af4c8cf461206d46a1e6006e4429ff3", - "sha256:5e84b6cc6a4a3d76c153a6b19270b3526a5a8ed6b09501d3af891daa2a9de7d6", - "sha256:6209bb41dc692ddfee4942517c19ee81b86c864b626dbfca272ec0f7cff5d9fb", - "sha256:673655af3eadf4df6b5457033f086e90299fdd7a47983a13827acf7459c15d94", - "sha256:6c762a5b0997f5659a5ef2266abc1d8851ad7749ad9a6a5506eb23d314e4f46b", - "sha256:7086cc1d5eebb91ad24ded9f58bec6c688e9f0ed7eb3dbbf1e4800280a896496", - "sha256:73664fe514b34c8f02452ffb73b7a92c6774e39a647087f83d67f010eb9a0cf0", - "sha256:76a911dfe51a36041f2e756b00f96ed84677cdeb75d25c767f296c1c1eda1319", - "sha256:780c072c2e11c9b2c7ca37f9a2ee8ba66f44367ac3e5c7832afcfe5104fd6d1b", - "sha256:7928ecbf1ece13956b95d9cbcfc77137652b02763ba384d9ab508099a2eca856", - "sha256:7970285ab628a3779aecc35823296a7869f889b8329c16ad5a71e4901a3dc4ef", - "sha256:7a8d4bade9952ea9a77d0c3e49cbd8b2890a399422258a77f357b9cc9be8d680", - "sha256:7c1ee6f42250df403c5f103cbd2768a28fe1a0ea1f0f03fe151c8741e1469c8b", - "sha256:7dfecdbad5c301d7b5bde160150b4db4c659cee2b69589705b6f8a0c509d9f42", - "sha256:812f7342b0eee081eaec84d91423d1b4650bb9828eb53d8511bcef8ce5aecf1e", - "sha256:866b6942a92f56300012f5fbac71f2d610312ee65e22f1aa2609e491284e5597", - "sha256:86dcb5a1eb778d8b25659d5e4341269e8590ad6b4e8b44d9f4b07f8d136c414a", - "sha256:87dd88ded2e6d74d31e1e0a99a726a6765cda32d00ba72dc37f0651f306daaa8", - "sha256:8bc1a764ed8c957a2e9cacf97c8b2b053b70307cf2996aafd70e91a082e70df3", - "sha256:8d4d5063501b6dd4024b8ac2f04962d661222d120381272deea52e3fc52d3736", - "sha256:8f0aef4ef59694b12cadee839e2ba6afeab89c0f39a3adc02ed51d109117b8da", - "sha256:930044bb7679ab003b14023138b50181899da3f25de50e9dbee23b61b4de2126", - "sha256:950be4d8ba92aca4b2bb0741285a46bfae3ca699ef913ec8416c1b78eadd64cd", - "sha256:961a7293b2457b405967af9c77dcaa43cc1a8cd50d23c532e62d48ab6cdd56f5", - "sha256:9b885f89040bb8c4a1573566bbb2f44f5c505ef6e74cec7ab9068c900047f04b", - "sha256:9f4727572e2918acaa9077c919cbbeb73bd2b3ebcfe033b72f858fc9fbef0026", - "sha256:a02364621fe369e06200d4a16558e056fe2805d3468350df3aef21e00d26214b", - "sha256:a985e028fc183bf12a77a8bbf36318db4238a3ded7fa9df1b9a133f1cb79f8fc", - "sha256:ac1452d2fbe4978c2eec89fb5a23b8387aba707ac72810d9490118817d9c0b46", - "sha256:b15e02e9bb4c21e39876698abf233c8c579127986f8207200bc8a8f6bb27acf2", - "sha256:b2724fdb354a868ddf9a880cb84d102da914e99119211ef7ecbdc613b8c96b3c", - "sha256:bbc527b519bd3aa9d7f429d152fea69f9ad37c95f0b02aebddff592688998abe", - "sha256:bcd5e41a859bf2e84fdc42f4edb7d9aba0a13d29a2abadccafad99de3feff984", - "sha256:bd2880a07482090a3bcb01f4265f1936a903d70bc740bfcb1fd4e8a2ffe5cf5a", - "sha256:bee197b30783295d2eb680b311af15a20a8b24024a19c3a26431ff83eb8d1f70", - "sha256:bf2342ac639c4cf38799a44950bbc2dfcb685f052b9e262f446482afaf4bffca", - "sha256:c76e5786951e72ed3686e122d14c5d7012f16c8303a674d18cdcd6d89557fc5b", - "sha256:cbed61494057c0f83b83eb3a310f0bf774b09513307c434d4366ed64f4128a91", - "sha256:cfdd747216947628af7b259d274771d84db2268ca062dd5faf373639d00113a3", - "sha256:d7480af14364494365e89d6fddc510a13e5a2c3584cb19ef65415ca57252fb84", - "sha256:dbc6ae66518ab3c5847659e9988c3b60dc94ffb48ef9168656e0019a93dbf8a1", - "sha256:dc3e2db6ba09ffd7d02ae9141cfa0ae23393ee7687248d46a7507b75d610f4f5", - "sha256:dfe91cb65544a1321e631e696759491ae04a2ea11d36715eca01ce07284738be", - "sha256:e4d49b85c4348ea0b31ea63bc75a9f3857869174e2bf17e7aba02945cd218e6f", - "sha256:e4db64794ccdf6cb83a59d73405f63adbe2a1887012e308828596100a0b2f6cc", - "sha256:e553cad5179a66ba15bb18b353a19020e73a7921296a7979c4a2b7f6a5cd57f9", - "sha256:e88d5e6ad0d026fba7bdab8c3f225a69f063f116462c49892b0149e21b6c0a0e", - "sha256:ecd85a8d3e79cd7158dec1c9e5808e821feea088e2f69a974db5edf84dc53141", - "sha256:f5b92f4d70791b4a67157321c4e8225d60b119c5cc9aee8ecf153aace4aad4ef", - "sha256:f5f0c3e969c8f12dd2bb7e0b15d5c468b51e5017e01e2e867335c81903046a22", - "sha256:f7baece4ce06bade126fb84b8af1c33439a76d8a6fd818970215e0560ca28c27", - "sha256:ff25afb18123cea58a591ea0244b92eb1e61a1fd497bf6d6384f09bc3262ec3e", - "sha256:ff337c552345e95702c5fde3158acb0625111017d0e5f24bf3acdb9cc16b90d1" + "sha256:0304004f8067386b477d20a518b50f3fa658a28d44e4116970abfcd94fac34a8", + "sha256:0689b5a8c5288bc0504d9fcee48f61a6a586b9b98514d7d29b840143d6734f39", + "sha256:0eae2073305f451d8ecacb5474997c08569fb4eb4ac231ffa4ad7d342fdc25ac", + "sha256:0fb3e7fc88a14eacd303e90481ad983fd5b69c761e9e6ef94c983f91025da869", + "sha256:11fa2e5984b949b0dd6d7a94d967743d87c577ff0b83392f17cb3990d0d2fd6e", + "sha256:127cee571038f252a552760076407f9cff79761c3d436a12af6000cd182a9d04", + "sha256:154e939c5f0053a383de4fd3d3da48d9427a7e985f58af8e94d0b3c9fcfcf4f9", + "sha256:15587643b9e5eb26c48e49a7b33659790d28f190fc514a322d55da2fb5c2950e", + "sha256:170aeb00224ab3dc54230c797f8404507240dd868cf52066f66a41b33169bdbe", + "sha256:1b5e1b74d1bd1b78bc3477528919414874748dd363e6272efd5abf7654e68bef", + "sha256:1da3b2703afd040cf65ec97efea81cfba59cdbed9c11d8efc5ab09df9509fc56", + "sha256:1e23412b5c41e58cec602f1135c57dfcf15482013ce6e5f093a86db69646a5aa", + "sha256:2247178effb34a77c11c0e8ac355c7a741ceca0a732b27bf11e747bbc950722f", + "sha256:257d8788df5ca62c980314053197f4d46eefedf4e6175bc9412f14412ec4ea2f", + "sha256:3031709084b6e7852d00479fd1d310b07d0ba82765f973b543c8af5061cf990e", + "sha256:322209c642aabdd6207517e9739c704dc9f9db943015535783239022002f054a", + "sha256:322bdf3c9b556e9ffb18f93462e5f749d3444ce081290352c6070d014c93feb2", + "sha256:33870dc4653c5017bf4c8873e5488d8f8d5f8935e2f1fb9a2208c47cdd66efd2", + "sha256:35bb52c37f256f662abdfa49d2dfa6ce5d93281d323a9af377a120e89a9eafb5", + "sha256:3c31822339516fb3c82d03f30e22b1d038da87ef27b6a78c9549888f8ceda39a", + "sha256:3eedd52442c0a5ff4f887fab0c1c0bb164d8635b32c894bc1faf4c618dd89df2", + "sha256:3ff074fc97dd4e80543a3e91f69d58889baf2002b6be64347ea8cf5533188213", + "sha256:47c0995fc4e7f79b5cfcab1fc437ff2890b770440f7696a3ba065ee0fd496563", + "sha256:49d9ba1ed0ef3e061088cd1e7538a0759aab559e2e0a80a36f9fd9d8c0c21591", + "sha256:51f1a1bffc50e2e9492e87d8e09a17c5eea8409cda8d3f277eb6edc82813c17c", + "sha256:52a50aa3fb3acb9cf7213573ef55d31d6eca37f5709c69e6858fe3bc04a5c2a2", + "sha256:54f1852cd531aa981bc0965b7d609f5f6cc8ce8c41b1139f6ed6b3c54ab82bfb", + "sha256:609448742444d9290fd687940ac0b57fb35e6fd92bdb65386e08e99af60bf757", + "sha256:69ffdd6120a4737710a9eee73e1d2e37db89b620f702754b8f6e62594471dee0", + "sha256:6fad5ff2f13d69b7e74ce5b4ecd12cc0ec530fcee76356cac6742785ff71c452", + "sha256:7049e301399273a0136ff39b84c3678e314f2158f50f517bc50285fb5ec847ad", + "sha256:70c61d4c475835a19b3a5aa42492409878bbca7438554a1f89d20d58a7c75c01", + "sha256:716d30ed977be8b37d3ef185fecb9e5a1d62d110dfbdcd1e2a122ab46fddb03f", + "sha256:753cd8f2086b2b80180d9b3010dd4ed147efc167c90d3bf593fe2af21265e5a5", + "sha256:773efe0603db30c281521a7c0214cad7836c03b8ccff897beae9b47c0b657d61", + "sha256:7823bdd049099efa16e4246bdf15e5a13dbb18a51b68fa06d6c1d4d8b99a796e", + "sha256:7c8f97e8e7a9009bcacbe3766a36175056c12f9a44e6e6f2d5caad06dcfbf03b", + "sha256:823ef7a27cf86df6597fa0671066c1b596f69eba53efa3d1e1cb8b30f3533068", + "sha256:8373c6c251f7ef8bda6675dd6d2b3a0fcc31edf1201266b5cf608b62a37407f9", + "sha256:83b2021f2ade7d1ed556bc50a399127d7fb245e725aa0113ebd05cfe88aaf588", + "sha256:870ea1ada0899fd0b79643990809323b389d4d1d46c192f97342eeb6ee0b8483", + "sha256:8d12251f02d69d8310b046e82572ed486685c38f02176bd08baf216746eb947f", + "sha256:9c23f307202661071d94b5e384e1e1dc7dfb972a28a2310e4ee16103e66ddb67", + "sha256:9d189550615b4948f45252d7f005e53c2040cea1af5b60d6f79491a6e147eef7", + "sha256:a086c2af425c5f62a65e12fbf385f7c9fcb8f107d0849dba5839461a129cf311", + "sha256:a2b56ba36e05f973d450582fb015594aaa78834fefe8dfb8fcd79b93e64ba4c6", + "sha256:aebb6044806f2e16ecc07b2a2637ee1ef67a11840a66752751714a0d924adf72", + "sha256:b1b3020d90c2d8e1dae29cf3ce54f8094f7938460fb5ce8bc5c01450b01fbaf6", + "sha256:b4b6b1e20608493548b1f32bce8cca185bf0480983890403d3b8753e44077129", + "sha256:b6f491cdf80ae540738859d9766783e3b3c8e5bd37f5dfa0b76abdecc5081f13", + "sha256:b792a349405fbc0163190fde0dc7b3fef3c9268292586cf5645598b48e63dc67", + "sha256:b7c2286c23cd350b80d2fc9d424fc797575fb16f854b831d16fd47ceec078f2c", + "sha256:babf5acfede515f176833ed6028754cbcd0d206f7f614ea3447d67c33be12516", + "sha256:c365fd1703040de1ec284b176d6af5abe21b427cb3a5ff68e0759e1e313a5e7e", + "sha256:c4225f5220f46b2fde568c74fca27ae9771536c2e29d7c04f4fb62c83275ac4e", + "sha256:c570f24be1e468e3f0ce7ef56a89a60f0e05b30a3669a459e419c6eac2c35364", + "sha256:c6dafac9e0f2b3c78df97e79af707cdc5ef8e88208d686a4847bab8266870023", + "sha256:c8de2789052ed501dd829e9cae8d3dcce7acb4777ea4a479c14521c942d395b1", + "sha256:cb28c753fd5eb3dd859b4ee95de66cc62af91bcff5db5f2571d32a520baf1f04", + "sha256:cb4c38abeef13c61d6916f264d4845fab99d7b711be96c326b84df9e3e0ff62d", + "sha256:d1b35bcd6c5543b9cb547dee3150c93008f8dd0f1fef78fc0cd2b141c5baf58a", + "sha256:d8e6aeb9201e655354b3ad049cb77d19813ad4ece0df1249d3c793de3774f8c7", + "sha256:d8ecd059fdaf60c1963c58ceb8997b32e9dc1b911f5da5307aab614f1ce5c2fb", + "sha256:da2b52b37dad6d9ec64e653637a096905b258d2fc2b984c41ae7d08b938a67e4", + "sha256:e87f0b2c78157e12d7686b27d63c070fd65d994e8ddae6f328e0dcf4a0cd007e", + "sha256:edca80cbfb2b68d7b56930b84a0e45ae1694aeba0541f798e908a49d66b837f1", + "sha256:f379abd2f1e3dddb2b61bc67977a6b5a0a3f7485538bcc6f39ec76163891ee48", + "sha256:fe4c15f6c9285dc54ce6553a3ce908ed37c8f3825b5a51a15c91442bb955b868" ], "markers": "python_version >= '3.8'", - "version": "==10.4.0" + "version": "==10.2.0" }, "platformdirs": { "hashes": [ - "sha256:2d7a1657e36a80ea911db832a8a6ece5ee53d8de21edd5cc5879af6530b1bfee", - "sha256:38b7b51f512eed9e84a22788b4bce1de17c0adb134d6becb09836e37d8654cd3" + "sha256:11c8f37bcca40db96d8144522d925583bdb7a31f7b0e37e3ed4318400a8e2380", + "sha256:906d548203468492d432bcb294d4bc2fff751bf84971fbb2c10918cc206ee420" ], "markers": "python_version >= '3.8'", - "version": "==4.2.2" + "version": "==4.1.0" }, "pluggy": { "hashes": [ - "sha256:2cffa88e94fdc978c4c574f15f9e59b7f4201d439195c3715ca9e2486f1d0cf1", - "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669" + "sha256:cf61ae8f126ac6f7c451172cf30e3e43d3ca77615509771b3a984a0730651e12", + "sha256:d89c696a773f8bd377d18e5ecda92b7a3793cbe66c87060a6fb58c7b6e1061f7" ], "markers": "python_version >= '3.8'", - "version": "==1.5.0" + "version": "==1.3.0" }, "pycodestyle": { "hashes": [ @@ -922,46 +900,45 @@ }, "pydantic": { "hashes": [ - "sha256:0fe8a415cea8f340e7a9af9c54fc71a649b43e8ca3cc732986116b3cb135d303", - "sha256:1289c180abd4bd4555bb927c42ee42abc3aee02b0fb2d1223fb7c6e5bef87dbe", - "sha256:1eb2085c13bce1612da8537b2d90f549c8cbb05c67e8f22854e201bde5d98a47", - "sha256:2031de0967c279df0d8a1c72b4ffc411ecd06bac607a212892757db7462fc494", - "sha256:2a7bac939fa326db1ab741c9d7f44c565a1d1e80908b3797f7f81a4f86bc8d33", - "sha256:2d5a58feb9a39f481eda4d5ca220aa8b9d4f21a41274760b9bc66bfd72595b86", - "sha256:2f9a6fab5f82ada41d56b0602606a5506aab165ca54e52bc4545028382ef1c5d", - "sha256:2fcfb5296d7877af406ba1547dfde9943b1256d8928732267e2653c26938cd9c", - "sha256:549a8e3d81df0a85226963611950b12d2d334f214436a19537b2efed61b7639a", - "sha256:598da88dfa127b666852bef6d0d796573a8cf5009ffd62104094a4fe39599565", - "sha256:5d1197e462e0364906cbc19681605cb7c036f2475c899b6f296104ad42b9f5fb", - "sha256:69328e15cfda2c392da4e713443c7dbffa1505bc9d566e71e55abe14c97ddc62", - "sha256:6a9dfa722316f4acf4460afdf5d41d5246a80e249c7ff475c43a3a1e9d75cf62", - "sha256:6b30bcb8cbfccfcf02acb8f1a261143fab622831d9c0989707e0e659f77a18e0", - "sha256:6c076be61cd0177a8433c0adcb03475baf4ee91edf5a4e550161ad57fc90f523", - "sha256:771735dc43cf8383959dc9b90aa281f0b6092321ca98677c5fb6125a6f56d58d", - "sha256:795e34e6cc065f8f498c89b894a3c6da294a936ee71e644e4bd44de048af1405", - "sha256:87afda5539d5140cb8ba9e8b8c8865cb5b1463924d38490d73d3ccfd80896b3f", - "sha256:8fb2aa3ab3728d950bcc885a2e9eff6c8fc40bc0b7bb434e555c215491bcf48b", - "sha256:a1fcb59f2f355ec350073af41d927bf83a63b50e640f4dbaa01053a28b7a7718", - "sha256:a5e7add47a5b5a40c49b3036d464e3c7802f8ae0d1e66035ea16aa5b7a3923ed", - "sha256:a73f489aebd0c2121ed974054cb2759af8a9f747de120acd2c3394cf84176ccb", - "sha256:ab26038b8375581dc832a63c948f261ae0aa21f1d34c1293469f135fa92972a5", - "sha256:b0d191db0f92dfcb1dec210ca244fdae5cbe918c6050b342d619c09d31eea0cc", - "sha256:b749a43aa51e32839c9d71dc67eb1e4221bb04af1033a32e3923d46f9effa942", - "sha256:b7ccf02d7eb340b216ec33e53a3a629856afe1c6e0ef91d84a4e6f2fb2ca70fe", - "sha256:ba5b2e6fe6ca2b7e013398bc7d7b170e21cce322d266ffcd57cca313e54fb246", - "sha256:ba5c4a8552bff16c61882db58544116d021d0b31ee7c66958d14cf386a5b5350", - "sha256:c79e6a11a07da7374f46970410b41d5e266f7f38f6a17a9c4823db80dadf4303", - "sha256:ca48477862372ac3770969b9d75f1bf66131d386dba79506c46d75e6b48c1e09", - "sha256:dea7adcc33d5d105896401a1f37d56b47d443a2b2605ff8a969a0ed5543f7e33", - "sha256:e0a16d274b588767602b7646fa05af2782576a6cf1022f4ba74cbb4db66f6ca8", - "sha256:e4129b528c6baa99a429f97ce733fff478ec955513630e61b49804b6cf9b224a", - "sha256:e5f805d2d5d0a41633651a73fa4ecdd0b3d7a49de4ec3fadf062fe16501ddbf1", - "sha256:ef6c96b2baa2100ec91a4b428f80d8f28a3c9e53568219b6c298c1125572ebc6", - "sha256:fdbdd1d630195689f325c9ef1a12900524dceb503b00a987663ff4f58669b93d" + "sha256:1061c6ee6204f4f5a27133126854948e3b3d51fcc16ead2e5d04378c199b2f44", + "sha256:19b5686387ea0d1ea52ecc4cffb71abb21702c5e5b2ac626fd4dbaa0834aa49d", + "sha256:2bd446bdb7755c3a94e56d7bdfd3ee92396070efa8ef3a34fab9579fe6aa1d84", + "sha256:328558c9f2eed77bd8fffad3cef39dbbe3edc7044517f4625a769d45d4cf7555", + "sha256:32e0b4fb13ad4db4058a7c3c80e2569adbd810c25e6ca3bbd8b2a9cc2cc871d7", + "sha256:3ee0d69b2a5b341fc7927e92cae7ddcfd95e624dfc4870b32a85568bd65e6131", + "sha256:4aafd4e55e8ad5bd1b19572ea2df546ccace7945853832bb99422a79c70ce9b8", + "sha256:4b3946f87e5cef3ba2e7bd3a4eb5a20385fe36521d6cc1ebf3c08a6697c6cfb3", + "sha256:4de71c718c9756d679420c69f216776c2e977459f77e8f679a4a961dc7304a56", + "sha256:5565a49effe38d51882cb7bac18bda013cdb34d80ac336428e8908f0b72499b0", + "sha256:5803ad846cdd1ed0d97eb00292b870c29c1f03732a010e66908ff48a762f20e4", + "sha256:5da164119602212a3fe7e3bc08911a89db4710ae51444b4224c2382fd09ad453", + "sha256:615661bfc37e82ac677543704437ff737418e4ea04bef9cf11c6d27346606044", + "sha256:78a4d6bdfd116a559aeec9a4cfe77dda62acc6233f8b56a716edad2651023e5e", + "sha256:7d0f183b305629765910eaad707800d2f47c6ac5bcfb8c6397abdc30b69eeb15", + "sha256:7ead3cd020d526f75b4188e0a8d71c0dbbe1b4b6b5dc0ea775a93aca16256aeb", + "sha256:84d76ecc908d917f4684b354a39fd885d69dd0491be175f3465fe4b59811c001", + "sha256:8cb0bc509bfb71305d7a59d00163d5f9fc4530f0881ea32c74ff4f74c85f3d3d", + "sha256:91089b2e281713f3893cd01d8e576771cd5bfdfbff5d0ed95969f47ef6d676c3", + "sha256:9c9e04a6cdb7a363d7cb3ccf0efea51e0abb48e180c0d31dca8d247967d85c6e", + "sha256:a8c5360a0297a713b4123608a7909e6869e1b56d0e96eb0d792c27585d40757f", + "sha256:afacf6d2a41ed91fc631bade88b1d319c51ab5418870802cedb590b709c5ae3c", + "sha256:b34ba24f3e2d0b39b43f0ca62008f7ba962cff51efa56e64ee25c4af6eed987b", + "sha256:bd67cb2c2d9602ad159389c29e4ca964b86fa2f35c2faef54c3eb28b4efd36c8", + "sha256:c0f5e142ef8217019e3eef6ae1b6b55f09a7a15972958d44fbd228214cede567", + "sha256:cdb4272678db803ddf94caa4f94f8672e9a46bae4a44f167095e4d06fec12979", + "sha256:d70916235d478404a3fa8c997b003b5f33aeac4686ac1baa767234a0f8ac2326", + "sha256:d8ce3fb0841763a89322ea0432f1f59a2d3feae07a63ea2c958b2315e1ae8adb", + "sha256:e0b214e57623a535936005797567231a12d0da0c29711eb3514bc2b3cd008d0f", + "sha256:e631c70c9280e3129f071635b81207cad85e6c08e253539467e4ead0e5b219aa", + "sha256:e78578f0c7481c850d1c969aca9a65405887003484d24f6110458fb02cca7747", + "sha256:f0ca86b525264daa5f6b192f216a0d1e860b7383e3da1c65a1908f9c02f42801", + "sha256:f1a68f4f65a9ee64b6ccccb5bf7e17db07caebd2730109cb8a95863cfa9c4e55", + "sha256:fafe841be1103f340a24977f61dee76172e4ae5f647ab9e7fd1e1fca51524f08", + "sha256:ff68fc85355532ea77559ede81f35fff79a6a5543477e168ab3a381887caea76" ], "index": "pypi", - "markers": "python_version >= '3.7'", - "version": "==1.10.12" + "markers": "python_full_version >= '3.6.1'", + "version": "==1.9.2" }, "pydocstyle": { "hashes": [ @@ -981,11 +958,49 @@ }, "pyparsing": { "hashes": [ - "sha256:a1bac0ce561155ecc3ed78ca94d3c9378656ad4c94c1270de543f621420f94ad", - "sha256:f9db75911801ed778fe61bb643079ff86601aca99fcae6345aa67292038fb742" + "sha256:32c7c0b711493c72ff18a981d24f28aaf9c1fb7ed5e9667c9e84e3db623bdbfb", + "sha256:ede28a1a32462f5a9705e07aea48001a08f7cf81a021585011deba701581a0db" ], "markers": "python_full_version >= '3.6.8'", - "version": "==3.1.2" + "version": "==3.1.1" + }, + "pyrsistent": { + "hashes": [ + "sha256:0724c506cd8b63c69c7f883cc233aac948c1ea946ea95996ad8b1380c25e1d3f", + "sha256:09848306523a3aba463c4b49493a760e7a6ca52e4826aa100ee99d8d39b7ad1e", + "sha256:0f3b1bcaa1f0629c978b355a7c37acd58907390149b7311b5db1b37648eb6958", + "sha256:21cc459636983764e692b9eba7144cdd54fdec23ccdb1e8ba392a63666c60c34", + "sha256:2e14c95c16211d166f59c6611533d0dacce2e25de0f76e4c140fde250997b3ca", + "sha256:2e2c116cc804d9b09ce9814d17df5edf1df0c624aba3b43bc1ad90411487036d", + "sha256:4021a7f963d88ccd15b523787d18ed5e5269ce57aa4037146a2377ff607ae87d", + "sha256:4c48f78f62ab596c679086084d0dd13254ae4f3d6c72a83ffdf5ebdef8f265a4", + "sha256:4f5c2d012671b7391803263419e31b5c7c21e7c95c8760d7fc35602353dee714", + "sha256:58b8f6366e152092194ae68fefe18b9f0b4f89227dfd86a07770c3d86097aebf", + "sha256:59a89bccd615551391f3237e00006a26bcf98a4d18623a19909a2c48b8e986ee", + "sha256:5cdd7ef1ea7a491ae70d826b6cc64868de09a1d5ff9ef8d574250d0940e275b8", + "sha256:6288b3fa6622ad8a91e6eb759cfc48ff3089e7c17fb1d4c59a919769314af224", + "sha256:6d270ec9dd33cdb13f4d62c95c1a5a50e6b7cdd86302b494217137f760495b9d", + "sha256:79ed12ba79935adaac1664fd7e0e585a22caa539dfc9b7c7c6d5ebf91fb89054", + "sha256:7d29c23bdf6e5438c755b941cef867ec2a4a172ceb9f50553b6ed70d50dfd656", + "sha256:8441cf9616d642c475684d6cf2520dd24812e996ba9af15e606df5f6fd9d04a7", + "sha256:881bbea27bbd32d37eb24dd320a5e745a2a5b092a17f6debc1349252fac85423", + "sha256:8c3aba3e01235221e5b229a6c05f585f344734bd1ad42a8ac51493d74722bbce", + "sha256:a14798c3005ec892bbada26485c2eea3b54109cb2533713e355c806891f63c5e", + "sha256:b14decb628fac50db5e02ee5a35a9c0772d20277824cfe845c8a8b717c15daa3", + "sha256:b318ca24db0f0518630e8b6f3831e9cba78f099ed5c1d65ffe3e023003043ba0", + "sha256:c1beb78af5423b879edaf23c5591ff292cf7c33979734c99aa66d5914ead880f", + "sha256:c55acc4733aad6560a7f5f818466631f07efc001fd023f34a6c203f8b6df0f0b", + "sha256:ca52d1ceae015859d16aded12584c59eb3825f7b50c6cfd621d4231a6cc624ce", + "sha256:cae40a9e3ce178415040a0383f00e8d68b569e97f31928a3a8ad37e3fde6df6a", + "sha256:e78d0c7c1e99a4a45c99143900ea0546025e41bb59ebc10182e947cf1ece9174", + "sha256:ef3992833fbd686ee783590639f4b8343a57f1f75de8633749d984dc0eb16c86", + "sha256:f058a615031eea4ef94ead6456f5ec2026c19fb5bd6bfe86e9665c4158cf802f", + "sha256:f5ac696f02b3fc01a710427585c855f65cd9c640e14f52abe52020722bb4906b", + "sha256:f920385a11207dc372a028b3f1e1038bb244b3ec38d448e6d8e43c6b3ba20e98", + "sha256:fed2c3216a605dc9a6ea50c7e84c82906e3684c4e80d2908208f662a6cbf9022" + ], + "markers": "python_version >= '3.8'", + "version": "==0.20.0" }, "pytest": { "hashes": [ @@ -998,12 +1013,12 @@ }, "pytest-asyncio": { "hashes": [ - "sha256:50265d892689a5faefb84df80819d1ecef566eb3549cf915dfb33569359d1ce2", - "sha256:759b10b33a6dc61cce40a8bd5205e302978bbbcc00e279a8b61d9a6a3c82e4d3" + "sha256:37a9d912e8338ee7b4a3e917381d1c95bfc8682048cb0fbc35baba316ec1faba", + "sha256:af313ce900a62fbe2b1aed18e37ad757f1ef9940c6b6a88e2954de38d6b1fb9f" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==0.23.8" + "version": "==0.23.3" }, "pytest-cov": { "hashes": [ @@ -1024,124 +1039,19 @@ }, "python-dateutil": { "hashes": [ - "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3", - "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427" + "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86", + "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==2.9.0.post0" + "version": "==2.8.2" }, - "referencing": { - "hashes": [ - "sha256:25b42124a6c8b632a425174f24087783efb348a6f1e0008e63cd4466fedf703c", - "sha256:eda6d3234d62814d1c64e305c1331c9a3a6132da475ab6382eaa997b21ee75de" - ], - "markers": "python_version >= '3.8'", - "version": "==0.35.1" - }, - "rpds-py": { + "setuptools": { "hashes": [ - "sha256:0121803b0f424ee2109d6e1f27db45b166ebaa4b32ff47d6aa225642636cd834", - "sha256:06925c50f86da0596b9c3c64c3837b2481337b83ef3519e5db2701df695453a4", - "sha256:071d4adc734de562bd11d43bd134330fb6249769b2f66b9310dab7460f4bf714", - "sha256:1540d807364c84516417115c38f0119dfec5ea5c0dd9a25332dea60b1d26fc4d", - "sha256:15e65395a59d2e0e96caf8ee5389ffb4604e980479c32742936ddd7ade914b22", - "sha256:19d02c45f2507b489fd4df7b827940f1420480b3e2e471e952af4d44a1ea8e34", - "sha256:1c26da90b8d06227d7769f34915913911222d24ce08c0ab2d60b354e2d9c7aff", - "sha256:1d16089dfa58719c98a1c06f2daceba6d8e3fb9b5d7931af4a990a3c486241cb", - "sha256:1dd46f309e953927dd018567d6a9e2fb84783963650171f6c5fe7e5c41fd5666", - "sha256:2575efaa5d949c9f4e2cdbe7d805d02122c16065bfb8d95c129372d65a291a0b", - "sha256:3208f9aea18991ac7f2b39721e947bbd752a1abbe79ad90d9b6a84a74d44409b", - "sha256:329c719d31362355a96b435f4653e3b4b061fcc9eba9f91dd40804ca637d914e", - "sha256:3384d278df99ec2c6acf701d067147320b864ef6727405d6470838476e44d9e8", - "sha256:34a01a4490e170376cd79258b7f755fa13b1a6c3667e872c8e35051ae857a92b", - "sha256:354f3a91718489912f2e0fc331c24eaaf6a4565c080e00fbedb6015857c00582", - "sha256:37f46bb11858717e0efa7893c0f7055c43b44c103e40e69442db5061cb26ed34", - "sha256:3b4cf5a9497874822341c2ebe0d5850fed392034caadc0bad134ab6822c0925b", - "sha256:3f148c3f47f7f29a79c38cc5d020edcb5ca780020fab94dbc21f9af95c463581", - "sha256:443cec402ddd650bb2b885113e1dcedb22b1175c6be223b14246a714b61cd521", - "sha256:462b0c18fbb48fdbf980914a02ee38c423a25fcc4cf40f66bacc95a2d2d73bc8", - "sha256:474bc83233abdcf2124ed3f66230a1c8435896046caa4b0b5ab6013c640803cc", - "sha256:4d438e4c020d8c39961deaf58f6913b1bf8832d9b6f62ec35bd93e97807e9cbc", - "sha256:4fdc9afadbeb393b4bbbad75481e0ea78e4469f2e1d713a90811700830b553a9", - "sha256:5039e3cef7b3e7a060de468a4a60a60a1f31786da94c6cb054e7a3c75906111c", - "sha256:5095a7c838a8647c32aa37c3a460d2c48debff7fc26e1136aee60100a8cd8f68", - "sha256:52e466bea6f8f3a44b1234570244b1cff45150f59a4acae3fcc5fd700c2993ca", - "sha256:535d4b52524a961d220875688159277f0e9eeeda0ac45e766092bfb54437543f", - "sha256:57dbc9167d48e355e2569346b5aa4077f29bf86389c924df25c0a8b9124461fb", - "sha256:5a4b07cdf3f84310c08c1de2c12ddadbb7a77568bcb16e95489f9c81074322ed", - "sha256:5c872814b77a4e84afa293a1bee08c14daed1068b2bb1cc312edbf020bbbca2b", - "sha256:5f83689a38e76969327e9b682be5521d87a0c9e5a2e187d2bc6be4765f0d4600", - "sha256:688aa6b8aa724db1596514751ffb767766e02e5c4a87486ab36b8e1ebc1aedac", - "sha256:6b130bd4163c93798a6b9bb96be64a7c43e1cec81126ffa7ffaa106e1fc5cef5", - "sha256:6b31f059878eb1f5da8b2fd82480cc18bed8dcd7fb8fe68370e2e6285fa86da6", - "sha256:6d45080095e585f8c5097897313def60caa2046da202cdb17a01f147fb263b81", - "sha256:6f2f78ef14077e08856e788fa482107aa602636c16c25bdf59c22ea525a785e9", - "sha256:6fe87efd7f47266dfc42fe76dae89060038f1d9cb911f89ae7e5084148d1cc08", - "sha256:75969cf900d7be665ccb1622a9aba225cf386bbc9c3bcfeeab9f62b5048f4a07", - "sha256:75a6076289b2df6c8ecb9d13ff79ae0cad1d5fb40af377a5021016d58cd691ec", - "sha256:78d57546bad81e0da13263e4c9ce30e96dcbe720dbff5ada08d2600a3502e526", - "sha256:79e205c70afddd41f6ee79a8656aec738492a550247a7af697d5bd1aee14f766", - "sha256:7c98298a15d6b90c8f6e3caa6457f4f022423caa5fa1a1ca7a5e9e512bdb77a4", - "sha256:7ec72df7354e6b7f6eb2a17fa6901350018c3a9ad78e48d7b2b54d0412539a67", - "sha256:81ea573aa46d3b6b3d890cd3c0ad82105985e6058a4baed03cf92518081eec8c", - "sha256:8344127403dea42f5970adccf6c5957a71a47f522171fafaf4c6ddb41b61703a", - "sha256:8445f23f13339da640d1be8e44e5baf4af97e396882ebbf1692aecd67f67c479", - "sha256:850720e1b383df199b8433a20e02b25b72f0fded28bc03c5bd79e2ce7ef050be", - "sha256:88cb4bac7185a9f0168d38c01d7a00addece9822a52870eee26b8d5b61409213", - "sha256:8a790d235b9d39c70a466200d506bb33a98e2ee374a9b4eec7a8ac64c2c261fa", - "sha256:8b1a94b8afc154fbe36978a511a1f155f9bd97664e4f1f7a374d72e180ceb0ae", - "sha256:8d6ad132b1bc13d05ffe5b85e7a01a3998bf3a6302ba594b28d61b8c2cf13aaf", - "sha256:8eb488ef928cdbc05a27245e52de73c0d7c72a34240ef4d9893fdf65a8c1a955", - "sha256:90bf55d9d139e5d127193170f38c584ed3c79e16638890d2e36f23aa1630b952", - "sha256:9133d75dc119a61d1a0ded38fb9ba40a00ef41697cc07adb6ae098c875195a3f", - "sha256:93a91c2640645303e874eada51f4f33351b84b351a689d470f8108d0e0694210", - "sha256:959179efb3e4a27610e8d54d667c02a9feaa86bbabaf63efa7faa4dfa780d4f1", - "sha256:9625367c8955e4319049113ea4f8fee0c6c1145192d57946c6ffcd8fe8bf48dd", - "sha256:9da6f400eeb8c36f72ef6646ea530d6d175a4f77ff2ed8dfd6352842274c1d8b", - "sha256:9e65489222b410f79711dc3d2d5003d2757e30874096b2008d50329ea4d0f88c", - "sha256:a3e2fd14c5d49ee1da322672375963f19f32b3d5953f0615b175ff7b9d38daed", - "sha256:a5a7c1062ef8aea3eda149f08120f10795835fc1c8bc6ad948fb9652a113ca55", - "sha256:a5da93debdfe27b2bfc69eefb592e1831d957b9535e0943a0ee8b97996de21b5", - "sha256:a6e605bb9edcf010f54f8b6a590dd23a4b40a8cb141255eec2a03db249bc915b", - "sha256:a707b158b4410aefb6b054715545bbb21aaa5d5d0080217290131c49c2124a6e", - "sha256:a8b6683a37338818646af718c9ca2a07f89787551057fae57c4ec0446dc6224b", - "sha256:aa5476c3e3a402c37779e95f7b4048db2cb5b0ed0b9d006983965e93f40fe05a", - "sha256:ab1932ca6cb8c7499a4d87cb21ccc0d3326f172cfb6a64021a889b591bb3045c", - "sha256:ae8b6068ee374fdfab63689be0963333aa83b0815ead5d8648389a8ded593378", - "sha256:b0906357f90784a66e89ae3eadc2654f36c580a7d65cf63e6a616e4aec3a81be", - "sha256:b0da31853ab6e58a11db3205729133ce0df26e6804e93079dee095be3d681dc1", - "sha256:b1c30841f5040de47a0046c243fc1b44ddc87d1b12435a43b8edff7e7cb1e0d0", - "sha256:b228e693a2559888790936e20f5f88b6e9f8162c681830eda303bad7517b4d5a", - "sha256:b7cc6cb44f8636fbf4a934ca72f3e786ba3c9f9ba4f4d74611e7da80684e48d2", - "sha256:ba0ed0dc6763d8bd6e5de5cf0d746d28e706a10b615ea382ac0ab17bb7388633", - "sha256:bc9128e74fe94650367fe23f37074f121b9f796cabbd2f928f13e9661837296d", - "sha256:bcf426a8c38eb57f7bf28932e68425ba86def6e756a5b8cb4731d8e62e4e0223", - "sha256:bec35eb20792ea64c3c57891bc3ca0bedb2884fbac2c8249d9b731447ecde4fa", - "sha256:c3444fe52b82f122d8a99bf66777aed6b858d392b12f4c317da19f8234db4533", - "sha256:c5c9581019c96f865483d031691a5ff1cc455feb4d84fc6920a5ffc48a794d8a", - "sha256:c6feacd1d178c30e5bc37184526e56740342fd2aa6371a28367bad7908d454fc", - "sha256:c8f77e661ffd96ff104bebf7d0f3255b02aa5d5b28326f5408d6284c4a8b3248", - "sha256:cb0f6eb3a320f24b94d177e62f4074ff438f2ad9d27e75a46221904ef21a7b05", - "sha256:ce84a7efa5af9f54c0aa7692c45861c1667080814286cacb9958c07fc50294fb", - "sha256:cf902878b4af334a09de7a45badbff0389e7cf8dc2e4dcf5f07125d0b7c2656d", - "sha256:dab8d921b55a28287733263c0e4c7db11b3ee22aee158a4de09f13c93283c62d", - "sha256:dc9ac4659456bde7c567107556ab065801622396b435a3ff213daef27b495388", - "sha256:dd36b712d35e757e28bf2f40a71e8f8a2d43c8b026d881aa0c617b450d6865c9", - "sha256:e19509145275d46bc4d1e16af0b57a12d227c8253655a46bbd5ec317e941279d", - "sha256:e21cc693045fda7f745c790cb687958161ce172ffe3c5719ca1764e752237d16", - "sha256:e54548e0be3ac117595408fd4ca0ac9278fde89829b0b518be92863b17ff67a2", - "sha256:e5b9fc03bf76a94065299d4a2ecd8dfbae4ae8e2e8098bbfa6ab6413ca267709", - "sha256:e8481b946792415adc07410420d6fc65a352b45d347b78fec45d8f8f0d7496f0", - "sha256:ebcbf356bf5c51afc3290e491d3722b26aaf5b6af3c1c7f6a1b757828a46e336", - "sha256:ef9101f3f7b59043a34f1dccbb385ca760467590951952d6701df0da9893ca0c", - "sha256:f2afd2164a1e85226fcb6a1da77a5c8896c18bfe08e82e8ceced5181c42d2179", - "sha256:f629ecc2db6a4736b5ba95a8347b0089240d69ad14ac364f557d52ad68cf94b0", - "sha256:f68eea5df6347d3f1378ce992d86b2af16ad7ff4dcb4a19ccdc23dea901b87fb", - "sha256:f757f359f30ec7dcebca662a6bd46d1098f8b9fb1fcd661a9e13f2e8ce343ba1", - "sha256:fb37bd599f031f1a6fb9e58ec62864ccf3ad549cf14bac527dbfa97123edcca4" + "sha256:385eb4edd9c9d5c17540511303e39a147ce2fc04bc55289c322b9e5904fe2c05", + "sha256:be1af57fc409f93647f2e8e4573a142ed38724b8cdd389706a867bb4efcf1e78" ], "markers": "python_version >= '3.8'", - "version": "==0.19.0" + "version": "==69.0.3" }, "six": { "hashes": [ @@ -1175,21 +1085,21 @@ }, "types-mock": { "hashes": [ - "sha256:5281a645d72e827d70043e3cc144fe33b1c003db084f789dc203aa90e812a5a4", - "sha256:d586a01d39ad919d3ddcd73de6cde73ca7f3c69707219f722d1b8d7733641ad7" + "sha256:13ca379d5710ccb3f18f69ade5b08881874cb83383d8fb49b1d4dac9d5c5d090", + "sha256:3d116955495935b0bcba14954b38d97e507cd43eca3e3700fc1b8e4f5c6bf2c7" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==5.1.0.20240425" + "version": "==5.1.0.20240106" }, "typing-extensions": { "hashes": [ - "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d", - "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8" + "sha256:23478f88c37f27d76ac8aee6c905017a143b0b1b886c3c9f66bc2fd94f9f5783", + "sha256:af72aea155e91adfc61c3ae9e0e342dbc0cba726d6cba4b6c72c1f34e47291cd" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==4.12.2" + "version": "==4.9.0" } } } diff --git a/hardware/opentrons_hardware/drivers/can_bus/settings.py b/hardware/opentrons_hardware/drivers/can_bus/settings.py index 9ebb990d9c9..c36bd791fb9 100644 --- a/hardware/opentrons_hardware/drivers/can_bus/settings.py +++ b/hardware/opentrons_hardware/drivers/can_bus/settings.py @@ -50,21 +50,21 @@ class DriverSettings(BaseSettings): """Settings for driver building.""" interface: str = Field( - default=DEFAULT_INTERFACE, + DEFAULT_INTERFACE, description=f"Can either be {OPENTRONS_INTERFACE} for simple socket " f"or a python can interface.", ) bit_rate: int = Field( - default=DEFAULT_BITRATE, + DEFAULT_BITRATE, description=f"Bit rate. Not applicable to {OPENTRONS_INTERFACE} interface.", ) - channel: str = Field(default=DEFAULT_CHANNEL, description="The SocketCan channel.") + channel: str = Field(DEFAULT_CHANNEL, description="The SocketCan channel.") - host: str = Field(default=DEFAULT_HOST, description=f"{OPENTRONS_INTERFACE} only.") - port: int = Field(default=DEFAULT_PORT, description=f"{OPENTRONS_INTERFACE} only.") - fcan_clock: int = Field(default=DEFAULT_FDCAN_CLK, description="pcan only.") - sample_rate: float = Field(default=DEFAULT_SAMPLE_RATE, description="pcan only.") - jump_width: int = Field(default=DEFAULT_JUMP_WIDTH_SEG, description="pcan only.") + host: str = Field(DEFAULT_HOST, description=f"{OPENTRONS_INTERFACE} only.") + port: int = Field(DEFAULT_PORT, description=f"{OPENTRONS_INTERFACE} only.") + fcan_clock: int = Field(DEFAULT_FDCAN_CLK, description="pcan only.") + sample_rate: float = Field(DEFAULT_SAMPLE_RATE, description="pcan only.") + jump_width: int = Field(DEFAULT_JUMP_WIDTH_SEG, descript="pcan only.") class Config: # noqa: D106 env_prefix = "OT3_CAN_DRIVER_" diff --git a/performance-metrics/Pipfile b/performance-metrics/Pipfile index d3104f8c27c..b57cd339814 100644 --- a/performance-metrics/Pipfile +++ b/performance-metrics/Pipfile @@ -6,9 +6,7 @@ name = "pypi" [packages] performance-metrics = {file = ".", editable = true} psutil = "==6.0.0" -# systemd-python errors upon installation if the host machine does not have systemd. -# Mark it as Linux-only so we can have dev environments on such machines. -systemd-python = { version = "==234", markers="sys_platform=='linux'" } +systemd-python = "234" [dev-packages] pytest = "==7.4.4" diff --git a/performance-metrics/Pipfile.lock b/performance-metrics/Pipfile.lock index 3680792e6bf..9bb77d23330 100644 --- a/performance-metrics/Pipfile.lock +++ b/performance-metrics/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "259baa6d557d8069c945cca566e64d0135576284244f14461c67e044366535b5" + "sha256": "a123325c3bebd1451774ebfadc532a4ca78a92897b80cf1d20ded030e145bac2" }, "pipfile-spec": 6, "requires": { @@ -49,7 +49,6 @@ "sha256:fd0e44bf70eadae45aadc292cb0a7eb5b0b6372cd1b391228047d33895db83e7" ], "index": "pypi", - "markers": "sys_platform == 'linux'", "version": "==234" } }, @@ -102,11 +101,11 @@ }, "exceptiongroup": { "hashes": [ - "sha256:3111b9d131c238bec2f8f516e123e14ba243563fb135d3fe885990585aa7795b", - "sha256:47c2edf7c6738fafb49fd34290706d1a1a2f4d1c6df275526b62cbb4aa5393cc" + "sha256:5258b9ed329c5bbdd31a309f53cbfb0b155341807f6ff7606a1e801a891b29ad", + "sha256:a4785e48b045528f5bfe627b6ad554ff32def154f42372786903b7abcfe1aa16" ], "markers": "python_version < '3.11'", - "version": "==1.2.2" + "version": "==1.2.1" }, "flake8": { "hashes": [ @@ -269,12 +268,12 @@ }, "pytest-asyncio": { "hashes": [ - "sha256:50265d892689a5faefb84df80819d1ecef566eb3549cf915dfb33569359d1ce2", - "sha256:759b10b33a6dc61cce40a8bd5205e302978bbbcc00e279a8b61d9a6a3c82e4d3" + "sha256:009b48127fbe44518a547bddd25611551b0e43ccdbf1e67d12479f569832c20b", + "sha256:5f5c72948f4c49e7db4f29f2521d4031f1c27f86e57b046126654083d4770268" ], "index": "pypi", "markers": "python_version >= '3.8'", - "version": "==0.23.8" + "version": "==0.23.7" }, "snowballstemmer": { "hashes": [ diff --git a/shared-data/python/Pipfile b/shared-data/python/Pipfile index 70cc1cc48c7..0d11a1d68c9 100644 --- a/shared-data/python/Pipfile +++ b/shared-data/python/Pipfile @@ -26,5 +26,5 @@ pytest-clarity = "~=1.0.0" [packages] opentrons-shared-data = { editable = true, path = "." } -jsonschema = "==4.17.3" +jsonschema = "==4.21.1" pydantic = "==1.10.12" diff --git a/shared-data/python/Pipfile.lock b/shared-data/python/Pipfile.lock index 957723991c1..a125943127f 100644 --- a/shared-data/python/Pipfile.lock +++ b/shared-data/python/Pipfile.lock @@ -1,7 +1,7 @@ { "_meta": { "hash": { - "sha256": "861a8ab74e5057cb08962b9e5753828739acf1aaf11f9c883d9042281f201880" + "sha256": "9b5174a247c5fe717a5db26f523afd532a6b0fc27943d86f6588839785ef51f4" }, "pipfile-spec": 6, "requires": {}, @@ -24,16 +24,24 @@ }, "jsonschema": { "hashes": [ - "sha256:0f864437ab8b6076ba6707453ef8f98a6a0d512a80e93f8abdb676f737ecb60d", - "sha256:a870ad254da1a8ca84b6a2905cac29d265f805acc57af304784962a2aa6508f6" + "sha256:7996507afae316306f9e2290407761157c6f78002dcf7419acb99822143d1c6f", + "sha256:85727c00279f5fa6bedbe6238d2aa6403bedd8b4864ab11207d07df3cc1b2ee5" ], "index": "pypi", - "markers": "python_version >= '3.7'", - "version": "==4.17.3" + "markers": "python_version >= '3.8'", + "version": "==4.21.1" + }, + "jsonschema-specifications": { + "hashes": [ + "sha256:48a76787b3e70f5ed53f1160d2b81f586e4ca6d1548c5de7085d1682674764cc", + "sha256:87e4fdf3a94858b8a2ba2778d9ba57d8a9cafca7c7489c46ba0d30a8bc6a9c3c" + ], + "markers": "python_version >= '3.8'", + "version": "==2023.12.1" }, "opentrons-shared-data": { "editable": true, - "markers": "python_version >= '3.10'", + "markers": "python_version >= '3.7'", "path": "." }, "pydantic": { @@ -79,51 +87,126 @@ "markers": "python_version >= '3.7'", "version": "==1.10.12" }, - "pyrsistent": { - "hashes": [ - "sha256:0724c506cd8b63c69c7f883cc233aac948c1ea946ea95996ad8b1380c25e1d3f", - "sha256:09848306523a3aba463c4b49493a760e7a6ca52e4826aa100ee99d8d39b7ad1e", - "sha256:0f3b1bcaa1f0629c978b355a7c37acd58907390149b7311b5db1b37648eb6958", - "sha256:21cc459636983764e692b9eba7144cdd54fdec23ccdb1e8ba392a63666c60c34", - "sha256:2e14c95c16211d166f59c6611533d0dacce2e25de0f76e4c140fde250997b3ca", - "sha256:2e2c116cc804d9b09ce9814d17df5edf1df0c624aba3b43bc1ad90411487036d", - "sha256:4021a7f963d88ccd15b523787d18ed5e5269ce57aa4037146a2377ff607ae87d", - "sha256:4c48f78f62ab596c679086084d0dd13254ae4f3d6c72a83ffdf5ebdef8f265a4", - "sha256:4f5c2d012671b7391803263419e31b5c7c21e7c95c8760d7fc35602353dee714", - "sha256:58b8f6366e152092194ae68fefe18b9f0b4f89227dfd86a07770c3d86097aebf", - "sha256:59a89bccd615551391f3237e00006a26bcf98a4d18623a19909a2c48b8e986ee", - "sha256:5cdd7ef1ea7a491ae70d826b6cc64868de09a1d5ff9ef8d574250d0940e275b8", - "sha256:6288b3fa6622ad8a91e6eb759cfc48ff3089e7c17fb1d4c59a919769314af224", - "sha256:6d270ec9dd33cdb13f4d62c95c1a5a50e6b7cdd86302b494217137f760495b9d", - "sha256:79ed12ba79935adaac1664fd7e0e585a22caa539dfc9b7c7c6d5ebf91fb89054", - "sha256:7d29c23bdf6e5438c755b941cef867ec2a4a172ceb9f50553b6ed70d50dfd656", - "sha256:8441cf9616d642c475684d6cf2520dd24812e996ba9af15e606df5f6fd9d04a7", - "sha256:881bbea27bbd32d37eb24dd320a5e745a2a5b092a17f6debc1349252fac85423", - "sha256:8c3aba3e01235221e5b229a6c05f585f344734bd1ad42a8ac51493d74722bbce", - "sha256:a14798c3005ec892bbada26485c2eea3b54109cb2533713e355c806891f63c5e", - "sha256:b14decb628fac50db5e02ee5a35a9c0772d20277824cfe845c8a8b717c15daa3", - "sha256:b318ca24db0f0518630e8b6f3831e9cba78f099ed5c1d65ffe3e023003043ba0", - "sha256:c1beb78af5423b879edaf23c5591ff292cf7c33979734c99aa66d5914ead880f", - "sha256:c55acc4733aad6560a7f5f818466631f07efc001fd023f34a6c203f8b6df0f0b", - "sha256:ca52d1ceae015859d16aded12584c59eb3825f7b50c6cfd621d4231a6cc624ce", - "sha256:cae40a9e3ce178415040a0383f00e8d68b569e97f31928a3a8ad37e3fde6df6a", - "sha256:e78d0c7c1e99a4a45c99143900ea0546025e41bb59ebc10182e947cf1ece9174", - "sha256:ef3992833fbd686ee783590639f4b8343a57f1f75de8633749d984dc0eb16c86", - "sha256:f058a615031eea4ef94ead6456f5ec2026c19fb5bd6bfe86e9665c4158cf802f", - "sha256:f5ac696f02b3fc01a710427585c855f65cd9c640e14f52abe52020722bb4906b", - "sha256:f920385a11207dc372a028b3f1e1038bb244b3ec38d448e6d8e43c6b3ba20e98", - "sha256:fed2c3216a605dc9a6ea50c7e84c82906e3684c4e80d2908208f662a6cbf9022" + "referencing": { + "hashes": [ + "sha256:3c57da0513e9563eb7e203ebe9bb3a1b509b042016433bd1e45a2853466c3dd3", + "sha256:7e4dc12271d8e15612bfe35792f5ea1c40970dadf8624602e33db2758f7ee554" + ], + "markers": "python_version >= '3.8'", + "version": "==0.32.1" + }, + "rpds-py": { + "hashes": [ + "sha256:01f58a7306b64e0a4fe042047dd2b7d411ee82e54240284bab63e325762c1147", + "sha256:0210b2668f24c078307260bf88bdac9d6f1093635df5123789bfee4d8d7fc8e7", + "sha256:02866e060219514940342a1f84303a1ef7a1dad0ac311792fbbe19b521b489d2", + "sha256:0387ce69ba06e43df54e43968090f3626e231e4bc9150e4c3246947567695f68", + "sha256:060f412230d5f19fc8c8b75f315931b408d8ebf56aec33ef4168d1b9e54200b1", + "sha256:071bc28c589b86bc6351a339114fb7a029f5cddbaca34103aa573eba7b482382", + "sha256:0bfb09bf41fe7c51413f563373e5f537eaa653d7adc4830399d4e9bdc199959d", + "sha256:10162fe3f5f47c37ebf6d8ff5a2368508fe22007e3077bf25b9c7d803454d921", + "sha256:149c5cd24f729e3567b56e1795f74577aa3126c14c11e457bec1b1c90d212e38", + "sha256:1701fc54460ae2e5efc1dd6350eafd7a760f516df8dbe51d4a1c79d69472fbd4", + "sha256:1957a2ab607f9added64478a6982742eb29f109d89d065fa44e01691a20fc20a", + "sha256:1a746a6d49665058a5896000e8d9d2f1a6acba8a03b389c1e4c06e11e0b7f40d", + "sha256:1bfcad3109c1e5ba3cbe2f421614e70439f72897515a96c462ea657261b96518", + "sha256:1d36b2b59e8cc6e576f8f7b671e32f2ff43153f0ad6d0201250a7c07f25d570e", + "sha256:1db228102ab9d1ff4c64148c96320d0be7044fa28bd865a9ce628ce98da5973d", + "sha256:1dc29db3900cb1bb40353772417800f29c3d078dbc8024fd64655a04ee3c4bdf", + "sha256:1e626b365293a2142a62b9a614e1f8e331b28f3ca57b9f05ebbf4cf2a0f0bdc5", + "sha256:1f3c3461ebb4c4f1bbc70b15d20b565759f97a5aaf13af811fcefc892e9197ba", + "sha256:20de7b7179e2031a04042e85dc463a93a82bc177eeba5ddd13ff746325558aa6", + "sha256:24e4900a6643f87058a27320f81336d527ccfe503984528edde4bb660c8c8d59", + "sha256:2528ff96d09f12e638695f3a2e0c609c7b84c6df7c5ae9bfeb9252b6fa686253", + "sha256:25f071737dae674ca8937a73d0f43f5a52e92c2d178330b4c0bb6ab05586ffa6", + "sha256:270987bc22e7e5a962b1094953ae901395e8c1e1e83ad016c5cfcfff75a15a3f", + "sha256:292f7344a3301802e7c25c53792fae7d1593cb0e50964e7bcdcc5cf533d634e3", + "sha256:2953937f83820376b5979318840f3ee47477d94c17b940fe31d9458d79ae7eea", + "sha256:2a792b2e1d3038daa83fa474d559acfd6dc1e3650ee93b2662ddc17dbff20ad1", + "sha256:2a7b2f2f56a16a6d62e55354dd329d929560442bd92e87397b7a9586a32e3e76", + "sha256:2f4eb548daf4836e3b2c662033bfbfc551db58d30fd8fe660314f86bf8510b93", + "sha256:3664d126d3388a887db44c2e293f87d500c4184ec43d5d14d2d2babdb4c64cad", + "sha256:3677fcca7fb728c86a78660c7fb1b07b69b281964673f486ae72860e13f512ad", + "sha256:380e0df2e9d5d5d339803cfc6d183a5442ad7ab3c63c2a0982e8c824566c5ccc", + "sha256:3ac732390d529d8469b831949c78085b034bff67f584559340008d0f6041a049", + "sha256:4128980a14ed805e1b91a7ed551250282a8ddf8201a4e9f8f5b7e6225f54170d", + "sha256:4341bd7579611cf50e7b20bb8c2e23512a3dc79de987a1f411cb458ab670eb90", + "sha256:436474f17733c7dca0fbf096d36ae65277e8645039df12a0fa52445ca494729d", + "sha256:4dc889a9d8a34758d0fcc9ac86adb97bab3fb7f0c4d29794357eb147536483fd", + "sha256:4e21b76075c01d65d0f0f34302b5a7457d95721d5e0667aea65e5bb3ab415c25", + "sha256:516fb8c77805159e97a689e2f1c80655c7658f5af601c34ffdb916605598cda2", + "sha256:5576ee2f3a309d2bb403ec292d5958ce03953b0e57a11d224c1f134feaf8c40f", + "sha256:5a024fa96d541fd7edaa0e9d904601c6445e95a729a2900c5aec6555fe921ed6", + "sha256:5d0e8a6434a3fbf77d11448c9c25b2f25244226cfbec1a5159947cac5b8c5fa4", + "sha256:5e7d63ec01fe7c76c2dbb7e972fece45acbb8836e72682bde138e7e039906e2c", + "sha256:60e820ee1004327609b28db8307acc27f5f2e9a0b185b2064c5f23e815f248f8", + "sha256:637b802f3f069a64436d432117a7e58fab414b4e27a7e81049817ae94de45d8d", + "sha256:65dcf105c1943cba45d19207ef51b8bc46d232a381e94dd38719d52d3980015b", + "sha256:698ea95a60c8b16b58be9d854c9f993c639f5c214cf9ba782eca53a8789d6b19", + "sha256:70fcc6c2906cfa5c6a552ba7ae2ce64b6c32f437d8f3f8eea49925b278a61453", + "sha256:720215373a280f78a1814becb1312d4e4d1077b1202a56d2b0815e95ccb99ce9", + "sha256:7450dbd659fed6dd41d1a7d47ed767e893ba402af8ae664c157c255ec6067fde", + "sha256:7b7d9ca34542099b4e185b3c2a2b2eda2e318a7dbde0b0d83357a6d4421b5296", + "sha256:7fbd70cb8b54fe745301921b0816c08b6d917593429dfc437fd024b5ba713c58", + "sha256:81038ff87a4e04c22e1d81f947c6ac46f122e0c80460b9006e6517c4d842a6ec", + "sha256:810685321f4a304b2b55577c915bece4c4a06dfe38f6e62d9cc1d6ca8ee86b99", + "sha256:82ada4a8ed9e82e443fcef87e22a3eed3654dd3adf6e3b3a0deb70f03e86142a", + "sha256:841320e1841bb53fada91c9725e766bb25009cfd4144e92298db296fb6c894fb", + "sha256:8587fd64c2a91c33cdc39d0cebdaf30e79491cc029a37fcd458ba863f8815383", + "sha256:8ffe53e1d8ef2520ebcf0c9fec15bb721da59e8ef283b6ff3079613b1e30513d", + "sha256:9051e3d2af8f55b42061603e29e744724cb5f65b128a491446cc029b3e2ea896", + "sha256:91e5a8200e65aaac342a791272c564dffcf1281abd635d304d6c4e6b495f29dc", + "sha256:93432e747fb07fa567ad9cc7aaadd6e29710e515aabf939dfbed8046041346c6", + "sha256:938eab7323a736533f015e6069a7d53ef2dcc841e4e533b782c2bfb9fb12d84b", + "sha256:9584f8f52010295a4a417221861df9bea4c72d9632562b6e59b3c7b87a1522b7", + "sha256:9737bdaa0ad33d34c0efc718741abaafce62fadae72c8b251df9b0c823c63b22", + "sha256:99da0a4686ada4ed0f778120a0ea8d066de1a0a92ab0d13ae68492a437db78bf", + "sha256:99f567dae93e10be2daaa896e07513dd4bf9c2ecf0576e0533ac36ba3b1d5394", + "sha256:9bdf1303df671179eaf2cb41e8515a07fc78d9d00f111eadbe3e14262f59c3d0", + "sha256:9f0e4dc0f17dcea4ab9d13ac5c666b6b5337042b4d8f27e01b70fae41dd65c57", + "sha256:a000133a90eea274a6f28adc3084643263b1e7c1a5a66eb0a0a7a36aa757ed74", + "sha256:a3264e3e858de4fc601741498215835ff324ff2482fd4e4af61b46512dd7fc83", + "sha256:a71169d505af63bb4d20d23a8fbd4c6ce272e7bce6cc31f617152aa784436f29", + "sha256:a967dd6afda7715d911c25a6ba1517975acd8d1092b2f326718725461a3d33f9", + "sha256:aa5bfb13f1e89151ade0eb812f7b0d7a4d643406caaad65ce1cbabe0a66d695f", + "sha256:ae35e8e6801c5ab071b992cb2da958eee76340e6926ec693b5ff7d6381441745", + "sha256:b686f25377f9c006acbac63f61614416a6317133ab7fafe5de5f7dc8a06d42eb", + "sha256:b760a56e080a826c2e5af09002c1a037382ed21d03134eb6294812dda268c811", + "sha256:b86b21b348f7e5485fae740d845c65a880f5d1eda1e063bc59bef92d1f7d0c55", + "sha256:b9412abdf0ba70faa6e2ee6c0cc62a8defb772e78860cef419865917d86c7342", + "sha256:bd345a13ce06e94c753dab52f8e71e5252aec1e4f8022d24d56decd31e1b9b23", + "sha256:be22ae34d68544df293152b7e50895ba70d2a833ad9566932d750d3625918b82", + "sha256:bf046179d011e6114daf12a534d874958b039342b347348a78b7cdf0dd9d6041", + "sha256:c3d2010656999b63e628a3c694f23020322b4178c450dc478558a2b6ef3cb9bb", + "sha256:c64602e8be701c6cfe42064b71c84ce62ce66ddc6422c15463fd8127db3d8066", + "sha256:d65e6b4f1443048eb7e833c2accb4fa7ee67cc7d54f31b4f0555b474758bee55", + "sha256:d8bbd8e56f3ba25a7d0cf980fc42b34028848a53a0e36c9918550e0280b9d0b6", + "sha256:da1ead63368c04a9bded7904757dfcae01eba0e0f9bc41d3d7f57ebf1c04015a", + "sha256:dbbb95e6fc91ea3102505d111b327004d1c4ce98d56a4a02e82cd451f9f57140", + "sha256:dbc56680ecf585a384fbd93cd42bc82668b77cb525343170a2d86dafaed2a84b", + "sha256:df3b6f45ba4515632c5064e35ca7f31d51d13d1479673185ba8f9fefbbed58b9", + "sha256:dfe07308b311a8293a0d5ef4e61411c5c20f682db6b5e73de6c7c8824272c256", + "sha256:e796051f2070f47230c745d0a77a91088fbee2cc0502e9b796b9c6471983718c", + "sha256:efa767c220d94aa4ac3a6dd3aeb986e9f229eaf5bce92d8b1b3018d06bed3772", + "sha256:f0b8bf5b8db49d8fd40f54772a1dcf262e8be0ad2ab0206b5a2ec109c176c0a4", + "sha256:f175e95a197f6a4059b50757a3dca33b32b61691bdbd22c29e8a8d21d3914cae", + "sha256:f2f3b28b40fddcb6c1f1f6c88c6f3769cd933fa493ceb79da45968a21dccc920", + "sha256:f6c43b6f97209e370124baf2bf40bb1e8edc25311a158867eb1c3a5d449ebc7a", + "sha256:f7f4cb1f173385e8a39c29510dd11a78bf44e360fb75610594973f5ea141028b", + "sha256:fad059a4bd14c45776600d223ec194e77db6c20255578bb5bcdd7c18fd169361", + "sha256:ff1dcb8e8bc2261a088821b2595ef031c91d499a0c1b031c152d43fe0a6ecec8", + "sha256:ffee088ea9b593cc6160518ba9bd319b5475e5f3e578e4552d63818773c6f56a" ], "markers": "python_version >= '3.8'", - "version": "==0.20.0" + "version": "==0.17.1" }, "typing-extensions": { "hashes": [ - "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d", - "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8" + "sha256:23478f88c37f27d76ac8aee6c905017a143b0b1b886c3c9f66bc2fd94f9f5783", + "sha256:af72aea155e91adfc61c3ae9e0e342dbc0cba726d6cba4b6c72c1f34e47291cd" ], "markers": "python_version >= '3.8'", - "version": "==4.12.2" + "version": "==4.9.0" } }, "develop": { @@ -143,14 +226,6 @@ "markers": "python_version >= '3.7'", "version": "==23.2.0" }, - "backports.tarfile": { - "hashes": [ - "sha256:77e284d754527b01fb1e6fa8a1afe577858ebe4e9dad8919e34c862cb399bc34", - "sha256:d75e02c268746e1b8144c278978b6e98e85de6ad16f8e4b0844a154557eca991" - ], - "markers": "python_version < '3.12'", - "version": "==1.2.0" - }, "black": { "hashes": [ "sha256:06f9d8846f2340dfac80ceb20200ea5d1b3f181dd0556b47af4e8e0b24fa0a6b", @@ -183,11 +258,11 @@ }, "certifi": { "hashes": [ - "sha256:5a1e7645bc0ec61a09e26c36f6106dd4cf40c6db3a1fb6352b0244e7fb057c7b", - "sha256:c198e21b1289c2ab85ee4e67bb4b4ef3ead0892059901a8d5b622f24a1101e90" + "sha256:9b469f3a900bf28dc19b8cfbf8019bf47f7fdd1a65a1d4ffb98fc14166beb4d1", + "sha256:e036ab49d5b79556f99cfc2d9320b34cfbe5be05c5871b51de9329f0603b0474" ], "markers": "python_version >= '3.6'", - "version": "==2024.7.4" + "version": "==2023.11.17" }, "charset-normalizer": { "hashes": [ @@ -306,85 +381,85 @@ "toml" ], "hashes": [ - "sha256:0086cd4fc71b7d485ac93ca4239c8f75732c2ae3ba83f6be1c9be59d9e2c6382", - "sha256:01c322ef2bbe15057bc4bf132b525b7e3f7206f071799eb8aa6ad1940bcf5fb1", - "sha256:03cafe82c1b32b770a29fd6de923625ccac3185a54a5e66606da26d105f37dac", - "sha256:044a0985a4f25b335882b0966625270a8d9db3d3409ddc49a4eb00b0ef5e8cee", - "sha256:07ed352205574aad067482e53dd606926afebcb5590653121063fbf4e2175166", - "sha256:0d1b923fc4a40c5832be4f35a5dab0e5ff89cddf83bb4174499e02ea089daf57", - "sha256:0e7b27d04131c46e6894f23a4ae186a6a2207209a05df5b6ad4caee6d54a222c", - "sha256:1fad32ee9b27350687035cb5fdf9145bc9cf0a094a9577d43e909948ebcfa27b", - "sha256:289cc803fa1dc901f84701ac10c9ee873619320f2f9aff38794db4a4a0268d51", - "sha256:3c59105f8d58ce500f348c5b56163a4113a440dad6daa2294b5052a10db866da", - "sha256:46c3d091059ad0b9c59d1034de74a7f36dcfa7f6d3bde782c49deb42438f2450", - "sha256:482855914928c8175735a2a59c8dc5806cf7d8f032e4820d52e845d1f731dca2", - "sha256:49c76cdfa13015c4560702574bad67f0e15ca5a2872c6a125f6327ead2b731dd", - "sha256:4b03741e70fb811d1a9a1d75355cf391f274ed85847f4b78e35459899f57af4d", - "sha256:4bea27c4269234e06f621f3fac3925f56ff34bc14521484b8f66a580aacc2e7d", - "sha256:4d5fae0a22dc86259dee66f2cc6c1d3e490c4a1214d7daa2a93d07491c5c04b6", - "sha256:543ef9179bc55edfd895154a51792b01c017c87af0ebaae092720152e19e42ca", - "sha256:54dece71673b3187c86226c3ca793c5f891f9fc3d8aa183f2e3653da18566169", - "sha256:6379688fb4cfa921ae349c76eb1a9ab26b65f32b03d46bb0eed841fd4cb6afb1", - "sha256:65fa405b837060db569a61ec368b74688f429b32fa47a8929a7a2f9b47183713", - "sha256:6616d1c9bf1e3faea78711ee42a8b972367d82ceae233ec0ac61cc7fec09fa6b", - "sha256:6fe885135c8a479d3e37a7aae61cbd3a0fb2deccb4dda3c25f92a49189f766d6", - "sha256:7221f9ac9dad9492cecab6f676b3eaf9185141539d5c9689d13fd6b0d7de840c", - "sha256:76d5f82213aa78098b9b964ea89de4617e70e0d43e97900c2778a50856dac605", - "sha256:7792f0ab20df8071d669d929c75c97fecfa6bcab82c10ee4adb91c7a54055463", - "sha256:831b476d79408ab6ccfadaaf199906c833f02fdb32c9ab907b1d4aa0713cfa3b", - "sha256:9146579352d7b5f6412735d0f203bbd8d00113a680b66565e205bc605ef81bc6", - "sha256:9cc44bf0315268e253bf563f3560e6c004efe38f76db03a1558274a6e04bf5d5", - "sha256:a73d18625f6a8a1cbb11eadc1d03929f9510f4131879288e3f7922097a429f63", - "sha256:a8659fd33ee9e6ca03950cfdcdf271d645cf681609153f218826dd9805ab585c", - "sha256:a94925102c89247530ae1dab7dc02c690942566f22e189cbd53579b0693c0783", - "sha256:ad4567d6c334c46046d1c4c20024de2a1c3abc626817ae21ae3da600f5779b44", - "sha256:b2e16f4cd2bc4d88ba30ca2d3bbf2f21f00f382cf4e1ce3b1ddc96c634bc48ca", - "sha256:bbdf9a72403110a3bdae77948b8011f644571311c2fb35ee15f0f10a8fc082e8", - "sha256:beb08e8508e53a568811016e59f3234d29c2583f6b6e28572f0954a6b4f7e03d", - "sha256:c4cbe651f3904e28f3a55d6f371203049034b4ddbce65a54527a3f189ca3b390", - "sha256:c7b525ab52ce18c57ae232ba6f7010297a87ced82a2383b1afd238849c1ff933", - "sha256:ca5d79cfdae420a1d52bf177de4bc2289c321d6c961ae321503b2ca59c17ae67", - "sha256:cdab02a0a941af190df8782aafc591ef3ad08824f97850b015c8c6a8b3877b0b", - "sha256:d17c6a415d68cfe1091d3296ba5749d3d8696e42c37fca5d4860c5bf7b729f03", - "sha256:d39bd10f0ae453554798b125d2f39884290c480f56e8a02ba7a6ed552005243b", - "sha256:d4b3cd1ca7cd73d229487fa5caca9e4bc1f0bca96526b922d61053ea751fe791", - "sha256:d50a252b23b9b4dfeefc1f663c568a221092cbaded20a05a11665d0dbec9b8fb", - "sha256:da8549d17489cd52f85a9829d0e1d91059359b3c54a26f28bec2c5d369524807", - "sha256:dcd070b5b585b50e6617e8972f3fbbee786afca71b1936ac06257f7e178f00f6", - "sha256:ddaaa91bfc4477d2871442bbf30a125e8fe6b05da8a0015507bfbf4718228ab2", - "sha256:df423f351b162a702c053d5dddc0fc0ef9a9e27ea3f449781ace5f906b664428", - "sha256:dff044f661f59dace805eedb4a7404c573b6ff0cdba4a524141bc63d7be5c7fd", - "sha256:e7e128f85c0b419907d1f38e616c4f1e9f1d1b37a7949f44df9a73d5da5cd53c", - "sha256:ed8d1d1821ba5fc88d4a4f45387b65de52382fa3ef1f0115a4f7a20cdfab0e94", - "sha256:f2501d60d7497fd55e391f423f965bbe9e650e9ffc3c627d5f0ac516026000b8", - "sha256:f7db0b6ae1f96ae41afe626095149ecd1b212b424626175a6633c2999eaad45b" + "sha256:04387a4a6ecb330c1878907ce0dc04078ea72a869263e53c72a1ba5bbdf380ca", + "sha256:0676cd0ba581e514b7f726495ea75aba3eb20899d824636c6f59b0ed2f88c471", + "sha256:0e8d06778e8fbffccfe96331a3946237f87b1e1d359d7fbe8b06b96c95a5407a", + "sha256:0eb3c2f32dabe3a4aaf6441dde94f35687224dfd7eb2a7f47f3fd9428e421058", + "sha256:109f5985182b6b81fe33323ab4707011875198c41964f014579cf82cebf2bb85", + "sha256:13eaf476ec3e883fe3e5fe3707caeb88268a06284484a3daf8250259ef1ba143", + "sha256:164fdcc3246c69a6526a59b744b62e303039a81e42cfbbdc171c91a8cc2f9446", + "sha256:26776ff6c711d9d835557ee453082025d871e30b3fd6c27fcef14733f67f0590", + "sha256:26f66da8695719ccf90e794ed567a1549bb2644a706b41e9f6eae6816b398c4a", + "sha256:29f3abe810930311c0b5d1a7140f6395369c3db1be68345638c33eec07535105", + "sha256:316543f71025a6565677d84bc4df2114e9b6a615aa39fb165d697dba06a54af9", + "sha256:36b0ea8ab20d6a7564e89cb6135920bc9188fb5f1f7152e94e8300b7b189441a", + "sha256:3cc9d4bc55de8003663ec94c2f215d12d42ceea128da8f0f4036235a119c88ac", + "sha256:485e9f897cf4856a65a57c7f6ea3dc0d4e6c076c87311d4bc003f82cfe199d25", + "sha256:5040148f4ec43644702e7b16ca864c5314ccb8ee0751ef617d49aa0e2d6bf4f2", + "sha256:51456e6fa099a8d9d91497202d9563a320513fcf59f33991b0661a4a6f2ad450", + "sha256:53d7d9158ee03956e0eadac38dfa1ec8068431ef8058fe6447043db1fb40d932", + "sha256:5a10a4920def78bbfff4eff8a05c51be03e42f1c3735be42d851f199144897ba", + "sha256:5b14b4f8760006bfdb6e08667af7bc2d8d9bfdb648351915315ea17645347137", + "sha256:5b2ccb7548a0b65974860a78c9ffe1173cfb5877460e5a229238d985565574ae", + "sha256:697d1317e5290a313ef0d369650cfee1a114abb6021fa239ca12b4849ebbd614", + "sha256:6ae8c9d301207e6856865867d762a4b6fd379c714fcc0607a84b92ee63feff70", + "sha256:707c0f58cb1712b8809ece32b68996ee1e609f71bd14615bd8f87a1293cb610e", + "sha256:74775198b702868ec2d058cb92720a3c5a9177296f75bd97317c787daf711505", + "sha256:756ded44f47f330666843b5781be126ab57bb57c22adbb07d83f6b519783b870", + "sha256:76f03940f9973bfaee8cfba70ac991825611b9aac047e5c80d499a44079ec0bc", + "sha256:79287fd95585ed36e83182794a57a46aeae0b64ca53929d1176db56aacc83451", + "sha256:799c8f873794a08cdf216aa5d0531c6a3747793b70c53f70e98259720a6fe2d7", + "sha256:7d360587e64d006402b7116623cebf9d48893329ef035278969fa3bbf75b697e", + "sha256:80b5ee39b7f0131ebec7968baa9b2309eddb35b8403d1869e08f024efd883566", + "sha256:815ac2d0f3398a14286dc2cea223a6f338109f9ecf39a71160cd1628786bc6f5", + "sha256:83c2dda2666fe32332f8e87481eed056c8b4d163fe18ecc690b02802d36a4d26", + "sha256:846f52f46e212affb5bcf131c952fb4075b55aae6b61adc9856222df89cbe3e2", + "sha256:936d38794044b26c99d3dd004d8af0035ac535b92090f7f2bb5aa9c8e2f5cd42", + "sha256:9864463c1c2f9cb3b5db2cf1ff475eed2f0b4285c2aaf4d357b69959941aa555", + "sha256:995ea5c48c4ebfd898eacb098164b3cc826ba273b3049e4a889658548e321b43", + "sha256:a1526d265743fb49363974b7aa8d5899ff64ee07df47dd8d3e37dcc0818f09ed", + "sha256:a56de34db7b7ff77056a37aedded01b2b98b508227d2d0979d373a9b5d353daa", + "sha256:a7c97726520f784239f6c62506bc70e48d01ae71e9da128259d61ca5e9788516", + "sha256:b8e99f06160602bc64da35158bb76c73522a4010f0649be44a4e167ff8555952", + "sha256:bb1de682da0b824411e00a0d4da5a784ec6496b6850fdf8c865c1d68c0e318dd", + "sha256:bf477c355274a72435ceb140dc42de0dc1e1e0bf6e97195be30487d8eaaf1a09", + "sha256:bf635a52fc1ea401baf88843ae8708591aa4adff875e5c23220de43b1ccf575c", + "sha256:bfd5db349d15c08311702611f3dccbef4b4e2ec148fcc636cf8739519b4a5c0f", + "sha256:c530833afc4707fe48524a44844493f36d8727f04dcce91fb978c414a8556cc6", + "sha256:cc6d65b21c219ec2072c1293c505cf36e4e913a3f936d80028993dd73c7906b1", + "sha256:cd3c1e4cb2ff0083758f09be0f77402e1bdf704adb7f89108007300a6da587d0", + "sha256:cfd2a8b6b0d8e66e944d47cdec2f47c48fef2ba2f2dff5a9a75757f64172857e", + "sha256:d0ca5c71a5a1765a0f8f88022c52b6b8be740e512980362f7fdbb03725a0d6b9", + "sha256:e7defbb9737274023e2d7af02cac77043c86ce88a907c58f42b580a97d5bcca9", + "sha256:e9d1bf53c4c8de58d22e0e956a79a5b37f754ed1ffdbf1a260d9dcfa2d8a325e", + "sha256:ea81d8f9691bb53f4fb4db603203029643caffc82bf998ab5b59ca05560f4c06" ], "markers": "python_version >= '3.8'", - "version": "==7.6.0" + "version": "==7.4.0" }, "docutils": { "hashes": [ - "sha256:3a6b18732edf182daa3cd12775bbb338cf5691468f91eeeb109deff6ebfa986f", - "sha256:dafca5b9e384f0e419294eb4d2ff9fa826435bf15f15b7bd45723e8ad76811b2" + "sha256:96f387a2c5562db4476f09f13bbab2192e764cac08ebbf3a34a95d9b1e4a59d6", + "sha256:f08a4e276c3a1583a86dce3e34aba3fe04d02bba2dd51ed16106244e8a923e3b" ], - "markers": "python_version >= '3.9'", - "version": "==0.21.2" + "markers": "python_version >= '3.7'", + "version": "==0.20.1" }, "exceptiongroup": { "hashes": [ - "sha256:3111b9d131c238bec2f8f516e123e14ba243563fb135d3fe885990585aa7795b", - "sha256:47c2edf7c6738fafb49fd34290706d1a1a2f4d1c6df275526b62cbb4aa5393cc" + "sha256:4bfd3996ac73b41e9b9628b04e079f193850720ea5945fc96a08633c66912f14", + "sha256:91f5c769735f051a4290d52edd0858999b57e5876e9f85937691bd4c9fa3ed68" ], "markers": "python_version < '3.11'", - "version": "==1.2.2" + "version": "==1.2.0" }, "execnet": { "hashes": [ - "sha256:26dee51f1b80cebd6d0ca8e74dd8745419761d3bef34163928cbebbdc4749fdc", - "sha256:5189b52c6121c24feae288166ab41b32549c7e2348652736540b9e6e7d4e72e3" + "sha256:88256416ae766bc9e8895c76a87928c0012183da3cc4fc18016e6f050e025f41", + "sha256:cc59bc4423742fd71ad227122eb0dd44db51efb3dc4095b45ac9a08c770096af" ], - "markers": "python_version >= '3.8'", - "version": "==2.1.1" + "markers": "python_version >= '3.7'", + "version": "==2.0.2" }, "flake8": { "hashes": [ @@ -424,19 +499,19 @@ }, "idna": { "hashes": [ - "sha256:028ff3aadf0609c1fd278d8ea3089299412a7a8b9bd005dd08b9f8285bcb5cfc", - "sha256:82fee1fc78add43492d3a1898bfa6d8a904cc97d8427f683ed8e798d07761aa0" + "sha256:9ecdbbd083b06798ae1e86adcbfe8ab1479cf864e4ee30fe4e46a003d12491ca", + "sha256:c05567e9c24a6b9faaa835c4821bad0590fbb9d5779e7caa6e1cc4978e7eb24f" ], "markers": "python_version >= '3.5'", - "version": "==3.7" + "version": "==3.6" }, "importlib-metadata": { "hashes": [ - "sha256:15584cf2b1bf449d98ff8a6ff1abef57bf20f3ac6454f431736cd3e660921b2f", - "sha256:188bd24e4c346d3f0a933f275c2fec67050326a856b9a359881d7c2a697e8812" + "sha256:4805911c3a4ec7c3966410053e9ec6a1fecd629117df5adee56dfc9432a1081e", + "sha256:f238736bb06590ae52ac1fab06a3a9ef1d8dce2b7a35b5ab329371d6c8f5d2cc" ], "markers": "python_version >= '3.8'", - "version": "==8.0.0" + "version": "==7.0.1" }, "iniconfig": { "hashes": [ @@ -448,35 +523,19 @@ }, "jaraco.classes": { "hashes": [ - "sha256:47a024b51d0239c0dd8c8540c6c7f484be3b8fcf0b2d85c13825780d3b3f3acd", - "sha256:f662826b6bed8cace05e7ff873ce0f9283b5c924470fe664fff1c2f00f581790" + "sha256:10afa92b6743f25c0cf5f37c6bb6e18e2c5bb84a16527ccfc0040ea377e7aaeb", + "sha256:c063dd08e89217cee02c8d5e5ec560f2c8ce6cdc2fcdc2e68f7b2e5547ed3621" ], "markers": "python_version >= '3.8'", - "version": "==3.4.0" - }, - "jaraco.context": { - "hashes": [ - "sha256:3e16388f7da43d384a1a7cd3452e72e14732ac9fe459678773a3608a812bf266", - "sha256:c2f67165ce1f9be20f32f650f25d8edfc1646a8aeee48ae06fb35f90763576d2" - ], - "markers": "python_version >= '3.8'", - "version": "==5.3.0" - }, - "jaraco.functools": { - "hashes": [ - "sha256:3b24ccb921d6b593bdceb56ce14799204f473976e2a9d4b15b04d0f2c2326664", - "sha256:d33fa765374c0611b52f8b3a795f8900869aa88c84769d4d1746cd68fb28c3e8" - ], - "markers": "python_version >= '3.8'", - "version": "==4.0.1" + "version": "==3.3.0" }, "keyring": { "hashes": [ - "sha256:2458681cdefc0dbc0b7eb6cf75d0b98e59f9ad9b2d4edd319d18f68bdca95e50", - "sha256:daaffd42dbda25ddafb1ad5fec4024e5bbcfe424597ca1ca452b299861e49f1b" + "sha256:4446d35d636e6a10b8bce7caa66913dd9eca5fd222ca03a3d42c38608ac30836", + "sha256:e730ecffd309658a08ee82535a3b5ec4b4c8669a9be11efb66249d8e0aeb9a25" ], "markers": "python_version >= '3.8'", - "version": "==25.2.1" + "version": "==24.3.0" }, "markdown-it-py": { "hashes": [ @@ -504,11 +563,11 @@ }, "more-itertools": { "hashes": [ - "sha256:e5d93ef411224fbcef366a6e8ddc4c5781bc6359d43412a65dd5964e46111463", - "sha256:ea6a02e24a9161e51faad17a8782b92a0df82c12c1c8886fec7f0c3fa1a1b320" + "sha256:686b06abe565edfab151cb8fd385a05651e1fdf8f0a14191e4439283421f8684", + "sha256:8fccb480c43d3e99a00087634c06dd02b0d50fbf088b380de5a41a015ec239e1" ], "markers": "python_version >= '3.8'", - "version": "==10.3.0" + "version": "==10.2.0" }, "mypy": { "hashes": [ @@ -554,32 +613,32 @@ }, "nh3": { "hashes": [ - "sha256:0411beb0589eacb6734f28d5497ca2ed379eafab8ad8c84b31bb5c34072b7164", - "sha256:14c5a72e9fe82aea5fe3072116ad4661af5cf8e8ff8fc5ad3450f123e4925e86", - "sha256:19aaba96e0f795bd0a6c56291495ff59364f4300d4a39b29a0abc9cb3774a84b", - "sha256:34c03fa78e328c691f982b7c03d4423bdfd7da69cd707fe572f544cf74ac23ad", - "sha256:36c95d4b70530b320b365659bb5034341316e6a9b30f0b25fa9c9eff4c27a204", - "sha256:3a157ab149e591bb638a55c8c6bcb8cdb559c8b12c13a8affaba6cedfe51713a", - "sha256:42c64511469005058cd17cc1537578eac40ae9f7200bedcfd1fc1a05f4f8c200", - "sha256:5f36b271dae35c465ef5e9090e1fdaba4a60a56f0bb0ba03e0932a66f28b9189", - "sha256:6955369e4d9f48f41e3f238a9e60f9410645db7e07435e62c6a9ea6135a4907f", - "sha256:7b7c2a3c9eb1a827d42539aa64091640bd275b81e097cd1d8d82ef91ffa2e811", - "sha256:8ce0f819d2f1933953fca255db2471ad58184a60508f03e6285e5114b6254844", - "sha256:94a166927e53972a9698af9542ace4e38b9de50c34352b962f4d9a7d4c927af4", - "sha256:a7f1b5b2c15866f2db413a3649a8fe4fd7b428ae58be2c0f6bca5eefd53ca2be", - "sha256:c8b3a1cebcba9b3669ed1a84cc65bf005728d2f0bc1ed2a6594a992e817f3a50", - "sha256:de3ceed6e661954871d6cd78b410213bdcb136f79aafe22aa7182e028b8c7307", - "sha256:f0eca9ca8628dbb4e916ae2491d72957fdd35f7a5d326b7032a345f111ac07fe" - ], - "version": "==0.2.18" + "sha256:0d02d0ff79dfd8208ed25a39c12cbda092388fff7f1662466e27d97ad011b770", + "sha256:3277481293b868b2715907310c7be0f1b9d10491d5adf9fce11756a97e97eddf", + "sha256:3b803a5875e7234907f7d64777dfde2b93db992376f3d6d7af7f3bc347deb305", + "sha256:427fecbb1031db085eaac9931362adf4a796428ef0163070c484b5a768e71601", + "sha256:5f0d77272ce6d34db6c87b4f894f037d55183d9518f948bba236fe81e2bb4e28", + "sha256:60684857cfa8fdbb74daa867e5cad3f0c9789415aba660614fe16cd66cbb9ec7", + "sha256:6f42f99f0cf6312e470b6c09e04da31f9abaadcd3eb591d7d1a88ea931dca7f3", + "sha256:86e447a63ca0b16318deb62498db4f76fc60699ce0a1231262880b38b6cff911", + "sha256:8d595df02413aa38586c24811237e95937ef18304e108b7e92c890a06793e3bf", + "sha256:9c0d415f6b7f2338f93035bba5c0d8c1b464e538bfbb1d598acd47d7969284f0", + "sha256:a5167a6403d19c515217b6bcaaa9be420974a6ac30e0da9e84d4fc67a5d474c5", + "sha256:ac19c0d68cd42ecd7ead91a3a032fdfff23d29302dbb1311e641a130dfefba97", + "sha256:b1e97221cedaf15a54f5243f2c5894bb12ca951ae4ddfd02a9d4ea9df9e1a29d", + "sha256:bc2d086fb540d0fa52ce35afaded4ea526b8fc4d3339f783db55c95de40ef02e", + "sha256:d1e30ff2d8d58fb2a14961f7aac1bbb1c51f9bdd7da727be35c63826060b0bf3", + "sha256:f3b53ba93bb7725acab1e030bc2ecd012a817040fd7851b332f86e2f9bb98dc6" + ], + "version": "==0.2.15" }, "packaging": { "hashes": [ - "sha256:026ed72c8ed3fcce5bf8950572258698927fd1dbda10a5e981cdf0ac37f4f002", - "sha256:5b8f2217dbdbd2f7f384c41c628544e6d52f2d0f53c6d0c3ea61aa5d1d7ff124" + "sha256:048fb0e9405036518eaaf48a55953c750c11e1a1b68e0dd1a9d62ed0c092cfc5", + "sha256:8c491190033a9af7e1d931d0b5dacc2ef47509b34dd0de67ed209b5203fc88c7" ], - "markers": "python_version >= '3.8'", - "version": "==24.1" + "markers": "python_version >= '3.7'", + "version": "==23.2" }, "pathspec": { "hashes": [ @@ -591,27 +650,27 @@ }, "pkginfo": { "hashes": [ - "sha256:2e0dca1cf4c8e39644eed32408ea9966ee15e0d324c62ba899a393b3c6b467aa", - "sha256:bfa76a714fdfc18a045fcd684dbfc3816b603d9d075febef17cb6582bea29573" + "sha256:4b7a555a6d5a22169fcc9cf7bfd78d296b0361adad412a346c1226849af5e546", + "sha256:8fd5896e8718a4372f0ea9cc9d96f6417c9b986e23a4d116dda26b62cc29d046" ], - "markers": "python_version >= '3.8'", - "version": "==1.11.1" + "markers": "python_version >= '3.6'", + "version": "==1.9.6" }, "platformdirs": { "hashes": [ - "sha256:2d7a1657e36a80ea911db832a8a6ece5ee53d8de21edd5cc5879af6530b1bfee", - "sha256:38b7b51f512eed9e84a22788b4bce1de17c0adb134d6becb09836e37d8654cd3" + "sha256:11c8f37bcca40db96d8144522d925583bdb7a31f7b0e37e3ed4318400a8e2380", + "sha256:906d548203468492d432bcb294d4bc2fff751bf84971fbb2c10918cc206ee420" ], "markers": "python_version >= '3.8'", - "version": "==4.2.2" + "version": "==4.1.0" }, "pluggy": { "hashes": [ - "sha256:2cffa88e94fdc978c4c574f15f9e59b7f4201d439195c3715ca9e2486f1d0cf1", - "sha256:44e1ad92c8ca002de6377e165f3e0f1be63266ab4d554740532335b9d75ea669" + "sha256:cf61ae8f126ac6f7c451172cf30e3e43d3ca77615509771b3a984a0730651e12", + "sha256:d89c696a773f8bd377d18e5ecda92b7a3793cbe66c87060a6fb58c7b6e1061f7" ], "markers": "python_version >= '3.8'", - "version": "==1.5.0" + "version": "==1.3.0" }, "pprintpp": { "hashes": [ @@ -646,11 +705,11 @@ }, "pygments": { "hashes": [ - "sha256:786ff802f32e91311bff3889f6e9a86e81505fe99f2735bb6d60ae0c5004f199", - "sha256:b8e6aca0523f3ab76fee51799c488e38782ac06eafcf95e7ba832985c8e7b13a" + "sha256:b27c2826c47d0f3219f29554824c30c5e8945175d888647acd804ddd04af846c", + "sha256:da46cec9fd2de5be3a8a784f434e4c4ab670b4ff54d605c4c2717e9d49c4c367" ], - "markers": "python_version >= '3.8'", - "version": "==2.18.0" + "markers": "python_version >= '3.7'", + "version": "==2.17.2" }, "pytest": { "hashes": [ @@ -689,19 +748,19 @@ }, "readme-renderer": { "hashes": [ - "sha256:2fbca89b81a08526aadf1357a8c2ae889ec05fb03f5da67f9769c9a592166151", - "sha256:8712034eabbfa6805cacf1402b4eeb2a73028f72d1166d6f5cb7f9c047c5d1e1" + "sha256:13d039515c1f24de668e2c93f2e877b9dbe6c6c32328b90a40a49d8b2b85f36d", + "sha256:2d55489f83be4992fe4454939d1a051c33edbab778e82761d060c9fc6b308cd1" ], - "markers": "python_version >= '3.9'", - "version": "==44.0" + "markers": "python_version >= '3.8'", + "version": "==42.0" }, "requests": { "hashes": [ - "sha256:55365417734eb18255590a9ff9eb97e9e1da868d4ccd6402399eaf68af20a760", - "sha256:70761cfe03c773ceb22aa2f671b4757976145175cdfca038c02654d061d6dcc6" + "sha256:58cd2187c01e70e6e26505bca751777aa9f2ee0b7f4300988b709f44e013003f", + "sha256:942c5a758f98d790eaed1a29cb6eefc7ffb0d1cf7af05c3d2791656dbd6ad1e1" ], - "markers": "python_version >= '3.8'", - "version": "==2.32.3" + "markers": "python_version >= '3.7'", + "version": "==2.31.0" }, "requests-toolbelt": { "hashes": [ @@ -721,11 +780,11 @@ }, "rich": { "hashes": [ - "sha256:4edbae314f59eb482f54e9e30bf00d33350aaa94f4bfcd4e9e3110e64d0d7222", - "sha256:9be308cb1fe2f1f57d67ce99e95af38a1e2bc71ad9813b0e247cf7ffbcc3a432" + "sha256:5cb5123b5cf9ee70584244246816e9114227e0b98ad9176eede6ad54bf5403fa", + "sha256:6da14c108c4866ee9520bbffa71f6fe3962e193b7da68720583850cd4548e235" ], "markers": "python_full_version >= '3.7.0'", - "version": "==13.7.1" + "version": "==13.7.0" }, "snowballstemmer": { "hashes": [ @@ -762,19 +821,19 @@ }, "typing-extensions": { "hashes": [ - "sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d", - "sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8" + "sha256:23478f88c37f27d76ac8aee6c905017a143b0b1b886c3c9f66bc2fd94f9f5783", + "sha256:af72aea155e91adfc61c3ae9e0e342dbc0cba726d6cba4b6c72c1f34e47291cd" ], "markers": "python_version >= '3.8'", - "version": "==4.12.2" + "version": "==4.9.0" }, "urllib3": { "hashes": [ - "sha256:a448b2f64d686155468037e1ace9f2d2199776e17f0a46610480d311f73e3472", - "sha256:dd505485549a7a552833da5e6063639d0d177c04f23bc3864e41e5dc5f612168" + "sha256:55901e917a5896a349ff771be919f8bd99aff50b79fe58fec595eb37bbc56bb3", + "sha256:df7aa8afb0148fa78488e7899b2c59b5f4ffcfa82e6c54ccb9dd37c1d7b52d54" ], "markers": "python_version >= '3.8'", - "version": "==2.2.2" + "version": "==2.1.0" }, "wheel": { "hashes": [ @@ -787,11 +846,11 @@ }, "zipp": { "hashes": [ - "sha256:bf1dcf6450f873a13e952a29504887c89e6de7506209e5b1bcc3460135d4de19", - "sha256:f091755f667055f2d02b32c53771a7a6c8b47e1fdbc4b72a8b9072b3eef8015c" + "sha256:0e923e726174922dce09c53c59ad483ff7bbb8e572e00c7f7c46b88556409f31", + "sha256:84e64a1c28cf7e91ed2078bb8cc8c259cb19b76942096c8d7b84947690cabaf0" ], "markers": "python_version >= '3.8'", - "version": "==3.19.2" + "version": "==3.17.0" } } } diff --git a/shared-data/python/setup.py b/shared-data/python/setup.py index c82e4de1777..4e1720cb610 100644 --- a/shared-data/python/setup.py +++ b/shared-data/python/setup.py @@ -141,7 +141,7 @@ def get_version(): ) PACKAGES = find_packages(where=".", exclude=["tests", "tests.*"]) INSTALL_REQUIRES = [ - "jsonschema>=3.0.1,<5", + "jsonschema>=3.0.1,<4.18.0", "typing-extensions>=4.0.0,<5", "pydantic>=1.10.9,<2.0.0", ] From d3a8001741fd26db826d78f2e2933aeec6a722eb Mon Sep 17 00:00:00 2001 From: Shlok Amin Date: Tue, 23 Jul 2024 13:51:48 -0400 Subject: [PATCH 19/50] feat(api, app): delete references to robot log aggregation setting (#15635) closes AUTH-531 --- api/src/opentrons/config/advanced_settings.py | 31 ++++---- .../test_advanced_settings_migration.py | 9 +-- app/src/App/types.ts | 1 - .../RobotSettingsFeatureFlags.tsx | 1 - .../RobotSettings/RobotSettingsPrivacy.tsx | 71 ------------------- .../RobotSettingsFeatureFlags.test.tsx | 1 - .../RobotSettingsDashboard/Privacy.tsx | 28 +------- .../__tests__/Privacy.test.tsx | 14 +--- .../__tests__/RobotSettings.test.tsx | 14 ---- app/src/pages/Devices/RobotSettings/index.tsx | 11 +-- 10 files changed, 17 insertions(+), 164 deletions(-) delete mode 100644 app/src/organisms/Devices/RobotSettings/RobotSettingsPrivacy.tsx diff --git a/api/src/opentrons/config/advanced_settings.py b/api/src/opentrons/config/advanced_settings.py index f1d95d76c7e..6e3647939cb 100644 --- a/api/src/opentrons/config/advanced_settings.py +++ b/api/src/opentrons/config/advanced_settings.py @@ -16,7 +16,7 @@ List, ) -from opentrons.config import CONFIG, ARCHITECTURE, SystemArchitecture +from opentrons.config import CONFIG from opentrons_shared_data.robot.dev_types import RobotTypeEnum if TYPE_CHECKING: @@ -97,18 +97,6 @@ async def on_change(self, value: Optional[bool]) -> None: set_restart_required() -class DisableLogIntegrationSettingDefinition(SettingDefinition): - def __init__(self) -> None: - super().__init__( - _id="disableLogAggregation", - title="Disable Opentrons Log Collection", - description="Prevent the robot from sending its logs to Opentrons" - " for analysis. Opentrons uses these logs to" - " troubleshoot robot issues and spot error trends.", - robot_type=[RobotTypeEnum.OT2, RobotTypeEnum.FLEX], - ) - - class Setting(NamedTuple): value: Optional[bool] definition: SettingDefinition @@ -236,12 +224,6 @@ class Setting(NamedTuple): ), ] -if ( - ARCHITECTURE == SystemArchitecture.BUILDROOT - or ARCHITECTURE == SystemArchitecture.YOCTO -): - settings.append(DisableLogIntegrationSettingDefinition()) - settings_by_id: Dict[str, SettingDefinition] = {s.id: s for s in settings} settings_by_old_id: Dict[str, SettingDefinition] = { @@ -723,6 +705,16 @@ def _migrate33to34(previous: SettingsMap) -> SettingsMap: return newmap +def _migrate34to35(previous: SettingsMap) -> SettingsMap: + """Migrate to version 35 of the feature flags file. + + - Removes disableLogAggregation + """ + removals = ["disableLogAggregation"] + newmap = {k: v for k, v in previous.items() if k not in removals} + return newmap + + _MIGRATIONS = [ _migrate0to1, _migrate1to2, @@ -758,6 +750,7 @@ def _migrate33to34(previous: SettingsMap) -> SettingsMap: _migrate31to32, _migrate32to33, _migrate33to34, + _migrate34to35, ] """ List of all migrations to apply, indexed by (version - 1). See _migrate below diff --git a/api/tests/opentrons/config/test_advanced_settings_migration.py b/api/tests/opentrons/config/test_advanced_settings_migration.py index 283d11a3000..92a45a6d610 100644 --- a/api/tests/opentrons/config/test_advanced_settings_migration.py +++ b/api/tests/opentrons/config/test_advanced_settings_migration.py @@ -8,7 +8,7 @@ @pytest.fixture def migrated_file_version() -> int: - return 34 + return 35 # make sure to set a boolean value in default_file_settings only if @@ -20,7 +20,6 @@ def default_file_settings() -> Dict[str, Any]: "deckCalibrationDots": None, "disableHomeOnBoot": None, "useOldAspirationFunctions": None, - "disableLogAggregation": None, "enableDoorSafetySwitch": None, "enableOT3HardwareController": None, "rearPanelIntegration": True, @@ -69,7 +68,6 @@ def v2_config(v1_config: Dict[str, Any]) -> Dict[str, Any]: r.update( { "_version": 2, - "disableLogAggregation": True, } ) return r @@ -525,15 +523,12 @@ def test_ignores_invalid_keys( def test_ensures_config() -> None: - assert _ensure( - {"_version": 3, "shortFixedTrash": False, "disableLogAggregation": True} - ) == { + assert _ensure({"_version": 3, "shortFixedTrash": False}) == { "_version": 3, "shortFixedTrash": False, "deckCalibrationDots": None, "disableHomeOnBoot": None, "useOldAspirationFunctions": None, - "disableLogAggregation": True, "enableDoorSafetySwitch": None, "enableOT3HardwareController": None, "rearPanelIntegration": None, diff --git a/app/src/App/types.ts b/app/src/App/types.ts index b75f19aedb7..87d8f77d4a1 100644 --- a/app/src/App/types.ts +++ b/app/src/App/types.ts @@ -22,7 +22,6 @@ export type RobotSettingsTab = | 'networking' | 'advanced' | 'feature-flags' - | 'privacy' export type AppSettingsTab = | 'general' diff --git a/app/src/organisms/Devices/RobotSettings/RobotSettingsFeatureFlags.tsx b/app/src/organisms/Devices/RobotSettings/RobotSettingsFeatureFlags.tsx index ab3ae2d479e..90954fc1315 100644 --- a/app/src/organisms/Devices/RobotSettings/RobotSettingsFeatureFlags.tsx +++ b/app/src/organisms/Devices/RobotSettings/RobotSettingsFeatureFlags.tsx @@ -34,7 +34,6 @@ const NON_FEATURE_FLAG_SETTINGS = [ 'deckCalibrationDots', 'shortFixedTrash', 'useOldAspirationFunctions', - 'disableLogAggregation', 'disableFastProtocolUpload', 'disableStatusBar', ] diff --git a/app/src/organisms/Devices/RobotSettings/RobotSettingsPrivacy.tsx b/app/src/organisms/Devices/RobotSettings/RobotSettingsPrivacy.tsx deleted file mode 100644 index 9a0f0164fc5..00000000000 --- a/app/src/organisms/Devices/RobotSettings/RobotSettingsPrivacy.tsx +++ /dev/null @@ -1,71 +0,0 @@ -import * as React from 'react' -import { useSelector, useDispatch } from 'react-redux' -import { useTranslation } from 'react-i18next' - -import { getRobotSettings, fetchSettings } from '../../../redux/robot-settings' - -import type { State, Dispatch } from '../../../redux/types' -import type { - RobotSettings, - RobotSettingsField, -} from '../../../redux/robot-settings/types' -import { SettingToggle } from './SettingToggle' - -interface RobotSettingsPrivacyProps { - robotName: string -} - -const PRIVACY_SETTINGS = ['disableLogAggregation'] - -const INFO_BY_SETTING_ID: { - [id: string]: { - titleKey: string - descriptionKey: string - invert: boolean - } -} = { - disableLogAggregation: { - titleKey: 'branded:share_logs_with_opentrons', - descriptionKey: 'branded:share_logs_with_opentrons_description', - invert: true, - }, -} - -export function RobotSettingsPrivacy({ - robotName, -}: RobotSettingsPrivacyProps): JSX.Element { - const { t } = useTranslation(['device_settings', 'branded']) - const settings = useSelector((state: State) => - getRobotSettings(state, robotName) - ) - const privacySettings = settings.filter(({ id }) => - PRIVACY_SETTINGS.includes(id) - ) - const translatedPrivacySettings: Array< - RobotSettingsField & { invert: boolean } - > = privacySettings.map(s => { - const { titleKey, descriptionKey, invert } = INFO_BY_SETTING_ID[s.id] - return s.id in INFO_BY_SETTING_ID - ? { - ...s, - title: t(titleKey), - description: t(descriptionKey), - invert, - } - : { ...s, invert: false } - }) - - const dispatch = useDispatch() - - React.useEffect(() => { - dispatch(fetchSettings(robotName)) - }, [dispatch, robotName]) - - return ( - <> - {translatedPrivacySettings.map(field => ( - - ))} - - ) -} diff --git a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx index b2b922be1b0..1c1e6d8a2ea 100644 --- a/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx +++ b/app/src/organisms/Devices/RobotSettings/__tests__/RobotSettingsFeatureFlags.test.tsx @@ -37,7 +37,6 @@ describe('RobotSettings Advanced tab', () => { 'deckCalibrationDots', 'shortFixedTrash', 'useOldAspirationFunctions', - 'disableLogAggregation', 'disableFastProtocolUpload', ].map(id => ({ id, diff --git a/app/src/organisms/RobotSettingsDashboard/Privacy.tsx b/app/src/organisms/RobotSettingsDashboard/Privacy.tsx index d8b30cdf8ec..51a096af3a3 100644 --- a/app/src/organisms/RobotSettingsDashboard/Privacy.tsx +++ b/app/src/organisms/RobotSettingsDashboard/Privacy.tsx @@ -17,13 +17,10 @@ import { getAnalyticsOptedIn, toggleAnalyticsOptedIn, } from '../../redux/analytics' -import { getRobotSettings, updateSetting } from '../../redux/robot-settings' -import type { Dispatch, State } from '../../redux/types' +import type { Dispatch } from '../../redux/types' import type { SetSettingOption } from '../../pages/RobotSettingsDashboard' -const ROBOT_ANALYTICS_SETTING_ID = 'disableLogAggregation' - interface PrivacyProps { robotName: string setCurrentOption: SetSettingOption @@ -36,16 +33,8 @@ export function Privacy({ const { t } = useTranslation(['app_settings', 'branded']) const dispatch = useDispatch() - const allRobotSettings = useSelector((state: State) => - getRobotSettings(state, robotName) - ) - const appAnalyticsOptedIn = useSelector(getAnalyticsOptedIn) - const isRobotAnalyticsDisabled = - allRobotSettings.find(({ id }) => id === ROBOT_ANALYTICS_SETTING_ID) - ?.value ?? false - return ( - } - onClick={() => - dispatch( - updateSetting( - robotName, - ROBOT_ANALYTICS_SETTING_ID, - !isRobotAnalyticsDisabled - ) - ) - } - /> { screen.getByText( 'Opentrons cares about your privacy. We anonymize all data and only use it to improve our products.' ) - screen.getByText('Share robot logs') - screen.getByText('Data on actions the robot does, like running protocols.') screen.getByText('Share display usage') screen.getByText('Data on how you interact with the touchscreen on Flex.') }) @@ -49,14 +47,4 @@ describe('Privacy', () => { fireEvent.click(screen.getByText('Share display usage')) expect(vi.mocked(toggleAnalyticsOptedIn)).toBeCalled() }) - - it('should toggle robot logs sharing on click', () => { - render(props) - fireEvent.click(screen.getByText('Share robot logs')) - expect(vi.mocked(updateSetting)).toBeCalledWith( - 'Otie', - 'disableLogAggregation', - true - ) - }) }) diff --git a/app/src/pages/Devices/RobotSettings/__tests__/RobotSettings.test.tsx b/app/src/pages/Devices/RobotSettings/__tests__/RobotSettings.test.tsx index 23b55dc1c84..6961f74ad22 100644 --- a/app/src/pages/Devices/RobotSettings/__tests__/RobotSettings.test.tsx +++ b/app/src/pages/Devices/RobotSettings/__tests__/RobotSettings.test.tsx @@ -8,7 +8,6 @@ import { i18n } from '../../../../i18n' import { RobotSettingsCalibration } from '../../../../organisms/RobotSettingsCalibration' import { RobotSettingsNetworking } from '../../../../organisms/Devices/RobotSettings/RobotSettingsNetworking' import { RobotSettingsAdvanced } from '../../../../organisms/Devices/RobotSettings/RobotSettingsAdvanced' -import { RobotSettingsPrivacy } from '../../../../organisms/Devices/RobotSettings/RobotSettingsPrivacy' import { useRobot } from '../../../../organisms/Devices/hooks' import { RobotSettings } from '..' import { when } from 'vitest-when' @@ -22,7 +21,6 @@ import { getRobotUpdateSession } from '../../../../redux/robot-update' vi.mock('../../../../organisms/RobotSettingsCalibration') vi.mock('../../../../organisms/Devices/RobotSettings/RobotSettingsNetworking') vi.mock('../../../../organisms/Devices/RobotSettings/RobotSettingsAdvanced') -vi.mock('../../../../organisms/Devices/RobotSettings/RobotSettingsPrivacy') vi.mock('../../../../organisms/Devices/hooks') vi.mock('../../../../redux/discovery/selectors') vi.mock('../../../../redux/robot-update') @@ -61,9 +59,6 @@ describe('RobotSettings', () => { vi.mocked(RobotSettingsAdvanced).mockReturnValue(
Mock RobotSettingsAdvanced
) - vi.mocked(RobotSettingsPrivacy).mockReturnValue( -
Mock RobotSettingsPrivacy
- ) }) afterEach(() => { vi.resetAllMocks() @@ -160,13 +155,4 @@ describe('RobotSettings', () => { fireEvent.click(AdvancedTab) screen.getByText('Mock RobotSettingsAdvanced') }) - - it('renders privacy content when the privacy tab is clicked', () => { - render('/devices/otie/robot-settings/calibration') - - const PrivacyTab = screen.getByText('Privacy') - expect(screen.queryByText('Mock RobotSettingsPrivacy')).toBeFalsy() - fireEvent.click(PrivacyTab) - screen.getByText('Mock RobotSettingsPrivacy') - }) }) diff --git a/app/src/pages/Devices/RobotSettings/index.tsx b/app/src/pages/Devices/RobotSettings/index.tsx index b3b7580377b..fd2e089a7d4 100644 --- a/app/src/pages/Devices/RobotSettings/index.tsx +++ b/app/src/pages/Devices/RobotSettings/index.tsx @@ -33,7 +33,6 @@ import { RobotSettingsCalibration } from '../../../organisms/RobotSettingsCalibr import { RobotSettingsAdvanced } from '../../../organisms/Devices/RobotSettings/RobotSettingsAdvanced' import { RobotSettingsNetworking } from '../../../organisms/Devices/RobotSettings/RobotSettingsNetworking' import { RobotSettingsFeatureFlags } from '../../../organisms/Devices/RobotSettings/RobotSettingsFeatureFlags' -import { RobotSettingsPrivacy } from '../../../organisms/Devices/RobotSettings/RobotSettingsPrivacy' import { ReachableBanner } from '../../../organisms/Devices/ReachableBanner' import type { DesktopRouteParams, RobotSettingsTab } from '../../../App/types' @@ -45,7 +44,6 @@ export function RobotSettings(): JSX.Element | null { >() as DesktopRouteParams const robot = useRobot(robotName) const isCalibrationDisabled = robot?.status !== CONNECTABLE - const isPrivacyDisabled = robot?.status === UNREACHABLE const isNetworkingDisabled = robot?.status === UNREACHABLE const [showRobotBusyBanner, setShowRobotBusyBanner] = React.useState( false @@ -78,7 +76,6 @@ export function RobotSettings(): JSX.Element | null { /> ), 'feature-flags': , - privacy: , } const devToolsOn = useSelector(getDevtoolsEnabled) @@ -95,8 +92,7 @@ export function RobotSettings(): JSX.Element | null { robotSettingsTab === 'calibration' && isCalibrationDisabled const cannotViewFeatureFlags = robotSettingsTab === 'feature-flags' && !devToolsOn - const cannotViewPrivacy = robotSettingsTab === 'privacy' && isPrivacyDisabled - if (cannotViewCalibration || cannotViewFeatureFlags || cannotViewPrivacy) { + if (cannotViewCalibration || cannotViewFeatureFlags) { return } @@ -146,11 +142,6 @@ export function RobotSettings(): JSX.Element | null { tabName={t('networking')} disabled={isNetworkingDisabled} /> - Date: Tue, 23 Jul 2024 14:23:29 -0400 Subject: [PATCH 20/50] feat(robot_server): add a GET endpoint to get all data files (#15744) re AUTH-569 # Overview Adding a GET endpoint to access a list of all data file info on the robot server. # Test Plan # Changelog -Added a get_all_data_files function in the dataFiles/router.py -Updated test # Review requests # Risk assessment --------- Co-authored-by: shiyaochen --- .../data_files/data_files_store.py | 5 ++- .../robot_server/data_files/router.py | 35 +++++++++++++++ robot-server/tests/data_files/test_router.py | 43 +++++++++++++++++++ 3 files changed, 81 insertions(+), 2 deletions(-) diff --git a/robot-server/robot_server/data_files/data_files_store.py b/robot-server/robot_server/data_files/data_files_store.py index fb84d15634e..4427f8efb91 100644 --- a/robot-server/robot_server/data_files/data_files_store.py +++ b/robot-server/robot_server/data_files/data_files_store.py @@ -35,7 +35,7 @@ def __init__( def get_file_info_by_hash(self, file_hash: str) -> Optional[DataFileInfo]: """Get the ID of data file having the provided hash.""" - for file in self._sql_get_all_from_engine(): + for file in self.sql_get_all_from_engine(): if file.file_hash == file_hash: return file return None @@ -65,7 +65,8 @@ def get(self, data_file_id: str) -> DataFileInfo: return _convert_row_data_file_info(data_file_row) - def _sql_get_all_from_engine(self) -> List[DataFileInfo]: + def sql_get_all_from_engine(self) -> List[DataFileInfo]: + """Get all data file entries from the database.""" statement = sqlalchemy.select(data_files_table).order_by(sqlite_rowid) with self._sql_engine.begin() as transaction: all_rows = transaction.execute(statement).all() diff --git a/robot-server/robot_server/data_files/router.py b/robot-server/robot_server/data_files/router.py index e997059230f..afd63487b16 100644 --- a/robot-server/robot_server/data_files/router.py +++ b/robot-server/robot_server/data_files/router.py @@ -9,7 +9,9 @@ from robot_server.service.json_api import ( SimpleBody, + SimpleMultiBody, PydanticResponse, + MultiBodyMeta, ) from robot_server.errors.error_responses import ErrorDetails, ErrorBody from .dependencies import get_data_files_directory, get_data_files_store @@ -218,3 +220,36 @@ async def get_data_file( content=buffered_file.contents.decode("utf-8"), media_type="text/plain", ) + + +@PydanticResponse.wrap_route( + datafiles_router.get, + path="/dataFiles", + summary="Get a list of all data files stored on the robot server", + responses={status.HTTP_200_OK: {"model": SimpleMultiBody[str]}}, +) +async def get_all_data_files( + data_files_store: DataFilesStore = Depends(get_data_files_store), +) -> PydanticResponse[SimpleMultiBody[DataFile]]: + """Get a list of all data files stored on the robot server. + + Args: + data_files_store: In-memory database of data file resources. + """ + data_files = data_files_store.sql_get_all_from_engine() + + meta = MultiBodyMeta(cursor=0, totalLength=len(data_files)) + + return await PydanticResponse.create( + content=SimpleMultiBody.construct( + data=[ + DataFile.construct( + id=data_file_info.id, + name=data_file_info.name, + createdAt=data_file_info.created_at, + ) + for data_file_info in data_files + ], + meta=meta, + ), + ) diff --git a/robot-server/tests/data_files/test_router.py b/robot-server/tests/data_files/test_router.py index 2594ec40329..0bb9f4c7bee 100644 --- a/robot-server/tests/data_files/test_router.py +++ b/robot-server/tests/data_files/test_router.py @@ -8,12 +8,15 @@ from fastapi import UploadFile from opentrons.protocol_reader import FileHasher, FileReaderWriter, BufferedFile +from robot_server.service.json_api import MultiBodyMeta + from robot_server.data_files.data_files_store import DataFilesStore, DataFileInfo from robot_server.data_files.models import DataFile, FileIdNotFoundError from robot_server.data_files.router import ( upload_data_file, get_data_file_info_by_id, get_data_file, + get_all_data_files, ) from robot_server.errors.error_responses import ApiError @@ -322,3 +325,43 @@ async def test_get_data_file( assert result.status_code == 200 assert result.body == b"some_content" assert result.media_type == "text/plain" + + +async def test_get_all_data_file_info( + decoy: Decoy, + data_files_store: DataFilesStore, +) -> None: + """Get a list of all data file info from the database.""" + decoy.when(data_files_store.sql_get_all_from_engine()).then_return( + [ + DataFileInfo( + id="qwerty", + name="abc.xyz", + file_hash="123", + created_at=datetime(year=2024, month=7, day=15), + ), + DataFileInfo( + id="hfhcjdeowjfie", + name="mcd.kfc", + file_hash="124", + created_at=datetime(year=2024, month=7, day=22), + ), + ] + ) + + result = await get_all_data_files(data_files_store=data_files_store) + + assert result.status_code == 200 + assert result.content.data == [ + DataFile( + id="qwerty", + name="abc.xyz", + createdAt=datetime(year=2024, month=7, day=15), + ), + DataFile( + id="hfhcjdeowjfie", + name="mcd.kfc", + createdAt=datetime(year=2024, month=7, day=22), + ), + ] + assert result.content.meta == MultiBodyMeta(cursor=0, totalLength=2) From ea31e34e0076341690e71a41167e98d32ea3d7ef Mon Sep 17 00:00:00 2001 From: aaron-kulkarni <107003644+aaron-kulkarni@users.noreply.github.com> Date: Tue, 23 Jul 2024 14:32:39 -0400 Subject: [PATCH 21/50] fix(api): throw error if no valid nozzle has tip (#15721) Provide a check in all LLD functions(called manually or automatically during aspirate) that makes sure that if it's a 96-channel pipette, that at least one of the nozzles with a pressure sensor has a tip on it. # Overview # Test Plan # Changelog # Review requests # Risk assessment --- .../protocol_api/instrument_context.py | 19 ++++++ .../protocol_api/test_instrument_context.py | 59 +++++++++++++++++++ 2 files changed, 78 insertions(+) diff --git a/api/src/opentrons/protocol_api/instrument_context.py b/api/src/opentrons/protocol_api/instrument_context.py index 05a8ecdc80c..c39a4aba2ac 100644 --- a/api/src/opentrons/protocol_api/instrument_context.py +++ b/api/src/opentrons/protocol_api/instrument_context.py @@ -2,6 +2,7 @@ import logging from contextlib import ExitStack from typing import Any, List, Optional, Sequence, Union, cast, Dict +from opentrons.protocol_engine.errors.exceptions import TipNotAttachedError from opentrons_shared_data.errors.exceptions import ( CommandPreconditionViolated, CommandParameterLimitViolated, @@ -264,6 +265,7 @@ def aspirate( self.api_version >= APIVersion(2, 20) and well is not None and self.liquid_presence_detection + and self._96_tip_config_valid() ): self.require_liquid_presence(well=well) self.prepare_to_aspirate() @@ -1874,6 +1876,19 @@ def _get_last_location_by_api_version(self) -> Optional[types.Location]: else: return self._protocol_core.get_last_location() + def _96_tip_config_valid(self) -> bool: + n_map = self._core.get_nozzle_map() + channels = self._core.get_active_channels() + if channels == 96: + if ( + n_map.back_left != n_map.full_instrument_back_left + and n_map.front_right != n_map.full_instrument_front_right + ): + raise TipNotAttachedError( + "Either the front right or the back left nozzle must have a tip attached to do LLD." + ) + return True + def __repr__(self) -> str: return "<{}: {} in {}>".format( self.__class__.__name__, @@ -2110,6 +2125,7 @@ def detect_liquid_presence(self, well: labware.Well) -> bool: :returns: A boolean. """ loc = well.top() + self._96_tip_config_valid() return self._core.detect_liquid_presence(well._core, loc) @requires_version(2, 20) @@ -2119,6 +2135,7 @@ def require_liquid_presence(self, well: labware.Well) -> None: :returns: None. """ loc = well.top() + self._96_tip_config_valid() self._core.liquid_probe_with_recovery(well._core, loc) @requires_version(2, 20) @@ -2131,6 +2148,8 @@ def measure_liquid_height(self, well: labware.Well) -> float: This is intended for Opentrons internal use only and is not a guaranteed API. """ + loc = well.top() + self._96_tip_config_valid() height = self._core.liquid_probe_without_recovery(well._core, loc) return height diff --git a/api/tests/opentrons/protocol_api/test_instrument_context.py b/api/tests/opentrons/protocol_api/test_instrument_context.py index d98b99a9a6d..0e85082c3e2 100644 --- a/api/tests/opentrons/protocol_api/test_instrument_context.py +++ b/api/tests/opentrons/protocol_api/test_instrument_context.py @@ -17,6 +17,12 @@ from opentrons.legacy_broker import LegacyBroker +from opentrons.protocol_engine.errors.exceptions import TipNotAttachedError +from tests.opentrons.protocol_engine.pipette_fixtures import ( + NINETY_SIX_COLS, + NINETY_SIX_MAP, + NINETY_SIX_ROWS, +) from opentrons.protocols.api_support import instrument as mock_instrument_support from opentrons.protocols.api_support.types import APIVersion from opentrons.protocols.api_support.util import ( @@ -1351,3 +1357,56 @@ def test_measure_liquid_height( with pytest.raises(ProtocolCommandFailedError) as pcfe: subject.measure_liquid_height(mock_well) assert pcfe.value is errorToRaise + + +def test_96_tip_config_valid( + decoy: Decoy, mock_instrument_core: InstrumentCore, subject: InstrumentContext +) -> None: + """It should error when there's no tips on the correct corner nozzles.""" + nozzle_map = NozzleMap.build( + physical_nozzles=NINETY_SIX_MAP, + physical_rows=NINETY_SIX_ROWS, + physical_columns=NINETY_SIX_COLS, + starting_nozzle="A5", + back_left_nozzle="A5", + front_right_nozzle="H5", + valid_nozzle_maps=ValidNozzleMaps(maps={"Column12": NINETY_SIX_COLS["5"]}), + ) + decoy.when(mock_instrument_core.get_nozzle_map()).then_return(nozzle_map) + decoy.when(mock_instrument_core.get_active_channels()).then_return(96) + with pytest.raises(TipNotAttachedError): + subject._96_tip_config_valid() + + +def test_96_tip_config_invalid( + decoy: Decoy, mock_instrument_core: InstrumentCore, subject: InstrumentContext +) -> None: + """It should return True when there are tips on the correct corner nozzles.""" + nozzle_map = NozzleMap.build( + physical_nozzles=NINETY_SIX_MAP, + physical_rows=NINETY_SIX_ROWS, + physical_columns=NINETY_SIX_COLS, + starting_nozzle="A1", + back_left_nozzle="A1", + front_right_nozzle="H12", + valid_nozzle_maps=ValidNozzleMaps( + maps={ + "Full": sum( + [ + NINETY_SIX_ROWS["A"], + NINETY_SIX_ROWS["B"], + NINETY_SIX_ROWS["C"], + NINETY_SIX_ROWS["D"], + NINETY_SIX_ROWS["E"], + NINETY_SIX_ROWS["F"], + NINETY_SIX_ROWS["G"], + NINETY_SIX_ROWS["H"], + ], + [], + ) + } + ), + ) + decoy.when(mock_instrument_core.get_nozzle_map()).then_return(nozzle_map) + decoy.when(mock_instrument_core.get_active_channels()).then_return(96) + assert subject._96_tip_config_valid() is True From 3b26b137fd527af0205168b4228ddc2994c8a320 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Tue, 23 Jul 2024 14:50:40 -0400 Subject: [PATCH 22/50] fix(analyses-snapshot-testing): PLAT-337-update-force-to-duty-cycle-for-new-gripper-motor-in-shared-data snapshot failure capture (#15653) 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. --------- Co-authored-by: ahiuchingau <20424172+ahiuchingau@users.noreply.github.com> From 34d35f53206ac939373fde8405c7090f951aeb9f Mon Sep 17 00:00:00 2001 From: TamarZanzouri Date: Tue, 23 Jul 2024 15:22:21 -0400 Subject: [PATCH 23/50] chore(app): wire up NoLiquidDetected ER on desktop (#15765) # Overview closes https://opentrons.atlassian.net/browse/EXEC-556. Wire up desktop designs for NoLiquidDetected ER flow. # Test Plan - Make sure designs match desktop app and logic works as expected. - Make sure designs match ODD app and logic works as expected. # Risk assessment low. mainly css changes. should affect desktop app only. --- .../RecoveryOptions/IgnoreErrorSkipStep.tsx | 65 +++++++++++++++++-- .../RecoveryOptions/ManageTips.tsx | 3 - .../RecoveryOptions/SelectRecoveryOption.tsx | 5 ++ .../__tests__/IgnoreErrorSkipStep.test.tsx | 4 +- .../TwoColTextAndFailedStepNextStep.tsx | 4 +- 5 files changed, 68 insertions(+), 13 deletions(-) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/IgnoreErrorSkipStep.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/IgnoreErrorSkipStep.tsx index c5ecf84a61b..b2c6e167f6f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/IgnoreErrorSkipStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/IgnoreErrorSkipStep.tsx @@ -1,19 +1,27 @@ import * as React from 'react' import head from 'lodash/head' +import { css } from 'styled-components' import { useTranslation } from 'react-i18next' import { DIRECTION_COLUMN, Flex, SPACING, - LegacyStyledText, + StyledText, + RESPONSIVENESS, } from '@opentrons/components' -import { ODD_SECTION_TITLE_STYLE, RECOVERY_MAP } from '../constants' +import { + ODD_SECTION_TITLE_STYLE, + RECOVERY_MAP, + ODD_ONLY, + DESKTOP_ONLY, +} from '../constants' import { SelectRecoveryOption } from './SelectRecoveryOption' import { RecoveryFooterButtons, RecoverySingleColumnContentWrapper, + RecoveryRadioGroup, } from '../shared' import { RadioButton } from '../../../atoms/buttons' @@ -81,17 +89,50 @@ export function IgnoreErrorStepHome({ } return ( - - + + {t('ignore_similar_errors_later_in_run')} - - + + + + ) => { + setSelectedOption(e.currentTarget.value as IgnoreOption) + }} + options={IGNORE_OPTIONS_IN_ORDER.map(option => { + return { + value: option, + children: ( + + {t(option)} + + ), + } + })} + /> + ) => { setSelected(e.currentTarget.value as RemovalOptions) diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx index a33f2fb7abc..ed9de710db8 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/SelectRecoveryOption.tsx @@ -148,6 +148,7 @@ export function DesktopRecoveryOptions({ }: RecoveryOptionsProps): JSX.Element { return ( { setSelectedRoute(e.currentTarget.value) }} @@ -221,3 +222,7 @@ export const GENERAL_ERROR_OPTIONS: RecoveryRoute[] = [ RECOVERY_MAP.RETRY_FAILED_COMMAND.ROUTE, RECOVERY_MAP.CANCEL_RUN.ROUTE, ] + +const RADIO_GAP = ` + gap: ${SPACING.spacing4}; +` diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx index d6241b7dcd9..466c9f65dc7 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/IgnoreErrorSkipStep.test.tsx @@ -105,7 +105,7 @@ describe('IgnoreErrorStepHome', () => { it('calls ignoreOnce when "ignore_only_this_error" is selected and primary button is clicked', async () => { renderIgnoreErrorStepHome(props) - fireEvent.click(screen.getByText('Ignore only this error')) + fireEvent.click(screen.queryAllByText('Ignore only this error')[0]) clickButtonLabeled('Continue') await waitFor(() => { expect(mockProceedToRouteAndStep).toHaveBeenCalledWith( @@ -117,7 +117,7 @@ describe('IgnoreErrorStepHome', () => { it('calls ignoreAlways when "ignore_all_errors_of_this_type" is selected and primary button is clicked', async () => { renderIgnoreErrorStepHome(props) - fireEvent.click(screen.getByText('Ignore all errors of this type')) + fireEvent.click(screen.queryAllByText('Ignore all errors of this type')[0]) clickButtonLabeled('Continue') await waitFor(() => { expect(mockIgnoreErrorKindThisRun).toHaveBeenCalled() diff --git a/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx b/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx index 4ed62e8ff8d..a1f959c72fb 100644 --- a/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/shared/TwoColTextAndFailedStepNextStep.tsx @@ -46,9 +46,9 @@ export function TwoColTextAndFailedStepNextStep( From cad42e0b0018deb0795797658cdd31d9f94af2cd Mon Sep 17 00:00:00 2001 From: Max Marrone Date: Tue, 23 Jul 2024 15:29:53 -0400 Subject: [PATCH 24/50] chore(ci): Add user input to enable/disable opening an analyses snapshot PR (#15764) --- .github/workflows/analyses-snapshot-test.yaml | 107 ++++++++++-------- 1 file changed, 58 insertions(+), 49 deletions(-) diff --git a/.github/workflows/analyses-snapshot-test.yaml b/.github/workflows/analyses-snapshot-test.yaml index bf1451932a9..86f6ccab31c 100644 --- a/.github/workflows/analyses-snapshot-test.yaml +++ b/.github/workflows/analyses-snapshot-test.yaml @@ -11,8 +11,13 @@ on: description: 'Branch or tag that provides the snapshot and test code at test runtime' required: true default: 'edge' + OPEN_PR_ON_FAILURE: + description: 'If the test fails, open a PR to update the snapshots' + type: boolean + required: true + default: false schedule: - - cron: '26 7 * * *' # 7:26 AM UTC + - cron: '26 7 * * *' # 7:26 AM UTC pull_request: paths: - 'api/**' @@ -32,66 +37,70 @@ jobs: env: ANALYSIS_REF: ${{ github.event.inputs.ANALYSIS_REF || github.head_ref || 'edge' }} SNAPSHOT_REF: ${{ github.event.inputs.SNAPSHOT_REF || github.head_ref || 'edge' }} + # 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') }} steps: - - name: Checkout Repository - uses: actions/checkout@v4 - with: - ref: ${{ env.SNAPSHOT_REF }} + - name: Checkout Repository + uses: actions/checkout@v4 + with: + ref: ${{ env.SNAPSHOT_REF }} - - name: Docker Build - working-directory: analyses-snapshot-testing - run: make build-opentrons-analysis + - name: Docker Build + working-directory: analyses-snapshot-testing + run: make build-opentrons-analysis - - name: Set up Python 3.12 - uses: actions/setup-python@v5 - with: - python-version: '3.12' - cache: 'pipenv' - cache-dependency-path: analyses-snapshot-testing/Pipfile.lock + - name: Set up Python 3.12 + uses: actions/setup-python@v5 + with: + python-version: '3.12' + cache: 'pipenv' + cache-dependency-path: analyses-snapshot-testing/Pipfile.lock - - name: Setup Python Dependencies - working-directory: analyses-snapshot-testing - run: make setup + - name: Setup Python Dependencies + working-directory: analyses-snapshot-testing + run: make setup - - name: Run Test - id: run_test - working-directory: analyses-snapshot-testing - run: make snapshot-test + - name: Run Test + id: run_test + working-directory: analyses-snapshot-testing + run: make snapshot-test - - name: Upload Report - if: '!cancelled()' - uses: actions/upload-artifact@v4 - with: - name: test-report - path: analyses-snapshot-testing/results/ + - name: Upload Report + if: '!cancelled()' + uses: actions/upload-artifact@v4 + with: + name: test-report + path: analyses-snapshot-testing/results/ - - name: Handle Test Failure - id: handle_failure - if: always() && steps.run_test.outcome == 'failure' - working-directory: analyses-snapshot-testing - run: make snapshot-test-update + - name: Handle Test Failure + id: handle_failure + if: always() && steps.run_test.outcome == 'failure' + 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' - uses: peter-evans/create-pull-request@v6 - with: + - name: Create Snapshot update Request + id: create_pull_request + if: always() && steps.handle_failure.outcome == 'success' && env.OPEN_PR_ON_FAILURE + 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.' branch: 'analyses-snapshot-testing/${{ env.ANALYSIS_REF }}-from-${{ env.SNAPSHOT_REF}}' base: ${{ env.SNAPSHOT_REF}} - - name: Comment on 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 }}'; - github.rest.issues.createComment({ - owner: context.repo.owner, - repo: context.repo.repo, - issue_number: context.issue.number, - body: message - }); + - name: Comment on 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 }}'; + github.rest.issues.createComment({ + owner: context.repo.owner, + repo: context.repo.repo, + issue_number: context.issue.number, + body: message + }); From e0e018fc604acf3a75383cb09c5632ba6bf5dde5 Mon Sep 17 00:00:00 2001 From: Ryan Howard Date: Tue, 23 Jul 2024 15:55:38 -0400 Subject: [PATCH 25/50] fix(hardware): didn't mean to take this out (#15770) # Overview We had removed this as part of another PR aimed at reducing magic numbers and put it in ot3api, but then that caused a circular dependency problem and we reverted. we just missed this line when we reverted # Test Plan # Changelog # Review requests # Risk assessment --- .../opentrons_hardware/hardware_control/tool_sensors.py | 2 ++ scripts/push.mk | 6 +++--- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/hardware/opentrons_hardware/hardware_control/tool_sensors.py b/hardware/opentrons_hardware/hardware_control/tool_sensors.py index eeb4736a6d9..c1b2f1650f5 100644 --- a/hardware/opentrons_hardware/hardware_control/tool_sensors.py +++ b/hardware/opentrons_hardware/hardware_control/tool_sensors.py @@ -82,6 +82,8 @@ # FIXME we should organize all of these functions to use the sensor drivers. # FIXME we should restrict some of these functions by instrument type. +PLUNGER_SOLO_MOVE_TIME = 0.2 + def _fix_pass_step_for_buffer( move_group: MoveGroupStep, diff --git a/scripts/push.mk b/scripts/push.mk index a6d191a229a..0f693f9cb53 100644 --- a/scripts/push.mk +++ b/scripts/push.mk @@ -1,7 +1,7 @@ # utilities for pushing things to robots in a reusable fashion find_robot=$(shell yarn run -s discovery find -i 169.254) -default_ssh_key := ~/.ssh/robot_key +default_ssh_key := ~/.ssh/id_rsa default_ssh_opts := -o stricthostkeychecking=no -o userknownhostsfile=/dev/null version_dict=$(shell ssh $(call id-file-arg,$(2)) $(3) root@$(1) cat /etc/VERSION.json) is-ot3=$(findstring OT-3, $(version_dict)) @@ -19,7 +19,7 @@ is-in-version=$(findstring $(firstword $(checked-ssh-version)),$(allowed-ssh-ver # when using an OpenSSH version larger than 8.9, # we need to add a flag to use legacy scp with SFTP protocol scp-legacy-option-flag = $(if $(is-in-version),,-O) -# when using windows, make is running against a different openSSH than the OS. +# when using windows, make is running against a different openSSH than the OS. # adding the -O flag to scp will fail if the openSSH on OS is less than 9. # if openSSH on OS is 9 or more please add the -O flag to scp. PLATFORM := $(shell uname -s) @@ -80,7 +80,7 @@ ssh $(call id-file-arg,$(2)) $(3) root@$(1) \ endef # push-systemd-unit: move a systemd unit file to the robot -# +# # argument 1 is the host to push to # argument 2 is the identity file to use, if any # argument 3 is any further ssh options, quoted From 603dc2f30b94825dcf438c632df847aff5ee8c5f Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 24 Jul 2024 11:57:32 -0400 Subject: [PATCH 26/50] refactor(app): enable recovery errors for failed pickUpTip retry commands (#15779) --- .../hooks/__tests__/useRecoveryCommands.test.ts | 2 +- .../organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts | 3 +-- 2 files changed, 2 insertions(+), 3 deletions(-) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts index d75387a99d4..1a54da00103 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts @@ -217,7 +217,7 @@ describe('useRecoveryCommands', () => { expect(mockChainRunCommands).toHaveBeenCalledWith( [buildPickUpTipsCmd], - true + false ) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts index 3ef8f5b3809..58c398b33f6 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -84,7 +84,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,7 +96,7 @@ 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]) From e4ff49aea5755afb3439592085a50c09be3cdffa Mon Sep 17 00:00:00 2001 From: aaron-kulkarni <107003644+aaron-kulkarni@users.noreply.github.com> Date: Wed, 24 Jul 2024 13:12:25 -0400 Subject: [PATCH 27/50] feat(robot-server): start logic for dynamic error recovery policy (#15707) # Overview Create robot-server function that turns a List of ErrorRecoveryRules into a full ErrorRecoveryPolicy. A future PR will implement the HTTP calls that actually create the list of rules. EXEC-589 # Test Plan # Changelog # Review requests # Risk assessment --- .../protocol_engine/error_recovery_policy.py | 6 +- .../protocol_engine/state/commands.py | 5 +- .../runs/error_recovery_mapping.py | 46 +++++++ .../runs/error_recovery_models.py | 69 ++++++++++ robot-server/robot_server/service/errors.py | 1 + .../tests/runs/test_error_recovery_mapping.py | 118 ++++++++++++++++++ 6 files changed, 240 insertions(+), 5 deletions(-) create mode 100644 robot-server/robot_server/runs/error_recovery_mapping.py create mode 100644 robot-server/robot_server/runs/error_recovery_models.py create mode 100644 robot-server/tests/runs/test_error_recovery_mapping.py diff --git a/api/src/opentrons/protocol_engine/error_recovery_policy.py b/api/src/opentrons/protocol_engine/error_recovery_policy.py index f7468961131..2623a696c4d 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): diff --git a/api/src/opentrons/protocol_engine/state/commands.py b/api/src/opentrons/protocol_engine/state/commands.py index a558210cbff..53a11a66ba0 100644 --- a/api/src/opentrons/protocol_engine/state/commands.py +++ b/api/src/opentrons/protocol_engine/state/commands.py @@ -337,7 +337,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) 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..dfef53ad05f --- /dev/null +++ b/robot-server/robot_server/runs/error_recovery_mapping.py @@ -0,0 +1,46 @@ +"""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, + standard_run_policy, +) + + +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: + for i, criteria in enumerate(rule.matchCriteria): + command_type_matches = ( + failed_command.commandType == criteria.command.commandType + ) + error_type_matches = ( + defined_error_data is not None + and defined_error_data.public.errorType + == criteria.command.error.errorType + ) + if command_type_matches and error_type_matches: + if rule.ifMatch[i] == ReactionIfMatch.IGNORE_AND_CONTINUE: + raise NotImplementedError # No protocol engine support for this yet. It's in EXEC-302. + elif rule.ifMatch[i] == ReactionIfMatch.FAIL_RUN: + return ErrorRecoveryType.FAIL_RUN + elif rule.ifMatch[i] == ReactionIfMatch.WAIT_FOR_RECOVERY: + return ErrorRecoveryType.WAIT_FOR_RECOVERY + + return standard_run_policy(config, failed_command, defined_error_data) + + return _policy 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..95a5a1e5631 --- /dev/null +++ b/robot-server/robot_server/runs/error_recovery_models.py @@ -0,0 +1,69 @@ +"""Request and response models for dealing with error recovery policies.""" +from enum import Enum +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): + """Request/Response model for new error recovery rule creation.""" + + matchCriteria: list[MatchCriteria] = Field( + default_factory=list, + description="The criteria that must be met for this rule to be applied.", + ) + ifMatch: list[ReactionIfMatch] = Field( + default_factory=list, + description="The specific recovery setting that will be in use if the type parameters match.", + ) 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/tests/runs/test_error_recovery_mapping.py b/robot-server/tests/runs/test_error_recovery_mapping.py new file mode 100644 index 00000000000..4d01ad50085 --- /dev/null +++ b/robot-server/tests/runs/test_error_recovery_mapping.py @@ -0,0 +1,118 @@ +"""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, + ) + with pytest.raises(NotImplementedError): + policy(exampleConfig, mock_command, mock_error_data) + + +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.""" + rule1 = ErrorRecoveryRule(matchCriteria=[], ifMatch=[]) + + policy = create_error_recovery_policy_from_rules([rule1]) + 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.""" + rule1 = ErrorRecoveryRule(matchCriteria=[], ifMatch=[]) + + policy = create_error_recovery_policy_from_rules([rule1]) + exampleConfig = Config( + robot_type="OT-3 Standard", + deck_type=DeckType.OT3_STANDARD, + ) + + assert ( + policy(exampleConfig, mock_command, mock_error_data) + == ErrorRecoveryType.WAIT_FOR_RECOVERY + ) From 75e2ee68cb5cfab3eea5c07c47fd367e5f03d6f9 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 24 Jul 2024 13:55:30 -0400 Subject: [PATCH 28/50] feat(app): add error recovery "error while dispensing" flows to desktop (#15774) Closes EXEC-559 --- .../localization/en/error_recovery.json | 2 +- .../DropTipWizardFlows/DropTipWizard.tsx | 4 +++ app/src/organisms/DropTipWizardFlows/types.ts | 1 + .../RecoveryOptions/ManageTips.tsx | 27 ++++++++++++++++- .../__tests__/ManageTips.test.tsx | 25 +++++++++++++++- .../ErrorRecoveryFlows/shared/SelectTips.tsx | 29 +++++++++++++++---- 6 files changed, 80 insertions(+), 8 deletions(-) diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index c139f21acd2..8bf00af168d 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -17,7 +17,7 @@ "continue_to_drop_tip": "Continue to drop tip", "error": "Error", "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", 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/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/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx index 87a195a12a8..5e6d6685195 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx @@ -30,7 +30,7 @@ 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. @@ -251,6 +251,7 @@ export function useDropTipFlowUtils({ const { t } = useTranslation('error_recovery') const { RETRY_NEW_TIPS, + SKIP_STEP_WITH_NEW_TIPS, ERROR_WHILE_RECOVERING, DROP_TIP_FLOWS, } = RECOVERY_MAP @@ -263,12 +264,35 @@ export function useDropTipFlowUtils({ const buildTipDropCompleteBtn = (): string => { 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 +327,7 @@ export function useDropTipFlowUtils({ goBackBeforeBeginning: () => { return proceedToRouteAndStep(DROP_TIP_FLOWS.ROUTE) }, + tipDropComplete: buildTipDropCompleteRouting(), } } diff --git a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx index ed58d7e597a..aa8a7063463 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/__tests__/ManageTips.test.tsx @@ -264,11 +264,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/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')} /> - + From 6864eabfc92bdccdaa1657d3912cf21d850f73a2 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 24 Jul 2024 13:55:57 -0400 Subject: [PATCH 29/50] feat(app): Add Error Recovery ErrorDetails desktop support (#15784) Closes EXEC-627 --- .../localization/en/error_recovery.json | 5 +- app/src/molecules/InterventionModal/index.tsx | 2 + .../Devices/ProtocolRun/ProtocolRunHeader.tsx | 1 - .../ErrorRecoveryWizard.tsx | 18 +- .../RecoveryOptions/ManageTips.tsx | 5 +- .../ErrorRecoveryFlows/RunPausedSplash.tsx | 6 +- .../ErrorRecoveryFlows/__fixtures__/index.ts | 1 - .../__tests__/ErrorRecoveryFlows.test.tsx | 1 - .../organisms/ErrorRecoveryFlows/constants.ts | 6 +- .../ErrorRecoveryFlows/hooks/useERUtils.ts | 4 +- .../organisms/ErrorRecoveryFlows/index.tsx | 1 - .../shared/ErrorDetailsModal.tsx | 159 +++++++++++++++--- .../shared/RecoveryInterventionModal.tsx | 3 +- .../ErrorRecoveryFlows/shared/StepInfo.tsx | 23 ++- .../__tests__/ErrorDetailsModal.test.tsx | 59 +++---- .../shared/__tests__/StepInfo.test.tsx | 2 +- app/src/organisms/ErrorRecoveryFlows/types.ts | 2 + app/src/pages/RunningProtocol/index.tsx | 1 - 18 files changed, 212 insertions(+), 87 deletions(-) diff --git a/app/src/assets/localization/en/error_recovery.json b/app/src/assets/localization/en/error_recovery.json index 8bf00af168d..65f1fa50314 100644 --- a/app/src/assets/localization/en/error_recovery.json +++ b/app/src/assets/localization/en/error_recovery.json @@ -16,6 +16,7 @@ "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 with the attached tips.Close the robot door before proceeding.", "failed_step": "Failed step", @@ -40,6 +41,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_tips_from_pipette": "Remove tips from {{mount}} pipette before canceling the run?", "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}}", @@ -72,6 +74,5 @@ "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/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/Devices/ProtocolRun/ProtocolRunHeader.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx index 9b090a67259..f8338f2521b 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx @@ -297,7 +297,6 @@ export function ProtocolRunHeader({ <> {isERActive ? ( ( - + {t('view_error_details')} ) @@ -119,6 +126,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/ManageTips.tsx b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx index 5e6d6685195..9fe5f22c413 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RecoveryOptions/ManageTips.tsx @@ -11,7 +11,6 @@ import { StyledText, RESPONSIVENESS, } from '@opentrons/components' -import { FLEX_ROBOT_TYPE, OT2_ROBOT_TYPE } from '@opentrons/shared-data' import { RadioButton } from '../../../atoms/buttons' import { @@ -187,10 +186,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 +228,7 @@ function DropTipFlowsContainer( return ( {title} { runStatus: RUN_STATUS_AWAITING_RECOVERY, failedCommand: mockFailedCommand, runId: 'MOCK_RUN_ID', - isFlex: true, protocolAnalysis: {} as any, } vi.mocked(ErrorRecoveryWizard).mockReturnValue(
MOCK WIZARD
) diff --git a/app/src/organisms/ErrorRecoveryFlows/constants.ts b/app/src/organisms/ErrorRecoveryFlows/constants.ts index 846f7e2efc0..d61805d1777 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 { SPACING, RESPONSIVENESS } 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}; ` diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts index ff05642ff18..c0d867ea25a 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' @@ -56,7 +57,6 @@ export interface ERUtilsResults { const SUBSEQUENT_COMMAND_DEPTH = 2 // Builds various Error Recovery utilities. export function useERUtils({ - isFlex, failedCommand, runId, toggleERWizard, @@ -96,7 +96,7 @@ export function useERUtils({ const tipStatusUtils = useRecoveryTipStatus({ runId, - isFlex, + isFlex: robotType === FLEX_ROBOT_TYPE, runRecord, attachedInstruments, }) diff --git a/app/src/organisms/ErrorRecoveryFlows/index.tsx b/app/src/organisms/ErrorRecoveryFlows/index.tsx index 6e4e2bf1fd3..677cd512986 100644 --- a/app/src/organisms/ErrorRecoveryFlows/index.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/index.tsx @@ -106,7 +106,6 @@ export interface ErrorRecoveryFlowsProps { runId: string runStatus: RunStatus | null failedCommand: FailedCommand | null - isFlex: boolean protocolAnalysis: CompletedProtocolAnalysis | 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/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/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__/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/pages/RunningProtocol/index.tsx b/app/src/pages/RunningProtocol/index.tsx index 5240196e9e1..869fc7c61d8 100644 --- a/app/src/pages/RunningProtocol/index.tsx +++ b/app/src/pages/RunningProtocol/index.tsx @@ -164,7 +164,6 @@ export function RunningProtocol(): JSX.Element { {isERActive ? ( Date: Wed, 24 Jul 2024 15:04:46 -0400 Subject: [PATCH 30/50] feat(app,api): Move to right position before retrying in place command (#15776) This PR adds something in between a nice feature and a hack for retrying commands like `AspirateInPlace` in error recovery. The fundamental problem is that these commands' parameters do not contain the location where you're running them, and we know that we've moved the gantry since they failed. The fix is to add an optional `retryLocation` value to the error info of the well-defined overpressure command (and subsequently to any other well-defined error that will be raised from these commands), as in 2ae195619b6e19e3163528b05b3ba3e8300fac49 ; then use that location as the target of a MoveToCoordinates before we actually retry the target command if necessary, as we do in 9183301e27d351ad4a2b560ad29a52c73ed99c36 ## Testing - [x] Pass tests, mostly. I wanted to keep this separate from actually implementing the errors in those commands just to have a nicer commit log, so you can't really test it right now. I figured I'd follow up with a PR that implements it in those commands and apply any needed fixes there. --- .../protocol_engine/commands/aspirate.py | 1 + .../commands/pipetting_common.py | 10 ++- .../errors/error_occurrence.py | 4 +- .../protocol_engine/commands/test_aspirate.py | 5 +- .../state/test_pipette_store.py | 2 + .../__tests__/useRecoveryCommands.test.ts | 53 +++++++++++++++ .../hooks/useRecoveryCommands.ts | 65 +++++++++++++++++-- .../utils/__tests__/getErrorKind.test.ts | 4 +- .../__tests__/RunFailedModal.test.tsx | 20 +++--- shared-data/command/types/index.ts | 25 +++++-- shared-data/errors/index.ts | 4 ++ shared-data/errors/types/index.ts | 15 +++++ shared-data/tsconfig.json | 5 +- 13 files changed, 185 insertions(+), 28 deletions(-) create mode 100644 shared-data/errors/index.ts create mode 100644 shared-data/errors/types/index.ts 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/pipetting_common.py b/api/src/opentrons/protocol_engine/commands/pipetting_common.py index 408b1c71478..898cb6f1850 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,6 +123,12 @@ 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. @@ -138,6 +144,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/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/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/state/test_pipette_store.py b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py index 8ccfc06fd07..c60adb96fdb 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py @@ -330,6 +330,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) @@ -818,6 +819,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) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts index 1a54da00103..fb7b7367842 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts @@ -125,6 +125,59 @@ 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, + }) + ) + 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(() => diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts index 58c398b33f6..d28fcb6396a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -9,7 +9,16 @@ import { import { useChainRunCommands } from '../../../resources/runs' import { RECOVERY_MAP } from '../constants' -import type { CreateCommand, LoadedLabware } from '@opentrons/shared-data' +import type { + CreateCommand, + LoadedLabware, + MoveToCoordinatesCreateCommand, + AspirateInPlaceRunTimeCommand, + BlowoutInPlaceRunTimeCommand, + DispenseInPlaceRunTimeCommand, + DropTipInPlaceRunTimeCommand, + PrepareToAspirateRunTimeCommand, +} from '@opentrons/shared-data' import type { CommandData } from '@opentrons/api-client' import type { WellGroup } from '@opentrons/components' import type { FailedCommand } from '../types' @@ -56,6 +65,49 @@ export function useRecoveryCommands({ const { stopRun } = useStopRunMutation() 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[], @@ -72,10 +124,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. 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/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/shared-data/command/types/index.ts b/shared-data/command/types/index.ts index 5d6503761a3..2970a8e5185 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, @@ -78,14 +79,28 @@ export type RunTimeCommand = | AnnotationRunTimeCommand // annotating command execution | IncidentalRunTimeCommand // command with only incidental effects (status bar animations) +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/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/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" ] } From 8357916365cdf5e456611a8dc176826e7b8939da Mon Sep 17 00:00:00 2001 From: Seth Foster Date: Wed, 24 Jul 2024 16:23:12 -0400 Subject: [PATCH 31/50] feat(api): Handle overpressures in dispense (#15754) As in aspirate, we handle overpressure errors and turn them into DefinedError return values, allowing us to recover from overpressures via client-driven error recovery. Closes EXEC-498 --- .../protocol_engine/commands/dispense.py | 85 ++++++++++++----- .../protocol_engine/state/pipettes.py | 5 +- .../protocol_engine/commands/test_dispense.py | 91 ++++++++++++++++++- .../state/test_pipette_store.py | 68 ++++++++++++++ 4 files changed, 220 insertions(+), 29 deletions(-) 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/state/pipettes.py b/api/src/opentrons/protocol_engine/state/pipettes.py index 92344dd9600..35cfca94f33 100644 --- a/api/src/opentrons/protocol_engine/state/pipettes.py +++ b/api/src/opentrons/protocol_engine/state/pipettes.py @@ -13,6 +13,7 @@ ) 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.command import DefinedErrorData from opentrons.protocol_engine.commands.pipetting_common import ( OverpressureError, @@ -316,7 +317,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 +413,7 @@ def _update_deck_point( ) 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) ): 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/state/test_pipette_store.py b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py index c60adb96fdb..c132ea56c73 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py @@ -426,6 +426,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( @@ -902,6 +939,37 @@ 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, + ), ), ) def test_movement_commands_update_deck_point( From e4c829c207a5b1eba3ea195c924468ce72e0b862 Mon Sep 17 00:00:00 2001 From: Ryan Howard Date: Wed, 24 Jul 2024 16:28:31 -0400 Subject: [PATCH 32/50] feat(hardware): add sensor max value getter (#15786) # Overview Only used in can control right now but this provides a way to get the max sensor value reading, which we can use to differentiate pipettes with the old and new sensor boards. # Test Plan # Changelog # Review requests # Risk assessment --- .../firmware_bindings/constants.py | 3 +++ .../messages/message_definitions.py | 22 +++++++++++++++++++ .../firmware_bindings/messages/messages.py | 2 ++ 3 files changed, 27 insertions(+) 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, From 93446728c5dd52282bbf8fe8401b2d69e4358ed8 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 24 Jul 2024 17:12:14 -0400 Subject: [PATCH 33/50] fix(app): prevent eternal intervention purgatory (#15788) If you click on a run that has an intervention modal due to a pause, move labware, etc., you shouldn't have to stare at that run until you close or your app, deal with the robot, or contemplate your navigation actions for all eternity. Navigating away should be a legitimate option. --- app/src/organisms/RunProgressMeter/index.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) 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} From 8063e76eae0baf7d9834bd3458cbbe9c31460c65 Mon Sep 17 00:00:00 2001 From: aaron-kulkarni <107003644+aaron-kulkarni@users.noreply.github.com> Date: Thu, 25 Jul 2024 09:14:53 -0400 Subject: [PATCH 34/50] refactor(robot-server): refactor error recovery policy model (#15785) # Overview Previously, an ErrorRecoveryPolicy was of the format: ``` rules matchCriteria criteria 1 (with one commandType and one errorType) criteria 2 (with one commandType and one errorType) criteria 3 (with one commandType and one errorType) ifMatch reaction 1 reaction 2 reaction 3 ``` This PR moves it to the format of: ``` rules rule 1 matchCriteria (with one commandType and one errorType) ifMatch reaction rule 2 matchCriteria (with one commandType and one errorType) ifMatch reaction ``` # Test Plan Make sure old unit test pass. # Changelog # Review requests # Risk assessment --- .../runs/error_recovery_mapping.py | 34 ++++++++++--------- .../runs/error_recovery_models.py | 8 ++--- .../tests/runs/test_error_recovery_mapping.py | 12 +++---- 3 files changed, 26 insertions(+), 28 deletions(-) diff --git a/robot-server/robot_server/runs/error_recovery_mapping.py b/robot-server/robot_server/runs/error_recovery_mapping.py index dfef53ad05f..28664079a1d 100644 --- a/robot-server/robot_server/runs/error_recovery_mapping.py +++ b/robot-server/robot_server/runs/error_recovery_mapping.py @@ -23,23 +23,25 @@ def _policy( failed_command: Command, defined_error_data: Optional[CommandDefinedErrorData], ) -> ErrorRecoveryType: + if rules is None: + return standard_run_policy(config, failed_command, defined_error_data) for rule in rules: - for i, criteria in enumerate(rule.matchCriteria): - command_type_matches = ( - failed_command.commandType == criteria.command.commandType - ) - error_type_matches = ( - defined_error_data is not None - and defined_error_data.public.errorType - == criteria.command.error.errorType - ) - if command_type_matches and error_type_matches: - if rule.ifMatch[i] == ReactionIfMatch.IGNORE_AND_CONTINUE: - raise NotImplementedError # No protocol engine support for this yet. It's in EXEC-302. - elif rule.ifMatch[i] == ReactionIfMatch.FAIL_RUN: - return ErrorRecoveryType.FAIL_RUN - elif rule.ifMatch[i] == ReactionIfMatch.WAIT_FOR_RECOVERY: - return ErrorRecoveryType.WAIT_FOR_RECOVERY + 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: + raise NotImplementedError # No protocol engine support for this yet. It's in EXEC-302. + elif rule.ifMatch == ReactionIfMatch.FAIL_RUN: + return ErrorRecoveryType.FAIL_RUN + elif rule.ifMatch == ReactionIfMatch.WAIT_FOR_RECOVERY: + return ErrorRecoveryType.WAIT_FOR_RECOVERY return standard_run_policy(config, failed_command, defined_error_data) diff --git a/robot-server/robot_server/runs/error_recovery_models.py b/robot-server/robot_server/runs/error_recovery_models.py index 95a5a1e5631..65f0e9552db 100644 --- a/robot-server/robot_server/runs/error_recovery_models.py +++ b/robot-server/robot_server/runs/error_recovery_models.py @@ -59,11 +59,11 @@ class MatchCriteria(BaseModel): class ErrorRecoveryRule(BaseModel): """Request/Response model for new error recovery rule creation.""" - matchCriteria: list[MatchCriteria] = Field( - default_factory=list, + matchCriteria: MatchCriteria = Field( + ..., description="The criteria that must be met for this rule to be applied.", ) - ifMatch: list[ReactionIfMatch] = Field( - default_factory=list, + ifMatch: ReactionIfMatch = Field( + ..., description="The specific recovery setting that will be in use if the type parameters match.", ) diff --git a/robot-server/tests/runs/test_error_recovery_mapping.py b/robot-server/tests/runs/test_error_recovery_mapping.py index 4d01ad50085..a2e7575c306 100644 --- a/robot-server/tests/runs/test_error_recovery_mapping.py +++ b/robot-server/tests/runs/test_error_recovery_mapping.py @@ -64,8 +64,8 @@ def mock_criteria(decoy: Decoy) -> MatchCriteria: 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]) + decoy.when(mock.ifMatch).then_return(ReactionIfMatch.IGNORE_AND_CONTINUE) + decoy.when(mock.matchCriteria).then_return(mock_criteria) return mock @@ -89,9 +89,7 @@ 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.""" - rule1 = ErrorRecoveryRule(matchCriteria=[], ifMatch=[]) - - policy = create_error_recovery_policy_from_rules([rule1]) + policy = create_error_recovery_policy_from_rules(rules=[]) exampleConfig = Config( robot_type="OT-3 Standard", deck_type=DeckType.OT3_STANDARD, @@ -104,9 +102,7 @@ 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.""" - rule1 = ErrorRecoveryRule(matchCriteria=[], ifMatch=[]) - - policy = create_error_recovery_policy_from_rules([rule1]) + policy = create_error_recovery_policy_from_rules(rules=[]) exampleConfig = Config( robot_type="OT-3 Standard", deck_type=DeckType.OT3_STANDARD, From e99426da04d83f83bc24005037fdaed3734d4bcc Mon Sep 17 00:00:00 2001 From: Max Marrone Date: Thu, 25 Jul 2024 09:28:18 -0400 Subject: [PATCH 35/50] feat(robot-server): Add `/clientData` endpoints (#15727) --- .../robot_server/client_data/__init__.py | 1 + .../robot_server/client_data/router.py | 128 ++++++++++++++++++ .../robot_server/client_data/store.py | 55 ++++++++ robot-server/robot_server/router.py | 7 + robot-server/tests/integration/conftest.py | 6 + .../http_api/test_client_data.tavern.yaml | 112 +++++++++++++++ .../tests/integration/robot_client.py | 5 + 7 files changed, 314 insertions(+) create mode 100644 robot-server/robot_server/client_data/__init__.py create mode 100644 robot-server/robot_server/client_data/router.py create mode 100644 robot-server/robot_server/client_data/store.py create mode 100644 robot-server/tests/integration/http_api/test_client_data.tavern.yaml 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..35ec8c6e06c --- /dev/null +++ b/robot-server/robot_server/client_data/router.py @@ -0,0 +1,128 @@ +"""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 + +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: ClientDataStore = fastapi.Depends(get_client_data_store), +) -> SimpleBody[ClientData]: + store.put(key, request_body.data) + 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: ClientDataStore = fastapi.Depends(get_client_data_store), +) -> 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: + 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: ClientDataStore = fastapi.Depends(get_client_data_store), +) -> SimpleEmptyBody: + store.delete_all() + 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..29372f31e17 --- /dev/null +++ b/robot-server/robot_server/client_data/store.py @@ -0,0 +1,55 @@ +"""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 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/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/tests/integration/conftest.py b/robot-server/tests/integration/conftest.py index 2c8d66853a1..14de4bed0ce 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()) @@ -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/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/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 From 6b88258557d4fe2511b4f73589314776d31e369f Mon Sep 17 00:00:00 2001 From: koji Date: Thu, 25 Jul 2024 09:30:38 -0400 Subject: [PATCH 36/50] fix(app): fix LabwareListItem padding (#15742) * fix(app): fix LabwareListItem padding --- .../Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/LabwareListItem.tsx index 80141c4f1e5..32c3d434c35 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 { From baac901c60f2b1c1c36814616ce41b5fde1a64cd Mon Sep 17 00:00:00 2001 From: Jethary Rader <66035149+jerader@users.noreply.github.com> Date: Thu, 25 Jul 2024 10:23:48 -0400 Subject: [PATCH 37/50] =?UTF-8?q?refactor(labware-library):=20aluminum=20f?= =?UTF-8?q?lat=20bottom=20plate=20compatible=20for=20=E2=80=A6=20(#15794)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit …multiple types closes AUTH-572 --- .../labware-creator/components/sections/StackingOffsets.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) 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 && From 7a7bdf8cb598f8952576f479018c0b1682e57a3b Mon Sep 17 00:00:00 2001 From: aaron-kulkarni <107003644+aaron-kulkarni@users.noreply.github.com> Date: Thu, 25 Jul 2024 10:36:23 -0400 Subject: [PATCH 38/50] refactor(shared-data): remove .dev_types submodules (#15793) # Overview The dev_types submodule is vestigial. This PR moves dev_types definitions to the respective types file and deletes dev_types. If there was no existing types file, dev_types just gets renamed as types. [EXEC-220](https://opentrons.atlassian.net/browse/EXEC-220) Originally this was #15761 but I just made this new PR since it was easier to start from scratch. # Test Plan # Changelog # Review requests # Risk assessment [EXEC-220]: https://opentrons.atlassian.net/browse/EXEC-220?atlOrigin=eyJpIjoiNWRkNTljNzYxNjVmNDY3MDlhMDU5Y2ZhYzA5YTRkZjUiLCJwIjoiZ2l0aHViLWNvbS1KU1cifQ --- .../opentrons/calibration_storage/helpers.py | 4 +- .../calibration_storage/ot2/models/v1.py | 2 +- .../calibration_storage/ot2/tip_length.py | 4 +- .../calibration_storage/ot3/models/v1.py | 2 +- api/src/opentrons/cli/analyze.py | 2 +- api/src/opentrons/config/advanced_settings.py | 2 +- api/src/opentrons/config/feature_flags.py | 2 +- api/src/opentrons/config/reset.py | 2 +- api/src/opentrons/execute.py | 4 +- api/src/opentrons/hardware_control/api.py | 4 +- .../hardware_control/backends/controller.py | 4 +- .../backends/flex_protocol.py | 2 +- .../backends/ot3controller.py | 2 +- .../hardware_control/backends/ot3simulator.py | 2 +- .../hardware_control/backends/simulator.py | 2 +- .../opentrons/hardware_control/dev_types.py | 2 +- .../instruments/ot2/instrument_calibration.py | 4 +- .../instruments/ot2/pipette.py | 2 +- .../instruments/ot2/pipette_handler.py | 2 +- .../instruments/ot3/pipette.py | 2 +- .../instruments/ot3/pipette_handler.py | 2 +- .../hardware_control/modules/types.py | 2 +- .../hardware_control/motion_utilities.py | 2 +- api/src/opentrons/hardware_control/ot3api.py | 4 +- .../protocols/instrument_configurer.py | 2 +- .../motion_planning/adjacent_slots_getters.py | 2 +- .../motion_planning/deck_conflict.py | 4 +- .../protocol_api/core/engine/instrument.py | 2 +- .../protocol_api/core/engine/labware.py | 2 +- .../protocol_api/core/engine/protocol.py | 8 +- .../opentrons/protocol_api/core/labware.py | 2 +- .../protocol_api/core/legacy/deck.py | 4 +- .../core/legacy/legacy_labware_core.py | 2 +- .../core/legacy/legacy_protocol_core.py | 8 +- .../protocol_api/core/legacy/load_info.py | 2 +- .../core/legacy/module_geometry.py | 2 +- .../protocol_api/core/legacy/well_geometry.py | 2 +- .../legacy_simulator/legacy_protocol_core.py | 2 +- .../opentrons/protocol_api/core/protocol.py | 8 +- .../protocol_api/create_protocol_context.py | 2 +- api/src/opentrons/protocol_api/deck.py | 4 +- api/src/opentrons/protocol_api/labware.py | 2 +- .../opentrons/protocol_api/module_contexts.py | 4 +- .../protocol_api/protocol_context.py | 4 +- api/src/opentrons/protocol_api/validation.py | 4 +- .../protocol_engine/clients/sync_client.py | 2 +- .../protocol_engine/clients/transports.py | 2 +- .../protocol_engine/commands/load_pipette.py | 4 +- .../protocol_engine/execution/equipment.py | 2 +- .../resources/deck_configuration_provider.py | 2 +- .../resources/deck_data_provider.py | 2 +- .../resources/pipette_data_provider.py | 2 +- .../protocol_engine/slot_standardization.py | 2 +- .../state/addressable_areas.py | 4 +- .../opentrons/protocol_engine/state/config.py | 2 +- .../protocol_engine/state/geometry.py | 4 +- .../protocol_engine/state/labware.py | 4 +- .../opentrons/protocol_engine/state/state.py | 4 +- api/src/opentrons/protocol_engine/types.py | 8 +- .../protocol_reader/file_identifier.py | 2 +- .../protocol_reader/protocol_source.py | 2 +- .../create_simulating_orchestrator.py | 2 +- .../protocol_runner/json_translator.py | 2 +- .../protocol_runner/legacy_command_mapper.py | 2 +- .../python_protocol_wrappers.py | 2 +- .../protocol_runner/run_orchestrator.py | 4 +- .../protocols/api_support/deck_type.py | 2 +- .../protocols/api_support/instrument.py | 4 +- .../opentrons/protocols/api_support/util.py | 2 +- api/src/opentrons/protocols/bundle.py | 2 +- .../protocols/execution/dev_types.py | 2 +- .../protocols/execution/execute_json_v3.py | 2 +- .../protocols/execution/execute_json_v4.py | 2 +- .../protocols/execution/execute_json_v5.py | 2 +- .../protocols/geometry/labware_geometry.py | 2 +- api/src/opentrons/protocols/labware.py | 2 +- .../protocols/models/json_protocol.py | 56 +++--- api/src/opentrons/protocols/parse.py | 6 +- api/src/opentrons/protocols/types.py | 6 +- api/src/opentrons/simulate.py | 4 +- api/src/opentrons/types.py | 2 +- api/src/opentrons/util/entrypoint_util.py | 2 +- api/src/opentrons/util/performance_helpers.py | 2 +- .../calibration_storage/test_deck_attitude.py | 2 +- .../test_pipette_offset_ot2.py | 2 +- .../test_pipette_offset_ot3.py | 2 +- .../test_tip_length_ot2.py | 4 +- .../config/test_advanced_settings.py | 2 +- api/tests/opentrons/config/test_reset.py | 4 +- api/tests/opentrons/conftest.py | 10 +- .../test_instrument_calibration.py | 2 +- .../hardware_control/test_ot3_api.py | 2 +- .../hardware_control/test_ot3_transforms.py | 2 +- .../hardware_control/test_simulator_setup.py | 2 +- .../test_adjacent_slots_getters.py | 2 +- .../motion_planning/test_deck_conflict.py | 4 +- .../core/engine/test_deck_conflict.py | 4 +- .../core/engine/test_instrument_core.py | 2 +- .../core/engine/test_labware_core.py | 2 +- .../core/engine/test_protocol_core.py | 8 +- .../protocol_api/core/legacy/test_deck.py | 2 +- .../core/legacy/test_module_geometry.py | 2 +- .../test_protocol_context_implementation.py | 6 +- api/tests/opentrons/protocol_api/test_deck.py | 2 +- .../opentrons/protocol_api/test_labware.py | 2 +- .../protocol_api/test_module_context.py | 2 +- .../protocol_api/test_protocol_context.py | 4 +- .../opentrons/protocol_api/test_validation.py | 4 +- .../core/simulator/conftest.py | 4 +- .../core/simulator/test_protocol_context.py | 2 +- .../protocol_api_old/test_context.py | 2 +- .../protocol_api_old/test_labware.py | 2 +- .../protocol_api_old/test_labware_load.py | 2 +- .../protocol_api_old/test_module_context.py | 2 +- .../protocol_api_old/test_offsets.py | 2 +- .../clients/test_child_thread_transport.py | 2 +- .../clients/test_sync_client.py | 2 +- .../commands/test_configure_for_volume.py | 2 +- .../commands/test_load_module.py | 4 +- .../commands/test_load_pipette.py | 4 +- .../opentrons/protocol_engine/conftest.py | 2 +- .../execution/test_equipment_handler.py | 4 +- .../test_heater_shaker_movement_flagger.py | 2 +- .../protocol_engine/pipette_fixtures.py | 2 +- .../test_deck_configuration_provider.py | 2 +- .../resources/test_deck_data_provider.py | 2 +- .../resources/test_labware_data_provider.py | 2 +- .../resources/test_pipette_data_provider.py | 2 +- .../protocol_engine/state/command_fixtures.py | 2 +- .../state/test_addressable_area_state.py | 2 +- .../state/test_addressable_area_store.py | 2 +- .../state/test_addressable_area_view.py | 4 +- .../state/test_geometry_view.py | 6 +- .../state/test_labware_store.py | 2 +- .../state/test_labware_view.py | 4 +- .../state/test_module_store.py | 4 +- .../protocol_engine/state/test_module_view.py | 4 +- .../protocol_engine/state/test_motion_view.py | 4 +- .../state/test_pipette_store.py | 2 +- .../state/test_pipette_view.py | 2 +- .../protocol_engine/state/test_state_store.py | 2 +- .../protocol_engine/state/test_tip_state.py | 2 +- .../test_create_protocol_engine.py | 4 +- .../protocol_engine/test_protocol_engine.py | 2 +- .../test_slot_standardization.py | 2 +- .../protocol_reader/test_file_identifier.py | 2 +- .../smoke_tests/test_legacy_command_mapper.py | 2 +- .../smoke_tests/test_protocol_runner.py | 2 +- .../protocol_runner/test_json_translator.py | 2 +- .../test_legacy_command_mapper.py | 6 +- .../test_legacy_context_plugin.py | 2 +- .../protocol_runner/test_protocol_runner.py | 4 +- .../api_support/test_labware_like.py | 2 +- api/tests/opentrons/protocols/test_parse.py | 2 +- api/tests/opentrons/test_execute.py | 2 +- .../opentrons/util/test_entrypoint_util.py | 2 +- .../g_code_parsing/g_code_engine.py | 2 +- .../hardware_testing/gravimetric/helpers.py | 2 +- .../opentrons_api/p1000_gen3_ul_per_mm.py | 2 +- .../production_qc/z_stage_qc_ot3.py | 2 +- .../scripts/visualize_pipette_function.py | 2 +- .../robot_server/deck_configuration/router.py | 2 +- .../deck_configuration/validation.py | 2 +- robot-server/robot_server/hardware.py | 4 +- robot-server/robot_server/health/models.py | 2 +- robot-server/robot_server/health/router.py | 2 +- .../instruments/instrument_models.py | 2 +- .../maintenance_runs/dependencies.py | 2 +- .../maintenance_run_orchestrator_store.py | 4 +- .../persistence/_legacy_pickle.py | 2 +- .../robot_server/protocols/analysis_models.py | 2 +- .../robot_server/protocols/analysis_store.py | 2 +- .../protocols/protocol_analyzer.py | 2 +- .../robot_server/protocols/protocol_models.py | 2 +- robot-server/robot_server/protocols/router.py | 2 +- .../robot/calibration/check/user_flow.py | 2 +- .../robot/calibration/deck/user_flow.py | 4 +- .../robot/calibration/helper_classes.py | 2 +- .../calibration/pipette_offset/user_flow.py | 4 +- .../robot/calibration/tip_length/user_flow.py | 4 +- .../robot_server/robot/calibration/util.py | 4 +- .../robot_server/robot/control/router.py | 4 +- .../robot_server/runs/dependencies.py | 2 +- .../runs/run_orchestrator_store.py | 6 +- .../service/legacy/routers/settings.py | 2 +- .../pipette_offset_calibration.py | 2 +- .../session_types/tip_length_calibration.py | 2 +- .../robot_server/service/tip_length/router.py | 2 +- robot-server/tests/conftest.py | 2 +- robot-server/tests/instruments/test_router.py | 2 +- robot-server/tests/integration/fixtures.py | 2 +- .../router/test_labware_router.py | 2 +- .../maintenance_runs/test_engine_store.py | 2 +- .../tests/protocols/test_analyses_manager.py | 2 +- .../tests/protocols/test_analysis_store.py | 2 +- .../tests/protocols/test_protocol_analyzer.py | 4 +- .../tests/runs/router/test_labware_router.py | 2 +- robot-server/tests/runs/test_engine_store.py | 2 +- robot-server/tests/runs/test_run_store.py | 2 +- .../service/legacy/routers/test_settings.py | 2 +- shared-data/protocol/README.md | 2 +- .../opentrons_shared_data/deck/__init__.py | 2 +- .../deck/{dev_types.py => types.py} | 4 +- .../opentrons_shared_data/labware/__init__.py | 2 +- .../labware/{dev_types.py => types.py} | 2 +- .../opentrons_shared_data/module/__init__.py | 2 +- .../module/{dev_types.py => types.py} | 2 +- .../opentrons_shared_data/pipette/__init__.py | 2 +- .../pipette/dev_types.py | 4 +- .../pipette/mutable_configurations.py | 2 +- .../pipette/pipette_definition.py | 4 +- .../pipette/pipette_load_name_conversions.py | 2 +- .../pipette/scripts/build_json_script.py | 2 +- .../scripts/update_configuration_files.py | 2 +- .../opentrons_shared_data/pipette/types.py | 172 +++++++++++++++++- .../protocol/constants.py | 2 +- .../protocol/{dev_types.py => types.py} | 8 +- .../opentrons_shared_data/robot/__init__.py | 2 +- .../robot/{dev_types.py => types.py} | 2 +- .../python/tests/deck/test_position.py | 2 +- .../python/tests/deck/test_typechecks.py | 2 +- .../python/tests/labware/test_typechecks.py | 2 +- .../python/tests/module/test_typechecks.py | 8 +- .../python/tests/pipette/test_load_data.py | 4 +- .../pipette/test_max_flow_rates_per_volume.py | 2 +- .../pipette/test_mutable_configurations.py | 5 +- .../test_pipette_load_name_conversions.py | 2 +- .../python/tests/pipette/test_typechecks.py | 2 +- .../tests/pipette/test_validate_schema.py | 2 +- .../python/tests/protocol/test_typechecks.py | 2 +- .../python/tests/robot/test_typechecks.py | 2 +- 231 files changed, 514 insertions(+), 349 deletions(-) rename shared-data/python/opentrons_shared_data/deck/{dev_types.py => types.py} (97%) rename shared-data/python/opentrons_shared_data/labware/{dev_types.py => types.py} (97%) rename shared-data/python/opentrons_shared_data/module/{dev_types.py => types.py} (98%) rename shared-data/python/opentrons_shared_data/protocol/{dev_types.py => types.py} (98%) rename shared-data/python/opentrons_shared_data/robot/{dev_types.py => types.py} (95%) 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..2beecb37cd3 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 ( 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..df4f5e71b55 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 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/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/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/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..434f461d524 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 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/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/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/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/state.py b/api/src/opentrons/protocol_engine/state/state.py index e343a4dfde1..e0d04b3d050 100644 --- a/api/src/opentrons/protocol_engine/state/state.py +++ b/api/src/opentrons/protocol_engine/state/state.py @@ -5,8 +5,8 @@ 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.types import ModuleOffsetData from opentrons.util.change_notifier import ChangeNotifier diff --git a/api/src/opentrons/protocol_engine/types.py b/api/src/opentrons/protocol_engine/types.py index 17a18a8ae4f..ca2525c6efe 100644 --- a/api/src/opentrons/protocol_engine/types.py +++ b/api/src/opentrons/protocol_engine/types.py @@ -16,7 +16,7 @@ from typing import Optional, Union, List, Dict, Any, NamedTuple, Tuple, FrozenSet 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 +26,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 +347,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.""" 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/python_protocol_wrappers.py b/api/src/opentrons/protocol_runner/python_protocol_wrappers.py index 4a022ca184f..5b7c43cae46 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 diff --git a/api/src/opentrons/protocol_runner/run_orchestrator.py b/api/src/opentrons/protocol_runner/run_orchestrator.py index 18ad48b6d5b..1aa60385e74 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 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/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/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..ec581f71875 100644 --- a/api/src/opentrons/util/performance_helpers.py +++ b/api/src/opentrons/util/performance_helpers.py @@ -3,7 +3,7 @@ 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, 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/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_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/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_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..66c6a34fe9f 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 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_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 c132ea56c73..6296b30911f 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 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..3eb7e2194ed 100644 --- a/api/tests/opentrons/protocol_engine/state/test_state_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_state_store.py @@ -5,7 +5,7 @@ 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 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..b4b9968a82d 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, 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..bafeab0279e 100644 --- a/api/tests/opentrons/protocol_engine/test_protocol_engine.py +++ b/api/tests/opentrons/protocol_engine/test_protocol_engine.py @@ -7,7 +7,7 @@ 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.types import DeckSlotName from opentrons.hardware_control import HardwareControlAPI, OT2HardwareControlAPI 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..3bae3c16d81 100644 --- a/api/tests/opentrons/protocol_runner/test_protocol_runner.py +++ b/api/tests/opentrons/protocol_runner/test_protocol_runner.py @@ -9,11 +9,11 @@ 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 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/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/g-code-testing/g_code_parsing/g_code_engine.py b/g-code-testing/g_code_parsing/g_code_engine.py index 29e5b046e7f..a42110f1d40 100644 --- a/g-code-testing/g_code_parsing/g_code_engine.py +++ b/g-code-testing/g_code_parsing/g_code_engine.py @@ -33,7 +33,7 @@ ) from g_code_parsing.g_code_watcher import GCodeWatcher from g_code_parsing.utils import get_configuration_dir -from opentrons_shared_data.robot.dev_types import RobotType +from opentrons_shared_data.robot.types import RobotType Protocol = namedtuple("Protocol", ["text", "filename", "filelike"]) diff --git a/hardware-testing/hardware_testing/gravimetric/helpers.py b/hardware-testing/hardware_testing/gravimetric/helpers.py index 31541d59f5a..5c89712ac81 100644 --- a/hardware-testing/hardware_testing/gravimetric/helpers.py +++ b/hardware-testing/hardware_testing/gravimetric/helpers.py @@ -26,7 +26,7 @@ from opentrons import execute, simulate from opentrons.types import Point, Location, Mount from opentrons.config.types import OT3Config, RobotConfig -from opentrons_shared_data.labware.dev_types import LabwareDefinition +from opentrons_shared_data.labware.types import LabwareDefinition from hardware_testing.opentrons_api import helpers_ot3 from opentrons.protocol_api import ProtocolContext, InstrumentContext diff --git a/hardware-testing/hardware_testing/opentrons_api/p1000_gen3_ul_per_mm.py b/hardware-testing/hardware_testing/opentrons_api/p1000_gen3_ul_per_mm.py index 66df1dae6d5..9da7424eda4 100644 --- a/hardware-testing/hardware_testing/opentrons_api/p1000_gen3_ul_per_mm.py +++ b/hardware-testing/hardware_testing/opentrons_api/p1000_gen3_ul_per_mm.py @@ -6,7 +6,7 @@ from opentrons.hardware_control.instruments.ot3.pipette import Pipette -from opentrons_shared_data.pipette.dev_types import UlPerMm +from opentrons_shared_data.pipette.types import UlPerMm from .types import OT3Mount diff --git a/hardware-testing/hardware_testing/production_qc/z_stage_qc_ot3.py b/hardware-testing/hardware_testing/production_qc/z_stage_qc_ot3.py index 7806561568a..8a55a831c45 100644 --- a/hardware-testing/hardware_testing/production_qc/z_stage_qc_ot3.py +++ b/hardware-testing/hardware_testing/production_qc/z_stage_qc_ot3.py @@ -23,7 +23,7 @@ from opentrons_shared_data.errors.exceptions import MoveConditionNotMetError from opentrons.config.advanced_settings import get_adv_setting, set_adv_setting -from opentrons_shared_data.robot.dev_types import RobotTypeEnum +from opentrons_shared_data.robot.types import RobotTypeEnum import logging diff --git a/hardware-testing/hardware_testing/scripts/visualize_pipette_function.py b/hardware-testing/hardware_testing/scripts/visualize_pipette_function.py index 59d2dde68e1..62229aabb5a 100644 --- a/hardware-testing/hardware_testing/scripts/visualize_pipette_function.py +++ b/hardware-testing/hardware_testing/scripts/visualize_pipette_function.py @@ -31,7 +31,7 @@ PipetteOffsetByPipetteMount, ) from opentrons_shared_data.pipette import model_config -from opentrons_shared_data.pipette.dev_types import PipetteModel +from opentrons_shared_data.pipette.types import PipetteModel def _user_select_model(model_includes: Optional[str] = None) -> str: 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/_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/protocols/analysis_models.py b/robot-server/robot_server/protocols/analysis_models.py index 8eeeb7fad76..e9366eb92bb 100644 --- a/robot-server/robot_server/protocols/analysis_models.py +++ b/robot-server/robot_server/protocols/analysis_models.py @@ -3,7 +3,7 @@ from enum import Enum from opentrons.protocol_engine.types import RunTimeParameter, RunTimeParamValuesType -from opentrons_shared_data.robot.dev_types import RobotType +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 diff --git a/robot-server/robot_server/protocols/analysis_store.py b/robot-server/robot_server/protocols/analysis_store.py index 4480cf0adaa..36df45d7e19 100644 --- a/robot-server/robot_server/protocols/analysis_store.py +++ b/robot-server/robot_server/protocols/analysis_store.py @@ -6,7 +6,7 @@ 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, diff --git a/robot-server/robot_server/protocols/protocol_analyzer.py b/robot-server/robot_server/protocols/protocol_analyzer.py index 19b32140cd0..f3974715757 100644 --- a/robot-server/robot_server/protocols/protocol_analyzer.py +++ b/robot-server/robot_server/protocols/protocol_analyzer.py @@ -2,7 +2,7 @@ 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 diff --git a/robot-server/robot_server/protocols/protocol_models.py b/robot-server/robot_server/protocols/protocol_models.py index b0c28e5e4eb..decef1e8e26 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 diff --git a/robot-server/robot_server/protocols/router.py b/robot-server/robot_server/protocols/router.py index 706f4f41ece..8ef8516e03f 100644 --- a/robot-server/robot_server/protocols/router.py +++ b/robot-server/robot_server/protocols/router.py @@ -31,7 +31,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 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/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/run_orchestrator_store.py b/robot-server/robot_server/runs/run_orchestrator_store.py index de346a72968..89fcb9ec402 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 @@ -44,7 +44,7 @@ RunTimeParamValuesType, EngineStatus, ) -from opentrons_shared_data.labware.dev_types import LabwareUri +from opentrons_shared_data.labware.types import LabwareUri _log = logging.getLogger(__name__) 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/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/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/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/protocols/test_analyses_manager.py b/robot-server/tests/protocols/test_analyses_manager.py index 766908adce0..339258306fd 100644 --- a/robot-server/tests/protocols/test_analyses_manager.py +++ b/robot-server/tests/protocols/test_analyses_manager.py @@ -8,7 +8,7 @@ import opentrons.protocol_runner.create_simulating_orchestrator as simulating_runner 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.robot.types import RobotType from robot_server.protocols import protocol_analyzer from robot_server.protocols.protocol_models import ProtocolKind diff --git a/robot-server/tests/protocols/test_analysis_store.py b/robot-server/tests/protocols/test_analysis_store.py index 606d6f13f10..69b1b812311 100644 --- a/robot-server/tests/protocols/test_analysis_store.py +++ b/robot-server/tests/protocols/test_analysis_store.py @@ -11,7 +11,7 @@ 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 diff --git a/robot-server/tests/protocols/test_protocol_analyzer.py b/robot-server/tests/protocols/test_protocol_analyzer.py index 44fde5bd6c2..e4b70e57027 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 ( 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_run_store.py b/robot-server/tests/runs/test_run_store.py index ee7697107f6..2ed63ff49b0 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 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/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/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/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/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..494541c0d0b 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]: 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"]) From 8b6494f56766194b80dbd6a5e8b427a10e0c4130 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Thu, 25 Jul 2024 12:07:12 -0400 Subject: [PATCH 39/50] feat(app): add error recovery analytics (#15787) Closes EXEC-588 --- .../Devices/ProtocolRun/ProtocolRunHeader.tsx | 7 + .../ErrorRecoveryWizard.tsx | 17 +- .../RecoveryOptions/SelectRecoveryOption.tsx | 2 + .../ErrorRecoveryFlows/RunPausedSplash.tsx | 27 +++- .../ErrorRecoveryFlows/__fixtures__/index.ts | 8 + .../__tests__/ErrorRecoveryFlows.test.tsx | 4 + .../__tests__/useRecoveryCommands.test.ts | 153 ++++++------------ .../ErrorRecoveryFlows/hooks/index.ts | 2 + .../ErrorRecoveryFlows/hooks/useERUtils.ts | 5 + .../hooks/useRecoveryAnalytics.ts | 127 +++++++++++++++ .../hooks/useRecoveryCommands.ts | 12 ++ .../organisms/ErrorRecoveryFlows/index.tsx | 11 +- app/src/pages/RunSummary/index.tsx | 2 + app/src/redux/analytics/constants.ts | 13 ++ 14 files changed, 280 insertions(+), 110 deletions(-) create mode 100644 app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryAnalytics.ts diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx index f8338f2521b..11e184c2931 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx @@ -108,6 +108,7 @@ import { useErrorRecoveryFlows, ErrorRecoveryFlows, } from '../../ErrorRecoveryFlows' +import { useRecoveryAnalytics } from '../../ErrorRecoveryFlows/hooks' import type { Run, RunError, RunStatus } from '@opentrons/api-client' import type { IconName } from '@opentrons/components' @@ -148,6 +149,7 @@ export function ProtocolRunHeader({ protocolKey, isProtocolAnalyzing, } = useProtocolDetailsForRun(runId) + const { reportRecoveredRunResult } = useRecoveryAnalytics() const { trackProtocolRunEvent } = useTrackProtocolRunEvent(runId, robotName) const robotAnalyticsData = useRobotAnalyticsData(robotName) @@ -161,6 +163,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 = @@ -234,6 +237,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 +248,9 @@ export function ProtocolRunHeader({ }) closeCurrentRun() } + if (runStatus === RUN_STATUS_AWAITING_RECOVERY) { + setEnteredER(true) + } }, [runStatus, isRunCurrent, runId, closeCurrentRun]) const startedAtTimestamp = diff --git a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx index 7f02436c2db..f8d063aad9a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/ErrorRecoveryWizard.tsx @@ -29,7 +29,7 @@ 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 { @@ -59,6 +59,7 @@ export type ErrorRecoveryWizardProps = ErrorRecoveryFlowsProps & robotType: RobotType isOnDevice: boolean isDoorOpen: boolean + analytics: UseRecoveryAnalyticsResult } export function ErrorRecoveryWizard( @@ -84,11 +85,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 ( 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/RunPausedSplash.tsx b/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx index 814491d702b..427489f66c1 100644 --- a/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/RunPausedSplash.tsx @@ -22,6 +22,7 @@ import { PrimaryButton, SecondaryButton, } from '@opentrons/components' +import { useHost } from '@opentrons/react-api-client' import { useErrorName } from './hooks' import { getErrorKind } from './utils' @@ -35,8 +36,7 @@ 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 } from './hooks' export function useRunPausedSplash( isOnDevice: boolean, @@ -53,17 +53,25 @@ type RunPausedSplashProps = ERUtilsResults & { protocolAnalysis: ErrorRecoveryFlowsProps['protocolAnalysis'] robotType: RobotType toggleERWiz: (launchER: boolean) => Promise + analytics: UseRecoveryAnalyticsResult } export function RunPausedSplash( props: RunPausedSplashProps ): JSX.Element | null { - const { isOnDevice, toggleERWiz, routeUpdateActions, failedCommand } = props + const { + isOnDevice, + toggleERWiz, + routeUpdateActions, + failedCommand, + analytics, + } = 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 ( @@ -75,12 +83,17 @@ export function RunPausedSplash( // 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 toggleERWiz(false).then(() => { + reportInitialActionEvent('cancel-run') + void proceedToRouteAndStep(RECOVERY_MAP.CANCEL_RUN.ROUTE) + }) } - const onLaunchERClick = (): Promise => toggleERWiz(true) + const onLaunchERClick = (): Promise => { + return toggleERWiz(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 diff --git a/app/src/organisms/ErrorRecoveryFlows/__fixtures__/index.ts b/app/src/organisms/ErrorRecoveryFlows/__fixtures__/index.ts index c5f36d77ea7..e7d3a85c484 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__fixtures__/index.ts +++ b/app/src/organisms/ErrorRecoveryFlows/__fixtures__/index.ts @@ -81,4 +81,12 @@ export const mockRecoveryContentProps: RecoveryContentProps = { mockRobotSideAnalysis.commands[mockRobotSideAnalysis.commands.length - 1], ], recoveryActionMutationUtils: {} as any, + analytics: { + reportRecoveredRunResult: () => {}, + 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 54b4f4d814f..d19e0d8e31f 100644 --- a/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/__tests__/ErrorRecoveryFlows.test.tsx @@ -16,6 +16,7 @@ import { useCurrentlyRecoveringFrom, useERUtils, useShowDoorInfo, + useRecoveryAnalytics, } from '../hooks' import { useFeatureFlag } from '../../../redux/config' import { useERWizard, ErrorRecoveryWizard } from '../ErrorRecoveryWizard' @@ -144,6 +145,9 @@ 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) }) it('renders the wizard when the wizard is toggled on', () => { diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts index fb7b7367842..df6ccebaa87 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts @@ -17,28 +17,42 @@ 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 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({ @@ -53,18 +67,10 @@ describe('useRecoveryCommands', () => { }) 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 +85,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 +104,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() @@ -154,6 +144,11 @@ describe('useRecoveryCommands', () => { failedLabwareUtils: mockFailedLabwareUtils, routeUpdateActions: mockRouteUpdateActions, recoveryToastUtils: {} as any, + analytics: { + reportActionSelectedResult: mockReportActionSelectedResult, + reportRecoveredRunResult: mockReportRecoveredRunResult, + } as any, + selectedRecoveryOption: RECOVERY_MAP.RETRY_NEW_TIPS.ROUTE, }) ) await act(async () => { @@ -180,15 +175,7 @@ describe('useRecoveryCommands', () => { }) 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() @@ -199,15 +186,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() @@ -215,15 +194,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() @@ -251,18 +222,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() @@ -275,15 +244,7 @@ describe('useRecoveryCommands', () => { }) 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() @@ -294,15 +255,7 @@ describe('useRecoveryCommands', () => { }) it('should call ignoreErrorKindThisRun and resolve immediately', async () => { - const { result } = renderHook(() => - useRecoveryCommands({ - runId: mockRunId, - failedCommand: mockFailedCommand, - failedLabwareUtils: mockFailedLabwareUtils, - routeUpdateActions: mockRouteUpdateActions, - recoveryToastUtils: {} as any, - }) - ) + const { result } = renderHook(() => useRecoveryCommands(props)) const consoleSpy = vi.spyOn(console, 'log') diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts index 31d9ebb4367..fadc81bd0e9 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/index.ts @@ -5,9 +5,11 @@ export { useShowDoorInfo } from './useShowDoorInfo' export { useRecoveryCommands } from './useRecoveryCommands' export { useRouteUpdateActions } from './useRouteUpdateActions' export { useERUtils } from './useERUtils' +export { useRecoveryAnalytics } from './useRecoveryAnalytics' 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' diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts index c0d867ea25a..98efc5636c2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useERUtils.ts @@ -29,12 +29,14 @@ 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' type ERUtilsProps = ErrorRecoveryFlowsProps & { toggleERWizard: (launchER: boolean) => Promise hasLaunchedRecovery: boolean isOnDevice: boolean robotType: RobotType + analytics: UseRecoveryAnalyticsResult } export interface ERUtilsResults { @@ -64,6 +66,7 @@ export function useERUtils({ protocolAnalysis, isOnDevice, robotType, + analytics, }: ERUtilsProps): ERUtilsResults { const { data: attachedInstruments } = useInstrumentsQuery() const { data: runRecord } = useNotifyRunQuery(runId) @@ -128,6 +131,8 @@ export function useERUtils({ failedLabwareUtils, routeUpdateActions, recoveryToastUtils, + analytics, + selectedRecoveryOption: currentRecoveryOptionUtils.selectedRecoveryOption, }) const deckMapUtils = useDeckMapUtils({ 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 d28fcb6396a..c33bce43416 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -25,6 +25,8 @@ 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 @@ -32,6 +34,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" */ @@ -49,6 +53,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, @@ -56,6 +62,8 @@ export function useRecoveryCommands({ failedLabwareUtils, routeUpdateActions, recoveryToastUtils, + analytics, + selectedRecoveryOption, }: UseRecoveryCommandsParams): UseRecoveryCommandsResult { const { proceedToRouteAndStep } = routeUpdateActions const { chainRunCommands } = useChainRunCommands(runId, failedCommand?.id) @@ -115,6 +123,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}`)) @@ -157,16 +166,19 @@ export function useRecoveryCommands({ 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]) diff --git a/app/src/organisms/ErrorRecoveryFlows/index.tsx b/app/src/organisms/ErrorRecoveryFlows/index.tsx index 677cd512986..75b8dab7c28 100644 --- a/app/src/organisms/ErrorRecoveryFlows/index.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/index.tsx @@ -21,6 +21,7 @@ import { RunPausedSplash, useRunPausedSplash } from './RunPausedSplash' import { useCurrentlyRecoveringFrom, useERUtils, + useRecoveryAnalytics, useShowDoorInfo, } from './hooks' @@ -112,13 +113,18 @@ export interface ErrorRecoveryFlowsProps { export function ErrorRecoveryFlows( props: ErrorRecoveryFlowsProps ): JSX.Element | null { - const { protocolAnalysis, runStatus } = props + const { protocolAnalysis, runStatus, failedCommand } = props const { hasLaunchedRecovery, toggleERWizard, showERWizard } = useERWizard() const isOnDevice = useSelector(getIsOnDevice) const robotType = protocolAnalysis?.robotType ?? OT2_ROBOT_TYPE const showSplash = useRunPausedSplash(isOnDevice, showERWizard) + const analytics = useRecoveryAnalytics() + + React.useEffect(() => { + analytics.reportErrorEvent(failedCommand) + }, [analytics, failedCommand]) const isDoorOpen = useShowDoorInfo(runStatus) @@ -128,6 +134,7 @@ export function ErrorRecoveryFlows( toggleERWizard, isOnDevice, robotType, + analytics, }) return ( @@ -139,6 +146,7 @@ export function ErrorRecoveryFlows( robotType={robotType} isOnDevice={isOnDevice} isDoorOpen={isDoorOpen} + analytics={analytics} /> ) : null} {showSplash ? ( @@ -148,6 +156,7 @@ export function ErrorRecoveryFlows( robotType={robotType} isOnDevice={isOnDevice} toggleERWiz={toggleERWizard} + analytics={analytics} /> ) : null} diff --git a/app/src/pages/RunSummary/index.tsx b/app/src/pages/RunSummary/index.tsx index 348d0c6031f..07fbba7b11c 100644 --- a/app/src/pages/RunSummary/index.tsx +++ b/app/src/pages/RunSummary/index.tsx @@ -149,6 +149,8 @@ export function RunSummary(): JSX.Element { navigate('/') } + // TODO(jh, 07-24-24): After EXEC-504, add reportRecoveredRunResult here. + // 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 => { setShowRunAgainSpinner(true) 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' From ba7b705aa69294ec1fa96cd0b60df6b20ac61257 Mon Sep 17 00:00:00 2001 From: Max Marrone Date: Thu, 25 Jul 2024 14:17:25 -0400 Subject: [PATCH 40/50] feat(robot-server): Notify clients of updates to `GET /clientData/{key}` (#15790) --- .../robot_server/client_data/router.py | 24 +++++++-- .../robot_server/client_data/store.py | 4 ++ .../service/notifications/__init__.py | 2 - .../notifications/notification_client.py | 6 ++- .../publishers/client_data_publisher.py | 29 +++++++++++ .../deck_configuration_publisher.py | 4 +- .../publishers/maintenance_runs_publisher.py | 4 +- .../publishers/runs_publisher.py | 20 +++++--- .../service/notifications/topics.py | 49 ++++++++++++++----- .../test_deck_configuration_publisher.py | 4 +- .../test_maintenance_runs_publisher.py | 4 +- .../publishers/test_runs_publisher.py | 26 +++++----- 12 files changed, 128 insertions(+), 48 deletions(-) create mode 100644 robot-server/robot_server/service/notifications/publishers/client_data_publisher.py diff --git a/robot-server/robot_server/client_data/router.py b/robot-server/robot_server/client_data/router.py index 35ec8c6e06c..d04eaeea9ad 100644 --- a/robot-server/robot_server/client_data/router.py +++ b/robot-server/robot_server/client_data/router.py @@ -13,6 +13,10 @@ 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() @@ -62,9 +66,13 @@ class ClientDataKeyDoesNotExist(ErrorDetails): async def put_client_data( # noqa: D103 key: Key, request_body: RequestModel[ClientData], - store: ClientDataStore = fastapi.Depends(get_client_data_store), + 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)) @@ -104,7 +112,10 @@ async def get_client_data( # noqa: D103 ) async def delete_client_data( # noqa: D103 key: Key, - store: ClientDataStore = fastapi.Depends(get_client_data_store), + 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) @@ -113,6 +124,7 @@ async def delete_client_data( # noqa: D103 fastapi.status.HTTP_404_NOT_FOUND ) from e else: + await client_data_publisher.publish_client_data(key) return SimpleEmptyBody.construct() @@ -122,7 +134,13 @@ async def delete_client_data( # noqa: D103 description="Delete all client-defined data. See `PUT /clientData` for background.", ) async def delete_all_client_data( # noqa: D103 - store: ClientDataStore = fastapi.Depends(get_client_data_store), + 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 index 29372f31e17..2fcd90e8feb 100644 --- a/robot-server/robot_server/client_data/store.py +++ b/robot-server/robot_server/client_data/store.py @@ -29,6 +29,10 @@ def get(self, key: str) -> ClientData: """ 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. 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/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" ) From 7c31fa79063375acef9e0a53db842477d399c3c4 Mon Sep 17 00:00:00 2001 From: koji Date: Thu, 25 Jul 2024 15:47:32 -0400 Subject: [PATCH 41/50] fix(app): add null and length check for array (#15799) * fix(app): add null and length check for array --- app/src/organisms/Devices/HistoricalProtocolRun.tsx | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/app/src/organisms/Devices/HistoricalProtocolRun.tsx b/app/src/organisms/Devices/HistoricalProtocolRun.tsx index 217e4e9380d..ef3a8434cfc 100644 --- a/app/src/organisms/Devices/HistoricalProtocolRun.tsx +++ b/app/src/organisms/Devices/HistoricalProtocolRun.tsx @@ -89,7 +89,9 @@ export function HistoricalProtocolRun( > {protocolName} - {enableCsvFile ? ( + {enableCsvFile && + allProtocolDataFiles != null && + allProtocolDataFiles.length > 0 ? ( Date: Thu, 25 Jul 2024 12:52:08 -0700 Subject: [PATCH 42/50] chore: rename RobotContextTracker to RobotActivityTracker (#15800) # Overview Closes https://opentrons.atlassian.net/browse/EXEC-623 Renames RobotContextTracker and RobotContextState to RobotActivityTracker and RobotActivityState. This is to prevent confusion with the `RobotContext` class added in https://github.com/Opentrons/opentrons/pull/15745 # Test Plan - [x] Make sure linting and testing passes --- api/src/opentrons/util/performance_helpers.py | 30 ++--- .../util/test_performance_helpers.py | 6 +- .../src/performance_metrics/__init__.py | 8 +- .../src/performance_metrics/_data_shapes.py | 18 +-- .../src/performance_metrics/_metrics_store.py | 18 +-- ..._tracker.py => _robot_activity_tracker.py} | 30 ++--- .../src/performance_metrics/_types.py | 6 +- .../performance_metrics/test_metrics_store.py | 26 ++-- .../test_robot_context_tracker.py | 117 +++++++++--------- 9 files changed, 132 insertions(+), 127 deletions(-) rename performance-metrics/src/performance_metrics/{_robot_context_tracker.py => _robot_activity_tracker.py} (82%) diff --git a/api/src/opentrons/util/performance_helpers.py b/api/src/opentrons/util/performance_helpers.py index ec581f71875..416e6766d02 100644 --- a/api/src/opentrons/util/performance_helpers.py +++ b/api/src/opentrons/util/performance_helpers.py @@ -1,4 +1,4 @@ -"""Performance helpers for tracking robot context.""" +"""Performance helpers for tracking robot activity.""" import functools from pathlib import Path @@ -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,35 +72,35 @@ 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. @@ -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/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/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..0bcde3584e3 100644 --- a/performance-metrics/src/performance_metrics/_data_shapes.py +++ b/performance-metrics/src/performance_metrics/_data_shapes.py @@ -4,34 +4,34 @@ import typing from pathlib import Path -from ._types import SupportsCSVStorage, StorableData, RobotContextState +from ._types import SupportsCSVStorage, 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 RawActivityData(SupportsCSVStorage): + """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. """ - 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.""" + """Returns the headers for the raw activity 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.""" + """Returns the raw activity data as a string.""" return ( self.state, self.func_start, @@ -40,9 +40,9 @@ def csv_row(self) -> typing.Tuple[str, int, int]: @classmethod def from_csv_row(cls, row: typing.Sequence[StorableData]) -> SupportsCSVStorage: - """Returns a RawContextData object from a CSV row.""" + """Returns a RawActivityData object from a CSV row.""" return cls( - state=typing.cast(RobotContextState, row[0]), + state=typing.cast(RobotActivityState, row[0]), func_start=int(row[1]), duration=int(row[2]), ) diff --git a/performance-metrics/src/performance_metrics/_metrics_store.py b/performance-metrics/src/performance_metrics/_metrics_store.py index 09bcce50e29..e09fb917a81 100644 --- a/performance-metrics/src/performance_metrics/_metrics_store.py +++ b/performance-metrics/src/performance_metrics/_metrics_store.py @@ -13,20 +13,20 @@ 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 +40,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..dbc8ab002a1 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], 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." From d8f8023908afd3b57c75e514bc18a86f8ea2267a Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Thu, 25 Jul 2024 17:03:06 -0400 Subject: [PATCH 43/50] feat(app): add userInfo to redux config (#15797) Closes EXEC-628 In the (near) future, apps will stake ownership in error recovery and perhaps later on, protocol runs. A basis for staking ownership requires some sort of unique identification, which we can do via UUIDs. Let's have clients generate their own UUID that's persisted in the electron config file. --- .../src/config/__fixtures__/index.ts | 12 ++++ .../src/config/__tests__/migrate.test.ts | 47 ++++++++---- app-shell-odd/src/config/migrate.ts | 17 ++++- app-shell/src/__fixtures__/config.ts | 12 ++++ .../src/config/__tests__/migrate.test.ts | 71 ++++++++++++------- app-shell/src/config/migrate.ts | 17 ++++- app/src/organisms/Devices/RobotOverview.tsx | 2 +- .../Devices/__tests__/RobotOverview.test.tsx | 2 +- .../redux/config/__tests__/selectors.test.ts | 16 +++++ app/src/redux/config/schema-types.ts | 9 ++- app/src/redux/config/selectors.ts | 5 ++ 11 files changed, 163 insertions(+), 47 deletions(-) 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/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/src/organisms/Devices/RobotOverview.tsx b/app/src/organisms/Devices/RobotOverview.tsx index 85ae49581cb..1e84891d524 100644 --- a/app/src/organisms/Devices/RobotOverview.tsx +++ b/app/src/organisms/Devices/RobotOverview.tsx @@ -66,7 +66,7 @@ export function RobotOverview({ const isRobotViewable = useIsRobotViewable(robot?.name ?? '') const { lightsOn, toggleLights } = useLights() - const userId = useSelector(getConfig)?.support?.userId ?? 'Opentrons-user' + const userId = useSelector(getConfig)?.userInfo?.userId ?? 'Opentrons-user' const addresses = useSelector((state: State) => getRobotAddressesByName(state, robot?.name ?? '') diff --git a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx index 66f6d18b7d0..6aaa236d49c 100644 --- a/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx +++ b/app/src/organisms/Devices/__tests__/RobotOverview.test.tsx @@ -151,7 +151,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({ 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 ?? '' +) From eaf9657becda810fc380d817f0b26e2efcdfabe5 Mon Sep 17 00:00:00 2001 From: Josh McVey Date: Thu, 25 Jul 2024 17:17:13 -0500 Subject: [PATCH 44/50] feat(ci): analyses snapshot skip pr on fail label (#15792) ## Overview - ~~Keep making the analyses snapshot test workflow more flexible allowing a label on the PR to prevent failure PRs from opening. Adding the label `no-analyses-snapshot-pr` on your PR will cause the workflow to skip opening a PR for snapshot updates if the snapshot tests fail.~~ - Added a step to diff the snapshots in the PR target branch against the PR branch. No action is taken based on the outcome. Purely informational. Will this information be noticed and used as a reminder to synchronize with the target branch? ### Updates from feedback - default `OPEN_PR_ON_FAILURE` to false - as is the case currently if the analyses snapshot test fails you will see a red X on that test and you can read the logs or download and view the HTML test report. - if you want to open a PR into your branch that would heal the snapshots, add `gen-analyses-snapshot-pr` label and a PR will get opened - create a specific step to handle PRs for the overnight scheduled test of edge. ### Risk No risk or change to application code, CI only --- .github/workflows/analyses-snapshot-test.yaml | 49 +++++++++++++++---- 1 file changed, 40 insertions(+), 9 deletions(-) diff --git a/.github/workflows/analyses-snapshot-test.yaml b/.github/workflows/analyses-snapshot-test.yaml index 86f6ccab31c..0fab6740f7d 100644 --- a/.github/workflows/analyses-snapshot-test.yaml +++ b/.github/workflows/analyses-snapshot-test.yaml @@ -23,8 +23,16 @@ 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' + types: + - opened #default + - synchronize #default + - reopened #default + - labeled concurrency: group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }} @@ -40,14 +48,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 +98,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}} From 2e11766b8e8b70b784eeaa03bc857b1e6403f952 Mon Sep 17 00:00:00 2001 From: Seth Foster Date: Fri, 26 Jul 2024 10:57:30 -0400 Subject: [PATCH 45/50] feat(api): Handle overpressures in AspirateInPlace, DispenseInPlace (#15791) Use DefinedErrorData for pipette overpressures in AspirateInPlace and DispenseInPlace. This is mostly similar to adding the same handling for Aspirate and Dispense, but the big difference is that these commands are not associated with wells or with points via their params. We get the current point by asking the hardware where it is; it's important to do this rather than getting it from the state because someone could conceivably have called AspirateInPlace/DispenseInPlace as the first command after a move command with a different pipette, and we wouldn't have the point in state. We don't get the current well at all. That's because this command isn't associated with a well, so there's not really anything to change. Closes EXEC-496 Closes EXEC-622 ## Testing - [x] Run a protocol that uses in place aspirate and dispense - [x] cause an overpressure error and make sure it now goes into error recovery - [x] make sure that error recovery moves back to the right place when retrying --- .../commands/aspirate_in_place.py | 84 +++++++++++++---- .../commands/dispense_in_place.py | 90 +++++++++++++++---- .../protocol_engine/state/pipettes.py | 7 +- .../protocol_engine/commands/conftest.py | 7 ++ .../commands/test_aspirate_in_place.py | 76 +++++++++++++++- .../commands/test_dispense_in_place.py | 76 +++++++++++++++- .../state/test_pipette_store.py | 58 ++++++++++++ 7 files changed, 358 insertions(+), 40 deletions(-) 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/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/state/pipettes.py b/api/src/opentrons/protocol_engine/state/pipettes.py index 35cfca94f33..e1dd00bbfb2 100644 --- a/api/src/opentrons/protocol_engine/state/pipettes.py +++ b/api/src/opentrons/protocol_engine/state/pipettes.py @@ -14,6 +14,8 @@ 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, @@ -413,7 +415,10 @@ def _update_deck_point( ) elif ( isinstance(action, FailCommandAction) - and isinstance(action.running_command, (Aspirate, Dispense)) + and isinstance( + action.running_command, + (Aspirate, Dispense, AspirateInPlace, DispenseInPlace), + ) and isinstance(action.error, DefinedErrorData) and isinstance(action.error.public, OverpressureError) ): 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_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_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/state/test_pipette_store.py b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py index 6296b30911f..6e7428719ec 100644 --- a/api/tests/opentrons/protocol_engine/state/test_pipette_store.py +++ b/api/tests/opentrons/protocol_engine/state/test_pipette_store.py @@ -970,6 +970,64 @@ def test_add_pipette_config( 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( From adb75f53b28cd65f7f9e8bdf3d0825b606a53b1a Mon Sep 17 00:00:00 2001 From: Derek Maggio Date: Fri, 26 Jul 2024 08:12:45 -0700 Subject: [PATCH 46/50] test(analysis-snapshot-testing): add 2.19 smoke test protocols (#15805) --- ...S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py | 654 ++++++++++++++++++ ...S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py | 605 ++++++++++++++++ 2 files changed, 1259 insertions(+) create mode 100644 analyses-snapshot-testing/files/protocols/Flex_S_v2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke.py create mode 100644 analyses-snapshot-testing/files/protocols/OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3.py 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() From 5615d1ef0cb3047723d2a45ebc2739412d35d985 Mon Sep 17 00:00:00 2001 From: Jethary Rader <66035149+jerader@users.noreply.github.com> Date: Fri, 26 Jul 2024 12:06:56 -0400 Subject: [PATCH 47/50] fix(protocol-designer): fix well depth for blowout field (#15809) closes RESC-309 --- .../StepEditForm/fields/BlowoutZOffsetField.tsx | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) 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} Date: Fri, 26 Jul 2024 11:09:49 -0500 Subject: [PATCH 48/50] fix(analyses-snapshot-testing): trigger test run on test library change (#15807) # Overview Changes to `analyses-snapshot-testing` should trigger the tests to run. # Test Plan - [x] ~~due to merge of #15805 test should fail~~ doesn't fail because I forgot how file addition works - [x] trigger update PR with the label (added during PR creation) - [x] added the files to the test but no snapshot so the test should fail - [x] merge the snapshot heal - [x] see the test pass --- .github/workflows/analyses-snapshot-test.yaml | 2 + .../automation/data/protocols.py | 12 + ...2_19_P300M_P20S_HS_TC_TM_SmokeTestV3].json | 9684 +++++++++++ ...2_19_P1000_96_GRIP_HS_MB_TC_TM_Smoke].json | 13236 ++++++++++++++++ 4 files changed, 22934 insertions(+) create mode 100644 analyses-snapshot-testing/tests/__snapshots__/analyses_snapshot_test/test_analysis_snapshot[057de2035d][OT2_S_v2_19_P300M_P20S_HS_TC_TM_SmokeTestV3].json create mode 100644 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 diff --git a/.github/workflows/analyses-snapshot-test.yaml b/.github/workflows/analyses-snapshot-test.yaml index 0fab6740f7d..26576899188 100644 --- a/.github/workflows/analyses-snapshot-test.yaml +++ b/.github/workflows/analyses-snapshot-test.yaml @@ -28,6 +28,8 @@ on: - 'shared-data/**/*' - '!shared-data/js/**' - '.github/workflows/analyses-snapshot-test.yaml' + - 'analyses-snapshot-testing/**' + types: - opened #default - synchronize #default 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/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" + } + ] +} From 13a9f5899859e3947b13aa7ae7e7b736f0559704 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Fri, 26 Jul 2024 14:51:11 -0400 Subject: [PATCH 49/50] feat(app): add app support for /clientData endpoints (#15811) Closes EXEC-630 Adds HTTP and MQTT support for GET /clientData/{key} and HTTP support for PUT /clientData/{key}. --- api-client/src/client_data/getClientData.ts | 12 +++++ api-client/src/client_data/index.ts | 8 ++++ api-client/src/client_data/types.ts | 9 ++++ .../src/client_data/updateClientData.ts | 22 +++++++++ api-client/src/index.ts | 1 + app/src/redux/shell/types.ts | 1 + app/src/resources/client_data/constants.ts | 9 ++++ app/src/resources/client_data/index.ts | 1 + .../resources/client_data/recovery/index.ts | 4 ++ .../resources/client_data/recovery/types.ts | 7 +++ .../recovery/useClientDataRecovery.ts | 22 +++++++++ .../recovery/useNotifyClientDataRecovery.ts | 32 +++++++++++++ .../recovery/useUpdateClientDataRecovery.ts | 47 ++++++++++++++++++ react-api-client/src/client_data/index.ts | 4 ++ .../src/client_data/useClientData.ts | 27 +++++++++++ .../src/client_data/useUpdateClientData.ts | 48 +++++++++++++++++++ react-api-client/src/index.ts | 1 + 17 files changed, 255 insertions(+) create mode 100644 api-client/src/client_data/getClientData.ts create mode 100644 api-client/src/client_data/index.ts create mode 100644 api-client/src/client_data/types.ts create mode 100644 api-client/src/client_data/updateClientData.ts create mode 100644 app/src/resources/client_data/constants.ts create mode 100644 app/src/resources/client_data/index.ts create mode 100644 app/src/resources/client_data/recovery/index.ts create mode 100644 app/src/resources/client_data/recovery/types.ts create mode 100644 app/src/resources/client_data/recovery/useClientDataRecovery.ts create mode 100644 app/src/resources/client_data/recovery/useNotifyClientDataRecovery.ts create mode 100644 app/src/resources/client_data/recovery/useUpdateClientDataRecovery.ts create mode 100644 react-api-client/src/client_data/index.ts create mode 100644 react-api-client/src/client_data/useClientData.ts create mode 100644 react-api-client/src/client_data/useUpdateClientData.ts 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/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/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..a0ab38ded1b --- /dev/null +++ b/app/src/resources/client_data/recovery/useUpdateClientDataRecovery.ts @@ -0,0 +1,47 @@ +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'] + ) => ReturnType< + UseUpdateClientDataMutationResult['updateClientData'] + > +} + +// 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'] + ): ReturnType< + UseUpdateClientDataMutationResult['updateClientData'] + > => { + updateClientData({ userId: thisUserId, intent }) + } + + return { + ...mutate, + updateWithIntent, + } +} 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/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' From 2f590fe16f52e960e295c1cb12ff33569cceeae4 Mon Sep 17 00:00:00 2001 From: koji Date: Fri, 26 Jul 2024 15:41:46 -0400 Subject: [PATCH 50/50] refactor(app): add id prop for a case that there are 2 same label (#15814) * refactor(app): add id prop for a case that there are 2 same label --- app/src/atoms/buttons/RadioButton.tsx | 6 +++-- .../buttons/__tests__/RadioButton.test.tsx | 25 ++++++++++++++++++- 2 files changed, 28 insertions(+), 3 deletions(-) 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() + }) })