Struct RetryLayer
pub struct RetryLayer<I = DefaultRetryInterceptor> {
builder: ExponentialBuilder,
notify: Arc<I>,
}
Expand description
Add retry for temporary failed operations.
§Notes
This layer will retry failed operations when Error::is_temporary
returns true. If operation still failed, this layer will set error to
Persistent
which means error has been retried.
§Panics
While retrying Reader
or Writer
operations, please make sure either:
- All futures generated by
Reader::read
orWriter::close
are resolved toReady
. - Or, won’t call any
Reader
orWriter
methods after retry returns final error.
Otherwise, RetryLayer
could panic while hitting in bad states.
For example, while composing RetryLayer
with TimeoutLayer
. The order of layer is sensitive.
let op = Operator::new(services::Memory::default())?
// This is fine, since timeout happen during retry.
.layer(TimeoutLayer::new().with_io_timeout(Duration::from_nanos(1)))
.layer(RetryLayer::new())
// This is wrong. Since timeout layer will drop future, leaving retry layer in a bad state.
.layer(TimeoutLayer::new().with_io_timeout(Duration::from_nanos(1)))
.finish();
Ok(())
§Examples
let _ = Operator::new(services::Memory::default())?
.layer(RetryLayer::new())
.finish();
Ok(())
§Customize retry interceptor
RetryLayer accepts RetryInterceptor
to allow users to customize
their own retry interceptor logic.
struct MyRetryInterceptor;
impl RetryInterceptor for MyRetryInterceptor {
fn intercept(&self, err: &Error, dur: Duration) {
// do something
}
}
let _ = Operator::new(services::Memory::default())?
.layer(RetryLayer::new().with_notify(MyRetryInterceptor))
.finish();
Ok(())
Fields§
§builder: ExponentialBuilder
§notify: Arc<I>
Implementations§
§impl RetryLayer
impl RetryLayer
pub fn new() -> RetryLayer
pub fn new() -> RetryLayer
Create a new retry layer.
§Examples
use anyhow::Result;
use opendal::layers::RetryLayer;
use opendal::services;
use opendal::Operator;
use opendal::Scheme;
let _ = Operator::new(services::Memory::default())
.expect("must init")
.layer(RetryLayer::new());
pub fn with_notify<I>(self, notify: I) -> RetryLayer<I>where
I: RetryInterceptor,
pub fn with_notify<I>(self, notify: I) -> RetryLayer<I>where
I: RetryInterceptor,
Set the retry interceptor as new notify.
use std::time::Duration;
use anyhow::Result;
use opendal::layers::RetryInterceptor;
use opendal::layers::RetryLayer;
use opendal::services;
use opendal::Error;
use opendal::Operator;
use opendal::Scheme;
struct MyRetryInterceptor;
impl RetryInterceptor for MyRetryInterceptor {
fn intercept(&self, err: &Error, dur: Duration) {
// do something
}
}
let _ = Operator::new(services::Memory::default())
.expect("must init")
.layer(RetryLayer::new().with_notify(MyRetryInterceptor))
.finish();
pub fn with_jitter(self) -> RetryLayer
pub fn with_jitter(self) -> RetryLayer
Set jitter of current backoff.
If jitter is enabled, ExponentialBackoff will add a random jitter in `[0, min_delay) to current delay.
pub fn with_factor(self, factor: f32) -> RetryLayer
pub fn with_factor(self, factor: f32) -> RetryLayer
pub fn with_min_delay(self, min_delay: Duration) -> RetryLayer
pub fn with_min_delay(self, min_delay: Duration) -> RetryLayer
Set min_delay of current backoff.
pub fn with_max_delay(self, max_delay: Duration) -> RetryLayer
pub fn with_max_delay(self, max_delay: Duration) -> RetryLayer
Set max_delay of current backoff.
Delay will not increasing if current delay is larger than max_delay.
pub fn with_max_times(self, max_times: usize) -> RetryLayer
pub fn with_max_times(self, max_times: usize) -> RetryLayer
Set max_times of current backoff.
Backoff will return None
if max times is reaching.
Trait Implementations§
§impl<I> Clone for RetryLayer<I>
impl<I> Clone for RetryLayer<I>
§fn clone(&self) -> RetryLayer<I>
fn clone(&self) -> RetryLayer<I>
1.0.0 · Source§fn clone_from(&mut self, source: &Self)
fn clone_from(&mut self, source: &Self)
source
. Read more§impl Default for RetryLayer
impl Default for RetryLayer
§fn default() -> RetryLayer
fn default() -> RetryLayer
§impl<A, I> Layer<A> for RetryLayer<I>where
A: Access,
I: RetryInterceptor,
impl<A, I> Layer<A> for RetryLayer<I>where
A: Access,
I: RetryInterceptor,
§type LayeredAccess = RetryAccessor<A, I>
type LayeredAccess = RetryAccessor<A, I>
§fn layer(&self, inner: A) -> <RetryLayer<I> as Layer<A>>::LayeredAccess
fn layer(&self, inner: A) -> <RetryLayer<I> as Layer<A>>::LayeredAccess
Auto Trait Implementations§
impl<I> Freeze for RetryLayer<I>
impl<I> RefUnwindSafe for RetryLayer<I>where
I: RefUnwindSafe,
impl<I> Send for RetryLayer<I>
impl<I> Sync for RetryLayer<I>
impl<I> Unpin for RetryLayer<I>
impl<I> UnwindSafe for RetryLayer<I>where
I: RefUnwindSafe,
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> CloneToUninit for Twhere
T: Clone,
impl<T> CloneToUninit for Twhere
T: Clone,
§impl<T> FutureExt for T
impl<T> FutureExt for T
§fn with_context(self, otel_cx: Context) -> WithContext<Self>
fn with_context(self, otel_cx: Context) -> WithContext<Self>
§fn with_current_context(self) -> WithContext<Self>
fn with_current_context(self) -> WithContext<Self>
§impl<T> Instrument for T
impl<T> Instrument for T
§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
Source§impl<T> IntoRequest<T> for T
impl<T> IntoRequest<T> for T
Source§fn into_request(self) -> Request<T>
fn into_request(self) -> Request<T>
T
in a tonic::Request
Source§impl<T> IntoRequest<T> for T
impl<T> IntoRequest<T> for T
Source§fn into_request(self) -> Request<T>
fn into_request(self) -> Request<T>
T
in a tonic::Request