2018-12-10 12:36:52 +00:00
|
|
|
use failure::*;
|
|
|
|
use std::collections::HashMap;
|
2018-12-12 06:51:43 +00:00
|
|
|
use std::collections::HashSet;
|
2018-12-10 12:36:52 +00:00
|
|
|
|
|
|
|
use crate::api::schema::*;
|
|
|
|
use crate::api::router::*;
|
|
|
|
use crate::api::config::*;
|
|
|
|
use crate::getopts;
|
|
|
|
|
|
|
|
pub fn print_cli_usage() {
|
|
|
|
|
|
|
|
eprintln!("Usage: TODO");
|
|
|
|
}
|
|
|
|
|
|
|
|
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)?;
|
|
|
|
|
|
|
|
if !rest.is_empty() {
|
|
|
|
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(())
|
|
|
|
}
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
fn find_command<'a>(def: &'a CliCommandMap, name: &str) -> Option<&'a CommandLineInterface> {
|
2018-12-10 17:13:55 +00:00
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
if let Some(sub_cmd) = def.commands.get(name) {
|
2018-12-10 17:13:55 +00:00
|
|
|
return Some(sub_cmd);
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut matches: Vec<&str> = vec![];
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
for cmd in def.commands.keys() {
|
2018-12-10 17:13:55 +00:00
|
|
|
if cmd.starts_with(name) {
|
|
|
|
matches.push(cmd); }
|
|
|
|
}
|
|
|
|
|
|
|
|
if matches.len() != 1 { return None; }
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
if let Some(sub_cmd) = def.commands.get(matches[0]) {
|
2018-12-10 17:13:55 +00:00
|
|
|
return Some(sub_cmd);
|
|
|
|
};
|
|
|
|
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
fn handle_nested_command(def: &CliCommandMap, mut args: Vec<String>) -> Result<(), Error> {
|
2018-12-10 12:36:52 +00:00
|
|
|
|
|
|
|
if args.len() < 1 {
|
2018-12-11 10:31:36 +00:00
|
|
|
let mut cmds: Vec<&String> = def.commands.keys().collect();
|
2018-12-10 12:36:52 +00:00
|
|
|
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);
|
|
|
|
|
2018-12-10 17:13:55 +00:00
|
|
|
let sub_cmd = match find_command(def, &command) {
|
2018-12-10 12:36:52 +00:00
|
|
|
Some(cmd) => cmd,
|
2018-12-10 17:13:55 +00:00
|
|
|
None => bail!("no such command '{}'", command),
|
2018-12-10 12:36:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
match sub_cmd {
|
2018-12-10 12:40:10 +00:00
|
|
|
CommandLineInterface::Simple(cli_cmd) => {
|
2018-12-10 12:36:52 +00:00
|
|
|
handle_simple_command(cli_cmd, args)?;
|
|
|
|
}
|
2018-12-10 12:40:10 +00:00
|
|
|
CommandLineInterface::Nested(map) => {
|
2018-12-10 12:36:52 +00:00
|
|
|
handle_nested_command(map, args)?;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
fn print_property_completion(schema: &Schema, arg: &str) {
|
2018-12-12 07:33:50 +00:00
|
|
|
// fixme: implement completion functions
|
|
|
|
if let Schema::String(StringSchema { format: Some(format), ..} ) = schema {
|
|
|
|
if let ApiStringFormat::Enum(list) = format.as_ref() {
|
|
|
|
for value in list {
|
|
|
|
if value.starts_with(arg) {
|
|
|
|
println!("{}", value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2018-12-12 06:51:43 +00:00
|
|
|
println!("");
|
|
|
|
}
|
2018-12-11 13:21:05 +00:00
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
fn record_done_arguments(done: &mut HashSet<String>, parameters: &ObjectSchema, list: &[String]) {
|
|
|
|
|
|
|
|
for arg in list {
|
|
|
|
if arg.starts_with("--") && arg.len() > 2 {
|
|
|
|
let prop_name = arg[2..].to_owned();
|
|
|
|
if let Some((_, schema)) = parameters.properties.get::<str>(&prop_name) {
|
|
|
|
match schema.as_ref() {
|
|
|
|
Schema::Array(_) => { /* do nothing */ }
|
|
|
|
_ => { done.insert(prop_name); }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_simple_completion(
|
|
|
|
cli_cmd: &CliCommand,
|
2018-12-12 09:37:03 +00:00
|
|
|
done: &mut HashSet<String>,
|
2018-12-12 06:51:43 +00:00
|
|
|
arg_param: &[&str],
|
|
|
|
mut args: Vec<String>,
|
|
|
|
) {
|
|
|
|
// fixme: arg_param, fixed_param
|
|
|
|
//eprintln!("COMPL: {:?} {:?} {}", arg_param, args, args.len());
|
|
|
|
|
|
|
|
if !arg_param.is_empty() {
|
|
|
|
let prop_name = arg_param[0];
|
|
|
|
done.insert(prop_name.into());
|
|
|
|
if args.len() > 1 {
|
|
|
|
args.remove(0);
|
|
|
|
print_simple_completion(cli_cmd, done, &arg_param[1..], args);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if let Some((_, schema)) = cli_cmd.info.parameters.properties.get(prop_name) {
|
2018-12-11 13:21:05 +00:00
|
|
|
if args.is_empty() {
|
2018-12-12 06:51:43 +00:00
|
|
|
print_property_completion(schema, "");
|
|
|
|
} else {
|
|
|
|
print_property_completion(schema, &args[0]);
|
2018-12-11 13:21:05 +00:00
|
|
|
}
|
2018-12-12 06:51:43 +00:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if args.is_empty() { return; }
|
|
|
|
|
2018-12-12 09:37:03 +00:00
|
|
|
record_done_arguments(done, &cli_cmd.info.parameters, &args);
|
2018-12-12 07:18:38 +00:00
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
let prefix = args.pop().unwrap(); // match on last arg
|
|
|
|
|
2018-12-12 07:18:38 +00:00
|
|
|
// complete option-name or option-value ?
|
|
|
|
if !prefix.starts_with("-") && args.len() > 0 {
|
|
|
|
let last = &args[args.len()-1];
|
|
|
|
if last.starts_with("--") && last.len() > 2 {
|
2018-12-12 09:37:03 +00:00
|
|
|
let prop_name = &last[2..];
|
|
|
|
if let Some((_, schema)) = cli_cmd.info.parameters.properties.get(prop_name) {
|
2018-12-12 07:18:38 +00:00
|
|
|
print_property_completion(schema, &prefix);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2018-12-12 06:51:43 +00:00
|
|
|
|
|
|
|
for (name, (optional, schema)) in &cli_cmd.info.parameters.properties {
|
|
|
|
if done.contains(*name) { continue; }
|
2018-12-12 07:18:38 +00:00
|
|
|
let option = String::from("--") + name;
|
|
|
|
if option.starts_with(&prefix) {
|
|
|
|
println!("{}", option);
|
2018-12-12 06:51:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn print_nested_completion(def: &CommandLineInterface, mut args: Vec<String>) {
|
|
|
|
|
|
|
|
match def {
|
|
|
|
CommandLineInterface::Simple(cli_cmd) => {
|
|
|
|
let mut done = HashSet::new();
|
|
|
|
let fixed: Vec<String> = cli_cmd.fixed_param.iter().map(|s| s.to_string()).collect();
|
|
|
|
record_done_arguments(&mut done, &cli_cmd.info.parameters, &fixed);
|
|
|
|
print_simple_completion(cli_cmd, &mut done, &cli_cmd.arg_param, args);
|
2018-12-11 13:21:05 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
CommandLineInterface::Nested(map) => {
|
|
|
|
if args.is_empty() {
|
|
|
|
for cmd in map.commands.keys() {
|
|
|
|
println!("{}", cmd);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let first = args.remove(0);
|
|
|
|
if let Some(sub_cmd) = map.commands.get(&first) {
|
2018-12-12 06:51:43 +00:00
|
|
|
print_nested_completion(sub_cmd, args);
|
2018-12-11 13:21:05 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
for cmd in map.commands.keys() {
|
|
|
|
if cmd.starts_with(&first) {
|
|
|
|
println!("{}", cmd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn print_bash_completion(def: &CommandLineInterface) {
|
|
|
|
|
|
|
|
let comp_point: usize = match std::env::var("COMP_POINT") {
|
|
|
|
Ok(val) => {
|
|
|
|
match usize::from_str_radix(&val, 10) {
|
|
|
|
Ok(i) => i,
|
|
|
|
Err(e) => return,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
let cmdline = match std::env::var("COMP_LINE") {
|
|
|
|
Ok(val) => val[0..comp_point].to_owned(),
|
|
|
|
Err(e) => return,
|
|
|
|
};
|
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
|
2018-12-11 13:21:05 +00:00
|
|
|
let mut args = match shellwords::split(&cmdline) {
|
|
|
|
Ok(v) => v,
|
|
|
|
Err(_) => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
args.remove(0); //no need for program name
|
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
if cmdline.ends_with(char::is_whitespace) {
|
|
|
|
//eprintln!("CMDLINE {:?}", cmdline);
|
|
|
|
args.push("".into());
|
|
|
|
}
|
|
|
|
|
2018-12-11 13:21:05 +00:00
|
|
|
//eprintln!("COMP_ARGS {:?}", args);
|
|
|
|
|
2018-12-12 06:51:43 +00:00
|
|
|
print_nested_completion(def, args);
|
2018-12-11 13:21:05 +00:00
|
|
|
}
|
|
|
|
|
2018-12-10 12:40:10 +00:00
|
|
|
pub fn run_cli_command(def: &CommandLineInterface) -> Result<(), Error> {
|
2018-12-10 12:36:52 +00:00
|
|
|
|
|
|
|
let args: Vec<String> = std::env::args().skip(1).collect();
|
|
|
|
|
2018-12-11 11:54:10 +00:00
|
|
|
if !args.is_empty() && args[0] == "bashcomplete" {
|
2018-12-11 13:21:05 +00:00
|
|
|
print_bash_completion(def);
|
2018-12-11 11:54:10 +00:00
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
2018-12-10 12:36:52 +00:00
|
|
|
match def {
|
2018-12-10 12:40:10 +00:00
|
|
|
CommandLineInterface::Simple(cli_cmd) => handle_simple_command(cli_cmd, args),
|
|
|
|
CommandLineInterface::Nested(map) => handle_nested_command(map, args),
|
2018-12-10 12:36:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct CliCommand {
|
|
|
|
pub info: ApiMethod,
|
|
|
|
pub arg_param: Vec<&'static str>,
|
|
|
|
pub fixed_param: Vec<&'static str>,
|
|
|
|
}
|
|
|
|
|
2018-12-11 10:12:13 +00:00
|
|
|
impl CliCommand {
|
|
|
|
|
|
|
|
pub fn new(info: ApiMethod) -> Self {
|
|
|
|
Self { info, arg_param: vec![], fixed_param: vec![] }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn arg_param(mut self, names: Vec<&'static str>) -> Self {
|
|
|
|
self.arg_param = names;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fixed_param(mut self, args: Vec<&'static str>) -> Self {
|
|
|
|
self.fixed_param = args;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
pub struct CliCommandMap {
|
|
|
|
pub commands: HashMap<String, CommandLineInterface>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CliCommandMap {
|
|
|
|
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self { commands: HashMap:: new() }
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn insert<S: Into<String>>(mut self, name: S, cli: CommandLineInterface) -> Self {
|
|
|
|
self.commands.insert(name.into(), cli);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-10 12:40:10 +00:00
|
|
|
pub enum CommandLineInterface {
|
2018-12-10 12:36:52 +00:00
|
|
|
Simple(CliCommand),
|
2018-12-11 10:31:36 +00:00
|
|
|
Nested(CliCommandMap),
|
2018-12-10 12:36:52 +00:00
|
|
|
}
|
2018-12-10 12:51:10 +00:00
|
|
|
|
|
|
|
impl From<CliCommand> for CommandLineInterface {
|
|
|
|
fn from(cli_cmd: CliCommand) -> Self {
|
|
|
|
CommandLineInterface::Simple(cli_cmd)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-11 10:31:36 +00:00
|
|
|
impl From<CliCommandMap> for CommandLineInterface {
|
|
|
|
fn from(list: CliCommandMap) -> Self {
|
|
|
|
CommandLineInterface::Nested(list)
|
2018-12-10 12:51:10 +00:00
|
|
|
}
|
|
|
|
}
|