use super::{Repl, InterpreterDirectiveOutput}; use colored::*; pub type BoxedCommandFunction = Box<(fn(&mut Repl, &[&str]) -> Option)>; /// A CommandTree is either a `Terminal` or a `NonTerminal`. When command parsing reaches the first /// Terminal, it will use the `DirectiveAction` found there to find an appropriate function to execute, /// and then execute it with any remaining arguments //TODO remove function: BoxedCommandFunction #[derive(Clone)] pub enum CommandTree { Terminal { name: String, children: Vec, help_msg: Option, function: BoxedCommandFunction, action: DirectiveAction, }, NonTerminal { name: String, children: Vec, help_msg: Option, action: DirectiveAction, }, //TODO get rid of Top Top(Vec), } #[derive(Debug, Clone)] pub enum DirectiveAction { Null, Help, QuitProgram } impl DirectiveAction { fn perform(&self, repl: &mut Repl, arguments: &[&str]) -> InterpreterDirectiveOutput { use DirectiveAction::*; match self { Null => None, Help => Some(repl.print_help_message(arguments)), QuitProgram => { repl.save_before_exit(); ::std::process::exit(0) } } } } impl CommandTree { pub fn nonterm_no_further_tab_completions(s: &str, help: Option<&str>) -> CommandTree { CommandTree::NonTerminal {name: s.to_string(), help_msg: help.map(|x| x.to_string()), children: vec![], action: DirectiveAction::Null } } pub fn terminal(s: &str, help: Option<&str>, children: Vec, function: BoxedCommandFunction) -> CommandTree { CommandTree::Terminal {name: s.to_string(), help_msg: help.map(|x| x.to_string()), function, children, action: DirectiveAction::Null } } pub fn terminal_act(s: &str, help: Option<&str>, children: Vec, action: DirectiveAction) -> CommandTree { CommandTree::Terminal {name: s.to_string(), help_msg: help.map(|x| x.to_string()), function: Box::new(|_,_| { None }), children, action} } pub fn nonterm(s: &str, help: Option<&str>, children: Vec) -> CommandTree { CommandTree::NonTerminal { name: s.to_string(), help_msg: help.map(|x| x.to_string()), children, action: DirectiveAction::Null } } pub fn get_cmd(&self) -> &str { match self { CommandTree::Terminal { name, .. } => name.as_str(), CommandTree::NonTerminal {name, ..} => name.as_str(), CommandTree::Top(_) => "", } } pub fn get_help(&self) -> &str { match self { CommandTree::Terminal { help_msg, ..} => help_msg.as_ref().map(|s| s.as_str()).unwrap_or(""), CommandTree::NonTerminal { help_msg, .. } => help_msg.as_ref().map(|s| s.as_str()).unwrap_or(""), CommandTree::Top(_) => "" } } pub fn get_children(&self) -> Vec<&str> { use CommandTree::*; match self { Terminal { children, .. } | NonTerminal { children, .. } | Top(children) => children.iter().map(|x| x.get_cmd()).collect() } } pub fn perform(&self, repl: &mut Repl, arguments: &Vec<&str>) -> InterpreterDirectiveOutput { let mut dir_pointer: &CommandTree = self; let mut idx = 0; let res: Result<(DirectiveAction, usize), String> = loop { match dir_pointer { CommandTree::Top(subcommands) | CommandTree::NonTerminal { children: subcommands, .. } => { let next_command = match arguments.get(idx) { Some(cmd) => cmd, None => break Err(format!("Command requires arguments")) }; idx += 1; match subcommands.iter().find(|sc| sc.get_cmd() == *next_command) { Some(command_tree) => { dir_pointer = command_tree; }, None => break Err(format!("Command {} not found", next_command)) }; }, CommandTree::Terminal { action, .. } => { break Ok((action.clone(), idx)); }, } }; match res { Ok((action, idx)) => action.perform(repl, &arguments[idx..]), Err(err) => Some(err.red().to_string()) } } }