2021-03-29 09:31:22 +02:00
|
|
|
//! A module to parse ICal files
|
|
|
|
|
|
|
|
use std::error::Error;
|
|
|
|
|
|
|
|
use ical::parser::ical::component::{IcalCalendar, IcalEvent, IcalTodo};
|
2021-04-14 21:19:32 +02:00
|
|
|
use chrono::{DateTime, TimeZone, Utc};
|
2021-03-29 09:31:22 +02:00
|
|
|
|
|
|
|
use crate::Item;
|
|
|
|
use crate::item::SyncStatus;
|
|
|
|
use crate::item::ItemId;
|
|
|
|
use crate::Task;
|
|
|
|
use crate::Event;
|
|
|
|
|
|
|
|
|
|
|
|
/// Parse an iCal file into the internal representation [`crate::Item`]
|
|
|
|
pub fn parse(content: &str, item_id: ItemId, sync_status: SyncStatus) -> Result<Item, Box<dyn Error>> {
|
|
|
|
let mut reader = ical::IcalParser::new(content.as_bytes());
|
|
|
|
let parsed_item = match reader.next() {
|
|
|
|
None => return Err(format!("Invalid uCal data to parse for item {}", item_id).into()),
|
|
|
|
Some(item) => match item {
|
|
|
|
Err(err) => return Err(format!("Unable to parse uCal data for item {}: {}", item_id, err).into()),
|
|
|
|
Ok(item) => item,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
let item = match assert_single_type(&parsed_item)? {
|
|
|
|
CurrentType::Event(_) => {
|
|
|
|
Item::Event(Event::new())
|
|
|
|
},
|
|
|
|
|
|
|
|
CurrentType::Todo(todo) => {
|
|
|
|
let mut name = None;
|
2021-04-12 09:21:50 +02:00
|
|
|
let mut uid = None;
|
2021-04-02 08:27:47 +02:00
|
|
|
let mut completed = false;
|
2021-04-14 21:19:32 +02:00
|
|
|
let mut last_modified = None;
|
2021-04-14 23:09:41 +02:00
|
|
|
let mut completion_date = None;
|
2021-04-16 09:05:30 +02:00
|
|
|
let mut creation_date = None;
|
2021-03-29 09:31:22 +02:00
|
|
|
for prop in &todo.properties {
|
|
|
|
if prop.name == "SUMMARY" {
|
|
|
|
name = prop.value.clone();
|
2021-04-02 08:27:47 +02:00
|
|
|
}
|
|
|
|
if prop.name == "STATUS" {
|
|
|
|
// Possible values:
|
|
|
|
// "NEEDS-ACTION" ;Indicates to-do needs action.
|
|
|
|
// "COMPLETED" ;Indicates to-do completed.
|
|
|
|
// "IN-PROCESS" ;Indicates to-do in process of.
|
|
|
|
// "CANCELLED" ;Indicates to-do was cancelled.
|
|
|
|
if prop.value.as_ref().map(|s| s.as_str()) == Some("COMPLETED") {
|
|
|
|
completed = true;
|
|
|
|
}
|
2021-03-29 09:31:22 +02:00
|
|
|
}
|
2021-04-12 09:21:50 +02:00
|
|
|
if prop.name == "UID" {
|
|
|
|
uid = prop.value.clone();
|
|
|
|
}
|
2021-04-14 21:19:32 +02:00
|
|
|
if prop.name == "DTSTAMP" {
|
|
|
|
// this property specifies the date and time that the information associated with
|
|
|
|
// the calendar component was last revised in the calendar store.
|
2021-04-14 23:09:41 +02:00
|
|
|
last_modified = parse_date_time_from_property(&prop.value)
|
|
|
|
}
|
|
|
|
if prop.name == "COMPLETED" {
|
|
|
|
// this property specifies the date and time that the information associated with
|
|
|
|
// the calendar component was last revised in the calendar store.
|
|
|
|
completion_date = parse_date_time_from_property(&prop.value)
|
2021-04-14 21:19:32 +02:00
|
|
|
}
|
2021-04-16 09:05:30 +02:00
|
|
|
if prop.name == "CREATED" {
|
|
|
|
// The property can be specified once, but is not mandatory
|
|
|
|
creation_date = parse_date_time_from_property(&prop.value)
|
|
|
|
}
|
2021-03-29 09:31:22 +02:00
|
|
|
}
|
|
|
|
let name = match name {
|
|
|
|
Some(name) => name,
|
|
|
|
None => return Err(format!("Missing name for item {}", item_id).into()),
|
|
|
|
};
|
2021-04-12 09:21:50 +02:00
|
|
|
let uid = match uid {
|
|
|
|
Some(uid) => uid,
|
|
|
|
None => return Err(format!("Missing UID for item {}", item_id).into()),
|
|
|
|
};
|
2021-04-14 21:19:32 +02:00
|
|
|
let last_modified = match last_modified {
|
|
|
|
Some(dt) => dt,
|
|
|
|
None => return Err(format!("Missing DTSTAMP for item {}, but this is required by RFC5545", item_id).into()),
|
|
|
|
};
|
2021-04-14 23:09:41 +02:00
|
|
|
if completion_date.is_none() && completed ||
|
|
|
|
completion_date.is_some() && !completed
|
|
|
|
{
|
|
|
|
log::warn!("Inconsistant iCal data: completion date is {:?} but completion status is {:?}", completion_date, completed);
|
|
|
|
}
|
2021-03-29 09:31:22 +02:00
|
|
|
|
2021-04-16 09:05:30 +02:00
|
|
|
Item::Task(Task::new_with_parameters(name, uid, item_id, sync_status, creation_date, last_modified, completion_date))
|
2021-03-29 09:31:22 +02:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// What to do with multiple items?
|
|
|
|
if reader.next().map(|r| r.is_ok()) == Some(true) {
|
|
|
|
return Err("Parsing multiple items are not supported".into());
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(item)
|
|
|
|
}
|
|
|
|
|
2021-04-14 21:19:32 +02:00
|
|
|
fn parse_date_time(dt: &str) -> Result<DateTime<Utc>, chrono::format::ParseError> {
|
|
|
|
Utc.datetime_from_str(dt, "%Y%m%dT%H%M%S")
|
|
|
|
}
|
|
|
|
|
2021-04-14 23:09:41 +02:00
|
|
|
fn parse_date_time_from_property(value: &Option<String>) -> Option<DateTime<Utc>> {
|
|
|
|
value.as_ref()
|
|
|
|
.and_then(|s| {
|
|
|
|
parse_date_time(s)
|
|
|
|
.map_err(|err| {
|
|
|
|
log::warn!("Invalid timestamp: {}", s);
|
|
|
|
err
|
|
|
|
})
|
|
|
|
.ok()
|
|
|
|
})
|
|
|
|
}
|
2021-04-14 21:19:32 +02:00
|
|
|
|
|
|
|
|
2021-03-29 09:31:22 +02:00
|
|
|
enum CurrentType<'a> {
|
|
|
|
Event(&'a IcalEvent),
|
|
|
|
Todo(&'a IcalTodo),
|
|
|
|
}
|
|
|
|
|
|
|
|
fn assert_single_type<'a>(item: &'a IcalCalendar) -> Result<CurrentType<'a>, Box<dyn Error>> {
|
|
|
|
let n_events = item.events.len();
|
|
|
|
let n_todos = item.todos.len();
|
|
|
|
let n_journals = item.journals.len();
|
|
|
|
|
|
|
|
if n_events == 1 {
|
|
|
|
if n_todos != 0 || n_journals != 0 {
|
|
|
|
return Err("Only a single TODO or a single EVENT is supported".into());
|
|
|
|
} else {
|
|
|
|
return Ok(CurrentType::Event(&item.events[0]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if n_todos == 1 {
|
|
|
|
if n_events != 0 || n_journals != 0 {
|
|
|
|
return Err("Only a single TODO or a single EVENT is supported".into());
|
|
|
|
} else {
|
|
|
|
return Ok(CurrentType::Todo(&item.todos[0]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Err("Only a single TODO or a single EVENT is supported".into());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
const EXAMPLE_ICAL: &str = r#"BEGIN:VCALENDAR
|
|
|
|
VERSION:2.0
|
|
|
|
PRODID:-//Nextcloud Tasks v0.13.6
|
|
|
|
BEGIN:VTODO
|
2021-04-12 09:21:50 +02:00
|
|
|
UID:0633de27-8c32-42be-bcb8-63bc879c6185@some-domain.com
|
2021-03-29 09:31:22 +02:00
|
|
|
CREATED:20210321T001600
|
|
|
|
LAST-MODIFIED:20210321T001600
|
|
|
|
DTSTAMP:20210321T001600
|
|
|
|
SUMMARY:Do not forget to do this
|
|
|
|
END:VTODO
|
|
|
|
END:VCALENDAR
|
2021-04-02 08:27:47 +02:00
|
|
|
"#;
|
|
|
|
|
|
|
|
const EXAMPLE_ICAL_COMPLETED: &str = r#"BEGIN:VCALENDAR
|
|
|
|
VERSION:2.0
|
|
|
|
PRODID:-//Nextcloud Tasks v0.13.6
|
|
|
|
BEGIN:VTODO
|
2021-04-12 09:21:50 +02:00
|
|
|
UID:19960401T080045Z-4000F192713-0052@example.com
|
2021-04-02 08:27:47 +02:00
|
|
|
CREATED:20210321T001600
|
|
|
|
LAST-MODIFIED:20210402T081557
|
|
|
|
DTSTAMP:20210402T081557
|
|
|
|
SUMMARY:Clean up your room or Mom will be angry
|
|
|
|
PERCENT-COMPLETE:100
|
|
|
|
COMPLETED:20210402T081557
|
|
|
|
STATUS:COMPLETED
|
|
|
|
END:VTODO
|
|
|
|
END:VCALENDAR
|
2021-03-29 09:31:22 +02:00
|
|
|
"#;
|
|
|
|
|
|
|
|
const EXAMPLE_MULTIPLE_ICAL: &str = r#"BEGIN:VCALENDAR
|
|
|
|
VERSION:2.0
|
|
|
|
PRODID:-//Nextcloud Tasks v0.13.6
|
|
|
|
BEGIN:VTODO
|
|
|
|
UID:0633de27-8c32-42be-bcb8-63bc879c6185
|
|
|
|
CREATED:20210321T001600
|
|
|
|
LAST-MODIFIED:20210321T001600
|
|
|
|
DTSTAMP:20210321T001600
|
|
|
|
SUMMARY:Call Mom
|
|
|
|
END:VTODO
|
|
|
|
END:VCALENDAR
|
|
|
|
BEGIN:VCALENDAR
|
|
|
|
BEGIN:VTODO
|
|
|
|
UID:0633de27-8c32-42be-bcb8-63bc879c6185
|
|
|
|
CREATED:20210321T001600
|
|
|
|
LAST-MODIFIED:20210321T001600
|
|
|
|
DTSTAMP:20210321T001600
|
|
|
|
SUMMARY:Buy a gift for Mom
|
|
|
|
END:VTODO
|
|
|
|
END:VCALENDAR
|
|
|
|
"#;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
use crate::item::VersionTag;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_ical_parsing() {
|
|
|
|
let version_tag = VersionTag::from(String::from("test-tag"));
|
|
|
|
let sync_status = SyncStatus::Synced(version_tag);
|
|
|
|
let item_id: ItemId = "http://some.id/for/testing".parse().unwrap();
|
|
|
|
|
|
|
|
let item = parse(EXAMPLE_ICAL, item_id.clone(), sync_status.clone()).unwrap();
|
|
|
|
let task = item.unwrap_task();
|
|
|
|
|
|
|
|
assert_eq!(task.name(), "Do not forget to do this");
|
|
|
|
assert_eq!(task.id(), &item_id);
|
2021-04-12 09:21:50 +02:00
|
|
|
assert_eq!(task.uid(), "0633de27-8c32-42be-bcb8-63bc879c6185@some-domain.com");
|
2021-03-29 09:31:22 +02:00
|
|
|
assert_eq!(task.completed(), false);
|
|
|
|
assert_eq!(task.sync_status(), &sync_status);
|
2021-04-14 21:19:32 +02:00
|
|
|
assert_eq!(task.last_modified(), &Utc.ymd(2021, 03, 21).and_hms(0, 16, 0));
|
2021-03-29 09:31:22 +02:00
|
|
|
}
|
|
|
|
|
2021-04-02 08:27:47 +02:00
|
|
|
#[test]
|
|
|
|
fn test_ical_completion_parsing() {
|
|
|
|
let version_tag = VersionTag::from(String::from("test-tag"));
|
|
|
|
let sync_status = SyncStatus::Synced(version_tag);
|
|
|
|
let item_id: ItemId = "http://some.id/for/testing".parse().unwrap();
|
|
|
|
|
|
|
|
let item = parse(EXAMPLE_ICAL_COMPLETED, item_id.clone(), sync_status.clone()).unwrap();
|
|
|
|
let task = item.unwrap_task();
|
|
|
|
|
|
|
|
assert_eq!(task.completed(), true);
|
|
|
|
}
|
|
|
|
|
2021-03-29 09:31:22 +02:00
|
|
|
#[test]
|
|
|
|
fn test_multiple_items_in_ical() {
|
|
|
|
let version_tag = VersionTag::from(String::from("test-tag"));
|
|
|
|
let sync_status = SyncStatus::Synced(version_tag);
|
|
|
|
let item_id: ItemId = "http://some.id/for/testing".parse().unwrap();
|
|
|
|
|
|
|
|
let item = parse(EXAMPLE_MULTIPLE_ICAL, item_id.clone(), sync_status.clone());
|
|
|
|
assert!(item.is_err());
|
|
|
|
}
|
|
|
|
}
|