Skip to content

Commit

Permalink
Internal support for waiting for a specific command's recovery.
Browse files Browse the repository at this point in the history
  • Loading branch information
SyntaxColoring committed Mar 28, 2024
1 parent be10a4c commit 34a759d
Show file tree
Hide file tree
Showing 2 changed files with 41 additions and 0 deletions.
21 changes: 21 additions & 0 deletions api/src/opentrons/protocol_engine/protocol_engine.py
Original file line number Diff line number Diff line change
Expand Up @@ -242,6 +242,27 @@ async def add_and_execute_command(
await self.wait_for_command(command.id)
return self._state_store.commands.get(command.id)

async def add_and_execute_command_wait_for_recovery(
self, request: commands.CommandCreate
) -> None:
"""Like `add_and_execute_command()`, except wait for error recovery.
Unlike `add_and_execute_command()`, if the command fails, this will not
immediately return the failed command. Instead, if the error is recoverable,
it will wait until error recovery has completed (e.g. when some other task
calls `self.resume_from_recovery()`).
"""
command = self.add_command(request)
await self.wait_for_command(command_id=command.id)
await self._state_store.wait_for(
# Per the warnings on `wait_for()`, we want our condition function to
# specifically check that *this* command's recovery has completed,
# rather than just checking that the overall run state
# != "awaiting-recovery."
self.state_view.commands.get_command_recovery_complete,
command_id=command.id,
)

def estop(self, maintenance_run: bool) -> None:
"""Signal to the engine that an estop event occurred.
Expand Down
20 changes: 20 additions & 0 deletions api/src/opentrons/protocol_engine/state/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -687,6 +687,26 @@ def get_all_commands_final(self) -> bool:

return no_command_running and no_command_to_execute

def get_command_recovery_complete(self, command_id: str) -> bool:
"""Return whether we're finished with error recovery for the given command.
If the given command didn't have a recovery phase (because it hasn't completed
yet, or because it succeeded without an error, or because it failed with an
error that wasn't recoverable), that counts as `True`.
If the given command did have a recovery phase, but it was interrupted by a
stop, that also counts as `True`.
"""
command_is_most_recent_to_fail = (
self._state.failed_command is not None
and command_id == self._state.failed_command.command.id
)
if command_is_most_recent_to_fail:
recovery_is_ongoing = self.get_status() == EngineStatus.AWAITING_RECOVERY
return not recovery_is_ongoing
else:
return True

def raise_fatal_command_error(self) -> None:
"""Raise the run's fatal command error, if there was one, as an exception.
Expand Down

0 comments on commit 34a759d

Please sign in to comment.