From 4968bc3ac0d006069061aef65aaf8b62f6a6e377 Mon Sep 17 00:00:00 2001 From: Wolfgang Bumiller Date: Thu, 3 Jan 2019 14:36:31 +0100 Subject: [PATCH] cli::command: wrap usage errors in a UsageError So we can distinguish them and show usage output conditionally. Signed-off-by: Wolfgang Bumiller --- src/bin/backup-client.rs | 4 +++- src/cli/command.rs | 38 +++++++++++++++++++++++++------------- 2 files changed, 28 insertions(+), 14 deletions(-) diff --git a/src/bin/backup-client.rs b/src/bin/backup-client.rs index c18e78ae..05d286ff 100644 --- a/src/bin/backup-client.rs +++ b/src/bin/backup-client.rs @@ -146,7 +146,9 @@ fn main() { if let Err(err) = run_cli_command(&cmd_def.into()) { eprintln!("Error: {}", err); - print_cli_usage(); + if err.downcast::().is_ok() { + print_cli_usage(); + } std::process::exit(-1); } diff --git a/src/cli/command.rs b/src/cli/command.rs index b808d16f..e76ee0cf 100644 --- a/src/cli/command.rs +++ b/src/cli/command.rs @@ -2,6 +2,8 @@ use failure::*; use std::collections::HashMap; use std::collections::HashSet; +use serde_json::Value; + use crate::api::schema::*; use crate::api::router::*; //use crate::api::config::*; @@ -12,7 +14,13 @@ pub fn print_cli_usage() { eprintln!("Usage: TODO"); } -fn handle_simple_command(cli_cmd: &CliCommand, args: Vec) -> Result<(), Error> { +#[derive(Debug, Fail)] +#[fail(display = "Usage error: {}", _0)] +pub struct UsageError(Error); + +pub struct Invocation<'a>(&'a CliCommand, Value); + +fn handle_simple_command(cli_cmd: &CliCommand, args: Vec) -> Result { let (params, rest) = getopts::parse_arguments( &args, &cli_cmd.arg_param, &cli_cmd.info.parameters)?; @@ -21,11 +29,7 @@ fn handle_simple_command(cli_cmd: &CliCommand, args: Vec) -> Result<(), bail!("got additional arguments: {:?}", rest); } - let res = (cli_cmd.info.handler)(params, &cli_cmd.info)?; - - println!("Result: {}", serde_json::to_string_pretty(&res).unwrap()); - - Ok(()) + Ok(Invocation(cli_cmd, params)) } fn find_command<'a>(def: &'a CliCommandMap, name: &str) -> Option<&'a CommandLineInterface> { @@ -50,7 +54,7 @@ fn find_command<'a>(def: &'a CliCommandMap, name: &str) -> Option<&'a CommandLin None } -fn handle_nested_command(def: &CliCommandMap, mut args: Vec) -> Result<(), Error> { +fn handle_nested_command(def: &CliCommandMap, mut args: Vec) -> Result { if args.len() < 1 { let mut cmds: Vec<&String> = def.commands.keys().collect(); @@ -74,14 +78,12 @@ fn handle_nested_command(def: &CliCommandMap, mut args: Vec) -> Result<( match sub_cmd { CommandLineInterface::Simple(cli_cmd) => { - handle_simple_command(cli_cmd, args)?; + handle_simple_command(cli_cmd, args) } CommandLineInterface::Nested(map) => { - handle_nested_command(map, args)?; + handle_nested_command(map, args) } } - - Ok(()) } fn print_property_completion( @@ -253,10 +255,20 @@ pub fn run_cli_command(def: &CommandLineInterface) -> Result<(), Error> { return Ok(()); } - match def { + let invocation = match def { CommandLineInterface::Simple(cli_cmd) => handle_simple_command(cli_cmd, args), CommandLineInterface::Nested(map) => handle_nested_command(map, args), - } + }; + + + let res = match invocation { + Err(e) => return Err(UsageError(e).into()), + Ok(invocation) => (invocation.0.info.handler)(invocation.1, &invocation.0.info)?, + }; + + println!("Result: {}", serde_json::to_string_pretty(&res).unwrap()); + + Ok(()) } pub type CompletionFunction = fn() -> Vec;