From 61478d28e170d85e1eb430365bc7c4abe4143ff4 Mon Sep 17 00:00:00 2001 From: TamarZanzouri Date: Thu, 1 Aug 2024 15:20:21 -0400 Subject: [PATCH 1/2] feature(robot-server, api): add whether the run has entered recovery mode in GET run by id (#15843) # Overview part of https://opentrons.atlassian.net/browse/EXEC-504. add full command error list to commands state. add whether the run has entered recovery mode and return it in GET /run/{runId} ## Test Plan and Hands on Testing Tested on dev server, Should test on a robot as well. - manually raise `TipNotAttachedError` from pickup tip command. - posted a protocol with a pickup tip command. - GET `/run/{runId}` should return the full error list in the response. `{ "data": { "id": "2a3375f6-0773-445f-aefd-2108bc7d376e", "ok": true, "createdAt": "2024-07-31T13:36:25.543796+00:00", "status": "stopped", "current": true, "actions": [ { "id": "3a3204f0-a8f0-4fbf-a5c3-19261f7a465f", "createdAt": "2024-07-31T13:36:39.326489+00:00", "actionType": "play" }, { "id": "6d66da93-fddb-4a9d-a4e1-5966158c1945", "createdAt": "2024-07-31T13:37:18.836018+00:00", "actionType": "resume-from-recovery" }, { "id": "8a54bc88-e5f3-48bc-ba80-411f2870515d", "createdAt": "2024-07-31T13:37:54.278171+00:00", "actionType": "stop" } ], "errors": [], "hasEverEnteredErrorRecovery": true, "pipettes": [ { "id": "2a652542-3b81-4c03-8a60-d3f7c33cb5b1", "pipetteName": "p1000_single_flex", "mount": "left" } ], "modules": [], "labware": [ { "id": "562293ac-6120-4bdb-9a0f-3a2736c4dae3", "loadName": "opentrons_flex_96_tiprack_50ul", "definitionUri": "opentrons/opentrons_flex_96_tiprack_50ul/1", "location": { "slotName": "C2" } }, { "id": "5e7c2ff5-46dd-443c-abff-f4e2c2e82473", "loadName": "opentrons_96_wellplate_200ul_pcr_full_skirt", "definitionUri": "opentrons/opentrons_96_wellplate_200ul_pcr_full_skirt/2", "location": { "slotName": "C3" } } ], "liquids": [], "labwareOffsets": [], "runTimeParameters": [], "protocolId": "b93d9f73-bb43-4c57-ba2c-16aedc36d06f", "completedAt": "2024-07-31T13:37:54.295804+00:00", "startedAt": "2024-07-31T13:36:39.330993+00:00" } }` ## Changelog - every `FailCommandAction` will now append the `ErrorOccurenece` to the full error list. - set state has_entered_error_recovery to True when issuing a `ResumeFromRecoveryAction`. - `StateSummery` will get hasEverEnteredErrorRecovery. - `hasEverEnteredErrorRecovery` added the `BadRun` and `Run` result. - `hasEverEnteredErrorRecovery` added to ts LegacyRun ## Review requests changes make sense? ## Risk assessment low. should not effect existing code. --------- Co-authored-by: Max Marrone Co-authored-by: Seth Foster --- api-client/src/runs/types.ts | 1 + api/src/opentrons/cli/analyze.py | 1 + .../protocol_engine/state/commands.py | 18 +++++++ .../opentrons/protocol_engine/state/state.py | 1 + .../protocol_engine/state/state_summary.py | 1 + .../state/test_command_state.py | 54 +++++++++++++++++-- .../state/test_command_store_old.py | 22 ++++++++ .../state/test_command_view_old.py | 4 ++ .../InterventionModal/__fixtures__/index.ts | 1 + .../RecentRunProtocolCarousel.test.tsx | 1 + .../RunTimeControl/__fixtures__/index.ts | 8 +++ .../src/runs/__fixtures__/runs.ts | 2 + .../maintenance_run_data_manager.py | 2 + .../maintenance_run_models.py | 4 ++ .../robot_server/runs/run_data_manager.py | 3 ++ robot-server/robot_server/runs/run_models.py | 8 +++ .../test_json_v6_protocol_run.tavern.yaml | 1 + .../test_json_v7_protocol_run.tavern.yaml | 1 + .../runs/test_protocol_run.tavern.yaml | 2 + ...t_run_queued_protocol_commands.tavern.yaml | 1 + ...t_run_with_run_time_parameters.tavern.yaml | 1 + .../router/test_base_router.py | 4 ++ .../router/test_labware_router.py | 1 + .../maintenance_runs/test_run_data_manager.py | 4 ++ .../tests/protocols/test_protocol_analyzer.py | 1 + .../tests/runs/router/test_base_router.py | 8 +++ .../tests/runs/router/test_labware_router.py | 1 + .../tests/runs/test_run_controller.py | 1 + .../tests/runs/test_run_data_manager.py | 12 +++++ robot-server/tests/runs/test_run_store.py | 2 + 30 files changed, 167 insertions(+), 4 deletions(-) diff --git a/api-client/src/runs/types.ts b/api-client/src/runs/types.ts index 1986a34a0b8..19127b70bba 100644 --- a/api-client/src/runs/types.ts +++ b/api-client/src/runs/types.ts @@ -47,6 +47,7 @@ export interface LegacyGoodRunData { status: RunStatus actions: RunAction[] errors: RunError[] + hasEverEnteredErrorRecovery: boolean pipettes: LoadedPipette[] labware: LoadedLabware[] liquids: Liquid[] diff --git a/api/src/opentrons/cli/analyze.py b/api/src/opentrons/cli/analyze.py index 0daacd711b3..7270e517644 100644 --- a/api/src/opentrons/cli/analyze.py +++ b/api/src/opentrons/cli/analyze.py @@ -275,6 +275,7 @@ async def _do_analyze(protocol_source: ProtocolSource) -> RunResult: modules=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ), parameters=[], ) diff --git a/api/src/opentrons/protocol_engine/state/commands.py b/api/src/opentrons/protocol_engine/state/commands.py index 9989f9aec01..1ad17867450 100644 --- a/api/src/opentrons/protocol_engine/state/commands.py +++ b/api/src/opentrons/protocol_engine/state/commands.py @@ -203,6 +203,12 @@ class CommandState: This value can be used to generate future hashes. """ + failed_command_errors: List[ErrorOccurrence] + """List of command errors that occurred during run execution.""" + + has_entered_error_recovery: bool + """Whether the run has entered error recovery.""" + stopped_by_estop: bool """If this is set to True, the engine was stopped by an estop event.""" @@ -238,7 +244,9 @@ def __init__( run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=error_recovery_policy, + has_entered_error_recovery=False, ) def handle_action(self, action: Action) -> None: @@ -330,6 +338,7 @@ def _handle_fail_command_action(self, action: FailCommandAction) -> None: notes=action.notes, ) self._state.failed_command = self._state.command_history.get(action.command_id) + self._state.failed_command_errors.append(public_error_occurrence) other_command_ids_to_fail: List[str] if prev_entry.command.intent == CommandIntent.SETUP: @@ -373,6 +382,7 @@ def _handle_fail_command_action(self, action: FailCommandAction) -> None: ): self._state.queue_status = QueueStatus.AWAITING_RECOVERY self._state.recovery_target_command_id = action.command_id + self._state.has_entered_error_recovery = True def _handle_play_action(self, action: PlayAction) -> None: if not self._state.run_result: @@ -635,6 +645,14 @@ def get_error(self) -> Optional[ErrorOccurrence]: else: return run_error or finish_error + def get_all_errors(self) -> List[ErrorOccurrence]: + """Get the run's full error list, if there was none, returns an empty list.""" + return self._state.failed_command_errors + + def get_has_entered_recovery_mode(self) -> bool: + """Get whether the run has entered recovery mode.""" + return self._state.has_entered_error_recovery + def get_running_command_id(self) -> Optional[str]: """Return the ID of the command that's currently running, if there is one.""" running_command = self._state.command_history.get_running_command() diff --git a/api/src/opentrons/protocol_engine/state/state.py b/api/src/opentrons/protocol_engine/state/state.py index 430ca1e5738..a05d529f50a 100644 --- a/api/src/opentrons/protocol_engine/state/state.py +++ b/api/src/opentrons/protocol_engine/state/state.py @@ -129,6 +129,7 @@ def get_summary(self) -> StateSummary: completedAt=self._state.commands.run_completed_at, startedAt=self._state.commands.run_started_at, liquids=self._liquid.get_all(), + hasEverEnteredErrorRecovery=self._commands.get_has_entered_recovery_mode(), ) diff --git a/api/src/opentrons/protocol_engine/state/state_summary.py b/api/src/opentrons/protocol_engine/state/state_summary.py index c7185cc2c0d..7e6e003aaa8 100644 --- a/api/src/opentrons/protocol_engine/state/state_summary.py +++ b/api/src/opentrons/protocol_engine/state/state_summary.py @@ -21,6 +21,7 @@ class StateSummary(BaseModel): # errors is a list for historical reasons. (This model needs to stay compatible with # robot-server's database.) It shouldn't have more than 1 element. errors: List[ErrorOccurrence] + hasEverEnteredErrorRecovery: bool = Field(default=False) labware: List[LoadedLabware] pipettes: List[LoadedPipette] modules: List[LoadedModule] diff --git a/api/tests/opentrons/protocol_engine/state/test_command_state.py b/api/tests/opentrons/protocol_engine/state/test_command_state.py index 9ebb338d85c..afafcc3cabe 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_state.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_state.py @@ -192,6 +192,7 @@ def test_command_failure(error_recovery_type: ErrorRecoveryType) -> None: ) assert subject_view.get("command-id") == expected_failed_command + assert subject.state.failed_command_errors == [expected_error_occurrence] def test_command_failure_clears_queues() -> None: @@ -227,12 +228,20 @@ def test_command_failure_clears_queues() -> None: started_at=datetime(year=2022, month=2, day=2), ) subject.handle_action(run_1) + expected_error = errors.ProtocolEngineError(message="oh no") + expected_error_occurance = errors.ErrorOccurrence( + id="error-id", + errorType="ProtocolEngineError", + createdAt=datetime(year=2023, month=3, day=3), + detail="oh no", + errorCode=ErrorCodes.GENERAL_ERROR.value.code, + ) fail_1 = actions.FailCommandAction( command_id="command-id-1", running_command=subject_view.get("command-id-1"), error_id="error-id", failed_at=datetime(year=2023, month=3, day=3), - error=errors.ProtocolEngineError(message="oh no"), + error=expected_error, notes=[], type=ErrorRecoveryType.FAIL_RUN, ) @@ -245,6 +254,7 @@ def test_command_failure_clears_queues() -> None: assert subject_view.get_running_command_id() is None assert subject_view.get_queue_ids() == OrderedSet() assert subject_view.get_next_to_execute() is None + assert subject.state.failed_command_errors == [expected_error_occurance] def test_setup_command_failure_only_clears_setup_command_queue() -> None: @@ -489,12 +499,20 @@ def test_door_during_error_recovery() -> None: started_at=datetime(year=2022, month=2, day=2), ) subject.handle_action(run_1) + expected_error = errors.ProtocolEngineError(message="oh no") + expected_error_occurance = errors.ErrorOccurrence( + id="error-id", + errorType="ProtocolEngineError", + createdAt=datetime(year=2023, month=3, day=3), + detail="oh no", + errorCode=ErrorCodes.GENERAL_ERROR.value.code, + ) fail_1 = actions.FailCommandAction( command_id="command-id-1", running_command=subject_view.get("command-id-1"), error_id="error-id", failed_at=datetime(year=2023, month=3, day=3), - error=errors.ProtocolEngineError(message="oh no"), + error=expected_error, notes=[], type=ErrorRecoveryType.WAIT_FOR_RECOVERY, ) @@ -536,6 +554,7 @@ def test_door_during_error_recovery() -> None: subject.handle_action(play) assert subject_view.get_status() == EngineStatus.AWAITING_RECOVERY assert subject_view.get_next_to_execute() == "command-id-2" + assert subject.state.failed_command_errors == [expected_error_occurance] @pytest.mark.parametrize( @@ -605,7 +624,7 @@ def test_error_recovery_type_tracking() -> None: command_id="c1", running_command=running_command_1, error_id="c1-error", - failed_at=datetime.now(), + failed_at=datetime(year=2023, month=3, day=3), error=PythonException(RuntimeError("new sheriff in town")), notes=[], type=ErrorRecoveryType.WAIT_FOR_RECOVERY, @@ -620,7 +639,7 @@ def test_error_recovery_type_tracking() -> None: command_id="c2", running_command=running_command_2, error_id="c2-error", - failed_at=datetime.now(), + failed_at=datetime(year=2023, month=3, day=3), error=PythonException(RuntimeError("new sheriff in town")), notes=[], type=ErrorRecoveryType.FAIL_RUN, @@ -631,6 +650,19 @@ def test_error_recovery_type_tracking() -> None: assert view.get_error_recovery_type("c1") == ErrorRecoveryType.WAIT_FOR_RECOVERY assert view.get_error_recovery_type("c2") == ErrorRecoveryType.FAIL_RUN + exception = PythonException(RuntimeError("new sheriff in town")) + error_occurrence_1 = ErrorOccurrence.from_failed( + id="c1-error", createdAt=datetime(year=2023, month=3, day=3), error=exception + ) + error_occurrence_2 = ErrorOccurrence.from_failed( + id="c2-error", createdAt=datetime(year=2023, month=3, day=3), error=exception + ) + + assert subject.state.failed_command_errors == [ + error_occurrence_1, + error_occurrence_2, + ] + def test_recovery_target_tracking() -> None: """It should keep track of the command currently undergoing error recovery.""" @@ -729,6 +761,8 @@ def test_recovery_target_tracking() -> None: assert subject_view.get_recovery_target() is None assert not subject_view.get_recovery_in_progress_for_command("c3") + assert subject_view.get_has_entered_recovery_mode() is True + def test_final_state_after_estop() -> None: """Test the final state of the run after it's E-stopped.""" @@ -761,6 +795,7 @@ def test_final_state_after_estop() -> None: assert subject_view.get_status() == EngineStatus.FAILED assert subject_view.get_error() == expected_error_occurrence + assert subject_view.get_all_errors() == [] def test_final_state_after_stop() -> None: @@ -833,6 +868,13 @@ def test_final_state_after_error_recovery_stop() -> None: notes=[], type=ErrorRecoveryType.WAIT_FOR_RECOVERY, ) + expected_error_occurrence_1 = ErrorOccurrence( + id="error-id", + createdAt=datetime(year=2023, month=3, day=3), + errorCode=ErrorCodes.GENERAL_ERROR.value.code, + errorType="ProtocolEngineError", + detail="oh no", + ) subject.handle_action(fail_1) assert subject_view.get_status() == EngineStatus.AWAITING_RECOVERY @@ -856,9 +898,13 @@ def test_final_state_after_error_recovery_stop() -> None: finish_error_details=None, ) ) + assert subject_view.get_status() == EngineStatus.STOPPED assert subject_view.get_recovery_target() is None assert subject_view.get_error() is None + assert subject_view.get_all_errors() == [ + expected_error_occurrence_1, + ] def test_set_and_get_error_recovery_policy() -> None: diff --git a/api/tests/opentrons/protocol_engine/state/test_command_store_old.py b/api/tests/opentrons/protocol_engine/state/test_command_store_old.py index 92fba9b4851..018634db435 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_store_old.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_store_old.py @@ -337,7 +337,9 @@ def test_command_store_handles_pause_action(pause_source: PauseSource) -> None: recovery_target_command_id=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) @@ -365,7 +367,9 @@ def test_command_store_handles_play_action(pause_source: PauseSource) -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -398,7 +402,9 @@ def test_command_store_handles_finish_action() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -451,7 +457,9 @@ def test_command_store_handles_stop_action( run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=from_estop, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -487,7 +495,9 @@ def test_command_store_handles_stop_action_when_awaiting_recovery() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -519,7 +529,9 @@ def test_command_store_cannot_restart_after_should_stop() -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -664,7 +676,9 @@ def test_command_store_wraps_unknown_errors() -> None: recovery_target_command_id=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -732,7 +746,9 @@ def __init__(self, message: str) -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -766,7 +782,9 @@ def test_command_store_ignores_stop_after_graceful_finish() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -800,7 +818,9 @@ def test_command_store_ignores_finish_after_non_graceful_stop() -> None: run_started_at=datetime(year=2021, month=1, day=1), latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] @@ -834,7 +854,9 @@ def test_handles_hardware_stopped() -> None: run_started_at=None, latest_protocol_command_hash=None, stopped_by_estop=False, + failed_command_errors=[], error_recovery_policy=matchers.Anything(), + has_entered_error_recovery=False, ) assert subject.state.command_history.get_running_command() is None assert subject.state.command_history.get_all_ids() == [] diff --git a/api/tests/opentrons/protocol_engine/state/test_command_view_old.py b/api/tests/opentrons/protocol_engine/state/test_command_view_old.py index f64f4a09d2d..2b86fe9259f 100644 --- a/api/tests/opentrons/protocol_engine/state/test_command_view_old.py +++ b/api/tests/opentrons/protocol_engine/state/test_command_view_old.py @@ -72,6 +72,8 @@ def get_command_view( # noqa: C901 finish_error: Optional[errors.ErrorOccurrence] = None, commands: Sequence[cmd.Command] = (), latest_command_hash: Optional[str] = None, + failed_command_errors: Optional[List[ErrorOccurrence]] = None, + has_entered_error_recovery: bool = False, ) -> CommandView: """Get a command view test subject.""" command_history = CommandHistory() @@ -108,6 +110,8 @@ def get_command_view( # noqa: C901 run_started_at=run_started_at, latest_protocol_command_hash=latest_command_hash, stopped_by_estop=False, + failed_command_errors=failed_command_errors or [], + has_entered_error_recovery=has_entered_error_recovery, error_recovery_policy=_placeholder_error_recovery_policy, ) diff --git a/app/src/organisms/InterventionModal/__fixtures__/index.ts b/app/src/organisms/InterventionModal/__fixtures__/index.ts index 7e9f57090e4..2594aace49e 100644 --- a/app/src/organisms/InterventionModal/__fixtures__/index.ts +++ b/app/src/organisms/InterventionModal/__fixtures__/index.ts @@ -192,6 +192,7 @@ export const mockRunData: RunData = { status: 'running', actions: [], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [mockLabwareOnModule, mockLabwareOnSlot, mockLabwareOffDeck], modules: [mockModule], diff --git a/app/src/organisms/OnDeviceDisplay/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx b/app/src/organisms/OnDeviceDisplay/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx index 8bc3a481843..f029d739806 100644 --- a/app/src/organisms/OnDeviceDisplay/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx +++ b/app/src/organisms/OnDeviceDisplay/RobotDashboard/__tests__/RecentRunProtocolCarousel.test.tsx @@ -18,6 +18,7 @@ const mockRun = { createdAt: '2023-04-12T15:13:52.110602+00:00', current: false, errors: [], + hasEverEnteredErrorRecovery: false, id: '853a3fae-8043-47de-8f03-5d28b3ee3d35', labware: [], labwareOffsets: [], diff --git a/app/src/organisms/RunTimeControl/__fixtures__/index.ts b/app/src/organisms/RunTimeControl/__fixtures__/index.ts index 235c49fbcde..49dadfe85ea 100644 --- a/app/src/organisms/RunTimeControl/__fixtures__/index.ts +++ b/app/src/organisms/RunTimeControl/__fixtures__/index.ts @@ -37,6 +37,7 @@ export const mockPausedRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -68,6 +69,7 @@ export const mockRunningRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -110,6 +112,7 @@ export const mockFailedRun: RunData = { errorCode: '4000', }, ], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -146,6 +149,7 @@ export const mockStopRequestedRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -182,6 +186,7 @@ export const mockStoppedRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -213,6 +218,7 @@ export const mockSucceededRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -228,6 +234,7 @@ export const mockIdleUnstartedRun: RunData = { protocolId: PROTOCOL_ID, actions: [], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -259,6 +266,7 @@ export const mockIdleStartedRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], diff --git a/react-api-client/src/runs/__fixtures__/runs.ts b/react-api-client/src/runs/__fixtures__/runs.ts index 33ae7cb4b4d..2222e6562d7 100644 --- a/react-api-client/src/runs/__fixtures__/runs.ts +++ b/react-api-client/src/runs/__fixtures__/runs.ts @@ -27,6 +27,7 @@ export const mockPausedRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], @@ -58,6 +59,7 @@ export const mockRunningRun: RunData = { }, ], errors: [], + hasEverEnteredErrorRecovery: false, pipettes: [], labware: [], modules: [], diff --git a/robot-server/robot_server/maintenance_runs/maintenance_run_data_manager.py b/robot-server/robot_server/maintenance_runs/maintenance_run_data_manager.py index 7b255a6f79d..6f2cddd1835 100644 --- a/robot-server/robot_server/maintenance_runs/maintenance_run_data_manager.py +++ b/robot-server/robot_server/maintenance_runs/maintenance_run_data_manager.py @@ -32,6 +32,7 @@ def _build_run( pipettes=[], modules=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) return MaintenanceRun.construct( id=run_id, @@ -47,6 +48,7 @@ def _build_run( completedAt=state_summary.completedAt, startedAt=state_summary.startedAt, liquids=state_summary.liquids, + hasEverEnteredErrorRecovery=state_summary.hasEverEnteredErrorRecovery, ) diff --git a/robot-server/robot_server/maintenance_runs/maintenance_run_models.py b/robot-server/robot_server/maintenance_runs/maintenance_run_models.py index 766c717e7bf..e4c5971f5d1 100644 --- a/robot-server/robot_server/maintenance_runs/maintenance_run_models.py +++ b/robot-server/robot_server/maintenance_runs/maintenance_run_models.py @@ -47,6 +47,10 @@ class MaintenanceRun(ResourceModel): " but it won't have more than one element." ), ) + hasEverEnteredErrorRecovery: bool = Field( + ..., + description=("Whether the run has entered error recovery."), + ) pipettes: List[LoadedPipette] = Field( ..., description="Pipettes that have been loaded into the run.", diff --git a/robot-server/robot_server/runs/run_data_manager.py b/robot-server/robot_server/runs/run_data_manager.py index 5ae7581d952..de5eea82e45 100644 --- a/robot-server/robot_server/runs/run_data_manager.py +++ b/robot-server/robot_server/runs/run_data_manager.py @@ -44,6 +44,7 @@ def _build_run( actions=run_resource.actions, status=state_summary.status, errors=state_summary.errors, + hasEverEnteredErrorRecovery=state_summary.hasEverEnteredErrorRecovery, labware=state_summary.labware, labwareOffsets=state_summary.labwareOffsets, pipettes=state_summary.pipettes, @@ -65,6 +66,7 @@ def _build_run( pipettes=[], modules=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) errors.append(state_summary.dataError) else: @@ -107,6 +109,7 @@ def _build_run( startedAt=state.startedAt, liquids=state.liquids, runTimeParameters=run_time_parameters, + hasEverEnteredErrorRecovery=state.hasEverEnteredErrorRecovery, ) diff --git a/robot-server/robot_server/runs/run_models.py b/robot-server/robot_server/runs/run_models.py index 5d2fb0a41ca..45ad22e3167 100644 --- a/robot-server/robot_server/runs/run_models.py +++ b/robot-server/robot_server/runs/run_models.py @@ -110,6 +110,10 @@ class Run(ResourceModel): " but it won't have more than one element." ), ) + hasEverEnteredErrorRecovery: bool = Field( + ..., + description=("Whether the run has entered error recovery."), + ) pipettes: List[LoadedPipette] = Field( ..., description="Pipettes that have been loaded into the run.", @@ -183,6 +187,10 @@ class BadRun(ResourceModel): " but it won't have more than one element." ), ) + hasEverEnteredErrorRecovery: bool = Field( + ..., + description=("Whether the run has entered error recovery."), + ) pipettes: List[LoadedPipette] = Field( ..., description="Pipettes that have been loaded into the run.", diff --git a/robot-server/tests/integration/http_api/runs/test_json_v6_protocol_run.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_json_v6_protocol_run.tavern.yaml index a22c3c3d74a..28d39bcfa77 100644 --- a/robot-server/tests/integration/http_api/runs/test_json_v6_protocol_run.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_json_v6_protocol_run.tavern.yaml @@ -37,6 +37,7 @@ stages: current: True actions: [] errors: [] + hasEverEnteredErrorRecovery: False pipettes: [] modules: [] labware: diff --git a/robot-server/tests/integration/http_api/runs/test_json_v7_protocol_run.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_json_v7_protocol_run.tavern.yaml index 7aaec1dd822..70737a7f6c3 100644 --- a/robot-server/tests/integration/http_api/runs/test_json_v7_protocol_run.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_json_v7_protocol_run.tavern.yaml @@ -37,6 +37,7 @@ stages: current: True actions: [] errors: [] + hasEverEnteredErrorRecovery: False pipettes: [] modules: [] labware: diff --git a/robot-server/tests/integration/http_api/runs/test_protocol_run.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_protocol_run.tavern.yaml index 159b1238986..2bfa2ccd552 100644 --- a/robot-server/tests/integration/http_api/runs/test_protocol_run.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_protocol_run.tavern.yaml @@ -34,6 +34,7 @@ stages: current: True actions: [] errors: [] + hasEverEnteredErrorRecovery: False pipettes: [] modules: [] labware: @@ -241,6 +242,7 @@ stages: runTimeParameters: [] completedAt: !re_fullmatch "\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d+(Z|([+-]\\d{2}:\\d{2}))" errors: [] + hasEverEnteredErrorRecovery: False pipettes: [] modules: [] protocolId: '{protocol_id}' diff --git a/robot-server/tests/integration/http_api/runs/test_run_queued_protocol_commands.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_run_queued_protocol_commands.tavern.yaml index 074f68b5456..edec26c4e03 100644 --- a/robot-server/tests/integration/http_api/runs/test_run_queued_protocol_commands.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_run_queued_protocol_commands.tavern.yaml @@ -90,6 +90,7 @@ stages: createdAt: !re_fullmatch "\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d+(Z|([+-]\\d{2}:\\d{2}))" current: True errors: [] + hasEverEnteredErrorRecovery: False id: '{run_id}' labware: [] labwareOffsets: [] diff --git a/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml b/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml index 2a71179b37d..9d91abea32f 100644 --- a/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml +++ b/robot-server/tests/integration/http_api/runs/test_run_with_run_time_parameters.tavern.yaml @@ -43,6 +43,7 @@ stages: current: True actions: [] errors: [] + hasEverEnteredErrorRecovery: False pipettes: [] modules: [] labware: [] diff --git a/robot-server/tests/maintenance_runs/router/test_base_router.py b/robot-server/tests/maintenance_runs/router/test_base_router.py index 87881f982c2..b363cd1e6ac 100644 --- a/robot-server/tests/maintenance_runs/router/test_base_router.py +++ b/robot-server/tests/maintenance_runs/router/test_base_router.py @@ -75,6 +75,7 @@ async def test_create_run( labwareOffsets=[], status=pe_types.EngineStatus.IDLE, liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when( @@ -146,6 +147,7 @@ async def test_get_run_data_from_url( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(mock_maintenance_run_data_manager.get("run-id")).then_return( @@ -195,6 +197,7 @@ async def test_get_run() -> None: labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) result = await get_run(run_data=run_data) @@ -220,6 +223,7 @@ async def test_get_current_run( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(mock_maintenance_run_data_manager.current_run_id).then_return( "current-run-id" diff --git a/robot-server/tests/maintenance_runs/router/test_labware_router.py b/robot-server/tests/maintenance_runs/router/test_labware_router.py index cb88688731e..d8a8fdab603 100644 --- a/robot-server/tests/maintenance_runs/router/test_labware_router.py +++ b/robot-server/tests/maintenance_runs/router/test_labware_router.py @@ -38,6 +38,7 @@ def run() -> MaintenanceRun: modules=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) diff --git a/robot-server/tests/maintenance_runs/test_run_data_manager.py b/robot-server/tests/maintenance_runs/test_run_data_manager.py index f1127f287fb..7baffe86a29 100644 --- a/robot-server/tests/maintenance_runs/test_run_data_manager.py +++ b/robot-server/tests/maintenance_runs/test_run_data_manager.py @@ -63,6 +63,7 @@ def engine_state_summary() -> StateSummary: return StateSummary( status=EngineStatus.IDLE, errors=[ErrorOccurrence.construct(id="some-error-id")], # type: ignore[call-arg] + hasEverEnteredErrorRecovery=False, labware=[LoadedLabware.construct(id="some-labware-id")], # type: ignore[call-arg] labwareOffsets=[LabwareOffset.construct(id="some-labware-offset-id")], # type: ignore[call-arg] pipettes=[LoadedPipette.construct(id="some-pipette-id")], # type: ignore[call-arg] @@ -132,6 +133,7 @@ async def test_create( status=engine_state_summary.status, actions=[], errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -184,6 +186,7 @@ async def test_create_with_options( status=engine_state_summary.status, actions=[], errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -252,6 +255,7 @@ async def test_get_current_run( status=engine_state_summary.status, actions=[], errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, diff --git a/robot-server/tests/protocols/test_protocol_analyzer.py b/robot-server/tests/protocols/test_protocol_analyzer.py index 1e9c004999a..a5eb40b95bc 100644 --- a/robot-server/tests/protocols/test_protocol_analyzer.py +++ b/robot-server/tests/protocols/test_protocol_analyzer.py @@ -189,6 +189,7 @@ async def test_analyze( modules=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ), parameters=[bool_parameter], ) diff --git a/robot-server/tests/runs/router/test_base_router.py b/robot-server/tests/runs/router/test_base_router.py index 8ee37d64b9d..979d3a92371 100644 --- a/robot-server/tests/runs/router/test_base_router.py +++ b/robot-server/tests/runs/router/test_base_router.py @@ -84,6 +84,7 @@ async def test_create_run( labwareOffsets=[], status=pe_types.EngineStatus.IDLE, liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when( await mock_deck_configuration_store.get_deck_configuration() @@ -159,6 +160,7 @@ async def test_create_protocol_run( labwareOffsets=[], status=pe_types.EngineStatus.IDLE, liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when( await mock_deck_configuration_store.get_deck_configuration() @@ -280,6 +282,7 @@ async def test_get_run_data_from_url( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(mock_run_data_manager.get("run-id")).then_return(expected_response) @@ -326,6 +329,7 @@ async def test_get_run() -> None: labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) result = await get_run(run_data=run_data) @@ -371,6 +375,7 @@ async def test_get_runs_not_empty( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) response_2 = Run( @@ -386,6 +391,7 @@ async def test_get_runs_not_empty( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(mock_run_data_manager.get_all(20)).then_return([response_1, response_2]) @@ -464,6 +470,7 @@ async def test_update_run_to_not_current( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(await mock_run_data_manager.update("run-id", current=False)).then_return( @@ -498,6 +505,7 @@ async def test_update_current_none_noop( labware=[], labwareOffsets=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) decoy.when(await mock_run_data_manager.update("run-id", current=None)).then_return( diff --git a/robot-server/tests/runs/router/test_labware_router.py b/robot-server/tests/runs/router/test_labware_router.py index 09811f20a38..9a38ce6cd0f 100644 --- a/robot-server/tests/runs/router/test_labware_router.py +++ b/robot-server/tests/runs/router/test_labware_router.py @@ -40,6 +40,7 @@ def run() -> Run: labwareOffsets=[], protocolId=None, liquids=[], + hasEverEnteredErrorRecovery=False, ) diff --git a/robot-server/tests/runs/test_run_controller.py b/robot-server/tests/runs/test_run_controller.py index c6d58b229dc..a901c988168 100644 --- a/robot-server/tests/runs/test_run_controller.py +++ b/robot-server/tests/runs/test_run_controller.py @@ -65,6 +65,7 @@ def engine_state_summary() -> StateSummary: pipettes=[], modules=[], liquids=[], + hasEverEnteredErrorRecovery=False, ) diff --git a/robot-server/tests/runs/test_run_data_manager.py b/robot-server/tests/runs/test_run_data_manager.py index 63716a8ebd5..ba0b457f9f6 100644 --- a/robot-server/tests/runs/test_run_data_manager.py +++ b/robot-server/tests/runs/test_run_data_manager.py @@ -86,6 +86,7 @@ def engine_state_summary() -> StateSummary: return StateSummary( status=EngineStatus.IDLE, errors=[ErrorOccurrence.construct(id="some-error-id")], # type: ignore[call-arg] + hasEverEnteredErrorRecovery=False, labware=[LoadedLabware.construct(id="some-labware-id")], # type: ignore[call-arg] labwareOffsets=[LabwareOffset.construct(id="some-labware-offset-id")], # type: ignore[call-arg] pipettes=[LoadedPipette.construct(id="some-pipette-id")], # type: ignore[call-arg] @@ -195,6 +196,7 @@ async def test_create( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -266,6 +268,7 @@ async def test_create_with_options( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -347,6 +350,7 @@ async def test_get_current_run( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -388,6 +392,7 @@ async def test_get_historical_run( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -430,6 +435,7 @@ async def test_get_historical_run_no_data( actions=run_resource.actions, status=EngineStatus.STOPPED, errors=[], + hasEverEnteredErrorRecovery=False, labware=[], labwareOffsets=[], pipettes=[], @@ -449,6 +455,7 @@ async def test_get_all_runs( current_run_data = StateSummary( status=EngineStatus.IDLE, errors=[ErrorOccurrence.construct(id="current-error-id")], # type: ignore[call-arg] + hasEverEnteredErrorRecovery=False, labware=[LoadedLabware.construct(id="current-labware-id")], # type: ignore[call-arg] labwareOffsets=[LabwareOffset.construct(id="current-labware-offset-id")], # type: ignore[call-arg] pipettes=[LoadedPipette.construct(id="current-pipette-id")], # type: ignore[call-arg] @@ -467,6 +474,7 @@ async def test_get_all_runs( historical_run_data = StateSummary( status=EngineStatus.STOPPED, errors=[ErrorOccurrence.construct(id="old-error-id")], # type: ignore[call-arg] + hasEverEnteredErrorRecovery=False, labware=[LoadedLabware.construct(id="old-labware-id")], # type: ignore[call-arg] labwareOffsets=[LabwareOffset.construct(id="old-labware-offset-id")], # type: ignore[call-arg] pipettes=[LoadedPipette.construct(id="old-pipette-id")], # type: ignore[call-arg] @@ -526,6 +534,7 @@ async def test_get_all_runs( actions=historical_run_resource.actions, status=historical_run_data.status, errors=historical_run_data.errors, + hasEverEnteredErrorRecovery=historical_run_data.hasEverEnteredErrorRecovery, labware=historical_run_data.labware, labwareOffsets=historical_run_data.labwareOffsets, pipettes=historical_run_data.pipettes, @@ -541,6 +550,7 @@ async def test_get_all_runs( actions=current_run_resource.actions, status=current_run_data.status, errors=current_run_data.errors, + hasEverEnteredErrorRecovery=current_run_data.hasEverEnteredErrorRecovery, labware=current_run_data.labware, labwareOffsets=current_run_data.labwareOffsets, pipettes=current_run_data.pipettes, @@ -630,6 +640,7 @@ async def test_update_current( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, @@ -685,6 +696,7 @@ async def test_update_current_noop( actions=run_resource.actions, status=engine_state_summary.status, errors=engine_state_summary.errors, + hasEverEnteredErrorRecovery=engine_state_summary.hasEverEnteredErrorRecovery, labware=engine_state_summary.labware, labwareOffsets=engine_state_summary.labwareOffsets, pipettes=engine_state_summary.pipettes, diff --git a/robot-server/tests/runs/test_run_store.py b/robot-server/tests/runs/test_run_store.py index 94899c5c20e..7e4155ef1b5 100644 --- a/robot-server/tests/runs/test_run_store.py +++ b/robot-server/tests/runs/test_run_store.py @@ -117,6 +117,7 @@ def state_summary() -> StateSummary: labwareOffsets=[], status=EngineStatus.IDLE, liquids=liquids, + hasEverEnteredErrorRecovery=False, ) @@ -189,6 +190,7 @@ def invalid_state_summary() -> StateSummary: return StateSummary( errors=[analysis_error], + hasEverEnteredErrorRecovery=False, labware=[analysis_labware], pipettes=[analysis_pipette], # TODO(mc, 2022-02-14): evaluate usage of modules in the analysis resp. From a5204fbfce6cc3914e22e2f669312ebdcc1588e5 Mon Sep 17 00:00:00 2001 From: Nick Diehl <47604184+ncdiehl11@users.noreply.github.com> Date: Thu, 1 Aug 2024 15:34:28 -0400 Subject: [PATCH 2/2] Fix app rtpfiles post (#15868) From both `ChooseRobotToRunProtocolSlideout` and `ChooseProtocolSlideout` components, we render a hook `useCreateRunFromProtocol`, which synchronously handles posting to /protocols and /runs. This PR fixes a bug where `runTimeParameterFiles` overrides were being passed to /protocols, but not to /runs in this hook. --- .../__tests__/ChooseProtocolSlideout.test.tsx | 4 ++++ .../__tests__/ChooseRobotToRunProtocolSlideout.test.tsx | 5 +++++ .../useCreateRunFromProtocol.ts | 3 ++- 3 files changed, 11 insertions(+), 1 deletion(-) diff --git a/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx b/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx index af2ce216663..fe0c4dd748b 100644 --- a/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx +++ b/app/src/organisms/ChooseProtocolSlideout/__tests__/ChooseProtocolSlideout.test.tsx @@ -7,6 +7,7 @@ import { simpleAnalysisFileFixture } from '@opentrons/api-client' import { OT2_ROBOT_TYPE } from '@opentrons/shared-data' import { renderWithProviders } from '../../../__testing-utils__' import { i18n } from '../../../i18n' +import { useFeatureFlag } from '../../../redux/config' import { getStoredProtocols } from '../../../redux/protocol-storage' import { mockConnectableRobot } from '../../../redux/discovery/__fixtures__' import { @@ -18,6 +19,7 @@ import { useCreateRunFromProtocol } from '../../ChooseRobotToRunProtocolSlideout import { ChooseProtocolSlideout } from '../' import { useNotifyDataReady } from '../../../resources/useNotifyDataReady' import type { ProtocolAnalysisOutput } from '@opentrons/shared-data' +import { when } from 'vitest-when' vi.mock('../../ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol') vi.mock('../../../redux/protocol-storage') @@ -67,6 +69,7 @@ describe('ChooseProtocolSlideout', () => { trackCreateProtocolRunEvent: mockTrackCreateProtocolRunEvent, }) vi.mocked(useNotifyDataReady).mockReturnValue({} as any) + when(vi.mocked(useFeatureFlag)).calledWith('enableCsvFile').thenReturn(true) }) it('renders slideout if showSlideout true', () => { @@ -127,6 +130,7 @@ describe('ChooseProtocolSlideout', () => { files: [expect.any(File)], protocolKey: storedProtocolDataFixture.protocolKey, runTimeParameterValues: expect.any(Object), + runTimeParameterFiles: expect.any(Object), }) ) expect(mockTrackCreateProtocolRunEvent).toHaveBeenCalled() diff --git a/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx b/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx index 9dc260f1b85..d7ead4bd4dc 100644 --- a/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx +++ b/app/src/organisms/ChooseRobotToRunProtocolSlideout/__tests__/ChooseRobotToRunProtocolSlideout.test.tsx @@ -9,6 +9,7 @@ import { i18n } from '../../../i18n' import { useTrackCreateProtocolRunEvent } from '../../../organisms/Devices/hooks' import { useCloseCurrentRun } from '../../../organisms/ProtocolUpload/hooks' import { useCurrentRunStatus } from '../../../organisms/RunTimeControl/hooks' +import { useFeatureFlag } from '../../../redux/config' import { getConnectableRobots, getReachableRobots, @@ -38,6 +39,7 @@ import type { State } from '../../../redux/types' vi.mock('../../../organisms/Devices/hooks') vi.mock('../../../organisms/ProtocolUpload/hooks') vi.mock('../../../organisms/RunTimeControl/hooks') +vi.mock('../../../redux/config') vi.mock('../../../redux/discovery') vi.mock('../../../redux/robot-update') vi.mock('../../../redux/networking') @@ -190,6 +192,8 @@ describe('ChooseRobotToRunProtocolSlideout', () => { { ...mockConnectableRobot, name: 'otherRobot', ip: 'otherIp' }, mockConnectableRobot, ]) + vi.mocked(useFeatureFlag).mockReturnValue(true) + provideNullCurrentRunIdFor('otherIp') render({ storedProtocolData: storedProtocolDataFixture, @@ -213,6 +217,7 @@ describe('ChooseRobotToRunProtocolSlideout', () => { files: [expect.any(File)], protocolKey: storedProtocolDataFixture.protocolKey, runTimeParameterValues: expect.any(Object), + runTimeParameterFiles: expect.any(Object), }) ) expect(mockTrackCreateProtocolRunEvent).toHaveBeenCalled() diff --git a/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts b/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts index 24d4d5d0eba..1dcddfe12ce 100644 --- a/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts +++ b/app/src/organisms/ChooseRobotToRunProtocolSlideout/useCreateRunFromProtocol.ts @@ -71,11 +71,12 @@ export function useCreateRunFromProtocol( reset: resetProtocolMutation, } = useCreateProtocolMutation( { - onSuccess: (data, { runTimeParameterValues }) => { + onSuccess: (data, { runTimeParameterValues, runTimeParameterFiles }) => { createRun({ protocolId: data.data.id, labwareOffsets, runTimeParameterValues, + runTimeParameterFiles, }) }, },