imag-notes: Do not call exit() but propagate error to main function

Signed-off-by: Matthias Beyer <mail@beyermatthias.de>
This commit is contained in:
Matthias Beyer 2019-11-15 15:27:50 +01:00
parent 68befe23c6
commit c2d4ec5fef
2 changed files with 46 additions and 73 deletions

View file

@ -23,6 +23,7 @@ maintenance = { status = "actively-developed" }
log = "0.4.6" log = "0.4.6"
itertools = "0.8.0" itertools = "0.8.0"
failure = "0.1.5" failure = "0.1.5"
resiter = "0.4.0"
libimagrt = { version = "0.10.0", path = "../../../lib/core/libimagrt" } libimagrt = { version = "0.10.0", path = "../../../lib/core/libimagrt" }
libimagerror = { version = "0.10.0", path = "../../../lib/core/libimagerror" } libimagerror = { version = "0.10.0", path = "../../../lib/core/libimagerror" }

View file

@ -37,7 +37,8 @@
extern crate clap; extern crate clap;
#[macro_use] extern crate log; #[macro_use] extern crate log;
extern crate itertools; extern crate itertools;
extern crate failure; #[macro_use] extern crate failure;
extern crate resiter;
extern crate libimagnotes; extern crate libimagnotes;
extern crate libimagrt; extern crate libimagrt;
@ -47,11 +48,13 @@ extern crate libimagutil;
extern crate libimagstore; extern crate libimagstore;
use std::io::Write; use std::io::Write;
use std::process::exit;
use itertools::Itertools; use itertools::Itertools;
use clap::App; use clap::App;
use failure::Error;
use failure::err_msg;
use failure::Fallible as Result; use failure::Fallible as Result;
use resiter::IterInnerOkOrElse;
use libimagentryedit::edit::Edit; use libimagentryedit::edit::Edit;
use libimagrt::runtime::Runtime; use libimagrt::runtime::Runtime;
@ -59,11 +62,6 @@ use libimagrt::application::ImagApplication;
use libimagstore::iter::get::StoreIdGetIteratorExtension; use libimagstore::iter::get::StoreIdGetIteratorExtension;
use libimagnotes::note::Note; use libimagnotes::note::Note;
use libimagnotes::notestore::*; use libimagnotes::notestore::*;
use libimagerror::trace::MapErrTrace;
use libimagerror::exit::ExitUnwrap;
use libimagerror::io::ToExitCode;
use libimagerror::iter::TraceIterator;
use libimagutil::info_result::*;
use libimagutil::warn_result::WarnResult; use libimagutil::warn_result::WarnResult;
@ -76,25 +74,20 @@ mod ui;
pub enum ImagNotes {} pub enum ImagNotes {}
impl ImagApplication for ImagNotes { impl ImagApplication for ImagNotes {
fn run(rt: Runtime) -> Result<()> { fn run(rt: Runtime) -> Result<()> {
if let Some(name) = rt.cli().subcommand_name() { match rt.cli().subcommand_name().ok_or_else(|| err_msg("No subcommand called"))? {
"create" => create(&rt),
debug!("Call: {}", name); "delete" => delete(&rt),
match name { "edit" => edit(&rt),
"create" => create(&rt), "list" => list(&rt),
"delete" => delete(&rt), other => {
"edit" => edit(&rt), debug!("Unknown command");
"list" => list(&rt), if rt.handle_unknown_subcommand("imag-notes", other, rt.cli())?.success() {
other => { Ok(())
debug!("Unknown command"); } else {
let _ = rt.handle_unknown_subcommand("imag-notes", other, rt.cli()) Err(err_msg("Failed to handle unknown subcommand"))
.map_err_trace_exit_unwrap() }
.code() },
.map(::std::process::exit);
},
};
} }
Ok(())
} }
fn build_cli<'a>(app: App<'a, 'a>) -> App<'a, 'a> { fn build_cli<'a>(app: App<'a, 'a>) -> App<'a, 'a> {
@ -118,74 +111,53 @@ fn name_from_cli(rt: &Runtime, subcmd: &str) -> String {
rt.cli().subcommand_matches(subcmd).unwrap().value_of("name").map(String::from).unwrap() rt.cli().subcommand_matches(subcmd).unwrap().value_of("name").map(String::from).unwrap()
} }
fn create(rt: &Runtime) { fn create(rt: &Runtime) -> Result<()> {
let name = name_from_cli(rt, "create"); let name = name_from_cli(rt, "create");
let mut note = rt let mut note = rt.store().new_note(name.clone(), String::new())?;
.store()
.new_note(name.clone(), String::new())
.map_err_trace_exit_unwrap();
if rt.cli().subcommand_matches("create").unwrap().is_present("edit") { if rt.cli().subcommand_matches("create").unwrap().is_present("edit") {
note note.edit_content(rt)?
.edit_content(rt)
.map_warn_err_str("Editing failed")
.map_err_trace_exit_unwrap();
} }
rt.report_touched(note.get_location()).unwrap_or_exit(); rt.report_touched(note.get_location()).map_err(Error::from)
} }
fn delete(rt: &Runtime) { fn delete(rt: &Runtime) -> Result<()> {
rt.store() rt.store().delete_note(name_from_cli(rt, "delete")).map(|_| ())
.delete_note(name_from_cli(rt, "delete"))
.map_info_str("Ok")
.map_err_trace_exit_unwrap();
} }
fn edit(rt: &Runtime) { fn edit(rt: &Runtime) -> Result<()> {
let name = name_from_cli(rt, "edit"); let name = name_from_cli(rt, "edit");
rt rt
.store() .store()
.get_note(name.clone()) .get_note(name.clone())?
.map_err_trace_exit_unwrap() .ok_or_else(|| format_err!("Name '{}' not found", name))
.map(|mut note| { .and_then(|mut note| {
note note.edit_content(rt).map_warn_err_str("Editing failed")?;
.edit_content(rt) rt.report_touched(note.get_location()).map_err(Error::from)
.map_warn_err_str("Editing failed")
.map_err_trace_exit_unwrap();
rt.report_touched(note.get_location()).unwrap_or_exit();
}) })
.unwrap_or_else(|| {
error!("Cannot find note with name '{}'", name);
});
} }
fn list(rt: &Runtime) { fn list(rt: &Runtime) -> Result<()> {
use std::cmp::Ordering; use std::cmp::Ordering;
rt rt
.store() .store()
.all_notes() .all_notes()?
.map_err_trace_exit_unwrap()
.into_get_iter(rt.store()) .into_get_iter(rt.store())
.trace_unwrap_exit() .map_inner_ok_or_else(|| err_msg("Did not find one entry"))
.map(|opt| opt.unwrap_or_else(|| { .collect::<Result<Vec<_>>>()?
error!("Fatal: Nonexistent entry where entry should exist"); .into_iter()
exit(1) .sorted_by(|a, b| match (a.get_name(), b.get_name()) {
})) (Ok(a), Ok(b)) => a.cmp(&b),
.sorted_by(|note_a, note_b| if let (Ok(a), Ok(b)) = (note_a.get_name(), note_b.get_name()) { _ => Ordering::Greater,
a.cmp(&b)
} else {
Ordering::Greater
}) })
.for_each(|note| { .map(|note| {
let name = note.get_name().map_err_trace_exit_unwrap(); let name = note.get_name().map_err(Error::from)?;
writeln!(rt.stdout(), "{}", name) writeln!(rt.stdout(), "{}", name)?;
.to_exit_code() rt.report_touched(note.get_location()).map_err(Error::from)
.unwrap_or_exit(); })
.collect::<Result<Vec<_>>>()
rt.report_touched(note.get_location()).unwrap_or_exit(); .map(|_| ())
});
} }