isar 1.15.0__py3-none-any.whl → 1.34.9__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- isar/__init__.py +2 -5
- isar/apis/api.py +159 -66
- isar/apis/models/__init__.py +0 -1
- isar/apis/models/models.py +22 -12
- isar/apis/models/start_mission_definition.py +128 -123
- isar/apis/robot_control/robot_controller.py +41 -0
- isar/apis/schedule/scheduling_controller.py +135 -121
- isar/apis/security/authentication.py +5 -5
- isar/config/certs/ca-cert.pem +32 -32
- isar/config/keyvault/keyvault_service.py +1 -2
- isar/config/log.py +47 -39
- isar/config/logging.conf +16 -31
- isar/config/open_telemetry.py +102 -0
- isar/config/predefined_mission_definition/default_exr.json +49 -0
- isar/config/predefined_mission_definition/default_mission.json +1 -5
- isar/config/predefined_mission_definition/default_turtlebot.json +4 -11
- isar/config/predefined_missions/default.json +67 -87
- isar/config/predefined_missions/default_extra_capabilities.json +107 -0
- isar/config/settings.py +119 -142
- isar/eventhandlers/eventhandler.py +123 -0
- isar/mission_planner/local_planner.py +6 -20
- isar/mission_planner/mission_planner_interface.py +1 -1
- isar/models/events.py +184 -0
- isar/models/status.py +18 -0
- isar/modules.py +118 -205
- isar/robot/robot.py +377 -0
- isar/robot/robot_battery.py +60 -0
- isar/robot/robot_monitor_mission.py +357 -0
- isar/robot/robot_pause_mission.py +74 -0
- isar/robot/robot_resume_mission.py +67 -0
- isar/robot/robot_start_mission.py +66 -0
- isar/robot/robot_status.py +61 -0
- isar/robot/robot_stop_mission.py +68 -0
- isar/robot/robot_upload_inspection.py +75 -0
- isar/script.py +171 -0
- isar/services/service_connections/mqtt/mqtt_client.py +47 -11
- isar/services/service_connections/mqtt/robot_heartbeat_publisher.py +32 -0
- isar/services/service_connections/mqtt/robot_info_publisher.py +4 -3
- isar/services/service_connections/persistent_memory.py +69 -0
- isar/services/utilities/mqtt_utilities.py +93 -0
- isar/services/utilities/robot_utilities.py +20 -0
- isar/services/utilities/scheduling_utilities.py +393 -65
- isar/state_machine/state_machine.py +227 -486
- isar/state_machine/states/__init__.py +0 -7
- isar/state_machine/states/await_next_mission.py +114 -0
- isar/state_machine/states/blocked_protective_stop.py +60 -0
- isar/state_machine/states/going_to_lockdown.py +95 -0
- isar/state_machine/states/going_to_recharging.py +92 -0
- isar/state_machine/states/home.py +115 -0
- isar/state_machine/states/intervention_needed.py +77 -0
- isar/state_machine/states/lockdown.py +38 -0
- isar/state_machine/states/maintenance.py +36 -0
- isar/state_machine/states/monitor.py +137 -166
- isar/state_machine/states/offline.py +60 -0
- isar/state_machine/states/paused.py +92 -23
- isar/state_machine/states/pausing.py +48 -0
- isar/state_machine/states/pausing_return_home.py +48 -0
- isar/state_machine/states/recharging.py +80 -0
- isar/state_machine/states/resuming.py +57 -0
- isar/state_machine/states/resuming_return_home.py +64 -0
- isar/state_machine/states/return_home_paused.py +109 -0
- isar/state_machine/states/returning_home.py +217 -0
- isar/state_machine/states/stopping.py +61 -0
- isar/state_machine/states/stopping_due_to_maintenance.py +61 -0
- isar/state_machine/states/stopping_go_to_lockdown.py +60 -0
- isar/state_machine/states/stopping_go_to_recharge.py +51 -0
- isar/state_machine/states/stopping_return_home.py +77 -0
- isar/state_machine/states/unknown_status.py +72 -0
- isar/state_machine/states_enum.py +22 -5
- isar/state_machine/transitions/mission.py +192 -0
- isar/state_machine/transitions/return_home.py +106 -0
- isar/state_machine/transitions/robot_status.py +80 -0
- isar/state_machine/utils/common_event_handlers.py +73 -0
- isar/storage/blob_storage.py +71 -45
- isar/storage/local_storage.py +28 -14
- isar/storage/storage_interface.py +28 -6
- isar/storage/uploader.py +184 -55
- isar/storage/utilities.py +35 -27
- isar-1.34.9.dist-info/METADATA +496 -0
- isar-1.34.9.dist-info/RECORD +135 -0
- {isar-1.15.0.dist-info → isar-1.34.9.dist-info}/WHEEL +1 -1
- isar-1.34.9.dist-info/entry_points.txt +3 -0
- robot_interface/models/exceptions/__init__.py +0 -7
- robot_interface/models/exceptions/robot_exceptions.py +274 -4
- robot_interface/models/initialize/__init__.py +0 -1
- robot_interface/models/inspection/__init__.py +0 -13
- robot_interface/models/inspection/inspection.py +43 -34
- robot_interface/models/mission/mission.py +18 -14
- robot_interface/models/mission/status.py +20 -25
- robot_interface/models/mission/task.py +156 -92
- robot_interface/models/robots/battery_state.py +6 -0
- robot_interface/models/robots/media.py +13 -0
- robot_interface/models/robots/robot_model.py +7 -7
- robot_interface/robot_interface.py +135 -66
- robot_interface/telemetry/mqtt_client.py +84 -12
- robot_interface/telemetry/payloads.py +111 -12
- robot_interface/utilities/json_service.py +7 -1
- isar/config/predefined_missions/default_turtlebot.json +0 -110
- isar/config/predefined_poses/__init__.py +0 -0
- isar/config/predefined_poses/predefined_poses.py +0 -616
- isar/config/settings.env +0 -26
- isar/mission_planner/sequential_task_selector.py +0 -23
- isar/mission_planner/task_selector_interface.py +0 -31
- isar/models/communication/__init__.py +0 -0
- isar/models/communication/message.py +0 -12
- isar/models/communication/queues/__init__.py +0 -4
- isar/models/communication/queues/queue_io.py +0 -12
- isar/models/communication/queues/queue_timeout_error.py +0 -2
- isar/models/communication/queues/queues.py +0 -19
- isar/models/communication/queues/status_queue.py +0 -20
- isar/models/mission_metadata/__init__.py +0 -0
- isar/services/readers/__init__.py +0 -0
- isar/services/readers/base_reader.py +0 -37
- isar/services/service_connections/mqtt/robot_status_publisher.py +0 -93
- isar/services/service_connections/stid/__init__.py +0 -0
- isar/services/service_connections/stid/stid_service.py +0 -45
- isar/services/utilities/queue_utilities.py +0 -39
- isar/state_machine/states/idle.py +0 -40
- isar/state_machine/states/initialize.py +0 -60
- isar/state_machine/states/initiate.py +0 -129
- isar/state_machine/states/off.py +0 -18
- isar/state_machine/states/stop.py +0 -78
- isar/storage/slimm_storage.py +0 -181
- isar-1.15.0.dist-info/METADATA +0 -417
- isar-1.15.0.dist-info/RECORD +0 -113
- robot_interface/models/initialize/initialize_params.py +0 -9
- robot_interface/models/mission/step.py +0 -211
- {isar-1.15.0.dist-info → isar-1.34.9.dist-info/licenses}/LICENSE +0 -0
- {isar-1.15.0.dist-info → isar-1.34.9.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,60 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, Callable, List, Optional
|
|
2
|
+
|
|
3
|
+
from isar.apis.models.models import LockdownResponse
|
|
4
|
+
from isar.eventhandlers.eventhandler import EventHandlerBase, EventHandlerMapping
|
|
5
|
+
from isar.models.events import Event
|
|
6
|
+
from robot_interface.models.exceptions.robot_exceptions import ErrorMessage
|
|
7
|
+
|
|
8
|
+
if TYPE_CHECKING:
|
|
9
|
+
from isar.state_machine.state_machine import StateMachine
|
|
10
|
+
|
|
11
|
+
|
|
12
|
+
class StoppingGoToLockdown(EventHandlerBase):
|
|
13
|
+
|
|
14
|
+
def __init__(self, state_machine: "StateMachine"):
|
|
15
|
+
events = state_machine.events
|
|
16
|
+
|
|
17
|
+
def _failed_stop_event_handler(
|
|
18
|
+
event: Event[ErrorMessage],
|
|
19
|
+
) -> Optional[Callable]:
|
|
20
|
+
error_message: Optional[ErrorMessage] = event.consume_event()
|
|
21
|
+
if error_message is None:
|
|
22
|
+
return None
|
|
23
|
+
|
|
24
|
+
events.api_requests.send_to_lockdown.response.trigger_event(
|
|
25
|
+
LockdownResponse(
|
|
26
|
+
lockdown_started=False,
|
|
27
|
+
failure_reason="Failed to stop ongoing mission",
|
|
28
|
+
)
|
|
29
|
+
)
|
|
30
|
+
return state_machine.mission_stopping_failed # type: ignore
|
|
31
|
+
|
|
32
|
+
def _successful_stop_event_handler(event: Event[bool]) -> Optional[Callable]:
|
|
33
|
+
if not event.consume_event():
|
|
34
|
+
return None
|
|
35
|
+
|
|
36
|
+
state_machine.publish_mission_aborted("Robot being sent to lockdown", True)
|
|
37
|
+
|
|
38
|
+
events.api_requests.send_to_lockdown.response.trigger_event(
|
|
39
|
+
LockdownResponse(lockdown_started=True)
|
|
40
|
+
)
|
|
41
|
+
state_machine.start_return_home_mission()
|
|
42
|
+
return state_machine.start_lockdown_mission_monitoring # type: ignore
|
|
43
|
+
|
|
44
|
+
event_handlers: List[EventHandlerMapping] = [
|
|
45
|
+
EventHandlerMapping(
|
|
46
|
+
name="failed_stop_event",
|
|
47
|
+
event=events.robot_service_events.mission_failed_to_stop,
|
|
48
|
+
handler=_failed_stop_event_handler,
|
|
49
|
+
),
|
|
50
|
+
EventHandlerMapping(
|
|
51
|
+
name="successful_stop_event",
|
|
52
|
+
event=events.robot_service_events.mission_successfully_stopped,
|
|
53
|
+
handler=_successful_stop_event_handler,
|
|
54
|
+
),
|
|
55
|
+
]
|
|
56
|
+
super().__init__(
|
|
57
|
+
state_name="stopping_go_to_lockdown",
|
|
58
|
+
state_machine=state_machine,
|
|
59
|
+
event_handler_mappings=event_handlers,
|
|
60
|
+
)
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, Callable, List, Optional
|
|
2
|
+
|
|
3
|
+
from isar.eventhandlers.eventhandler import EventHandlerBase, EventHandlerMapping
|
|
4
|
+
from isar.models.events import Event
|
|
5
|
+
from robot_interface.models.exceptions.robot_exceptions import ErrorMessage
|
|
6
|
+
|
|
7
|
+
if TYPE_CHECKING:
|
|
8
|
+
from isar.state_machine.state_machine import StateMachine
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
class StoppingGoToRecharge(EventHandlerBase):
|
|
12
|
+
|
|
13
|
+
def __init__(self, state_machine: "StateMachine"):
|
|
14
|
+
events = state_machine.events
|
|
15
|
+
|
|
16
|
+
def _failed_stop_event_handler(
|
|
17
|
+
event: Event[ErrorMessage],
|
|
18
|
+
) -> Optional[Callable]:
|
|
19
|
+
error_message: Optional[ErrorMessage] = event.consume_event()
|
|
20
|
+
if error_message is None:
|
|
21
|
+
return None
|
|
22
|
+
|
|
23
|
+
return state_machine.mission_stopping_failed # type: ignore
|
|
24
|
+
|
|
25
|
+
def _successful_stop_event_handler(event: Event[bool]) -> Optional[Callable]:
|
|
26
|
+
if not event.consume_event():
|
|
27
|
+
return None
|
|
28
|
+
|
|
29
|
+
state_machine.publish_mission_aborted(
|
|
30
|
+
"Robot battery too low to continue mission", True
|
|
31
|
+
)
|
|
32
|
+
state_machine.start_return_home_mission()
|
|
33
|
+
return state_machine.start_recharging_mission_monitoring # type: ignore
|
|
34
|
+
|
|
35
|
+
event_handlers: List[EventHandlerMapping] = [
|
|
36
|
+
EventHandlerMapping(
|
|
37
|
+
name="failed_stop_event",
|
|
38
|
+
event=events.robot_service_events.mission_failed_to_stop,
|
|
39
|
+
handler=_failed_stop_event_handler,
|
|
40
|
+
),
|
|
41
|
+
EventHandlerMapping(
|
|
42
|
+
name="successful_stop_event",
|
|
43
|
+
event=events.robot_service_events.mission_successfully_stopped,
|
|
44
|
+
handler=_successful_stop_event_handler,
|
|
45
|
+
),
|
|
46
|
+
]
|
|
47
|
+
super().__init__(
|
|
48
|
+
state_name="stopping_go_to_recharge",
|
|
49
|
+
state_machine=state_machine,
|
|
50
|
+
event_handler_mappings=event_handlers,
|
|
51
|
+
)
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import logging
|
|
2
|
+
from typing import TYPE_CHECKING, Callable, List, Optional
|
|
3
|
+
|
|
4
|
+
from isar.apis.models.models import MissionStartResponse
|
|
5
|
+
from isar.eventhandlers.eventhandler import EventHandlerBase, EventHandlerMapping
|
|
6
|
+
from isar.models.events import Event
|
|
7
|
+
from robot_interface.models.exceptions.robot_exceptions import ErrorMessage
|
|
8
|
+
from robot_interface.models.mission.mission import Mission
|
|
9
|
+
|
|
10
|
+
if TYPE_CHECKING:
|
|
11
|
+
from isar.state_machine.state_machine import StateMachine
|
|
12
|
+
|
|
13
|
+
|
|
14
|
+
class StoppingReturnHome(EventHandlerBase):
|
|
15
|
+
|
|
16
|
+
def __init__(self, state_machine: "StateMachine"):
|
|
17
|
+
logger = logging.getLogger("state_machine")
|
|
18
|
+
events = state_machine.events
|
|
19
|
+
|
|
20
|
+
def _failed_stop_event_handler(
|
|
21
|
+
event: Event[ErrorMessage],
|
|
22
|
+
) -> Optional[Callable]:
|
|
23
|
+
error_message: Optional[ErrorMessage] = event.consume_event()
|
|
24
|
+
if error_message is None:
|
|
25
|
+
return None
|
|
26
|
+
|
|
27
|
+
logger.warning(error_message.error_description)
|
|
28
|
+
mission: Mission = (
|
|
29
|
+
state_machine.events.api_requests.start_mission.request.consume_event()
|
|
30
|
+
)
|
|
31
|
+
state_machine.events.api_requests.start_mission.response.trigger_event(
|
|
32
|
+
MissionStartResponse(
|
|
33
|
+
mission_id=mission.id,
|
|
34
|
+
mission_started=False,
|
|
35
|
+
mission_not_started_reason="Failed to cancel return home mission",
|
|
36
|
+
)
|
|
37
|
+
)
|
|
38
|
+
return state_machine.return_home_mission_stopping_failed # type: ignore
|
|
39
|
+
|
|
40
|
+
def _successful_stop_event_handler(event: Event[bool]) -> Optional[Callable]:
|
|
41
|
+
if not event.consume_event():
|
|
42
|
+
return None
|
|
43
|
+
|
|
44
|
+
mission: Mission = (
|
|
45
|
+
state_machine.events.api_requests.start_mission.request.consume_event()
|
|
46
|
+
)
|
|
47
|
+
|
|
48
|
+
if mission:
|
|
49
|
+
state_machine.start_mission(mission=mission)
|
|
50
|
+
state_machine.events.api_requests.start_mission.response.trigger_event(
|
|
51
|
+
MissionStartResponse(mission_started=True)
|
|
52
|
+
)
|
|
53
|
+
return state_machine.start_mission_monitoring # type: ignore
|
|
54
|
+
|
|
55
|
+
state_machine.logger.error(
|
|
56
|
+
"Stopped return home without a new mission to start"
|
|
57
|
+
)
|
|
58
|
+
state_machine.start_return_home_mission()
|
|
59
|
+
return state_machine.start_return_home_monitoring # type: ignore
|
|
60
|
+
|
|
61
|
+
event_handlers: List[EventHandlerMapping] = [
|
|
62
|
+
EventHandlerMapping(
|
|
63
|
+
name="failed_stop_event",
|
|
64
|
+
event=events.robot_service_events.mission_failed_to_stop,
|
|
65
|
+
handler=_failed_stop_event_handler,
|
|
66
|
+
),
|
|
67
|
+
EventHandlerMapping(
|
|
68
|
+
name="successful_stop_event",
|
|
69
|
+
event=events.robot_service_events.mission_successfully_stopped,
|
|
70
|
+
handler=_successful_stop_event_handler,
|
|
71
|
+
),
|
|
72
|
+
]
|
|
73
|
+
super().__init__(
|
|
74
|
+
state_name="stopping_return_home",
|
|
75
|
+
state_machine=state_machine,
|
|
76
|
+
event_handler_mappings=event_handlers,
|
|
77
|
+
)
|
|
@@ -0,0 +1,72 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, Callable, List, Optional
|
|
2
|
+
|
|
3
|
+
from isar.apis.models.models import MaintenanceResponse
|
|
4
|
+
from isar.eventhandlers.eventhandler import EventHandlerBase, EventHandlerMapping
|
|
5
|
+
from isar.models.events import Event
|
|
6
|
+
from isar.state_machine.utils.common_event_handlers import stop_mission_event_handler
|
|
7
|
+
from robot_interface.models.mission.status import RobotStatus
|
|
8
|
+
|
|
9
|
+
if TYPE_CHECKING:
|
|
10
|
+
from isar.state_machine.state_machine import StateMachine
|
|
11
|
+
|
|
12
|
+
|
|
13
|
+
class UnknownStatus(EventHandlerBase):
|
|
14
|
+
|
|
15
|
+
def __init__(self, state_machine: "StateMachine"):
|
|
16
|
+
events = state_machine.events
|
|
17
|
+
shared_state = state_machine.shared_state
|
|
18
|
+
|
|
19
|
+
def _set_maintenance_mode_event_handler(event: Event[bool]):
|
|
20
|
+
should_set_maintenande_mode: bool = event.consume_event()
|
|
21
|
+
if should_set_maintenande_mode:
|
|
22
|
+
events.api_requests.set_maintenance_mode.response.trigger_event(
|
|
23
|
+
MaintenanceResponse(is_maintenance_mode=True)
|
|
24
|
+
)
|
|
25
|
+
return state_machine.set_maintenance_mode # type: ignore
|
|
26
|
+
return None
|
|
27
|
+
|
|
28
|
+
def _robot_status_event_handler(
|
|
29
|
+
status_changed_event: Event[bool],
|
|
30
|
+
) -> Optional[Callable]:
|
|
31
|
+
has_changed = status_changed_event.consume_event()
|
|
32
|
+
if not has_changed:
|
|
33
|
+
return None
|
|
34
|
+
robot_status: Optional[RobotStatus] = shared_state.robot_status.check()
|
|
35
|
+
|
|
36
|
+
if robot_status == RobotStatus.Home:
|
|
37
|
+
return state_machine.robot_status_home # type: ignore
|
|
38
|
+
elif robot_status == RobotStatus.Available:
|
|
39
|
+
return state_machine.robot_status_available # type: ignore
|
|
40
|
+
elif robot_status == RobotStatus.Offline:
|
|
41
|
+
return state_machine.robot_status_offline # type: ignore
|
|
42
|
+
elif robot_status == RobotStatus.BlockedProtectiveStop:
|
|
43
|
+
return state_machine.robot_status_blocked_protective_stop # type: ignore
|
|
44
|
+
return None
|
|
45
|
+
|
|
46
|
+
def _reset_status_check():
|
|
47
|
+
# Ensures that we will check the status immediately instead of waiting for it to change
|
|
48
|
+
self.events.robot_service_events.robot_status_changed.trigger_event(True)
|
|
49
|
+
|
|
50
|
+
event_handlers: List[EventHandlerMapping] = [
|
|
51
|
+
EventHandlerMapping(
|
|
52
|
+
name="stop_mission_event",
|
|
53
|
+
event=events.api_requests.stop_mission.request,
|
|
54
|
+
handler=lambda event: stop_mission_event_handler(state_machine, event),
|
|
55
|
+
),
|
|
56
|
+
EventHandlerMapping(
|
|
57
|
+
name="robot_status_event",
|
|
58
|
+
event=events.robot_service_events.robot_status_changed,
|
|
59
|
+
handler=_robot_status_event_handler,
|
|
60
|
+
),
|
|
61
|
+
EventHandlerMapping(
|
|
62
|
+
name="set_maintenance_mode",
|
|
63
|
+
event=events.api_requests.set_maintenance_mode.request,
|
|
64
|
+
handler=_set_maintenance_mode_event_handler,
|
|
65
|
+
),
|
|
66
|
+
]
|
|
67
|
+
super().__init__(
|
|
68
|
+
state_name="unknown_status",
|
|
69
|
+
state_machine=state_machine,
|
|
70
|
+
event_handler_mappings=event_handlers,
|
|
71
|
+
on_entry=_reset_status_check,
|
|
72
|
+
)
|
|
@@ -2,13 +2,30 @@ from enum import Enum
|
|
|
2
2
|
|
|
3
3
|
|
|
4
4
|
class States(str, Enum):
|
|
5
|
-
Off = "off"
|
|
6
|
-
Idle = "idle"
|
|
7
|
-
Initiate = "initiate"
|
|
8
|
-
Initialize = "initialize"
|
|
9
5
|
Monitor = "monitor"
|
|
6
|
+
ReturningHome = "returning_home"
|
|
7
|
+
Stopping = "stopping"
|
|
8
|
+
StoppingReturnHome = "stopping_return_home"
|
|
10
9
|
Paused = "paused"
|
|
11
|
-
|
|
10
|
+
Pausing = "pausing"
|
|
11
|
+
Resuming = "resuming"
|
|
12
|
+
PausingReturnHome = "pausing_return_home"
|
|
13
|
+
ResumingReturnHome = "resuming_return_home"
|
|
14
|
+
ReturnHomePaused = "return_home_paused"
|
|
15
|
+
AwaitNextMission = "await_next_mission"
|
|
16
|
+
Home = "home"
|
|
17
|
+
Offline = "offline"
|
|
18
|
+
BlockedProtectiveStop = "blocked_protective_stop"
|
|
19
|
+
UnknownStatus = "unknown_status"
|
|
20
|
+
InterventionNeeded = "intervention_needed"
|
|
21
|
+
Recharging = "recharging"
|
|
22
|
+
StoppingGoToLockdown = "stopping_go_to_lockdown"
|
|
23
|
+
GoingToLockdown = "going_to_lockdown"
|
|
24
|
+
Lockdown = "lockdown"
|
|
25
|
+
GoingToRecharging = "going_to_recharging"
|
|
26
|
+
StoppingGoToRecharge = "stopping_go_to_recharge"
|
|
27
|
+
Maintenance = "maintenance"
|
|
28
|
+
StoppingDueToMaintenance = "stopping_due_to_maintenance"
|
|
12
29
|
|
|
13
30
|
def __repr__(self):
|
|
14
31
|
return self.value
|
|
@@ -0,0 +1,192 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, List
|
|
2
|
+
|
|
3
|
+
if TYPE_CHECKING:
|
|
4
|
+
from isar.state_machine.state_machine import StateMachine
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
def get_mission_transitions(state_machine: "StateMachine") -> List[dict]:
|
|
8
|
+
mission_transitions: List[dict] = [
|
|
9
|
+
{
|
|
10
|
+
"trigger": "pause",
|
|
11
|
+
"source": state_machine.monitor_state,
|
|
12
|
+
"dest": state_machine.pausing_state,
|
|
13
|
+
},
|
|
14
|
+
{
|
|
15
|
+
"trigger": "mission_paused",
|
|
16
|
+
"source": state_machine.pausing_state,
|
|
17
|
+
"dest": state_machine.paused_state,
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
"trigger": "mission_pausing_failed",
|
|
21
|
+
"source": state_machine.pausing_state,
|
|
22
|
+
"dest": state_machine.monitor_state,
|
|
23
|
+
},
|
|
24
|
+
{
|
|
25
|
+
"trigger": "pause_return_home",
|
|
26
|
+
"source": state_machine.returning_home_state,
|
|
27
|
+
"dest": state_machine.pausing_return_home_state,
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
"trigger": "return_home_mission_pausing_failed",
|
|
31
|
+
"source": state_machine.pausing_return_home_state,
|
|
32
|
+
"dest": state_machine.returning_home_state,
|
|
33
|
+
},
|
|
34
|
+
{
|
|
35
|
+
"trigger": "return_home_mission_paused",
|
|
36
|
+
"source": state_machine.pausing_return_home_state,
|
|
37
|
+
"dest": state_machine.return_home_paused_state,
|
|
38
|
+
},
|
|
39
|
+
{
|
|
40
|
+
"trigger": "resume",
|
|
41
|
+
"source": state_machine.paused_state,
|
|
42
|
+
"dest": state_machine.resuming_state,
|
|
43
|
+
},
|
|
44
|
+
{
|
|
45
|
+
"trigger": "mission_resumed",
|
|
46
|
+
"source": state_machine.resuming_state,
|
|
47
|
+
"dest": state_machine.monitor_state,
|
|
48
|
+
},
|
|
49
|
+
{
|
|
50
|
+
"trigger": "mission_resuming_failed",
|
|
51
|
+
"source": state_machine.resuming_state,
|
|
52
|
+
"dest": state_machine.await_next_mission_state,
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
"trigger": "resume",
|
|
56
|
+
"source": state_machine.return_home_paused_state,
|
|
57
|
+
"dest": state_machine.resuming_return_home_state,
|
|
58
|
+
},
|
|
59
|
+
{
|
|
60
|
+
"trigger": "return_home_mission_resumed",
|
|
61
|
+
"source": state_machine.resuming_return_home_state,
|
|
62
|
+
"dest": state_machine.returning_home_state,
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
"trigger": "return_home_mission_resuming_failed",
|
|
66
|
+
"source": state_machine.resuming_return_home_state,
|
|
67
|
+
"dest": state_machine.await_next_mission_state,
|
|
68
|
+
},
|
|
69
|
+
{
|
|
70
|
+
"trigger": "resume_lockdown",
|
|
71
|
+
"source": state_machine.return_home_paused_state,
|
|
72
|
+
"dest": state_machine.going_to_lockdown_state,
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
"trigger": "stop",
|
|
76
|
+
"source": [
|
|
77
|
+
state_machine.await_next_mission_state,
|
|
78
|
+
state_machine.monitor_state,
|
|
79
|
+
state_machine.paused_state,
|
|
80
|
+
],
|
|
81
|
+
"dest": state_machine.stopping_state,
|
|
82
|
+
},
|
|
83
|
+
{
|
|
84
|
+
"trigger": "stop_go_to_lockdown",
|
|
85
|
+
"source": [
|
|
86
|
+
state_machine.monitor_state,
|
|
87
|
+
state_machine.paused_state,
|
|
88
|
+
],
|
|
89
|
+
"dest": state_machine.stopping_go_to_lockdown_state,
|
|
90
|
+
},
|
|
91
|
+
{
|
|
92
|
+
"trigger": "stop_go_to_recharge",
|
|
93
|
+
"source": state_machine.monitor_state,
|
|
94
|
+
"dest": state_machine.stopping_go_to_recharge_state,
|
|
95
|
+
},
|
|
96
|
+
{
|
|
97
|
+
"trigger": "stop_due_to_maintenance",
|
|
98
|
+
"source": [
|
|
99
|
+
state_machine.monitor_state,
|
|
100
|
+
state_machine.paused_state,
|
|
101
|
+
# state_machine.pausing_return_home_state, # Not neccessary since it will become paused and then the maintenance can trigger.
|
|
102
|
+
# state_machine.pausing_state, # Not neccessary since it will become paused and then the maintenance can trigger.
|
|
103
|
+
state_machine.return_home_paused_state,
|
|
104
|
+
state_machine.returning_home_state,
|
|
105
|
+
# state_machine.stopping_return_home_state, # Not neccessary since it will become monitor and then the maintenance can trigger.
|
|
106
|
+
# state_machine.stopping_state, # Not neccessary since it will become await next mission and then the maintenance can trigger.
|
|
107
|
+
],
|
|
108
|
+
"dest": state_machine.stopping_due_to_maintenance_state,
|
|
109
|
+
},
|
|
110
|
+
{
|
|
111
|
+
"trigger": "stop_return_home",
|
|
112
|
+
"source": [
|
|
113
|
+
state_machine.returning_home_state,
|
|
114
|
+
state_machine.return_home_paused_state,
|
|
115
|
+
],
|
|
116
|
+
"dest": state_machine.stopping_return_home_state,
|
|
117
|
+
},
|
|
118
|
+
{
|
|
119
|
+
"trigger": "mission_stopped",
|
|
120
|
+
"source": state_machine.stopping_state,
|
|
121
|
+
"dest": state_machine.await_next_mission_state,
|
|
122
|
+
},
|
|
123
|
+
{
|
|
124
|
+
"trigger": "mission_stopped",
|
|
125
|
+
"source": state_machine.stopping_go_to_lockdown_state,
|
|
126
|
+
"dest": state_machine.going_to_lockdown_state,
|
|
127
|
+
},
|
|
128
|
+
{
|
|
129
|
+
"trigger": "mission_stopped",
|
|
130
|
+
"source": state_machine.stopping_due_to_maintenance_state,
|
|
131
|
+
"dest": state_machine.maintenance_state,
|
|
132
|
+
},
|
|
133
|
+
{
|
|
134
|
+
"trigger": "mission_stopping_failed",
|
|
135
|
+
"source": [
|
|
136
|
+
state_machine.stopping_go_to_lockdown_state,
|
|
137
|
+
state_machine.stopping_go_to_recharge_state,
|
|
138
|
+
state_machine.stopping_state,
|
|
139
|
+
],
|
|
140
|
+
"dest": state_machine.monitor_state,
|
|
141
|
+
},
|
|
142
|
+
{
|
|
143
|
+
"trigger": "mission_stopping_failed",
|
|
144
|
+
"source": state_machine.stopping_due_to_maintenance_state,
|
|
145
|
+
"dest": state_machine.unknown_status_state, # We do not know if we need to go to monitor or return_home state
|
|
146
|
+
},
|
|
147
|
+
{
|
|
148
|
+
"trigger": "return_home_mission_stopping_failed",
|
|
149
|
+
"source": state_machine.stopping_return_home_state,
|
|
150
|
+
"dest": state_machine.returning_home_state,
|
|
151
|
+
},
|
|
152
|
+
{
|
|
153
|
+
"trigger": "start_mission_monitoring",
|
|
154
|
+
"source": [
|
|
155
|
+
state_machine.await_next_mission_state,
|
|
156
|
+
state_machine.home_state,
|
|
157
|
+
state_machine.stopping_return_home_state,
|
|
158
|
+
],
|
|
159
|
+
"dest": state_machine.monitor_state,
|
|
160
|
+
},
|
|
161
|
+
{
|
|
162
|
+
"trigger": "mission_failed_to_start",
|
|
163
|
+
"source": state_machine.monitor_state,
|
|
164
|
+
"dest": state_machine.await_next_mission_state,
|
|
165
|
+
},
|
|
166
|
+
{
|
|
167
|
+
"trigger": "mission_finished",
|
|
168
|
+
"source": state_machine.monitor_state,
|
|
169
|
+
"dest": state_machine.await_next_mission_state,
|
|
170
|
+
},
|
|
171
|
+
{
|
|
172
|
+
"trigger": "set_maintenance_mode",
|
|
173
|
+
"source": [
|
|
174
|
+
state_machine.await_next_mission_state,
|
|
175
|
+
state_machine.blocked_protective_stopping_state,
|
|
176
|
+
state_machine.home_state,
|
|
177
|
+
state_machine.intervention_needed_state,
|
|
178
|
+
state_machine.offline_state,
|
|
179
|
+
state_machine.recharging_state,
|
|
180
|
+
state_machine.unknown_status_state,
|
|
181
|
+
],
|
|
182
|
+
"dest": state_machine.maintenance_state,
|
|
183
|
+
},
|
|
184
|
+
{
|
|
185
|
+
"trigger": "release_from_maintenance",
|
|
186
|
+
"source": [
|
|
187
|
+
state_machine.maintenance_state,
|
|
188
|
+
],
|
|
189
|
+
"dest": state_machine.home_state,
|
|
190
|
+
},
|
|
191
|
+
]
|
|
192
|
+
return mission_transitions
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, List
|
|
2
|
+
|
|
3
|
+
if TYPE_CHECKING:
|
|
4
|
+
from isar.state_machine.state_machine import StateMachine
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
def get_return_home_transitions(state_machine: "StateMachine") -> List[dict]:
|
|
8
|
+
return_home_transitions: List[dict] = [
|
|
9
|
+
{
|
|
10
|
+
"trigger": "start_return_home_monitoring",
|
|
11
|
+
"source": [
|
|
12
|
+
state_machine.await_next_mission_state,
|
|
13
|
+
state_machine.home_state,
|
|
14
|
+
state_machine.intervention_needed_state,
|
|
15
|
+
state_machine.monitor_state,
|
|
16
|
+
state_machine.stopping_state,
|
|
17
|
+
state_machine.stopping_return_home_state,
|
|
18
|
+
],
|
|
19
|
+
"dest": state_machine.returning_home_state,
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
"trigger": "returned_home",
|
|
23
|
+
"source": state_machine.returning_home_state,
|
|
24
|
+
"dest": state_machine.home_state,
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
"trigger": "starting_recharging",
|
|
28
|
+
"source": [
|
|
29
|
+
state_machine.lockdown_state,
|
|
30
|
+
state_machine.home_state,
|
|
31
|
+
state_machine.going_to_recharging_state,
|
|
32
|
+
],
|
|
33
|
+
"dest": state_machine.recharging_state,
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
"trigger": "return_home_failed",
|
|
37
|
+
"source": [
|
|
38
|
+
state_machine.returning_home_state,
|
|
39
|
+
state_machine.going_to_recharging_state,
|
|
40
|
+
],
|
|
41
|
+
"dest": state_machine.intervention_needed_state,
|
|
42
|
+
},
|
|
43
|
+
{
|
|
44
|
+
"trigger": "release_intervention_needed",
|
|
45
|
+
"source": state_machine.intervention_needed_state,
|
|
46
|
+
"dest": state_machine.unknown_status_state,
|
|
47
|
+
},
|
|
48
|
+
{
|
|
49
|
+
"trigger": "start_lockdown_mission_monitoring",
|
|
50
|
+
"source": [
|
|
51
|
+
state_machine.stopping_go_to_lockdown_state,
|
|
52
|
+
state_machine.await_next_mission_state,
|
|
53
|
+
],
|
|
54
|
+
"dest": state_machine.going_to_lockdown_state,
|
|
55
|
+
},
|
|
56
|
+
{
|
|
57
|
+
"trigger": "start_recharging_mission_monitoring",
|
|
58
|
+
"source": [
|
|
59
|
+
state_machine.stopping_go_to_recharge_state,
|
|
60
|
+
state_machine.await_next_mission_state,
|
|
61
|
+
],
|
|
62
|
+
"dest": state_machine.going_to_recharging_state,
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
"trigger": "go_to_lockdown",
|
|
66
|
+
"source": [
|
|
67
|
+
state_machine.returning_home_state,
|
|
68
|
+
state_machine.going_to_recharging_state,
|
|
69
|
+
],
|
|
70
|
+
"dest": state_machine.going_to_lockdown_state,
|
|
71
|
+
},
|
|
72
|
+
{
|
|
73
|
+
"trigger": "go_to_recharging",
|
|
74
|
+
"source": [
|
|
75
|
+
state_machine.returning_home_state,
|
|
76
|
+
state_machine.return_home_paused_state,
|
|
77
|
+
state_machine.home_state,
|
|
78
|
+
],
|
|
79
|
+
"dest": state_machine.going_to_recharging_state,
|
|
80
|
+
},
|
|
81
|
+
{
|
|
82
|
+
"trigger": "reached_lockdown",
|
|
83
|
+
"source": [
|
|
84
|
+
state_machine.home_state,
|
|
85
|
+
state_machine.going_to_lockdown_state,
|
|
86
|
+
state_machine.recharging_state,
|
|
87
|
+
],
|
|
88
|
+
"dest": state_machine.lockdown_state,
|
|
89
|
+
},
|
|
90
|
+
{
|
|
91
|
+
"trigger": "lockdown_mission_failed",
|
|
92
|
+
"source": state_machine.going_to_lockdown_state,
|
|
93
|
+
"dest": state_machine.intervention_needed_state,
|
|
94
|
+
},
|
|
95
|
+
{
|
|
96
|
+
"trigger": "release_from_lockdown",
|
|
97
|
+
"source": state_machine.lockdown_state,
|
|
98
|
+
"dest": state_machine.home_state,
|
|
99
|
+
},
|
|
100
|
+
{
|
|
101
|
+
"trigger": "go_to_home",
|
|
102
|
+
"source": state_machine.intervention_needed_state,
|
|
103
|
+
"dest": state_machine.home_state,
|
|
104
|
+
},
|
|
105
|
+
]
|
|
106
|
+
return return_home_transitions
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
from typing import TYPE_CHECKING, List
|
|
2
|
+
|
|
3
|
+
if TYPE_CHECKING:
|
|
4
|
+
from isar.state_machine.state_machine import StateMachine
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
def get_robot_status_transitions(state_machine: "StateMachine") -> List[dict]:
|
|
8
|
+
robot_status_transitions: List[dict] = [
|
|
9
|
+
{
|
|
10
|
+
"trigger": "initial_transition",
|
|
11
|
+
"source": state_machine.unknown_status_state,
|
|
12
|
+
"dest": state_machine.unknown_status_state,
|
|
13
|
+
},
|
|
14
|
+
{
|
|
15
|
+
"trigger": "initial_transition",
|
|
16
|
+
"source": state_machine.maintenance_state,
|
|
17
|
+
"dest": state_machine.maintenance_state,
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
"trigger": "robot_status_available",
|
|
21
|
+
"source": [
|
|
22
|
+
state_machine.unknown_status_state,
|
|
23
|
+
],
|
|
24
|
+
"dest": state_machine.await_next_mission_state,
|
|
25
|
+
},
|
|
26
|
+
{
|
|
27
|
+
"trigger": "robot_status_available",
|
|
28
|
+
"source": [
|
|
29
|
+
state_machine.offline_state,
|
|
30
|
+
state_machine.blocked_protective_stopping_state,
|
|
31
|
+
state_machine.home_state,
|
|
32
|
+
],
|
|
33
|
+
"dest": state_machine.intervention_needed_state,
|
|
34
|
+
},
|
|
35
|
+
{
|
|
36
|
+
"trigger": "robot_status_home",
|
|
37
|
+
"source": [
|
|
38
|
+
state_machine.home_state,
|
|
39
|
+
state_machine.blocked_protective_stopping_state,
|
|
40
|
+
state_machine.offline_state,
|
|
41
|
+
state_machine.unknown_status_state,
|
|
42
|
+
],
|
|
43
|
+
"dest": state_machine.home_state,
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
"trigger": "robot_status_blocked_protective_stop",
|
|
47
|
+
"source": [
|
|
48
|
+
state_machine.home_state,
|
|
49
|
+
state_machine.offline_state,
|
|
50
|
+
state_machine.unknown_status_state,
|
|
51
|
+
],
|
|
52
|
+
"dest": state_machine.blocked_protective_stopping_state,
|
|
53
|
+
},
|
|
54
|
+
{
|
|
55
|
+
"trigger": "robot_status_offline",
|
|
56
|
+
"source": [
|
|
57
|
+
state_machine.home_state,
|
|
58
|
+
state_machine.blocked_protective_stopping_state,
|
|
59
|
+
state_machine.unknown_status_state,
|
|
60
|
+
state_machine.recharging_state,
|
|
61
|
+
],
|
|
62
|
+
"dest": state_machine.offline_state,
|
|
63
|
+
},
|
|
64
|
+
{
|
|
65
|
+
"trigger": "robot_status_unknown",
|
|
66
|
+
"source": [
|
|
67
|
+
state_machine.home_state,
|
|
68
|
+
state_machine.blocked_protective_stopping_state,
|
|
69
|
+
state_machine.offline_state,
|
|
70
|
+
state_machine.unknown_status_state,
|
|
71
|
+
],
|
|
72
|
+
"dest": state_machine.unknown_status_state,
|
|
73
|
+
},
|
|
74
|
+
{
|
|
75
|
+
"trigger": "robot_recharged",
|
|
76
|
+
"source": state_machine.recharging_state,
|
|
77
|
+
"dest": state_machine.home_state,
|
|
78
|
+
},
|
|
79
|
+
]
|
|
80
|
+
return robot_status_transitions
|