|
@@ -19,34 +19,87 @@ import functools
|
|
|
import sys
|
|
|
import typing as t
|
|
|
|
|
|
+import tenacity
|
|
|
from tenacity import AttemptManager
|
|
|
from tenacity import BaseRetrying
|
|
|
from tenacity import DoAttempt
|
|
|
from tenacity import DoSleep
|
|
|
from tenacity import RetryCallState
|
|
|
+from tenacity import RetryError
|
|
|
+from tenacity import after_nothing
|
|
|
+from tenacity import before_nothing
|
|
|
from tenacity import _utils
|
|
|
|
|
|
+# Import all built-in retry strategies for easier usage.
|
|
|
+from .retry import RetryBaseT
|
|
|
+from .retry import retry_all # noqa
|
|
|
+from .retry import retry_any # noqa
|
|
|
+from .retry import retry_if_exception # noqa
|
|
|
+from .retry import retry_if_result # noqa
|
|
|
+from ..retry import RetryBaseT as SyncRetryBaseT
|
|
|
+
|
|
|
+if t.TYPE_CHECKING:
|
|
|
+ from tenacity.stop import StopBaseT
|
|
|
+ from tenacity.wait import WaitBaseT
|
|
|
+
|
|
|
WrappedFnReturnT = t.TypeVar("WrappedFnReturnT")
|
|
|
WrappedFn = t.TypeVar("WrappedFn", bound=t.Callable[..., t.Awaitable[t.Any]])
|
|
|
|
|
|
|
|
|
-def asyncio_sleep(duration: float) -> t.Awaitable[None]:
|
|
|
+def _portable_async_sleep(seconds: float) -> t.Awaitable[None]:
|
|
|
+ # If trio is already imported, then importing it is cheap.
|
|
|
+ # If trio isn't already imported, then it's definitely not running, so we
|
|
|
+ # can skip further checks.
|
|
|
+ if "trio" in sys.modules:
|
|
|
+ # If trio is available, then sniffio is too
|
|
|
+ import trio
|
|
|
+ import sniffio
|
|
|
+
|
|
|
+ if sniffio.current_async_library() == "trio":
|
|
|
+ return trio.sleep(seconds)
|
|
|
+ # Otherwise, assume asyncio
|
|
|
# Lazy import asyncio as it's expensive (responsible for 25-50% of total import overhead).
|
|
|
import asyncio
|
|
|
|
|
|
- return asyncio.sleep(duration)
|
|
|
+ return asyncio.sleep(seconds)
|
|
|
|
|
|
|
|
|
class AsyncRetrying(BaseRetrying):
|
|
|
- sleep: t.Callable[[float], t.Awaitable[t.Any]]
|
|
|
-
|
|
|
def __init__(
|
|
|
self,
|
|
|
- sleep: t.Callable[[float], t.Awaitable[t.Any]] = asyncio_sleep,
|
|
|
- **kwargs: t.Any,
|
|
|
+ sleep: t.Callable[
|
|
|
+ [t.Union[int, float]], t.Union[None, t.Awaitable[None]]
|
|
|
+ ] = _portable_async_sleep,
|
|
|
+ stop: "StopBaseT" = tenacity.stop.stop_never,
|
|
|
+ wait: "WaitBaseT" = tenacity.wait.wait_none(),
|
|
|
+ retry: "t.Union[SyncRetryBaseT, RetryBaseT]" = tenacity.retry_if_exception_type(),
|
|
|
+ before: t.Callable[
|
|
|
+ ["RetryCallState"], t.Union[None, t.Awaitable[None]]
|
|
|
+ ] = before_nothing,
|
|
|
+ after: t.Callable[
|
|
|
+ ["RetryCallState"], t.Union[None, t.Awaitable[None]]
|
|
|
+ ] = after_nothing,
|
|
|
+ before_sleep: t.Optional[
|
|
|
+ t.Callable[["RetryCallState"], t.Union[None, t.Awaitable[None]]]
|
|
|
+ ] = None,
|
|
|
+ reraise: bool = False,
|
|
|
+ retry_error_cls: t.Type["RetryError"] = RetryError,
|
|
|
+ retry_error_callback: t.Optional[
|
|
|
+ t.Callable[["RetryCallState"], t.Union[t.Any, t.Awaitable[t.Any]]]
|
|
|
+ ] = None,
|
|
|
) -> None:
|
|
|
- super().__init__(**kwargs)
|
|
|
- self.sleep = sleep
|
|
|
+ super().__init__(
|
|
|
+ sleep=sleep, # type: ignore[arg-type]
|
|
|
+ stop=stop,
|
|
|
+ wait=wait,
|
|
|
+ retry=retry, # type: ignore[arg-type]
|
|
|
+ before=before, # type: ignore[arg-type]
|
|
|
+ after=after, # type: ignore[arg-type]
|
|
|
+ before_sleep=before_sleep, # type: ignore[arg-type]
|
|
|
+ reraise=reraise,
|
|
|
+ retry_error_cls=retry_error_cls,
|
|
|
+ retry_error_callback=retry_error_callback,
|
|
|
+ )
|
|
|
|
|
|
async def __call__( # type: ignore[override]
|
|
|
self, fn: WrappedFn, *args: t.Any, **kwargs: t.Any
|
|
@@ -65,31 +118,21 @@ class AsyncRetrying(BaseRetrying):
|
|
|
retry_state.set_result(result)
|
|
|
elif isinstance(do, DoSleep):
|
|
|
retry_state.prepare_for_next_attempt()
|
|
|
- await self.sleep(do)
|
|
|
+ await self.sleep(do) # type: ignore[misc]
|
|
|
else:
|
|
|
return do # type: ignore[no-any-return]
|
|
|
|
|
|
- @classmethod
|
|
|
- def _wrap_action_func(cls, fn: t.Callable[..., t.Any]) -> t.Callable[..., t.Any]:
|
|
|
- if _utils.is_coroutine_callable(fn):
|
|
|
- return fn
|
|
|
-
|
|
|
- async def inner(*args: t.Any, **kwargs: t.Any) -> t.Any:
|
|
|
- return fn(*args, **kwargs)
|
|
|
-
|
|
|
- return inner
|
|
|
-
|
|
|
def _add_action_func(self, fn: t.Callable[..., t.Any]) -> None:
|
|
|
- self.iter_state.actions.append(self._wrap_action_func(fn))
|
|
|
+ self.iter_state.actions.append(_utils.wrap_to_async_func(fn))
|
|
|
|
|
|
async def _run_retry(self, retry_state: "RetryCallState") -> None: # type: ignore[override]
|
|
|
- self.iter_state.retry_run_result = await self._wrap_action_func(self.retry)(
|
|
|
+ self.iter_state.retry_run_result = await _utils.wrap_to_async_func(self.retry)(
|
|
|
retry_state
|
|
|
)
|
|
|
|
|
|
async def _run_wait(self, retry_state: "RetryCallState") -> None: # type: ignore[override]
|
|
|
if self.wait:
|
|
|
- sleep = await self._wrap_action_func(self.wait)(retry_state)
|
|
|
+ sleep = await _utils.wrap_to_async_func(self.wait)(retry_state)
|
|
|
else:
|
|
|
sleep = 0.0
|
|
|
|
|
@@ -97,7 +140,7 @@ class AsyncRetrying(BaseRetrying):
|
|
|
|
|
|
async def _run_stop(self, retry_state: "RetryCallState") -> None: # type: ignore[override]
|
|
|
self.statistics["delay_since_first_attempt"] = retry_state.seconds_since_start
|
|
|
- self.iter_state.stop_run_result = await self._wrap_action_func(self.stop)(
|
|
|
+ self.iter_state.stop_run_result = await _utils.wrap_to_async_func(self.stop)(
|
|
|
retry_state
|
|
|
)
|
|
|
|
|
@@ -127,7 +170,7 @@ class AsyncRetrying(BaseRetrying):
|
|
|
return AttemptManager(retry_state=self._retry_state)
|
|
|
elif isinstance(do, DoSleep):
|
|
|
self._retry_state.prepare_for_next_attempt()
|
|
|
- await self.sleep(do)
|
|
|
+ await self.sleep(do) # type: ignore[misc]
|
|
|
else:
|
|
|
raise StopAsyncIteration
|
|
|
|
|
@@ -146,3 +189,13 @@ class AsyncRetrying(BaseRetrying):
|
|
|
async_wrapped.retry_with = fn.retry_with # type: ignore[attr-defined]
|
|
|
|
|
|
return async_wrapped # type: ignore[return-value]
|
|
|
+
|
|
|
+
|
|
|
+__all__ = [
|
|
|
+ "retry_all",
|
|
|
+ "retry_any",
|
|
|
+ "retry_if_exception",
|
|
|
+ "retry_if_result",
|
|
|
+ "WrappedFn",
|
|
|
+ "AsyncRetrying",
|
|
|
+]
|