From ab123758db68222d26f55059b34e4accacbb37d1 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 31 Jul 2024 10:43:29 -0400 Subject: [PATCH 1/2] implement the command in ER --- .../__tests__/useRecoveryCommands.test.ts | 46 ++++++++++++++++--- .../hooks/useRecoveryCommands.ts | 37 +++++++++++++-- 2 files changed, 73 insertions(+), 10 deletions(-) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts index df6ccebaa87..a55f3ef43f2 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/__tests__/useRecoveryCommands.test.ts @@ -4,6 +4,7 @@ import { renderHook, act } from '@testing-library/react' import { useResumeRunFromRecoveryMutation, useStopRunMutation, + useUpdateErrorRecoveryPolicy, } from '@opentrons/react-api-client' import { useChainRunCommands } from '../../../../resources/runs' @@ -11,6 +12,7 @@ import { useRecoveryCommands, HOME_PIPETTE_Z_AXES, buildPickUpTips, + buildIgnorePolicyRules, } from '../useRecoveryCommands' import { RECOVERY_MAP } from '../../constants' @@ -40,6 +42,7 @@ describe('useRecoveryCommands', () => { const mockChainRunCommands = vi.fn().mockResolvedValue([]) const mockReportActionSelectedResult = vi.fn() const mockReportRecoveredRunResult = vi.fn() + const mockUpdateErrorRecoveryPolicy = vi.fn() const props = { runId: mockRunId, @@ -64,6 +67,9 @@ describe('useRecoveryCommands', () => { vi.mocked(useChainRunCommands).mockReturnValue({ chainRunCommands: mockChainRunCommands, } as any) + vi.mocked(useUpdateErrorRecoveryPolicy).mockReturnValue({ + updateErrorRecoveryPolicy: mockUpdateErrorRecoveryPolicy, + } as any) }) it('should call chainRunRecoveryCommands with continuePastCommandFailure set to false', async () => { @@ -254,18 +260,46 @@ describe('useRecoveryCommands', () => { expect(mockMakeSuccessToast).toHaveBeenCalled() }) - it('should call ignoreErrorKindThisRun and resolve immediately', async () => { - const { result } = renderHook(() => useRecoveryCommands(props)) + it('should call updateErrorRecoveryPolicy with correct policy rules when failedCommand has an error', async () => { + const mockFailedCommandWithError = { + ...mockFailedCommand, + commandType: 'aspirateInPlace', + error: { + errorType: 'mockErrorType', + }, + } - const consoleSpy = vi.spyOn(console, 'log') + const testProps = { + ...props, + failedCommand: mockFailedCommandWithError, + } + + const { result } = renderHook(() => useRecoveryCommands(testProps)) await act(async () => { await result.current.ignoreErrorKindThisRun() }) - expect(consoleSpy).toHaveBeenCalledWith( - 'IGNORING ALL ERRORS OF THIS KIND THIS RUN' + const expectedPolicyRules = buildIgnorePolicyRules( + 'aspirateInPlace', + 'mockErrorType' + ) + + expect(mockUpdateErrorRecoveryPolicy).toHaveBeenCalledWith( + expectedPolicyRules + ) + }) + + it('should reject with an error when failedCommand or error is null', async () => { + const testProps = { + ...props, + failedCommand: null, + } + + const { result } = renderHook(() => useRecoveryCommands(testProps)) + + await expect(result.current.ignoreErrorKindThisRun()).rejects.toThrow( + 'Could not execute command. No failed command.' ) - expect(result.current.ignoreErrorKindThisRun()).resolves.toBeUndefined() }) }) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts index c33bce43416..803bdf18f6a 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useRecoveryCommands.ts @@ -4,6 +4,7 @@ import head from 'lodash/head' import { useResumeRunFromRecoveryMutation, useStopRunMutation, + useUpdateErrorRecoveryPolicy, } from '@opentrons/react-api-client' import { useChainRunCommands } from '../../../resources/runs' @@ -19,7 +20,10 @@ import type { DropTipInPlaceRunTimeCommand, PrepareToAspirateRunTimeCommand, } from '@opentrons/shared-data' -import type { CommandData } from '@opentrons/api-client' +import type { + CommandData, + RecoveryPolicyRulesParams, +} from '@opentrons/api-client' import type { WellGroup } from '@opentrons/components' import type { FailedCommand } from '../types' import type { UseFailedLabwareUtilsResult } from './useFailedLabwareUtils' @@ -71,6 +75,7 @@ export function useRecoveryCommands({ mutateAsync: resumeRunFromRecovery, } = useResumeRunFromRecoveryMutation() const { stopRun } = useStopRunMutation() + const { updateErrorRecoveryPolicy } = useUpdateErrorRecoveryPolicy(runId) const { makeSuccessToast } = recoveryToastUtils const buildRetryPrepMove = (): MoveToCoordinatesCreateCommand | null => { @@ -184,9 +189,20 @@ export function useRecoveryCommands({ }, [runId, resumeRunFromRecovery, makeSuccessToast]) const ignoreErrorKindThisRun = React.useCallback((): Promise => { - console.log('IGNORING ALL ERRORS OF THIS KIND THIS RUN') - return Promise.resolve() - }, []) + if (failedCommand?.error != null) { + const ignorePolicyRules = buildIgnorePolicyRules( + failedCommand.commandType, + failedCommand.error.errorType + ) + + updateErrorRecoveryPolicy(ignorePolicyRules) + return Promise.resolve() + } else { + return Promise.reject( + new Error('Could not execute command. No failed command.') + ) + } + }, [failedCommand?.error?.errorType, failedCommand?.commandType]) return { resumeRun, @@ -230,3 +246,16 @@ export const buildPickUpTips = ( } } } + +export const buildIgnorePolicyRules = ( + commandType: FailedCommand['commandType'], + errorType: string +): RecoveryPolicyRulesParams => { + return [ + { + commandType, + errorType, + ifMatch: 'ignoreAndContinue', + }, + ] +} From ea1a9277d88ebcd9b15228f738d2a55cc92a30b7 Mon Sep 17 00:00:00 2001 From: Jamey Huffnagle Date: Wed, 31 Jul 2024 11:26:29 -0400 Subject: [PATCH 2/2] better use effects --- .../organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts | 2 +- app/src/organisms/ErrorRecoveryFlows/index.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts index a5cdcc8ca19..927b867752b 100644 --- a/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts +++ b/app/src/organisms/ErrorRecoveryFlows/hooks/useFailedLabwareUtils.ts @@ -57,7 +57,7 @@ export function useFailedLabwareUtils({ }: UseFailedLabwareUtilsProps): UseFailedLabwareUtilsResult { const recentRelevantFailedLabwareCmd = React.useMemo( () => getRelevantFailedLabwareCmdFrom({ failedCommand, runCommands }), - [failedCommand?.key, runCommands] + [failedCommand?.error?.errorType, runCommands] ) const tipSelectionUtils = useTipSelectionUtils(recentRelevantFailedLabwareCmd) diff --git a/app/src/organisms/ErrorRecoveryFlows/index.tsx b/app/src/organisms/ErrorRecoveryFlows/index.tsx index 9ccea140663..76b5d6b07bc 100644 --- a/app/src/organisms/ErrorRecoveryFlows/index.tsx +++ b/app/src/organisms/ErrorRecoveryFlows/index.tsx @@ -121,7 +121,7 @@ export function ErrorRecoveryFlows( const analytics = useRecoveryAnalytics() React.useEffect(() => { analytics.reportErrorEvent(failedCommand) - }, [failedCommand?.key]) + }, [failedCommand?.error?.detail]) const { hasLaunchedRecovery, toggleERWizard, showERWizard } = useERWizard() const isOnDevice = useSelector(getIsOnDevice)