From bd1f8da7ad28309df7847106188f467c074a1ebe Mon Sep 17 00:00:00 2001 From: Jethary Rader <66035149+jerader@users.noreply.github.com> Date: Wed, 24 Apr 2024 22:16:26 -0400 Subject: [PATCH] feat(step-generation, shared-data): pipette collision warnings (#14989) closes AUTH-19 --- .../__tests__/getFlexSurroundingSlots.test.ts | 30 ++ .../js/helpers/getFlexSurroundingSlots.ts | 63 +++ shared-data/js/helpers/index.ts | 1 + .../getIsSafePipetteMovement.test.ts | 167 ++++++++ .../ninetySixChannelCollision.test.ts | 146 ------- .../src/commandCreators/atomic/replaceTip.ts | 17 +- .../commandCreators/compound/consolidate.ts | 82 ++-- .../commandCreators/compound/distribute.ts | 73 ++-- .../src/commandCreators/compound/mix.ts | 34 +- .../src/commandCreators/compound/transfer.ts | 75 ++-- step-generation/src/errorCreators.ts | 10 +- step-generation/src/types.ts | 2 +- step-generation/src/utils/index.ts | 2 +- .../src/utils/ninetySixChannelCollision.ts | 78 ---- .../src/utils/safePipetteMovements.ts | 376 ++++++++++++++++++ 15 files changed, 768 insertions(+), 388 deletions(-) create mode 100644 shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts create mode 100644 shared-data/js/helpers/getFlexSurroundingSlots.ts create mode 100644 step-generation/src/__tests__/getIsSafePipetteMovement.test.ts delete mode 100644 step-generation/src/__tests__/ninetySixChannelCollision.test.ts delete mode 100644 step-generation/src/utils/ninetySixChannelCollision.ts create mode 100644 step-generation/src/utils/safePipetteMovements.ts diff --git a/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts b/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts new file mode 100644 index 00000000000..a91d2f737c5 --- /dev/null +++ b/shared-data/js/helpers/__tests__/getFlexSurroundingSlots.test.ts @@ -0,0 +1,30 @@ +import { describe, it, expect } from 'vitest' +import { getFlexSurroundingSlots } from '../getFlexSurroundingSlots' + +describe('getFlexSurroundingSlots', () => { + it('returns slots when slot is D2', () => { + const results = getFlexSurroundingSlots('D2', []) + expect(results).toStrictEqual(['C1', 'C2', 'C3', 'D1', 'D3']) + }) + it('returns slots when selected is a center slot', () => { + const results = getFlexSurroundingSlots('C2', []) + expect(results).toStrictEqual([ + 'B1', + 'B2', + 'B3', + 'C1', + 'C3', + 'D1', + 'D2', + 'D3', + ]) + }) + it('returns slots when selected is a column 3 with staging areas present', () => { + const results = getFlexSurroundingSlots('B3', ['A4']) + expect(results).toStrictEqual(['A2', 'A3', 'A4', 'B2', 'C2', 'C3']) + }) + it('returns slots when selected is a corner, A1', () => { + const results = getFlexSurroundingSlots('A1', ['A4']) + expect(results).toStrictEqual(['A2', 'B1', 'B2']) + }) +}) diff --git a/shared-data/js/helpers/getFlexSurroundingSlots.ts b/shared-data/js/helpers/getFlexSurroundingSlots.ts new file mode 100644 index 00000000000..9900cee9880 --- /dev/null +++ b/shared-data/js/helpers/getFlexSurroundingSlots.ts @@ -0,0 +1,63 @@ +import type { DeckSlotId } from '../types' + +const FLEX_GRID = [ + ['A1', 'A2', 'A3'], + ['B1', 'B2', 'B3'], + ['C1', 'C2', 'C3'], + ['D1', 'D2', 'D3'], +] + +const LETTER_TO_ROW_MAP: Record = { + A: 0, + B: 1, + C: 2, + D: 3, +} + +let COLS = 3 // Initial number of columns in each row +const ROWS = 4 + +const DIRECTIONS = [ + [-1, -1], // NW + [-1, 0], // N + [-1, 1], // NE + [0, -1], // W + [0, 1], // E + [1, -1], // SW + [1, 0], // S + [1, 1], // SE +] + +export const getFlexSurroundingSlots = ( + slot: DeckSlotId, + stagingAreaSlots: DeckSlotId[] +): DeckSlotId[] => { + // Handle staging area slots + if (stagingAreaSlots.length > 0) { + stagingAreaSlots.forEach((stagingSlot, index) => { + if (stagingSlot) { + FLEX_GRID[index].push(stagingSlot) + } + }) + COLS = Math.max(COLS, FLEX_GRID[0].length) // Update COLS to the maximum row length + } + + const letter = slot.charAt(0) + const col = parseInt(slot.charAt(1)) - 1 // Convert the column to a 0-based index + const row = LETTER_TO_ROW_MAP[letter] + + const surroundingSlots: DeckSlotId[] = [] + + // Iterate through both directions + DIRECTIONS.forEach(([dRow, dCol]) => { + const newRow = row + dRow + const newCol = col + dCol + + if (newRow >= 0 && newRow < ROWS && newCol >= 0 && newCol < COLS) { + surroundingSlots.push(FLEX_GRID[newRow][newCol]) + } + }) + + // Filter out any undefined values from the staging area slots that are not added + return surroundingSlots.filter(slot => slot !== undefined) +} diff --git a/shared-data/js/helpers/index.ts b/shared-data/js/helpers/index.ts index a07d10472f6..791fa1f5db1 100644 --- a/shared-data/js/helpers/index.ts +++ b/shared-data/js/helpers/index.ts @@ -27,6 +27,7 @@ export * from './getLoadedLabwareDefinitionsByUri' export * from './getOccludedSlotCountForModule' export * from './labwareInference' export * from './getAddressableAreasInProtocol' +export * from './getFlexSurroundingSlots' export * from './getSimplestFlexDeckConfig' export * from './formatRunTimeParameterDefaultValue' export * from './formatRunTimeParameterValue' diff --git a/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts new file mode 100644 index 00000000000..b0d40489178 --- /dev/null +++ b/step-generation/src/__tests__/getIsSafePipetteMovement.test.ts @@ -0,0 +1,167 @@ +import { expect, describe, it } from 'vitest' +import { getIsSafePipetteMovement } from '../utils' +import { + LabwareDefinition2, + TEMPERATURE_MODULE_TYPE, + TEMPERATURE_MODULE_V2, + fixture96Plate, + fixtureP100096V2Specs, + fixtureTiprack1000ul, + fixtureTiprackAdapter, +} from '@opentrons/shared-data' +import { InvariantContext, RobotState } from '../types' + +const mockLabwareId = 'labwareId' +const mockPipId = 'pip' +const mockTiprackId = 'tiprackId' +const mockModule = 'moduleId' +const mockLabware2 = 'labwareId2' +const mockAdapter = 'adapterId' +const mockInvariantProperties: InvariantContext = { + pipetteEntities: { + pip: { + name: 'p1000_96', + id: 'pip', + tiprackDefURI: ['mockDefUri'], + tiprackLabwareDef: [fixtureTiprack1000ul as LabwareDefinition2], + spec: fixtureP100096V2Specs, + }, + }, + labwareEntities: { + [mockLabwareId]: { + id: mockLabwareId, + labwareDefURI: 'mockDefUri', + def: fixture96Plate as LabwareDefinition2, + }, + [mockTiprackId]: { + id: mockTiprackId, + labwareDefURI: 'mockTipUri', + def: fixtureTiprack1000ul as LabwareDefinition2, + }, + [mockAdapter]: { + id: mockAdapter, + labwareDefURI: 'mockAdapterUri', + def: fixtureTiprackAdapter as LabwareDefinition2, + }, + [mockLabware2]: { + id: mockLabware2, + labwareDefURI: 'mockDefUri', + def: fixture96Plate as LabwareDefinition2, + }, + }, + moduleEntities: {}, + additionalEquipmentEntities: {}, + config: { + OT_PD_DISABLE_MODULE_RESTRICTIONS: false, + }, +} + +const mockRobotState: RobotState = { + pipettes: { pip: { mount: 'left' } }, + labware: { [mockLabwareId]: { slot: 'D2' }, [mockTiprackId]: { slot: 'A2' } }, + modules: {}, + tipState: { tipracks: {}, pipettes: {} }, + liquidState: { pipettes: {}, labware: {}, additionalEquipment: {} }, +} +describe('getIsSafePipetteMovement', () => { + it('returns true when the labware id is a trash bin', () => { + const result = getIsSafePipetteMovement( + { + labware: {}, + pipettes: {}, + modules: {}, + tipState: {}, + liquidState: {}, + } as any, + { + labwareEntities: {}, + pipetteEntities: {}, + moduleEntities: {}, + additionalEquipmentEntities: { + trashBin: { name: 'trashBin', location: 'A3', id: 'trashBin' }, + }, + config: {} as any, + }, + 'mockId', + 'mockTrashBin', + 'mockTiprackId', + { x: 0, y: 0, z: 0 } + ) + expect(result).toEqual(true) + }) + it('returns false when within pipette extents is false', () => { + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -12, y: -100, z: 20 } + ) + expect(result).toEqual(false) + }) + it('returns true when there are no collisions and a module near it', () => { + mockRobotState.modules = { + [mockModule]: { slot: 'D1', moduleState: {} as any }, + } + mockInvariantProperties.moduleEntities = { + [mockModule]: { + id: mockModule, + type: TEMPERATURE_MODULE_TYPE, + model: TEMPERATURE_MODULE_V2, + }, + } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -1, y: 5, z: 20 } + ) + expect(result).toEqual(true) + }) + it('returns false when there is a tip that collides', () => { + mockRobotState.tipState.tipracks = { mockTiprackId: { A1: true } } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: -1, y: 5, z: 0 } + ) + expect(result).toEqual(false) + }) + it('returns false when there is a tall module nearby in a diagonal slot with adapter and labware', () => { + mockRobotState.modules = { + [mockModule]: { slot: 'C1', moduleState: {} as any }, + } + mockRobotState.labware = { + [mockLabwareId]: { slot: 'D2' }, + [mockAdapter]: { + slot: mockModule, + }, + [mockLabware2]: { + slot: mockAdapter, + }, + } + mockInvariantProperties.moduleEntities = { + [mockModule]: { + id: mockModule, + type: TEMPERATURE_MODULE_TYPE, + model: TEMPERATURE_MODULE_V2, + }, + } + const result = getIsSafePipetteMovement( + mockRobotState, + mockInvariantProperties, + mockPipId, + mockLabwareId, + mockTiprackId, + { x: 0, y: 0, z: 0 } + ) + expect(result).toEqual(false) + }) + // todo(jr, 4/23/24): add more test cases, test thermocycler collision - i'll do this in a follow up +}) diff --git a/step-generation/src/__tests__/ninetySixChannelCollision.test.ts b/step-generation/src/__tests__/ninetySixChannelCollision.test.ts deleted file mode 100644 index aae8c8acab9..00000000000 --- a/step-generation/src/__tests__/ninetySixChannelCollision.test.ts +++ /dev/null @@ -1,146 +0,0 @@ -import { beforeEach, describe, it, expect } from 'vitest' -import { getIsTallLabwareWestOf96Channel } from '../utils/ninetySixChannelCollision' -import type { LabwareDefinition2 } from '@opentrons/shared-data' -import type { RobotState, InvariantContext } from '../types' - -let invariantContext: InvariantContext -let robotState: RobotState - -const mockSourceId = 'sourceId' -const mockWestId = 'westId' -const mockPipetteId = 'pipetteId' -const mockTiprackId = 'tiprackId' -const mockSourceDef: LabwareDefinition2 = { - dimensions: { zDimension: 100 }, -} as any -const mockWestDef: LabwareDefinition2 = { - dimensions: { zDimension: 90 }, -} as any -const mockWestDefTall: LabwareDefinition2 = { - dimensions: { zDimension: 101 }, -} as any -const mockTiprackDefinition: LabwareDefinition2 = { - parameters: { tipLength: 10 }, -} as any -describe('getIsTallLabwareWestOf96Channel ', () => { - beforeEach(() => { - invariantContext = { - labwareEntities: { - [mockSourceId]: { - id: mockSourceId, - labwareDefURI: 'mockDefUri', - def: mockSourceDef, - }, - }, - additionalEquipmentEntities: {}, - moduleEntities: {}, - config: {} as any, - pipetteEntities: { - [mockPipetteId]: { - name: 'p1000_96', - id: mockPipetteId, - tiprackDefURI: ['mockUri'], - tiprackLabwareDef: [mockTiprackDefinition], - spec: {} as any, - }, - }, - } - robotState = { - labware: { [mockSourceId]: { slot: 'A1' } }, - pipettes: {}, - modules: {}, - tipState: { pipettes: { [mockPipetteId]: false } } as any, - liquidState: {} as any, - } - }) - it('should return false when the slot is in column is 1', () => { - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when source id is a waste chute', () => { - invariantContext = { - ...invariantContext, - additionalEquipmentEntities: { - [mockSourceId]: { - id: mockSourceId, - name: 'wasteChute', - location: 'D3', - }, - }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when there is no labware west of source labware', () => { - robotState.labware = { [mockSourceId]: { slot: 'A2' } } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return false when the west labware height is not tall enough', () => { - invariantContext.labwareEntities = { - ...invariantContext.labwareEntities, - [mockWestId]: { - id: mockWestId, - labwareDefURI: 'mockDefUri', - def: mockWestDef, - }, - } - robotState.labware = { - [mockSourceId]: { slot: 'A2' }, - [mockWestId]: { slot: 'A1' }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(false) - }) - it('should return true when the west labware height is tall enough', () => { - invariantContext.labwareEntities = { - ...invariantContext.labwareEntities, - [mockWestId]: { - id: mockWestId, - labwareDefURI: 'mockDefUri', - def: mockWestDefTall, - }, - } - robotState.labware = { - [mockSourceId]: { slot: 'A2' }, - [mockWestId]: { slot: 'A1' }, - } - expect( - getIsTallLabwareWestOf96Channel( - robotState, - invariantContext, - mockSourceId, - mockPipetteId, - mockTiprackId - ) - ).toBe(true) - }) -}) diff --git a/step-generation/src/commandCreators/atomic/replaceTip.ts b/step-generation/src/commandCreators/atomic/replaceTip.ts index 85160be713c..7aae3b98be1 100644 --- a/step-generation/src/commandCreators/atomic/replaceTip.ts +++ b/step-generation/src/commandCreators/atomic/replaceTip.ts @@ -7,7 +7,7 @@ import { curryCommandCreator, getIsHeaterShakerEastWestMultiChannelPipette, getIsHeaterShakerEastWestWithLatchOpen, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getLabwareSlot, modulePipetteCollision, pipetteAdjacentHeaterShakerWhileShaking, @@ -160,23 +160,18 @@ export const replaceTip: CommandCreator = ( if ( channels === 96 && nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, nextTiprack.tiprackId, pipette, - tipRack + tipRack, + // we don't adjust the offset when moving to the tiprack + { x: 0, y: 0 } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'tiprack', - labware: - invariantContext.labwareEntities[nextTiprack.tiprackId].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } diff --git a/step-generation/src/commandCreators/compound/consolidate.ts b/step-generation/src/commandCreators/compound/consolidate.ts index b37f2ede1b0..f7fc4c85f9d 100644 --- a/step-generation/src/commandCreators/compound/consolidate.ts +++ b/step-generation/src/commandCreators/compound/consolidate.ts @@ -18,7 +18,7 @@ import { airGapHelper, dispenseLocationHelper, moveHelper, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -56,6 +56,31 @@ export const consolidate: CommandCreator = ( * 'once': get a new tip at the beginning of the consolidate step, and use it throughout * 'never': reuse the tip from the last step */ + + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + blowoutFlowRateUlSec, + blowoutOffsetFromTopMm, + dispenseAirGapVolume, + dispenseDelay, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + mixFirstAspirate, + mixInDestination, + dropTipLocation, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + const actionName = 'consolidate' const pipetteData = prevRobotState.pipettes[args.pipette] const is96Channel = @@ -91,72 +116,37 @@ export const consolidate: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) ) { return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }), - ], + errors: [errorCreators.possiblePipetteCollision()], } } - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - blowoutFlowRateUlSec, - blowoutOffsetFromTopMm, - dispenseAirGapVolume, - dispenseDelay, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - mixFirstAspirate, - mixInDestination, - dropTipLocation, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const maxWellsPerChunk = Math.floor( getPipetteWithTipMaxVol(args.pipette, invariantContext, args.tipRack) / diff --git a/step-generation/src/commandCreators/compound/distribute.ts b/step-generation/src/commandCreators/compound/distribute.ts index 520ce06aeb4..eae11c1452f 100644 --- a/step-generation/src/commandCreators/compound/distribute.ts +++ b/step-generation/src/commandCreators/compound/distribute.ts @@ -16,7 +16,7 @@ import { blowoutUtil, wasteChuteCommandsUtil, getDispenseAirGapLocation, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -53,6 +53,26 @@ export const distribute: CommandCreator = ( * 'once': get a new tip at the beginning of the distribute step, and use it throughout * 'never': reuse the tip from the last step */ + + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + dispenseDelay, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + blowoutLocation, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + // TODO Ian 2018-05-03 next ~20 lines match consolidate.js const actionName = 'distribute' const errors: CommandCreatorError[] = [] @@ -91,67 +111,38 @@ export const distribute: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if (errors.length > 0) return { errors, } - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - dispenseDelay, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - blowoutLocation, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args + const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const dispenseAirGapVolume = args.dispenseAirGapVolume || 0 // TODO error on negative args.disposalVolume? diff --git a/step-generation/src/commandCreators/compound/mix.ts b/step-generation/src/commandCreators/compound/mix.ts index 284529c7c1f..734be8c1a39 100644 --- a/step-generation/src/commandCreators/compound/mix.ts +++ b/step-generation/src/commandCreators/compound/mix.ts @@ -5,7 +5,7 @@ import { blowoutUtil, curryCommandCreator, reduceCommandCreators, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, } from '../../utils' import * as errorCreators from '../../errorCreators' import { @@ -178,25 +178,29 @@ export const mix: CommandCreator = ( return { errors: [errorCreators.dropTipLocationDoesNotExist()] } } - if ( - is96Channel && - data.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + console.log(invariantContext.pipetteEntities[pipette]) + + if (is96Channel && data.nozzles === COLUMN) { + const isAspirateSafePipetteMovement = getIsSafePipetteMovement( prevRobotState, invariantContext, + pipette, labware, + tipRack, + { x: aspirateXOffset, y: aspirateYOffset } + ) + const isDispenseSafePipetteMovement = getIsSafePipetteMovement( + prevRobotState, + invariantContext, pipette, - tipRack + labware, + tipRack, + { x: dispenseXOffset, y: dispenseYOffset } ) - ) { - return { - errors: [ - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'mix', - labware: - invariantContext.labwareEntities[labware].def.metadata.displayName, - }), - ], + if (!isAspirateSafePipetteMovement && !isDispenseSafePipetteMovement) { + return { + errors: [errorCreators.possiblePipetteCollision()], + } } } const stateNozzles = prevRobotState.pipettes[pipette].nozzles diff --git a/step-generation/src/commandCreators/compound/transfer.ts b/step-generation/src/commandCreators/compound/transfer.ts index 2d16c8064bf..9c59d301aa4 100644 --- a/step-generation/src/commandCreators/compound/transfer.ts +++ b/step-generation/src/commandCreators/compound/transfer.ts @@ -18,7 +18,7 @@ import { getTrashOrLabware, dispenseLocationHelper, moveHelper, - getIsTallLabwareWestOf96Channel, + getIsSafePipetteMovement, getWasteChuteAddressableAreaNamePip, } from '../../utils' import { @@ -63,6 +63,27 @@ export const transfer: CommandCreator = ( NOTE: In some situations, different changeTip options have equivalent outcomes. That's OK. */ + // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound + // to the action of aspiration of dispensing in a given command, they are actually values bound + // to a given labware associated with a command (e.g. Source, Destination). For this reason we + // currently remapping the inner mix values. Those calls to mixUtil should become easier to read + // when we decide to rename these fields/args... probably all the way up to the UI level. + const { + aspirateDelay, + dispenseDelay, + aspirateFlowRateUlSec, + aspirateOffsetFromBottomMm, + blowoutFlowRateUlSec, + blowoutOffsetFromTopMm, + dispenseFlowRateUlSec, + dispenseOffsetFromBottomMm, + tipRack, + aspirateXOffset, + aspirateYOffset, + dispenseXOffset, + dispenseYOffset, + } = args + const trashOrLabware = getTrashOrLabware( invariantContext.labwareEntities, invariantContext.additionalEquipmentEntities, @@ -130,43 +151,31 @@ export const transfer: CommandCreator = ( if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.sourceLabware, args.pipette, - args.tipRack + args.sourceLabware, + args.tipRack, + { x: aspirateXOffset, y: aspirateYOffset, z: aspirateOffsetFromBottomMm } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'aspirate', - labware: - invariantContext.labwareEntities[args.sourceLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if ( is96Channel && args.nozzles === COLUMN && - getIsTallLabwareWestOf96Channel( + !getIsSafePipetteMovement( prevRobotState, invariantContext, - args.destLabware, args.pipette, - args.tipRack + args.destLabware, + args.tipRack, + { x: dispenseXOffset, y: dispenseYOffset, z: dispenseOffsetFromBottomMm } ) ) { - errors.push( - errorCreators.tallLabwareWestOf96ChannelPipetteLabware({ - source: 'dispense', - labware: - invariantContext.labwareEntities[args.destLabware].def.metadata - .displayName, - }) - ) + errors.push(errorCreators.possiblePipetteCollision()) } if (errors.length > 0) @@ -190,26 +199,6 @@ export const transfer: CommandCreator = ( pipetteSpec.channels ) - // TODO: BC 2019-07-08 these argument names are a bit misleading, instead of being values bound - // to the action of aspiration of dispensing in a given command, they are actually values bound - // to a given labware associated with a command (e.g. Source, Destination). For this reason we - // currently remapping the inner mix values. Those calls to mixUtil should become easier to read - // when we decide to rename these fields/args... probably all the way up to the UI level. - const { - aspirateDelay, - dispenseDelay, - aspirateFlowRateUlSec, - aspirateOffsetFromBottomMm, - blowoutFlowRateUlSec, - blowoutOffsetFromTopMm, - dispenseFlowRateUlSec, - dispenseOffsetFromBottomMm, - tipRack, - aspirateXOffset, - aspirateYOffset, - dispenseXOffset, - dispenseYOffset, - } = args const aspirateAirGapVolume = args.aspirateAirGapVolume || 0 const dispenseAirGapVolume = args.dispenseAirGapVolume || 0 const effectiveTransferVol = diff --git a/step-generation/src/errorCreators.ts b/step-generation/src/errorCreators.ts index 50a271effe0..581b04d72f9 100644 --- a/step-generation/src/errorCreators.ts +++ b/step-generation/src/errorCreators.ts @@ -175,13 +175,11 @@ export const tallLabwareEastWestOfHeaterShaker = ( } } -export const tallLabwareWestOf96ChannelPipetteLabware = (args: { - source: string - labware: string -}): CommandCreatorError => { +export const possiblePipetteCollision = (): CommandCreatorError => { return { - type: 'TALL_LABWARE_WEST_OF_96_CHANNEL_LABWARE', - message: `Labware to the left of the ${args.source} ${args.labware} is too tall and will collide with the 96-channel.`, + type: 'POSSIBLE_PIPETTE_COLLISION', + message: + 'There is a possibility that the Pipette will collide with the a labware or module on the deck', } } diff --git a/step-generation/src/types.ts b/step-generation/src/types.ts index 6cef80c43ed..e63360a3f27 100644 --- a/step-generation/src/types.ts +++ b/step-generation/src/types.ts @@ -539,9 +539,9 @@ export type ErrorType = | 'PIPETTE_HAS_TIP' | 'PIPETTE_VOLUME_EXCEEDED' | 'PIPETTING_INTO_COLUMN_4' + | 'POSSIBLE_PIPETTE_COLLISION' | 'REMOVE_96_CHANNEL_TIPRACK_ADAPTER' | 'TALL_LABWARE_EAST_WEST_OF_HEATER_SHAKER' - | 'TALL_LABWARE_WEST_OF_96_CHANNEL_LABWARE' | 'THERMOCYCLER_LID_CLOSED' | 'TIP_VOLUME_EXCEEDED' diff --git a/step-generation/src/utils/index.ts b/step-generation/src/utils/index.ts index ac363cbcd97..9c8ab222c57 100644 --- a/step-generation/src/utils/index.ts +++ b/step-generation/src/utils/index.ts @@ -20,6 +20,6 @@ export * from './commandCreatorArgsGetters' export * from './heaterShakerCollision' export * from './misc' export * from './movableTrashCommandsUtil' -export * from './ninetySixChannelCollision' +export * from './safePipetteMovements' export * from './wasteChuteCommandsUtil' export const uuid: () => string = uuidv4 diff --git a/step-generation/src/utils/ninetySixChannelCollision.ts b/step-generation/src/utils/ninetySixChannelCollision.ts deleted file mode 100644 index 7a2b7f3e0c1..00000000000 --- a/step-generation/src/utils/ninetySixChannelCollision.ts +++ /dev/null @@ -1,78 +0,0 @@ -import toNumber from 'lodash/toNumber' -import { getModuleDef2 } from '@opentrons/shared-data' -import type { RobotState, InvariantContext } from '../types' - -const SAFETY_MARGIN = 10 -const targetNumbers = ['2', '3', '4'] - -export const getIsTallLabwareWestOf96Channel = ( - robotState: RobotState, - invariantContext: InvariantContext, - sourceLabwareId: string, - pipetteId: string, - tipRackId: string -): boolean => { - const { labwareEntities, additionalEquipmentEntities } = invariantContext - const { labware: labwareState, tipState } = robotState - const pipetteHasTip = tipState.pipettes[pipetteId] - const tipLength = pipetteHasTip - ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 - : 0 - // early exit if source labware is the waste chute or trash bin - if (additionalEquipmentEntities[sourceLabwareId] != null) { - return false - } - - const labwareSlot = labwareState[sourceLabwareId].slot - const letter = labwareSlot.charAt(0) - const number = labwareSlot.charAt(1) - - if (targetNumbers.includes(number)) { - const westNumber = toNumber(number) - 1 - const westSlot = letter + westNumber - - const westLabwareState = Object.entries(labwareState).find( - ([id, labware]) => labware.slot === westSlot - ) - if (westLabwareState != null) { - const westLabwareId = westLabwareState[0] - if (labwareEntities[westLabwareId] == null) { - console.error( - `expected to find labware west of source labware but could not, with labware id ${westLabwareId}` - ) - } - if (labwareEntities[westLabwareId] != null) { - const westLabwareHeight = - labwareEntities[westLabwareId].def.dimensions.zDimension - const westLabwareSlot = robotState.labware[westLabwareId].slot - let adapterHeight: number = 0 - let moduleHeight: number = 0 - // if labware is on an adapter + or on an adapter + module - if (robotState.labware[westLabwareSlot] != null) { - const adapterSlot = robotState.labware[westLabwareSlot]?.slot - adapterHeight = - invariantContext.labwareEntities[westLabwareSlot]?.def.dimensions - .zDimension - const moduleModel = - invariantContext.moduleEntities[adapterSlot]?.model - const moduleDimensions = - moduleModel != null ? getModuleDef2(moduleModel)?.dimensions : null - moduleHeight = - moduleDimensions != null ? moduleDimensions.bareOverallHeight : 0 - // if labware is on a module - } else if (invariantContext.moduleEntities[westLabwareSlot] != null) { - const moduleModel = - invariantContext.moduleEntities[westLabwareSlot].model - moduleHeight = getModuleDef2(moduleModel).dimensions.bareOverallHeight - } - const totalHighestZ = westLabwareHeight + adapterHeight + moduleHeight - const sourceLabwareHeight = - labwareEntities[sourceLabwareId].def.dimensions.zDimension - - return totalHighestZ + SAFETY_MARGIN > sourceLabwareHeight + tipLength - } - } - } - - return false -} diff --git a/step-generation/src/utils/safePipetteMovements.ts b/step-generation/src/utils/safePipetteMovements.ts new file mode 100644 index 00000000000..ea1d7d0cadc --- /dev/null +++ b/step-generation/src/utils/safePipetteMovements.ts @@ -0,0 +1,376 @@ +import { + FLEX_ROBOT_TYPE, + THERMOCYCLER_MODULE_TYPE, + getAddressableAreaFromSlotId, + getDeckDefFromRobotType, + getFlexSurroundingSlots, + getModuleDef2, + getPositionFromSlotId, +} from '@opentrons/shared-data' +import type { + AddressableArea, + CoordinateTuple, + NozzleConfigurationStyle, +} from '@opentrons/shared-data' +import type { + RobotState, + InvariantContext, + PipetteEntity, + ModuleEntities, + LabwareEntity, +} from '../types' + +const A12_column_front_left_bound = { x: -11.03, y: 2 } +const A12_column_back_right_bound = { x: 526.77, y: 506.2 } +const PRIMARY_NOZZLE = 'A12' +const NOZZLE_CONFIGURATION = 'COLUMN' +const FLEX_TC_LID_COLLISION_ZONE = { + back_left: { x: -43.25, y: 454.9, z: 211.91 }, + front_right: { x: 128.75, y: 402, z: 211.91 }, +} +const FLEX_TC_LID_BACK_LEFT_PT = { + x: FLEX_TC_LID_COLLISION_ZONE.back_left.x, + y: FLEX_TC_LID_COLLISION_ZONE.back_left.y, + z: FLEX_TC_LID_COLLISION_ZONE.back_left.z, +} + +const FLEX_TC_LID_FRONT_RIGHT_PT = { + x: FLEX_TC_LID_COLLISION_ZONE.front_right.x, + y: FLEX_TC_LID_COLLISION_ZONE.front_right.y, + z: FLEX_TC_LID_COLLISION_ZONE.front_right.z, +} + +interface SlotInfo { + addressableArea: AddressableArea | null + position: CoordinateTuple | null +} +interface Point { + x: number + y: number + z?: number +} + +// check if nozzle(s) are inbounds +const getIsWithinPipetteExtents = ( + location: Point, + nozzleConfiguration: NozzleConfigurationStyle, + primaryNozzle: string +): boolean => { + if (nozzleConfiguration === 'COLUMN' && primaryNozzle === 'A12') { + const isWithinBounds = + A12_column_front_left_bound.x <= location.x && + location.x <= A12_column_back_right_bound.x && + A12_column_front_left_bound.y <= location.y && + location.y <= A12_column_back_right_bound.y + + return isWithinBounds + } else { + // TODO: Handle other configurations such as 8-channel partial tip, and eventually all pipettes. + return true + } +} + +// return pipette bounds at a sepcific position +const getPipetteBoundsAtSpecifiedMoveToPosition = ( + pipetteEntity: PipetteEntity, + tipLength: number, + destinationPosition: Point +): Point[] => { + const primaryNozzleOffset = + pipetteEntity.spec.nozzleMap != null + ? pipetteEntity.spec.nozzleMap.A1 + : pipetteEntity.spec.nozzleOffset + const primaryNozzlePosition = { + x: destinationPosition.x, + y: destinationPosition.y, + z: (destinationPosition.z ?? 0) + tipLength, + } + const pipetteBoundsOffsets = pipetteEntity.spec.pipetteBoundingBoxOffsets + const backLeftBound = { + x: + primaryNozzlePosition.x - + primaryNozzleOffset[0] + + pipetteBoundsOffsets.backLeftCorner[0], + y: + primaryNozzlePosition.y - + primaryNozzleOffset[1] + + pipetteBoundsOffsets.backLeftCorner[1], + z: + primaryNozzlePosition.z - + primaryNozzleOffset[2] + + pipetteBoundsOffsets.backLeftCorner[2], + } + const frontRightBound = { + x: + primaryNozzlePosition.x - + primaryNozzleOffset[0] + + pipetteBoundsOffsets.frontRightCorner[0], + y: + primaryNozzlePosition.y - + primaryNozzleOffset[1] + + pipetteBoundsOffsets.frontRightCorner[1], + z: + primaryNozzlePosition.z - + primaryNozzleOffset[2] + + pipetteBoundsOffsets.frontRightCorner[2], + } + + const backRightBound: Point = { + x: frontRightBound.x, + y: backLeftBound.y, + z: frontRightBound.z, + } + const frontLeftBound: Point = { + x: backLeftBound.x, + y: frontRightBound.y, + z: backLeftBound.z, + } + + return [backLeftBound, frontRightBound, backRightBound, frontLeftBound] +} + +// return whether the two provided rectangles are overlapping in the 2d space. +const getHasOverlappingRectangles = ( + rectangle1: Point[], + rectangle2: Point[] +): boolean => { + const xCoords = [ + rectangle1[0].x, + rectangle1[1].x, + rectangle2[0].x, + rectangle2[1].x, + ] + const xLengthRect1 = Math.abs(rectangle1[1].x - rectangle1[0].x) + const xLengthRect2 = Math.abs((rectangle2[1].x = rectangle2[0].x)) + const overlappingInX = + Math.abs(Math.max(...xCoords) - Math.min(...xCoords)) < + xLengthRect1 + xLengthRect2 + const yCoordinates = [ + rectangle1[0].y, + rectangle1[1].y, + rectangle2[0].y, + rectangle2[1].y, + ] + const yLengthRect1 = Math.abs(rectangle1[1].y - rectangle1[0].y) + const yLengthRect2 = Math.abs(rectangle2[1].y - rectangle2[0].y) + const overlappingInY = + Math.abs(Math.max(...yCoordinates) - Math.min(...yCoordinates)) < + yLengthRect1 + yLengthRect2 + + return overlappingInX && overlappingInY +} + +// check the highest Z-point of all items stacked given a deck slot (including modules, +// adapters, and modules on adapters) +const getHighestZInSlot = ( + robotState: RobotState, + invariantContext: InvariantContext, + labwareId: string +): number => { + const { modules, labware } = robotState + const { moduleEntities, labwareEntities } = invariantContext + if (modules[labwareId] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[labwareId].model) + .dimensions + return ( + // labware + module + labwareEntities[labwareId].def.dimensions.zDimension + + moduleDimensions.bareOverallHeight + + (moduleDimensions.lidHeight ?? 0) + ) + } else if (labware[labwareId] != null) { + const adapterId = labware[labwareId].slot + if (labwareEntities[adapterId] != null) { + if (modules[adapterId] != null) { + const moduleDimensions = getModuleDef2(moduleEntities[adapterId].model) + .dimensions + return ( + // labware + adapter + module + labwareEntities[labwareId].def.dimensions.zDimension + + labwareEntities[adapterId].def.dimensions.zDimension + + moduleDimensions.bareOverallHeight + + (moduleDimensions.lidHeight ?? 0) + ) + } else { + return ( + // labware + adapter + labwareEntities[labwareId].def.dimensions.zDimension + + labwareEntities[adapterId].def.dimensions.zDimension + ) + } + } else { + // labware + return labwareEntities[labwareId].def.dimensions.zDimension + } + // shouldn't hit here! + } else { + console.error('something went wrong, this shoud not be hit') + return 0 + } +} + +// check if the slot overlaps with the pipette position +const getSlotHasPotentialCollidingObject = ( + pipetteBounds: Point[], + slotInfo: SlotInfo[], + robotState: RobotState, + invariantContext: InvariantContext, + labwareId: string +): boolean => { + for (const slot of slotInfo) { + const slotBounds = slot.addressableArea?.boundingBox + const slotPosition = slot.position + + // If slotPosition or slotBounds is null, continue to the next iteration + if (slotPosition == null || slotBounds == null) { + continue + } + + const backLeftCoords = { + x: slotPosition[0], + y: slotBounds.yDimension + slotPosition[1], + z: slotPosition[2], + } + const frontRightCoords = { + x: slotPosition[0] + slotBounds.xDimension, + y: slotPosition[1], + z: slotPosition[2], + } + // Check for overlapping rectangles and pipette z-coordinate if slot overlaps with pipette bounds + if ( + getHasOverlappingRectangles( + [pipetteBounds[0], pipetteBounds[1]], + [backLeftCoords, frontRightCoords] + ) && + pipetteBounds[0].z != null + ) { + const highestZInSlot = getHighestZInSlot( + robotState, + invariantContext, + labwareId + ) + return highestZInSlot >= pipetteBounds[0]?.z + } + } + return false +} + +const getWillCollideWithThermocyclerLid = ( + pipetteBounds: Point[], + moduleEntities: ModuleEntities +): boolean => { + if ( + Object.values(moduleEntities).find( + module => module.type === THERMOCYCLER_MODULE_TYPE + ) + ) { + return ( + getHasOverlappingRectangles( + [FLEX_TC_LID_BACK_LEFT_PT, FLEX_TC_LID_FRONT_RIGHT_PT], + [pipetteBounds[0], pipetteBounds[1]] + ) && pipetteBounds[0].x <= FLEX_TC_LID_BACK_LEFT_PT.z + ) + } else { + return false + } +} + +const getWellPosition = ( + labwareEntity: LabwareEntity, + wellLocationOffset: Point +): Point => { + const { dimensions: wellDimensions, cornerOffsetFromSlot } = labwareEntity.def + + // getting location from the bottom of the well since PD only supports aspirate/dispense from bottom + // note: api includes calibration data here which PD does not have knowledge of at the moment + return { + x: + cornerOffsetFromSlot.x + wellLocationOffset.x + wellDimensions.xDimension, + y: + cornerOffsetFromSlot.y + wellLocationOffset.y + wellDimensions.yDimension, + z: + cornerOffsetFromSlot.z + + (wellLocationOffset.z ?? 0) + + wellDimensions.zDimension, + } +} + +// util to use in step-generation for if the pipette movement is safe +export const getIsSafePipetteMovement = ( + robotState: RobotState, + invariantContext: InvariantContext, + pipetteId: string, + labwareId: string, + tipRackId: string, + wellLocationOffset: Point +): boolean => { + const deckDefinition = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) + const { + pipetteEntities, + labwareEntities, + additionalEquipmentEntities, + moduleEntities, + } = invariantContext + const { labware: labwareState, tipState } = robotState + + // early exit if labwareId is a trashBin or wasteChute + if (labwareEntities[labwareId] == null) { + return true + } + + const stagingAreaSlots = Object.values(additionalEquipmentEntities) + .filter(ae => ae.name === 'stagingArea') + .map(stagingArea => stagingArea.location as string) + const pipetteEntity = pipetteEntities[pipetteId] + const pipetteHasTip = tipState.pipettes[pipetteId] + const tipLength = pipetteHasTip + ? labwareEntities[tipRackId].def.parameters.tipLength ?? 0 + : 0 + const wellLocationPoint = getWellPosition( + labwareEntities[labwareId], + wellLocationOffset + ) + + const isWithinPipetteExtents = getIsWithinPipetteExtents( + wellLocationPoint, + // TODO(jr, 4/22/24): PD only supports A12 as a primary nozzle for now + // and only for 96-channel column pick up + NOZZLE_CONFIGURATION, + PRIMARY_NOZZLE + ) + if (!isWithinPipetteExtents) { + return false + } else { + const labwareSlot = labwareState[labwareId].slot + const pipetteBoundsAtWellLocation = getPipetteBoundsAtSpecifiedMoveToPosition( + pipetteEntity, + tipLength, + wellLocationOffset + ) + const surroundingSlots = getFlexSurroundingSlots( + labwareSlot, + stagingAreaSlots + ) + const slotInfos: SlotInfo[] = surroundingSlots.map(slot => { + const addressableArea = getAddressableAreaFromSlotId(slot, deckDefinition) + const position = getPositionFromSlotId(slot, deckDefinition) + return { + addressableArea, + position, + } + }) + return ( + !getWillCollideWithThermocyclerLid( + pipetteBoundsAtWellLocation, + moduleEntities + ) && + !getSlotHasPotentialCollidingObject( + pipetteBoundsAtWellLocation, + slotInfos, + robotState, + invariantContext, + labwareId + ) + ) + } +}