e2216db41d
Signed-off-by: flip1995 <hello@philkrones.com> Signed-off-by: Matthias Beyer <mail@beyermatthias.de>
330 lines
10 KiB
Rust
330 lines
10 KiB
Rust
//
|
|
// imag - the personal information management suite for the commandline
|
|
// Copyright (C) 2015-2019 Matthias Beyer <mail@beyermatthias.de> and contributors
|
|
//
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
// License as published by the Free Software Foundation; version
|
|
// 2.1 of the License.
|
|
//
|
|
// This library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
// Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this library; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
//
|
|
|
|
use chrono::naive::NaiveDateTime;
|
|
use toml_query::delete::TomlValueDeleteExt;
|
|
use toml_query::insert::TomlValueInsertExt;
|
|
use toml_query::read::TomlValueReadTypeExt;
|
|
use toml::Value;
|
|
|
|
use libimagstore::store::Entry;
|
|
use libimagerror::errors::ErrorMsg as EM;
|
|
|
|
use failure::Error;
|
|
use failure::Fallible as Result;
|
|
use failure::ResultExt;
|
|
use failure::err_msg;
|
|
use crate::range::DateTimeRange;
|
|
|
|
pub trait EntryDate {
|
|
|
|
fn delete_date(&mut self) -> Result<()>;
|
|
fn read_date(&self) -> Result<NaiveDateTime>;
|
|
fn set_date(&mut self, d: NaiveDateTime) -> Result<Option<Result<NaiveDateTime>>>;
|
|
|
|
fn delete_date_range(&mut self) -> Result<()>;
|
|
fn read_date_range(&self) -> Result<DateTimeRange>;
|
|
fn set_date_range(&mut self, start: NaiveDateTime, end: NaiveDateTime) -> Result<Option<Result<DateTimeRange>>>;
|
|
|
|
}
|
|
|
|
const DATE_HEADER_LOCATION : &str = "datetime.value";
|
|
const DATE_RANGE_START_HEADER_LOCATION : &str = "datetime.range.start";
|
|
const DATE_RANGE_END_HEADER_LOCATION : &str = "datetime.range.end";
|
|
const DATE_FMT : &str = "%Y-%m-%dT%H:%M:%S";
|
|
|
|
impl EntryDate for Entry {
|
|
|
|
fn delete_date(&mut self) -> Result<()> {
|
|
self.get_header_mut()
|
|
.delete(&DATE_HEADER_LOCATION)
|
|
.map(|_| ())
|
|
.context("Delete date error")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
fn read_date(&self) -> Result<NaiveDateTime> {
|
|
self.get_header()
|
|
.read_string(&DATE_HEADER_LOCATION)
|
|
.context("Error while reading date")?
|
|
.ok_or_else(|| err_msg("Error reading date"))?
|
|
.parse::<NaiveDateTime>()
|
|
.context("Datetime parse error")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
/// Set a Date for this entry
|
|
///
|
|
/// # Return value
|
|
///
|
|
/// This function returns funny things, I know. But I find it more attractive to be explicit
|
|
/// what failed when here, instead of beeing nice to the user here.
|
|
///
|
|
/// So here's a list how things are returned:
|
|
///
|
|
/// - Err(_) if the inserting failed
|
|
/// - Ok(None) if the inserting succeeded and _did not replace an existing value_.
|
|
/// - Ok(Some(Ok(_))) if the inserting succeeded, but replaced an existing value which then got
|
|
/// parsed into a NaiveDateTime object
|
|
/// - Ok(Some(Err(_))) if the inserting succeeded, but replaced an existing value which then
|
|
/// got parsed into a NaiveDateTime object, where the parsing failed for some reason.
|
|
///
|
|
fn set_date(&mut self, d: NaiveDateTime) -> Result<Option<Result<NaiveDateTime>>> {
|
|
let date = d.format(&DATE_FMT).to_string();
|
|
|
|
self.get_header_mut()
|
|
.insert(&DATE_HEADER_LOCATION, Value::String(date))
|
|
.context(format_err!("Failed to insert header '{}' in '{}'",
|
|
DATE_HEADER_LOCATION,
|
|
self.get_location()))
|
|
.map_err(Error::from)
|
|
.map(|opt| opt.map(|stri| {
|
|
stri.as_str()
|
|
.ok_or_else(|| Error::from(EM::EntryHeaderTypeError))?
|
|
.parse::<NaiveDateTime>()
|
|
.context("Datetime parse error")
|
|
.map_err(Error::from)
|
|
}))
|
|
.context("Error setting date")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
|
|
/// Deletes the date range
|
|
///
|
|
/// # Warning
|
|
///
|
|
/// First deletes the start, then the end. If the first operation fails, this might leave the
|
|
/// header in an inconsistent state.
|
|
///
|
|
fn delete_date_range(&mut self) -> Result<()> {
|
|
self
|
|
.get_header_mut()
|
|
.delete(&DATE_RANGE_START_HEADER_LOCATION)
|
|
.map(|_| ())
|
|
.context("Delete Datetime range error")?;
|
|
|
|
self.get_header_mut()
|
|
.delete(&DATE_RANGE_END_HEADER_LOCATION)
|
|
.map(|_| ())
|
|
.context("Delete Datetime range error")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
fn read_date_range(&self) -> Result<DateTimeRange> {
|
|
let start = self
|
|
.get_header()
|
|
.read_string(&DATE_RANGE_START_HEADER_LOCATION)
|
|
.context("Error while reading Datetime range")?
|
|
.ok_or_else(|| err_msg("Error reading date"))
|
|
.and_then(str_to_ndt)?;
|
|
|
|
let end = self
|
|
.get_header()
|
|
.read_string(&DATE_RANGE_START_HEADER_LOCATION)
|
|
.context("Error reading Datetime range")?
|
|
.ok_or_else(|| err_msg("Error reading date"))
|
|
.and_then(str_to_ndt)?;
|
|
|
|
DateTimeRange::new(start, end)
|
|
.context("Datetime Range error")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
/// Set the date range
|
|
///
|
|
/// # Warning
|
|
///
|
|
/// This first sets the start, then the end. If the first operation fails, this might leave the
|
|
/// header in an inconsistent state.
|
|
///
|
|
fn set_date_range(&mut self, start: NaiveDateTime, end: NaiveDateTime)
|
|
-> Result<Option<Result<DateTimeRange>>>
|
|
{
|
|
let start = start.format(&DATE_FMT).to_string();
|
|
let end = end.format(&DATE_FMT).to_string();
|
|
|
|
let opt_old_start = self
|
|
.get_header_mut()
|
|
.insert(&DATE_RANGE_START_HEADER_LOCATION, Value::String(start))
|
|
.map(|opt| opt.as_ref().map(val_to_ndt))
|
|
.context("Error setting Datetime range")?;
|
|
|
|
let opt_old_end = self
|
|
.get_header_mut()
|
|
.insert(&DATE_RANGE_END_HEADER_LOCATION, Value::String(end))
|
|
.map(|opt| opt.as_ref().map(val_to_ndt))
|
|
.context("Error setting Datetime range")?;
|
|
|
|
match (opt_old_start, opt_old_end) {
|
|
(Some(Ok(old_start)), Some(Ok(old_end))) => {
|
|
let dr = DateTimeRange::new(old_start, old_end)
|
|
.context("Error processing Datetime range")
|
|
.map_err(Error::from);
|
|
|
|
Ok(Some(dr))
|
|
},
|
|
|
|
(Some(Err(e)), _) => Err(e),
|
|
(_, Some(Err(e))) => Err(e),
|
|
_ => {
|
|
Ok(None)
|
|
},
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
#[inline]
|
|
fn str_to_ndt(v: String) -> Result<NaiveDateTime> {
|
|
v.parse::<NaiveDateTime>()
|
|
.context("Error parsing Datetime")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
#[inline]
|
|
fn val_to_ndt(v: &Value) -> Result<NaiveDateTime> {
|
|
v.as_str()
|
|
.ok_or_else(|| Error::from(EM::EntryHeaderTypeError))?
|
|
.parse::<NaiveDateTime>()
|
|
.context("Datetime parsing error")
|
|
.map_err(Error::from)
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use std::path::PathBuf;
|
|
|
|
use super::*;
|
|
|
|
use libimagstore::store::Store;
|
|
|
|
use chrono::naive::NaiveDateTime;
|
|
use chrono::naive::NaiveDate;
|
|
use chrono::naive::NaiveTime;
|
|
use toml_query::read::TomlValueReadExt;
|
|
|
|
pub fn get_store() -> Store {
|
|
Store::new_inmemory(PathBuf::from("/"), &None).unwrap()
|
|
}
|
|
|
|
#[test]
|
|
fn test_set_date() {
|
|
let store = get_store();
|
|
|
|
#[allow(clippy::zero_prefixed_literal)]
|
|
let date = {
|
|
let date = NaiveDate::from_ymd(2000, 01, 02);
|
|
let time = NaiveTime::from_hms(03, 04, 05);
|
|
|
|
NaiveDateTime::new(date, time)
|
|
};
|
|
|
|
let mut entry = store.create(PathBuf::from("test")).unwrap();
|
|
let res = entry.set_date(date);
|
|
|
|
assert!(res.is_ok(), format!("Error: {:?}", res));
|
|
let res = res.unwrap();
|
|
|
|
assert!(res.is_none()); // There shouldn't be an existing value
|
|
|
|
// Check whether the header is set correctly
|
|
|
|
let hdr_field = entry.get_header().read(&DATE_HEADER_LOCATION);
|
|
|
|
assert!(hdr_field.is_ok());
|
|
let hdr_field = hdr_field.unwrap();
|
|
|
|
assert!(hdr_field.is_some());
|
|
let hdr_field = hdr_field.unwrap();
|
|
|
|
match *hdr_field {
|
|
Value::String(ref s) => assert_eq!("2000-01-02T03:04:05", s),
|
|
_ => panic!("Wrong header type"),
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
#[allow(clippy::zero_prefixed_literal)]
|
|
fn test_read_date() {
|
|
use chrono::Datelike;
|
|
use chrono::Timelike;
|
|
|
|
let store = get_store();
|
|
|
|
let date = {
|
|
let date = NaiveDate::from_ymd(2000, 01, 02);
|
|
let time = NaiveTime::from_hms(03, 04, 05);
|
|
|
|
NaiveDateTime::new(date, time)
|
|
};
|
|
|
|
let mut entry = store.create(PathBuf::from("test")).unwrap();
|
|
let res = entry.set_date(date);
|
|
|
|
assert!(res.is_ok(), format!("Expected Ok(_), got: {:?}", res));
|
|
let res = res.unwrap();
|
|
|
|
assert!(res.is_none()); // There shouldn't be an existing value
|
|
|
|
// same as the test above ...
|
|
|
|
let d = entry.read_date();
|
|
|
|
assert!(d.is_ok(), format!("Expected Ok(_), got: {:?}", d));
|
|
let d = d.unwrap();
|
|
|
|
assert_eq!(d.date().year() , 2000);
|
|
assert_eq!(d.date().month() , 01);
|
|
assert_eq!(d.date().day() , 02);
|
|
assert_eq!(d.time().hour() , 03);
|
|
assert_eq!(d.time().minute() , 04);
|
|
assert_eq!(d.time().second() , 05);
|
|
}
|
|
|
|
#[test]
|
|
fn test_delete_date() {
|
|
let store = get_store();
|
|
|
|
#[allow(clippy::zero_prefixed_literal)]
|
|
let date = {
|
|
let date = NaiveDate::from_ymd(2000, 01, 02);
|
|
let time = NaiveTime::from_hms(03, 04, 05);
|
|
|
|
NaiveDateTime::new(date, time)
|
|
};
|
|
|
|
let mut entry = store.create(PathBuf::from("test")).unwrap();
|
|
let res = entry.set_date(date);
|
|
|
|
assert!(res.is_ok(), format!("Expected Ok(_), got: {:?}", res));
|
|
let res = res.unwrap();
|
|
assert!(res.is_none()); // There shouldn't be an existing value
|
|
|
|
assert!(entry.delete_date().is_ok());
|
|
|
|
let hdr_field = entry.get_header().read(&DATE_HEADER_LOCATION);
|
|
|
|
assert!(hdr_field.is_ok());
|
|
let hdr_field = hdr_field.unwrap();
|
|
|
|
assert!(hdr_field.is_none());
|
|
}
|
|
}
|
|
|