use clap::{App, AppSettings, Arg, ArgMatches}; use crate::config::Config; use crate::error::Result; // TODO --add-site (in addition to defaults) // TODO set_api_key should probably just be a bool, since we have config pub struct Opts { pub list_sites: bool, pub print_config_path: bool, pub update_sites: bool, pub set_api_key: Option, pub query: Option, pub config: Config, } /// Get CLI opts and args, with defaults pulled from user configuration pub fn get_opts() -> Result { get_opts_with(Config::new, |a| a.get_matches()) } /// Get CLI opts, starting with defaults produced from `mk_config` and matching args with /// `get_matches`. fn get_opts_with(mk_config: F, get_matches: G) -> Result where F: FnOnce() -> Result, G: for<'a> FnOnce(App<'a, '_>) -> ArgMatches<'a>, { let config = mk_config()?; let limit = &config.limit.to_string(); let sites = &config.sites.join(";"); let engine = &config.search_engine.to_string(); let clapp = App::new("so") .setting(AppSettings::ColoredHelp) .version(clap::crate_version!()) .author(clap::crate_authors!()) .about(clap::crate_description!()) .arg( Arg::with_name("list-sites") .long("list-sites") .help("Print available StackExchange sites"), ) .arg( Arg::with_name("update-sites") .long("update-sites") .help("Update cache of StackExchange sites"), ) .arg( Arg::with_name("set-api-key") .long("set-api-key") .number_of_values(1) .takes_value(true) .value_name("key") .help("Set StackExchange API key"), ) .arg( Arg::with_name("print-config-path") .long("print-config-path") .help("Print path to config file") .hidden(true), ) .arg( Arg::with_name("site") .long("site") .short("s") .multiple(true) .number_of_values(1) .takes_value(true) .default_value(sites) .value_name("site-code") .help("StackExchange site to search"), ) .arg( Arg::with_name("limit") .long("limit") .short("l") .number_of_values(1) .takes_value(true) .default_value(limit) .value_name("int") .validator(|s| s.parse::().map(|_| ()).map_err(|e| e.to_string())) .help("Question limit"), ) .arg( Arg::with_name("lucky") .long("lucky") .help("Print the top-voted answer of the most relevant question"), ) .arg( Arg::with_name("no-lucky") .long("no-lucky") .help("Disable lucky") .conflicts_with("lucky") .hidden(!config.lucky), ) .arg( Arg::with_name("query") .multiple(true) .index(1) .required_unless_one(&[ "list-sites", "update-sites", "set-api-key", "print-config-path", ]), ) .arg( Arg::with_name("search-engine") .long("search-engine") .short("e") .number_of_values(1) .takes_value(true) .default_value(engine) .value_name("engine") .possible_values(&["duckduckgo", "google", "stackexchange"]) .help("Use specified search engine") .next_line_help(true), ); let matches = get_matches(clapp); let lucky = match (matches.is_present("lucky"), matches.is_present("no-lucky")) { (true, _) => true, (_, true) => false, _ => config.lucky, }; Ok(Opts { list_sites: matches.is_present("list-sites"), print_config_path: matches.is_present("print-config-path"), update_sites: matches.is_present("update-sites"), set_api_key: matches.value_of("set-api-key").map(String::from), query: matches .values_of("query") .map(|q| q.collect::>().join(" ")), config: Config { // these unwraps are safe via clap default values & validators limit: matches.value_of("limit").unwrap().parse::().unwrap(), search_engine: serde_yaml::from_str(matches.value_of("search-engine").unwrap())?, sites: matches .values_of("site") .unwrap() .map(|s| s.split(';')) .flatten() .map(String::from) .collect(), api_key: matches .value_of("set-api-key") .map(String::from) .or(config.api_key), lucky, }, }) } #[cfg(test)] mod tests { use super::*; use crate::config::SearchEngine; fn defaults() -> Config { Config { api_key: Some(String::from("my key")), limit: 64, lucky: false, sites: vec![ String::from("some"), String::from("sites"), String::from("yeah"), ], search_engine: SearchEngine::DuckDuckGo, } } fn mk_config() -> Result { Ok(defaults()) } #[test] fn test_defaults() { let opts = get_opts_with(mk_config, |a| { a.get_matches_from(vec!["so", "how do I exit Vim"]) }); assert_eq!(opts.unwrap().config, defaults()); } #[test] fn test_overrides() { let opts = get_opts_with(mk_config, |a| { a.get_matches_from(vec!["so", "-s", "english", "how do I exit Vim"]) }); assert_eq!( opts.unwrap().config, Config { sites: vec![String::from("english")], ..defaults() } ); let opts = get_opts_with(mk_config, |a| { a.get_matches_from(vec!["so", "-l", "5", "--lucky", "how do I exit Vim"]) }); assert_eq!( opts.unwrap().config, Config { limit: 5, lucky: true, ..defaults() } ); } #[test] fn test_set_api_key() { let opts = get_opts_with(mk_config, |a| { a.get_matches_from(vec!["so", "--set-api-key", "new key"]) }) .unwrap(); // Uses key in new config assert_eq!( opts.config, Config { api_key: Some(String::from("new key")), ..defaults() } ); // Flags it in opts assert_eq!(opts.set_api_key, Some(String::from("new key"))); } #[test] #[should_panic] fn test_conflicts() { get_opts_with(mk_config, |a| { a.get_matches_from_safe(vec!["so", "--lucky", "--no-lucky"]) .unwrap() }) .unwrap(); } }