From 38fd04c3a47951d19cbf646ddf81d3d4dbb0b44c Mon Sep 17 00:00:00 2001 From: daladim Date: Fri, 8 Oct 2021 23:02:37 +0200 Subject: [PATCH 1/5] Moved SyncResult --- src/{provider.rs => provider/mod.rs} | 32 ++-------------------------- src/provider/sync_progress.rs | 30 ++++++++++++++++++++++++++ 2 files changed, 32 insertions(+), 30 deletions(-) rename src/{provider.rs => provider/mod.rs} (95%) create mode 100644 src/provider/sync_progress.rs diff --git a/src/provider.rs b/src/provider/mod.rs similarity index 95% rename from src/provider.rs rename to src/provider/mod.rs index aad647e..e9cec70 100644 --- a/src/provider.rs +++ b/src/provider/mod.rs @@ -10,36 +10,8 @@ use crate::traits::CompleteCalendar; use crate::item::SyncStatus; use crate::calendar::CalendarId; -/// A counter of errors that happen during a sync -struct SyncResult { - n_errors: u32, -} -impl SyncResult { - pub fn new() -> Self { - Self { n_errors: 0 } - } - pub fn is_success(&self) -> bool { - self.n_errors == 0 - } - - pub fn error(&mut self, text: &str) { - log::error!("{}", text); - self.n_errors += 1; - } - pub fn warn(&mut self, text: &str) { - log::warn!("{}", text); - self.n_errors += 1; - } - pub fn info(&mut self, text: &str) { - log::info!("{}", text); - } - pub fn debug(&mut self, text: &str) { - log::debug!("{}", text); - } - pub fn trace(&mut self, text: &str) { - log::trace!("{}", text); - } -} +mod sync_progress; +use sync_progress::SyncResult; /// A data source that combines two `CalDavSource`s, which is able to sync both sources. /// diff --git a/src/provider/sync_progress.rs b/src/provider/sync_progress.rs new file mode 100644 index 0000000..1399233 --- /dev/null +++ b/src/provider/sync_progress.rs @@ -0,0 +1,30 @@ +/// A counter of errors that happen during a sync +pub struct SyncResult { + n_errors: u32, +} +impl SyncResult { + pub fn new() -> Self { + Self { n_errors: 0 } + } + pub fn is_success(&self) -> bool { + self.n_errors == 0 + } + + pub fn error(&mut self, text: &str) { + log::error!("{}", text); + self.n_errors += 1; + } + pub fn warn(&mut self, text: &str) { + log::warn!("{}", text); + self.n_errors += 1; + } + pub fn info(&mut self, text: &str) { + log::info!("{}", text); + } + pub fn debug(&mut self, text: &str) { + log::debug!("{}", text); + } + pub fn trace(&mut self, text: &str) { + log::trace!("{}", text); + } +} From f7ec5d29be62e74192dfdd5724ce498e852cb24c Mon Sep 17 00:00:00 2001 From: daladim Date: Fri, 8 Oct 2021 23:03:43 +0200 Subject: [PATCH 2/5] [doc] --- src/provider/mod.rs | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/provider/mod.rs b/src/provider/mod.rs index e9cec70..787b9a8 100644 --- a/src/provider/mod.rs +++ b/src/provider/mod.rs @@ -1,4 +1,6 @@ //! This modules abstracts data sources and merges them in a single virtual one +//! +//! It is also responsible for syncing them together use std::error::Error; use std::collections::HashSet; From 04c8b3a2ee4659cd912d2b4fc1977d5d4295c540 Mon Sep 17 00:00:00 2001 From: daladim Date: Fri, 8 Oct 2021 23:05:02 +0200 Subject: [PATCH 3/5] Renamed SyncResult -> SyncProgress --- src/provider/mod.rs | 98 +++++++++++++++++------------------ src/provider/sync_progress.rs | 4 +- 2 files changed, 51 insertions(+), 51 deletions(-) diff --git a/src/provider/mod.rs b/src/provider/mod.rs index 787b9a8..55e1127 100644 --- a/src/provider/mod.rs +++ b/src/provider/mod.rs @@ -13,7 +13,7 @@ use crate::item::SyncStatus; use crate::calendar::CalendarId; mod sync_progress; -use sync_progress::SyncResult; +use sync_progress::SyncProgress; /// A data source that combines two `CalDavSource`s, which is able to sync both sources. /// @@ -72,15 +72,15 @@ where /// It returns whether the sync was totally successful (details about errors are logged using the `log::*` macros). /// In case errors happened, the sync might have been partially executed, and you can safely run this function again, since it has been designed to gracefully recover from errors. pub async fn sync(&mut self) -> bool { - let mut result = SyncResult::new(); + let mut result = SyncProgress::new(); if let Err(err) = self.run_sync(&mut result).await { result.error(&format!("Sync terminated because of an error: {}", err)); } result.is_success() } - async fn run_sync(&mut self, result: &mut SyncResult) -> Result<(), Box> { - result.info("Starting a sync."); + async fn run_sync(&mut self, progress: &mut SyncProgress) -> Result<(), Box> { + progress.info("Starting a sync."); let mut handled_calendars = HashSet::new(); @@ -89,14 +89,14 @@ where for (cal_id, cal_remote) in cals_remote { let counterpart = match self.get_or_insert_local_counterpart_calendar(&cal_id, cal_remote.clone()).await { Err(err) => { - result.warn(&format!("Unable to get or insert local counterpart calendar for {} ({}). Skipping this time", cal_id, err)); + progress.warn(&format!("Unable to get or insert local counterpart calendar for {} ({}). Skipping this time", cal_id, err)); continue; }, Ok(arc) => arc, }; - if let Err(err) = Self::sync_calendar_pair(counterpart, cal_remote, result).await { - result.warn(&format!("Unable to sync calendar {}: {}, skipping this time.", cal_id, err)); + if let Err(err) = Self::sync_calendar_pair(counterpart, cal_remote, progress).await { + progress.warn(&format!("Unable to sync calendar {}: {}, skipping this time.", cal_id, err)); continue; } handled_calendars.insert(cal_id); @@ -111,14 +111,14 @@ where let counterpart = match self.get_or_insert_remote_counterpart_calendar(&cal_id, cal_local.clone()).await { Err(err) => { - result.warn(&format!("Unable to get or insert remote counterpart calendar for {} ({}). Skipping this time", cal_id, err)); + progress.warn(&format!("Unable to get or insert remote counterpart calendar for {} ({}). Skipping this time", cal_id, err)); continue; }, Ok(arc) => arc, }; - if let Err(err) = Self::sync_calendar_pair(cal_local, counterpart, result).await { - result.warn(&format!("Unable to sync calendar {}: {}, skipping this time.", cal_id, err)); + if let Err(err) = Self::sync_calendar_pair(cal_local, counterpart, progress).await { + progress.warn(&format!("Unable to sync calendar {}: {}, skipping this time.", cal_id, err)); continue; } } @@ -135,12 +135,12 @@ where } - async fn sync_calendar_pair(cal_local: Arc>, cal_remote: Arc>, result: &mut SyncResult) -> Result<(), Box> { + async fn sync_calendar_pair(cal_local: Arc>, cal_remote: Arc>, progress: &mut SyncProgress) -> Result<(), Box> { let mut cal_remote = cal_remote.lock().unwrap(); let mut cal_local = cal_local.lock().unwrap(); // Step 1 - find the differences - result.debug("Finding the differences to sync..."); + progress.debug("Finding the differences to sync..."); let mut local_del = HashSet::new(); let mut remote_del = HashSet::new(); let mut local_changes = HashSet::new(); @@ -151,49 +151,49 @@ where let remote_items = cal_remote.get_item_version_tags().await?; let mut local_items_to_handle = cal_local.get_item_ids().await?; for (id, remote_tag) in remote_items { - result.trace(&format!("***** Considering remote item {}...", id)); + progress.trace(&format!("***** Considering remote item {}...", id)); match cal_local.get_item_by_id(&id).await { None => { // This was created on the remote - result.debug(&format!("* {} is a remote addition", id)); + progress.debug(&format!("* {} is a remote addition", id)); remote_additions.insert(id); }, Some(local_item) => { if local_items_to_handle.remove(&id) == false { - result.error(&format!("Inconsistent state: missing task {} from the local tasks", id)); + progress.error(&format!("Inconsistent state: missing task {} from the local tasks", id)); } match local_item.sync_status() { SyncStatus::NotSynced => { - result.error(&format!("ID reuse between remote and local sources ({}). Ignoring this item in the sync", id)); + progress.error(&format!("ID reuse between remote and local sources ({}). Ignoring this item in the sync", id)); continue; }, SyncStatus::Synced(local_tag) => { if &remote_tag != local_tag { // This has been modified on the remote - result.debug(&format!("* {} is a remote change", id)); + progress.debug(&format!("* {} is a remote change", id)); remote_changes.insert(id); } }, SyncStatus::LocallyModified(local_tag) => { if &remote_tag == local_tag { // This has been changed locally - result.debug(&format!("* {} is a local change", id)); + progress.debug(&format!("* {} is a local change", id)); local_changes.insert(id); } else { - result.info(&format!("Conflict: task {} has been modified in both sources. Using the remote version.", id)); - result.debug(&format!("* {} is considered a remote change", id)); + progress.info(&format!("Conflict: task {} has been modified in both sources. Using the remote version.", id)); + progress.debug(&format!("* {} is considered a remote change", id)); remote_changes.insert(id); } }, SyncStatus::LocallyDeleted(local_tag) => { if &remote_tag == local_tag { // This has been locally deleted - result.debug(&format!("* {} is a local deletion", id)); + progress.debug(&format!("* {} is a local deletion", id)); local_del.insert(id); } else { - result.info(&format!("Conflict: task {} has been locally deleted and remotely modified. Reverting to the remote version.", id)); - result.debug(&format!("* {} is a considered a remote change", id)); + progress.info(&format!("Conflict: task {} has been locally deleted and remotely modified. Reverting to the remote version.", id)); + progress.debug(&format!("* {} is a considered a remote change", id)); remote_changes.insert(id); } }, @@ -204,10 +204,10 @@ where // Also iterate on the local tasks that are not on the remote for id in local_items_to_handle { - result.trace(&format!("##### Considering local item {}...", id)); + progress.trace(&format!("##### Considering local item {}...", id)); let local_item = match cal_local.get_item_by_id(&id).await { None => { - result.error(&format!("Inconsistent state: missing task {} from the local tasks", id)); + progress.error(&format!("Inconsistent state: missing task {} from the local tasks", id)); continue; }, Some(item) => item, @@ -216,21 +216,21 @@ where match local_item.sync_status() { SyncStatus::Synced(_) => { // This item has been removed from the remote - result.debug(&format!("# {} is a deletion from the server", id)); + progress.debug(&format!("# {} is a deletion from the server", id)); remote_del.insert(id); }, SyncStatus::NotSynced => { // This item has just been locally created - result.debug(&format!("# {} has been locally created", id)); + progress.debug(&format!("# {} has been locally created", id)); local_additions.insert(id); }, SyncStatus::LocallyDeleted(_) => { // This item has been deleted from both sources - result.debug(&format!("# {} has been deleted from both sources", id)); + progress.debug(&format!("# {} has been deleted from both sources", id)); remote_del.insert(id); }, SyncStatus::LocallyModified(_) => { - result.info(&format!("Conflict: item {} has been deleted from the server and locally modified. Deleting the local copy", id)); + progress.info(&format!("Conflict: item {} has been deleted from the server and locally modified. Deleting the local copy", id)); remote_del.insert(id); }, } @@ -238,44 +238,44 @@ where // Step 2 - commit changes - result.trace("Committing changes..."); + progress.trace("Committing changes..."); for id_del in local_del { - result.debug(&format!("> Pushing local deletion {} to the server", id_del)); + progress.debug(&format!("> Pushing local deletion {} to the server", id_del)); match cal_remote.delete_item(&id_del).await { Err(err) => { - result.warn(&format!("Unable to delete remote item {}: {}", id_del, err)); + progress.warn(&format!("Unable to delete remote item {}: {}", id_del, err)); }, Ok(()) => { // Change the local copy from "marked to deletion" to "actually deleted" if let Err(err) = cal_local.immediately_delete_item(&id_del).await { - result.error(&format!("Unable to permanently delete local item {}: {}", id_del, err)); + progress.error(&format!("Unable to permanently delete local item {}: {}", id_del, err)); } }, } } for id_del in remote_del { - result.debug(&format!("> Applying remote deletion {} locally", id_del)); + progress.debug(&format!("> Applying remote deletion {} locally", id_del)); if let Err(err) = cal_local.immediately_delete_item(&id_del).await { - result.warn(&format!("Unable to delete local item {}: {}", id_del, err)); + progress.warn(&format!("Unable to delete local item {}: {}", id_del, err)); } } for id_add in remote_additions { - result.debug(&format!("> Applying remote addition {} locally", id_add)); + progress.debug(&format!("> Applying remote addition {} locally", id_add)); match cal_remote.get_item_by_id(&id_add).await { Err(err) => { - result.warn(&format!("Unable to get remote item {}: {}. Skipping it.", id_add, err)); + progress.warn(&format!("Unable to get remote item {}: {}. Skipping it.", id_add, err)); continue; }, Ok(item) => match item { None => { - result.error(&format!("Inconsistency: new item {} has vanished from the remote end", id_add)); + progress.error(&format!("Inconsistency: new item {} has vanished from the remote end", id_add)); continue; }, Some(new_item) => { if let Err(err) = cal_local.add_item(new_item.clone()).await { - result.error(&format!("Not able to add item {} to local calendar: {}", id_add, err)); + progress.error(&format!("Not able to add item {} to local calendar: {}", id_add, err)); } }, }, @@ -283,20 +283,20 @@ where } for id_change in remote_changes { - result.debug(&format!("> Applying remote change {} locally", id_change)); + progress.debug(&format!("> Applying remote change {} locally", id_change)); match cal_remote.get_item_by_id(&id_change).await { Err(err) => { - result.warn(&format!("Unable to get remote item {}: {}. Skipping it", id_change, err)); + progress.warn(&format!("Unable to get remote item {}: {}. Skipping it", id_change, err)); continue; }, Ok(item) => match item { None => { - result.error(&format!("Inconsistency: modified item {} has vanished from the remote end", id_change)); + progress.error(&format!("Inconsistency: modified item {} has vanished from the remote end", id_change)); continue; }, Some(item) => { if let Err(err) = cal_local.update_item(item.clone()).await { - result.error(&format!("Unable to update item {} in local calendar: {}", id_change, err)); + progress.error(&format!("Unable to update item {} in local calendar: {}", id_change, err)); } }, } @@ -305,15 +305,15 @@ where for id_add in local_additions { - result.debug(&format!("> Pushing local addition {} to the server", id_add)); + progress.debug(&format!("> Pushing local addition {} to the server", id_add)); match cal_local.get_item_by_id_mut(&id_add).await { None => { - result.error(&format!("Inconsistency: created item {} has been marked for upload but is locally missing", id_add)); + progress.error(&format!("Inconsistency: created item {} has been marked for upload but is locally missing", id_add)); continue; }, Some(item) => { match cal_remote.add_item(item.clone()).await { - Err(err) => result.error(&format!("Unable to add item {} to remote calendar: {}", id_add, err)), + Err(err) => progress.error(&format!("Unable to add item {} to remote calendar: {}", id_add, err)), Ok(new_ss) => { // Update local sync status item.set_sync_status(new_ss); @@ -324,15 +324,15 @@ where } for id_change in local_changes { - result.debug(&format!("> Pushing local change {} to the server", id_change)); + progress.debug(&format!("> Pushing local change {} to the server", id_change)); match cal_local.get_item_by_id_mut(&id_change).await { None => { - result.error(&format!("Inconsistency: modified item {} has been marked for upload but is locally missing", id_change)); + progress.error(&format!("Inconsistency: modified item {} has been marked for upload but is locally missing", id_change)); continue; }, Some(item) => { match cal_remote.update_item(item.clone()).await { - Err(err) => result.error(&format!("Unable to update item {} in remote calendar: {}", id_change, err)), + Err(err) => progress.error(&format!("Unable to update item {} in remote calendar: {}", id_change, err)), Ok(new_ss) => { // Update local sync status item.set_sync_status(new_ss); diff --git a/src/provider/sync_progress.rs b/src/provider/sync_progress.rs index 1399233..c0897f1 100644 --- a/src/provider/sync_progress.rs +++ b/src/provider/sync_progress.rs @@ -1,8 +1,8 @@ /// A counter of errors that happen during a sync -pub struct SyncResult { +pub struct SyncProgress { n_errors: u32, } -impl SyncResult { +impl SyncProgress { pub fn new() -> Self { Self { n_errors: 0 } } From 7fb98a471bb71dec8092432a7d9f5c0541c928e7 Mon Sep 17 00:00:00 2001 From: daladim Date: Fri, 8 Oct 2021 23:58:17 +0200 Subject: [PATCH 4/5] Feedback infrastructure --- src/provider/mod.rs | 29 +++++++++++++----- src/provider/sync_progress.rs | 58 +++++++++++++++++++++++++++++++++-- 2 files changed, 77 insertions(+), 10 deletions(-) diff --git a/src/provider/mod.rs b/src/provider/mod.rs index 55e1127..348a104 100644 --- a/src/provider/mod.rs +++ b/src/provider/mod.rs @@ -14,6 +14,7 @@ use crate::calendar::CalendarId; mod sync_progress; use sync_progress::SyncProgress; +pub use sync_progress::FeedbackSender; /// A data source that combines two `CalDavSource`s, which is able to sync both sources. /// @@ -64,22 +65,34 @@ where /// To be sure `local` accurately mirrors the `remote` source, you can run [`Provider::sync`] pub fn remote(&self) -> &R { &self.remote } - /// Performs a synchronisation between `local` and `remote`. + /// Performs a synchronisation between `local` and `remote`, and provide feeedback to the user about the progress. /// /// This bidirectional sync applies additions/deletions made on a source to the other source. /// In case of conflicts (the same item has been modified on both ends since the last sync, `remote` always wins) /// /// It returns whether the sync was totally successful (details about errors are logged using the `log::*` macros). /// In case errors happened, the sync might have been partially executed, and you can safely run this function again, since it has been designed to gracefully recover from errors. - pub async fn sync(&mut self) -> bool { - let mut result = SyncProgress::new(); - if let Err(err) = self.run_sync(&mut result).await { - result.error(&format!("Sync terminated because of an error: {}", err)); - } - result.is_success() + pub async fn sync_with_feedback(&mut self, feedback_sender: FeedbackSender) -> bool { + let mut progress = SyncProgress::new_with_feedback_channel(feedback_sender); + self.run_sync(&mut progress).await } - async fn run_sync(&mut self, progress: &mut SyncProgress) -> Result<(), Box> { + /// Performs a synchronisation between `local` and `remote`, without giving any feedback. + /// + /// See [sync_with_feedback] + pub async fn sync(&mut self) -> bool { + let mut progress = SyncProgress::new(); + self.run_sync(&mut progress).await + } + + async fn run_sync(&mut self, progress: &mut SyncProgress) -> bool { + if let Err(err) = self.run_sync_inner(progress).await { + progress.error(&format!("Sync terminated because of an error: {}", err)); + } + progress.is_success() + } + + async fn run_sync_inner(&mut self, progress: &mut SyncProgress) -> Result<(), Box> { progress.info("Starting a sync."); let mut handled_calendars = HashSet::new(); diff --git a/src/provider/sync_progress.rs b/src/provider/sync_progress.rs index c0897f1..2140bd0 100644 --- a/src/provider/sync_progress.rs +++ b/src/provider/sync_progress.rs @@ -1,11 +1,65 @@ -/// A counter of errors that happen during a sync +//! Utilities to track the progression of a sync + +use std::fmt::{Display, Error, Formatter}; + +/// An event that happens during a sync +pub enum SyncEvent { + /// Sync has not started + NotStarted, + /// Sync has just started but no calendar is handled yet + Started, + /// Sync is in progress. + InProgress{ calendar: String, details: String}, + /// Sync is finished + Finished{ success: bool }, +} + +impl Display for SyncEvent { + fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error> { + match self { + SyncEvent::NotStarted => write!(f, "Not started"), + SyncEvent::Started => write!(f, "Sync has started"), + SyncEvent::InProgress{calendar, details} => write!(f, "[{}] {}", calendar, details), + SyncEvent::Finished{success} => match success { + true => write!(f, "Sync successfully finished"), + false => write!(f, "Sync finished with errors"), + } + } + } +} + +impl Default for SyncEvent { + fn default() -> Self { + Self::NotStarted + } +} + + + +pub type FeedbackSender = tokio::sync::watch::Sender; +pub type FeedbackReceiver = tokio::sync::watch::Receiver; + +pub fn feedback_channel() -> (FeedbackSender, FeedbackReceiver) { + tokio::sync::watch::channel(SyncEvent::default()) +} + + + + +/// A structure that tracks the progression and the errors that happen during a sync pub struct SyncProgress { n_errors: u32, + feedback_channel: Option } impl SyncProgress { pub fn new() -> Self { - Self { n_errors: 0 } + Self { n_errors: 0, feedback_channel: None } } + pub fn new_with_feedback_channel(channel: FeedbackSender) -> Self { + Self { n_errors: 0, feedback_channel: Some(channel) } + } + + pub fn is_success(&self) -> bool { self.n_errors == 0 } From b404fc68e8e8d0d5cebda5864939b7077cd011c8 Mon Sep 17 00:00:00 2001 From: daladim Date: Sat, 9 Oct 2021 00:25:23 +0200 Subject: [PATCH 5/5] Sending sync progress --- src/provider/mod.rs | 50 ++++++++++++++++++++++++++++++++--- src/provider/sync_progress.rs | 13 +++++++++ 2 files changed, 60 insertions(+), 3 deletions(-) diff --git a/src/provider/mod.rs b/src/provider/mod.rs index 348a104..33ca360 100644 --- a/src/provider/mod.rs +++ b/src/provider/mod.rs @@ -9,12 +9,12 @@ use std::sync::{Arc, Mutex}; use crate::traits::{BaseCalendar, CalDavSource, DavCalendar}; use crate::traits::CompleteCalendar; -use crate::item::SyncStatus; +use crate::item::{ItemId, SyncStatus}; use crate::calendar::CalendarId; -mod sync_progress; +pub mod sync_progress; use sync_progress::SyncProgress; -pub use sync_progress::FeedbackSender; +use sync_progress::{FeedbackSender, SyncEvent}; /// A data source that combines two `CalDavSource`s, which is able to sync both sources. /// @@ -89,11 +89,13 @@ where if let Err(err) = self.run_sync_inner(progress).await { progress.error(&format!("Sync terminated because of an error: {}", err)); } + progress.feedback(SyncEvent::Finished{ success: progress.is_success() }); progress.is_success() } async fn run_sync_inner(&mut self, progress: &mut SyncProgress) -> Result<(), Box> { progress.info("Starting a sync."); + progress.feedback(SyncEvent::Started); let mut handled_calendars = HashSet::new(); @@ -151,6 +153,13 @@ where async fn sync_calendar_pair(cal_local: Arc>, cal_remote: Arc>, progress: &mut SyncProgress) -> Result<(), Box> { let mut cal_remote = cal_remote.lock().unwrap(); let mut cal_local = cal_local.lock().unwrap(); + let cal_name = cal_local.name().to_string(); + + progress.info(&format!("Syncing calendar {}", cal_name)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: "started".to_string() + }); // Step 1 - find the differences progress.debug("Finding the differences to sync..."); @@ -162,6 +171,11 @@ where let mut remote_additions = HashSet::new(); let remote_items = cal_remote.get_item_version_tags().await?; + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: format!("{} remote items", remote_items.len()), + }); + let mut local_items_to_handle = cal_local.get_item_ids().await?; for (id, remote_tag) in remote_items { progress.trace(&format!("***** Considering remote item {}...", id)); @@ -254,6 +268,10 @@ where progress.trace("Committing changes..."); for id_del in local_del { progress.debug(&format!("> Pushing local deletion {} to the server", id_del)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_del).await, + }); match cal_remote.delete_item(&id_del).await { Err(err) => { progress.warn(&format!("Unable to delete remote item {}: {}", id_del, err)); @@ -269,6 +287,10 @@ where for id_del in remote_del { progress.debug(&format!("> Applying remote deletion {} locally", id_del)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_del).await, + }); if let Err(err) = cal_local.immediately_delete_item(&id_del).await { progress.warn(&format!("Unable to delete local item {}: {}", id_del, err)); } @@ -276,6 +298,10 @@ where for id_add in remote_additions { progress.debug(&format!("> Applying remote addition {} locally", id_add)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_add).await, + }); match cal_remote.get_item_by_id(&id_add).await { Err(err) => { progress.warn(&format!("Unable to get remote item {}: {}. Skipping it.", id_add, err)); @@ -297,6 +323,10 @@ where for id_change in remote_changes { progress.debug(&format!("> Applying remote change {} locally", id_change)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_change).await, + }); match cal_remote.get_item_by_id(&id_change).await { Err(err) => { progress.warn(&format!("Unable to get remote item {}: {}. Skipping it", id_change, err)); @@ -319,6 +349,10 @@ where for id_add in local_additions { progress.debug(&format!("> Pushing local addition {} to the server", id_add)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_add).await, + }); match cal_local.get_item_by_id_mut(&id_add).await { None => { progress.error(&format!("Inconsistency: created item {} has been marked for upload but is locally missing", id_add)); @@ -338,6 +372,10 @@ where for id_change in local_changes { progress.debug(&format!("> Pushing local change {} to the server", id_change)); + progress.feedback(SyncEvent::InProgress{ + calendar: cal_name.clone(), + details: Self::item_name(&cal_local, &id_change).await, + }); match cal_local.get_item_by_id_mut(&id_change).await { None => { progress.error(&format!("Inconsistency: modified item {} has been marked for upload but is locally missing", id_change)); @@ -357,6 +395,12 @@ where Ok(()) } + + + async fn item_name(cal: &T, id: &ItemId) -> String { + cal.get_item_by_id(id).await.map(|item| item.name()).unwrap_or_default().to_string() + } + } diff --git a/src/provider/sync_progress.rs b/src/provider/sync_progress.rs index 2140bd0..d2a24bf 100644 --- a/src/provider/sync_progress.rs +++ b/src/provider/sync_progress.rs @@ -64,21 +64,34 @@ impl SyncProgress { self.n_errors == 0 } + /// Log an error pub fn error(&mut self, text: &str) { log::error!("{}", text); self.n_errors += 1; } + /// Log a warning pub fn warn(&mut self, text: &str) { log::warn!("{}", text); self.n_errors += 1; } + /// Log an info pub fn info(&mut self, text: &str) { log::info!("{}", text); } + /// Log a debug message pub fn debug(&mut self, text: &str) { log::debug!("{}", text); } + /// Log a trace message pub fn trace(&mut self, text: &str) { log::trace!("{}", text); } + /// Send an event as a feedback to the listener (if any). + pub fn feedback(&mut self, event: SyncEvent) { + self.feedback_channel + .as_ref() + .map(|sender| { + sender.send(event) + }); + } }