|
@@ -1,31 +1,44 @@
|
|
|
from __future__ import annotations
|
|
|
|
|
|
-from asyncio import Task
|
|
|
-from typing import Callable, Optional
|
|
|
+from typing import Callable
|
|
|
|
|
|
from . import background_tasks
|
|
|
|
|
|
|
|
|
class AwaitableResponse:
|
|
|
|
|
|
- def __init__(self, fire_and_forget: Optional[Callable], wait_for_result: Optional[Callable]) -> None:
|
|
|
+ def __init__(self, fire_and_forget: Callable, wait_for_result: Callable) -> None:
|
|
|
"""Awaitable Response
|
|
|
|
|
|
This class can be used to run one of two different callables, depending on whether the response is awaited or not.
|
|
|
+ It must be awaited immediately after creation or not at all.
|
|
|
|
|
|
:param fire_and_forget: The callable to run if the response is not awaited.
|
|
|
:param wait_for_result: The callable to run if the response is awaited.
|
|
|
"""
|
|
|
- self.fire_and_forget_task: Optional[Task] = \
|
|
|
- background_tasks.create(self._start(fire_and_forget), name='fire and forget') if fire_and_forget else None
|
|
|
+ self.fire_and_forget = fire_and_forget
|
|
|
self.wait_for_result = wait_for_result
|
|
|
+ self._is_fired = False
|
|
|
+ self._is_awaited = False
|
|
|
+ background_tasks.create(self._fire(), name='fire')
|
|
|
|
|
|
- async def _start(self, command: Callable) -> None:
|
|
|
- command()
|
|
|
+ async def _fire(self) -> None:
|
|
|
+ if self._is_awaited:
|
|
|
+ return
|
|
|
+ self._is_fired = True
|
|
|
+ self.fire_and_forget()
|
|
|
|
|
|
def __await__(self):
|
|
|
- if self.fire_and_forget_task is not None:
|
|
|
- self.fire_and_forget_task.cancel()
|
|
|
- if self.wait_for_result is None:
|
|
|
- raise ValueError('AwaitableResponse has no result to await')
|
|
|
+ if self._is_fired:
|
|
|
+ raise RuntimeError('AwaitableResponse must be awaited immediately after creation or not at all')
|
|
|
+ self._is_awaited = True
|
|
|
return self.wait_for_result().__await__()
|
|
|
+
|
|
|
+
|
|
|
+class NullResponse(AwaitableResponse):
|
|
|
+
|
|
|
+ def __init__(self) -> None:
|
|
|
+ pass
|
|
|
+
|
|
|
+ def __await__(self):
|
|
|
+ raise RuntimeError('NullResponse cannot be awaited')
|