Merge branch 'imag-todo-taskwarrior-import' into master
Signed-off-by: Matthias Beyer <mail@beyermatthias.de>
This commit is contained in:
commit
99fedd5abd
7 changed files with 362 additions and 34 deletions
|
@ -43,6 +43,26 @@ version = "2.33.0"
|
||||||
default-features = false
|
default-features = false
|
||||||
features = ["color", "suggestions", "wrap_help"]
|
features = ["color", "suggestions", "wrap_help"]
|
||||||
|
|
||||||
|
[dependencies.task-hookrs]
|
||||||
|
version = "0.7.0"
|
||||||
|
optional = true
|
||||||
|
|
||||||
|
[dependencies.uuid]
|
||||||
|
version = "0.7.4"
|
||||||
|
features = ["v4"]
|
||||||
|
optional = true
|
||||||
|
|
||||||
|
[dependencies.libimagentrytag]
|
||||||
|
version = "0.10.0"
|
||||||
|
path = "../../../lib/entry/libimagentrytag"
|
||||||
|
optional = true
|
||||||
|
|
||||||
|
[dependencies.libimagentrylink]
|
||||||
|
version = "0.10.0"
|
||||||
|
path = "../../../lib/entry/libimagentrylink"
|
||||||
|
optional = true
|
||||||
|
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
name = "libimagtodofrontend"
|
name = "libimagtodofrontend"
|
||||||
path = "src/lib.rs"
|
path = "src/lib.rs"
|
||||||
|
@ -50,3 +70,8 @@ path = "src/lib.rs"
|
||||||
[[bin]]
|
[[bin]]
|
||||||
name = "imag-todo"
|
name = "imag-todo"
|
||||||
path = "src/bin.rs"
|
path = "src/bin.rs"
|
||||||
|
|
||||||
|
[features]
|
||||||
|
default = []
|
||||||
|
import-taskwarrior = [ "task-hookrs", "uuid", "libimagentrytag", "libimagentrylink" ]
|
||||||
|
|
||||||
|
|
162
bin/domain/imag-todo/src/import.rs
Normal file
162
bin/domain/imag-todo/src/import.rs
Normal file
|
@ -0,0 +1,162 @@
|
||||||
|
//
|
||||||
|
// 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 failure::Fallible as Result;
|
||||||
|
use failure::err_msg;
|
||||||
|
|
||||||
|
use libimagrt::runtime::Runtime;
|
||||||
|
|
||||||
|
pub fn import(rt: &Runtime) -> Result<()> {
|
||||||
|
let scmd = rt.cli().subcommand().1.unwrap();
|
||||||
|
|
||||||
|
match scmd.subcommand_name() {
|
||||||
|
None => Err(err_msg("No subcommand called")),
|
||||||
|
Some("taskwarrior") => import_taskwarrior(rt),
|
||||||
|
Some(other) => {
|
||||||
|
debug!("Unknown command");
|
||||||
|
if rt.handle_unknown_subcommand("imag-todo-import", other, rt.cli())?.success() {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(err_msg("Failed to handle unknown subcommand"))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(unused_variables)]
|
||||||
|
fn import_taskwarrior(rt: &Runtime) -> Result<()> {
|
||||||
|
#[cfg(not(feature = "import-taskwarrior"))]
|
||||||
|
{
|
||||||
|
Err(err_msg("Binary not compiled with taskwarrior import functionality"))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(feature = "import-taskwarrior")]
|
||||||
|
{
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use std::ops::Deref;
|
||||||
|
|
||||||
|
use uuid::Uuid;
|
||||||
|
|
||||||
|
use libimagtodo::status::Status;
|
||||||
|
use libimagtodo::priority::Priority;
|
||||||
|
use libimagtodo::store::TodoStore;
|
||||||
|
use libimagentrytag::tagable::Tagable;
|
||||||
|
use libimagentrylink::linkable::Linkable;
|
||||||
|
|
||||||
|
use task_hookrs::import::import as taskwarrior_import;
|
||||||
|
use task_hookrs::priority::TaskPriority;
|
||||||
|
use task_hookrs::status::TaskStatus;
|
||||||
|
|
||||||
|
let store = rt.store();
|
||||||
|
if !rt.input_is_pipe() {
|
||||||
|
return Err(err_msg("Cannot get stdin for importing tasks"))
|
||||||
|
}
|
||||||
|
let stdin = ::std::io::stdin();
|
||||||
|
|
||||||
|
let translate_status = |twstatus: &TaskStatus| -> Option<Status> {
|
||||||
|
match *twstatus {
|
||||||
|
TaskStatus::Pending => Some(Status::Pending),
|
||||||
|
TaskStatus::Completed => Some(Status::Done),
|
||||||
|
TaskStatus::Deleted => Some(Status::Deleted),
|
||||||
|
_ => Some(Status::Deleted), // default to deleted if taskwarrior data does not have a status
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let translate_prio = |p: &TaskPriority| -> Priority {
|
||||||
|
match p {
|
||||||
|
TaskPriority::Low => Priority::Low,
|
||||||
|
TaskPriority::Medium => Priority::Medium,
|
||||||
|
TaskPriority::High => Priority::High,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
taskwarrior_import(stdin)?
|
||||||
|
.into_iter()
|
||||||
|
.map(|task| {
|
||||||
|
let hash = task.uuid().clone();
|
||||||
|
let mut todo = store
|
||||||
|
.todo_builder()
|
||||||
|
.with_check_sanity(false) // data should be imported, even if it is not sane
|
||||||
|
.with_status(translate_status(task.status()))
|
||||||
|
.with_uuid(Some(task.uuid().clone()))
|
||||||
|
.with_due(task.due().map(Deref::deref).cloned())
|
||||||
|
.with_scheduled(task.scheduled().map(Deref::deref).cloned())
|
||||||
|
.with_hidden(task.wait().map(Deref::deref).cloned())
|
||||||
|
.with_prio(task.priority().map(|p| translate_prio(p)))
|
||||||
|
.build(rt.store())?;
|
||||||
|
|
||||||
|
todo.set_content(task.description().clone());
|
||||||
|
|
||||||
|
if let Some(tags) = task.tags() {
|
||||||
|
tags.into_iter().map(|tag| {
|
||||||
|
let tag = tag.clone();
|
||||||
|
if libimagentrytag::tag::is_tag_str(&tag).is_err() {
|
||||||
|
warn!("Not a valid tag, ignoring: {}", tag);
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
todo.add_tag(tag)
|
||||||
|
}
|
||||||
|
}).collect::<Result<Vec<_>>>()?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(annos) = task.annotations() {
|
||||||
|
// We do not import annotations as imag annotations, but add them as text to
|
||||||
|
// the entry, which is more sane IMO.
|
||||||
|
//
|
||||||
|
// this could be changed into a configurable thing later.
|
||||||
|
let anno = annos.iter()
|
||||||
|
.map(|anno| anno.description())
|
||||||
|
.map(String::clone)
|
||||||
|
.collect::<Vec<String>>()
|
||||||
|
.join("\n");
|
||||||
|
todo.get_content_mut().push('\n');
|
||||||
|
todo.get_content_mut().push_str(&anno);
|
||||||
|
}
|
||||||
|
|
||||||
|
let dependends = task.depends().cloned().unwrap_or_else(|| vec![]);
|
||||||
|
Ok((hash, dependends))
|
||||||
|
})
|
||||||
|
.collect::<Result<HashMap<Uuid, Vec<Uuid>>>>()?
|
||||||
|
|
||||||
|
//
|
||||||
|
// We actually _have_ to collect here, because we must ensure that all imported Todo
|
||||||
|
// entries are in the store before we can continue and link them together (which is
|
||||||
|
// what happens next)
|
||||||
|
//
|
||||||
|
|
||||||
|
.iter()
|
||||||
|
.filter(|(_, list)| !list.is_empty())
|
||||||
|
.map(|(key, list)| {
|
||||||
|
let mut entry = store.get_todo_by_uuid(key)?.ok_or_else(|| {
|
||||||
|
format_err!("Cannot find todo by UUID: {}", key)
|
||||||
|
})?;
|
||||||
|
|
||||||
|
list.iter()
|
||||||
|
.map(move |element| {
|
||||||
|
store.get_todo_by_uuid(element)?
|
||||||
|
.ok_or_else(|| {
|
||||||
|
format_err!("Cannot find todo by UUID: {}", key)
|
||||||
|
})
|
||||||
|
.and_then(|mut target| entry.add_link(&mut target))
|
||||||
|
})
|
||||||
|
.collect::<Result<_>>()
|
||||||
|
})
|
||||||
|
.collect::<Result<_>>()
|
||||||
|
}
|
||||||
|
}
|
|
@ -44,6 +44,18 @@ extern crate kairos;
|
||||||
#[macro_use] extern crate failure;
|
#[macro_use] extern crate failure;
|
||||||
extern crate resiter;
|
extern crate resiter;
|
||||||
|
|
||||||
|
#[cfg(feature = "import-taskwarrior")]
|
||||||
|
extern crate task_hookrs;
|
||||||
|
|
||||||
|
#[cfg(feature = "import-taskwarrior")]
|
||||||
|
extern crate uuid;
|
||||||
|
|
||||||
|
#[cfg(feature = "import-taskwarrior")]
|
||||||
|
extern crate libimagentrytag;
|
||||||
|
|
||||||
|
#[cfg(feature = "import-taskwarrior")]
|
||||||
|
extern crate libimagentrylink;
|
||||||
|
|
||||||
extern crate libimagrt;
|
extern crate libimagrt;
|
||||||
extern crate libimagstore;
|
extern crate libimagstore;
|
||||||
extern crate libimagerror;
|
extern crate libimagerror;
|
||||||
|
@ -79,6 +91,7 @@ use libimagtodo::store::TodoStore;
|
||||||
use libimagutil::date::datetime_to_string;
|
use libimagutil::date::datetime_to_string;
|
||||||
|
|
||||||
mod ui;
|
mod ui;
|
||||||
|
mod import;
|
||||||
|
|
||||||
/// Marker enum for implementing ImagApplication on
|
/// Marker enum for implementing ImagApplication on
|
||||||
///
|
///
|
||||||
|
@ -93,6 +106,7 @@ impl ImagApplication for ImagTodo {
|
||||||
Some("mark") => mark(&rt),
|
Some("mark") => mark(&rt),
|
||||||
Some("pending") | None => list_todos(&rt, &StatusMatcher::new().is(Status::Pending), false),
|
Some("pending") | None => list_todos(&rt, &StatusMatcher::new().is(Status::Pending), false),
|
||||||
Some("list") => list(&rt),
|
Some("list") => list(&rt),
|
||||||
|
Some("import") => import::import(&rt),
|
||||||
Some(other) => {
|
Some(other) => {
|
||||||
debug!("Unknown command");
|
debug!("Unknown command");
|
||||||
if rt.handle_unknown_subcommand("imag-todo", other, rt.cli())?.success() {
|
if rt.handle_unknown_subcommand("imag-todo", other, rt.cli())?.success() {
|
||||||
|
|
|
@ -178,6 +178,16 @@ pub fn build_ui<'a>(app: App<'a, 'a>) -> App<'a, 'a> {
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
.subcommand(SubCommand::with_name("import")
|
||||||
|
.about("Import todos from other tool")
|
||||||
|
.version("0.1")
|
||||||
|
.subcommand(SubCommand::with_name("taskwarrior")
|
||||||
|
.about("Import from taskwarrior by piping 'task export' to this subcommand.")
|
||||||
|
.version("0.1")
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct PathProvider;
|
pub struct PathProvider;
|
||||||
|
|
130
lib/domain/libimagtodo/src/builder.rs
Normal file
130
lib/domain/libimagtodo/src/builder.rs
Normal 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
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -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;
|
||||||
|
|
|
@ -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>>> {
|
||||||
|
|
Loading…
Reference in a new issue