From c5a8bf4308c97c0bfe29838de1af5f4e39f08ba0 Mon Sep 17 00:00:00 2001 From: Brent Hagen Date: Mon, 4 Dec 2023 16:24:16 -0500 Subject: [PATCH] feat(app,shared-data): get addressable area for legacy fixed trash (#14085) provides a special case for the addressable area of the legacy Flex fixed trash found in older protocols. parses the labware key of a protocol analysis and inserts addressable area moveableTrashA3 if a legacy fixed trash labware is found. moves the getSimplestDeckConfigForProtocol function to shared-data and updates references in app and components. closes RAUT-888 --- .../Devices/ProtocolRun/ProtocolRunHeader.tsx | 2 +- .../Devices/ProtocolRun/ProtocolRunSetup.tsx | 15 +- .../SetupLabware/SetupLabwareMap.tsx | 6 +- .../SetupLiquids/SetupLiquidsMap.tsx | 6 +- .../__tests__/SetupLiquidsMap.test.tsx | 11 +- .../SetupModuleAndDeck/SetupModulesList.tsx | 2 +- .../SetupModuleAndDeck/SetupModulesMap.tsx | 6 +- .../__tests__/SetupModulesAndDeck.test.tsx | 2 +- .../ProtocolRun/SetupModuleAndDeck/index.tsx | 11 +- .../__tests__/ProtocolRunSetup.test.tsx | 9 +- .../useModuleRenderInfoForProtocolById.ts | 2 +- .../RobotConfigurationDetails.tsx | 2 +- app/src/organisms/ProtocolDetails/index.tsx | 10 +- .../ProtocolSetupDeckConfiguration.test.tsx | 9 +- .../ProtocolSetupDeckConfiguration/index.tsx | 10 +- .../LabwareMapViewModal.tsx | 11 +- .../__tests__/LabwareMapViewModal.test.tsx | 13 +- .../FixtureTable.tsx | 8 +- .../ModulesAndDeckMapViewModal.tsx | 10 +- .../__tests__/FixtureTable.test.tsx | 4 +- .../ModulesAndDeckMapViewModal.test.tsx | 9 +- .../ProtocolSetupModulesAndDeck.test.tsx | 12 +- .../ProtocolSetupModulesAndDeck/index.tsx | 2 +- .../ProtocolDetails/__tests__/Deck.test.tsx | 1 + .../Protocols/hooks/__tests__/hooks.test.tsx | 2 +- app/src/pages/Protocols/hooks/index.ts | 22 +-- .../__tests__/utils.test.ts | 142 -------------- app/src/resources/deck_configuration/hooks.ts | 19 +- app/src/resources/deck_configuration/utils.ts | 185 +----------------- .../src/hardware-sim/BaseDeck/BaseDeck.tsx | 5 +- .../src/hardware-sim/ProtocolDeck/index.tsx | 14 +- .../src/hardware-sim/ProtocolDeck/types.ts | 13 -- .../hardware-sim/ProtocolDeck/utils/index.ts | 1 - .../getSimplestFlexDeckConfig.test.ts | 154 +++++++++++++++ .../helpers/getAddressableAreasInProtocol.ts | 146 +++++++++----- .../js/helpers}/getSimplestFlexDeckConfig.ts | 28 ++- shared-data/js/helpers/index.ts | 1 + 37 files changed, 382 insertions(+), 523 deletions(-) delete mode 100644 app/src/resources/deck_configuration/__tests__/utils.test.ts create mode 100644 shared-data/js/helpers/__tests__/getSimplestFlexDeckConfig.test.ts rename {components/src/hardware-sim/ProtocolDeck/utils => shared-data/js/helpers}/getSimplestFlexDeckConfig.ts (90%) diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx index 9e9d35b3518..904ecf726b8 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunHeader.tsx @@ -183,7 +183,7 @@ export function ProtocolRunHeader({ const robotType = isFlex ? FLEX_ROBOT_TYPE : OT2_ROBOT_TYPE const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - robotProtocolAnalysis?.commands ?? [] + robotProtocolAnalysis ) const isFixtureMismatch = getIsFixtureMismatch(deckConfigCompatibility) diff --git a/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx b/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx index 00fd64857a2..e6e07c9f3f8 100644 --- a/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx +++ b/app/src/organisms/Devices/ProtocolRun/ProtocolRunSetup.tsx @@ -13,7 +13,11 @@ import { SPACING, TYPOGRAPHY, } from '@opentrons/components' -import { FLEX_ROBOT_TYPE, OT2_ROBOT_TYPE } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, + OT2_ROBOT_TYPE, +} from '@opentrons/shared-data' import { Line } from '../../../atoms/structure' import { StyledText } from '../../../atoms/text' @@ -21,7 +25,6 @@ import { InfoMessage } from '../../../molecules/InfoMessage' import { getIsFixtureMismatch, getRequiredDeckConfig, - getSimplestDeckConfigForProtocolCommands, } from '../../../resources/deck_configuration/utils' import { useDeckConfigurationCompatibility } from '../../../resources/deck_configuration/hooks' import { @@ -89,7 +92,7 @@ export function ProtocolRunSetup({ const robotType = isFlex ? FLEX_ROBOT_TYPE : OT2_ROBOT_TYPE const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - protocolAnalysis?.commands ?? [] + protocolAnalysis ) const isFixtureMismatch = getIsFixtureMismatch(deckConfigCompatibility) @@ -134,9 +137,7 @@ export function ProtocolRunSetup({ protocolAnalysis != null && protocolAnalysis.liquids?.length > 0 const hasModules = protocolAnalysis != null && modules.length > 0 - const protocolDeckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis?.commands ?? [] - ) + const protocolDeckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) const requiredDeckConfig = getRequiredDeckConfig(protocolDeckConfig) @@ -185,7 +186,7 @@ export function ProtocolRunSetup({ robotName={robotName} runId={runId} hasModules={hasModules} - commands={protocolAnalysis?.commands ?? []} + protocolAnalysis={protocolAnalysis} /> ), description: moduleDescription, diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx index 630f0fbbf84..0505cf0c921 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLabware/SetupLabwareMap.tsx @@ -12,11 +12,11 @@ import { import { FLEX_ROBOT_TYPE, getDeckDefFromRobotType, + getSimplestDeckConfigForProtocol, THERMOCYCLER_MODULE_V1, } from '@opentrons/shared-data' import { getLabwareSetupItemGroups } from '../../../../pages/Protocols/utils' -import { getSimplestDeckConfigForProtocolCommands } from '../../../../resources/deck_configuration/utils' import { getAttachedProtocolModuleMatches } from '../../../ProtocolSetupModulesAndDeck/utils' import { useAttachedModules } from '../../hooks' import { LabwareInfoOverlay } from '../LabwareInfoOverlay' @@ -106,9 +106,7 @@ export function SetupLabwareMap({ const { offDeckItems } = getLabwareSetupItemGroups(commands) - const deckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis.commands - ) + const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) const labwareRenderInfo = getLabwareRenderInfo(protocolAnalysis, deckDef) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx index d549a45c47d..965c36f100c 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/SetupLiquidsMap.tsx @@ -17,6 +17,7 @@ import { import { FLEX_ROBOT_TYPE, getDeckDefFromRobotType, + getSimplestDeckConfigForProtocol, THERMOCYCLER_MODULE_V1, } from '@opentrons/shared-data' @@ -25,7 +26,6 @@ import { LabwareInfoOverlay } from '../LabwareInfoOverlay' import { LiquidsLabwareDetailsModal } from './LiquidsLabwareDetailsModal' import { getWellFillFromLabwareId } from './utils' import { getLabwareRenderInfo } from '../utils/getLabwareRenderInfo' -import { getSimplestDeckConfigForProtocolCommands } from '../../../../resources/deck_configuration/utils' import { getStandardDeckViewLayerBlockList } from '../utils/getStandardDeckViewLayerBlockList' import { getAttachedProtocolModuleMatches } from '../../../ProtocolSetupModulesAndDeck/utils' import { getProtocolModulesInfo } from '../utils/getProtocolModulesInfo' @@ -70,9 +70,7 @@ export function SetupLiquidsMap( const labwareByLiquidId = parseLabwareInfoByLiquidId( protocolAnalysis.commands ?? [] ) - const deckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis.commands - ) + const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) const deckLayerBlocklist = getStandardDeckViewLayerBlockList(robotType) const protocolModulesInfo = getProtocolModulesInfo(protocolAnalysis, deckDef) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx index c20b1288af2..65bbabf2b69 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupLiquids/__tests__/SetupLiquidsMap.test.tsx @@ -12,6 +12,7 @@ import fixture_tiprack_300_ul from '@opentrons/shared-data/labware/fixtures/2/fi import { FLEX_ROBOT_TYPE, getDeckDefFromRobotType, + getSimplestDeckConfigForProtocol, OT2_ROBOT_TYPE, } from '@opentrons/shared-data' import { @@ -29,7 +30,6 @@ import { getLabwareRenderInfo } from '../../utils/getLabwareRenderInfo' import { getStandardDeckViewLayerBlockList } from '../../utils/getStandardDeckViewLayerBlockList' import { getAttachedProtocolModuleMatches } from '../../../../ProtocolSetupModulesAndDeck/utils' import { getProtocolModulesInfo } from '../../utils/getProtocolModulesInfo' -import { getSimplestDeckConfigForProtocolCommands } from '../../../../../resources/deck_configuration/utils' import { mockProtocolModuleInfo } from '../../../../ProtocolSetupLabware/__fixtures__' import { mockFetchModulesSuccessActionPayloadModules } from '../../../../../redux/modules/__fixtures__' @@ -38,7 +38,6 @@ import { SetupLiquidsMap } from '../SetupLiquidsMap' import type { ModuleModel, ModuleType, - RunTimeCommand, LabwareDefinition2, } from '@opentrons/shared-data' @@ -92,8 +91,8 @@ const mockGetAttachedProtocolModuleMatches = getAttachedProtocolModuleMatches as const mockGetProtocolModulesInfo = getProtocolModulesInfo as jest.MockedFunction< typeof getProtocolModulesInfo > -const mockGetSimplestDeckConfigForProtocolCommands = getSimplestDeckConfigForProtocolCommands as jest.MockedFunction< - typeof getSimplestDeckConfigForProtocolCommands +const mockGetSimplestDeckConfigForProtocol = getSimplestDeckConfigForProtocol as jest.MockedFunction< + typeof getSimplestDeckConfigForProtocol > const RUN_ID = '1' @@ -163,8 +162,8 @@ describe('SetupLiquidsMap', () => { when(mockGetLabwareRenderInfo) .calledWith(mockProtocolAnalysis, ot2StandardDeckDef as any) .mockReturnValue({}) - when(mockGetSimplestDeckConfigForProtocolCommands) - .calledWith(mockProtocolAnalysis.commands as RunTimeCommand[]) + when(mockGetSimplestDeckConfigForProtocol) + .calledWith(mockProtocolAnalysis) // TODO(bh, 2023-11-13): mock the cutout config protocol spec .mockReturnValue([]) when(mockParseLiquidsInLoadOrder) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesList.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesList.tsx index 07c943e8341..109eafdd30b 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesList.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesList.tsx @@ -21,6 +21,7 @@ import { } from '@opentrons/components' import { FLEX_ROBOT_TYPE, + getCutoutIdForSlotName, getDeckDefFromRobotType, getModuleType, HEATERSHAKER_MODULE_TYPE, @@ -35,7 +36,6 @@ import { TertiaryButton } from '../../../../atoms/buttons' import { StatusLabel } from '../../../../atoms/StatusLabel' import { StyledText } from '../../../../atoms/text' import { Tooltip } from '../../../../atoms/Tooltip' -import { getCutoutIdForSlotName } from '../../../../resources/deck_configuration/utils' import { useChainLiveCommands } from '../../../../resources/runs/hooks' import { ModuleSetupModal } from '../../../ModuleCard/ModuleSetupModal' import { ModuleWizardFlows } from '../../../ModuleWizardFlows' diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx index bb7c87c8f56..a5c6ab1ecbd 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/SetupModulesMap.tsx @@ -10,9 +10,9 @@ import { import { FLEX_ROBOT_TYPE, getDeckDefFromRobotType, + getSimplestDeckConfigForProtocol, } from '@opentrons/shared-data' -import { getSimplestDeckConfigForProtocolCommands } from '../../../../resources/deck_configuration/utils' import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { getAttachedProtocolModuleMatches } from '../../../ProtocolSetupModulesAndDeck/utils' import { ModuleInfo } from '../../ModuleInfo' @@ -64,9 +64,7 @@ export const SetupModulesMap = ({ ), })) - const deckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis.commands - ) + const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) return ( { runId: MOCK_RUN_ID, expandLabwarePositionCheckStep: () => jest.fn(), hasModules: true, - commands: [], + protocolAnalysis: null, } mockSetupFixtureList.mockReturnValue(
Mock setup fixture list
) mockSetupModulesList.mockReturnValue(
Mock setup modules list
) diff --git a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx index 6a781bdcadb..ad5de80fe7a 100644 --- a/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx +++ b/app/src/organisms/Devices/ProtocolRun/SetupModuleAndDeck/index.tsx @@ -27,14 +27,17 @@ import { SetupModulesMap } from './SetupModulesMap' import { SetupModulesList } from './SetupModulesList' import { SetupFixtureList } from './SetupFixtureList' -import type { RunTimeCommand } from '@opentrons/shared-data' +import type { + CompletedProtocolAnalysis, + ProtocolAnalysisOutput, +} from '@opentrons/shared-data' interface SetupModuleAndDeckProps { expandLabwarePositionCheckStep: () => void robotName: string runId: string hasModules: boolean - commands: RunTimeCommand[] + protocolAnalysis: CompletedProtocolAnalysis | ProtocolAnalysisOutput | null } export const SetupModuleAndDeck = ({ @@ -42,7 +45,7 @@ export const SetupModuleAndDeck = ({ robotName, runId, hasModules, - commands, + protocolAnalysis, }: SetupModuleAndDeckProps): JSX.Element => { const { t } = useTranslation('protocol_setup') const [selectedValue, toggleGroup] = useToggleGroup( @@ -58,7 +61,7 @@ export const SetupModuleAndDeck = ({ const moduleCalibrationStatus = useModuleCalibrationStatus(robotName, runId) const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - commands + protocolAnalysis ) const isFixtureMismatch = getIsFixtureMismatch(deckConfigCompatibility) diff --git a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx index 43433667f00..bd50b4e78a1 100644 --- a/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx +++ b/app/src/organisms/Devices/ProtocolRun/__tests__/ProtocolRunSetup.test.tsx @@ -7,6 +7,7 @@ import { renderWithProviders, } from '@opentrons/components' import { + getSimplestDeckConfigForProtocol, ProtocolAnalysisOutput, STAGING_AREA_SLOT_WITH_WASTE_CHUTE_RIGHT_ADAPTER_NO_COVER_FIXTURE, } from '@opentrons/shared-data' @@ -18,7 +19,6 @@ import { mockConnectedRobot } from '../../../../redux/discovery/__fixtures__' import { getIsFixtureMismatch, getRequiredDeckConfig, - getSimplestDeckConfigForProtocolCommands, } from '../../../../resources/deck_configuration/utils' import { useMostRecentCompletedAnalysis } from '../../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useDeckConfigurationCompatibility } from '../../../../resources/deck_configuration/hooks' @@ -48,6 +48,7 @@ jest.mock('../SetupLiquids') jest.mock('../EmptySetupStep') jest.mock('../../../LabwarePositionCheck/useMostRecentCompletedAnalysis') jest.mock('@opentrons/shared-data/js/helpers/parseProtocolData') +jest.mock('@opentrons/shared-data/js/helpers/getSimplestFlexDeckConfig') jest.mock('../../../../redux/config') jest.mock('../../../../resources/deck_configuration/utils') jest.mock('../../../../resources/deck_configuration/hooks') @@ -90,8 +91,8 @@ const mockSetupLiquids = SetupLiquids as jest.MockedFunction< const mockEmptySetupStep = EmptySetupStep as jest.MockedFunction< typeof EmptySetupStep > -const mockGetSimplestDeckConfigForProtocolCommands = getSimplestDeckConfigForProtocolCommands as jest.MockedFunction< - typeof getSimplestDeckConfigForProtocolCommands +const mockGetSimplestDeckConfigForProtocol = getSimplestDeckConfigForProtocol as jest.MockedFunction< + typeof getSimplestDeckConfigForProtocol > const mockGetRequiredDeckConfig = getRequiredDeckConfig as jest.MockedFunction< typeof getRequiredDeckConfig @@ -168,7 +169,7 @@ describe('ProtocolRunSetup', () => { when(mockSetupModuleAndDeck).mockReturnValue(
Mock SetupModules
) when(mockSetupLiquids).mockReturnValue(
Mock SetupLiquids
) when(mockEmptySetupStep).mockReturnValue(
Mock EmptySetupStep
) - when(mockGetSimplestDeckConfigForProtocolCommands).mockReturnValue([]) + when(mockGetSimplestDeckConfigForProtocol).mockReturnValue([]) when(mockUseDeckConfigurationCompatibility).mockReturnValue([]) when(mockGetRequiredDeckConfig).mockReturnValue([]) when(mockUseUnmatchedModulesForProtocol) diff --git a/app/src/organisms/Devices/hooks/useModuleRenderInfoForProtocolById.ts b/app/src/organisms/Devices/hooks/useModuleRenderInfoForProtocolById.ts index 1c9570dfcde..9f4e49c73c5 100644 --- a/app/src/organisms/Devices/hooks/useModuleRenderInfoForProtocolById.ts +++ b/app/src/organisms/Devices/hooks/useModuleRenderInfoForProtocolById.ts @@ -1,12 +1,12 @@ import { checkModuleCompatibility, FLEX_ROBOT_TYPE, + getCutoutIdForSlotName, getDeckDefFromRobotType, SINGLE_SLOT_FIXTURES, } from '@opentrons/shared-data' import { useDeckConfigurationQuery } from '@opentrons/react-api-client/src/deck_configuration' -import { getCutoutIdForSlotName } from '../../../resources/deck_configuration/utils' import { getProtocolModulesInfo } from '../ProtocolRun/utils/getProtocolModulesInfo' import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { useAttachedModules } from './useAttachedModules' diff --git a/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx b/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx index e4da0e001c7..f25c7351cd8 100644 --- a/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx +++ b/app/src/organisms/ProtocolDetails/RobotConfigurationDetails.tsx @@ -29,12 +29,12 @@ import { getRobotTypeDisplayName } from '../ProtocolsLanding/utils' import { getSlotsForThermocycler } from './utils' import type { + CutoutConfigProtocolSpec, LoadModuleRunTimeCommand, PipetteName, RobotType, SingleSlotCutoutFixtureId, } from '@opentrons/shared-data' -import type { CutoutConfigProtocolSpec } from '../../resources/deck_configuration/utils' interface RobotConfigurationDetailsProps { leftMountPipetteName: PipetteName | null diff --git a/app/src/organisms/ProtocolDetails/index.tsx b/app/src/organisms/ProtocolDetails/index.tsx index 365e5bff520..a98f863b8c8 100644 --- a/app/src/organisms/ProtocolDetails/index.tsx +++ b/app/src/organisms/ProtocolDetails/index.tsx @@ -39,7 +39,10 @@ import { parseInitialLoadedLabwareByModuleId, parseInitialLoadedLabwareByAdapter, } from '@opentrons/api-client' -import { getGripperDisplayName } from '@opentrons/shared-data' +import { + getGripperDisplayName, + getSimplestDeckConfigForProtocol, +} from '@opentrons/shared-data' import { Portal } from '../../App/portal' import { Divider } from '../../atoms/structure' @@ -54,7 +57,6 @@ import { analyzeProtocol, } from '../../redux/protocol-storage' import { useFeatureFlag } from '../../redux/config' -import { getSimplestDeckConfigForProtocolCommands } from '../../resources/deck_configuration/utils' import { ChooseRobotToRunProtocolSlideout } from '../ChooseRobotToRunProtocolSlideout' import { SendProtocolToOT3Slideout } from '../SendProtocolToOT3Slideout' import { ProtocolAnalysisFailure } from '../ProtocolAnalysisFailure' @@ -230,8 +232,8 @@ export function ProtocolDetails( ? map(parseInitialLoadedModulesBySlot(mostRecentAnalysis.commands)) : [] - const requiredFixtureDetails = getSimplestDeckConfigForProtocolCommands( - mostRecentAnalysis?.commands ?? [] + const requiredFixtureDetails = getSimplestDeckConfigForProtocol( + mostRecentAnalysis ) const requiredLabwareDetails = diff --git a/app/src/organisms/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx b/app/src/organisms/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx index b8382ac1ef2..6621c8c5c22 100644 --- a/app/src/organisms/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx +++ b/app/src/organisms/ProtocolSetupDeckConfiguration/__tests__/ProtocolSetupDeckConfiguration.test.tsx @@ -8,6 +8,8 @@ import { i18n } from '../../../i18n' import { useMostRecentCompletedAnalysis } from '../../LabwarePositionCheck/useMostRecentCompletedAnalysis' import { ProtocolSetupDeckConfiguration } from '..' +import type { CompletedProtocolAnalysis } from '@opentrons/shared-data' + jest.mock('@opentrons/components/src/hardware-sim/BaseDeck/index') jest.mock('@opentrons/react-api-client') jest.mock('../../LabwarePositionCheck/useMostRecentCompletedAnalysis') @@ -16,7 +18,10 @@ const mockSetSetupScreen = jest.fn() const mockUpdateDeckConfiguration = jest.fn() const PROTOCOL_DETAILS = { displayName: 'fake protocol', - protocolData: [], + protocolData: ({ + commands: [], + labware: [], + } as unknown) as CompletedProtocolAnalysis, protocolKey: 'fakeProtocolKey', robotType: 'OT-3 Standard' as const, } @@ -50,7 +55,7 @@ describe('ProtocolSetupDeckConfiguration', () => { mockBaseDeck.mockReturnValue(
mock BaseDeck
) when(mockUseMostRecentCompletedAnalysis) .calledWith('mockRunId') - .mockReturnValue(PROTOCOL_DETAILS.protocolData as any) + .mockReturnValue(PROTOCOL_DETAILS.protocolData) mockUseUpdateDeckConfigurationMutation.mockReturnValue({ updateDeckConfiguration: mockUpdateDeckConfiguration, } as any) diff --git a/app/src/organisms/ProtocolSetupDeckConfiguration/index.tsx b/app/src/organisms/ProtocolSetupDeckConfiguration/index.tsx index 435498c53d9..cf03ae8e465 100644 --- a/app/src/organisms/ProtocolSetupDeckConfiguration/index.tsx +++ b/app/src/organisms/ProtocolSetupDeckConfiguration/index.tsx @@ -8,14 +8,16 @@ import { JUSTIFY_CENTER, SPACING, } from '@opentrons/components' -import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, +} from '@opentrons/shared-data' import { useUpdateDeckConfigurationMutation } from '@opentrons/react-api-client' import { ChildNavigation } from '../ChildNavigation' import { AddFixtureModal } from '../DeviceDetailsDeckConfiguration/AddFixtureModal' import { DeckConfigurationDiscardChangesModal } from '../DeviceDetailsDeckConfiguration/DeckConfigurationDiscardChangesModal' import { useMostRecentCompletedAnalysis } from '../LabwarePositionCheck/useMostRecentCompletedAnalysis' -import { getSimplestDeckConfigForProtocolCommands } from '../../resources/deck_configuration/utils' import { Portal } from '../../App/portal' import type { @@ -51,8 +53,8 @@ export function ProtocolSetupDeckConfiguration({ const mostRecentAnalysis = useMostRecentCompletedAnalysis(runId) - const simplestDeckConfig = getSimplestDeckConfigForProtocolCommands( - mostRecentAnalysis?.commands ?? [] + const simplestDeckConfig = getSimplestDeckConfigForProtocol( + mostRecentAnalysis ).map(({ cutoutId, cutoutFixtureId }) => ({ cutoutId, cutoutFixtureId })) const [ diff --git a/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx b/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx index a685bb3929b..efd97cb14ed 100644 --- a/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx +++ b/app/src/organisms/ProtocolSetupLabware/LabwareMapViewModal.tsx @@ -2,10 +2,13 @@ import * as React from 'react' import map from 'lodash/map' import { useTranslation } from 'react-i18next' import { BaseDeck } from '@opentrons/components' -import { FLEX_ROBOT_TYPE, THERMOCYCLER_MODULE_V1 } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, + THERMOCYCLER_MODULE_V1, +} from '@opentrons/shared-data' import { Modal } from '../../molecules/Modal' -import { getSimplestDeckConfigForProtocolCommands } from '../../resources/deck_configuration/utils' import { getStandardDeckViewLayerBlockList } from '../Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList' import { getLabwareRenderInfo } from '../Devices/ProtocolRun/utils/getLabwareRenderInfo' import { AttachedProtocolModuleMatch } from '../ProtocolSetupModulesAndDeck/utils' @@ -42,9 +45,7 @@ export function LabwareMapViewModal( mostRecentAnalysis, } = props const { t } = useTranslation('protocol_setup') - const deckConfig = getSimplestDeckConfigForProtocolCommands( - mostRecentAnalysis?.commands ?? [] - ) + const deckConfig = getSimplestDeckConfigForProtocol(mostRecentAnalysis) const labwareRenderInfo = mostRecentAnalysis != null ? getLabwareRenderInfo(mostRecentAnalysis, deckDef) diff --git a/app/src/organisms/ProtocolSetupLabware/__tests__/LabwareMapViewModal.test.tsx b/app/src/organisms/ProtocolSetupLabware/__tests__/LabwareMapViewModal.test.tsx index a74f09dfe98..aeff31977c6 100644 --- a/app/src/organisms/ProtocolSetupLabware/__tests__/LabwareMapViewModal.test.tsx +++ b/app/src/organisms/ProtocolSetupLabware/__tests__/LabwareMapViewModal.test.tsx @@ -6,11 +6,13 @@ import { BaseDeck, EXTENDED_DECK_CONFIG_FIXTURE, } from '@opentrons/components' -import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, +} from '@opentrons/shared-data' import deckDefFixture from '@opentrons/shared-data/deck/fixtures/3/deckExample.json' import fixture_tiprack_300_ul from '@opentrons/shared-data/labware/fixtures/2/fixture_tiprack_300_ul.json' import { i18n } from '../../../i18n' -import { getSimplestDeckConfigForProtocolCommands } from '../../../resources/deck_configuration/utils' import { getLabwareRenderInfo } from '../../Devices/ProtocolRun/utils/getLabwareRenderInfo' import { getStandardDeckViewLayerBlockList } from '../../Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList' import { mockProtocolModuleInfo } from '../__fixtures__' @@ -26,14 +28,15 @@ import type { jest.mock('../../Devices/ProtocolRun/utils/getLabwareRenderInfo') jest.mock('@opentrons/components/src/hardware-sim/Labware/LabwareRender') jest.mock('@opentrons/components/src/hardware-sim/BaseDeck') +jest.mock('@opentrons/shared-data/js/helpers/getSimplestFlexDeckConfig') jest.mock('../../../resources/deck_configuration/utils') jest.mock('../../../redux/config') const mockGetLabwareRenderInfo = getLabwareRenderInfo as jest.MockedFunction< typeof getLabwareRenderInfo > -const mockGetSimplestDeckConfigForProtocolCommands = getSimplestDeckConfigForProtocolCommands as jest.MockedFunction< - typeof getSimplestDeckConfigForProtocolCommands +const mockGetSimplestDeckConfigForProtocol = getSimplestDeckConfigForProtocol as jest.MockedFunction< + typeof getSimplestDeckConfigForProtocol > const mockBaseDeck = BaseDeck as jest.MockedFunction @@ -53,7 +56,7 @@ const render = (props: React.ComponentProps) => { describe('LabwareMapViewModal', () => { beforeEach(() => { mockGetLabwareRenderInfo.mockReturnValue({}) - mockGetSimplestDeckConfigForProtocolCommands.mockReturnValue([]) + mockGetSimplestDeckConfigForProtocol.mockReturnValue([]) }) afterEach(() => { diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/FixtureTable.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/FixtureTable.tsx index 910daa5db60..8b374b21048 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/FixtureTable.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/FixtureTable.tsx @@ -17,13 +17,13 @@ import { FLEX_SINGLE_SLOT_ADDRESSABLE_AREAS, getCutoutDisplayName, getFixtureDisplayName, + getSimplestDeckConfigForProtocol, SINGLE_SLOT_FIXTURES, } from '@opentrons/shared-data' import { useDeckConfigurationCompatibility } from '../../resources/deck_configuration/hooks' import { LocationConflictModal } from '../Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal' import { StyledText } from '../../atoms/text' import { Chip } from '../../atoms/Chip' -import { getSimplestDeckConfigForProtocolCommands } from '../../resources/deck_configuration/utils' import type { CompletedProtocolAnalysis, @@ -55,12 +55,12 @@ export function FixtureTable({ setShowLocationConflictModal, ] = React.useState(false) - const requiredFixtureDetails = getSimplestDeckConfigForProtocolCommands( - mostRecentAnalysis?.commands ?? [] + const requiredFixtureDetails = getSimplestDeckConfigForProtocol( + mostRecentAnalysis ) const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - mostRecentAnalysis?.commands ?? [] + mostRecentAnalysis ) const nonSingleSlotDeckConfigCompatibility = deckConfigCompatibility.filter( diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/ModulesAndDeckMapViewModal.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/ModulesAndDeckMapViewModal.tsx index 3dced10d3e2..37d9ba6c3b0 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/ModulesAndDeckMapViewModal.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/ModulesAndDeckMapViewModal.tsx @@ -2,11 +2,13 @@ import React from 'react' import { useTranslation } from 'react-i18next' import { BaseDeck } from '@opentrons/components' -import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, +} from '@opentrons/shared-data' import { Modal } from '../../molecules/Modal' import { ModuleInfo } from '../Devices/ModuleInfo' -import { getSimplestDeckConfigForProtocolCommands } from '../../resources/deck_configuration/utils' import { getStandardDeckViewLayerBlockList } from '../Devices/ProtocolRun/utils/getStandardDeckViewLayerBlockList' import type { CompletedProtocolAnalysis } from '@opentrons/shared-data' @@ -36,9 +38,7 @@ export function ModulesAndDeckMapViewModal({ if (protocolAnalysis == null) return null - const deckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis.commands - ) + const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) const modulesOnDeck = attachedProtocolModuleMatches.map(module => ({ moduleModel: module.moduleDef.model, diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx index c24bbbe2537..7bd357c8b7f 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/FixtureTable.test.tsx @@ -34,7 +34,7 @@ describe('FixtureTable', () => { let props: React.ComponentProps beforeEach(() => { props = { - mostRecentAnalysis: [] as any, + mostRecentAnalysis: { commands: [], labware: [] } as any, robotType: FLEX_ROBOT_TYPE, setSetupScreen: mockSetSetupScreen, setCutoutId: mockSetCutoutId, @@ -65,7 +65,7 @@ describe('FixtureTable', () => { props = { ...props, // TODO(bh, 2023-11-13): mock load labware etc commands - mostRecentAnalysis: { commands: [] } as any, + mostRecentAnalysis: { commands: [], labware: [] } as any, } const [{ getByText }] = render(props) getByText('Configured') diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapViewModal.test.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapViewModal.test.tsx index 343a692a4f4..7a403394e2f 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapViewModal.test.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ModulesAndDeckMapViewModal.test.tsx @@ -2,13 +2,14 @@ import * as React from 'react' import { when, resetAllWhenMocks } from 'jest-when' import { renderWithProviders, BaseDeck } from '@opentrons/components' +import { getSimplestDeckConfigForProtocol } from '@opentrons/shared-data' import { i18n } from '../../../i18n' -import { getSimplestDeckConfigForProtocolCommands } from '../../../resources/deck_configuration/utils' import { ModulesAndDeckMapViewModal } from '../ModulesAndDeckMapViewModal' jest.mock('@opentrons/components/src/hardware-sim/BaseDeck') jest.mock('@opentrons/api-client') +jest.mock('@opentrons/shared-data/js/helpers/getSimplestFlexDeckConfig') jest.mock('../../../redux/config') jest.mock('../../Devices/hooks') jest.mock('../../../resources/deck_configuration/utils') @@ -88,8 +89,8 @@ const render = ( } const mockBaseDeck = BaseDeck as jest.MockedFunction -const mockGetSimplestDeckConfigForProtocolCommands = getSimplestDeckConfigForProtocolCommands as jest.MockedFunction< - typeof getSimplestDeckConfigForProtocolCommands +const mockGetSimplestDeckConfigForProtocol = getSimplestDeckConfigForProtocol as jest.MockedFunction< + typeof getSimplestDeckConfigForProtocol > describe('ModulesAndDeckMapViewModal', () => { @@ -102,7 +103,7 @@ describe('ModulesAndDeckMapViewModal', () => { runId: mockRunId, protocolAnalysis: PROTOCOL_ANALYSIS, } - when(mockGetSimplestDeckConfigForProtocolCommands).mockReturnValue( + when(mockGetSimplestDeckConfigForProtocol).mockReturnValue( // TODO(bh, 2023-11-13): mock cutout config protocol spec [] ) diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx index 422b3df24dd..4999c59831e 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/__tests__/ProtocolSetupModulesAndDeck.test.tsx @@ -6,10 +6,7 @@ import { MemoryRouter } from 'react-router-dom' import { renderWithProviders } from '@opentrons/components' import { useDeckConfigurationQuery } from '@opentrons/react-api-client' -import { - getDeckDefFromRobotType, - WASTE_CHUTE_RIGHT_ADAPTER_NO_COVER_FIXTURE, -} from '@opentrons/shared-data' +import { WASTE_CHUTE_RIGHT_ADAPTER_NO_COVER_FIXTURE } from '@opentrons/shared-data' import ot3StandardDeckDef from '@opentrons/shared-data/deck/definitions/4/ot3_standard.json' import { i18n } from '../../../i18n' @@ -40,7 +37,6 @@ import type { CutoutConfig, DeckConfiguration } from '@opentrons/shared-data' jest.mock('@opentrons/react-api-client') jest.mock('../../../resources/runs/hooks') -jest.mock('@opentrons/shared-data/js/helpers') jest.mock('../../../redux/discovery') jest.mock('../../../organisms/Devices/hooks') jest.mock( @@ -54,9 +50,6 @@ jest.mock('../FixtureTable') jest.mock('../../Devices/ProtocolRun/SetupModuleAndDeck/LocationConflictModal') jest.mock('../ModulesAndDeckMapViewModal') -const mockGetDeckDefFromRobotType = getDeckDefFromRobotType as jest.MockedFunction< - typeof getDeckDefFromRobotType -> const mockUseAttachedModules = useAttachedModules as jest.MockedFunction< typeof useAttachedModules > @@ -158,9 +151,6 @@ describe('ProtocolSetupModulesAndDeck', () => { when(mockGetUnmatchedModulesForProtocol) .calledWith([], []) .mockReturnValue({ missingModuleIds: [], remainingAttachedModules: [] }) - when(mockGetDeckDefFromRobotType) - .calledWith('OT-3 Standard') - .mockReturnValue(ot3StandardDeckDef as any) mockSetupInstructionsModal.mockReturnValue(
mock SetupInstructionsModal
) diff --git a/app/src/organisms/ProtocolSetupModulesAndDeck/index.tsx b/app/src/organisms/ProtocolSetupModulesAndDeck/index.tsx index ab6c3fd5cd4..fd424664709 100644 --- a/app/src/organisms/ProtocolSetupModulesAndDeck/index.tsx +++ b/app/src/organisms/ProtocolSetupModulesAndDeck/index.tsx @@ -17,6 +17,7 @@ import { import { useDeckConfigurationQuery } from '@opentrons/react-api-client' import { FLEX_ROBOT_TYPE, + getCutoutIdForSlotName, getDeckDefFromRobotType, getModuleDisplayName, getModuleType, @@ -40,7 +41,6 @@ import { MultipleModulesModal } from '../Devices/ProtocolRun/SetupModuleAndDeck/ import { getProtocolModulesInfo } from '../../organisms/Devices/ProtocolRun/utils/getProtocolModulesInfo' import { useMostRecentCompletedAnalysis } from '../../organisms/LabwarePositionCheck/useMostRecentCompletedAnalysis' import { getLocalRobot } from '../../redux/discovery' -import { getCutoutIdForSlotName } from '../../resources/deck_configuration/utils' import { useChainLiveCommands } from '../../resources/runs/hooks' import { getModulePrepCommands, diff --git a/app/src/pages/OnDeviceDisplay/ProtocolDetails/__tests__/Deck.test.tsx b/app/src/pages/OnDeviceDisplay/ProtocolDetails/__tests__/Deck.test.tsx index 7311d3023f3..c766b9a9732 100644 --- a/app/src/pages/OnDeviceDisplay/ProtocolDetails/__tests__/Deck.test.tsx +++ b/app/src/pages/OnDeviceDisplay/ProtocolDetails/__tests__/Deck.test.tsx @@ -129,6 +129,7 @@ const MOCK_PROTOCOL_ANALYSIS = { completedAt: '2022-09-07T19:47:42.786412+00:00', }, ], + labware: [], liquids: [ { id: '1', diff --git a/app/src/pages/Protocols/hooks/__tests__/hooks.test.tsx b/app/src/pages/Protocols/hooks/__tests__/hooks.test.tsx index 4c7b7c150fb..ee2a8d6ff95 100644 --- a/app/src/pages/Protocols/hooks/__tests__/hooks.test.tsx +++ b/app/src/pages/Protocols/hooks/__tests__/hooks.test.tsx @@ -13,6 +13,7 @@ import { import { CompletedProtocolAnalysis, DeckConfiguration, + FLEX_SIMPLEST_DECK_CONFIG, LabwareDefinition2, WASTE_CHUTE_RIGHT_ADAPTER_NO_COVER_FIXTURE, } from '@opentrons/shared-data' @@ -21,7 +22,6 @@ import { useMissingProtocolHardware, useRequiredProtocolLabware } from '..' import type { Protocol } from '@opentrons/api-client' import { mockHeaterShaker } from '../../../../redux/modules/__fixtures__' -import { FLEX_SIMPLEST_DECK_CONFIG } from '../../../../resources/deck_configuration/utils' jest.mock('@opentrons/react-api-client') jest.mock('../../../../organisms/Devices/hooks') diff --git a/app/src/pages/Protocols/hooks/index.ts b/app/src/pages/Protocols/hooks/index.ts index a78b319ea29..730f484f605 100644 --- a/app/src/pages/Protocols/hooks/index.ts +++ b/app/src/pages/Protocols/hooks/index.ts @@ -10,12 +10,12 @@ import { FLEX_ROBOT_TYPE, FLEX_SINGLE_SLOT_ADDRESSABLE_AREAS, SINGLE_SLOT_FIXTURES, + getCutoutIdForSlotName, getDeckDefFromRobotType, } from '@opentrons/shared-data' import { getLabwareSetupItemGroups } from '../utils' import { getProtocolUsesGripper } from '../../../organisms/ProtocolSetupInstruments/utils' import { useDeckConfigurationCompatibility } from '../../../resources/deck_configuration/hooks' -import { getCutoutIdForSlotName } from '../../../resources/deck_configuration/utils' import type { CompletedProtocolAnalysis, @@ -23,8 +23,8 @@ import type { CutoutId, ModuleModel, PipetteName, + ProtocolAnalysisOutput, RobotType, - RunTimeCommand, } from '@opentrons/shared-data' import type { LabwareSetupItem } from '../utils' import type { AttachedModule } from '@opentrons/api-client' @@ -63,7 +63,7 @@ export type ProtocolHardware = | ProtocolFixture export const useRequiredProtocolHardwareFromAnalysis = ( - analysis?: CompletedProtocolAnalysis | null + analysis: CompletedProtocolAnalysis | null ): { requiredProtocolHardware: ProtocolHardware[]; isLoading: boolean } => { const { data: attachedModulesData, @@ -82,7 +82,7 @@ export const useRequiredProtocolHardwareFromAnalysis = ( const { data: deckConfig = [] } = useDeckConfigurationQuery() const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - analysis?.commands ?? [] + analysis ) if (analysis == null || analysis?.status !== 'completed') { @@ -195,7 +195,7 @@ export const useRequiredProtocolHardware = ( { enabled: protocolData != null } ) - return useRequiredProtocolHardwareFromAnalysis(analysis) + return useRequiredProtocolHardwareFromAnalysis(analysis ?? null) } /** @@ -234,7 +234,7 @@ const useMissingProtocolHardwareFromRequiredProtocolHardware = ( requiredProtocolHardware: ProtocolHardware[], isLoading: boolean, robotType: RobotType, - protocolCommands: RunTimeCommand[] + protocolAnalysis: CompletedProtocolAnalysis | ProtocolAnalysisOutput | null ): { missingProtocolHardware: ProtocolHardware[] conflictedSlots: string[] @@ -242,7 +242,7 @@ const useMissingProtocolHardwareFromRequiredProtocolHardware = ( } => { const deckConfigCompatibility = useDeckConfigurationCompatibility( robotType, - protocolCommands + protocolAnalysis ) // determine missing or conflicted hardware @@ -283,7 +283,7 @@ const useMissingProtocolHardwareFromRequiredProtocolHardware = ( export const useMissingProtocolHardwareFromAnalysis = ( robotType: RobotType, - analysis?: CompletedProtocolAnalysis | null + analysis: CompletedProtocolAnalysis | null ): { missingProtocolHardware: ProtocolHardware[] conflictedSlots: string[] @@ -298,7 +298,7 @@ export const useMissingProtocolHardwareFromAnalysis = ( requiredProtocolHardware, isLoading, robotType, - analysis?.commands ?? [] + analysis ?? null ) } @@ -318,12 +318,12 @@ export const useMissingProtocolHardware = ( const { requiredProtocolHardware, isLoading, - } = useRequiredProtocolHardwareFromAnalysis(analysis) + } = useRequiredProtocolHardwareFromAnalysis(analysis ?? null) return useMissingProtocolHardwareFromRequiredProtocolHardware( requiredProtocolHardware, isLoading, FLEX_ROBOT_TYPE, - analysis?.commands ?? [] + analysis ?? null ) } diff --git a/app/src/resources/deck_configuration/__tests__/utils.test.ts b/app/src/resources/deck_configuration/__tests__/utils.test.ts deleted file mode 100644 index 759733fd18c..00000000000 --- a/app/src/resources/deck_configuration/__tests__/utils.test.ts +++ /dev/null @@ -1,142 +0,0 @@ -import { RunTimeCommand } from '@opentrons/shared-data' -import { - FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC, - getSimplestDeckConfigForProtocolCommands, -} from '../utils' - -const RUN_TIME_COMMAND_STUB_MIXIN: Pick< - RunTimeCommand, - 'id' | 'createdAt' | 'startedAt' | 'completedAt' | 'status' -> = { - id: 'fake_id', - createdAt: 'fake_createdAt', - startedAt: 'fake_startedAt', - completedAt: 'fake_createdAt', - status: 'succeeded', -} - -describe('getSimplestDeckConfigForProtocolCommands', () => { - it('returns simplest deck if no commands alter addressable areas', () => { - expect(getSimplestDeckConfigForProtocolCommands([])).toEqual( - FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC - ) - }) - it('returns staging area fixtures if commands address column 4 areas', () => { - const cutoutConfigs = getSimplestDeckConfigForProtocolCommands([ - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'loadLabware', - params: { - loadName: 'fake_load_name', - location: { slotName: 'A4' }, - version: 1, - namespace: 'fake_namespace', - }, - }, - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'loadLabware', - params: { - loadName: 'fake_load_name', - location: { slotName: 'B4' }, - version: 1, - namespace: 'fake_namespace', - }, - }, - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'loadLabware', - params: { - loadName: 'fake_load_name', - location: { slotName: 'C4' }, - version: 1, - namespace: 'fake_namespace', - }, - }, - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'loadLabware', - params: { - loadName: 'fake_load_name', - location: { slotName: 'D4' }, - version: 1, - namespace: 'fake_namespace', - }, - }, - ]) - expect(cutoutConfigs).toEqual([ - ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 8), - { - cutoutId: 'cutoutA3', - cutoutFixtureId: 'stagingAreaRightSlot', - requiredAddressableAreas: ['A4'], - }, - { - cutoutId: 'cutoutB3', - cutoutFixtureId: 'stagingAreaRightSlot', - requiredAddressableAreas: ['B4'], - }, - { - cutoutId: 'cutoutC3', - cutoutFixtureId: 'stagingAreaRightSlot', - requiredAddressableAreas: ['C4'], - }, - { - cutoutId: 'cutoutD3', - cutoutFixtureId: 'stagingAreaRightSlot', - requiredAddressableAreas: ['D4'], - }, - ]) - }) - it('returns simplest cutout fixture where many are possible', () => { - const cutoutConfigs = getSimplestDeckConfigForProtocolCommands([ - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'moveLabware', - params: { - newLocation: { addressableAreaName: 'gripperWasteChute' }, - labwareId: 'fake_labwareId', - strategy: 'usingGripper', - }, - }, - ]) - expect(cutoutConfigs).toEqual([ - ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 11), - { - cutoutId: 'cutoutD3', - cutoutFixtureId: 'wasteChuteRightAdapterNoCover', - requiredAddressableAreas: ['gripperWasteChute'], - }, - ]) - }) - it('returns compatible cutout fixture where multiple addressable requirements present', () => { - const cutoutConfigs = getSimplestDeckConfigForProtocolCommands([ - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'moveLabware', - params: { - newLocation: { addressableAreaName: 'gripperWasteChute' }, - labwareId: 'fake_labwareId', - strategy: 'usingGripper', - }, - }, - { - ...RUN_TIME_COMMAND_STUB_MIXIN, - commandType: 'moveLabware', - params: { - newLocation: { addressableAreaName: 'D4' }, - labwareId: 'fake_labwareId', - strategy: 'usingGripper', - }, - }, - ]) - expect(cutoutConfigs).toEqual([ - ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 11), - { - cutoutId: 'cutoutD3', - cutoutFixtureId: 'stagingAreaSlotWithWasteChuteRightAdapterNoCover', - requiredAddressableAreas: ['gripperWasteChute', 'D4'], - }, - ]) - }) -}) diff --git a/app/src/resources/deck_configuration/hooks.ts b/app/src/resources/deck_configuration/hooks.ts index 1581e6e16d9..7ea95eae3ce 100644 --- a/app/src/resources/deck_configuration/hooks.ts +++ b/app/src/resources/deck_configuration/hooks.ts @@ -2,32 +2,33 @@ import { useDeckConfigurationQuery } from '@opentrons/react-api-client' import { FLEX_ROBOT_TYPE, getAddressableAreasInProtocol, - getDeckDefFromRobotType, -} from '@opentrons/shared-data' - -import { getCutoutFixturesForCutoutId, getCutoutIdForAddressableArea, -} from './utils' + getDeckDefFromRobotType, +} from '@opentrons/shared-data' import type { + CompletedProtocolAnalysis, + CutoutConfigProtocolSpec, CutoutFixtureId, + ProtocolAnalysisOutput, RobotType, - RunTimeCommand, } from '@opentrons/shared-data' -import type { CutoutConfigProtocolSpec } from './utils' export interface CutoutConfigAndCompatibility extends CutoutConfigProtocolSpec { compatibleCutoutFixtureIds: CutoutFixtureId[] } export function useDeckConfigurationCompatibility( robotType: RobotType, - protocolCommands: RunTimeCommand[] + protocolAnalysis: CompletedProtocolAnalysis | ProtocolAnalysisOutput | null ): CutoutConfigAndCompatibility[] { const deckConfig = useDeckConfigurationQuery().data ?? [] if (robotType !== FLEX_ROBOT_TYPE) return [] const deckDef = getDeckDefFromRobotType(robotType) - const allAddressableAreas = getAddressableAreasInProtocol(protocolCommands) + const allAddressableAreas = + protocolAnalysis != null + ? getAddressableAreasInProtocol(protocolAnalysis) + : [] return deckConfig.reduce( (acc, { cutoutId, cutoutFixtureId }) => { const fixturesThatMountToCutoutId = getCutoutFixturesForCutoutId( diff --git a/app/src/resources/deck_configuration/utils.ts b/app/src/resources/deck_configuration/utils.ts index fb0e4a230d7..a02efa373c5 100644 --- a/app/src/resources/deck_configuration/utils.ts +++ b/app/src/resources/deck_configuration/utils.ts @@ -1,192 +1,11 @@ -import { - FLEX_ROBOT_TYPE, - FLEX_SINGLE_SLOT_ADDRESSABLE_AREAS, - getAddressableAreaFromSlotId, - getAddressableAreasInProtocol, - getDeckDefFromRobotType, -} from '@opentrons/shared-data' +import { FLEX_SINGLE_SLOT_ADDRESSABLE_AREAS } from '@opentrons/shared-data' import type { - CutoutConfig, - CutoutId, - RunTimeCommand, - CutoutFixture, - AddressableAreaName, - DeckDefinition, - DeckConfiguration, + CutoutConfigProtocolSpec, CutoutFixtureId, } from '@opentrons/shared-data' import type { CutoutConfigAndCompatibility } from './hooks' -export interface CutoutConfigProtocolSpec extends CutoutConfig { - requiredAddressableAreas: AddressableAreaName[] -} - -export const FLEX_SIMPLEST_DECK_CONFIG: DeckConfiguration = [ - { cutoutId: 'cutoutA1', cutoutFixtureId: 'singleLeftSlot' }, - { cutoutId: 'cutoutB1', cutoutFixtureId: 'singleLeftSlot' }, - { cutoutId: 'cutoutC1', cutoutFixtureId: 'singleLeftSlot' }, - { cutoutId: 'cutoutD1', cutoutFixtureId: 'singleLeftSlot' }, - { cutoutId: 'cutoutA2', cutoutFixtureId: 'singleCenterSlot' }, - { cutoutId: 'cutoutB2', cutoutFixtureId: 'singleCenterSlot' }, - { cutoutId: 'cutoutC2', cutoutFixtureId: 'singleCenterSlot' }, - { cutoutId: 'cutoutD2', cutoutFixtureId: 'singleCenterSlot' }, - { cutoutId: 'cutoutA3', cutoutFixtureId: 'singleRightSlot' }, - { cutoutId: 'cutoutB3', cutoutFixtureId: 'singleRightSlot' }, - { cutoutId: 'cutoutC3', cutoutFixtureId: 'singleRightSlot' }, - { cutoutId: 'cutoutD3', cutoutFixtureId: 'singleRightSlot' }, -] - -export const FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC: CutoutConfigProtocolSpec[] = FLEX_SIMPLEST_DECK_CONFIG.map( - config => ({ ...config, requiredAddressableAreas: [] }) -) - -export function getSimplestDeckConfigForProtocolCommands( - protocolAnalysisCommands: RunTimeCommand[] -): CutoutConfigProtocolSpec[] { - // TODO(BC, 2023-11-06): abstract out the robot type - const deckDef = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) - - const addressableAreas = getAddressableAreasInProtocol( - protocolAnalysisCommands - ) - const simplestDeckConfig = addressableAreas.reduce< - CutoutConfigProtocolSpec[] - >((acc, addressableArea) => { - const cutoutFixturesForAddressableArea = getCutoutFixturesForAddressableAreas( - [addressableArea], - deckDef.cutoutFixtures - ) - const cutoutIdForAddressableArea = getCutoutIdForAddressableArea( - addressableArea, - cutoutFixturesForAddressableArea - ) - const cutoutFixturesForCutoutId = - cutoutIdForAddressableArea != null - ? getCutoutFixturesForCutoutId( - cutoutIdForAddressableArea, - deckDef.cutoutFixtures - ) - : null - - const existingCutoutConfig = acc.find( - cutoutConfig => cutoutConfig.cutoutId === cutoutIdForAddressableArea - ) - - if ( - existingCutoutConfig != null && - cutoutFixturesForCutoutId != null && - cutoutIdForAddressableArea != null - ) { - const indexOfExistingFixture = cutoutFixturesForCutoutId.findIndex( - ({ id }) => id === existingCutoutConfig.cutoutFixtureId - ) - const accIndex = acc.findIndex( - ({ cutoutId }) => cutoutId === cutoutIdForAddressableArea - ) - const previousRequiredAAs = acc[accIndex]?.requiredAddressableAreas - const allNextRequiredAddressableAreas = previousRequiredAAs.includes( - addressableArea - ) - ? previousRequiredAAs - : [...previousRequiredAAs, addressableArea] - const nextCompatibleCutoutFixture = getSimplestFixtureForAddressableAreas( - cutoutIdForAddressableArea, - allNextRequiredAddressableAreas, - cutoutFixturesForCutoutId - ) - const indexOfCurrentFixture = cutoutFixturesForCutoutId.findIndex( - ({ id }) => id === nextCompatibleCutoutFixture?.id - ) - - if ( - nextCompatibleCutoutFixture != null && - indexOfCurrentFixture > indexOfExistingFixture - ) { - return [ - ...acc.slice(0, accIndex), - { - cutoutId: cutoutIdForAddressableArea, - cutoutFixtureId: nextCompatibleCutoutFixture.id, - requiredAddressableAreas: allNextRequiredAddressableAreas, - }, - ...acc.slice(accIndex + 1), - ] - } - } - return acc - }, FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC) - - return simplestDeckConfig -} - -export function getCutoutFixturesForAddressableAreas( - addressableAreas: AddressableAreaName[], - cutoutFixtures: CutoutFixture[] -): CutoutFixture[] { - return cutoutFixtures.filter(cutoutFixture => - Object.values(cutoutFixture.providesAddressableAreas).some(providedAAs => - addressableAreas.every(aa => providedAAs.includes(aa)) - ) - ) -} - -export function getCutoutFixturesForCutoutId( - cutoutId: CutoutId, - cutoutFixtures: CutoutFixture[] -): CutoutFixture[] { - return cutoutFixtures.filter(cutoutFixture => - cutoutFixture.mayMountTo.some(mayMountTo => mayMountTo.includes(cutoutId)) - ) -} - -export function getCutoutIdForSlotName( - slotName: string, - deckDef: DeckDefinition -): CutoutId | null { - const addressableArea = getAddressableAreaFromSlotId(slotName, deckDef) - const cutoutIdForSlotName = - addressableArea != null - ? getCutoutIdForAddressableArea( - addressableArea.id, - deckDef.cutoutFixtures - ) - : null - - return cutoutIdForSlotName -} - -export function getCutoutIdForAddressableArea( - addressableArea: AddressableAreaName, - cutoutFixtures: CutoutFixture[] -): CutoutId | null { - return cutoutFixtures.reduce((acc, cutoutFixture) => { - const [cutoutId] = - Object.entries( - cutoutFixture.providesAddressableAreas - ).find(([_cutoutId, providedAAs]) => - providedAAs.includes(addressableArea) - ) ?? [] - return (cutoutId as CutoutId) ?? acc - }, null) -} - -export function getSimplestFixtureForAddressableAreas( - cutoutId: CutoutId, - requiredAddressableAreas: AddressableAreaName[], - allCutoutFixtures: CutoutFixture[] -): CutoutFixture | null { - const cutoutFixturesForCutoutId = getCutoutFixturesForCutoutId( - cutoutId, - allCutoutFixtures - ) - const nextCompatibleCutoutFixtures = getCutoutFixturesForAddressableAreas( - requiredAddressableAreas, - cutoutFixturesForCutoutId - ) - return nextCompatibleCutoutFixtures?.[0] ?? null -} - export function getRequiredDeckConfig( deckConfigProtocolSpec: T[] ): T[] { diff --git a/components/src/hardware-sim/BaseDeck/BaseDeck.tsx b/components/src/hardware-sim/BaseDeck/BaseDeck.tsx index fdf08844906..3de5dfb5071 100644 --- a/components/src/hardware-sim/BaseDeck/BaseDeck.tsx +++ b/components/src/hardware-sim/BaseDeck/BaseDeck.tsx @@ -253,7 +253,10 @@ export function BaseDeck(props: BaseDeckProps): JSX.Element { }) => { if ( labwareLocation === 'offDeck' || - !('slotName' in labwareLocation) + !('slotName' in labwareLocation) || + // for legacy protocols that list fixed trash as a labware, do not render + definition.parameters.loadName === + 'opentrons_1_trash_3200ml_fixed' ) { return null } diff --git a/components/src/hardware-sim/ProtocolDeck/index.tsx b/components/src/hardware-sim/ProtocolDeck/index.tsx index faa8fc8763b..8f7aa808ef9 100644 --- a/components/src/hardware-sim/ProtocolDeck/index.tsx +++ b/components/src/hardware-sim/ProtocolDeck/index.tsx @@ -1,12 +1,12 @@ import * as React from 'react' -import { FLEX_ROBOT_TYPE } from '@opentrons/shared-data' +import { + FLEX_ROBOT_TYPE, + getSimplestDeckConfigForProtocol, +} from '@opentrons/shared-data' import { BaseDeck } from '../BaseDeck' -import { - getStandardDeckViewLayerBlockList, - getSimplestDeckConfigForProtocolCommands, -} from './utils' +import { getStandardDeckViewLayerBlockList } from './utils' import { getLabwareOnDeck } from './utils/getLabwareOnDeck' import { getModulesOnDeck } from './utils/getModulesOnDeck' @@ -30,9 +30,7 @@ export function ProtocolDeck(props: ProtocolDeckProps): JSX.Element | null { return null const robotType = protocolAnalysis.robotType ?? FLEX_ROBOT_TYPE - const deckConfig = getSimplestDeckConfigForProtocolCommands( - protocolAnalysis.commands - ) + const deckConfig = getSimplestDeckConfigForProtocol(protocolAnalysis) return ( = { + id: 'fake_id', + createdAt: 'fake_createdAt', + startedAt: 'fake_startedAt', + completedAt: 'fake_createdAt', + status: 'succeeded', +} + +// TODO(bh, 2023-12-4): test cases for legacy fixed trash +describe('getSimplestDeckConfigForProtocol', () => { + it('returns simplest deck if no commands alter addressable areas', () => { + expect(getSimplestDeckConfigForProtocol(null)).toEqual( + FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC + ) + }) + it('returns staging area fixtures if commands address column 4 areas', () => { + const cutoutConfigs = getSimplestDeckConfigForProtocol(({ + commands: [ + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'loadLabware', + params: { + loadName: 'fake_load_name', + location: { slotName: 'A4' }, + version: 1, + namespace: 'fake_namespace', + }, + }, + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'loadLabware', + params: { + loadName: 'fake_load_name', + location: { slotName: 'B4' }, + version: 1, + namespace: 'fake_namespace', + }, + }, + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'loadLabware', + params: { + loadName: 'fake_load_name', + location: { slotName: 'C4' }, + version: 1, + namespace: 'fake_namespace', + }, + }, + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'loadLabware', + params: { + loadName: 'fake_load_name', + location: { slotName: 'D4' }, + version: 1, + namespace: 'fake_namespace', + }, + }, + ], + labware: [], + } as unknown) as CompletedProtocolAnalysis) + expect(cutoutConfigs).toEqual([ + ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 8), + { + cutoutId: 'cutoutA3', + cutoutFixtureId: 'stagingAreaRightSlot', + requiredAddressableAreas: ['A4'], + }, + { + cutoutId: 'cutoutB3', + cutoutFixtureId: 'stagingAreaRightSlot', + requiredAddressableAreas: ['B4'], + }, + { + cutoutId: 'cutoutC3', + cutoutFixtureId: 'stagingAreaRightSlot', + requiredAddressableAreas: ['C4'], + }, + { + cutoutId: 'cutoutD3', + cutoutFixtureId: 'stagingAreaRightSlot', + requiredAddressableAreas: ['D4'], + }, + ]) + }) + it('returns simplest cutout fixture where many are possible', () => { + const cutoutConfigs = getSimplestDeckConfigForProtocol(({ + commands: [ + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'moveLabware', + params: { + newLocation: { addressableAreaName: 'gripperWasteChute' }, + labwareId: 'fake_labwareId', + strategy: 'usingGripper', + }, + }, + ], + labware: [], + } as unknown) as CompletedProtocolAnalysis) + expect(cutoutConfigs).toEqual([ + ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 11), + { + cutoutId: 'cutoutD3', + cutoutFixtureId: 'wasteChuteRightAdapterNoCover', + requiredAddressableAreas: ['gripperWasteChute'], + }, + ]) + }) + it('returns compatible cutout fixture where multiple addressable requirements present', () => { + const cutoutConfigs = getSimplestDeckConfigForProtocol(({ + commands: [ + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'moveLabware', + params: { + newLocation: { addressableAreaName: 'gripperWasteChute' }, + labwareId: 'fake_labwareId', + strategy: 'usingGripper', + }, + }, + { + ...RUN_TIME_COMMAND_STUB_MIXIN, + commandType: 'moveLabware', + params: { + newLocation: { addressableAreaName: 'D4' }, + labwareId: 'fake_labwareId', + strategy: 'usingGripper', + }, + }, + ], + labware: [], + } as unknown) as CompletedProtocolAnalysis) + expect(cutoutConfigs).toEqual([ + ...FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC.slice(0, 11), + { + cutoutId: 'cutoutD3', + cutoutFixtureId: 'stagingAreaSlotWithWasteChuteRightAdapterNoCover', + requiredAddressableAreas: ['gripperWasteChute', 'D4'], + }, + ]) + }) +}) diff --git a/shared-data/js/helpers/getAddressableAreasInProtocol.ts b/shared-data/js/helpers/getAddressableAreasInProtocol.ts index f9e6e45c2cb..fae6fc0d276 100644 --- a/shared-data/js/helpers/getAddressableAreasInProtocol.ts +++ b/shared-data/js/helpers/getAddressableAreasInProtocol.ts @@ -1,59 +1,97 @@ +import { MOVABLE_TRASH_A3_ADDRESSABLE_AREA } from '../constants' + import type { AddressableAreaName } from '../../deck' -import type { RunTimeCommand } from '../../protocol' +import type { ProtocolAnalysisOutput } from '../../protocol' +import type { CompletedProtocolAnalysis } from '../types' export function getAddressableAreasInProtocol( - commands: RunTimeCommand[] + protocolAnalysis: CompletedProtocolAnalysis | ProtocolAnalysisOutput ): AddressableAreaName[] { - return commands.reduce((acc, command) => { - if ( - command.commandType === 'moveLabware' && - command.params.newLocation !== 'offDeck' && - 'slotName' in command.params.newLocation && - !acc.includes(command.params.newLocation.slotName as AddressableAreaName) - ) { - return [ - ...acc, - command.params.newLocation.slotName as AddressableAreaName, - ] - } else if ( - command.commandType === 'moveLabware' && - command.params.newLocation !== 'offDeck' && - 'addressableAreaName' in command.params.newLocation && - !acc.includes( - command.params.newLocation.addressableAreaName as AddressableAreaName - ) - ) { - return [ - ...acc, - command.params.newLocation.addressableAreaName as AddressableAreaName, - ] - } else if ( - (command.commandType === 'loadLabware' || - command.commandType === 'loadModule') && - command.params.location !== 'offDeck' && - 'slotName' in command.params.location && - !acc.includes(command.params.location.slotName as AddressableAreaName) - ) { - return [...acc, command.params.location.slotName as AddressableAreaName] - } else if ( - command.commandType === 'loadLabware' && - command.params.location !== 'offDeck' && - 'addressableAreaName' in command.params.location && - !acc.includes( - command.params.location.addressableAreaName as AddressableAreaName - ) - ) { - return [ - ...acc, - command.params.location.addressableAreaName as AddressableAreaName, - ] - } else if ( - command.commandType === 'moveToAddressableArea' && - !acc.includes(command.params.addressableAreaName as AddressableAreaName) - ) { - return [...acc, command.params.addressableAreaName as AddressableAreaName] - } else { - return acc - } - }, []) + const { commands, labware } = protocolAnalysis + + const addressableAreasFromCommands = commands.reduce( + (acc, command) => { + if ( + command.commandType === 'moveLabware' && + command.params.newLocation !== 'offDeck' && + 'slotName' in command.params.newLocation && + !acc.includes( + command.params.newLocation.slotName as AddressableAreaName + ) + ) { + return [ + ...acc, + command.params.newLocation.slotName as AddressableAreaName, + ] + } else if ( + command.commandType === 'moveLabware' && + command.params.newLocation !== 'offDeck' && + 'addressableAreaName' in command.params.newLocation && + !acc.includes( + command.params.newLocation.addressableAreaName as AddressableAreaName + ) + ) { + return [ + ...acc, + command.params.newLocation.addressableAreaName as AddressableAreaName, + ] + } else if ( + (command.commandType === 'loadLabware' || + command.commandType === 'loadModule') && + command.params.location !== 'offDeck' && + 'slotName' in command.params.location && + !acc.includes(command.params.location.slotName as AddressableAreaName) + ) { + // do not add addressable area name for legacy trash labware + if ( + 'loadName' in command.params && + command.params.loadName === 'opentrons_1_trash_3200ml_fixed' + ) { + return acc + } else { + // TODO(bh, 2023-12-4): use getAddressableAreaFromSlotId helper + return [ + ...acc, + command.params.location.slotName as AddressableAreaName, + ] + } + } else if ( + command.commandType === 'loadLabware' && + command.params.location !== 'offDeck' && + 'addressableAreaName' in command.params.location && + !acc.includes( + command.params.location.addressableAreaName as AddressableAreaName + ) + ) { + return [ + ...acc, + command.params.location.addressableAreaName as AddressableAreaName, + ] + } else if ( + command.commandType === 'moveToAddressableArea' && + !acc.includes(command.params.addressableAreaName as AddressableAreaName) + ) { + return [ + ...acc, + command.params.addressableAreaName as AddressableAreaName, + ] + } else { + return acc + } + }, + [] + ) + + // special-case the Flex trash labware load name in A3 for back compatibility with the legacy fixed trash load labware command + const legacyTrashAddressableArea = labware.some( + ({ loadName, location }) => + loadName === 'opentrons_1_trash_3200ml_fixed' && + location !== 'offDeck' && + 'slotName' in location && + location.slotName === 'A3' + ) + ? MOVABLE_TRASH_A3_ADDRESSABLE_AREA + : [] + + return addressableAreasFromCommands.concat(legacyTrashAddressableArea) } diff --git a/components/src/hardware-sim/ProtocolDeck/utils/getSimplestFlexDeckConfig.ts b/shared-data/js/helpers/getSimplestFlexDeckConfig.ts similarity index 90% rename from components/src/hardware-sim/ProtocolDeck/utils/getSimplestFlexDeckConfig.ts rename to shared-data/js/helpers/getSimplestFlexDeckConfig.ts index ad5c081bcae..c922d5fc64f 100644 --- a/components/src/hardware-sim/ProtocolDeck/utils/getSimplestFlexDeckConfig.ts +++ b/shared-data/js/helpers/getSimplestFlexDeckConfig.ts @@ -1,19 +1,16 @@ -import { - FLEX_ROBOT_TYPE, - getAddressableAreaFromSlotId, - getAddressableAreasInProtocol, - getDeckDefFromRobotType, -} from '@opentrons/shared-data' +import { FLEX_ROBOT_TYPE } from '../constants' +import { getAddressableAreaFromSlotId } from '../fixtures' +import { getAddressableAreasInProtocol, getDeckDefFromRobotType } from '.' +import type { AddressableAreaName, CutoutId } from '../../deck' +import type { ProtocolAnalysisOutput } from '../../protocol' import type { CutoutConfig, - CutoutId, - RunTimeCommand, CutoutFixture, - AddressableAreaName, DeckDefinition, DeckConfiguration, -} from '@opentrons/shared-data' + CompletedProtocolAnalysis, +} from '../types' export interface CutoutConfigProtocolSpec extends CutoutConfig { requiredAddressableAreas: AddressableAreaName[] @@ -38,15 +35,16 @@ export const FLEX_SIMPLEST_DECK_CONFIG_PROTOCOL_SPEC: CutoutConfigProtocolSpec[] config => ({ ...config, requiredAddressableAreas: [] }) ) -export function getSimplestDeckConfigForProtocolCommands( - protocolAnalysisCommands: RunTimeCommand[] +export function getSimplestDeckConfigForProtocol( + protocolAnalysis: CompletedProtocolAnalysis | ProtocolAnalysisOutput | null ): CutoutConfigProtocolSpec[] { // TODO(BC, 2023-11-06): abstract out the robot type const deckDef = getDeckDefFromRobotType(FLEX_ROBOT_TYPE) - const addressableAreas = getAddressableAreasInProtocol( - protocolAnalysisCommands - ) + const addressableAreas = + protocolAnalysis != null + ? getAddressableAreasInProtocol(protocolAnalysis) + : [] const simplestDeckConfig = addressableAreas.reduce< CutoutConfigProtocolSpec[] >((acc, addressableArea) => { diff --git a/shared-data/js/helpers/index.ts b/shared-data/js/helpers/index.ts index cccc025b040..f96f38ff2a5 100644 --- a/shared-data/js/helpers/index.ts +++ b/shared-data/js/helpers/index.ts @@ -28,6 +28,7 @@ export * from './getLoadedLabwareDefinitionsByUri' export * from './getOccludedSlotCountForModule' export * from './labwareInference' export * from './getAddressableAreasInProtocol' +export * from './getSimplestFlexDeckConfig' export const getLabwareDefIsStandard = (def: LabwareDefinition2): boolean => def?.namespace === OPENTRONS_LABWARE_NAMESPACE