From 4a1b391334ed77d953c4f94d17c1a1620af682d9 Mon Sep 17 00:00:00 2001 From: link2xt Date: Mon, 2 Dec 2024 00:41:39 +0000 Subject: [PATCH] fix: replace connectivity state "Connected" with "Preparing" This better reflects that this state means we just connected and there may me work to do. This state is converted to DC_CONNECTIVITY_WORKING instead of DC_CONNECTIVITY_CONNECTED state now. Before this change when IMAP connected to the server, it switched from DC_CONNECTIVITY_NOT_CONNECTED to DC_CONNECTIVITY_CONNECTING, then to DC_CONNECTIVITY_CONNECTED (actually preparing) then to DC_CONNECTIVITY_WORKING and then to DC_CONNECTIVITY_CONNECTED again (actually idle). On fast connections this resulted in flickering "Connected" string in the status bar right before "Updating..." and on slow connections this "Connected" state before "Updating..." lasted for a while leaving the user to wonder if there are no new messages or if Delta Chat will still switch to "Updating..." before going into "Connected" state again. --- python/tests/test_1_online.py | 2 +- src/imap.rs | 2 +- src/scheduler/connectivity.rs | 70 ++++++++++++++++++++++++----------- 3 files changed, 51 insertions(+), 23 deletions(-) diff --git a/python/tests/test_1_online.py b/python/tests/test_1_online.py index ea1e45e278..13b8b1a17a 100644 --- a/python/tests/test_1_online.py +++ b/python/tests/test_1_online.py @@ -1899,7 +1899,7 @@ def test_connectivity(acfactory, lp): ac1.start_io() ac1._evtracker.wait_for_connectivity(dc.const.DC_CONNECTIVITY_CONNECTING) - ac1._evtracker.wait_for_connectivity_change(dc.const.DC_CONNECTIVITY_CONNECTING, dc.const.DC_CONNECTIVITY_CONNECTED) + ac1._evtracker.wait_for_connectivity_change(dc.const.DC_CONNECTIVITY_CONNECTING, dc.const.DC_CONNECTIVITY_WORKING) lp.sec( "Test that after calling start_io(), maybe_network() and waiting for `all_work_done()`, " diff --git a/src/imap.rs b/src/imap.rs index 4eb432599d..2358f70865 100644 --- a/src/imap.rs +++ b/src/imap.rs @@ -407,7 +407,7 @@ impl Imap { "IMAP-LOGIN as {}", lp.user ))); - self.connectivity.set_connected(context).await; + self.connectivity.set_preparing(context).await; info!(context, "Successfully logged into IMAP server"); return Ok(session); } diff --git a/src/scheduler/connectivity.rs b/src/scheduler/connectivity.rs index 337b126caf..c230decfd8 100644 --- a/src/scheduler/connectivity.rs +++ b/src/scheduler/connectivity.rs @@ -14,12 +14,31 @@ use crate::{context::Context, log::LogExt}; use super::InnerSchedulerState; +/// Rough connectivity status for display in the status bar in the UI. #[derive(Debug, Clone, Copy, PartialEq, Eq, EnumProperty, PartialOrd, Ord)] pub enum Connectivity { + /// Not connected. + /// + /// This may be because we just started, + /// because we lost connection and + /// were not able to connect and log in yet + /// or because I/O is not started. NotConnected = 1000, + + /// Attempting to connect and log in. Connecting = 2000, - /// Fetching or sending messages + + /// Fetching or sending messages. Working = 3000, + + /// We are connected but not doing anything. + /// + /// This is the most common state, + /// so mobile UIs display the profile name + /// instead of connectivity status in this state. + /// Desktop UI displays "Connected" in the tooltip, + /// which signals that no more messages + /// are coming in. Connected = 4000, } @@ -32,13 +51,17 @@ enum DetailedConnectivity { Error(String), #[default] Uninitialized, + + /// Attempting to connect, + /// until we successfully log in. Connecting, - /// Connection is just established, but there may be work to do. - Connected, + /// Connection is just established, + /// there may be work to do. + Preparing, /// There is actual work to do, e.g. there are messages in SMTP queue - /// or we detected a message that should be downloaded. + /// or we detected a message on IMAP server that should be downloaded. Working, InterruptingIdle, @@ -58,7 +81,13 @@ impl DetailedConnectivity { DetailedConnectivity::Connecting => Some(Connectivity::Connecting), DetailedConnectivity::Working => Some(Connectivity::Working), DetailedConnectivity::InterruptingIdle => Some(Connectivity::Connected), - DetailedConnectivity::Connected => Some(Connectivity::Connected), + + // At this point IMAP has just connected, + // but does not know yet if there are messages to download. + // We still convert this to Working state + // so user can see "Updating..." and not "Connected" + // which is reserved for idle state. + DetailedConnectivity::Preparing => Some(Connectivity::Working), // Just don't return a connectivity, probably the folder is configured not to be // watched or there is e.g. no "Sent" folder, so we are not interested in it @@ -74,9 +103,9 @@ impl DetailedConnectivity { | DetailedConnectivity::Uninitialized | DetailedConnectivity::NotConfigured => "".to_string(), DetailedConnectivity::Connecting => "".to_string(), - DetailedConnectivity::Working + DetailedConnectivity::Preparing + | DetailedConnectivity::Working | DetailedConnectivity::InterruptingIdle - | DetailedConnectivity::Connected | DetailedConnectivity::Idle => "".to_string(), } } @@ -86,10 +115,12 @@ impl DetailedConnectivity { DetailedConnectivity::Error(e) => stock_str::error(context, e).await, DetailedConnectivity::Uninitialized => "Not started".to_string(), DetailedConnectivity::Connecting => stock_str::connecting(context).await, - DetailedConnectivity::Working => stock_str::updating(context).await, - DetailedConnectivity::InterruptingIdle - | DetailedConnectivity::Connected - | DetailedConnectivity::Idle => stock_str::connected(context).await, + DetailedConnectivity::Preparing | DetailedConnectivity::Working => { + stock_str::updating(context).await + } + DetailedConnectivity::InterruptingIdle | DetailedConnectivity::Idle => { + stock_str::connected(context).await + } DetailedConnectivity::NotConfigured => "Not configured".to_string(), } } @@ -107,7 +138,7 @@ impl DetailedConnectivity { // since sending the last message, connectivity could have changed, which we don't notice // until another message is sent DetailedConnectivity::InterruptingIdle - | DetailedConnectivity::Connected + | DetailedConnectivity::Preparing | DetailedConnectivity::Idle => stock_str::last_msg_sent_successfully(context).await, DetailedConnectivity::NotConfigured => "Not configured".to_string(), } @@ -120,7 +151,7 @@ impl DetailedConnectivity { DetailedConnectivity::Connecting => false, DetailedConnectivity::Working => false, DetailedConnectivity::InterruptingIdle => false, - DetailedConnectivity::Connected => false, // Just connected, there may still be work to do. + DetailedConnectivity::Preparing => false, // Just connected, there may still be work to do. DetailedConnectivity::NotConfigured => true, DetailedConnectivity::Idle => true, } @@ -148,8 +179,8 @@ impl ConnectivityStore { pub(crate) async fn set_working(&self, context: &Context) { self.set(context, DetailedConnectivity::Working).await; } - pub(crate) async fn set_connected(&self, context: &Context) { - self.set(context, DetailedConnectivity::Connected).await; + pub(crate) async fn set_preparing(&self, context: &Context) { + self.set(context, DetailedConnectivity::Preparing).await; } pub(crate) async fn set_not_configured(&self, context: &Context) { self.set(context, DetailedConnectivity::NotConfigured).await; @@ -169,7 +200,7 @@ impl ConnectivityStore { } } -/// Set all folder states to InterruptingIdle in case they were `Connected` before. +/// Set all folder states to InterruptingIdle in case they were `Idle` before. /// Called during `dc_maybe_network()` to make sure that `dc_all_work_done()` /// returns false immediately after `dc_maybe_network()`. pub(crate) async fn idle_interrupted(inbox: ConnectivityStore, oboxes: Vec) { @@ -179,8 +210,7 @@ pub(crate) async fn idle_interrupted(inbox: ConnectivityStore, oboxes: Vec