Add builder for todo entries

Signed-off-by: Matthias Beyer <mail@beyermatthias.de>
This commit is contained in:
Matthias Beyer 2019-11-04 20:34:55 +01:00
parent 7873d99df5
commit 5be95227c4
3 changed files with 151 additions and 34 deletions

View file

@ -0,0 +1,130 @@
//
// 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::NaiveDateTime;
use failure::Fallible as Result;
use failure::err_msg;
use toml_query::insert::TomlValueInsertExt;
use uuid::Uuid;
use libimagstore::store::Store;
use libimagstore::store::FileLockEntry;
use libimagentryutil::isa::Is;
use libimagutil::date::datetime_to_string;
use crate::priority::Priority;
use crate::status::Status;
use crate::entry::IsTodo;
use crate::entry::TodoHeader;
use crate::store::date_sanity_check;
pub struct TodoBuilder {
uuid: Option<Uuid>,
status: Option<Status>,
scheduled: Option<NaiveDateTime>,
hidden: Option<NaiveDateTime>,
due: Option<NaiveDateTime>,
prio: Option<Priority>,
check_sanity: bool,
}
impl TodoBuilder {
pub(crate) fn new() -> Self {
TodoBuilder {
uuid: None,
status: None,
scheduled: None,
hidden: None,
due: None,
prio: None,
check_sanity: true,
}
}
pub fn build<'a>(self, store: &'a Store) -> Result<FileLockEntry<'a>> {
let uuid = self.uuid.ok_or_else(|| err_msg("Uuid missing"))?;
let status = self.status.ok_or_else(|| err_msg("Status missing"))?;
if self.check_sanity {
trace!("Checking sanity before creating todo");
if let Err(s) = date_sanity_check(self.scheduled.as_ref(), self.hidden.as_ref(), self.due.as_ref()) {
trace!("Not sane.");
return Err(format_err!("{}", s))
}
}
let uuid_s = format!("{}", uuid.to_hyphenated_ref()); // TODO: not how it is supposed to be
debug!("Created new UUID for todo = {}", uuid_s);
let mut entry = crate::module_path::new_id(uuid_s).and_then(|id| store.create(id))?;
let header = TodoHeader {
uuid,
status,
scheduled: self.scheduled.as_ref().map(datetime_to_string),
hidden: self.hidden.as_ref().map(datetime_to_string),
due: self.due.as_ref().map(datetime_to_string),
priority: self.prio
};
debug!("Created header for todo: {:?}", header);
let _ = entry.get_header_mut().insert_serialized("todo", header)?;
let _ = entry.set_isflag::<IsTodo>()?;
Ok(entry)
}
pub fn with_uuid(mut self, uuid: Option<Uuid>) -> Self {
self.uuid = uuid;
self
}
pub fn with_status(mut self, status: Option<Status>) -> Self {
self.status = status;
self
}
pub fn with_scheduled(mut self, scheduled: Option<NaiveDateTime>) -> Self {
self.scheduled = scheduled;
self
}
pub fn with_hidden(mut self, hidden: Option<NaiveDateTime>) -> Self {
self.hidden = hidden;
self
}
pub fn with_due(mut self, due: Option<NaiveDateTime>) -> Self {
self.due = due;
self
}
pub fn with_prio(mut self, prio: Option<Priority>) -> Self {
self.prio = prio;
self
}
pub fn with_check_sanity(mut self, b: bool) -> Self {
self.check_sanity = b;
self
}
}

View file

@ -34,6 +34,7 @@ extern crate libimagutil;
#[macro_use] extern crate libimagstore; #[macro_use] extern crate libimagstore;
#[macro_use] extern crate libimagentryutil; #[macro_use] extern crate libimagentryutil;
pub mod builder;
pub mod entry; pub mod entry;
pub mod iter; pub mod iter;
pub mod priority; pub mod priority;

View file

@ -22,20 +22,19 @@ use std::result::Result as RResult;
use failure::Fallible as Result; use failure::Fallible as Result;
use chrono::NaiveDateTime; use chrono::NaiveDateTime;
use uuid::Uuid; use uuid::Uuid;
use toml_query::insert::TomlValueInsertExt;
use libimagstore::store::FileLockEntry; use libimagstore::store::FileLockEntry;
use libimagstore::store::Store; use libimagstore::store::Store;
use libimagstore::iter::Entries; use libimagstore::iter::Entries;
use libimagutil::date::datetime_to_string;
use libimagentryutil::isa::Is;
use crate::status::Status; use crate::status::Status;
use crate::priority::Priority; use crate::priority::Priority;
use crate::entry::TodoHeader; use crate::builder::TodoBuilder;
use crate::entry::IsTodo;
pub trait TodoStore<'a> { pub trait TodoStore<'a> {
fn todo_builder(&self) -> TodoBuilder;
fn create_todo(&'a self, fn create_todo(&'a self,
status: Status, status: Status,
scheduled: Option<NaiveDateTime>, scheduled: Option<NaiveDateTime>,
@ -51,6 +50,13 @@ pub trait TodoStore<'a> {
impl<'a> TodoStore<'a> for Store { impl<'a> TodoStore<'a> for Store {
/// Get a TodoBuilder instance, which can be used to build a todo object.
///
/// The TodoBuilder::new() constructor is not exposed, this function should be used instead.
fn todo_builder(&self) -> TodoBuilder {
TodoBuilder::new()
}
/// Create a new todo entry /// Create a new todo entry
/// ///
/// # Warning /// # Warning
@ -70,35 +76,15 @@ impl<'a> TodoStore<'a> for Store {
prio: Option<Priority>, prio: Option<Priority>,
check_sanity: bool) -> Result<FileLockEntry<'a>> check_sanity: bool) -> Result<FileLockEntry<'a>>
{ {
if check_sanity { TodoBuilder::new()
trace!("Checking sanity before creating todo"); .with_status(Some(status))
if let Err(s) = date_sanity_check(scheduled.as_ref(), hidden.as_ref(), due.as_ref()) { .with_uuid(Some(Uuid::new_v4()))
trace!("Not sane."); .with_scheduled(scheduled)
return Err(format_err!("{}", s)) .with_hidden(hidden)
} .with_due(due)
} .with_prio(prio)
.with_check_sanity(check_sanity)
let uuid = Uuid::new_v4(); .build(&self)
let uuid_s = format!("{}", uuid.to_hyphenated_ref()); // TODO: not how it is supposed to be
debug!("Created new UUID for todo = {}", uuid_s);
let mut entry = crate::module_path::new_id(uuid_s).and_then(|id| self.create(id))?;
let header = TodoHeader {
uuid,
status,
scheduled: scheduled.as_ref().map(datetime_to_string),
hidden: hidden.as_ref().map(datetime_to_string),
due: due.as_ref().map(datetime_to_string),
priority: prio
};
debug!("Created header for todo: {:?}", header);
let _ = entry.get_header_mut().insert_serialized("todo", header)?;
let _ = entry.set_isflag::<IsTodo>()?;
Ok(entry)
} }
fn get_todo_by_uuid(&'a self, uuid: &Uuid) -> Result<Option<FileLockEntry<'a>>> { fn get_todo_by_uuid(&'a self, uuid: &Uuid) -> Result<Option<FileLockEntry<'a>>> {