From 03fb6e095225992d6094fa2e5515ea805b80583c Mon Sep 17 00:00:00 2001 From: link2xt Date: Sun, 10 Dec 2023 00:35:20 +0000 Subject: [PATCH] fix: reset IDLE timeout when keepalive is received This allows to use shorter timeouts to detect lost connection earlier. E.g. if timeout is set to 5 minutes and the server sends keepalives every 2 minutes, IDLE will never be interrupted. If connection is lost, it will be noticed not later than 5 minutes later. `wait` method which does not accept the timeout now uses default timeout of 24 hours, but should probably be deprecated. --- src/extensions/idle.rs | 54 ++++++++++++++++++++---------------------- 1 file changed, 26 insertions(+), 28 deletions(-) diff --git a/src/extensions/idle.rs b/src/extensions/idle.rs index f54b58f..7a6ce29 100644 --- a/src/extensions/idle.rs +++ b/src/extensions/idle.rs @@ -119,11 +119,28 @@ impl Handle { ) -> ( impl Future> + '_, stop_token::StopSource, + ) { + self.wait_with_timeout(Duration::from_secs(24 * 60 * 60)) + } + + /// Start listening to the server side resonses. + /// + /// Stops after the passed in `timeout` without any response from the server. + /// Timeout is reset by any response, including `* OK Still here` keepalives. + /// + /// Must be called after [Handle::init]. + pub fn wait_with_timeout( + &mut self, + dur: Duration, + ) -> ( + impl Future> + '_, + stop_token::StopSource, ) { assert!( self.id.is_some(), "Cannot listen to response without starting IDLE" ); + let sender = self.session.unsolicited_responses_tx.clone(); let interrupt = stop_token::StopSource::new(); @@ -131,7 +148,15 @@ impl Handle { let mut interruptible_stream = raw_stream.timeout_at(interrupt.token()); let fut = async move { - while let Some(Ok(resp)) = interruptible_stream.next().await { + loop { + let Ok(res) = timeout(dur, interruptible_stream.next()).await else { + return Ok(IdleResponse::Timeout); + }; + + let Some(Ok(resp)) = res else { + return Ok(IdleResponse::ManualInterrupt); + }; + let resp = resp?; match resp.parsed() { Response::Data { @@ -148,33 +173,6 @@ impl Handle { _ => return Ok(IdleResponse::NewData(resp)), } } - - Ok(IdleResponse::ManualInterrupt) - }; - - (fut, interrupt) - } - - /// Start listening to the server side resonses, stops latest after the passed in `timeout`. - /// Must be called after [Handle::init]. - pub fn wait_with_timeout( - &mut self, - dur: Duration, - ) -> ( - impl Future> + '_, - stop_token::StopSource, - ) { - assert!( - self.id.is_some(), - "Cannot listen to response without starting IDLE" - ); - - let (waiter, interrupt) = self.wait(); - let fut = async move { - match timeout(dur, waiter).await { - Ok(res) => res, - Err(_err) => Ok(IdleResponse::Timeout), - } }; (fut, interrupt)