//! Code to connect to a Caldav server
//!
//! Some of it comes from https://github.com/marshalshi/caldav-client-rust.git
use std::error::Error;
use reqwest::Method;
use reqwest::header::CONTENT_TYPE;
use minidom::Element;
use url::Url;
static DAVCLIENT_BODY: &str = r#"
"#;
static HOMESET_BODY: &str = r#"
"#;
pub struct Client {
url: Url,
username: String,
password: String,
principal: Option,
calendar_home_set: Option,
}
impl Client {
pub fn new, T: ToString, U: ToString>(url: S, username: T, password: U) -> Result> {
let url = Url::parse(url.as_ref())?;
Ok(Self{
url,
username: username.to_string(),
password: password.to_string(),
principal: None,
calendar_home_set: None,
})
}
async fn sub_request(&self, url: &Url, body: String, item1: &str, item2: &str) -> Result> {
let method = Method::from_bytes(b"PROPFIND")
.expect("cannot create PROPFIND method.");
let res = reqwest::Client::new()
.request(method, url.as_str())
.header("Depth", 0)
.header(CONTENT_TYPE, "application/xml")
.basic_auth(self.username.clone(), Some(self.password.clone()))
.body(body)
.send()
.await?;
let text = res.text().await?;
let root: Element = text.parse().unwrap();
let elem1 = find_elem(&root, item1.to_string()).unwrap();
let elem2 = find_elem(elem1, item2.to_string()).unwrap();
let text = elem2.text();
Ok(text)
}
/// Return the Principal URL, or fetch it from server if not known yet
async fn get_principal(&mut self) -> Result> {
if let Some(p) = &self.principal {
return Ok(p.clone());
}
let href = self.sub_request(&self.url, DAVCLIENT_BODY.into(), "current-user-principal", "href").await?;
let mut principal_url = self.url.clone();
principal_url.set_path(&href);
self.principal = Some(principal_url.clone());
println!("URL is {}", href);
return Ok(principal_url);
}
/// Return the Homeset URL, or fetch it from server if not known yet
async fn get_cal_home_set(&mut self) -> Result> {
if let Some(h) = &self.calendar_home_set {
return Ok(h.clone());
}
let principal_url = self.get_principal().await?;
let href = self.sub_request(&principal_url, HOMESET_BODY.into(), "calendar-home-set", "href").await?;
let mut chs_url = self.url.clone();
chs_url.set_path(&href);
self.calendar_home_set = Some(chs_url.clone());
println!("Calendar home set {:?}", chs_url.path());
Ok(chs_url)
}
}
pub fn find_elems(root: &Element, tag: String) -> Vec<&Element> {
let mut elems: Vec<&Element> = Vec::new();
for el in root.children() {
if el.name() == tag {
elems.push(el);
} else {
let ret = find_elems(el, tag.clone());
elems.extend(ret);
}
}
elems
}
pub fn find_elem(root: &Element, tag: String) -> Option<&Element> {
if root.name() == tag {
return Some(root);
}
for el in root.children() {
if el.name() == tag {
return Some(el);
} else {
let ret = find_elem(el, tag.clone());
if ret.is_some() {
return ret;
}
}
}
None
}
#[cfg(test)]
mod test {
use super::*;
use crate::settings::URL;
use crate::settings::USERNAME;
use crate::settings::PASSWORD;
#[tokio::test]
async fn test_client() {
let mut client = Client::new(URL, USERNAME, PASSWORD).unwrap();
client.get_cal_home_set().await.unwrap();
}
}