kitchen-freezer/src/data/client.rs

212 lines
6.3 KiB
Rust
Raw Normal View History

2021-02-20 00:10:05 +01:00
//! 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#"
<d:propfind xmlns:d="DAV:">
<d:prop>
<d:current-user-principal />
</d:prop>
</d:propfind>
"#;
static HOMESET_BODY: &str = r#"
<d:propfind xmlns:d="DAV:" xmlns:c="urn:ietf:params:xml:ns:caldav" >
<d:self/>
<d:prop>
<c:calendar-home-set />
</d:prop>
</d:propfind>
"#;
2021-02-20 00:46:20 +01:00
static CAL_BODY: &str = r#"
<d:propfind xmlns:d="DAV:" xmlns:c="urn:ietf:params:xml:ns:caldav" >
<d:prop>
<d:displayname />
<d:resourcetype />
<c:supported-calendar-component-set />
</d:prop>
</d:propfind>
"#;
2021-02-20 00:10:05 +01:00
pub struct Client {
url: Url,
username: String,
password: String,
principal: Option<Url>,
calendar_home_set: Option<Url>,
2021-02-20 17:59:12 +01:00
calendars_url: Option<Vec<Url>>,
2021-02-20 00:10:05 +01:00
}
2021-02-18 12:02:04 +01:00
impl Client {
2021-02-20 17:58:46 +01:00
/// Create a client. This does not start a connection
2021-02-20 00:10:05 +01:00
pub fn new<S: AsRef<str>, T: ToString, U: ToString>(url: S, username: T, password: U) -> Result<Self, Box<dyn Error>> {
let url = Url::parse(url.as_ref())?;
Ok(Self{
url,
username: username.to_string(),
password: password.to_string(),
principal: None,
calendar_home_set: None,
2021-02-20 17:59:12 +01:00
calendars_url: None,
2021-02-20 00:10:05 +01:00
})
}
2021-02-20 00:46:20 +01:00
async fn sub_request(&self, url: &Url, body: String, depth: u32) -> Result<String, Box<dyn Error>> {
2021-02-20 00:10:05 +01:00
let method = Method::from_bytes(b"PROPFIND")
.expect("cannot create PROPFIND method.");
let res = reqwest::Client::new()
2021-02-20 00:18:51 +01:00
.request(method, url.as_str())
2021-02-20 00:46:20 +01:00
.header("Depth", depth)
2021-02-20 00:10:05 +01:00
.header(CONTENT_TYPE, "application/xml")
.basic_auth(self.username.clone(), Some(self.password.clone()))
2021-02-20 00:18:51 +01:00
.body(body)
2021-02-20 00:10:05 +01:00
.send()
.await?;
let text = res.text().await?;
2021-02-20 00:18:51 +01:00
Ok(text)
}
2021-02-20 00:10:05 +01:00
2021-02-20 00:46:20 +01:00
async fn sub_request_and_process(&self, url: &Url, body: String, items: &[&str]) -> Result<String, Box<dyn Error>> {
let text = self.sub_request(url, body, 0).await?;
let mut current_element: &Element = &text.parse().unwrap();
items.iter()
.map(|item| {
current_element = find_elem(&current_element, item.to_string()).unwrap();
})
.collect::<()>();
Ok(current_element.text())
}
2021-02-20 00:18:51 +01:00
/// Return the Principal URL, or fetch it from server if not known yet
async fn get_principal(&mut self) -> Result<Url, Box<dyn Error>> {
if let Some(p) = &self.principal {
return Ok(p.clone());
}
2021-02-20 00:10:05 +01:00
2021-02-20 00:46:20 +01:00
let href = self.sub_request_and_process(&self.url, DAVCLIENT_BODY.into(), &["current-user-principal", "href"]).await?;
2021-02-20 00:10:05 +01:00
let mut principal_url = self.url.clone();
2021-02-20 00:18:51 +01:00
principal_url.set_path(&href);
2021-02-20 00:10:05 +01:00
self.principal = Some(principal_url.clone());
2021-02-20 00:18:51 +01:00
println!("URL is {}", href);
2021-02-20 00:10:05 +01:00
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<Url, Box<dyn Error>> {
if let Some(h) = &self.calendar_home_set {
return Ok(h.clone());
}
let principal_url = self.get_principal().await?;
2021-02-20 00:46:20 +01:00
let href = self.sub_request_and_process(&principal_url, HOMESET_BODY.into(), &["calendar-home-set", "href"]).await?;
2021-02-20 00:10:05 +01:00
let mut chs_url = self.url.clone();
2021-02-20 00:18:51 +01:00
chs_url.set_path(&href);
2021-02-20 00:10:05 +01:00
self.calendar_home_set = Some(chs_url.clone());
2021-02-20 00:18:51 +01:00
println!("Calendar home set {:?}", chs_url.path());
2021-02-20 00:10:05 +01:00
Ok(chs_url)
}
2021-02-20 00:46:20 +01:00
2021-02-20 17:59:12 +01:00
/// Return the list of calendars, or fetch from server if not known yet
pub async fn get_calendars_url(&mut self) -> Result<Vec<Url>, Box<dyn Error>> {
if let Some(c) = &self.calendars_url {
return Ok(c.clone());
}
2021-02-20 00:46:20 +01:00
let cal_home_set = self.get_cal_home_set().await?;
let text = self.sub_request(&cal_home_set, CAL_BODY.into(), 1).await?;
let root: Element = text.parse().unwrap();
let reps = find_elems(&root, "response".to_string());
2021-02-20 17:59:12 +01:00
let mut calendars = Vec::new();
2021-02-20 00:46:20 +01:00
for rep in reps {
// TODO checking `displayname` here but may there are better way
let displayname = find_elem(rep, "displayname".to_string())
.unwrap()
.text();
if displayname == "" {
continue;
}
// TODO: filter by:
// <cal:supported-calendar-component-set>
// <cal:comp name=\"VEVENT\"/>
// <cal:comp name=\"VTODO\"/>
// </cal:supported-calendar-component-set>
let href = find_elem(rep, "href".to_string()).unwrap();
let href_text = href.text();
2021-02-20 17:59:12 +01:00
let mut this_calendar_url = self.url.clone();
this_calendar_url.set_path(&href_text);
calendars.push(this_calendar_url);
}
Ok(calendars)
2021-02-20 00:46:20 +01:00
}
2021-02-20 00:10:05 +01:00
}
2021-02-20 18:04:08 +01:00
/// Walks the tree and returns every element that has the given name
pub fn find_elems(root: &Element, searched_name: String) -> Vec<&Element> {
2021-02-20 00:10:05 +01:00
let mut elems: Vec<&Element> = Vec::new();
for el in root.children() {
2021-02-20 18:04:08 +01:00
if el.name() == searched_name {
2021-02-20 00:10:05 +01:00
elems.push(el);
} else {
2021-02-20 18:04:08 +01:00
let ret = find_elems(el, searched_name.clone());
2021-02-20 00:10:05 +01:00
elems.extend(ret);
}
}
elems
}
2021-02-20 18:04:08 +01:00
/// Walks the tree until it finds an elements with the given name
pub fn find_elem(root: &Element, searched_name: String) -> Option<&Element> {
if root.name() == searched_name {
2021-02-20 00:10:05 +01:00
return Some(root);
}
for el in root.children() {
2021-02-20 18:04:08 +01:00
if el.name() == searched_name {
2021-02-20 00:10:05 +01:00
return Some(el);
} else {
2021-02-20 18:04:08 +01:00
let ret = find_elem(el, searched_name.clone());
2021-02-20 00:10:05 +01:00
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();
2021-02-20 17:59:12 +01:00
client.get_calendars_url().await.unwrap();
2021-02-18 12:02:04 +01:00
}
2021-02-20 00:10:05 +01:00
}