mirror of
https://github.com/lennart-k/rustical.git
synced 2026-01-30 16:28:21 +00:00
Re-implement calendar export
This commit is contained in:
@@ -5,11 +5,11 @@ use axum::extract::State;
|
||||
use axum::{extract::Path, response::Response};
|
||||
use headers::{ContentType, HeaderMapExt};
|
||||
use http::{HeaderValue, Method, StatusCode, header};
|
||||
use ical::component::IcalCalendar;
|
||||
use ical::generator::Emitter;
|
||||
use ical::property::ContentLine;
|
||||
use percent_encoding::{CONTROLS, utf8_percent_encode};
|
||||
use rustical_store::{CalendarStore, SubscriptionStore, auth::Principal};
|
||||
use std::collections::HashMap;
|
||||
use std::str::FromStr;
|
||||
use tracing::instrument;
|
||||
|
||||
@@ -31,79 +31,62 @@ pub async fn route_get<C: CalendarStore, S: SubscriptionStore>(
|
||||
return Err(crate::Error::Unauthorized);
|
||||
}
|
||||
|
||||
// let mut vtimezones = HashMap::new();
|
||||
// let objects = cal_store.get_objects(&principal, &calendar_id).await?;
|
||||
let objects = cal_store
|
||||
.get_objects(&principal, &calendar_id)
|
||||
.await?
|
||||
.into_iter()
|
||||
.map(|(_, object)| object.into())
|
||||
.collect();
|
||||
|
||||
todo!()
|
||||
let mut props = vec![];
|
||||
|
||||
// let mut ical_calendar_builder = IcalCalendarBuilder::version("2.0")
|
||||
// .gregorian()
|
||||
// .prodid("RustiCal");
|
||||
// if let Some(displayname) = calendar.meta.displayname {
|
||||
// ical_calendar_builder = ical_calendar_builder.set(ContentLine {
|
||||
// name: "X-WR-CALNAME".to_owned(),
|
||||
// value: Some(displayname),
|
||||
// params: vec![].into(),
|
||||
// });
|
||||
// }
|
||||
// if let Some(description) = calendar.meta.description {
|
||||
// ical_calendar_builder = ical_calendar_builder.set(ContentLine {
|
||||
// name: "X-WR-CALDESC".to_owned(),
|
||||
// value: Some(description),
|
||||
// params: vec![].into(),
|
||||
// });
|
||||
// }
|
||||
// if let Some(timezone_id) = calendar.timezone_id {
|
||||
// ical_calendar_builder = ical_calendar_builder.set(ContentLine {
|
||||
// name: "X-WR-TIMEZONE".to_owned(),
|
||||
// value: Some(timezone_id),
|
||||
// params: vec![].into(),
|
||||
// });
|
||||
// }
|
||||
//
|
||||
// for object in &objects {
|
||||
// vtimezones.extend(object.get_vtimezones());
|
||||
// match object.get_data() {
|
||||
// CalendarObjectComponent::Event(EventObject { event, .. }, overrides) => {
|
||||
// ical_calendar_builder = ical_calendar_builder
|
||||
// .add_event(event.clone())
|
||||
// .add_events(overrides.iter().map(|ev| ev.event.clone()));
|
||||
// }
|
||||
// CalendarObjectComponent::Todo(todo, overrides) => {
|
||||
// ical_calendar_builder = ical_calendar_builder
|
||||
// .add_todo(todo.clone())
|
||||
// .add_todos(overrides.iter().cloned());
|
||||
// }
|
||||
// CalendarObjectComponent::Journal(journal, overrides) => {
|
||||
// ical_calendar_builder = ical_calendar_builder
|
||||
// .add_journal(journal.clone())
|
||||
// .add_journals(overrides.iter().cloned());
|
||||
// }
|
||||
// }
|
||||
// }
|
||||
//
|
||||
// ical_calendar_builder = ical_calendar_builder.add_timezones(vtimezones.into_values().cloned());
|
||||
//
|
||||
// let ical_calendar = ical_calendar_builder
|
||||
// .build()
|
||||
// .map_err(|parser_error| Error::IcalError(parser_error.into()))?;
|
||||
//
|
||||
// let mut resp = Response::builder().status(StatusCode::OK);
|
||||
// let hdrs = resp.headers_mut().unwrap();
|
||||
// hdrs.typed_insert(ContentType::from_str("text/calendar; charset=utf-8").unwrap());
|
||||
//
|
||||
// let filename = format!("{}_{}.ics", calendar.principal, calendar.id);
|
||||
// let filename = utf8_percent_encode(&filename, CONTROLS);
|
||||
// hdrs.insert(
|
||||
// header::CONTENT_DISPOSITION,
|
||||
// HeaderValue::from_str(&format!(
|
||||
// "attachement; filename*=UTF-8''{filename}; filename={filename}",
|
||||
// ))
|
||||
// .unwrap(),
|
||||
// );
|
||||
// if matches!(method, Method::HEAD) {
|
||||
// Ok(resp.body(Body::empty()).unwrap())
|
||||
// } else {
|
||||
// Ok(resp.body(Body::new(ical_calendar.generate())).unwrap())
|
||||
// }
|
||||
if let Some(displayname) = calendar.meta.displayname {
|
||||
props.push(ContentLine {
|
||||
name: "X-WR-CALNAME".to_owned(),
|
||||
value: Some(displayname),
|
||||
params: vec![].into(),
|
||||
});
|
||||
}
|
||||
if let Some(description) = calendar.meta.description {
|
||||
props.push(ContentLine {
|
||||
name: "X-WR-CALDESC".to_owned(),
|
||||
value: Some(description),
|
||||
params: vec![].into(),
|
||||
});
|
||||
}
|
||||
if let Some(color) = calendar.meta.color {
|
||||
props.push(ContentLine {
|
||||
name: "X-WR-CALCOLOR".to_owned(),
|
||||
value: Some(color),
|
||||
params: vec![].into(),
|
||||
});
|
||||
}
|
||||
if let Some(timezone_id) = calendar.timezone_id {
|
||||
props.push(ContentLine {
|
||||
name: "X-WR-TIMEZONE".to_owned(),
|
||||
value: Some(timezone_id),
|
||||
params: vec![].into(),
|
||||
});
|
||||
}
|
||||
|
||||
let export_calendar = IcalCalendar::from_objects(objects, props);
|
||||
|
||||
let mut resp = Response::builder().status(StatusCode::OK);
|
||||
let hdrs = resp.headers_mut().unwrap();
|
||||
hdrs.typed_insert(ContentType::from_str("text/calendar; charset=utf-8").unwrap());
|
||||
|
||||
let filename = format!("{}_{}.ics", calendar.principal, calendar.id);
|
||||
let filename = utf8_percent_encode(&filename, CONTROLS);
|
||||
hdrs.insert(
|
||||
header::CONTENT_DISPOSITION,
|
||||
HeaderValue::from_str(&format!(
|
||||
"attachement; filename*=UTF-8''{filename}; filename={filename}",
|
||||
))
|
||||
.unwrap(),
|
||||
);
|
||||
if matches!(method, Method::HEAD) {
|
||||
Ok(resp.body(Body::empty()).unwrap())
|
||||
} else {
|
||||
Ok(resp.body(Body::new(export_calendar.generate())).unwrap())
|
||||
}
|
||||
}
|
||||
|
||||
@@ -46,6 +46,9 @@ pub async fn route_import<C: CalendarStore, S: SubscriptionStore>(
|
||||
let description = cal
|
||||
.get_property("X-WR-CALDESC")
|
||||
.and_then(|prop| prop.value.clone());
|
||||
let color = cal
|
||||
.get_property("X-WR-CALCOLOR")
|
||||
.and_then(|prop| prop.value.clone());
|
||||
let timezone_id = cal
|
||||
.get_property("X-WR-TIMEZONE")
|
||||
.and_then(|prop| prop.value.clone());
|
||||
|
||||
@@ -92,7 +92,7 @@ pub async fn route_mkcalendar<C: CalendarStore, S: SubscriptionStore>(
|
||||
.ok_or_else(|| rustical_dav::Error::BadRequest("No timezone data provided".to_owned()))?
|
||||
.map_err(|_| rustical_dav::Error::BadRequest("Error parsing timezone".to_owned()))?;
|
||||
|
||||
let timezone = calendar.vtimezones.first().ok_or_else(|| {
|
||||
let timezone = calendar.vtimezones.values().next().ok_or_else(|| {
|
||||
rustical_dav::Error::BadRequest("No timezone data provided".to_owned())
|
||||
})?;
|
||||
let timezone: Option<chrono_tz::Tz> = timezone.into();
|
||||
|
||||
@@ -215,7 +215,7 @@ impl Resource for CalendarResource {
|
||||
)
|
||||
})?;
|
||||
|
||||
let timezone = calendar.vtimezones.first().ok_or_else(|| {
|
||||
let timezone = calendar.vtimezones.values().next().ok_or_else(|| {
|
||||
rustical_dav::Error::BadRequest("No timezone data provided".to_owned())
|
||||
})?;
|
||||
let timezone: Option<chrono_tz::Tz> = timezone.into();
|
||||
|
||||
Reference in New Issue
Block a user