try to support nested cli command definitions

This commit is contained in:
Dietmar Maurer 2018-12-10 13:28:38 +01:00
parent 34d3ba52eb
commit 211fabd795
1 changed files with 79 additions and 35 deletions

View File

@ -17,24 +17,8 @@ fn print_cli_usage() {
eprintln!("Usage: TODO");
}
type CliCommandDefinition = HashMap<String, CliCommand>;
fn run_cli_command(def: &CliCommandDefinition) -> Result<(), Error> {
let mut args: Vec<String> = std::env::args().skip(1).collect();
if args.len() < 1 {
bail!("no command specified.");
}
let command = args.remove(0);
let cli_cmd = match def.get(&command) {
Some(cmd) => cmd,
None => {
bail!("no such command '{}'", command);
}
};
fn handle_simple_command(cli_cmd: &CliCommand, args: Vec<String>) -> Result<(), Error> {
let (params, rest) = getopts::parse_arguments(
&args, &cli_cmd.arg_param, &cli_cmd.info.parameters)?;
@ -50,35 +34,95 @@ fn run_cli_command(def: &CliCommandDefinition) -> Result<(), Error> {
Ok(())
}
fn handle_nested_command(def: &HashMap<String, CmdDef>, mut args: Vec<String>) -> Result<(), Error> {
if args.len() < 1 {
let mut cmds: Vec<&String> = def.keys().collect();
cmds.sort();
let list = cmds.iter().fold(String::new(),|mut s,item| {
if !s.is_empty() { s+= ", "; }
s += item;
s
});
bail!("expected command argument, but no command specified.\nPossible commands: {}", list);
}
let command = args.remove(0);
let sub_cmd = match def.get(&command) {
Some(cmd) => cmd,
None => {
bail!("no such command '{}'", command);
}
};
match sub_cmd {
CmdDef::Simple(cli_cmd) => {
handle_simple_command(cli_cmd, args)?;
}
CmdDef::Nested(map) => {
handle_nested_command(map, args)?;
}
}
Ok(())
}
fn run_cli_command(def: &CmdDef) -> Result<(), Error> {
let args: Vec<String> = std::env::args().skip(1).collect();
match def {
CmdDef::Simple(cli_cmd) => handle_simple_command(cli_cmd, args),
CmdDef::Nested(map) => handle_nested_command(map, args),
}
}
struct CliCommand {
info: ApiMethod,
arg_param: Vec<&'static str>,
fixed_param: Vec<&'static str>,
}
enum CmdDef {
Simple(CliCommand),
Nested(HashMap<String, CmdDef>),
}
fn datastore_commands() -> CmdDef {
let mut cmd_def = HashMap::new();
cmd_def.insert("list".to_owned(), CmdDef::Simple(CliCommand {
info: api3::config::datastore::get(),
arg_param: vec![],
fixed_param: vec![],
}));
cmd_def.insert("create".to_owned(), CmdDef::Simple(CliCommand {
info: api3::config::datastore::post(),
arg_param: vec!["name", "path"],
fixed_param: vec![],
}));
cmd_def.insert("remove".to_owned(), CmdDef::Simple(CliCommand {
info: api3::config::datastore::delete(),
arg_param: vec!["name"],
fixed_param: vec![],
}));
CmdDef::Nested(cmd_def)
}
fn main() {
let mut cmd_def = HashMap::new();
cmd_def.insert("list".to_owned(), CliCommand {
info: api3::config::datastore::get(),
arg_param: vec![],
fixed_param: vec![],
});
cmd_def.insert("datastore".to_owned(), datastore_commands());
cmd_def.insert("create".to_owned(), CliCommand {
info: api3::config::datastore::post(),
arg_param: vec!["name", "path"],
fixed_param: vec![],
});
cmd_def.insert("remove".to_owned(), CliCommand {
info: api3::config::datastore::delete(),
arg_param: vec!["name"],
fixed_param: vec![],
});
if let Err(err) = run_cli_command(&cmd_def) {
if let Err(err) = run_cli_command(&CmdDef::Nested(cmd_def)) {
eprintln!("Error: {}", err);
print_cli_usage();
std::process::exit(-1);