Struct TimeoutLayer
pub struct TimeoutLayer {
timeout: Duration,
io_timeout: Duration,
}
Expand description
Add timeout for every operation to avoid slow or unexpected hang operations.
For example, a dead connection could hang a databases sql query. TimeoutLayer will break this connection and returns an error so users can handle it by retrying or print to users.
§Notes
TimeoutLayer
treats all operations in two kinds:
- Non IO Operation like
stat
,delete
they operate on a single file. We control them by settingtimeout
. - IO Operation like
read
,Reader::read
andWriter::write
, they operate on data directly, we control them by settingio_timeout
.
§Default
- timeout: 60 seconds
- io_timeout: 10 seconds
§Panics
TimeoutLayer will drop the future if the timeout is reached. This might cause the internal state of the future to be broken. If underlying future moves ownership into the future, it will be dropped and will neven return back.
For example, while using TimeoutLayer
with RetryLayer
at the same time, please make sure
timeout layer showed up before retry layer.
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
The following examples will create a timeout layer with 10 seconds timeout for all non-io operations, 3 seconds timeout for all io operations.
let _ = Operator::new(services::Memory::default())?
.layer(
TimeoutLayer::default()
.with_timeout(Duration::from_secs(10))
.with_io_timeout(Duration::from_secs(3)),
)
.finish();
Ok(())
§Implementation Notes
TimeoutLayer is using [tokio::time::timeout
] to implement timeout for operations. And IO
Operations insides reader
, writer
will use Pin<Box<tokio::time::Sleep>>
to track the
timeout.
This might introduce a bit overhead for IO operations, but it’s the only way to implement
timeout correctly. We used to implement timeout layer in zero cost way that only stores
a std::time::Instant
and check the timeout by comparing the instant with current time.
However, it doesn’t work for all cases.
For examples, users TCP connection could be in Busy ESTAB state. In this state, no IO event will be emitted. The runtime will never poll our future again. From the application side, this future is hanging forever until this TCP connection is closed for reaching the linux net.ipv4.tcp_retries2 times.
Fields§
§timeout: Duration
§io_timeout: Duration
Implementations§
§impl TimeoutLayer
impl TimeoutLayer
pub fn new() -> TimeoutLayer
pub fn new() -> TimeoutLayer
Create a new TimeoutLayer
with default settings.
pub fn with_timeout(self, timeout: Duration) -> TimeoutLayer
pub fn with_timeout(self, timeout: Duration) -> TimeoutLayer
Set timeout for TimeoutLayer with given value.
This timeout is for all non-io operations like stat
, delete
.
pub fn with_io_timeout(self, timeout: Duration) -> TimeoutLayer
pub fn with_io_timeout(self, timeout: Duration) -> TimeoutLayer
Set io timeout for TimeoutLayer with given value.
This timeout is for all io operations like read
, Reader::read
and Writer::write
.
pub fn with_speed(self, _: u64) -> TimeoutLayer
👎Deprecated: with speed is not supported anymore, please use with_io_timeout instead
pub fn with_speed(self, _: u64) -> TimeoutLayer
Trait Implementations§
§impl Clone for TimeoutLayer
impl Clone for TimeoutLayer
§fn clone(&self) -> TimeoutLayer
fn clone(&self) -> TimeoutLayer
1.0.0 · Source§fn clone_from(&mut self, source: &Self)
fn clone_from(&mut self, source: &Self)
source
. Read more§impl Default for TimeoutLayer
impl Default for TimeoutLayer
§fn default() -> TimeoutLayer
fn default() -> TimeoutLayer
§impl<A> Layer<A> for TimeoutLayerwhere
A: Access,
impl<A> Layer<A> for TimeoutLayerwhere
A: Access,
§type LayeredAccess = TimeoutAccessor<A>
type LayeredAccess = TimeoutAccessor<A>
§fn layer(&self, inner: A) -> <TimeoutLayer as Layer<A>>::LayeredAccess
fn layer(&self, inner: A) -> <TimeoutLayer as Layer<A>>::LayeredAccess
Auto Trait Implementations§
impl Freeze for TimeoutLayer
impl RefUnwindSafe for TimeoutLayer
impl Send for TimeoutLayer
impl Sync for TimeoutLayer
impl Unpin for TimeoutLayer
impl UnwindSafe for TimeoutLayer
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