From 0b65dff46d9b6900cea7ce82099bfff9006cc810 Mon Sep 17 00:00:00 2001 From: daladim Date: Tue, 30 Mar 2021 00:07:31 +0200 Subject: [PATCH] Caching version IDs from the server --- src/calendar/remote_calendar.rs | 23 +++++++++++++++++++++-- src/item.rs | 4 ++++ 2 files changed, 25 insertions(+), 2 deletions(-) diff --git a/src/calendar/remote_calendar.rs b/src/calendar/remote_calendar.rs index 1d9ab9c..6e8ec51 100644 --- a/src/calendar/remote_calendar.rs +++ b/src/calendar/remote_calendar.rs @@ -1,5 +1,6 @@ use std::collections::HashMap; use std::error::Error; +use std::sync::Mutex; use async_trait::async_trait; use reqwest::header::CONTENT_TYPE; @@ -31,19 +32,23 @@ static TASKS_BODY: &str = r#" /// A CalDAV calendar created by a [`Client`](crate::client::Client). -#[derive(Clone)] pub struct RemoteCalendar { name: String, resource: Resource, supported_components: SupportedComponents, + + cached_version_tags: Mutex>>, } impl RemoteCalendar { pub fn new(name: String, resource: Resource, supported_components: SupportedComponents) -> Self { Self { name, resource, supported_components, + cached_version_tags: Mutex::new(None), } } + + } #[async_trait] @@ -63,6 +68,11 @@ impl BaseCalendar for RemoteCalendar { #[async_trait] impl DavCalendar for RemoteCalendar { async fn get_item_version_tags(&self) -> Result, Box> { + if let Some(map) = &*self.cached_version_tags.lock().unwrap() { + log::debug!("Version tags are already cached."); + return Ok(map.clone()); + }; + let responses = crate::client::sub_request_and_extract_elems(&self.resource, "REPORT", TASKS_BODY.to_string(), "response").await?; let mut items = HashMap::new(); @@ -92,6 +102,8 @@ impl DavCalendar for RemoteCalendar { items.insert(item_id, version_tag); } + // Note: the mutex cannot be locked during this whole async function, but it can safely be re-entrant (this will just waste an unnecessary request) + *self.cached_version_tags.lock().unwrap() = Some(items.clone()); Ok(items) } @@ -105,7 +117,14 @@ impl DavCalendar for RemoteCalendar { let text = res.text().await?; - let item = crate::ical::parse(&text, id.clone(), SyncStatus::Synced())?; + // This is supposed to be cached + let version_tags = self.get_item_version_tags().await?; + let vt = match version_tags.get(id) { + None => return Err(format!("Inconsistent data: {} has no version tag", id).into()), + Some(vt) => vt, + }; + + let item = crate::ical::parse(&text, id.clone(), SyncStatus::Synced(vt.clone()))?; Ok(Some(item)) } diff --git a/src/item.rs b/src/item.rs index 6fd197d..011697a 100644 --- a/src/item.rs +++ b/src/item.rs @@ -94,6 +94,10 @@ impl ItemId{ let u = s.parse().unwrap(); Self { content:u } } + + pub fn as_url(&self) -> &Url { + &self.content + } } impl From for ItemId { fn from(url: Url) -> Self {