From ce0e4e4e8f889b282c3b86b092c0835528c60da5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A5kon=20Fadum?= Date: Thu, 24 Oct 2024 16:19:50 +0200 Subject: [PATCH] Allow stopping missions from idle state --- .../apis/schedule/scheduling_controller.py | 10 ++----- src/isar/state_machine/state_machine.py | 27 ++++++++++++++----- src/isar/state_machine/states/idle.py | 3 +++ .../apis/scheduler/test_scheduler_router.py | 9 +------ 4 files changed, 27 insertions(+), 22 deletions(-) diff --git a/src/isar/apis/schedule/scheduling_controller.py b/src/isar/apis/schedule/scheduling_controller.py index 1562353b..47fbe1cd 100644 --- a/src/isar/apis/schedule/scheduling_controller.py +++ b/src/isar/apis/schedule/scheduling_controller.py @@ -21,13 +21,7 @@ from isar.services.utilities.scheduling_utilities import SchedulingUtilities from isar.state_machine.states_enum import States from robot_interface.models.mission.mission import Mission -from robot_interface.models.mission.task import ( - TASKS, - Localize, - MoveArm, - ReturnToHome, -) -from robot_interface.models.mission.task import Task +from robot_interface.models.mission.task import TASKS, Localize, MoveArm, ReturnToHome class SchedulingController: @@ -192,7 +186,7 @@ def stop_mission(self) -> ControlMissionResponse: state: States = self.scheduling_utilities.get_state() - if state in [States.Off, States.Idle]: + if state == States.Off: error_message = ( f"Conflict - Stop command received in invalid state - State: {state}" ) diff --git a/src/isar/state_machine/state_machine.py b/src/isar/state_machine/state_machine.py index 0f53b0bf..124e4f4d 100644 --- a/src/isar/state_machine/state_machine.py +++ b/src/isar/state_machine/state_machine.py @@ -32,11 +32,7 @@ from robot_interface.models.exceptions.robot_exceptions import ErrorMessage from robot_interface.models.initialize.initialize_params import InitializeParams from robot_interface.models.mission.mission import Mission -from robot_interface.models.mission.status import ( - MissionStatus, - RobotStatus, - TaskStatus, -) +from robot_interface.models.mission.status import MissionStatus, RobotStatus, TaskStatus from robot_interface.models.mission.task import TASKS, Task from robot_interface.robot_interface import RobotInterface from robot_interface.telemetry.mqtt_client import MqttClientInterface @@ -134,7 +130,11 @@ def __init__( }, { "trigger": "stop", - "source": [self.initiate_state, self.monitor_state], + "source": [ + self.initiate_state, + self.monitor_state, + self.idle_state, + ], "dest": self.stop_state, "before": self._stop, }, @@ -371,6 +371,11 @@ def _initiate_infeasible(self) -> None: self.iterate_current_task() def _mission_stopped(self) -> None: + if self.current_mission is None: + self._queue_empty_response() + self.reset_state_machine() + return + self.current_mission.status = MissionStatus.Cancelled for task in self.current_mission.tasks: @@ -588,6 +593,16 @@ def _make_control_mission_response(self) -> ControlMissionResponse: task_status=self.current_task.status, ) + def _queue_empty_response(self): + self.queues.stop_mission.output.put( + ControlMissionResponse( + mission_id="None", + mission_status="None", + task_id="None", + task_status="None", + ) + ) + def main(state_machine: StateMachine): """Starts a state machine instance.""" diff --git a/src/isar/state_machine/states/idle.py b/src/isar/state_machine/states/idle.py index 39fe9837..a97314ae 100644 --- a/src/isar/state_machine/states/idle.py +++ b/src/isar/state_machine/states/idle.py @@ -36,6 +36,9 @@ def stop(self) -> None: def _run(self) -> None: while True: + if self.state_machine.should_stop_mission(): + transition = self.state_machine.stop # type: ignore + break start_mission: Optional[StartMissionMessage] = ( self.state_machine.should_start_mission() ) diff --git a/tests/isar/apis/scheduler/test_scheduler_router.py b/tests/isar/apis/scheduler/test_scheduler_router.py index 46472c27..ff0a3280 100644 --- a/tests/isar/apis/scheduler/test_scheduler_router.py +++ b/tests/isar/apis/scheduler/test_scheduler_router.py @@ -269,6 +269,7 @@ class TestStopMission: valid_states = [ States.Initiate, States.Initialize, + States.Idle, States.Monitor, States.Paused, States.Stop, @@ -286,14 +287,6 @@ def test_stop_mission( assert response.status_code == HTTPStatus.OK assert response.json() == jsonable_encoder(mock_control_mission_response) - @mock.patch.object(SchedulingUtilities, "get_state", mock_return_idle) - @mock.patch.object( - SchedulingUtilities, "stop_mission", mock_control_mission_response - ) - def test_can_not_stop_mission_in_idle(self, client: TestClient): - response = client.post(url=self.schedule_stop_mission_path) - assert response.status_code == HTTPStatus.CONFLICT - @mock.patch.object(SchedulingUtilities, "get_state", mock_return_off) @mock.patch.object( SchedulingUtilities, "stop_mission", mock_control_mission_response