use std::collections::HashMap; use crate::LightClient; pub trait Command { fn help(&self); fn short_help(&self) -> String; fn exec(&self, _args: &[&str], lightclient: &mut LightClient); } struct SyncCommand {} impl Command for SyncCommand { fn help(&self) { println!("Type sync for syncing"); } fn short_help(&self) -> String { "Download CompactBlocks and sync to the server".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { lightclient.do_sync(); } } struct RescanCommand {} impl Command for RescanCommand { fn help(&self) { println!("Rescan the wallet from it's initial state, rescanning and downloading all blocks and transactions."); } fn short_help(&self) -> String { "Rescan the wallet, downloading and scanning all blocks and transactions".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { lightclient.do_rescan(); } } struct HelpCommand {} impl Command for HelpCommand { fn help(&self) { println!("Lists all available commands"); } fn short_help(&self) -> String { "Lists all available commands".to_string() } fn exec(&self, _args: &[&str], _: &mut LightClient) { // Print a list of all commands println!("Available commands:"); get_commands().iter().for_each(| (cmd, obj) | { println!("{} - {}", cmd, obj.short_help()); }); } } struct InfoCommand {} impl Command for InfoCommand { fn help(&self) { println!("Gets server info"); } fn short_help(&self) -> String { "Get the lightwalletd server's info".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { lightclient.do_info(); } } struct AddressCommand {} impl Command for AddressCommand { fn help(&self) { println!("Show my addresses"); } fn short_help(&self) -> String { "List all current addresses".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { let res = lightclient.do_address(); println!("{}", res.pretty(2)); } } struct SendCommand {} impl Command for SendCommand { fn help(&self) { println!("Sends ZEC to an address"); println!("Usage:"); println!("send recipient_address value memo"); } fn short_help(&self) -> String { "Send ZEC to the given address".to_string() } fn exec(&self, args: &[&str], lightclient: &mut LightClient) { // Parse the args. // 1 - Destination address. T or Z address if args.len() < 2 || args.len() > 3 { self.help(); return; } // Make sure we can parse the amount let value = match args[1].parse::() { Ok(amt) => amt, Err(e) => { println!("Couldn't parse amount: {}", e); return; } }; let memo = if args.len() == 3 { Some(args[2].to_string()) } else {None}; lightclient.do_send(args[0], value, memo); } } struct SaveCommand {} impl Command for SaveCommand { fn help(&self) { println!("Save wallet to disk"); } fn short_help(&self) -> String { "Save wallet file to disk".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { lightclient.do_save(); } } struct SeedCommand {} impl Command for SeedCommand { fn help(&self) { println!("Show the seed phrase for the wallet"); } fn short_help(&self) -> String { "Display the seed phrase".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { let phrase = lightclient.do_seed_phrase(); println!("PLEASE SAVE YOUR SEED PHRASE CAREFULLY AND DO NOT SHARE IT"); println!(); println!("{}", phrase); println!(); } } struct TransactionsCommand {} impl Command for TransactionsCommand { fn help(&self) { println!("Show transactions"); } fn short_help(&self) -> String { "List all transactions in the wallet".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { let txns = lightclient.do_list_transactions(); println!("{}", txns.pretty(2)); } } struct NotesCommand {} impl Command for NotesCommand { fn help(&self) { println!("Show Notes"); } fn short_help(&self) -> String { "List all sapling notes in the wallet".to_string() } fn exec(&self, args: &[&str], lightclient: &mut LightClient) { // Parse the args. if args.len() > 1 { self.help(); return; } // Make sure we can parse the amount let all_notes = if args.len() == 1 { match args[0] { "all" => true, _ => { println!("Invalid argument. Specify 'all' to include unspent notes"); return; } } } else { false }; let txns = lightclient.do_list_notes(all_notes); println!("{}", txns.pretty(2)); } } struct QuitCommand {} impl Command for QuitCommand { fn help(&self) { println!("Quit"); } fn short_help(&self) -> String { "Quit the lightwallet, saving state to disk".to_string() } fn exec(&self, _args: &[&str], lightclient: &mut LightClient) { lightclient.do_save(); } } // TODO: Add rescan command // TODO: Add consolidate command to consolidate t inputs pub fn get_commands() -> Box>> { let mut map: HashMap> = HashMap::new(); map.insert("sync".to_string(), Box::new(SyncCommand{})); map.insert("rescan".to_string(), Box::new(RescanCommand{})); map.insert("help".to_string(), Box::new(HelpCommand{})); map.insert("address".to_string(), Box::new(AddressCommand{})); map.insert("info".to_string(), Box::new(InfoCommand{})); map.insert("send".to_string(), Box::new(SendCommand{})); map.insert("save".to_string(), Box::new(SaveCommand{})); map.insert("quit".to_string(), Box::new(QuitCommand{})); map.insert("list".to_string(), Box::new(TransactionsCommand{})); map.insert("notes".to_string(), Box::new(NotesCommand{})); map.insert("seed".to_string(), Box::new(SeedCommand{})); Box::new(map) } pub fn do_user_command(cmd: &str, args: &Vec<&str>, lightclient: &mut LightClient) { match get_commands().get(cmd) { Some(cmd) => cmd.exec(args, lightclient), None => { println!("Unknown command : {}. Type 'help' for a list of commands", cmd); } } }