use core::fmt;
#[cfg(feature = "alloc")]
use super::super::HookChain;
use super::super::execution::sync_exec::{NoSyncSleep, SyncRetryCore, SyncSleep};
use super::super::time::ElapsedTracker;
use super::super::{AttemptHook, BeforeAttemptHook, ExecutionHooks, ExitHook, RetryPolicy};
use crate::compat::Duration;
use crate::predicate::Predicate;
use crate::state::{AttemptState, ExitState, RetryState};
use crate::{
RetryError, RetryStats, predicate,
stop::{self, Stop},
wait::{self, Wait},
};
pub trait RetryExt<T, E>: FnMut() -> Result<T, E> + Sized {
fn retry(self) -> DefaultSyncRetryBuilder<Self, T, E>;
}
#[doc(hidden)]
pub struct StatelessOp<F>(F);
impl<T, E, F: FnMut() -> Result<T, E>> super::super::execution::common::RetryOp<T, E>
for StatelessOp<F>
{
fn call_op(&mut self, _state: RetryState) -> Result<T, E> {
(self.0)()
}
}
impl<T, E, F> RetryExt<T, E> for F
where
F: FnMut() -> Result<T, E> + Sized,
{
fn retry(self) -> DefaultSyncRetryBuilder<Self, T, E> {
SyncRetryBuilder {
inner: SyncRetryCore::new(
RetryPolicy::default(),
ExecutionHooks::new(),
StatelessOp(self),
NoSyncSleep,
ElapsedTracker::new(None),
),
}
}
}
pub type DefaultSyncRetryBuilder<F, T, E> = SyncRetryBuilder<
stop::StopAfterAttempts,
wait::WaitExponential,
predicate::PredicateAnyError,
(),
(),
(),
StatelessOp<F>,
NoSyncSleep,
T,
E,
>;
pub type DefaultSyncRetryBuilderWithStats<F, SleepFn, T, E> = SyncRetryBuilderWithStats<
stop::StopAfterAttempts,
wait::WaitExponential,
predicate::PredicateAnyError,
(),
(),
(),
StatelessOp<F>,
SleepFn,
T,
E,
>;
#[cfg(not(feature = "std"))]
#[doc(hidden)]
#[allow(dead_code)]
fn _sync_retry_builder_requires_sleep_in_no_std() {}
#[allow(clippy::type_complexity)]
pub struct SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E> {
inner: SyncRetryCore<RetryPolicy<S, W, P>, BA, AA, OX, F, SleepFn, T, E>,
}
impl<S, W, P, F, T, E> SyncRetryBuilder<S, W, P, (), (), (), F, NoSyncSleep, T, E> {
pub(crate) fn from_policy(policy: RetryPolicy<S, W, P>, op: F) -> Self {
SyncRetryBuilder {
inner: SyncRetryCore::new(
policy,
ExecutionHooks::new(),
op,
NoSyncSleep,
ElapsedTracker::new(None),
),
}
}
}
impl<S, W, P, BA, AA, OX, F, SleepFn, T, E> fmt::Debug
for SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E>
{
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("SyncRetryBuilder").finish_non_exhaustive()
}
}
pub struct SyncRetryBuilderWithStats<S, W, P, BA, AA, OX, F, SleepFn, T, E> {
inner: SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E>,
}
impl<S, W, P, BA, AA, OX, F, SleepFn, T, E> fmt::Debug
for SyncRetryBuilderWithStats<S, W, P, BA, AA, OX, F, SleepFn, T, E>
{
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("SyncRetryBuilderWithStats")
.finish_non_exhaustive()
}
}
#[cfg(feature = "alloc")]
type SyncBuilderWithBeforeHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> =
SyncRetryBuilder<S, W, P, HookChain<BA, Hook>, AA, OX, F, SleepFn, T, E>;
#[cfg(feature = "alloc")]
type SyncBuilderWithAfterHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> =
SyncRetryBuilder<S, W, P, BA, HookChain<AA, Hook>, OX, F, SleepFn, T, E>;
#[cfg(feature = "alloc")]
type SyncBuilderWithOnExitHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> =
SyncRetryBuilder<S, W, P, BA, AA, HookChain<OX, Hook>, F, SleepFn, T, E>;
impl<S, W, P, BA, AA, OX, F, SleepFn, T, E>
SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E>
{
fn map_hooks<NewBA, NewAA, NewOX>(
self,
map: impl FnOnce(ExecutionHooks<BA, AA, OX>) -> ExecutionHooks<NewBA, NewAA, NewOX>,
) -> SyncRetryBuilder<S, W, P, NewBA, NewAA, NewOX, F, SleepFn, T, E> {
SyncRetryBuilder {
inner: self.inner.map_hooks(map),
}
}
#[must_use]
pub fn stop<NewStop>(
self,
stop: NewStop,
) -> SyncRetryBuilder<NewStop, W, P, BA, AA, OX, F, SleepFn, T, E> {
SyncRetryBuilder {
inner: self.inner.map_policy(|policy| policy.stop(stop)),
}
}
#[must_use]
pub fn wait<NewWait>(
self,
wait: NewWait,
) -> SyncRetryBuilder<S, NewWait, P, BA, AA, OX, F, SleepFn, T, E> {
SyncRetryBuilder {
inner: self.inner.map_policy(|policy| policy.wait(wait)),
}
}
#[must_use]
pub fn when<NewPredicate>(
self,
predicate: NewPredicate,
) -> SyncRetryBuilder<S, W, NewPredicate, BA, AA, OX, F, SleepFn, T, E> {
SyncRetryBuilder {
inner: self.inner.map_policy(|policy| policy.when(predicate)),
}
}
#[must_use]
pub fn until<NewPredicate>(
self,
predicate: NewPredicate,
) -> SyncRetryBuilder<S, W, predicate::PredicateUntil<NewPredicate>, BA, AA, OX, F, SleepFn, T, E>
{
SyncRetryBuilder {
inner: self.inner.map_policy(|policy| policy.until(predicate)),
}
}
#[must_use]
pub fn elapsed_clock(self, clock: fn() -> Duration) -> Self {
SyncRetryBuilder {
inner: self.inner.set_elapsed_clock(clock),
}
}
#[cfg(feature = "alloc")]
#[must_use]
pub fn elapsed_clock_fn(self, clock: impl Fn() -> Duration + 'static) -> Self {
SyncRetryBuilder {
inner: self
.inner
.set_elapsed_clock_fn(crate::compat::Box::new(clock)),
}
}
#[must_use]
pub fn timeout(self, dur: Duration) -> Self {
SyncRetryBuilder {
inner: self.inner.set_timeout(dur),
}
}
#[must_use]
pub fn sleep<NewSleep>(
self,
sleeper: NewSleep,
) -> SyncRetryBuilder<S, W, P, BA, AA, OX, F, NewSleep, T, E> {
SyncRetryBuilder {
inner: self.inner.with_sleeper(sleeper),
}
}
}
impl_alloc_hook_chain! {
impl[S, W, P, BA, AA, OX, F, SleepFn, T, E]
SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E> =>
before_attempt -> { SyncBuilderWithBeforeHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> },
after_attempt -> { SyncBuilderWithAfterHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> },
on_exit -> { SyncBuilderWithOnExitHook<S, W, P, BA, AA, OX, F, SleepFn, T, E, Hook> },
}
#[cfg(not(feature = "alloc"))]
impl<S, W, P, AA, OX, F, SleepFn, T, E> SyncRetryBuilder<S, W, P, (), AA, OX, F, SleepFn, T, E> {
#[must_use]
pub fn before_attempt<Hook>(
self,
hook: Hook,
) -> SyncRetryBuilder<S, W, P, Hook, AA, OX, F, SleepFn, T, E>
where
Hook: FnMut(&RetryState),
{
self.map_hooks(|hooks| hooks.set_before_attempt(hook))
}
}
#[cfg(not(feature = "alloc"))]
impl<S, W, P, BA, OX, F, SleepFn, T, E> SyncRetryBuilder<S, W, P, BA, (), OX, F, SleepFn, T, E> {
#[must_use]
pub fn after_attempt<Hook>(
self,
hook: Hook,
) -> SyncRetryBuilder<S, W, P, BA, Hook, OX, F, SleepFn, T, E>
where
Hook: for<'a> FnMut(&AttemptState<'a, T, E>),
{
self.map_hooks(|hooks| hooks.set_after_attempt(hook))
}
}
#[cfg(not(feature = "alloc"))]
impl<S, W, P, BA, AA, F, SleepFn, T, E> SyncRetryBuilder<S, W, P, BA, AA, (), F, SleepFn, T, E> {
#[must_use]
pub fn on_exit<Hook>(
self,
hook: Hook,
) -> SyncRetryBuilder<S, W, P, BA, AA, Hook, F, SleepFn, T, E>
where
Hook: for<'a> FnMut(&ExitState<'a, T, E>),
{
self.map_hooks(|hooks| hooks.set_on_exit(hook))
}
}
use super::super::execution::common::RetryOp;
#[allow(private_bounds)]
impl<S, W, P, BA, AA, OX, F, SleepFn, T, E> SyncRetryBuilder<S, W, P, BA, AA, OX, F, SleepFn, T, E>
where
S: Stop,
W: Wait,
P: Predicate<T, E>,
BA: BeforeAttemptHook,
AA: AttemptHook<T, E>,
OX: ExitHook<T, E>,
F: RetryOp<T, E>,
SleepFn: SyncSleep,
{
pub fn call(self) -> Result<T, RetryError<T, E>> {
self.execute::<false>().0
}
#[must_use]
pub fn with_stats(self) -> SyncRetryBuilderWithStats<S, W, P, BA, AA, OX, F, SleepFn, T, E> {
SyncRetryBuilderWithStats { inner: self }
}
fn execute<const COLLECT_STATS: bool>(
self,
) -> (Result<T, RetryError<T, E>>, Option<RetryStats>) {
self.inner.execute::<S, W, P, COLLECT_STATS>()
}
}
#[allow(private_bounds)]
impl<S, W, P, BA, AA, OX, F, SleepFn, T, E>
SyncRetryBuilderWithStats<S, W, P, BA, AA, OX, F, SleepFn, T, E>
where
S: Stop,
W: Wait,
P: Predicate<T, E>,
BA: BeforeAttemptHook,
AA: AttemptHook<T, E>,
OX: ExitHook<T, E>,
F: RetryOp<T, E>,
SleepFn: SyncSleep,
{
pub fn call(self) -> (Result<T, RetryError<T, E>>, RetryStats) {
let (result, stats) = self.inner.execute::<true>();
(
result,
stats.expect("sync retry builder completed without stats"),
)
}
}