use async_trait::async_trait;
use http::Uri;
use hyper_util::client::legacy::connect::Connect;
use libdav::dav::mime_types;
use libdav::CardDavClient;
use crate::base::{
AddressBookProperty, Collection, FetchedItem, Item, ItemRef, ListedProperty, Storage, VcardItem,
};
use crate::dav::{
collection_href_for_item, collection_id_for_href, path_for_collection_in_home_set,
};
use crate::disco::{DiscoveredCollection, Discovery};
use crate::vdir::PropertyWithFilename;
use crate::{CollectionId, Error, ErrorKind, Etag, Href, Result};
impl<C> CardDavStorage<C>
where
C: Connect + Send + Sync + Clone + std::fmt::Debug,
{
pub async fn new(client: CardDavClient<C>) -> Result<CardDavStorage<C>> {
let principal = client
.find_current_user_principal()
.await
.map_err(|e| Error::new(ErrorKind::Io, e))?
.ok_or_else(|| Error::new(ErrorKind::Unavailable, "no user principal found"))?;
let address_book_home_set = client
.find_address_book_home_set(&principal)
.await
.map_err(|e| Error::new(ErrorKind::Io, e))?;
Ok(CardDavStorage {
client,
address_book_home_set,
})
}
}
pub struct CardDavStorage<C: Connect + Clone + Sync + Send + 'static> {
client: CardDavClient<C>,
address_book_home_set: Vec<Uri>,
}
#[async_trait]
impl<C> Storage<VcardItem> for CardDavStorage<C>
where
C: Connect + Clone + Sync + Send,
{
async fn check(&self) -> Result<()> {
self.client
.check_support(&self.client.base_url)
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))
}
async fn discover_collections(&self) -> Result<Discovery> {
let mut collections = Vec::new();
for home in &self.address_book_home_set {
collections.append(&mut self.client.find_addressbooks(home).await?);
}
collections
.into_iter()
.map(|collection| {
collection_id_for_href(&collection.href)
.map_err(|e| Error::new(ErrorKind::InvalidData, e))
.map(|id| DiscoveredCollection::new(collection.href, id))
})
.collect::<Result<Vec<_>>>()
.map(Discovery::from)
}
async fn create_collection(&self, href: &str) -> Result<Collection> {
self.client
.create_addressbook(href)
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))?;
Ok(Collection::new(href.to_string()))
}
async fn destroy_collection(&self, href: &str) -> Result<()> {
let mut results = self
.client
.get_address_book_resources(href, &[href])
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))?;
if results.len() > 1 {
return Err(ErrorKind::InvalidData.into());
}
let item = results
.pop()
.ok_or_else(|| Error::from(ErrorKind::InvalidData))?;
if item.href != href {
return Err(Error::new(
ErrorKind::InvalidData,
format!("Requested href: {}, got: {}", href, item.href,),
));
}
let etag = item
.content
.map_err(|e| Error::new(ErrorKind::Uncategorised, format!("Got status code: {e}")))?
.etag;
let items = self.list_items(href).await?;
if !items.is_empty() {
return Err(ErrorKind::CollectionNotEmpty.into());
}
self.client
.delete(href, etag)
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))?;
Ok(())
}
async fn list_items(&self, collection_href: &str) -> Result<Vec<ItemRef>> {
let response = self.client.list_resources(collection_href).await?;
let mut items = Vec::with_capacity(response.len());
for r in response {
items.push(ItemRef {
href: r.href,
etag: r
.details
.etag
.ok_or(Error::from(ErrorKind::InvalidData))?
.into(),
});
}
Ok(items)
}
async fn get_item(&self, href: &str) -> Result<(VcardItem, Etag)> {
let collection_href = collection_href_for_item(href)?;
let mut results = self
.client
.get_address_book_resources(collection_href, &[href])
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))?;
if results.len() != 1 {
return Err(ErrorKind::InvalidData.into());
}
let item = results.pop().expect("results has exactly one item");
if item.href != href {
return Err(Error::new(
ErrorKind::Uncategorised,
format!("Requested href: {}, got: {}", href, item.href,),
));
}
let content = item
.content
.map_err(|e| Error::new(ErrorKind::Uncategorised, format!("Got status code: {e}")))?;
Ok((VcardItem::from(content.data), content.etag.into()))
}
async fn get_many_items(&self, hrefs: &[&str]) -> Result<Vec<FetchedItem<VcardItem>>> {
if hrefs.is_empty() {
return Ok(Vec::new());
}
let collection_href = collection_href_for_item(hrefs[0])?;
self.client
.get_address_book_resources(collection_href, hrefs)
.await
.map_err(|e| Error::new(ErrorKind::Uncategorised, e))?
.into_iter()
.map(|resource| {
resource
.content
.map_err(|e| {
ErrorKind::Io.error(format!("Got status code {} for {}", e, resource.href))
})
.map(|content| FetchedItem {
href: resource.href,
item: VcardItem::from(content.data),
etag: content.etag.into(),
})
})
.collect()
}
async fn get_all_items(&self, collection_href: &str) -> Result<Vec<FetchedItem<VcardItem>>> {
let list = self.list_items(collection_href).await?;
let hrefs = list.iter().map(|i| i.href.as_str()).collect::<Vec<_>>();
self.get_many_items(&hrefs).await
}
async fn add_item(&self, collection_href: &str, item: &VcardItem) -> Result<ItemRef> {
let href = join_hrefs(collection_href, &item.ident());
let response = self
.client
.create_resource(
&href,
item.as_str().as_bytes().to_vec(),
mime_types::ADDRESSBOOK,
)
.await?;
let etag = match response {
Some(e) => e,
None => self.get_item(&href).await?.1.to_string(),
};
Ok(ItemRef {
href,
etag: Etag::from(etag),
})
}
async fn update_item(&self, href: &str, etag: &Etag, item: &VcardItem) -> Result<Etag> {
let raw_etag = self
.client
.update_resource(
href,
item.as_str().as_bytes().to_vec(),
etag,
mime_types::CALENDAR,
)
.await?;
if let Some(etag) = raw_etag {
return Ok(Etag::from(etag));
}
let (new_item, etag) = self.get_item(href).await?;
if new_item.hash() == item.hash() {
return Ok(etag);
}
return Err(ErrorKind::Io.error("Item was overwritten replaced before reading Etag"));
}
async fn set_property(&self, href: &str, prop: AddressBookProperty, value: &str) -> Result<()> {
self.client
.set_property(href, prop.dav_propname(), Some(value))
.await
.map(|_| ())
.map_err(Error::from)
}
async fn unset_property(&self, href: &str, prop: AddressBookProperty) -> Result<()> {
self.client
.set_property(href, prop.dav_propname(), None)
.await
.map(|_| ())
.map_err(Error::from)
}
async fn get_property(&self, href: &str, prop: AddressBookProperty) -> Result<Option<String>> {
self.client
.get_property(href, prop.dav_propname())
.await
.map_err(Error::from)
}
async fn delete_item(&self, href: &str, etag: &Etag) -> Result<()> {
self.client.delete(href, etag).await?;
Ok(())
}
fn collection_id(&self, collection_href: &str) -> Result<CollectionId> {
collection_id_for_href(collection_href).map_err(|e| Error::new(ErrorKind::InvalidInput, e))
}
fn href_for_collection_id(&self, id: &CollectionId) -> Result<Href> {
if let Some(home_set) = &self.address_book_home_set.first() {
Ok(path_for_collection_in_home_set(home_set, id.as_ref()))
} else {
Err(Error::new(
ErrorKind::PreconditionFailed,
"calendar home set not found in caldav server",
))
}
}
async fn list_properties(
&self,
collection_href: &str,
) -> Result<Vec<ListedProperty<AddressBookProperty>>> {
let prop_names = AddressBookProperty::known_properties()
.iter()
.map(|p| p.dav_propname())
.collect::<Vec<_>>();
let result = self
.client
.get_properties(collection_href, &prop_names)
.await?
.into_iter()
.zip(AddressBookProperty::known_properties())
.filter_map(|((_, v), p)| {
v.map(|value| ListedProperty {
property: p.clone(),
value,
})
})
.collect::<Vec<_>>();
return Ok(result);
}
}
fn join_hrefs(collection_href: &str, item_href: &str) -> String {
if item_href.starts_with('/') {
return item_href.to_string();
}
let mut href = collection_href
.strip_suffix('/')
.unwrap_or(collection_href)
.to_string();
href.push('/');
href.push_str(item_href);
href
}