//! This modules abstracts data sources and merges them in a single virtual one use std::error::Error; use std::collections::HashSet; use std::marker::PhantomData; use chrono::{DateTime, Utc}; use crate::traits::{CalDavSource, CompleteCalendar}; use crate::traits::PartialCalendar; use crate::Item; use crate::item::ItemId; /// A data source that combines two `CalDavSources` (usually a server and a local cache), which is able to sync both sources. pub struct Provider where L: CalDavSource, T: CompleteCalendar, R: CalDavSource, U: PartialCalendar + Sync + Send, { /// The remote source (usually a server) remote: R, /// The local cache local: L, phantom_t: PhantomData, phantom_u: PhantomData, } impl Provider where L: CalDavSource, T: CompleteCalendar, R: CalDavSource, U: PartialCalendar + Sync + Send, { /// Create a provider. /// /// `remote` is usually a [`Client`](crate::client::Client), `local` is usually a [`Cache`](crate::cache::Cache). /// However, both can be interchangeable. The only difference is that `remote` always wins in case of a sync conflict pub fn new(remote: R, local: L) -> Self { Self { remote, local, phantom_t: PhantomData, phantom_u: PhantomData, } } /// Returns the data source described as the `server` pub fn remote(&self) -> &R { &self.remote } /// Returns the data source described as the `local` pub fn local(&self) -> &L { &self.local } /// Performs a synchronisation between `local` and `server`. /// /// 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, `server` always wins) pub async fn sync(&mut self) -> Result<(), Box> { log::info!("Starting a sync."); let cals_remote = self.remote.get_calendars().await?; for (id, cal_remote) in cals_remote { let mut cal_remote = cal_remote.lock().unwrap(); let cal_local = match self.local.get_calendar(&id).await { None => { log::error!("TODO: implement here"); continue; }, Some(cal) => cal, }; let mut cal_local = cal_local.lock().unwrap(); // Step 1 - find the differences // let mut local_del = HashSet::new(); // let mut remote_del = HashSet::new(); // let mut local_changes = HashSet::new(); // let mut remote_change = HashSet::new(); // let mut local_additions = HashSet::new(); // let mut remote_additions = HashSet::new(); } Ok(()) } } async fn move_to_calendar(items: &mut Vec, calendar: &mut C) { while items.len() > 0 { let item = items.remove(0); log::warn!("Moving {} to calendar", item.name()); calendar.add_item(item).await; } } async fn remove_from_calendar(ids: &Vec, calendar: &mut C) { for id in ids { log::info!(" Removing {:?} from calendar", id); if let Err(err) = calendar.delete_item(id).await { log::warn!("Unable to delete item {:?} from calendar: {}", id, err); } } }