From 1b69ca4bee27ceb73fa2ed3effdbd5ffd60f5c3c Mon Sep 17 00:00:00 2001 From: Kevin K Date: Mon, 9 Nov 2015 03:48:49 -0500 Subject: [PATCH] refactor: moves code for ergonomics --- src/app/app.rs | 3471 ---------------------------------- src/app/mod.rs | 3383 ++++++++++++++++++++++++++++++++- src/{app => }/errors.rs | 22 +- src/lib.rs | 5 +- src/{app => }/suggestions.rs | 30 + 5 files changed, 3428 insertions(+), 3483 deletions(-) delete mode 100644 src/app/app.rs rename src/{app => }/errors.rs (96%) rename src/{app => }/suggestions.rs (60%) diff --git a/src/app/app.rs b/src/app/app.rs deleted file mode 100644 index 44b0cb86..00000000 --- a/src/app/app.rs +++ /dev/null @@ -1,3471 +0,0 @@ -use std::collections::{BTreeMap, HashMap, VecDeque}; -use std::env; -use std::io::{self, BufRead, BufWriter, Write}; -use std::path::Path; -use std::process; -use std::ffi::OsStr; -use std::borrow::Borrow; - -#[cfg(feature = "yaml")] -use yaml_rust::Yaml; -use vec_map::VecMap; - -use args::{Arg, ArgMatches, MatchedArg, SubCommand}; -use args::{FlagBuilder, OptBuilder, PosBuilder}; -use args::settings::{ArgFlags, ArgSettings}; -use args::ArgGroup; -use fmt::Format; -use super::settings::{AppFlags, AppSettings}; - -use super::suggestions::{DidYouMeanMessageStyle, did_you_mean}; -use super::errors::{ClapError, ClapErrorType, ClapResult, error_builder}; - - -const INTERNAL_ERROR_MSG: &'static str = "Fatal internal error. Please consider filing a bug \ - report at https://github.com/kbknapp/clap-rs/issues"; - -/// Used to create a representation of a command line program and all possible command line -/// arguments. -/// -/// Application settings are set using the "builder pattern" with `.get_matches()` being the -/// terminal method that starts the runtime-parsing process and returns information about -/// the user supplied arguments (or lack there of). -/// -/// There aren't any mandatory "options" that one must set. The "options" may also appear in any -/// order (so long as `.get_matches()` is the last method called). -/// -/// -/// # Examples -/// -/// ```no_run -/// # use clap::{App, Arg}; -/// let matches = App::new("myprog") -/// .author("Me, me@mail.com") -/// .version("1.0.2") -/// .about("Explains in brief what the program does") -/// .arg( -/// Arg::with_name("in_file").index(1) -/// ) -/// .after_help("Longer explaination to appear after the options when \ -/// displaying the help information from --help or -h") -/// .get_matches(); -/// -/// // Your program logic starts here... -/// ``` -#[allow(missing_debug_implementations)] -pub struct App<'a, 'v, 'ab, 'u, 'h, 'ar> { - // The name displayed to the user when showing version and help/usage information - name: String, - name_slice: &'ar str, - // A string of author(s) if desired. Displayed when showing help/usage information - author: Option<&'a str>, - // The version displayed to the user - version: Option<&'v str>, - // A brief explanation of the program that gets displayed to the user when shown - // help/usage - // information - about: Option<&'ab str>, - // Additional help information - more_help: Option<&'h str>, - // A list of possible flags - flags: Vec>, - // A list of possible options - opts: Vec>, - // A list of positional arguments - positionals: VecMap>, - // A list of subcommands - subcommands: Vec>, - help_short: Option, - version_short: Option, - required: Vec<&'ar str>, - short_list: Vec, - long_list: Vec<&'ar str>, - blacklist: Vec<&'ar str>, - usage_str: Option<&'u str>, - bin_name: Option, - usage: Option, - groups: HashMap<&'ar str, ArgGroup<'ar, 'ar>>, - global_args: Vec>, - help_str: Option<&'u str>, - settings: AppFlags, - overrides: Vec<&'ar str>, -} - -impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar> { - /// Creates a new instance of an application requiring a name (such as the binary). The name - /// will be displayed to the user when they request to print version or help and usage - /// information. The name should not contain spaces (hyphens '-' are ok). - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let prog = App::new("myprog") - /// # .get_matches(); - /// ``` - pub fn new(n: &'ar str) -> Self { - App { - name: n.to_owned(), - name_slice: n, - author: None, - about: None, - more_help: None, - version: None, - flags: vec![], - opts: vec![], - positionals: VecMap::new(), - subcommands: vec![], - help_short: None, - version_short: None, - required: vec![], - short_list: vec![], - long_list: vec![], - usage_str: None, - usage: None, - blacklist: vec![], - bin_name: None, - groups: HashMap::new(), - global_args: vec![], - help_str: None, - overrides: vec![], - settings: AppFlags::new(), - } - } - - /// Creates a new instace of `App` from a .yml (YAML) file. The YAML file must be properly - /// formatted or this function will panic!(). A full example of supported YAML objects can be - /// found in `examples/17_yaml.rs` and `examples/17_yaml.yml`. - /// - /// In order to use this function you must compile with the `features = ["yaml"]` in your - /// settings for `[dependencies.clap]` table of your `Cargo.toml` - /// - /// Note, due to how the YAML objects are built there is a convienience macro for loading the - /// YAML file (relative to the current file, like modules work). That YAML object can then be - /// passed to this function. - /// - /// # Examples - /// - /// ```ignore - /// # use clap::App; - /// let yml = load_yaml!("app.yml"); - /// let app = App::from_yaml(yml); - /// ``` - #[cfg(feature = "yaml")] - pub fn from_yaml<'y>(mut yaml: &'y Yaml) -> App<'y, 'y, 'y, 'y, 'y, 'y> { - // We WANT this to panic on error...so expect() is good. - let mut is_sc = None; - let mut a = if let Some(name) = yaml["name"].as_str() { - App::new(name) - } else { - let yaml_hash = yaml.as_hash().unwrap(); - let sc_key = yaml_hash.keys().nth(0).unwrap(); - is_sc = Some(yaml_hash.get(sc_key).unwrap()); - App::new(sc_key.as_str().unwrap()) - }; - yaml = if let Some(sc) = is_sc { - sc - } else { - yaml - }; - if let Some(v) = yaml["version"].as_str() { - a = a.version(v); - } - if let Some(v) = yaml["author"].as_str() { - a = a.author(v); - } - if let Some(v) = yaml["bin_name"].as_str() { - a = a.bin_name(v); - } - if let Some(v) = yaml["about"].as_str() { - a = a.about(v); - } - if let Some(v) = yaml["after_help"].as_str() { - a = a.after_help(v); - } - if let Some(v) = yaml["usage"].as_str() { - a = a.usage(v); - } - if let Some(v) = yaml["help"].as_str() { - a = a.help(v); - } - if let Some(v) = yaml["help_short"].as_str() { - a = a.help_short(v); - } - if let Some(v) = yaml["version_short"].as_str() { - a = a.version_short(v); - } - if let Some(v) = yaml["settings"].as_vec() { - for ys in v { - if let Some(s) = ys.as_str() { - a = a.setting(s.parse().ok().expect("unknown AppSetting found in YAML file")); - } - } - } - if let Some(v) = yaml["args"].as_vec() { - for arg_yaml in v { - a = a.arg(Arg::from_yaml(&arg_yaml.as_hash().unwrap())); - } - } - if let Some(v) = yaml["subcommands"].as_vec() { - for sc_yaml in v { - a = a.subcommand(SubCommand::from_yaml(&sc_yaml)); - } - } - if let Some(v) = yaml["arg_groups"].as_vec() { - for ag_yaml in v { - a = a.arg_group(ArgGroup::from_yaml(&ag_yaml.as_hash().unwrap())); - } - } - - a - } - - /// Sets a string of author(s) and will be displayed to the user when they request the help - /// information with `--help` or `-h`. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .author("Me, me@mymain.com") - /// # ; - /// ``` - pub fn author(mut self, a: &'a str) -> Self { - self.author = Some(a); - self - } - - /// Overrides the system-determined binary name. This should only be used when absolutely - /// neccessary, such as the binary name for your application is misleading, or perhaps *not* - /// how the user should invoke your program. - /// - /// **NOTE:** This command **should not** be used for SubCommands. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .bin_name("my_binary") - /// # ; - /// ``` - pub fn bin_name(mut self, a: &str) -> Self { - self.bin_name = Some(a.to_owned()); - self - } - - /// Sets a string briefly describing what the program does and will be displayed when - /// displaying help information. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .about("Does really amazing things to great people") - /// # ; - /// ``` - pub fn about(mut self, a: &'ab str) -> Self { - self.about = Some(a); - self - } - - /// Adds additional help information to be displayed in addition to and directly after - /// auto-generated help. This information is displayed **after** the auto-generated help - /// information. This additional help is often used to describe how to use the arguments, - /// or caveats to be noted. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::App; - /// App::new("myprog") - /// .after_help("Does really amazing things to great people") - /// # ; - /// ``` - pub fn after_help(mut self, h: &'h str) -> Self { - self.more_help = Some(h); - self - } - - /// Allows subcommands to override all requirements of the parent (this command). For example - /// if you had a subcommand or even top level application which had a required arguments that - /// are only required as long as there is no subcommand present. - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandsNegateReqs` instead. - /// This method will be removed at 2.x - /// - /// **NOTE:** This defaults to false (using subcommand does *not* negate requirements) - /// - /// # Examples - /// - /// ```no_run - /// # use clap::App; - /// App::new("myprog") - /// .subcommands_negate_reqs(true) - /// # ; - /// ``` - pub fn subcommands_negate_reqs(mut self, n: bool) -> Self { - if n { - self.settings.set(&AppSettings::SubcommandsNegateReqs); - } else { - self.settings.unset(&AppSettings::SubcommandsNegateReqs); - } - self - } - - /// Allows specifying that if no subcommand is present at runtime, error and exit gracefully - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandRequired` instead. This - /// method will be removed at 2.x - /// - /// **NOTE:** This defaults to false (subcommands do *not* need to be present) - /// - /// # Examples - /// - /// ```no_run - /// # use clap::App; - /// App::new("myprog") - /// .subcommand_required(true) - /// # ; - /// ``` - pub fn subcommand_required(mut self, n: bool) -> Self { - if n { - self.settings.set(&AppSettings::SubcommandRequired); - } else { - self.settings.unset(&AppSettings::SubcommandRequired); - } - self - } - - /// Sets a string of the version number to be displayed when displaying version or help - /// information. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .version("v0.1.24") - /// # ; - /// ``` - pub fn version(mut self, v: &'v str) -> Self { - self.version = Some(v); - self - } - - /// Sets a custom usage string to override the auto-generated usage string. - /// - /// This will be displayed to the user when errors are found in argument parsing, or when you - /// call `ArgMatches::usage()` - /// - /// **NOTE:** You do not need to specify the "USAGE: \n\t" portion, as that will - /// still be applied by `clap`, you only need to specify the portion starting - /// with the binary name. - /// - /// **NOTE:** This will not replace the entire help message, *only* the portion - /// showing the usage. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .usage("myapp [-clDas] ") - /// # ; - /// ``` - pub fn usage(mut self, u: &'u str) -> Self { - self.usage_str = Some(u); - self - } - - /// Sets a custom help message and overrides the auto-generated one. This should only be used - /// when the auto-generated message does not suffice. - /// - /// This will be displayed to the user when they use the default `--help` or `-h` - /// - /// **NOTE:** This replaces the **entire** help message, so nothing will be auto-generated. - /// - /// **NOTE:** This **only** replaces the help message for the current command, meaning if you - /// are using subcommands, those help messages will still be auto-generated unless you - /// specify a `.help()` for them as well. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myapp") - /// .help("myapp v1.0\n\ - /// Does awesome things\n\ - /// (C) me@mail.com\n\n\ - /// - /// USAGE: myapp \n\n\ - /// - /// Options:\n\ - /// -h, --helpe Dispay this message\n\ - /// -V, --version Display version info\n\ - /// -s Do something with stuff\n\ - /// -v Be verbose\n\n\ - /// - /// Commmands:\n\ - /// help Prints this message\n\ - /// work Do some work") - /// # ; - /// ``` - pub fn help(mut self, h: &'u str) -> Self { - self.help_str = Some(h); - self - } - - /// Sets the short version of the `help` argument without the preceding `-`. - /// - /// By default `clap` automatically assigns `h`, but this can be overridden - /// - /// **NOTE:** Any leading `-` characters will be stripped, and only the first - /// non `-` chacter will be used as the `short` version - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// // Using an uppercase `H` instead of the default lowercase `h` - /// .help_short("H") - /// # ; - pub fn help_short(mut self, s: &str) -> Self { - self.help_short = s.trim_left_matches(|c| c == '-') - .chars() - .nth(0); - self - } - - /// Sets the short version of the `version` argument without the preceding `-`. - /// - /// By default `clap` automatically assigns `V`, but this can be overridden - /// - /// **NOTE:** Any leading `-` characters will be stripped, and only the first - /// non `-` chacter will be used as the `short` version - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// // Using a lowercase `v` instead of the default capital `V` - /// .version_short("v") - /// # ; - pub fn version_short(mut self, s: &str) -> Self { - self.version_short = s.trim_left_matches(|c| c == '-') - .chars() - .nth(0); - self - } - - /// Specifies that the help text sould be displayed (and then exit gracefully), if no - /// arguments are present at runtime (i.e. an empty run such as, `$ myprog`. - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::ArgRequiredElseHelp` instead. This - /// method will be removed at 2.x - /// - /// **NOTE:** Subcommands count as arguments - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .arg_required_else_help(true) - /// # ; - /// ``` - pub fn arg_required_else_help(mut self, tf: bool) -> Self { - if tf { - self.settings.set(&AppSettings::ArgRequiredElseHelp); - } else { - self.settings.unset(&AppSettings::ArgRequiredElseHelp); - } - self - } - - /// Hides a subcommand from help message output. - /// - /// **NOTE:** This does **not** hide the subcommand from usage strings on error - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, SubCommand}; - /// # let matches = App::new("myprog") - /// # .subcommand( - /// # SubCommand::with_name("debug") - /// .hidden(true) - /// # ).get_matches(); - pub fn hidden(mut self, h: bool) -> Self { - if h { - self.settings.set(&AppSettings::Hidden); - } else { - self.settings.unset(&AppSettings::Hidden); - } - self - } - - /// Uses version of the current command for all subcommands. (Defaults to false; subcommands - /// have independant version strings) - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::GlobalVersion` instead. This - /// method will be removed at 2.x - /// - /// **NOTE:** The version for the current command and this setting must be set **prior** to - /// adding any subcommands - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, SubCommand}; - /// App::new("myprog") - /// .version("v1.1") - /// .global_version(true) - /// .subcommand(SubCommand::with_name("test")) - /// .get_matches(); - /// // running `myprog test --version` will display - /// // "myprog-test v1.1" - /// ``` - pub fn global_version(mut self, gv: bool) -> Self { - if gv { - self.settings.set(&AppSettings::GlobalVersion); - } else { - self.settings.unset(&AppSettings::GlobalVersion); - } - self - } - - /// Disables `-V` and `--version` for all subcommands (Defaults to false; subcommands have - /// version flags) - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::VersionlessSubcommands` instead. - /// This method will be removed at 2.x - /// - /// **NOTE:** This setting must be set **prior** adding any subcommands - /// - /// **NOTE:** Do not set this value to false, it will have undesired results! - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, SubCommand}; - /// App::new("myprog") - /// .version("v1.1") - /// .versionless_subcommands(true) - /// .subcommand(SubCommand::with_name("test")) - /// .get_matches(); - /// // running `myprog test --version` will display unknown argument error - /// ``` - pub fn versionless_subcommands(mut self, vers: bool) -> Self { - if vers { - self.settings.set(&AppSettings::VersionlessSubcommands); - } else { - self.settings.unset(&AppSettings::VersionlessSubcommands); - } - self - } - - /// By default the auto-generated help message groups flags, options, and positional arguments - /// separately. This setting disable that and groups flags and options together presenting a - /// more unified help message (a la getopts or docopt style). - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::UnifiedHelpMessage` instead. This - /// method will be removed at 2.x - /// - /// **NOTE:** This setting is cosmetic only and does not affect any functionality. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, SubCommand}; - /// App::new("myprog") - /// .unified_help_message(true) - /// .get_matches(); - /// // running `myprog --help` will display a unified "docopt" or "getopts" style help message - /// ``` - pub fn unified_help_message(mut self, uni_help: bool) -> Self { - if uni_help { - self.settings.set(&AppSettings::UnifiedHelpMessage); - } else { - self.settings.unset(&AppSettings::UnifiedHelpMessage); - } - self - } - - /// Will display a message "Press [ENTER]/[RETURN] to continue..." and wait user before - /// exiting - /// - /// This is most useful when writing an application which is run from a GUI shortcut, or on - /// Windows where a user tries to open the binary by double-clicking instead of using the - /// command line (i.e. set `.arg_required_else_help(true)` and `.wait_on_error(true)` to - /// display the help in such a case). - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::WaitOnError` instead. This - /// method will be removed at 2.x - /// - /// **NOTE:** This setting is **not** recursive with subcommands, meaning if you wish this - /// behavior for all subcommands, you must set this on each command (needing this is extremely - /// rare) - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .arg_required_else_help(true) - /// # ; - /// ``` - pub fn wait_on_error(mut self, w: bool) -> Self { - if w { - self.settings.set(&AppSettings::WaitOnError); - } else { - self.settings.unset(&AppSettings::WaitOnError); - } - self - } - - /// Specifies that the help text sould be displayed (and then exit gracefully), if no - /// subcommands are present at runtime (i.e. an empty run such as, `$ myprog`. - /// - /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandRequiredElseHelp` - /// instead. This method will be removed at 2.x - /// - /// **NOTE:** This should *not* be used with `.subcommand_required()` as they do the same - /// thing, except one prints the help text, and one prints an error. - /// - /// **NOTE:** If the user specifies arguments at runtime, but no subcommand the help text will - /// still be displayed and exit. If this is *not* the desired result, consider using - /// `.arg_required_else_help()` - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .subcommand_required_else_help(true) - /// # ; - /// ``` - pub fn subcommand_required_else_help(mut self, tf: bool) -> Self { - if tf { - self.settings.set(&AppSettings::SubcommandRequiredElseHelp); - } else { - self.settings.unset(&AppSettings::SubcommandRequiredElseHelp); - } - self - } - - /// Enables Application level settings, passed as argument - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, AppSettings}; - /// App::new("myprog") - /// .setting(AppSettings::SubcommandRequired) - /// .setting(AppSettings::WaitOnError) - /// # ; - /// ``` - pub fn setting(mut self, setting: AppSettings) -> Self { - self.settings.set(&setting); - self - } - - /// Enables multiple Application level settings, passed as argument - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, AppSettings}; - /// App::new("myprog") - /// .settings( &[AppSettings::SubcommandRequired, - /// AppSettings::WaitOnError]) - /// # ; - /// ``` - pub fn settings(mut self, settings: &[AppSettings]) -> Self { - for s in settings { - self.settings.set(s); - } - self - } - - /// Adds an argument to the list of valid possibilties manually. This method allows you full - /// control over the arguments settings and options (as well as dynamic generation). It also - /// allows you specify several more advanced configuration options such as relational rules - /// (exclusions and requirements). - /// - /// The only disadvantage to this method is that it's more verbose, and arguments must be added - /// one at a time. Using `Arg::from_usage` helps with the verbosity, and still allows full - /// control over the advanced configuration options. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// // Adding a single "flag" argument with a short and help text, using Arg::with_name() - /// .arg( - /// Arg::with_name("debug") - /// .short("d") - /// .help("turns on debugging mode") - /// ) - /// // Adding a single "option" argument with a short, a long, and help text using the less - /// // verbose Arg::from_usage() - /// .arg( - /// Arg::from_usage("-c --config=[CONFIG] 'Optionally sets a config file to use'") - /// ) - /// # ; - /// ``` - pub fn arg(mut self, a: Arg<'ar, 'ar, 'ar, 'ar, 'ar, 'ar>) -> Self { - self.add_arg(a); - self - } - - // actually adds the arguments - fn add_arg(&mut self, a: Arg<'ar, 'ar, 'ar, 'ar, 'ar, 'ar>) { - if self.flags.iter().any(|f| &f.name == &a.name) || - self.opts.iter().any(|o| o.name == a.name) || - self.positionals.values().any(|p| p.name == a.name) { - panic!("Non-unique argument name: {} is already in use", a.name); - } - if let Some(grp) = a.group { - let ag = self.groups.entry(grp).or_insert(ArgGroup::with_name(grp)); - ag.args.push(a.name); - } - if let Some(s) = a.short { - if self.short_list.contains(&s) { - panic!("Argument short must be unique\n\n\t-{} is already in use", - s); - } else { - self.short_list.push(s); - } - } - if let Some(l) = a.long { - if self.long_list.contains(&l) { - panic!("Argument long must be unique\n\n\t--{} is already in use", - l); - } else { - self.long_list.push(l); - } - if l == "help" { - self.settings.set(&AppSettings::NeedsLongHelp); - } else if l == "version" { - self.settings.set(&AppSettings::NeedsLongVersion); - } - } - if a.required { - self.required.push(a.name); - } - if a.index.is_some() || (a.short.is_none() && a.long.is_none()) { - let i = if a.index.is_none() { - (self.positionals.len() + 1) - } else { - a.index.unwrap() as usize - }; - if self.positionals.contains_key(&i) { - panic!("Argument \"{}\" has the same index as another positional \ - argument\n\n\tPerhaps try .multiple(true) to allow one positional argument \ - to take multiple values", - a.name); - } - let pb = PosBuilder::from_arg(&a, i as u8, &mut self.required); - // self.positionals_name.insert(pb.name, i); - self.positionals.insert(i, pb); - } else if a.takes_value { - let ob = OptBuilder::from_arg(&a, &mut self.required); - self.opts.push(ob); - } else { - let fb = FlagBuilder::from(&a); - self.flags.push(fb); - } - if a.global { - if a.required { - panic!("Global arguments cannot be required.\n\n\t'{}' is marked as global and \ - required", - a.name); - } - self.global_args.push(a); - } - } - - /// Adds multiple arguments to the list of valid possibilties by iterating over a Vec of Args - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .args( - /// vec![Arg::from_usage("[debug] -d 'turns on debugging info"), - /// Arg::with_name("input").index(1).help("the input file to use")] - /// ) - /// # ; - /// ``` - pub fn args(mut self, args: Vec>) -> Self { - for arg in args.into_iter() { - self = self.arg(arg); - } - self - } - - /// A convienience method for adding a single basic argument (one without advanced - /// relational rules) from a usage type string. The string used follows the same rules and - /// syntax as `Arg::from_usage()` - /// - /// The downside to using this method is that you can not set any additional properties of the - /// `Arg` other than what `Arg::from_usage()` supports. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .arg_from_usage("-c --conf= 'Sets a configuration file to use'") - /// # ; - /// ``` - pub fn arg_from_usage(mut self, usage: &'ar str) -> Self { - self = self.arg(Arg::from_usage(usage)); - self - } - - /// Adds multiple arguments at once from a usage string, one per line. See `Arg::from_usage()` - /// for details on the syntax and rules supported. - /// - /// Like `App::arg_from_usage()` the downside is you only set properties for the `Arg`s which - /// `Arg::from_usage()` supports. But here the benefit is pretty strong, as the readability is - /// greatly enhanced, especially if you don't need any of the more advanced configuration - /// options. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// App::new("myprog") - /// .args_from_usage( - /// "-c --conf=[config] 'Sets a configuration file to use' - /// [debug]... -d 'Sets the debugging level' - /// 'The input file to use'" - /// ) - /// # ; - /// ``` - pub fn args_from_usage(mut self, usage: &'ar str) -> Self { - for l in usage.lines() { - self = self.arg(Arg::from_usage(l.trim())); - } - self - } - - /// Adds an ArgGroup to the application. ArgGroups are a family of related arguments. By - /// placing them in a logical group, you make easier requirement and exclusion rules. For - /// instance, you can make an ArgGroup required, this means that one (and *only* one) argument - /// from that group must be present. Using more than one argument from an ArgGroup causes a - /// failure (graceful exit). - /// - /// You can also do things such as name an ArgGroup as a confliction, meaning any of the - /// arguments that belong to that group will cause a failure if present. - /// - /// Perhaps the most common use of ArgGroups is to require one and *only* one argument to be - /// present out of a given set. For example, lets say that you were building an application - /// where one could set a given version number by supplying a string using an option argument, - /// such as `--set-ver v1.2.3`, you also wanted to support automatically using a previous - /// version numer and simply incrementing one of the three numbers, so you create three flags - /// `--major`, `--minor`, and `--patch`. All of these arguments shouldn't be used at one time - /// but perhaps you want to specify that *at least one* of them is used. You can create a - /// group - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, ArgGroup}; - /// # App::new("app") - /// .args_from_usage("--set-ver [ver] 'set the version manually' - /// --major 'auto increase major' - /// --minor 'auto increase minor' - /// --patch 'auto increase patch") - /// .arg_group(ArgGroup::with_name("vers") - /// .add_all(&["ver", "major", "minor","patch"]) - /// .required(true)) - /// # ; - pub fn arg_group(mut self, group: ArgGroup<'ar, 'ar>) -> Self { - if group.required { - self.required.push(group.name); - if let Some(ref reqs) = group.requires { - for r in reqs { - self.required.push(r); - } - } - if let Some(ref bl) = group.conflicts { - for b in bl { - self.blacklist.push(b); - } - } - } - let mut found = false; - if let Some(ref mut grp) = self.groups.get_mut(group.name) { - for a in &group.args { - grp.args.push(a); - } - grp.requires = group.requires.clone(); - grp.conflicts = group.conflicts.clone(); - grp.required = group.required; - found = true; - } - if !found { - self.groups.insert(group.name, group); - } - self - } - - /// Adds a ArgGroups to the application. ArgGroups are a family of related arguments. By - /// placing them in a logical group, you make easier requirement and exclusion rules. For - /// instance, you can make an ArgGroup required, this means that one (and *only* one) argument - /// from that group must be present. Using more than one argument from an ArgGroup causes a - /// failure (graceful exit). - /// - /// You can also do things such as name an ArgGroup as a confliction, meaning any of the - /// arguments that belong to that group will cause a failure if present. - /// - /// Perhaps the most common use of ArgGroups is to require one and *only* one argument to be - /// present out of a given set. For example, lets say that you were building an application - /// where one could set a given version number by supplying a string using an option argument, - /// such as `--set-ver v1.2.3`, you also wanted to support automatically using a previous - /// version numer and simply incrementing one of the three numbers, so you create three flags - /// `--major`, `--minor`, and `--patch`. All of these arguments shouldn't be used at one time - /// but perhaps you want to specify that *at least one* of them is used. You can create a - /// group - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, ArgGroup}; - /// # App::new("app") - /// .args_from_usage("--set-ver [ver] 'set the version manually' - /// --major 'auto increase major' - /// --minor 'auto increase minor' - /// --patch 'auto increase patch") - /// .arg_group(ArgGroup::with_name("vers") - /// .add_all(&["ver", "major", "minor","patch"]) - /// .required(true)) - /// # ; - pub fn arg_groups(mut self, groups: Vec>) -> Self { - for g in groups { - self = self.arg_group(g); - } - self - } - - /// Adds a subcommand to the list of valid possibilties. Subcommands are effectively sub apps, - /// because they can contain their own arguments, subcommands, version, usage, etc. They also - /// function just like apps, in that they get their own auto generated help, version, and - /// usage. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, SubCommand}; - /// # App::new("myprog") - /// .subcommand(SubCommand::with_name("config") - /// .about("Controls configuration features") - /// .arg_from_usage(" 'Required configuration file to use'")) - /// // Additional subcommand configuration goes here, such as other arguments... - /// # ; - /// ``` - pub fn subcommand(mut self, mut subcmd: App<'a, 'v, 'ab, 'u, 'h, 'ar>) -> Self { - if subcmd.name == "help" { - self.settings.set(&AppSettings::NeedsSubcommandHelp); - } - if self.settings.is_set(&AppSettings::VersionlessSubcommands) { - subcmd.settings.set(&AppSettings::DisableVersion); - } - if self.settings.is_set(&AppSettings::GlobalVersion) && subcmd.version.is_none() && - self.version.is_some() { - subcmd.version = Some(self.version.unwrap()); - } - self.subcommands.push(subcmd); - self - } - - /// Adds multiple subcommands to the list of valid possibilties by iterating over a Vec of - /// `SubCommand`s - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg, SubCommand}; - /// # App::new("myprog") - /// .subcommands( vec![ - /// SubCommand::with_name("config").about("Controls configuration functionality") - /// .arg(Arg::with_name("config_file").index(1)), - /// SubCommand::with_name("debug").about("Controls debug functionality")]) - /// # ; - /// ``` - pub fn subcommands(mut self, subcmds: Vec>) -> Self { - for subcmd in subcmds.into_iter() { - self = self.subcommand(subcmd); - } - self - } - - // Creates a usage string if one was not provided by the user manually. This - // happens just - // after all arguments were parsed, but before any subcommands have been parsed - // (so as to - // give subcommands their own usage recursively) - fn create_usage(&self, matches: &[&'ar str]) -> ClapResult { - let mut usage = String::with_capacity(75); - usage.push_str("USAGE:\n\t"); - if let Some(u) = self.usage_str { - usage.push_str(u); - } else if !matches.is_empty() { - try!(self.usage_from_matches(&mut usage, matches)); - } else { - usage.push_str(&*self.usage - .as_ref() - .unwrap_or(self.bin_name - .as_ref() - .unwrap_or(&self.name))); - - let reqs = self.required.iter().map(|n| *n).collect::>(); - let req_strings = self.get_required_from(&reqs, None); - let req_string = req_strings.iter() - .fold(String::new(), |a, s| a + &format!(" {}", s)[..]); - - if !self.flags.is_empty() && !self.settings.is_set(&AppSettings::UnifiedHelpMessage) { - usage.push_str(" [FLAGS]"); - } else { - usage.push_str(" [OPTIONS]"); - } - if !self.settings.is_set(&AppSettings::UnifiedHelpMessage) && !self.opts.is_empty() && - self.opts.iter().any(|a| !a.settings.is_set(&ArgSettings::Required)) { - usage.push_str(" [OPTIONS]"); - } - - usage.push_str(&req_string[..]); - - // places a '--' in the usage string if there are args and options - // supporting multiple values - if !self.positionals.is_empty() && - (self.opts.iter().any(|a| a.settings.is_set(&ArgSettings::Multiple)) || - self.positionals.values().any(|a| a.settings.is_set(&ArgSettings::Multiple))) && - !self.opts.iter().any(|a| a.settings.is_set(&ArgSettings::Required)) && - self.subcommands.is_empty() { - usage.push_str(" [--]") - } - if !self.positionals.is_empty() && - self.positionals - .values() - .any(|a| !a.settings.is_set(&ArgSettings::Required)) { - usage.push_str(" [ARGS]"); - } - - - if !self.subcommands.is_empty() && - !self.settings.is_set(&AppSettings::SubcommandRequired) { - usage.push_str(" [SUBCOMMAND]"); - } else if self.settings.is_set(&AppSettings::SubcommandRequired) && - !self.subcommands.is_empty() { - usage.push_str(" "); - } - } - - usage.shrink_to_fit(); - Ok(usage) - } - - // Creates a context aware usage string, or "smart usage" from currently used - // args, and - // requirements - fn usage_from_matches(&self, usage: &mut String, matches: &[&'ar str]) -> ClapResult<()> { - use std::fmt::Write; - let mut hs: Vec<&str> = self.required.iter().map(|n| *n).collect(); - for n in matches { - hs.push(*n); - } - let reqs = self.get_required_from(&hs, None); - - let r_string = reqs.iter().fold(String::new(), |acc, s| acc + &format!(" {}", s)[..]); - - try!(write!(usage, - "{}{}", - self.usage - .as_ref() - .unwrap_or(self.bin_name.as_ref().unwrap_or(&self.name)), - r_string)); - if self.settings.is_set(&AppSettings::SubcommandRequired) { - try!(write!(usage, " ")); - } - - Ok(()) - } - - /// Prints the full help message to `io::stdout()` using a `BufWriter` - /// - /// # Examples - /// ```no_run - /// # use clap::App; - /// # use std::io; - /// let mut app = App::new("myprog"); - /// let mut out = io::stdout(); - /// app.write_help(&mut out).ok().expect("failed to write to stdout"); - /// ``` - pub fn print_help(&self) -> ClapResult<()> { - let out = io::stdout(); - let mut buf_w = BufWriter::new(out.lock()); - self.write_help(&mut buf_w) - } - - /// Writes the full help message to the user to a `io::Write` object - /// - /// ```no_run - /// # use clap::App; - /// # use std::io; - /// let mut app = App::new("myprog"); - /// let mut out = io::stdout(); - /// app.write_help(&mut out).ok().expect("failed to write to stdout"); - /// ``` - pub fn write_help(&self, w: &mut W) -> ClapResult<()> { - if let Some(h) = self.help_str { - return writeln!(w, "{}", h).map_err(ClapError::from); - } - - // Print the version - try!(write!(w, - "{} {}\n", - &self.bin_name.as_ref().unwrap_or(&self.name)[..].replace(" ", "-"), - self.version.unwrap_or(""))); - let flags = !self.flags.is_empty(); - let pos = !self.positionals.is_empty(); - let opts = !self.opts.is_empty(); - let subcmds = !self.subcommands.is_empty(); - let unified_help = self.settings.is_set(&AppSettings::UnifiedHelpMessage); - - let mut longest_flag = 0; - for fl in self.flags - .iter() - .filter(|f| f.long.is_some() && !f.settings.is_set(&ArgSettings::Hidden)) - .map(|a| a.to_string().len()) { - if fl > longest_flag { - longest_flag = fl; - } - } - let mut longest_opt = 0; - for ol in self.opts - .iter() - .filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) - .map(|a| a.to_string().len()) { - if ol > longest_opt { - longest_opt = ol; - } - } - let mut longest_pos = 0; - for pl in self.positionals - .values() - .filter(|p| !p.settings.is_set(&ArgSettings::Hidden)) - .map(|f| f.to_string().len()) { - if pl > longest_pos { - longest_pos = pl; - } - } - let mut longest_sc = 0; - for scl in self.subcommands - .iter() - .filter(|s| !s.settings.is_set(&AppSettings::Hidden)) - .map(|f| f.name.len()) { - if scl > longest_sc { - longest_sc = scl; - } - } - - if let Some(author) = self.author { - try!(write!(w, "{}\n", author)); - } - if let Some(about) = self.about { - try!(write!(w, "{}\n", about)); - } - - try!(write!(w, "\n{}", try!(self.create_usage(&[])))); - - if flags || opts || pos || subcmds { - try!(write!(w, "\n")); - } - - let tab = " "; - let longest = if !unified_help || longest_opt == 0 { - longest_flag - } else { - longest_opt - }; - if unified_help && (flags || opts) { - try!(write!(w, "\nOPTIONS:\n")); - let mut combined = BTreeMap::new(); - for f in self.flags.iter().filter(|f| !f.settings.is_set(&ArgSettings::Hidden)) { - let mut v = vec![]; - try!(f.write_help(&mut v, tab, longest)); - combined.insert(f.name, v); - } - for o in self.opts.iter().filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) { - let mut v = vec![]; - try!(o.write_help(&mut v, tab, longest)); - combined.insert(o.name, v); - } - for (_, a) in combined { - try!(write!(w, "{}", String::from_utf8_lossy(&*a))); - } - } else { - if flags { - try!(write!(w, "\nFLAGS:\n")); - for (_, f) in self.flags - .iter() - .filter(|f| !f.settings.is_set(&ArgSettings::Hidden)) - .map(|f| (f.name, f)) - .collect::>() { - try!(f.write_help(w, tab, longest)); - } - } - if opts { - try!(write!(w, "\nOPTIONS:\n")); - for (_, o) in self.opts - .iter() - .filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) - .map(|o| (o.name, o)) - .collect::>() { - try!(o.write_help(w, tab, longest_opt)); - } - } - } - if pos { - try!(write!(w, "\nARGS:\n")); - for v in self.positionals - .values() - .filter(|p| !p.settings.is_set(&ArgSettings::Hidden)) { - try!(v.write_help(w, tab, longest_pos)); - } - } - if subcmds { - try!(write!(w, "\nSUBCOMMANDS:\n")); - for (name, sc) in self.subcommands - .iter() - .filter(|s| !s.settings.is_set(&AppSettings::Hidden)) - .map(|s| (s.name_slice, s)) - .collect::>() { - try!(write!(w, "{}{}", tab, name)); - write_spaces!((longest_sc + 4) - (name.len()), w); - if let Some(a) = sc.about { - if a.contains("{n}") { - let mut ab = a.split("{n}"); - while let Some(part) = ab.next() { - try!(write!(w, "{}\n", part)); - write_spaces!(longest_sc + 8, w); - try!(write!(w, "{}", ab.next().unwrap_or(""))); - } - } else { - try!(write!(w, "{}", a)); - } - } - try!(write!(w, "\n")); - } - } - - if let Some(h) = self.more_help { - try!(write!(w, "\n{}", h)); - } - - // flush the buffer - debugln!("Flushing the buffer..."); - w.flush().map_err(ClapError::from) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands - /// - /// # Panics - /// - /// If any arguments contain invalid unicode characters. If this is not desired it is - /// recommended to use the `*_safe()` or `*_lossy()` versions of this method. - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches(); - /// ``` - pub fn get_matches(self) -> ArgMatches<'ar, 'ar> { - // Start the parsing - self.get_matches_from(env::args()) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands. Invalid unicode characters are replaced with - /// `U+FFFD REPLACEMENT CHARACTER` - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches(); - /// ``` - pub fn get_matches_lossy(self) -> ArgMatches<'ar, 'ar> { - // Start the parsing - self.get_matches_from_lossy(env::args_os()) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands - /// - /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are - /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` - /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or - /// perform a `std::process::exit` yourself. - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_safe() - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_safe(self) -> ClapResult> { - // Start the parsing - self.get_matches_from_safe(env::args_os()) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands. Invalid unicode characters are replaced with - /// `U+FFFD REPLACEMENT CHARACTER` - /// - /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are - /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` - /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or - /// perform a `std::process::exit` yourself. - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_safe() - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_safe_lossy(self) -> ClapResult> { - // Start the parsing - self.get_matches_from_safe_lossy(env::args_os()) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches()` - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_from(arg_vec); - /// ``` - pub fn get_matches_from(mut self, itr: I) -> ArgMatches<'ar, 'ar> - where I: IntoIterator, - T: AsRef - { - self.get_matches_from_safe_borrow(itr).unwrap_or_else(|e| { - // Otherwise, write to stderr and exit - self.maybe_wait_for_exit(e); - }) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands. Invalid unicode characters are replaced with - /// `U+FFFD REPLACEMENT CHARACTER` - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches()` - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_from(arg_vec); - /// ``` - pub fn get_matches_from_lossy(mut self, itr: I) -> ArgMatches<'ar, 'ar> - where I: IntoIterator, - T: AsRef - { - self.get_matches_from_safe_borrow_lossy(itr).unwrap_or_else(|e| { - // Otherwise, write to stderr and exit - self.maybe_wait_for_exit(e); - }) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands - /// - /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are - /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` - /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or - /// perform a `std::process::exit` yourself. - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches_safe()` - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// **NOTE:** Invalid unicode characters will result in an `Err` with type - /// `ClapErrorType::InvalidUnicode` - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_from_safe(arg_vec) - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_from_safe(mut self, itr: I) -> ClapResult> - where I: IntoIterator, - T: AsRef - { - self.get_matches_from_safe_borrow(itr) - } - - /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls - /// the real parsing function for all subcommands. Invalid unicode characters are replaced with - /// `U+FFFD REPLACEMENT CHARACTER` - /// - /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are - /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` - /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or - /// perform a `std::process::exit` yourself. - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches_safe()` - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let matches = App::new("myprog") - /// // Args and options go here... - /// .get_matches_from_safe(arg_vec) - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_from_safe_lossy(mut self, itr: I) -> ClapResult> - where I: IntoIterator, - T: AsRef - { - self._get_matches_from_safe_borrow(itr, true) - } - - /// Starts the parsing process without consuming the `App` struct `self`. This is normally not - /// the desired functionality, instead prefer `App::get_matches_from_safe` which *does* - /// consume `self`. - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches_safe()` - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// **NOTE:** Invalid unicode characters will result in an `Err` with type - /// `ClapErrorType::InvalidUnicode` - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let mut app = App::new("myprog"); - /// // Args and options go here... - /// let matches = app.get_matches_from_safe_borrow(arg_vec) - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_from_safe_borrow(&mut self, itr: I) -> ClapResult> - where I: IntoIterator, - T: AsRef - { - self._get_matches_from_safe_borrow(itr, false) - } - - /// Starts the parsing process without consuming the `App` struct `self`. This is normally not - /// the desired functionality, instead prefer `App::get_matches_from_safe` which *does* - /// consume `self`. Invalid unicode characters are replaced with `U+FFFD REPLACEMENT CHARACTER` - /// - /// **NOTE:** The first argument will be parsed as the binary name. - /// - /// **NOTE:** This method should only be used when absolutely necessary, such as needing to - /// parse arguments from something other than `std::env::args()`. If you are unsure, use - /// `App::get_matches_safe()` - /// - /// **NOTE:** This method should only be used when is absolutely necessary to handle errors - /// manually. - /// - /// **NOTE:** Invalid unicode characters will result in an `Err` with type - /// `ClapErrorType::InvalidUnicode` - /// - /// # Examples - /// - /// ```no_run - /// # use clap::{App, Arg}; - /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; - /// - /// let mut app = App::new("myprog"); - /// // Args and options go here... - /// let matches = app.get_matches_from_safe_borrow(arg_vec) - /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); - /// ``` - pub fn get_matches_from_safe_borrow_lossy(&mut self, - itr: I) - -> ClapResult> - where I: IntoIterator, - T: AsRef - { - self._get_matches_from_safe_borrow(itr, true) - } - - fn _get_matches_from_safe_borrow(&mut self, - itr: I, - lossy: bool) - -> ClapResult> - where I: IntoIterator, - T: AsRef - { - // Verify all positional assertions pass - self.verify_positionals(); - // If there are global arguments, we need to propgate them down to subcommands - // before parsing incase we run into a subcommand - self.propogate_globals(); - - let mut matches = ArgMatches::new(); - - let mut it = itr.into_iter(); - // Get the name of the program (argument 1 of env::args()) and determine the - // actual file - // that was used to execute the program. This is because a program called - // ./target/release/my_prog -a - // will have two arguments, './target/release/my_prog', '-a' but we don't want - // to display - // the full path when displaying help messages and such - if !self.settings.is_set(&AppSettings::NoBinaryName) { - if let Some(name) = it.next() { - let p = Path::new(name.as_ref()); - if let Some(f) = p.file_name() { - if let Ok(s) = f.to_os_string().into_string() { - if let None = self.bin_name { - self.bin_name = Some(s); - } - } - } - } - } - - // do the real parsing - if let Err(e) = self.get_matches_with(&mut matches, &mut it, lossy) { - return Err(e); - } - - Ok(matches) - } - - // The actual parsing function - #[cfg_attr(feature="lints", allow(while_let_on_iterator))] - fn get_matches_with(&mut self, - matches: &mut ArgMatches<'ar, 'ar>, - it: &mut I, - lossy: bool) - -> ClapResult<()> - where I: Iterator, - T: AsRef - { - // First we create the `--help` and `--version` arguments and add them if - // necessary - self.create_help_and_version(); - - let mut pos_only = false; - let mut subcmd_name: Option = None; - let mut needs_val_of: Option<&str> = None; - let mut pos_counter = 1; - let mut val_counter = 0; - while let Some(arg) = it.next() { - let arg_cow = match arg.as_ref().to_str() { - Some(s) => s.into(), - None => { - if !lossy { - return Err( - error_builder::InvalidUnicode( - &*try!(self.create_current_usage(matches))) - ); - } - arg.as_ref().to_string_lossy() - } - }; - let arg_slice: &str = arg_cow.borrow(); - let mut skip = false; - - // we need to know if we're parsing a new argument, or the value of previous - // argument, - // perhaps one with multiple values such as --option val1 val2. We determine - // when to - // stop parsing multiple values by finding a '-' - let new_arg = if arg_slice.starts_with("-") { - // If we come to a single `-` it's a value, not a new argument...this happens - // when - // one wants to use the Unix standard of '-' to mean 'stdin' - !(arg_slice.len() == 1) - } else { - false - }; - - // pos_only is determined later, and set true when a user uses the Unix - // standard of - // '--' to mean only positionals follow - if !pos_only && !new_arg && - !self.subcommands.iter().any(|s| s.name_slice == arg_slice) { - // Check to see if parsing a value from an option - if let Some(nvo) = needs_val_of { - // get the OptBuilder so we can check the settings - if let Some(ref opt) = self.opts.iter().filter(|o| &o.name == &nvo).next() { - try!(self.validate_option(opt, arg_slice, matches)); - - if let Some(ref vec) = self.groups_for_arg(opt.name) { - for grp in vec { - if let Some(ref mut o) = matches.args.get_mut(grp) { - o.occurrences = if opt.settings - .is_set(&ArgSettings::Multiple) { - o.occurrences + 1 - } else { - 1 - }; - // Values must be inserted in order...the user may care about - // that! - if let Some(ref mut vals) = o.values { - let len = vals.len() as u8 + 1; - vals.insert(len, arg_slice.to_owned()); - } - } - - } - } - // save the value to matched option - if let Some(ref mut o) = matches.args.get_mut(opt.name) { - // if it's multiple; the occurrences are increased when originally - // found - o.occurrences = if opt.settings.is_set(&ArgSettings::Multiple) { - o.occurrences + 1 - } else { - skip = true; - 1 - }; - - // Options always have values, even if it's empty, so we can unwrap() - if let Some(ref mut vals) = o.values { - - // Values must be inserted in order...the user may care about that! - let len = vals.len() as u8 + 1; - vals.insert(len, arg_slice.to_owned()); - - // Now that the values have been added, we can ensure we haven't - // gone over any max_limits, or if we've reached the exact number - // of values we can stop parsing values, and go back to arguments. - // - // For example, if we define an option with exactly 2 values and - // the users passes: - // $ my_prog --option val1 val2 pos1 - // we stop parsing values of --option after val2, if the user - // hadn't defined an exact or max value, pos1 would be parsed as a - // value of --option - if let Some(num) = opt.max_vals { - if len != num { - continue; - } - } else if let Some(num) = opt.num_vals { - if opt.settings.is_set(&ArgSettings::Multiple) { - val_counter += 1; - if val_counter != num { - continue; - } else { - val_counter = 0; - } - } else { - // if we need more values, get the next value - if len != num { - continue; - } - } - } else if !skip { - // get the next value from the iterator - continue; - } - } - } - skip = true; - } - } - } - - // if we're done getting values from an option, get the next arg from the - // iterator - if skip { - needs_val_of = None; - continue; - } else if let Some(ref name) = needs_val_of { - // We've reached more values for an option than it possibly accepts - if let Some(ref o) = self.opts.iter().filter(|o| &o.name == name).next() { - if !o.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::EmptyValue( - &*o.to_string(), - &*try!(self.create_current_usage(matches)) - )); - } - } - } - - if arg_slice.starts_with("--") && !pos_only { - if arg_slice.len() == 2 { - // The user has passed '--' which means only positional args follow no matter - // what they start with - pos_only = true; - continue; - } - - // This arg is either an option or flag using a long (i.e. '--something') - match self.parse_long_arg(matches, arg_slice) { - Ok(r) => needs_val_of = r, - Err(e) => return Err(e), - } - } else if arg_slice.starts_with("-") && arg_slice.len() != 1 && !pos_only { - // Multiple or single flag(s), or single option (could be '-SbG' or '-o') - match self.parse_short_arg(matches, arg_slice) { - Ok(r) => needs_val_of = r, - Err(e) => return Err(e), - } - } else { - // Positional or Subcommand - // - // If the user passed `--` we don't check for subcommands, because the argument - // they - // may be trying to pass might match a subcommand name - if !pos_only { - if self.subcommands.iter().any(|s| s.name_slice == arg_slice) { - if arg_slice == "help" && - self.settings.is_set(&AppSettings::NeedsSubcommandHelp) { - try!(self.print_help()); - return Err(ClapError { - error: String::new(), - error_type: ClapErrorType::HelpDisplayed, - }); - } - subcmd_name = Some(arg_slice.to_owned()); - break; - } else if let Some(candidate_subcommand) = did_you_mean(arg_slice, - self.subcommands - .iter() - .map(|s| &s.name)) { - return Err(error_builder::InvalidSubcommand( - arg_slice, - candidate_subcommand, - self.bin_name.as_ref().unwrap_or(&self.name), - &*try!(self.create_current_usage(matches)))); - } - } - - // Did the developer even define any valid positionals? Since we reached this - // far, it's not a subcommand - if self.positionals.is_empty() { - return Err(error_builder::UnexpectedArgument( - arg_slice, - self.bin_name.as_ref().unwrap_or(&self.name), - &*try!(self.create_current_usage(matches)))); - } else if let Some(p) = self.positionals.get(&pos_counter) { - // Make sure this one doesn't conflict with anything - if self.blacklist.contains(&p.name) { - return Err(error_builder::ArgumentConflict( - p.to_string(), - self.blacklisted_from(p.name, &matches), - try!(self.create_current_usage(matches)))); - } - - if let Some(ref p_vals) = p.possible_vals { - if !p_vals.contains(&arg_slice) { - return Err(self.possible_values_error(arg_slice, - &p.to_string(), - p_vals, - matches)); - } - } - - // Have we made the update yet? - let mut done = false; - if p.settings.is_set(&ArgSettings::Multiple) { - if let Some(num) = p.num_vals { - if let Some(ref ma) = matches.args.get(p.name) { - if let Some(ref vals) = ma.values { - if vals.len() as u8 == num { - return Err(error_builder::TooManyValues( - arg_slice, - &*p.to_string(), - &*try!(self.create_current_usage(matches)))); - } - } - } - } - // Check if it's already existing and update if so... - if let Some(ref mut pos) = matches.args.get_mut(p.name) { - done = true; - pos.occurrences += 1; - if let Some(ref mut vals) = pos.values { - let len = (vals.len() + 1) as u8; - vals.insert(len, arg_slice.to_owned()); - } - } - - if !pos_only && - (self.settings.is_set(&AppSettings::TrailingVarArg) && - pos_counter == self.positionals.len()) { - pos_only = true; - } - } else { - // Only increment the positional counter if it doesn't allow multiples - pos_counter += 1; - } - // Was an update made, or is this the first occurrence? - if !done { - if self.overrides.contains(&p.name) { - if let Some(ref name) = self.overriden_from(p.name, matches) { - matches.args.remove(name); - remove_overriden!(self, name); - } - } - if let Some(ref or) = p.overrides { - for pa in or { - matches.args.remove(pa); - remove_overriden!(self, pa); - self.overrides.push(pa); - } - } - let mut bm = BTreeMap::new(); - if let Some(ref vtor) = p.validator { - let f = &*vtor; - if let Err(ref e) = f(arg_slice.to_owned()) { - return Err(error_builder::ValueValidationError(&*e)); - } - } - bm.insert(1, arg_slice.to_owned()); - if let Some(ref vec) = self.groups_for_arg(p.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: Some(bm.clone()), - }); - } - } - matches.args.insert(p.name, - MatchedArg { - occurrences: 1, - values: Some(bm), - }); - - if let Some(ref bl) = p.blacklist { - for name in bl { - self.blacklist.push(name); - } - } - - // Because of the macro call, we have to create a temp variable - if let Some(ref reqs) = p.requires { - // Add all required args which aren't already found in matches to the - // final required list - for n in reqs { - if matches.args.contains_key(n) { - continue; - } - - self.required.push(n); - } - } - - if p.settings.is_set(&ArgSettings::Required) { - // for macro call - let name = &p.name; - vec_remove!(self.required, name); - } - - parse_group_reqs!(self, p); - } - } else { - return Err(error_builder::UnexpectedArgument( - arg_slice, - self.bin_name.as_ref().unwrap_or(&self.name), - &*try!(self.create_current_usage(matches)))); - } - } - } - - if let Some(ref a) = needs_val_of { - if let Some(o) = self.opts.iter().filter(|o| &o.name == a).next() { - if o.settings.is_set(&ArgSettings::Multiple) && self.required.is_empty() { - let should_err = match matches.values_of(o.name) { - Some(ref v) => v.is_empty(), - None => true, - }; - if should_err { - return Err(error_builder::EmptyValue( - &*o.to_string(), - &*try!(self.create_current_usage(matches)) - )); - } - } else if !o.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::EmptyValue( - &*o.to_string(), - &*try!(self.create_current_usage(matches)) - )); - } else { - debugln!("Remaining Required Arg..."); - debugln!("required={:#?}", self.required); - return Err(error_builder::MissingRequiredArgument( - self.get_required_from(&self.required, Some(matches)) - .iter() - .fold(String::new(), - |acc, s| acc + &format!("\n\t{}", Format::Error(s))[..]), - try!(self.create_current_usage(matches)))); - } - } else { - return Err(error_builder::EmptyValue(&*format!("{}", - self.positionals - .values() - .filter(|p| &p.name == a) - .next() - .unwrap()), - &*try!(self.create_current_usage(matches)))); - } - } - - let res = self.validate_blacklist(matches); - if res.is_err() { - return res; - } - - let res = self.validate_num_args(matches); - if res.is_err() { - return res; - } - - matches.usage = Some(try!(self.create_usage(&[]))); - - if let Some(sc_name) = subcmd_name { - use std::fmt::Write; - let mut mid_string = String::new(); - if !self.settings.is_set(&AppSettings::SubcommandsNegateReqs) { - let mut hs = self.required.iter().map(|n| *n).collect::>(); - for k in matches.args.keys() { - hs.push(*k); - } - let reqs = self.get_required_from(&hs, Some(matches)); - - for s in reqs.iter() { - write!(&mut mid_string, " {}", s).ok().expect(INTERNAL_ERROR_MSG); - } - } - mid_string.push_str(" "); - if let Some(ref mut sc) = self.subcommands - .iter_mut() - .filter(|s| s.name_slice == &sc_name) - .next() { - let mut new_matches = ArgMatches::new(); - // bin_name should be parent's bin_name + [] + the sc's name separated by - // a space - sc.usage = Some(format!("{}{}{}", - self.bin_name.as_ref().unwrap_or(&String::new()), - if self.bin_name.is_some() { - &*mid_string - } else { - "" - }, - &*sc.name)); - sc.bin_name = Some(format!("{}{}{}", - self.bin_name.as_ref().unwrap_or(&String::new()), - if self.bin_name.is_some() { - " " - } else { - "" - }, - &*sc.name)); - if let Err(e) = sc.get_matches_with(&mut new_matches, it, lossy) { - e.exit(); - } - matches.subcommand = Some(Box::new(SubCommand { - name: sc.name_slice, - matches: new_matches, - })); - } - } else if self.settings.is_set(&AppSettings::SubcommandRequired) { - let bn = self.bin_name.as_ref().unwrap_or(&self.name); - return Err(error_builder::MissingSubcommand(bn, - &try!(self.create_current_usage(matches)))); - } else if self.settings.is_set(&AppSettings::SubcommandRequiredElseHelp) { - let mut out = vec![]; - try!(self.write_help(&mut out)); - return Err(ClapError { - error: String::from_utf8_lossy(&*out).into_owned(), - error_type: ClapErrorType::MissingArgumentOrSubcommand, - }); - } - if (!self.settings.is_set(&AppSettings::SubcommandsNegateReqs) || - matches.subcommand_name().is_none()) && self.validate_required(&matches) { - return Err(error_builder::MissingRequiredArgument( - &*self.get_required_from(&self.required, Some(matches)) - .iter() - .fold(String::new(), - |acc, s| acc + &format!("\n\t{}", Format::Error(s))[..]), - &*try!(self.create_current_usage(matches)))); - } - if matches.args.is_empty() && matches.subcommand_name().is_none() && - self.settings.is_set(&AppSettings::ArgRequiredElseHelp) { - let mut out = vec![]; - try!(self.write_help(&mut out)); - return Err(ClapError { - error: String::from_utf8_lossy(&*out).into_owned(), - error_type: ClapErrorType::MissingArgumentOrSubcommand, - }); - } - Ok(()) - } - - // basically re-implements ClapError::exit except it checks if we should wait - // for input before - // exiting since ClapError doesn't have that info and the error message must be - // printed before - // exiting - fn maybe_wait_for_exit(&self, e: ClapError) -> ! { - if e.use_stderr() { - wlnerr!("{}", e.error); - if self.settings.is_set(&AppSettings::WaitOnError) { - wlnerr!("\nPress [ENTER] / [RETURN] to continue..."); - let mut s = String::new(); - let i = io::stdin(); - i.lock().read_line(&mut s).unwrap(); - } - process::exit(1); - } - - e.exit() - } - - // Prints the version to the user and exits if quit=true - fn print_version(&self, w: &mut W) -> ClapResult<()> { - // Print the binary name if existing, but replace all spaces with hyphens in - // case we're - // dealing with subcommands i.e. git mv is translated to git-mv - try!(writeln!(w, - "{} {}", - &self.bin_name.as_ref().unwrap_or(&self.name)[..].replace(" ", "-"), - self.version.unwrap_or(""))); - - w.flush().map_err(ClapError::from) - } - - fn create_current_usage(&self, matches: &ArgMatches) -> ClapResult { - self.create_usage(&matches.args - .keys() - .filter(|k| { - if let Some(o) = self.opts - .iter() - .filter(|o| &&o.name == k) - .next() { - !o.settings.is_set(&ArgSettings::Required) - } else if let Some(p) = self.positionals - .values() - .filter(|p| &&p.name == k) - .next() { - !p.settings.is_set(&ArgSettings::Required) - } else { - true - } - }) - .map(|k| *k) - .collect::>()) - } - - fn groups_for_arg(&self, name: &str) -> Option> { - if self.groups.is_empty() { - return None; - } - let mut res = vec![]; - for (g_name, grp) in &self.groups { - for a in &grp.args { - if a == &name { - res.push(*g_name); - } - } - } - if res.is_empty() { - return None; - } - - Some(res) - } - - fn args_in_group(&self, group: &str) -> Vec { - let mut g_vec = vec![]; - let mut args = vec![]; - - for n in &self.groups.get(group).unwrap().args { - if let Some(f) = self.flags.iter().filter(|f| &f.name == n).next() { - args.push(f.to_string()); - } else if let Some(f) = self.opts.iter().filter(|o| &o.name == n).next() { - args.push(f.to_string()); - } else if self.groups.contains_key(n) { - g_vec.push(*n); - } else { - if let Some(p) = self.positionals - .values() - .filter(|p| &p.name == n) - .next() { - args.push(p.to_string()); - } - } - } - - if !g_vec.is_empty() { - for av in g_vec.iter().map(|g| self.args_in_group(g)) { - for a in av { - args.push(a); - } - } - } - assert!(!args.is_empty(), - "ArgGroup '{}' doesn't contain any args", - group); - args.dedup(); - args.iter().map(ToOwned::to_owned).collect() - } - - fn arg_names_in_group(&self, group: &'ar str) -> Vec<&'ar str> { - let mut g_vec = vec![]; - let mut args = vec![]; - - for n in &self.groups.get(group).unwrap().args { - if self.flags.iter().any(|f| &f.name == n) { - args.push(*n); - } else if self.opts.iter().any(|o| &o.name == n) { - args.push(*n); - } else if self.groups.contains_key(n) { - g_vec.push(*n); - } else if self.positionals.values().any(|p| &p.name == n) { - args.push(*n); - } - } - - if !g_vec.is_empty() { - for av in g_vec.iter().map(|g| self.arg_names_in_group(g)) { - for a in av { - args.push(a); - } - } - } - assert!(!args.is_empty(), - "ArgGroup '{}' doesn't contain any args", - group); - args.dedup(); - args.iter().map(|s| *s).collect() - } - - fn get_required_from(&self, - reqs: &[&'ar str], - matches: Option<&ArgMatches>) - -> VecDeque { - let mut c_flags = vec![]; - let mut c_pos = vec![]; - let mut c_opt = vec![]; - let mut grps = vec![]; - for name in reqs { - if self.flags.iter().any(|f| &f.name == name) { - c_flags.push(name); - } else if self.opts.iter().any(|o| &o.name == name) { - c_opt.push(name); - } else if self.groups.contains_key(name) { - grps.push(*name); - } else { - c_pos.push(name); - } - } - let mut tmp_f = vec![]; - for f in &c_flags { - if let Some(f) = self.flags.iter().filter(|flg| &&flg.name == f).next() { - if let Some(ref rl) = f.requires { - for r in rl.iter() { - if !reqs.contains(r) { - if self.flags.iter().any(|f| &f.name == r) { - tmp_f.push(r); - } else if self.opts.iter().any(|o| &o.name == r) { - c_opt.push(r); - } else if self.groups.contains_key(r) { - grps.push(*r); - } else { - c_pos.push(r); - } - } - } - } - } - } - for f in tmp_f.into_iter() { - c_flags.push(f); - } - let mut tmp_o = vec![]; - for f in &c_opt { - if let Some(f) = self.opts.iter().filter(|o| &&o.name == f).next() { - if let Some(ref rl) = f.requires { - for r in rl.iter() { - if !reqs.contains(r) { - if self.flags.iter().any(|f| &f.name == r) { - c_flags.push(r); - } else if self.opts.iter().any(|o| &o.name == r) { - tmp_o.push(r); - } else if self.groups.contains_key(r) { - grps.push(*r); - } else { - c_pos.push(r); - } - } - } - } - } - } - for f in tmp_o.into_iter() { - c_opt.push(f); - } - let mut tmp_p = vec![]; - for p in c_pos.iter() { - if let Some(p) = self.positionals.values().filter(|pos| &&pos.name == p).next() { - if let Some(ref rl) = p.requires { - for r in rl.iter() { - if !reqs.contains(r) { - if self.flags.iter().any(|f| &f.name == r) { - c_flags.push(r); - } else if self.opts.iter().any(|o| &o.name == r) { - c_opt.push(r); - } else if self.groups.contains_key(r) { - grps.push(*r); - } else { - tmp_p.push(r); - } - } - } - } - } - } - for f in tmp_p.into_iter() { - c_pos.push(f); - } - - let mut ret_val = VecDeque::new(); - - let mut pmap = BTreeMap::new(); - for p in c_pos.into_iter() { - if matches.is_some() && matches.as_ref().unwrap().is_present(p) { - continue; - } - if let Some(p) = self.positionals.values().filter(|x| &x.name == p).next() { - pmap.insert(p.index, format!("{}", p)); - } - } - for (_, s) in pmap { - ret_val.push_back(s); - } - for f in c_flags.into_iter() { - if matches.is_some() && matches.as_ref().unwrap().is_present(f) { - continue; - } - ret_val.push_back(format!("{}", - self.flags - .iter() - .filter(|flg| &flg.name == f) - .next() - .unwrap())); - } - for o in c_opt.into_iter() { - if matches.is_some() && matches.as_ref().unwrap().is_present(o) { - continue; - } - ret_val.push_back(format!("{}", - self.opts - .iter() - .filter(|opt| &opt.name == o) - .next() - .unwrap())); - } - for g in grps.into_iter() { - let g_string = self.args_in_group(g) - .iter() - .fold(String::new(), |acc, s| acc + &format!(" {} |", s)[..]); - ret_val.push_back(format!("[{}]", &g_string[..g_string.len() - 1])); - } - - ret_val - } - - fn verify_positionals(&mut self) { - // Because you must wait until all arguments have been supplied, this is the first chance - // to make assertions on positional argument indexes - // - // Firt we verify that the index highest supplied index, is equal to the number of - // positional arguments to verify there are no gaps (i.e. supplying an index of 1 and 3 - // but no 2) - if let Some((idx, ref p)) = self.positionals.iter().rev().next() { - if idx != self.positionals.len() { - panic!("Found positional argument \"{}\" who's index is {} but there are only {} \ - positional arguments defined", - p.name, - idx, - self.positionals.len()); - } - } - - // Next we verify that only the highest index has a .multiple(true) (if any) - assert!(!self.positionals - .values() - .any(|a| { - a.settings.is_set(&ArgSettings::Multiple) && - (a.index as usize != self.positionals.len()) - }), - "Only the positional argument with the highest index may accept multiple values"); - - // If it's required we also need to ensure all previous positionals are - // required too - let mut found = false; - for p in self.positionals.values().rev() { - if !found { - if p.settings.is_set(&ArgSettings::Required) { - found = true; - continue; - } - } else { - assert!(p.settings.is_set(&ArgSettings::Required), - "Found positional argument which is not required with a lower index than a \ - required positional argument: {:?} index {}", - p.name, - p.index); - } - } - } - - fn propogate_globals(&mut self) { - for sc in self.subcommands.iter_mut() { - // We have to create a new scope in order to tell rustc the borrow of `sc` is - // done and - // to recursively call this method - { - for a in self.global_args.iter() { - sc.add_arg(a.into()); - } - } - sc.propogate_globals(); - } - } - - - fn possible_values_error(&self, - arg: &str, - opt: &str, - p_vals: &[&str], - matches: &ArgMatches<'ar, 'ar>) - -> ClapError { - let suffix = App::did_you_mean_suffix(arg, - p_vals.iter(), - DidYouMeanMessageStyle::EnumValue); - - let mut sorted = vec![]; - for v in p_vals { - sorted.push(v); - } - sorted.sort(); - let valid_values = sorted.iter() - .fold(String::new(), |acc, name| acc + &format!(" {}", name)[..]); - let usage = match self.create_current_usage(matches) { - Ok(s) => s, - Err(e) => return e, - }; - error_builder::InvalidValue(arg, opt, &*valid_values, &*suffix.0, &*usage) - } - - fn blacklisted_from(&self, name: &str, matches: &ArgMatches) -> Option { - for k in matches.args.keys() { - if let Some(f) = self.flags.iter().filter(|f| &f.name == k).next() { - if let Some(ref bl) = f.blacklist { - if bl.contains(&name) { - return Some(format!("{}", f)); - } - } - } - if let Some(o) = self.opts.iter().filter(|o| &o.name == k).next() { - if let Some(ref bl) = o.blacklist { - if bl.contains(&name) { - return Some(format!("{}", o)); - } - } - } - if let Some(pos) = self.positionals.values().filter(|p| &p.name == k).next() { - if let Some(ref bl) = pos.blacklist { - if bl.contains(&name) { - return Some(format!("{}", pos)); - } - } - } - } - None - } - - fn overriden_from(&self, name: &'ar str, matches: &ArgMatches) -> Option<&'ar str> { - for k in matches.args.keys() { - if let Some(f) = self.flags.iter().filter(|f| &f.name == k).next() { - if let Some(ref bl) = f.overrides { - if bl.contains(&name) { - return Some(f.name); - } - } - } - if let Some(o) = self.opts.iter().filter(|o| &o.name == k).next() { - if let Some(ref bl) = o.overrides { - if bl.contains(&name) { - return Some(o.name); - } - } - } - if let Some(pos) = self.positionals.values().filter(|p| &p.name == k).next() { - if let Some(ref bl) = pos.overrides { - if bl.contains(&name) { - return Some(pos.name); - } - } - } - } - None - } - - fn create_help_and_version(&mut self) { - // name is "hclap_help" because flags are sorted by name - if !self.flags.iter().any(|a| a.long.is_some() && a.long.unwrap() == "help") { - if self.help_short.is_none() && !self.short_list.contains(&'h') { - self.help_short = Some('h'); - } - let arg = FlagBuilder { - name: "hclap_help", - short: self.help_short, - long: Some("help"), - help: Some("Prints help information"), - blacklist: None, - requires: None, - overrides: None, - settings: ArgFlags::new(), - }; - self.long_list.push("help"); - self.flags.push(arg); - } - if !self.settings.is_set(&AppSettings::DisableVersion) && - !self.flags.iter().any(|a| a.long.is_some() && a.long.unwrap() == "version") { - if self.version_short.is_none() && !self.short_list.contains(&'V') { - self.version_short = Some('V'); - } - // name is "vclap_version" because flags are sorted by name - let arg = FlagBuilder { - name: "vclap_version", - short: self.version_short, - long: Some("version"), - help: Some("Prints version information"), - blacklist: None, - requires: None, - overrides: None, - settings: ArgFlags::new(), - }; - self.long_list.push("version"); - self.flags.push(arg); - } - if !self.subcommands.is_empty() && - !self.subcommands - .iter() - .any(|a| a.name_slice == "help") { - self.subcommands.push(App::new("help").about("Prints this message")); - } - } - - fn check_for_help_and_version(&self, arg: char) -> ClapResult<()> { - if let Some(h) = self.help_short { - if h == arg { - try!(self.print_help()); - return Err(ClapError { - error: String::new(), - error_type: ClapErrorType::HelpDisplayed, - }); - } - } - if let Some(v) = self.version_short { - if v == arg { - let out = io::stdout(); - let mut buf_w = BufWriter::new(out.lock()); - try!(self.print_version(&mut buf_w)); - return Err(ClapError { - error: String::new(), - error_type: ClapErrorType::VersionDisplayed, - }); - } - } - - Ok(()) - } - - fn parse_long_arg<'av>(&mut self, - matches: &mut ArgMatches<'ar, 'ar>, - full_arg: &'av str) - -> ClapResult> { - let mut arg = full_arg.trim_left_matches(|c| c == '-'); - - if arg == "help" && self.settings.is_set(&AppSettings::NeedsLongHelp) { - try!(self.print_help()); - return Err(ClapError { - error: String::new(), - error_type: ClapErrorType::HelpDisplayed, - }); - } else if arg == "version" && self.settings.is_set(&AppSettings::NeedsLongVersion) { - let out = io::stdout(); - let mut buf_w = BufWriter::new(out.lock()); - try!(self.print_version(&mut buf_w)); - return Err(ClapError { - error: String::new(), - error_type: ClapErrorType::VersionDisplayed, - }); - } - - let mut arg_val: Option<&'av str> = None; - - if arg.contains("=") { - let arg_vec: Vec<_> = arg.split("=").collect(); - arg = arg_vec[0]; - if let Some(ref v) = self.opts - .iter() - .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) - .next() { - // prevents "--config= value" typo - if arg_vec[1].len() == 0 && !v.settings.is_set(&ArgSettings::EmptyValues) { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: None, - }); - } - } - matches.args.insert(v.name, - MatchedArg { - occurrences: 1, - values: None, - }); - return Err(error_builder::EmptyValue( - &*format!("--{}", arg), - &*try!(self.create_current_usage(matches)) - )); - } - arg_val = Some(arg_vec[1]); - } - } - - if let Some(ref v) = self.opts - .iter() - .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) - .next() { - // Ensure this option isn't on the master mutually excludes list - if self.blacklist.contains(&v.name) { - matches.args.remove(v.name); - return Err(error_builder::ArgumentConflict( - format!("--{}", arg), - self.blacklisted_from(v.name, &matches), - try!(self.create_current_usage(matches)))); - } - if self.overrides.contains(&v.name) { - debugln!("it is..."); - debugln!("checking who defined it..."); - if let Some(ref name) = self.overriden_from(v.name, matches) { - debugln!("found {}", name); - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.remove(grp); - } - } - matches.args.remove(name); - remove_overriden!(self, name); - } - } - if let Some(ref or) = v.overrides { - for pa in or { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.remove(grp); - } - } - matches.args.remove(pa); - remove_overriden!(self, pa); - self.overrides.push(pa); - } - } - - if matches.args.contains_key(v.name) { - if !v.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::UnexpectedMultipleUsage( - &*format!("--{}", arg), - &*try!(self.create_current_usage(matches)))); - } - if let Some(av) = arg_val { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - if let Some(ref mut o) = matches.args.get_mut(grp) { - o.occurrences += 1; - if let Some(ref mut vals) = o.values { - let len = (vals.len() + 1) as u8; - vals.insert(len, av.to_owned()); - } - } - } - } - if let Some(ref mut o) = matches.args.get_mut(v.name) { - o.occurrences += 1; - if let Some(ref mut vals) = o.values { - let len = (vals.len() + 1) as u8; - vals.insert(len, av.to_owned()); - } - } - // The validation must come AFTER inserting into 'matches' or the usage string - // can't be built - if let Err(e) = self.validate_value(v, av, matches) { - return Err(e); - } - } - } else { - let mut bm = BTreeMap::new(); - if let Some(val) = arg_val { - bm.insert(1, val.to_owned()); - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: Some(bm.clone()), - }); - } - } - matches.args.insert(v.name, - MatchedArg { - occurrences: 1, - values: Some(bm), - }); - // The validation must come AFTER inserting into 'matches' or the usage string - // can't be built - if let Err(e) = self.validate_value(v, val, matches) { - return Err(e); - } - } else { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: Some(bm.clone()), - }); - } - } - matches.args.insert(v.name, - MatchedArg { - occurrences: 0, - values: Some(bm), - }); - } - } - if let Some(ref bl) = v.blacklist { - for name in bl { - self.blacklist.push(name); - vec_remove!(self.overrides, name); - vec_remove!(self.required, name); - } - } - - if let Some(ref reqs) = v.requires { - // Add all required args which aren't already found in matches to the - // final required list - for n in reqs { - if matches.args.contains_key(n) { - continue; - } - - self.required.push(n); - } - } - - parse_group_reqs!(self, v); - - match arg_val { - None => { - return Ok(Some(v.name)); - } - _ => { - return Ok(None); - } - } - } - - if let Some(v) = self.flags - .iter() - .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) - .next() { - // Ensure this flag isn't on the mutually excludes list - if self.blacklist.contains(&v.name) { - matches.args.remove(v.name); - return Err(error_builder::ArgumentConflict( - v.to_string(), - self.blacklisted_from(v.name, &matches), - try!(self.create_current_usage(matches)))); - } - if self.overrides.contains(&v.name) { - debugln!("it is..."); - debugln!("checking who defined it..."); - if let Some(ref name) = self.overriden_from(v.name, matches) { - debugln!("found {}", name); - matches.args.remove(name); - remove_overriden!(self, name); - } - } - if let Some(ref or) = v.overrides { - for pa in or { - matches.args.remove(pa); - remove_overriden!(self, pa); - self.overrides.push(pa); - } - } - - // Make sure this isn't one being added multiple times if it doesn't suppor it - if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::UnexpectedMultipleUsage( - &*v.to_string(), - &*try!(self.create_current_usage(matches)))); - } - - let mut done = false; - if let Some(ref mut f) = matches.args.get_mut(v.name) { - done = true; - f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { - f.occurrences + 1 - } else { - 1 - }; - } - if !done { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: None, - }); - } - } - matches.args.insert(v.name, - MatchedArg { - occurrences: 1, - values: None, - }); - } - - - // Add all of this flags "mutually excludes" list to the master list - if let Some(ref ov) = v.overrides { - for name in ov { - self.overrides.push(name); - } - } - if let Some(ref bl) = v.blacklist { - for name in bl { - self.blacklist.push(name); - vec_remove!(self.overrides, name); - vec_remove!(self.required, name); - } - } - - // Add all required args which aren't already found in matches to the master - // list - if let Some(ref reqs) = v.requires { - for n in reqs { - if matches.args.contains_key(n) { - continue; - } - - self.required.push(n); - } - } - - parse_group_reqs!(self, v); - - return Ok(None); - } - - let suffix = App::did_you_mean_suffix(arg, - self.long_list.iter(), - DidYouMeanMessageStyle::LongFlag); - if let Some(name) = suffix.1 { - if let Some(ref opt) = self.opts - .iter() - .filter_map(|o| { - if o.long.is_some() && o.long.unwrap() == name { - Some(o.name) - } else { - None - } - }) - .next() { - if let Some(ref vec) = self.groups_for_arg(opt) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: None, - }); - } - } - matches.args.insert(opt, - MatchedArg { - occurrences: 0, - values: None, - }); - } else if let Some(ref flg) = self.flags - .iter() - .filter_map(|f| { - if f.long.is_some() && f.long.unwrap() == name { - Some(f.name) - } else { - None - } - }) - .next() { - if let Some(ref vec) = self.groups_for_arg(flg) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: None, - }); - } - } - matches.args.insert(flg, - MatchedArg { - occurrences: 0, - values: None, - }); - } - } - - Err(error_builder::InvalidArgument(&*format!("--{}", arg), - Some(&*suffix.0), - &*try!(self.create_current_usage(matches)))) - } - - fn validate_value(&self, v: &OptBuilder, av: &str, matches: &ArgMatches) -> ClapResult<()> { - if let Some(ref p_vals) = v.possible_vals { - if !p_vals.contains(&av) { - return Err(self.possible_values_error(av, &v.to_string(), p_vals, matches)); - } - } - if !v.settings.is_set(&ArgSettings::EmptyValues) && av.is_empty() && - matches.args.contains_key(v.name) { - return Err(error_builder::EmptyValue(&*v.to_string(), - &*try!(self.create_current_usage(matches)))); - } - if let Some(ref vtor) = v.validator { - if let Err(e) = vtor(av.to_owned()) { - return Err(error_builder::ValueValidationError(&*e)); - } - } - Ok(()) - } - - fn parse_short_arg(&mut self, - matches: &mut ArgMatches<'ar, 'ar>, - full_arg: &str) - -> ClapResult> { - let arg = &full_arg[..].trim_left_matches(|c| c == '-'); - for c in arg.chars() { - try!(self.check_for_help_and_version(c)); - - // Check for matching short in options, and return the name - // (only ones with shorts, of course) - if let Some(v) = self.opts - .iter() - .filter(|&v| v.short.is_some() && v.short == Some(c)) - .next() { - let mut ret = Some(v.name); - // Ensure this option isn't on the master mutually excludes list - if self.blacklist.contains(&v.name) { - matches.args.remove(v.name); - return Err(error_builder::ArgumentConflict( - v.to_string(), - self.blacklisted_from(v.name, &matches), - try!(self.create_current_usage(matches)))); - } - if self.overrides.contains(&v.name) { - if let Some(ref name) = self.overriden_from(v.name, matches) { - matches.args.remove(&*name); - remove_overriden!(self, name); - } - } - if let Some(ref or) = v.overrides { - for pa in or { - matches.args.remove(pa); - remove_overriden!(self, pa); - self.overrides.push(pa); - } - } - - if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::UnexpectedMultipleUsage( - &*format!("-{}", arg), - &*try!(self.create_current_usage(matches)))); - } - - // New scope for lifetimes - let val: Vec<&str> = arg.splitn(2, c).collect(); - { - let ma = matches.args.entry(v.name).or_insert(MatchedArg { - // occurrences will be incremented on getting a value - occurrences: 0, - values: Some(BTreeMap::new()), - }); - if !val[1].is_empty() { - if !v.settings.is_set(&ArgSettings::Multiple) { - ret = None; - } - if let Some(ref mut vals) = ma.values { - let len = vals.len() as u8 + 1; - vals.insert(len, val[1].to_owned()); - } - ma.occurrences += 1; - } - } - - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - let ma_g = matches.args.entry(grp).or_insert(MatchedArg { - occurrences: 0, - values: Some(BTreeMap::new()), - }); - if !val[1].is_empty() { - if let Some(ref mut vals) = ma_g.values { - let len = vals.len() as u8 + 1; - vals.insert(len, val[1].to_owned()); - } - ma_g.occurrences += 1; - } - } - } - - if let Some(ref bl) = v.blacklist { - for name in bl { - self.blacklist.push(name); - vec_remove!(self.overrides, name); - vec_remove!(self.required, name); - } - } - - if let Some(ref reqs) = v.requires { - // Add all required args which aren't already found in matches to the - // final required list - for n in reqs { - if matches.args.contains_key(n) { - continue; - } - - self.required.push(n); - } - } - - parse_group_reqs!(self, v); - - return Ok(ret); - } - - match self.parse_single_short_flag(matches, c) { - Ok(b) => { - if !b { - return Err(error_builder::InvalidArgument( - &*format!("-{}", c), - None, - &*try!(self.create_current_usage(matches)))); - } - } - Err(e) => return Err(e), - } - } - Ok(None) - } - - fn parse_single_short_flag(&mut self, - matches: &mut ArgMatches<'ar, 'ar>, - arg: char) - -> ClapResult { - if let Some(v) = self.flags - .iter() - .filter(|&v| v.short.is_some() && v.short.unwrap() == arg) - .next() { - // Ensure this flag isn't on the mutually excludes list - if self.blacklist.contains(&v.name) { - matches.args.remove(v.name); - return Err(error_builder::ArgumentConflict( - v.to_string(), - self.blacklisted_from(v.name, &matches), - try!(self.create_current_usage(matches)))); - } - if self.overrides.contains(&v.name) { - debugln!("it is..."); - debugln!("checking who defined it..."); - if let Some(ref name) = self.overriden_from(v.name, matches) { - debugln!("found {}", name); - matches.args.remove(name); - remove_overriden!(self, name); - } - } - if let Some(ref or) = v.overrides { - for pa in or { - matches.args.remove(pa); - remove_overriden!(self, pa); - self.overrides.push(pa); - } - } - - // Make sure this isn't one being added multiple times if it doesn't suppor it - if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::UnexpectedMultipleUsage( - &*format!("-{}", arg), - &*try!(self.create_current_usage(matches)))); - } - - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - if let Some(ref mut f) = matches.args.get_mut(grp) { - f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { - f.occurrences + 1 - } else { - 1 - }; - } - } - } - let mut done = false; - if let Some(ref mut f) = matches.args.get_mut(v.name) { - done = true; - f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { - f.occurrences + 1 - } else { - 1 - }; - } - if !done { - if let Some(ref vec) = self.groups_for_arg(v.name) { - for grp in vec { - matches.args.insert(grp, - MatchedArg { - occurrences: 1, - values: None, - }); - } - } - matches.args.insert(v.name, - MatchedArg { - occurrences: 1, - values: None, - }); - } - - if let Some(ref bl) = v.blacklist { - for name in bl { - self.blacklist.push(name); - vec_remove!(self.overrides, name); - vec_remove!(self.required, name); - } - } - - // Add all required args which aren't already found in matches to the master - // list - if let Some(ref reqs) = v.requires { - for n in reqs { - if matches.args.contains_key(n) { - continue; - } - - self.required.push(n); - } - } - - parse_group_reqs!(self, v); - - return Ok(true); - } - Ok(false) - } - - fn validate_blacklist(&self, matches: &mut ArgMatches<'ar, 'ar>) -> ClapResult<()> { - for name in self.blacklist.iter() { - if matches.args.contains_key(name) { - matches.args.remove(name); - return Err(error_builder::ArgumentConflict( - format!("{}", Format::Warning( - if let Some(f) = self.flags.iter().filter(|f| &f.name == name).next() { - f.to_string() - } else if let Some(o) = self.opts.iter() - .filter(|o| &o.name == name) - .next() { - o.to_string() - } else { - match self.positionals.values() - .filter(|p| p.name == *name) - .next() { - Some(p) => p.to_string(), - None => format!("'{}'", name) - } - } - )), - self.blacklisted_from(name, &matches), - try!(self.create_current_usage(matches)))); - } else if self.groups.contains_key(name) { - for n in self.arg_names_in_group(name) { - if matches.args.contains_key(n) { - matches.args.remove(n); - return Err(error_builder::ArgumentConflict( - format!("{}", Format::Warning( - if let Some(f) = self.flags.iter() - .filter(|f| &f.name == name) - .next() { - f.to_string() - } else if let Some(o) = self.opts - .iter() - .filter(|o| &o.name == name) - .next() { - o.to_string() - } else { - match self.positionals.values() - .filter(|p| p.name == n) - .next() { - Some(p) => p.to_string(), - None => format!("'{}'", name) - } - } - )), - self.blacklisted_from(name, &matches), - try!(self.create_current_usage(matches)))); - } - } - } - } - Ok(()) - } - - fn validate_num_args(&self, matches: &mut ArgMatches<'ar, 'ar>) -> ClapResult<()> { - for (name, ma) in matches.args.iter() { - if self.groups.contains_key(name) { - continue; - } else if let Some(ref vals) = ma.values { - if let Some(f) = self.opts - .iter() - .filter(|o| &o.name == name) - .next() { - if let Some(num) = f.num_vals { - let should_err = if f.settings.is_set(&ArgSettings::Multiple) { - ((vals.len() as u8) % num) != 0 - } else { - num != (vals.len() as u8) - }; - if should_err { - return Err(error_builder::WrongNumValues( - &*f.to_string(), - num, - if f.settings.is_set(&ArgSettings::Multiple) { - (vals.len() % num as usize) - } else { - vals.len() - }, - if vals.len() == 1 || - (f.settings.is_set(&ArgSettings::Multiple) && - (vals.len() % num as usize) == 1) { - "as" - } else { - "ere" - }, - &*try!(self.create_current_usage(matches)))); - } - } - if let Some(num) = f.max_vals { - if (vals.len() as u8) > num { - return Err(error_builder::TooManyValues( - vals.get(vals.keys() - .last() - .expect("error getting last key. This is a bug")) - .expect("failed to retrieve last value. This is a bug"), - &f.to_string(), - &try!(self.create_current_usage(matches)))); - } - } - if let Some(num) = f.min_vals { - if (vals.len() as u8) < num { - return Err(error_builder::TooFewValues( - &*f.to_string(), - num, - vals.len(), - &*try!(self.create_current_usage(matches)))); - } - } - } else if let Some(f) = self.positionals - .values() - .filter(|p| &p.name == name) - .next() { - if let Some(num) = f.num_vals { - if num != vals.len() as u8 { - return Err(error_builder::WrongNumValues( - &*f.to_string(), - num, - vals.len(), - if vals.len() == 1 { - "as" - } else { - "ere" - }, - &*try!(self.create_current_usage(matches)))); - } - } - if let Some(max) = f.max_vals { - if (vals.len() as u8) > max { - return Err(error_builder::TooManyValues( - vals.get(vals.keys() - .last() - .expect("error getting last key. This is a bug")) - .expect("failed to retrieve last value. This is a bug"), - &f.to_string(), - &try!(self.create_current_usage(matches)))); - } - } - if let Some(min) = f.min_vals { - if (vals.len() as u8) < min { - return Err(error_builder::TooFewValues( - &*f.to_string(), - min, - vals.len(), - &*try!(self.create_current_usage(matches)))); - } - } - } - } - } - Ok(()) - } - - fn validate_required(&self, matches: &ArgMatches<'ar, 'ar>) -> bool { - 'outer: for name in self.required.iter() { - if matches.args.contains_key(name) { - continue 'outer; - } - for grp in self.groups.values() { - if grp.args.contains(name) { - continue 'outer; - } - } - if let Some(a) = self.flags.iter().filter(|f| &f.name == name).next() { - if let Some(ref bl) = a.blacklist { - for n in bl.iter() { - if matches.args.contains_key(n) { - continue 'outer; - } else if self.groups.contains_key(n) { - let grp = self.groups.get(n).unwrap(); - for an in grp.args.iter() { - if matches.args.contains_key(an) { - continue 'outer; - } - } - } - } - } - } - if let Some(a) = self.opts.iter().filter(|o| &o.name == name).next() { - if let Some(ref bl) = a.blacklist { - for n in bl.iter() { - if matches.args.contains_key(n) { - continue 'outer; - } else if self.groups.contains_key(n) { - let grp = self.groups.get(n).unwrap(); - for an in grp.args.iter() { - if matches.args.contains_key(an) { - continue 'outer; - } - } - } - } - } - } - // because positions use different keys, we dont use the macro - match self.positionals.values().filter(|p| &p.name == name).next() { - Some(p) => { - if let Some(ref bl) = p.blacklist { - for n in bl.iter() { - if matches.args.contains_key(n) { - continue 'outer; - } else if self.groups.contains_key(n) { - let grp = self.groups.get(n).unwrap(); - for an in grp.args.iter() { - if matches.args.contains_key(an) { - continue 'outer; - } - } - } - } - } - } - None => (), - } - return true; - } - false - } - - /// Returns a suffix that can be empty, or is the standard 'did you mean phrase - #[cfg_attr(feature = "lints", allow(needless_lifetimes))] - fn did_you_mean_suffix<'z, T, I>(arg: &str, - values: I, - style: DidYouMeanMessageStyle) - -> (String, Option<&'z str>) - where T: AsRef + 'z, - I: IntoIterator - { - match did_you_mean(arg, values) { - Some(candidate) => { - let mut suffix = "\n\tDid you mean ".to_owned(); - match style { - DidYouMeanMessageStyle::LongFlag => - suffix.push_str(&*format!("{}", Format::Good("--"))), - DidYouMeanMessageStyle::EnumValue => suffix.push('\''), - } - suffix.push_str(&Format::Good(candidate).to_string()[..]); - if let DidYouMeanMessageStyle::EnumValue = style { - suffix.push('\''); - } - suffix.push_str(" ?"); - (suffix, Some(candidate)) - } - None => (String::new(), None), - } - } - - fn validate_option(&self, - opt: &OptBuilder, - arg_slice: &str, - matches: &ArgMatches) - -> ClapResult<()> { - // Check the possible values - if let Some(ref p_vals) = opt.possible_vals { - if !p_vals.contains(&arg_slice) { - return Err(self.possible_values_error(arg_slice, - &opt.to_string(), - p_vals, - matches)); - } - } - // Check the required number of values - if let Some(num) = opt.num_vals { - if let Some(ref ma) = matches.args.get(opt.name) { - if let Some(ref vals) = ma.values { - if (vals.len() as u8) > num && !opt.settings.is_set(&ArgSettings::Multiple) { - return Err(error_builder::TooManyValues( - vals.get(vals.keys() - .last() - .expect("error getting last key. This is a bug")) - .expect("failed to retrieve last value. This is a bug"), - &opt.to_string(), - &try!(self.create_current_usage(matches)))); - } - } - } - } - - // if it's an empty value, and we don't allow that, report the error - if !opt.settings.is_set(&ArgSettings::EmptyValues) && - matches.args.contains_key(opt.name) && arg_slice.is_empty() { - return Err(error_builder::EmptyValue(&*opt.to_string(), - &*try!(self.create_current_usage(matches)))); - } - - if let Some(ref vtor) = opt.validator { - if let Err(e) = vtor(arg_slice.to_owned()) { - return Err(error_builder::ValueValidationError(&*e)); - } - } - - Ok(()) - } -} diff --git a/src/app/mod.rs b/src/app/mod.rs index c2357ee9..9711ba56 100644 --- a/src/app/mod.rs +++ b/src/app/mod.rs @@ -1,9 +1,3382 @@ #[allow(dead_code)] mod settings; -mod app; -mod suggestions; -mod errors; pub use self::settings::AppSettings; -pub use self::app::App; -pub use self::errors::{ClapError, ClapErrorType}; + +use std::collections::{BTreeMap, HashMap, VecDeque}; +use std::env; +use std::io::{self, BufRead, BufWriter, Write}; +use std::path::Path; +use std::process; +use std::ffi::OsStr; +use std::borrow::Borrow; + +#[cfg(feature = "yaml")] +use yaml_rust::Yaml; +use vec_map::VecMap; + +use args::{Arg, ArgMatches, MatchedArg, SubCommand}; +use args::{FlagBuilder, OptBuilder, PosBuilder}; +use args::settings::{ArgFlags, ArgSettings}; +use args::ArgGroup; +use fmt::Format; +use self::settings::AppFlags; +use suggestions; +use errors::{ClapError, ClapErrorType, ClapResult, error_builder}; + + +const INTERNAL_ERROR_MSG: &'static str = "Fatal internal error. Please consider filing a bug \ + report at https://github.com/kbknapp/clap-rs/issues"; + +/// Used to create a representation of a command line program and all possible command line +/// arguments. +/// +/// Application settings are set using the "builder pattern" with `.get_matches()` being the +/// terminal method that starts the runtime-parsing process and returns information about +/// the user supplied arguments (or lack there of). +/// +/// There aren't any mandatory "options" that one must set. The "options" may also appear in any +/// order (so long as `.get_matches()` is the last method called). +/// +/// +/// # Examples +/// +/// ```no_run +/// # use clap::{App, Arg}; +/// let matches = App::new("myprog") +/// .author("Me, me@mail.com") +/// .version("1.0.2") +/// .about("Explains in brief what the program does") +/// .arg( +/// Arg::with_name("in_file").index(1) +/// ) +/// .after_help("Longer explaination to appear after the options when \ +/// displaying the help information from --help or -h") +/// .get_matches(); +/// +/// // Your program logic starts here... +/// ``` +#[allow(missing_debug_implementations)] +pub struct App<'a, 'v, 'ab, 'u, 'h, 'ar> { + // The name displayed to the user when showing version and help/usage information + name: String, + name_slice: &'ar str, + // A string of author(s) if desired. Displayed when showing help/usage information + author: Option<&'a str>, + // The version displayed to the user + version: Option<&'v str>, + // A brief explanation of the program that gets displayed to the user when shown + // help/usage + // information + about: Option<&'ab str>, + // Additional help information + more_help: Option<&'h str>, + // A list of possible flags + flags: Vec>, + // A list of possible options + opts: Vec>, + // A list of positional arguments + positionals: VecMap>, + // A list of subcommands + subcommands: Vec>, + help_short: Option, + version_short: Option, + required: Vec<&'ar str>, + short_list: Vec, + long_list: Vec<&'ar str>, + blacklist: Vec<&'ar str>, + usage_str: Option<&'u str>, + bin_name: Option, + usage: Option, + groups: HashMap<&'ar str, ArgGroup<'ar, 'ar>>, + global_args: Vec>, + help_str: Option<&'u str>, + settings: AppFlags, + overrides: Vec<&'ar str>, +} + +impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar> { + /// Creates a new instance of an application requiring a name (such as the binary). The name + /// will be displayed to the user when they request to print version or help and usage + /// information. The name should not contain spaces (hyphens '-' are ok). + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let prog = App::new("myprog") + /// # .get_matches(); + /// ``` + pub fn new(n: &'ar str) -> Self { + App { + name: n.to_owned(), + name_slice: n, + author: None, + about: None, + more_help: None, + version: None, + flags: vec![], + opts: vec![], + positionals: VecMap::new(), + subcommands: vec![], + help_short: None, + version_short: None, + required: vec![], + short_list: vec![], + long_list: vec![], + usage_str: None, + usage: None, + blacklist: vec![], + bin_name: None, + groups: HashMap::new(), + global_args: vec![], + help_str: None, + overrides: vec![], + settings: AppFlags::new(), + } + } + + /// Creates a new instace of `App` from a .yml (YAML) file. The YAML file must be properly + /// formatted or this function will panic!(). A full example of supported YAML objects can be + /// found in `examples/17_yaml.rs` and `examples/17_yaml.yml`. + /// + /// In order to use this function you must compile with the `features = ["yaml"]` in your + /// settings for `[dependencies.clap]` table of your `Cargo.toml` + /// + /// Note, due to how the YAML objects are built there is a convienience macro for loading the + /// YAML file (relative to the current file, like modules work). That YAML object can then be + /// passed to this function. + /// + /// # Examples + /// + /// ```ignore + /// # use clap::App; + /// let yml = load_yaml!("app.yml"); + /// let app = App::from_yaml(yml); + /// ``` + #[cfg(feature = "yaml")] + pub fn from_yaml<'y>(mut yaml: &'y Yaml) -> App<'y, 'y, 'y, 'y, 'y, 'y> { + // We WANT this to panic on error...so expect() is good. + let mut is_sc = None; + let mut a = if let Some(name) = yaml["name"].as_str() { + App::new(name) + } else { + let yaml_hash = yaml.as_hash().unwrap(); + let sc_key = yaml_hash.keys().nth(0).unwrap(); + is_sc = Some(yaml_hash.get(sc_key).unwrap()); + App::new(sc_key.as_str().unwrap()) + }; + yaml = if let Some(sc) = is_sc { + sc + } else { + yaml + }; + if let Some(v) = yaml["version"].as_str() { + a = a.version(v); + } + if let Some(v) = yaml["author"].as_str() { + a = a.author(v); + } + if let Some(v) = yaml["bin_name"].as_str() { + a = a.bin_name(v); + } + if let Some(v) = yaml["about"].as_str() { + a = a.about(v); + } + if let Some(v) = yaml["after_help"].as_str() { + a = a.after_help(v); + } + if let Some(v) = yaml["usage"].as_str() { + a = a.usage(v); + } + if let Some(v) = yaml["help"].as_str() { + a = a.help(v); + } + if let Some(v) = yaml["help_short"].as_str() { + a = a.help_short(v); + } + if let Some(v) = yaml["version_short"].as_str() { + a = a.version_short(v); + } + if let Some(v) = yaml["settings"].as_vec() { + for ys in v { + if let Some(s) = ys.as_str() { + a = a.setting(s.parse().ok().expect("unknown AppSetting found in YAML file")); + } + } + } + if let Some(v) = yaml["args"].as_vec() { + for arg_yaml in v { + a = a.arg(Arg::from_yaml(&arg_yaml.as_hash().unwrap())); + } + } + if let Some(v) = yaml["subcommands"].as_vec() { + for sc_yaml in v { + a = a.subcommand(SubCommand::from_yaml(&sc_yaml)); + } + } + if let Some(v) = yaml["arg_groups"].as_vec() { + for ag_yaml in v { + a = a.arg_group(ArgGroup::from_yaml(&ag_yaml.as_hash().unwrap())); + } + } + + a + } + + /// Sets a string of author(s) and will be displayed to the user when they request the help + /// information with `--help` or `-h`. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .author("Me, me@mymain.com") + /// # ; + /// ``` + pub fn author(mut self, a: &'a str) -> Self { + self.author = Some(a); + self + } + + /// Overrides the system-determined binary name. This should only be used when absolutely + /// neccessary, such as the binary name for your application is misleading, or perhaps *not* + /// how the user should invoke your program. + /// + /// **NOTE:** This command **should not** be used for SubCommands. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .bin_name("my_binary") + /// # ; + /// ``` + pub fn bin_name(mut self, a: &str) -> Self { + self.bin_name = Some(a.to_owned()); + self + } + + /// Sets a string briefly describing what the program does and will be displayed when + /// displaying help information. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .about("Does really amazing things to great people") + /// # ; + /// ``` + pub fn about(mut self, a: &'ab str) -> Self { + self.about = Some(a); + self + } + + /// Adds additional help information to be displayed in addition to and directly after + /// auto-generated help. This information is displayed **after** the auto-generated help + /// information. This additional help is often used to describe how to use the arguments, + /// or caveats to be noted. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::App; + /// App::new("myprog") + /// .after_help("Does really amazing things to great people") + /// # ; + /// ``` + pub fn after_help(mut self, h: &'h str) -> Self { + self.more_help = Some(h); + self + } + + /// Allows subcommands to override all requirements of the parent (this command). For example + /// if you had a subcommand or even top level application which had a required arguments that + /// are only required as long as there is no subcommand present. + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandsNegateReqs` instead. + /// This method will be removed at 2.x + /// + /// **NOTE:** This defaults to false (using subcommand does *not* negate requirements) + /// + /// # Examples + /// + /// ```no_run + /// # use clap::App; + /// App::new("myprog") + /// .subcommands_negate_reqs(true) + /// # ; + /// ``` + pub fn subcommands_negate_reqs(mut self, n: bool) -> Self { + if n { + self.settings.set(&AppSettings::SubcommandsNegateReqs); + } else { + self.settings.unset(&AppSettings::SubcommandsNegateReqs); + } + self + } + + /// Allows specifying that if no subcommand is present at runtime, error and exit gracefully + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandRequired` instead. This + /// method will be removed at 2.x + /// + /// **NOTE:** This defaults to false (subcommands do *not* need to be present) + /// + /// # Examples + /// + /// ```no_run + /// # use clap::App; + /// App::new("myprog") + /// .subcommand_required(true) + /// # ; + /// ``` + pub fn subcommand_required(mut self, n: bool) -> Self { + if n { + self.settings.set(&AppSettings::SubcommandRequired); + } else { + self.settings.unset(&AppSettings::SubcommandRequired); + } + self + } + + /// Sets a string of the version number to be displayed when displaying version or help + /// information. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .version("v0.1.24") + /// # ; + /// ``` + pub fn version(mut self, v: &'v str) -> Self { + self.version = Some(v); + self + } + + /// Sets a custom usage string to override the auto-generated usage string. + /// + /// This will be displayed to the user when errors are found in argument parsing, or when you + /// call `ArgMatches::usage()` + /// + /// **NOTE:** You do not need to specify the "USAGE: \n\t" portion, as that will + /// still be applied by `clap`, you only need to specify the portion starting + /// with the binary name. + /// + /// **NOTE:** This will not replace the entire help message, *only* the portion + /// showing the usage. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .usage("myapp [-clDas] ") + /// # ; + /// ``` + pub fn usage(mut self, u: &'u str) -> Self { + self.usage_str = Some(u); + self + } + + /// Sets a custom help message and overrides the auto-generated one. This should only be used + /// when the auto-generated message does not suffice. + /// + /// This will be displayed to the user when they use the default `--help` or `-h` + /// + /// **NOTE:** This replaces the **entire** help message, so nothing will be auto-generated. + /// + /// **NOTE:** This **only** replaces the help message for the current command, meaning if you + /// are using subcommands, those help messages will still be auto-generated unless you + /// specify a `.help()` for them as well. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myapp") + /// .help("myapp v1.0\n\ + /// Does awesome things\n\ + /// (C) me@mail.com\n\n\ + /// + /// USAGE: myapp \n\n\ + /// + /// Options:\n\ + /// -h, --helpe Dispay this message\n\ + /// -V, --version Display version info\n\ + /// -s Do something with stuff\n\ + /// -v Be verbose\n\n\ + /// + /// Commmands:\n\ + /// help Prints this message\n\ + /// work Do some work") + /// # ; + /// ``` + pub fn help(mut self, h: &'u str) -> Self { + self.help_str = Some(h); + self + } + + /// Sets the short version of the `help` argument without the preceding `-`. + /// + /// By default `clap` automatically assigns `h`, but this can be overridden + /// + /// **NOTE:** Any leading `-` characters will be stripped, and only the first + /// non `-` chacter will be used as the `short` version + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// // Using an uppercase `H` instead of the default lowercase `h` + /// .help_short("H") + /// # ; + pub fn help_short(mut self, s: &str) -> Self { + self.help_short = s.trim_left_matches(|c| c == '-') + .chars() + .nth(0); + self + } + + /// Sets the short version of the `version` argument without the preceding `-`. + /// + /// By default `clap` automatically assigns `V`, but this can be overridden + /// + /// **NOTE:** Any leading `-` characters will be stripped, and only the first + /// non `-` chacter will be used as the `short` version + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// // Using a lowercase `v` instead of the default capital `V` + /// .version_short("v") + /// # ; + pub fn version_short(mut self, s: &str) -> Self { + self.version_short = s.trim_left_matches(|c| c == '-') + .chars() + .nth(0); + self + } + + /// Specifies that the help text sould be displayed (and then exit gracefully), if no + /// arguments are present at runtime (i.e. an empty run such as, `$ myprog`. + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::ArgRequiredElseHelp` instead. This + /// method will be removed at 2.x + /// + /// **NOTE:** Subcommands count as arguments + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .arg_required_else_help(true) + /// # ; + /// ``` + pub fn arg_required_else_help(mut self, tf: bool) -> Self { + if tf { + self.settings.set(&AppSettings::ArgRequiredElseHelp); + } else { + self.settings.unset(&AppSettings::ArgRequiredElseHelp); + } + self + } + + /// Hides a subcommand from help message output. + /// + /// **NOTE:** This does **not** hide the subcommand from usage strings on error + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, SubCommand}; + /// # let matches = App::new("myprog") + /// # .subcommand( + /// # SubCommand::with_name("debug") + /// .hidden(true) + /// # ).get_matches(); + pub fn hidden(mut self, h: bool) -> Self { + if h { + self.settings.set(&AppSettings::Hidden); + } else { + self.settings.unset(&AppSettings::Hidden); + } + self + } + + /// Uses version of the current command for all subcommands. (Defaults to false; subcommands + /// have independant version strings) + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::GlobalVersion` instead. This + /// method will be removed at 2.x + /// + /// **NOTE:** The version for the current command and this setting must be set **prior** to + /// adding any subcommands + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, SubCommand}; + /// App::new("myprog") + /// .version("v1.1") + /// .global_version(true) + /// .subcommand(SubCommand::with_name("test")) + /// .get_matches(); + /// // running `myprog test --version` will display + /// // "myprog-test v1.1" + /// ``` + pub fn global_version(mut self, gv: bool) -> Self { + if gv { + self.settings.set(&AppSettings::GlobalVersion); + } else { + self.settings.unset(&AppSettings::GlobalVersion); + } + self + } + + /// Disables `-V` and `--version` for all subcommands (Defaults to false; subcommands have + /// version flags) + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::VersionlessSubcommands` instead. + /// This method will be removed at 2.x + /// + /// **NOTE:** This setting must be set **prior** adding any subcommands + /// + /// **NOTE:** Do not set this value to false, it will have undesired results! + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, SubCommand}; + /// App::new("myprog") + /// .version("v1.1") + /// .versionless_subcommands(true) + /// .subcommand(SubCommand::with_name("test")) + /// .get_matches(); + /// // running `myprog test --version` will display unknown argument error + /// ``` + pub fn versionless_subcommands(mut self, vers: bool) -> Self { + if vers { + self.settings.set(&AppSettings::VersionlessSubcommands); + } else { + self.settings.unset(&AppSettings::VersionlessSubcommands); + } + self + } + + /// By default the auto-generated help message groups flags, options, and positional arguments + /// separately. This setting disable that and groups flags and options together presenting a + /// more unified help message (a la getopts or docopt style). + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::UnifiedHelpMessage` instead. This + /// method will be removed at 2.x + /// + /// **NOTE:** This setting is cosmetic only and does not affect any functionality. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, SubCommand}; + /// App::new("myprog") + /// .unified_help_message(true) + /// .get_matches(); + /// // running `myprog --help` will display a unified "docopt" or "getopts" style help message + /// ``` + pub fn unified_help_message(mut self, uni_help: bool) -> Self { + if uni_help { + self.settings.set(&AppSettings::UnifiedHelpMessage); + } else { + self.settings.unset(&AppSettings::UnifiedHelpMessage); + } + self + } + + /// Will display a message "Press [ENTER]/[RETURN] to continue..." and wait user before + /// exiting + /// + /// This is most useful when writing an application which is run from a GUI shortcut, or on + /// Windows where a user tries to open the binary by double-clicking instead of using the + /// command line (i.e. set `.arg_required_else_help(true)` and `.wait_on_error(true)` to + /// display the help in such a case). + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::WaitOnError` instead. This + /// method will be removed at 2.x + /// + /// **NOTE:** This setting is **not** recursive with subcommands, meaning if you wish this + /// behavior for all subcommands, you must set this on each command (needing this is extremely + /// rare) + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .arg_required_else_help(true) + /// # ; + /// ``` + pub fn wait_on_error(mut self, w: bool) -> Self { + if w { + self.settings.set(&AppSettings::WaitOnError); + } else { + self.settings.unset(&AppSettings::WaitOnError); + } + self + } + + /// Specifies that the help text sould be displayed (and then exit gracefully), if no + /// subcommands are present at runtime (i.e. an empty run such as, `$ myprog`. + /// + /// **Deprecated:** Use `App::setting()` with `AppSettings::SubcommandRequiredElseHelp` + /// instead. This method will be removed at 2.x + /// + /// **NOTE:** This should *not* be used with `.subcommand_required()` as they do the same + /// thing, except one prints the help text, and one prints an error. + /// + /// **NOTE:** If the user specifies arguments at runtime, but no subcommand the help text will + /// still be displayed and exit. If this is *not* the desired result, consider using + /// `.arg_required_else_help()` + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .subcommand_required_else_help(true) + /// # ; + /// ``` + pub fn subcommand_required_else_help(mut self, tf: bool) -> Self { + if tf { + self.settings.set(&AppSettings::SubcommandRequiredElseHelp); + } else { + self.settings.unset(&AppSettings::SubcommandRequiredElseHelp); + } + self + } + + /// Enables Application level settings, passed as argument + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, AppSettings}; + /// App::new("myprog") + /// .setting(AppSettings::SubcommandRequired) + /// .setting(AppSettings::WaitOnError) + /// # ; + /// ``` + pub fn setting(mut self, setting: AppSettings) -> Self { + self.settings.set(&setting); + self + } + + /// Enables multiple Application level settings, passed as argument + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, AppSettings}; + /// App::new("myprog") + /// .settings( &[AppSettings::SubcommandRequired, + /// AppSettings::WaitOnError]) + /// # ; + /// ``` + pub fn settings(mut self, settings: &[AppSettings]) -> Self { + for s in settings { + self.settings.set(s); + } + self + } + + /// Adds an argument to the list of valid possibilties manually. This method allows you full + /// control over the arguments settings and options (as well as dynamic generation). It also + /// allows you specify several more advanced configuration options such as relational rules + /// (exclusions and requirements). + /// + /// The only disadvantage to this method is that it's more verbose, and arguments must be added + /// one at a time. Using `Arg::from_usage` helps with the verbosity, and still allows full + /// control over the advanced configuration options. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// // Adding a single "flag" argument with a short and help text, using Arg::with_name() + /// .arg( + /// Arg::with_name("debug") + /// .short("d") + /// .help("turns on debugging mode") + /// ) + /// // Adding a single "option" argument with a short, a long, and help text using the less + /// // verbose Arg::from_usage() + /// .arg( + /// Arg::from_usage("-c --config=[CONFIG] 'Optionally sets a config file to use'") + /// ) + /// # ; + /// ``` + pub fn arg(mut self, a: Arg<'ar, 'ar, 'ar, 'ar, 'ar, 'ar>) -> Self { + self.add_arg(a); + self + } + + // actually adds the arguments + fn add_arg(&mut self, a: Arg<'ar, 'ar, 'ar, 'ar, 'ar, 'ar>) { + if self.flags.iter().any(|f| &f.name == &a.name) || + self.opts.iter().any(|o| o.name == a.name) || + self.positionals.values().any(|p| p.name == a.name) { + panic!("Non-unique argument name: {} is already in use", a.name); + } + if let Some(grp) = a.group { + let ag = self.groups.entry(grp).or_insert(ArgGroup::with_name(grp)); + ag.args.push(a.name); + } + if let Some(s) = a.short { + if self.short_list.contains(&s) { + panic!("Argument short must be unique\n\n\t-{} is already in use", + s); + } else { + self.short_list.push(s); + } + } + if let Some(l) = a.long { + if self.long_list.contains(&l) { + panic!("Argument long must be unique\n\n\t--{} is already in use", + l); + } else { + self.long_list.push(l); + } + if l == "help" { + self.settings.set(&AppSettings::NeedsLongHelp); + } else if l == "version" { + self.settings.set(&AppSettings::NeedsLongVersion); + } + } + if a.required { + self.required.push(a.name); + } + if a.index.is_some() || (a.short.is_none() && a.long.is_none()) { + let i = if a.index.is_none() { + (self.positionals.len() + 1) + } else { + a.index.unwrap() as usize + }; + if self.positionals.contains_key(&i) { + panic!("Argument \"{}\" has the same index as another positional \ + argument\n\n\tPerhaps try .multiple(true) to allow one positional argument \ + to take multiple values", + a.name); + } + let pb = PosBuilder::from_arg(&a, i as u8, &mut self.required); + // self.positionals_name.insert(pb.name, i); + self.positionals.insert(i, pb); + } else if a.takes_value { + let ob = OptBuilder::from_arg(&a, &mut self.required); + self.opts.push(ob); + } else { + let fb = FlagBuilder::from(&a); + self.flags.push(fb); + } + if a.global { + if a.required { + panic!("Global arguments cannot be required.\n\n\t'{}' is marked as global and \ + required", + a.name); + } + self.global_args.push(a); + } + } + + /// Adds multiple arguments to the list of valid possibilties by iterating over a Vec of Args + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .args( + /// vec![Arg::from_usage("[debug] -d 'turns on debugging info"), + /// Arg::with_name("input").index(1).help("the input file to use")] + /// ) + /// # ; + /// ``` + pub fn args(mut self, args: Vec>) -> Self { + for arg in args.into_iter() { + self = self.arg(arg); + } + self + } + + /// A convienience method for adding a single basic argument (one without advanced + /// relational rules) from a usage type string. The string used follows the same rules and + /// syntax as `Arg::from_usage()` + /// + /// The downside to using this method is that you can not set any additional properties of the + /// `Arg` other than what `Arg::from_usage()` supports. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .arg_from_usage("-c --conf= 'Sets a configuration file to use'") + /// # ; + /// ``` + pub fn arg_from_usage(mut self, usage: &'ar str) -> Self { + self = self.arg(Arg::from_usage(usage)); + self + } + + /// Adds multiple arguments at once from a usage string, one per line. See `Arg::from_usage()` + /// for details on the syntax and rules supported. + /// + /// Like `App::arg_from_usage()` the downside is you only set properties for the `Arg`s which + /// `Arg::from_usage()` supports. But here the benefit is pretty strong, as the readability is + /// greatly enhanced, especially if you don't need any of the more advanced configuration + /// options. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// App::new("myprog") + /// .args_from_usage( + /// "-c --conf=[config] 'Sets a configuration file to use' + /// [debug]... -d 'Sets the debugging level' + /// 'The input file to use'" + /// ) + /// # ; + /// ``` + pub fn args_from_usage(mut self, usage: &'ar str) -> Self { + for l in usage.lines() { + self = self.arg(Arg::from_usage(l.trim())); + } + self + } + + /// Adds an ArgGroup to the application. ArgGroups are a family of related arguments. By + /// placing them in a logical group, you make easier requirement and exclusion rules. For + /// instance, you can make an ArgGroup required, this means that one (and *only* one) argument + /// from that group must be present. Using more than one argument from an ArgGroup causes a + /// failure (graceful exit). + /// + /// You can also do things such as name an ArgGroup as a confliction, meaning any of the + /// arguments that belong to that group will cause a failure if present. + /// + /// Perhaps the most common use of ArgGroups is to require one and *only* one argument to be + /// present out of a given set. For example, lets say that you were building an application + /// where one could set a given version number by supplying a string using an option argument, + /// such as `--set-ver v1.2.3`, you also wanted to support automatically using a previous + /// version numer and simply incrementing one of the three numbers, so you create three flags + /// `--major`, `--minor`, and `--patch`. All of these arguments shouldn't be used at one time + /// but perhaps you want to specify that *at least one* of them is used. You can create a + /// group + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, ArgGroup}; + /// # App::new("app") + /// .args_from_usage("--set-ver [ver] 'set the version manually' + /// --major 'auto increase major' + /// --minor 'auto increase minor' + /// --patch 'auto increase patch") + /// .arg_group(ArgGroup::with_name("vers") + /// .add_all(&["ver", "major", "minor","patch"]) + /// .required(true)) + /// # ; + pub fn arg_group(mut self, group: ArgGroup<'ar, 'ar>) -> Self { + if group.required { + self.required.push(group.name); + if let Some(ref reqs) = group.requires { + for r in reqs { + self.required.push(r); + } + } + if let Some(ref bl) = group.conflicts { + for b in bl { + self.blacklist.push(b); + } + } + } + let mut found = false; + if let Some(ref mut grp) = self.groups.get_mut(group.name) { + for a in &group.args { + grp.args.push(a); + } + grp.requires = group.requires.clone(); + grp.conflicts = group.conflicts.clone(); + grp.required = group.required; + found = true; + } + if !found { + self.groups.insert(group.name, group); + } + self + } + + /// Adds a ArgGroups to the application. ArgGroups are a family of related arguments. By + /// placing them in a logical group, you make easier requirement and exclusion rules. For + /// instance, you can make an ArgGroup required, this means that one (and *only* one) argument + /// from that group must be present. Using more than one argument from an ArgGroup causes a + /// failure (graceful exit). + /// + /// You can also do things such as name an ArgGroup as a confliction, meaning any of the + /// arguments that belong to that group will cause a failure if present. + /// + /// Perhaps the most common use of ArgGroups is to require one and *only* one argument to be + /// present out of a given set. For example, lets say that you were building an application + /// where one could set a given version number by supplying a string using an option argument, + /// such as `--set-ver v1.2.3`, you also wanted to support automatically using a previous + /// version numer and simply incrementing one of the three numbers, so you create three flags + /// `--major`, `--minor`, and `--patch`. All of these arguments shouldn't be used at one time + /// but perhaps you want to specify that *at least one* of them is used. You can create a + /// group + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, ArgGroup}; + /// # App::new("app") + /// .args_from_usage("--set-ver [ver] 'set the version manually' + /// --major 'auto increase major' + /// --minor 'auto increase minor' + /// --patch 'auto increase patch") + /// .arg_group(ArgGroup::with_name("vers") + /// .add_all(&["ver", "major", "minor","patch"]) + /// .required(true)) + /// # ; + pub fn arg_groups(mut self, groups: Vec>) -> Self { + for g in groups { + self = self.arg_group(g); + } + self + } + + /// Adds a subcommand to the list of valid possibilties. Subcommands are effectively sub apps, + /// because they can contain their own arguments, subcommands, version, usage, etc. They also + /// function just like apps, in that they get their own auto generated help, version, and + /// usage. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, SubCommand}; + /// # App::new("myprog") + /// .subcommand(SubCommand::with_name("config") + /// .about("Controls configuration features") + /// .arg_from_usage(" 'Required configuration file to use'")) + /// // Additional subcommand configuration goes here, such as other arguments... + /// # ; + /// ``` + pub fn subcommand(mut self, mut subcmd: App<'a, 'v, 'ab, 'u, 'h, 'ar>) -> Self { + if subcmd.name == "help" { + self.settings.set(&AppSettings::NeedsSubcommandHelp); + } + if self.settings.is_set(&AppSettings::VersionlessSubcommands) { + subcmd.settings.set(&AppSettings::DisableVersion); + } + if self.settings.is_set(&AppSettings::GlobalVersion) && subcmd.version.is_none() && + self.version.is_some() { + subcmd.version = Some(self.version.unwrap()); + } + self.subcommands.push(subcmd); + self + } + + /// Adds multiple subcommands to the list of valid possibilties by iterating over a Vec of + /// `SubCommand`s + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg, SubCommand}; + /// # App::new("myprog") + /// .subcommands( vec![ + /// SubCommand::with_name("config").about("Controls configuration functionality") + /// .arg(Arg::with_name("config_file").index(1)), + /// SubCommand::with_name("debug").about("Controls debug functionality")]) + /// # ; + /// ``` + pub fn subcommands(mut self, subcmds: Vec>) -> Self { + for subcmd in subcmds.into_iter() { + self = self.subcommand(subcmd); + } + self + } + + // Creates a usage string if one was not provided by the user manually. This + // happens just + // after all arguments were parsed, but before any subcommands have been parsed + // (so as to + // give subcommands their own usage recursively) + fn create_usage(&self, matches: &[&'ar str]) -> ClapResult { + let mut usage = String::with_capacity(75); + usage.push_str("USAGE:\n\t"); + if let Some(u) = self.usage_str { + usage.push_str(u); + } else if !matches.is_empty() { + try!(self.usage_from_matches(&mut usage, matches)); + } else { + usage.push_str(&*self.usage + .as_ref() + .unwrap_or(self.bin_name + .as_ref() + .unwrap_or(&self.name))); + + let reqs = self.required.iter().map(|n| *n).collect::>(); + let req_strings = self.get_required_from(&reqs, None); + let req_string = req_strings.iter() + .fold(String::new(), |a, s| a + &format!(" {}", s)[..]); + + if !self.flags.is_empty() && !self.settings.is_set(&AppSettings::UnifiedHelpMessage) { + usage.push_str(" [FLAGS]"); + } else { + usage.push_str(" [OPTIONS]"); + } + if !self.settings.is_set(&AppSettings::UnifiedHelpMessage) && !self.opts.is_empty() && + self.opts.iter().any(|a| !a.settings.is_set(&ArgSettings::Required)) { + usage.push_str(" [OPTIONS]"); + } + + usage.push_str(&req_string[..]); + + // places a '--' in the usage string if there are args and options + // supporting multiple values + if !self.positionals.is_empty() && + (self.opts.iter().any(|a| a.settings.is_set(&ArgSettings::Multiple)) || + self.positionals.values().any(|a| a.settings.is_set(&ArgSettings::Multiple))) && + !self.opts.iter().any(|a| a.settings.is_set(&ArgSettings::Required)) && + self.subcommands.is_empty() { + usage.push_str(" [--]") + } + if !self.positionals.is_empty() && + self.positionals + .values() + .any(|a| !a.settings.is_set(&ArgSettings::Required)) { + usage.push_str(" [ARGS]"); + } + + + if !self.subcommands.is_empty() && + !self.settings.is_set(&AppSettings::SubcommandRequired) { + usage.push_str(" [SUBCOMMAND]"); + } else if self.settings.is_set(&AppSettings::SubcommandRequired) && + !self.subcommands.is_empty() { + usage.push_str(" "); + } + } + + usage.shrink_to_fit(); + Ok(usage) + } + + // Creates a context aware usage string, or "smart usage" from currently used + // args, and + // requirements + fn usage_from_matches(&self, usage: &mut String, matches: &[&'ar str]) -> ClapResult<()> { + use std::fmt::Write; + let mut hs: Vec<&str> = self.required.iter().map(|n| *n).collect(); + for n in matches { + hs.push(*n); + } + let reqs = self.get_required_from(&hs, None); + + let r_string = reqs.iter().fold(String::new(), |acc, s| acc + &format!(" {}", s)[..]); + + try!(write!(usage, + "{}{}", + self.usage + .as_ref() + .unwrap_or(self.bin_name.as_ref().unwrap_or(&self.name)), + r_string)); + if self.settings.is_set(&AppSettings::SubcommandRequired) { + try!(write!(usage, " ")); + } + + Ok(()) + } + + /// Prints the full help message to `io::stdout()` using a `BufWriter` + /// + /// # Examples + /// ```no_run + /// # use clap::App; + /// # use std::io; + /// let mut app = App::new("myprog"); + /// let mut out = io::stdout(); + /// app.write_help(&mut out).ok().expect("failed to write to stdout"); + /// ``` + pub fn print_help(&self) -> ClapResult<()> { + let out = io::stdout(); + let mut buf_w = BufWriter::new(out.lock()); + self.write_help(&mut buf_w) + } + + /// Writes the full help message to the user to a `io::Write` object + /// + /// ```no_run + /// # use clap::App; + /// # use std::io; + /// let mut app = App::new("myprog"); + /// let mut out = io::stdout(); + /// app.write_help(&mut out).ok().expect("failed to write to stdout"); + /// ``` + pub fn write_help(&self, w: &mut W) -> ClapResult<()> { + if let Some(h) = self.help_str { + return writeln!(w, "{}", h).map_err(ClapError::from); + } + + // Print the version + try!(write!(w, + "{} {}\n", + &self.bin_name.as_ref().unwrap_or(&self.name)[..].replace(" ", "-"), + self.version.unwrap_or(""))); + let flags = !self.flags.is_empty(); + let pos = !self.positionals.is_empty(); + let opts = !self.opts.is_empty(); + let subcmds = !self.subcommands.is_empty(); + let unified_help = self.settings.is_set(&AppSettings::UnifiedHelpMessage); + + let mut longest_flag = 0; + for fl in self.flags + .iter() + .filter(|f| f.long.is_some() && !f.settings.is_set(&ArgSettings::Hidden)) + .map(|a| a.to_string().len()) { + if fl > longest_flag { + longest_flag = fl; + } + } + let mut longest_opt = 0; + for ol in self.opts + .iter() + .filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) + .map(|a| a.to_string().len()) { + if ol > longest_opt { + longest_opt = ol; + } + } + let mut longest_pos = 0; + for pl in self.positionals + .values() + .filter(|p| !p.settings.is_set(&ArgSettings::Hidden)) + .map(|f| f.to_string().len()) { + if pl > longest_pos { + longest_pos = pl; + } + } + let mut longest_sc = 0; + for scl in self.subcommands + .iter() + .filter(|s| !s.settings.is_set(&AppSettings::Hidden)) + .map(|f| f.name.len()) { + if scl > longest_sc { + longest_sc = scl; + } + } + + if let Some(author) = self.author { + try!(write!(w, "{}\n", author)); + } + if let Some(about) = self.about { + try!(write!(w, "{}\n", about)); + } + + try!(write!(w, "\n{}", try!(self.create_usage(&[])))); + + if flags || opts || pos || subcmds { + try!(write!(w, "\n")); + } + + let tab = " "; + let longest = if !unified_help || longest_opt == 0 { + longest_flag + } else { + longest_opt + }; + if unified_help && (flags || opts) { + try!(write!(w, "\nOPTIONS:\n")); + let mut combined = BTreeMap::new(); + for f in self.flags.iter().filter(|f| !f.settings.is_set(&ArgSettings::Hidden)) { + let mut v = vec![]; + try!(f.write_help(&mut v, tab, longest)); + combined.insert(f.name, v); + } + for o in self.opts.iter().filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) { + let mut v = vec![]; + try!(o.write_help(&mut v, tab, longest)); + combined.insert(o.name, v); + } + for (_, a) in combined { + try!(write!(w, "{}", String::from_utf8_lossy(&*a))); + } + } else { + if flags { + try!(write!(w, "\nFLAGS:\n")); + for (_, f) in self.flags + .iter() + .filter(|f| !f.settings.is_set(&ArgSettings::Hidden)) + .map(|f| (f.name, f)) + .collect::>() { + try!(f.write_help(w, tab, longest)); + } + } + if opts { + try!(write!(w, "\nOPTIONS:\n")); + for (_, o) in self.opts + .iter() + .filter(|o| !o.settings.is_set(&ArgSettings::Hidden)) + .map(|o| (o.name, o)) + .collect::>() { + try!(o.write_help(w, tab, longest_opt)); + } + } + } + if pos { + try!(write!(w, "\nARGS:\n")); + for v in self.positionals + .values() + .filter(|p| !p.settings.is_set(&ArgSettings::Hidden)) { + try!(v.write_help(w, tab, longest_pos)); + } + } + if subcmds { + try!(write!(w, "\nSUBCOMMANDS:\n")); + for (name, sc) in self.subcommands + .iter() + .filter(|s| !s.settings.is_set(&AppSettings::Hidden)) + .map(|s| (s.name_slice, s)) + .collect::>() { + try!(write!(w, "{}{}", tab, name)); + write_spaces!((longest_sc + 4) - (name.len()), w); + if let Some(a) = sc.about { + if a.contains("{n}") { + let mut ab = a.split("{n}"); + while let Some(part) = ab.next() { + try!(write!(w, "{}\n", part)); + write_spaces!(longest_sc + 8, w); + try!(write!(w, "{}", ab.next().unwrap_or(""))); + } + } else { + try!(write!(w, "{}", a)); + } + } + try!(write!(w, "\n")); + } + } + + if let Some(h) = self.more_help { + try!(write!(w, "\n{}", h)); + } + + // flush the buffer + debugln!("Flushing the buffer..."); + w.flush().map_err(ClapError::from) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands + /// + /// # Panics + /// + /// If any arguments contain invalid unicode characters. If this is not desired it is + /// recommended to use the `*_safe()` or `*_lossy()` versions of this method. + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches(); + /// ``` + pub fn get_matches(self) -> ArgMatches<'ar, 'ar> { + // Start the parsing + self.get_matches_from(env::args()) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands. Invalid unicode characters are replaced with + /// `U+FFFD REPLACEMENT CHARACTER` + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches(); + /// ``` + pub fn get_matches_lossy(self) -> ArgMatches<'ar, 'ar> { + // Start the parsing + self.get_matches_from_lossy(env::args_os()) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands + /// + /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are + /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` + /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or + /// perform a `std::process::exit` yourself. + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_safe() + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_safe(self) -> ClapResult> { + // Start the parsing + self.get_matches_from_safe(env::args_os()) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands. Invalid unicode characters are replaced with + /// `U+FFFD REPLACEMENT CHARACTER` + /// + /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are + /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` + /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or + /// perform a `std::process::exit` yourself. + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_safe() + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_safe_lossy(self) -> ClapResult> { + // Start the parsing + self.get_matches_from_safe_lossy(env::args_os()) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches()` + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_from(arg_vec); + /// ``` + pub fn get_matches_from(mut self, itr: I) -> ArgMatches<'ar, 'ar> + where I: IntoIterator, + T: AsRef + { + self.get_matches_from_safe_borrow(itr).unwrap_or_else(|e| { + // Otherwise, write to stderr and exit + self.maybe_wait_for_exit(e); + }) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands. Invalid unicode characters are replaced with + /// `U+FFFD REPLACEMENT CHARACTER` + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches()` + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_from(arg_vec); + /// ``` + pub fn get_matches_from_lossy(mut self, itr: I) -> ArgMatches<'ar, 'ar> + where I: IntoIterator, + T: AsRef + { + self.get_matches_from_safe_borrow_lossy(itr).unwrap_or_else(|e| { + // Otherwise, write to stderr and exit + self.maybe_wait_for_exit(e); + }) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands + /// + /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are + /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` + /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or + /// perform a `std::process::exit` yourself. + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches_safe()` + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// **NOTE:** Invalid unicode characters will result in an `Err` with type + /// `ClapErrorType::InvalidUnicode` + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_from_safe(arg_vec) + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_from_safe(mut self, itr: I) -> ClapResult> + where I: IntoIterator, + T: AsRef + { + self.get_matches_from_safe_borrow(itr) + } + + /// Starts the parsing process. Called on top level parent app **ONLY** then recursively calls + /// the real parsing function for all subcommands. Invalid unicode characters are replaced with + /// `U+FFFD REPLACEMENT CHARACTER` + /// + /// **NOTE:** This method WILL NOT exit when `--help` or `--version` (or short versions) are + /// used. It will return an error, where the `error_type` is a `ClapErrorType::HelpDisplayed` + /// or `ClapErrorType::VersionDisplayed` respectively. You must call `error.exit()` or + /// perform a `std::process::exit` yourself. + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches_safe()` + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let matches = App::new("myprog") + /// // Args and options go here... + /// .get_matches_from_safe(arg_vec) + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_from_safe_lossy(mut self, itr: I) -> ClapResult> + where I: IntoIterator, + T: AsRef + { + self._get_matches_from_safe_borrow(itr, true) + } + + /// Starts the parsing process without consuming the `App` struct `self`. This is normally not + /// the desired functionality, instead prefer `App::get_matches_from_safe` which *does* + /// consume `self`. + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches_safe()` + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// **NOTE:** Invalid unicode characters will result in an `Err` with type + /// `ClapErrorType::InvalidUnicode` + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let mut app = App::new("myprog"); + /// // Args and options go here... + /// let matches = app.get_matches_from_safe_borrow(arg_vec) + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_from_safe_borrow(&mut self, itr: I) -> ClapResult> + where I: IntoIterator, + T: AsRef + { + self._get_matches_from_safe_borrow(itr, false) + } + + /// Starts the parsing process without consuming the `App` struct `self`. This is normally not + /// the desired functionality, instead prefer `App::get_matches_from_safe` which *does* + /// consume `self`. Invalid unicode characters are replaced with `U+FFFD REPLACEMENT CHARACTER` + /// + /// **NOTE:** The first argument will be parsed as the binary name. + /// + /// **NOTE:** This method should only be used when absolutely necessary, such as needing to + /// parse arguments from something other than `std::env::args()`. If you are unsure, use + /// `App::get_matches_safe()` + /// + /// **NOTE:** This method should only be used when is absolutely necessary to handle errors + /// manually. + /// + /// **NOTE:** Invalid unicode characters will result in an `Err` with type + /// `ClapErrorType::InvalidUnicode` + /// + /// # Examples + /// + /// ```no_run + /// # use clap::{App, Arg}; + /// let arg_vec = vec!["my_prog", "some", "args", "to", "parse"]; + /// + /// let mut app = App::new("myprog"); + /// // Args and options go here... + /// let matches = app.get_matches_from_safe_borrow(arg_vec) + /// .unwrap_or_else( |e| { panic!("An error occurs: {}", e) }); + /// ``` + pub fn get_matches_from_safe_borrow_lossy(&mut self, + itr: I) + -> ClapResult> + where I: IntoIterator, + T: AsRef + { + self._get_matches_from_safe_borrow(itr, true) + } + + fn _get_matches_from_safe_borrow(&mut self, + itr: I, + lossy: bool) + -> ClapResult> + where I: IntoIterator, + T: AsRef + { + // Verify all positional assertions pass + self.verify_positionals(); + // If there are global arguments, we need to propgate them down to subcommands + // before parsing incase we run into a subcommand + self.propogate_globals(); + + let mut matches = ArgMatches::new(); + + let mut it = itr.into_iter(); + // Get the name of the program (argument 1 of env::args()) and determine the + // actual file + // that was used to execute the program. This is because a program called + // ./target/release/my_prog -a + // will have two arguments, './target/release/my_prog', '-a' but we don't want + // to display + // the full path when displaying help messages and such + if !self.settings.is_set(&AppSettings::NoBinaryName) { + if let Some(name) = it.next() { + let p = Path::new(name.as_ref()); + if let Some(f) = p.file_name() { + if let Ok(s) = f.to_os_string().into_string() { + if let None = self.bin_name { + self.bin_name = Some(s); + } + } + } + } + } + + // do the real parsing + if let Err(e) = self.get_matches_with(&mut matches, &mut it, lossy) { + return Err(e); + } + + Ok(matches) + } + + // The actual parsing function + #[cfg_attr(feature="lints", allow(while_let_on_iterator))] + fn get_matches_with(&mut self, + matches: &mut ArgMatches<'ar, 'ar>, + it: &mut I, + lossy: bool) + -> ClapResult<()> + where I: Iterator, + T: AsRef + { + // First we create the `--help` and `--version` arguments and add them if + // necessary + self.create_help_and_version(); + + let mut pos_only = false; + let mut subcmd_name: Option = None; + let mut needs_val_of: Option<&str> = None; + let mut pos_counter = 1; + let mut val_counter = 0; + while let Some(arg) = it.next() { + let arg_cow = match arg.as_ref().to_str() { + Some(s) => s.into(), + None => { + if !lossy { + return Err( + error_builder::InvalidUnicode( + &*try!(self.create_current_usage(matches))) + ); + } + arg.as_ref().to_string_lossy() + } + }; + let arg_slice: &str = arg_cow.borrow(); + let mut skip = false; + + // we need to know if we're parsing a new argument, or the value of previous + // argument, + // perhaps one with multiple values such as --option val1 val2. We determine + // when to + // stop parsing multiple values by finding a '-' + let new_arg = if arg_slice.starts_with("-") { + // If we come to a single `-` it's a value, not a new argument...this happens + // when + // one wants to use the Unix standard of '-' to mean 'stdin' + !(arg_slice.len() == 1) + } else { + false + }; + + // pos_only is determined later, and set true when a user uses the Unix + // standard of + // '--' to mean only positionals follow + if !pos_only && !new_arg && + !self.subcommands.iter().any(|s| s.name_slice == arg_slice) { + // Check to see if parsing a value from an option + if let Some(nvo) = needs_val_of { + // get the OptBuilder so we can check the settings + if let Some(ref opt) = self.opts.iter().filter(|o| &o.name == &nvo).next() { + try!(opt.validate_value(arg_slice, matches, self)); + + if let Some(ref vec) = self.groups_for_arg(opt.name) { + for grp in vec { + if let Some(ref mut o) = matches.args.get_mut(grp) { + o.occurrences = if opt.settings + .is_set(&ArgSettings::Multiple) { + o.occurrences + 1 + } else { + 1 + }; + // Values must be inserted in order...the user may care about + // that! + if let Some(ref mut vals) = o.values { + let len = vals.len() as u8 + 1; + vals.insert(len, arg_slice.to_owned()); + } + } + + } + } + // save the value to matched option + if let Some(ref mut o) = matches.args.get_mut(opt.name) { + // if it's multiple; the occurrences are increased when originally + // found + o.occurrences = if opt.settings.is_set(&ArgSettings::Multiple) { + o.occurrences + 1 + } else { + skip = true; + 1 + }; + + // Options always have values, even if it's empty, so we can unwrap() + if let Some(ref mut vals) = o.values { + + // Values must be inserted in order...the user may care about that! + let len = vals.len() as u8 + 1; + vals.insert(len, arg_slice.to_owned()); + + // Now that the values have been added, we can ensure we haven't + // gone over any max_limits, or if we've reached the exact number + // of values we can stop parsing values, and go back to arguments. + // + // For example, if we define an option with exactly 2 values and + // the users passes: + // $ my_prog --option val1 val2 pos1 + // we stop parsing values of --option after val2, if the user + // hadn't defined an exact or max value, pos1 would be parsed as a + // value of --option + if let Some(num) = opt.max_vals { + if len != num { + continue; + } + } else if let Some(num) = opt.num_vals { + if opt.settings.is_set(&ArgSettings::Multiple) { + val_counter += 1; + if val_counter != num { + continue; + } else { + val_counter = 0; + } + } else { + // if we need more values, get the next value + if len != num { + continue; + } + } + } else if !skip { + // get the next value from the iterator + continue; + } + } + } + skip = true; + } + } + } + + // if we're done getting values from an option, get the next arg from the + // iterator + if skip { + needs_val_of = None; + continue; + } else if let Some(ref name) = needs_val_of { + // We've reached more values for an option than it possibly accepts + if let Some(ref o) = self.opts.iter().filter(|o| &o.name == name).next() { + if !o.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::EmptyValue( + &*o.to_string(), + &*try!(self.create_current_usage(matches)) + )); + } + } + } + + if arg_slice.starts_with("--") && !pos_only { + if arg_slice.len() == 2 { + // The user has passed '--' which means only positional args follow no matter + // what they start with + pos_only = true; + continue; + } + + // This arg is either an option or flag using a long (i.e. '--something') + match self.parse_long_arg(matches, arg_slice) { + Ok(r) => needs_val_of = r, + Err(e) => return Err(e), + } + } else if arg_slice.starts_with("-") && arg_slice.len() != 1 && !pos_only { + // Multiple or single flag(s), or single option (could be '-SbG' or '-o') + match self.parse_short_arg(matches, arg_slice) { + Ok(r) => needs_val_of = r, + Err(e) => return Err(e), + } + } else { + // Positional or Subcommand + // + // If the user passed `--` we don't check for subcommands, because the argument + // they + // may be trying to pass might match a subcommand name + if !pos_only { + if self.subcommands.iter().any(|s| s.name_slice == arg_slice) { + if arg_slice == "help" && + self.settings.is_set(&AppSettings::NeedsSubcommandHelp) { + try!(self.print_help()); + return Err(ClapError { + error: String::new(), + error_type: ClapErrorType::HelpDisplayed, + }); + } + subcmd_name = Some(arg_slice.to_owned()); + break; + } else if let Some(candidate) = suggestions::did_you_mean( + arg_slice, + self.subcommands + .iter() + .map(|s| &s.name)) { + return Err(error_builder::InvalidSubcommand( + arg_slice, + candidate, + self.bin_name.as_ref().unwrap_or(&self.name), + &*try!(self.create_current_usage(matches)))); + } + } + + // Did the developer even define any valid positionals? Since we reached this + // far, it's not a subcommand + if self.positionals.is_empty() { + return Err(error_builder::UnexpectedArgument( + arg_slice, + self.bin_name.as_ref().unwrap_or(&self.name), + &*try!(self.create_current_usage(matches)))); + } else if let Some(p) = self.positionals.get(&pos_counter) { + // Make sure this one doesn't conflict with anything + if self.blacklist.contains(&p.name) { + return Err(error_builder::ArgumentConflict( + p.to_string(), + self.blacklisted_from(p.name, &matches), + try!(self.create_current_usage(matches)))); + } + + if let Some(ref p_vals) = p.possible_vals { + if !p_vals.contains(&arg_slice) { + return Err( + error_builder::InvalidValue(arg_slice, + p_vals, + &p.to_string(), + &*try!(self.create_current_usage(matches)))); + } + } + + // Have we made the update yet? + let mut done = false; + if p.settings.is_set(&ArgSettings::Multiple) { + if let Some(num) = p.num_vals { + if let Some(ref ma) = matches.args.get(p.name) { + if let Some(ref vals) = ma.values { + if vals.len() as u8 == num { + return Err(error_builder::TooManyValues( + arg_slice, + &*p.to_string(), + &*try!(self.create_current_usage(matches)))); + } + } + } + } + // Check if it's already existing and update if so... + if let Some(ref mut pos) = matches.args.get_mut(p.name) { + done = true; + pos.occurrences += 1; + if let Some(ref mut vals) = pos.values { + let len = (vals.len() + 1) as u8; + vals.insert(len, arg_slice.to_owned()); + } + } + + if !pos_only && + (self.settings.is_set(&AppSettings::TrailingVarArg) && + pos_counter == self.positionals.len()) { + pos_only = true; + } + } else { + // Only increment the positional counter if it doesn't allow multiples + pos_counter += 1; + } + // Was an update made, or is this the first occurrence? + if !done { + if self.overrides.contains(&p.name) { + if let Some(ref name) = self.overriden_from(p.name, matches) { + matches.args.remove(name); + remove_overriden!(self, name); + } + } + if let Some(ref or) = p.overrides { + for pa in or { + matches.args.remove(pa); + remove_overriden!(self, pa); + self.overrides.push(pa); + } + } + let mut bm = BTreeMap::new(); + if let Some(ref vtor) = p.validator { + let f = &*vtor; + if let Err(ref e) = f(arg_slice.to_owned()) { + return Err(error_builder::ValueValidationError(&*e)); + } + } + bm.insert(1, arg_slice.to_owned()); + if let Some(ref vec) = self.groups_for_arg(p.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: Some(bm.clone()), + }); + } + } + matches.args.insert(p.name, + MatchedArg { + occurrences: 1, + values: Some(bm), + }); + + if let Some(ref bl) = p.blacklist { + for name in bl { + self.blacklist.push(name); + } + } + + // Because of the macro call, we have to create a temp variable + if let Some(ref reqs) = p.requires { + // Add all required args which aren't already found in matches to the + // final required list + for n in reqs { + if matches.args.contains_key(n) { + continue; + } + + self.required.push(n); + } + } + + if p.settings.is_set(&ArgSettings::Required) { + // for macro call + let name = &p.name; + vec_remove!(self.required, name); + } + + parse_group_reqs!(self, p); + } + } else { + return Err(error_builder::UnexpectedArgument( + arg_slice, + self.bin_name.as_ref().unwrap_or(&self.name), + &*try!(self.create_current_usage(matches)))); + } + } + } + + if let Some(ref a) = needs_val_of { + if let Some(o) = self.opts.iter().filter(|o| &o.name == a).next() { + if o.settings.is_set(&ArgSettings::Multiple) && self.required.is_empty() { + let should_err = match matches.values_of(o.name) { + Some(ref v) => v.is_empty(), + None => true, + }; + if should_err { + return Err(error_builder::EmptyValue( + &*o.to_string(), + &*try!(self.create_current_usage(matches)) + )); + } + } else if !o.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::EmptyValue( + &*o.to_string(), + &*try!(self.create_current_usage(matches)) + )); + } else { + debugln!("Remaining Required Arg..."); + debugln!("required={:#?}", self.required); + return Err(error_builder::MissingRequiredArgument( + self.get_required_from(&self.required, Some(matches)) + .iter() + .fold(String::new(), + |acc, s| acc + &format!("\n\t{}", Format::Error(s))[..]), + try!(self.create_current_usage(matches)))); + } + } else { + return Err(error_builder::EmptyValue(&*format!("{}", + self.positionals + .values() + .filter(|p| &p.name == a) + .next() + .unwrap()), + &*try!(self.create_current_usage(matches)))); + } + } + + let res = self.validate_blacklist(matches); + if res.is_err() { + return res; + } + + let res = self.validate_num_args(matches); + if res.is_err() { + return res; + } + + matches.usage = Some(try!(self.create_usage(&[]))); + + if let Some(sc_name) = subcmd_name { + use std::fmt::Write; + let mut mid_string = String::new(); + if !self.settings.is_set(&AppSettings::SubcommandsNegateReqs) { + let mut hs = self.required.iter().map(|n| *n).collect::>(); + for k in matches.args.keys() { + hs.push(*k); + } + let reqs = self.get_required_from(&hs, Some(matches)); + + for s in reqs.iter() { + write!(&mut mid_string, " {}", s).ok().expect(INTERNAL_ERROR_MSG); + } + } + mid_string.push_str(" "); + if let Some(ref mut sc) = self.subcommands + .iter_mut() + .filter(|s| s.name_slice == &sc_name) + .next() { + let mut new_matches = ArgMatches::new(); + // bin_name should be parent's bin_name + [] + the sc's name separated by + // a space + sc.usage = Some(format!("{}{}{}", + self.bin_name.as_ref().unwrap_or(&String::new()), + if self.bin_name.is_some() { + &*mid_string + } else { + "" + }, + &*sc.name)); + sc.bin_name = Some(format!("{}{}{}", + self.bin_name.as_ref().unwrap_or(&String::new()), + if self.bin_name.is_some() { + " " + } else { + "" + }, + &*sc.name)); + if let Err(e) = sc.get_matches_with(&mut new_matches, it, lossy) { + e.exit(); + } + matches.subcommand = Some(Box::new(SubCommand { + name: sc.name_slice, + matches: new_matches, + })); + } + } else if self.settings.is_set(&AppSettings::SubcommandRequired) { + let bn = self.bin_name.as_ref().unwrap_or(&self.name); + return Err(error_builder::MissingSubcommand(bn, + &try!(self.create_current_usage(matches)))); + } else if self.settings.is_set(&AppSettings::SubcommandRequiredElseHelp) { + let mut out = vec![]; + try!(self.write_help(&mut out)); + return Err(ClapError { + error: String::from_utf8_lossy(&*out).into_owned(), + error_type: ClapErrorType::MissingArgumentOrSubcommand, + }); + } + if (!self.settings.is_set(&AppSettings::SubcommandsNegateReqs) || + matches.subcommand_name().is_none()) && self.validate_required(&matches) { + return Err(error_builder::MissingRequiredArgument( + &*self.get_required_from(&self.required, Some(matches)) + .iter() + .fold(String::new(), + |acc, s| acc + &format!("\n\t{}", Format::Error(s))[..]), + &*try!(self.create_current_usage(matches)))); + } + if matches.args.is_empty() && matches.subcommand_name().is_none() && + self.settings.is_set(&AppSettings::ArgRequiredElseHelp) { + let mut out = vec![]; + try!(self.write_help(&mut out)); + return Err(ClapError { + error: String::from_utf8_lossy(&*out).into_owned(), + error_type: ClapErrorType::MissingArgumentOrSubcommand, + }); + } + Ok(()) + } + + // basically re-implements ClapError::exit except it checks if we should wait + // for input before + // exiting since ClapError doesn't have that info and the error message must be + // printed before + // exiting + fn maybe_wait_for_exit(&self, e: ClapError) -> ! { + if e.use_stderr() { + wlnerr!("{}", e.error); + if self.settings.is_set(&AppSettings::WaitOnError) { + wlnerr!("\nPress [ENTER] / [RETURN] to continue..."); + let mut s = String::new(); + let i = io::stdin(); + i.lock().read_line(&mut s).unwrap(); + } + process::exit(1); + } + + e.exit() + } + + // Prints the version to the user and exits if quit=true + fn print_version(&self, w: &mut W) -> ClapResult<()> { + // Print the binary name if existing, but replace all spaces with hyphens in + // case we're + // dealing with subcommands i.e. git mv is translated to git-mv + try!(writeln!(w, + "{} {}", + &self.bin_name.as_ref().unwrap_or(&self.name)[..].replace(" ", "-"), + self.version.unwrap_or(""))); + + w.flush().map_err(ClapError::from) + } + + #[doc(hidden)] + pub fn create_current_usage(&self, matches: &ArgMatches) -> ClapResult { + self.create_usage(&matches.args + .keys() + .filter(|k| { + if let Some(o) = self.opts + .iter() + .filter(|o| &&o.name == k) + .next() { + !o.settings.is_set(&ArgSettings::Required) + } else if let Some(p) = self.positionals + .values() + .filter(|p| &&p.name == k) + .next() { + !p.settings.is_set(&ArgSettings::Required) + } else { + true + } + }) + .map(|k| *k) + .collect::>()) + } + + fn groups_for_arg(&self, name: &str) -> Option> { + if self.groups.is_empty() { + return None; + } + let mut res = vec![]; + for (g_name, grp) in &self.groups { + for a in &grp.args { + if a == &name { + res.push(*g_name); + } + } + } + if res.is_empty() { + return None; + } + + Some(res) + } + + fn args_in_group(&self, group: &str) -> Vec { + let mut g_vec = vec![]; + let mut args = vec![]; + + for n in &self.groups.get(group).unwrap().args { + if let Some(f) = self.flags.iter().filter(|f| &f.name == n).next() { + args.push(f.to_string()); + } else if let Some(f) = self.opts.iter().filter(|o| &o.name == n).next() { + args.push(f.to_string()); + } else if self.groups.contains_key(n) { + g_vec.push(*n); + } else { + if let Some(p) = self.positionals + .values() + .filter(|p| &p.name == n) + .next() { + args.push(p.to_string()); + } + } + } + + if !g_vec.is_empty() { + for av in g_vec.iter().map(|g| self.args_in_group(g)) { + for a in av { + args.push(a); + } + } + } + assert!(!args.is_empty(), + "ArgGroup '{}' doesn't contain any args", + group); + args.dedup(); + args.iter().map(ToOwned::to_owned).collect() + } + + fn arg_names_in_group(&self, group: &'ar str) -> Vec<&'ar str> { + let mut g_vec = vec![]; + let mut args = vec![]; + + for n in &self.groups.get(group).unwrap().args { + if self.flags.iter().any(|f| &f.name == n) { + args.push(*n); + } else if self.opts.iter().any(|o| &o.name == n) { + args.push(*n); + } else if self.groups.contains_key(n) { + g_vec.push(*n); + } else if self.positionals.values().any(|p| &p.name == n) { + args.push(*n); + } + } + + if !g_vec.is_empty() { + for av in g_vec.iter().map(|g| self.arg_names_in_group(g)) { + for a in av { + args.push(a); + } + } + } + assert!(!args.is_empty(), + "ArgGroup '{}' doesn't contain any args", + group); + args.dedup(); + args.iter().map(|s| *s).collect() + } + + fn get_required_from(&self, + reqs: &[&'ar str], + matches: Option<&ArgMatches>) + -> VecDeque { + let mut c_flags = vec![]; + let mut c_pos = vec![]; + let mut c_opt = vec![]; + let mut grps = vec![]; + for name in reqs { + if self.flags.iter().any(|f| &f.name == name) { + c_flags.push(name); + } else if self.opts.iter().any(|o| &o.name == name) { + c_opt.push(name); + } else if self.groups.contains_key(name) { + grps.push(*name); + } else { + c_pos.push(name); + } + } + let mut tmp_f = vec![]; + for f in &c_flags { + if let Some(f) = self.flags.iter().filter(|flg| &&flg.name == f).next() { + if let Some(ref rl) = f.requires { + for r in rl.iter() { + if !reqs.contains(r) { + if self.flags.iter().any(|f| &f.name == r) { + tmp_f.push(r); + } else if self.opts.iter().any(|o| &o.name == r) { + c_opt.push(r); + } else if self.groups.contains_key(r) { + grps.push(*r); + } else { + c_pos.push(r); + } + } + } + } + } + } + for f in tmp_f.into_iter() { + c_flags.push(f); + } + let mut tmp_o = vec![]; + for f in &c_opt { + if let Some(f) = self.opts.iter().filter(|o| &&o.name == f).next() { + if let Some(ref rl) = f.requires { + for r in rl.iter() { + if !reqs.contains(r) { + if self.flags.iter().any(|f| &f.name == r) { + c_flags.push(r); + } else if self.opts.iter().any(|o| &o.name == r) { + tmp_o.push(r); + } else if self.groups.contains_key(r) { + grps.push(*r); + } else { + c_pos.push(r); + } + } + } + } + } + } + for f in tmp_o.into_iter() { + c_opt.push(f); + } + let mut tmp_p = vec![]; + for p in c_pos.iter() { + if let Some(p) = self.positionals.values().filter(|pos| &&pos.name == p).next() { + if let Some(ref rl) = p.requires { + for r in rl.iter() { + if !reqs.contains(r) { + if self.flags.iter().any(|f| &f.name == r) { + c_flags.push(r); + } else if self.opts.iter().any(|o| &o.name == r) { + c_opt.push(r); + } else if self.groups.contains_key(r) { + grps.push(*r); + } else { + tmp_p.push(r); + } + } + } + } + } + } + for f in tmp_p.into_iter() { + c_pos.push(f); + } + + let mut ret_val = VecDeque::new(); + + let mut pmap = BTreeMap::new(); + for p in c_pos.into_iter() { + if matches.is_some() && matches.as_ref().unwrap().is_present(p) { + continue; + } + if let Some(p) = self.positionals.values().filter(|x| &x.name == p).next() { + pmap.insert(p.index, format!("{}", p)); + } + } + for (_, s) in pmap { + ret_val.push_back(s); + } + for f in c_flags.into_iter() { + if matches.is_some() && matches.as_ref().unwrap().is_present(f) { + continue; + } + ret_val.push_back(format!("{}", + self.flags + .iter() + .filter(|flg| &flg.name == f) + .next() + .unwrap())); + } + for o in c_opt.into_iter() { + if matches.is_some() && matches.as_ref().unwrap().is_present(o) { + continue; + } + ret_val.push_back(format!("{}", + self.opts + .iter() + .filter(|opt| &opt.name == o) + .next() + .unwrap())); + } + for g in grps.into_iter() { + let g_string = self.args_in_group(g) + .iter() + .fold(String::new(), |acc, s| acc + &format!(" {} |", s)[..]); + ret_val.push_back(format!("[{}]", &g_string[..g_string.len() - 1])); + } + + ret_val + } + + fn verify_positionals(&mut self) { + // Because you must wait until all arguments have been supplied, this is the first chance + // to make assertions on positional argument indexes + // + // Firt we verify that the index highest supplied index, is equal to the number of + // positional arguments to verify there are no gaps (i.e. supplying an index of 1 and 3 + // but no 2) + if let Some((idx, ref p)) = self.positionals.iter().rev().next() { + if idx != self.positionals.len() { + panic!("Found positional argument \"{}\" who's index is {} but there are only {} \ + positional arguments defined", + p.name, + idx, + self.positionals.len()); + } + } + + // Next we verify that only the highest index has a .multiple(true) (if any) + assert!(!self.positionals + .values() + .any(|a| { + a.settings.is_set(&ArgSettings::Multiple) && + (a.index as usize != self.positionals.len()) + }), + "Only the positional argument with the highest index may accept multiple values"); + + // If it's required we also need to ensure all previous positionals are + // required too + let mut found = false; + for p in self.positionals.values().rev() { + if !found { + if p.settings.is_set(&ArgSettings::Required) { + found = true; + continue; + } + } else { + assert!(p.settings.is_set(&ArgSettings::Required), + "Found positional argument which is not required with a lower index than a \ + required positional argument: {:?} index {}", + p.name, + p.index); + } + } + } + + fn propogate_globals(&mut self) { + for sc in self.subcommands.iter_mut() { + // We have to create a new scope in order to tell rustc the borrow of `sc` is + // done and + // to recursively call this method + { + for a in self.global_args.iter() { + sc.add_arg(a.into()); + } + } + sc.propogate_globals(); + } + } + + fn blacklisted_from(&self, name: &str, matches: &ArgMatches) -> Option { + for k in matches.args.keys() { + if let Some(f) = self.flags.iter().filter(|f| &f.name == k).next() { + if let Some(ref bl) = f.blacklist { + if bl.contains(&name) { + return Some(format!("{}", f)); + } + } + } + if let Some(o) = self.opts.iter().filter(|o| &o.name == k).next() { + if let Some(ref bl) = o.blacklist { + if bl.contains(&name) { + return Some(format!("{}", o)); + } + } + } + if let Some(pos) = self.positionals.values().filter(|p| &p.name == k).next() { + if let Some(ref bl) = pos.blacklist { + if bl.contains(&name) { + return Some(format!("{}", pos)); + } + } + } + } + None + } + + fn overriden_from(&self, name: &'ar str, matches: &ArgMatches) -> Option<&'ar str> { + for k in matches.args.keys() { + if let Some(f) = self.flags.iter().filter(|f| &f.name == k).next() { + if let Some(ref bl) = f.overrides { + if bl.contains(&name) { + return Some(f.name); + } + } + } + if let Some(o) = self.opts.iter().filter(|o| &o.name == k).next() { + if let Some(ref bl) = o.overrides { + if bl.contains(&name) { + return Some(o.name); + } + } + } + if let Some(pos) = self.positionals.values().filter(|p| &p.name == k).next() { + if let Some(ref bl) = pos.overrides { + if bl.contains(&name) { + return Some(pos.name); + } + } + } + } + None + } + + fn create_help_and_version(&mut self) { + // name is "hclap_help" because flags are sorted by name + if !self.flags.iter().any(|a| a.long.is_some() && a.long.unwrap() == "help") { + if self.help_short.is_none() && !self.short_list.contains(&'h') { + self.help_short = Some('h'); + } + let arg = FlagBuilder { + name: "hclap_help", + short: self.help_short, + long: Some("help"), + help: Some("Prints help information"), + blacklist: None, + requires: None, + overrides: None, + settings: ArgFlags::new(), + }; + self.long_list.push("help"); + self.flags.push(arg); + } + if !self.settings.is_set(&AppSettings::DisableVersion) && + !self.flags.iter().any(|a| a.long.is_some() && a.long.unwrap() == "version") { + if self.version_short.is_none() && !self.short_list.contains(&'V') { + self.version_short = Some('V'); + } + // name is "vclap_version" because flags are sorted by name + let arg = FlagBuilder { + name: "vclap_version", + short: self.version_short, + long: Some("version"), + help: Some("Prints version information"), + blacklist: None, + requires: None, + overrides: None, + settings: ArgFlags::new(), + }; + self.long_list.push("version"); + self.flags.push(arg); + } + if !self.subcommands.is_empty() && + !self.subcommands + .iter() + .any(|a| a.name_slice == "help") { + self.subcommands.push(App::new("help").about("Prints this message")); + } + } + + fn check_for_help_and_version(&self, arg: char) -> ClapResult<()> { + if let Some(h) = self.help_short { + if h == arg { + try!(self.print_help()); + return Err(ClapError { + error: String::new(), + error_type: ClapErrorType::HelpDisplayed, + }); + } + } + if let Some(v) = self.version_short { + if v == arg { + let out = io::stdout(); + let mut buf_w = BufWriter::new(out.lock()); + try!(self.print_version(&mut buf_w)); + return Err(ClapError { + error: String::new(), + error_type: ClapErrorType::VersionDisplayed, + }); + } + } + + Ok(()) + } + + fn parse_long_arg<'av>(&mut self, + matches: &mut ArgMatches<'ar, 'ar>, + full_arg: &'av str) + -> ClapResult> { + let mut arg = full_arg.trim_left_matches(|c| c == '-'); + + if arg == "help" && self.settings.is_set(&AppSettings::NeedsLongHelp) { + try!(self.print_help()); + return Err(ClapError { + error: String::new(), + error_type: ClapErrorType::HelpDisplayed, + }); + } else if arg == "version" && self.settings.is_set(&AppSettings::NeedsLongVersion) { + let out = io::stdout(); + let mut buf_w = BufWriter::new(out.lock()); + try!(self.print_version(&mut buf_w)); + return Err(ClapError { + error: String::new(), + error_type: ClapErrorType::VersionDisplayed, + }); + } + + let mut arg_val: Option<&'av str> = None; + + if arg.contains("=") { + let arg_vec: Vec<_> = arg.split("=").collect(); + arg = arg_vec[0]; + if let Some(ref v) = self.opts + .iter() + .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) + .next() { + // prevents "--config= value" typo + if arg_vec[1].len() == 0 && !v.settings.is_set(&ArgSettings::EmptyValues) { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: None, + }); + } + } + matches.args.insert(v.name, + MatchedArg { + occurrences: 1, + values: None, + }); + return Err(error_builder::EmptyValue( + &*format!("--{}", arg), + &*try!(self.create_current_usage(matches)) + )); + } + arg_val = Some(arg_vec[1]); + } + } + + if let Some(ref v) = self.opts + .iter() + .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) + .next() { + // Ensure this option isn't on the master mutually excludes list + if self.blacklist.contains(&v.name) { + matches.args.remove(v.name); + return Err(error_builder::ArgumentConflict( + format!("--{}", arg), + self.blacklisted_from(v.name, &matches), + try!(self.create_current_usage(matches)))); + } + if self.overrides.contains(&v.name) { + debugln!("it is..."); + debugln!("checking who defined it..."); + if let Some(ref name) = self.overriden_from(v.name, matches) { + debugln!("found {}", name); + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.remove(grp); + } + } + matches.args.remove(name); + remove_overriden!(self, name); + } + } + if let Some(ref or) = v.overrides { + for pa in or { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.remove(grp); + } + } + matches.args.remove(pa); + remove_overriden!(self, pa); + self.overrides.push(pa); + } + } + + if matches.args.contains_key(v.name) { + if !v.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::UnexpectedMultipleUsage( + &*format!("--{}", arg), + &*try!(self.create_current_usage(matches)))); + } + if let Some(av) = arg_val { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + if let Some(ref mut o) = matches.args.get_mut(grp) { + o.occurrences += 1; + if let Some(ref mut vals) = o.values { + let len = (vals.len() + 1) as u8; + vals.insert(len, av.to_owned()); + } + } + } + } + if let Some(ref mut o) = matches.args.get_mut(v.name) { + o.occurrences += 1; + if let Some(ref mut vals) = o.values { + let len = (vals.len() + 1) as u8; + vals.insert(len, av.to_owned()); + } + } + // The validation must come AFTER inserting into 'matches' or the usage string + // can't be built + if let Err(e) = self.validate_value(v, av, matches) { + return Err(e); + } + } + } else { + let mut bm = BTreeMap::new(); + if let Some(val) = arg_val { + bm.insert(1, val.to_owned()); + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: Some(bm.clone()), + }); + } + } + matches.args.insert(v.name, + MatchedArg { + occurrences: 1, + values: Some(bm), + }); + // The validation must come AFTER inserting into 'matches' or the usage string + // can't be built + if let Err(e) = self.validate_value(v, val, matches) { + return Err(e); + } + } else { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: Some(bm.clone()), + }); + } + } + matches.args.insert(v.name, + MatchedArg { + occurrences: 0, + values: Some(bm), + }); + } + } + if let Some(ref bl) = v.blacklist { + for name in bl { + self.blacklist.push(name); + vec_remove!(self.overrides, name); + vec_remove!(self.required, name); + } + } + + if let Some(ref reqs) = v.requires { + // Add all required args which aren't already found in matches to the + // final required list + for n in reqs { + if matches.args.contains_key(n) { + continue; + } + + self.required.push(n); + } + } + + parse_group_reqs!(self, v); + + match arg_val { + None => { + return Ok(Some(v.name)); + } + _ => { + return Ok(None); + } + } + } + + if let Some(v) = self.flags + .iter() + .filter(|&v| v.long.is_some() && v.long.unwrap() == arg) + .next() { + // Ensure this flag isn't on the mutually excludes list + if self.blacklist.contains(&v.name) { + matches.args.remove(v.name); + return Err(error_builder::ArgumentConflict( + v.to_string(), + self.blacklisted_from(v.name, &matches), + try!(self.create_current_usage(matches)))); + } + if self.overrides.contains(&v.name) { + debugln!("it is..."); + debugln!("checking who defined it..."); + if let Some(ref name) = self.overriden_from(v.name, matches) { + debugln!("found {}", name); + matches.args.remove(name); + remove_overriden!(self, name); + } + } + if let Some(ref or) = v.overrides { + for pa in or { + matches.args.remove(pa); + remove_overriden!(self, pa); + self.overrides.push(pa); + } + } + + // Make sure this isn't one being added multiple times if it doesn't suppor it + if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::UnexpectedMultipleUsage( + &*v.to_string(), + &*try!(self.create_current_usage(matches)))); + } + + let mut done = false; + if let Some(ref mut f) = matches.args.get_mut(v.name) { + done = true; + f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { + f.occurrences + 1 + } else { + 1 + }; + } + if !done { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: None, + }); + } + } + matches.args.insert(v.name, + MatchedArg { + occurrences: 1, + values: None, + }); + } + + + // Add all of this flags "mutually excludes" list to the master list + if let Some(ref ov) = v.overrides { + for name in ov { + self.overrides.push(name); + } + } + if let Some(ref bl) = v.blacklist { + for name in bl { + self.blacklist.push(name); + vec_remove!(self.overrides, name); + vec_remove!(self.required, name); + } + } + + // Add all required args which aren't already found in matches to the master + // list + if let Some(ref reqs) = v.requires { + for n in reqs { + if matches.args.contains_key(n) { + continue; + } + + self.required.push(n); + } + } + + parse_group_reqs!(self, v); + + return Ok(None); + } + + let suffix = suggestions::did_you_mean_suffix(arg, + self.long_list.iter(), + suggestions::DidYouMeanMessageStyle::LongFlag); + if let Some(name) = suffix.1 { + if let Some(ref opt) = self.opts + .iter() + .filter_map(|o| { + if o.long.is_some() && o.long.unwrap() == name { + Some(o.name) + } else { + None + } + }) + .next() { + if let Some(ref vec) = self.groups_for_arg(opt) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: None, + }); + } + } + matches.args.insert(opt, + MatchedArg { + occurrences: 0, + values: None, + }); + } else if let Some(ref flg) = self.flags + .iter() + .filter_map(|f| { + if f.long.is_some() && f.long.unwrap() == name { + Some(f.name) + } else { + None + } + }) + .next() { + if let Some(ref vec) = self.groups_for_arg(flg) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: None, + }); + } + } + matches.args.insert(flg, + MatchedArg { + occurrences: 0, + values: None, + }); + } + } + + Err(error_builder::InvalidArgument(&*format!("--{}", arg), + Some(&*suffix.0), + &*try!(self.create_current_usage(matches)))) + } + + fn validate_value(&self, v: &OptBuilder, av: &str, matches: &ArgMatches) -> ClapResult<()> { + if let Some(ref p_vals) = v.possible_vals { + if !p_vals.contains(&av) { + return Err( + error_builder::InvalidValue(av, + p_vals, + &v.to_string(), + &*try!(self.create_current_usage(matches)))); + } + } + if !v.settings.is_set(&ArgSettings::EmptyValues) && av.is_empty() && + matches.args.contains_key(v.name) { + return Err(error_builder::EmptyValue(&*v.to_string(), + &*try!(self.create_current_usage(matches)))); + } + if let Some(ref vtor) = v.validator { + if let Err(e) = vtor(av.to_owned()) { + return Err(error_builder::ValueValidationError(&*e)); + } + } + Ok(()) + } + + fn parse_short_arg(&mut self, + matches: &mut ArgMatches<'ar, 'ar>, + full_arg: &str) + -> ClapResult> { + let arg = &full_arg[..].trim_left_matches(|c| c == '-'); + for c in arg.chars() { + try!(self.check_for_help_and_version(c)); + + // Check for matching short in options, and return the name + // (only ones with shorts, of course) + if let Some(v) = self.opts + .iter() + .filter(|&v| v.short.is_some() && v.short == Some(c)) + .next() { + let mut ret = Some(v.name); + // Ensure this option isn't on the master mutually excludes list + if self.blacklist.contains(&v.name) { + matches.args.remove(v.name); + return Err(error_builder::ArgumentConflict( + v.to_string(), + self.blacklisted_from(v.name, &matches), + try!(self.create_current_usage(matches)))); + } + if self.overrides.contains(&v.name) { + if let Some(ref name) = self.overriden_from(v.name, matches) { + matches.args.remove(&*name); + remove_overriden!(self, name); + } + } + if let Some(ref or) = v.overrides { + for pa in or { + matches.args.remove(pa); + remove_overriden!(self, pa); + self.overrides.push(pa); + } + } + + if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::UnexpectedMultipleUsage( + &*format!("-{}", arg), + &*try!(self.create_current_usage(matches)))); + } + + // New scope for lifetimes + let val: Vec<&str> = arg.splitn(2, c).collect(); + { + let ma = matches.args.entry(v.name).or_insert(MatchedArg { + // occurrences will be incremented on getting a value + occurrences: 0, + values: Some(BTreeMap::new()), + }); + if !val[1].is_empty() { + if !v.settings.is_set(&ArgSettings::Multiple) { + ret = None; + } + if let Some(ref mut vals) = ma.values { + let len = vals.len() as u8 + 1; + vals.insert(len, val[1].to_owned()); + } + ma.occurrences += 1; + } + } + + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + let ma_g = matches.args.entry(grp).or_insert(MatchedArg { + occurrences: 0, + values: Some(BTreeMap::new()), + }); + if !val[1].is_empty() { + if let Some(ref mut vals) = ma_g.values { + let len = vals.len() as u8 + 1; + vals.insert(len, val[1].to_owned()); + } + ma_g.occurrences += 1; + } + } + } + + if let Some(ref bl) = v.blacklist { + for name in bl { + self.blacklist.push(name); + vec_remove!(self.overrides, name); + vec_remove!(self.required, name); + } + } + + if let Some(ref reqs) = v.requires { + // Add all required args which aren't already found in matches to the + // final required list + for n in reqs { + if matches.args.contains_key(n) { + continue; + } + + self.required.push(n); + } + } + + parse_group_reqs!(self, v); + + return Ok(ret); + } + + match self.parse_single_short_flag(matches, c) { + Ok(b) => { + if !b { + return Err(error_builder::InvalidArgument( + &*format!("-{}", c), + None, + &*try!(self.create_current_usage(matches)))); + } + } + Err(e) => return Err(e), + } + } + Ok(None) + } + + fn parse_single_short_flag(&mut self, + matches: &mut ArgMatches<'ar, 'ar>, + arg: char) + -> ClapResult { + if let Some(v) = self.flags + .iter() + .filter(|&v| v.short.is_some() && v.short.unwrap() == arg) + .next() { + // Ensure this flag isn't on the mutually excludes list + if self.blacklist.contains(&v.name) { + matches.args.remove(v.name); + return Err(error_builder::ArgumentConflict( + v.to_string(), + self.blacklisted_from(v.name, &matches), + try!(self.create_current_usage(matches)))); + } + if self.overrides.contains(&v.name) { + debugln!("it is..."); + debugln!("checking who defined it..."); + if let Some(ref name) = self.overriden_from(v.name, matches) { + debugln!("found {}", name); + matches.args.remove(name); + remove_overriden!(self, name); + } + } + if let Some(ref or) = v.overrides { + for pa in or { + matches.args.remove(pa); + remove_overriden!(self, pa); + self.overrides.push(pa); + } + } + + // Make sure this isn't one being added multiple times if it doesn't suppor it + if matches.args.contains_key(v.name) && !v.settings.is_set(&ArgSettings::Multiple) { + return Err(error_builder::UnexpectedMultipleUsage( + &*format!("-{}", arg), + &*try!(self.create_current_usage(matches)))); + } + + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + if let Some(ref mut f) = matches.args.get_mut(grp) { + f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { + f.occurrences + 1 + } else { + 1 + }; + } + } + } + let mut done = false; + if let Some(ref mut f) = matches.args.get_mut(v.name) { + done = true; + f.occurrences = if v.settings.is_set(&ArgSettings::Multiple) { + f.occurrences + 1 + } else { + 1 + }; + } + if !done { + if let Some(ref vec) = self.groups_for_arg(v.name) { + for grp in vec { + matches.args.insert(grp, + MatchedArg { + occurrences: 1, + values: None, + }); + } + } + matches.args.insert(v.name, + MatchedArg { + occurrences: 1, + values: None, + }); + } + + if let Some(ref bl) = v.blacklist { + for name in bl { + self.blacklist.push(name); + vec_remove!(self.overrides, name); + vec_remove!(self.required, name); + } + } + + // Add all required args which aren't already found in matches to the master + // list + if let Some(ref reqs) = v.requires { + for n in reqs { + if matches.args.contains_key(n) { + continue; + } + + self.required.push(n); + } + } + + parse_group_reqs!(self, v); + + return Ok(true); + } + Ok(false) + } + + fn validate_blacklist(&self, matches: &mut ArgMatches<'ar, 'ar>) -> ClapResult<()> { + for name in self.blacklist.iter() { + if matches.args.contains_key(name) { + matches.args.remove(name); + return Err(error_builder::ArgumentConflict( + format!("{}", Format::Warning( + if let Some(f) = self.flags.iter().filter(|f| &f.name == name).next() { + f.to_string() + } else if let Some(o) = self.opts.iter() + .filter(|o| &o.name == name) + .next() { + o.to_string() + } else { + match self.positionals.values() + .filter(|p| p.name == *name) + .next() { + Some(p) => p.to_string(), + None => format!("'{}'", name) + } + } + )), + self.blacklisted_from(name, &matches), + try!(self.create_current_usage(matches)))); + } else if self.groups.contains_key(name) { + for n in self.arg_names_in_group(name) { + if matches.args.contains_key(n) { + matches.args.remove(n); + return Err(error_builder::ArgumentConflict( + format!("{}", Format::Warning( + if let Some(f) = self.flags.iter() + .filter(|f| &f.name == name) + .next() { + f.to_string() + } else if let Some(o) = self.opts + .iter() + .filter(|o| &o.name == name) + .next() { + o.to_string() + } else { + match self.positionals.values() + .filter(|p| p.name == n) + .next() { + Some(p) => p.to_string(), + None => format!("'{}'", name) + } + } + )), + self.blacklisted_from(name, &matches), + try!(self.create_current_usage(matches)))); + } + } + } + } + Ok(()) + } + + fn validate_num_args(&self, matches: &mut ArgMatches<'ar, 'ar>) -> ClapResult<()> { + for (name, ma) in matches.args.iter() { + if self.groups.contains_key(name) { + continue; + } else if let Some(ref vals) = ma.values { + if let Some(f) = self.opts + .iter() + .filter(|o| &o.name == name) + .next() { + if let Some(num) = f.num_vals { + let should_err = if f.settings.is_set(&ArgSettings::Multiple) { + ((vals.len() as u8) % num) != 0 + } else { + num != (vals.len() as u8) + }; + if should_err { + return Err(error_builder::WrongNumValues( + &*f.to_string(), + num, + if f.settings.is_set(&ArgSettings::Multiple) { + (vals.len() % num as usize) + } else { + vals.len() + }, + if vals.len() == 1 || + (f.settings.is_set(&ArgSettings::Multiple) && + (vals.len() % num as usize) == 1) { + "as" + } else { + "ere" + }, + &*try!(self.create_current_usage(matches)))); + } + } + if let Some(num) = f.max_vals { + if (vals.len() as u8) > num { + return Err(error_builder::TooManyValues( + vals.get(vals.keys() + .last() + .expect("error getting last key. This is a bug")) + .expect("failed to retrieve last value. This is a bug"), + &f.to_string(), + &try!(self.create_current_usage(matches)))); + } + } + if let Some(num) = f.min_vals { + if (vals.len() as u8) < num { + return Err(error_builder::TooFewValues( + &*f.to_string(), + num, + vals.len(), + &*try!(self.create_current_usage(matches)))); + } + } + } else if let Some(f) = self.positionals + .values() + .filter(|p| &p.name == name) + .next() { + if let Some(num) = f.num_vals { + if num != vals.len() as u8 { + return Err(error_builder::WrongNumValues( + &*f.to_string(), + num, + vals.len(), + if vals.len() == 1 { + "as" + } else { + "ere" + }, + &*try!(self.create_current_usage(matches)))); + } + } + if let Some(max) = f.max_vals { + if (vals.len() as u8) > max { + return Err(error_builder::TooManyValues( + vals.get(vals.keys() + .last() + .expect("error getting last key. This is a bug")) + .expect("failed to retrieve last value. This is a bug"), + &f.to_string(), + &try!(self.create_current_usage(matches)))); + } + } + if let Some(min) = f.min_vals { + if (vals.len() as u8) < min { + return Err(error_builder::TooFewValues( + &*f.to_string(), + min, + vals.len(), + &*try!(self.create_current_usage(matches)))); + } + } + } + } + } + Ok(()) + } + + fn validate_required(&self, matches: &ArgMatches<'ar, 'ar>) -> bool { + 'outer: for name in self.required.iter() { + if matches.args.contains_key(name) { + continue 'outer; + } + for grp in self.groups.values() { + if grp.args.contains(name) { + continue 'outer; + } + } + if let Some(a) = self.flags.iter().filter(|f| &f.name == name).next() { + if let Some(ref bl) = a.blacklist { + for n in bl.iter() { + if matches.args.contains_key(n) { + continue 'outer; + } else if self.groups.contains_key(n) { + let grp = self.groups.get(n).unwrap(); + for an in grp.args.iter() { + if matches.args.contains_key(an) { + continue 'outer; + } + } + } + } + } + } + if let Some(a) = self.opts.iter().filter(|o| &o.name == name).next() { + if let Some(ref bl) = a.blacklist { + for n in bl.iter() { + if matches.args.contains_key(n) { + continue 'outer; + } else if self.groups.contains_key(n) { + let grp = self.groups.get(n).unwrap(); + for an in grp.args.iter() { + if matches.args.contains_key(an) { + continue 'outer; + } + } + } + } + } + } + // because positions use different keys, we dont use the macro + match self.positionals.values().filter(|p| &p.name == name).next() { + Some(p) => { + if let Some(ref bl) = p.blacklist { + for n in bl.iter() { + if matches.args.contains_key(n) { + continue 'outer; + } else if self.groups.contains_key(n) { + let grp = self.groups.get(n).unwrap(); + for an in grp.args.iter() { + if matches.args.contains_key(an) { + continue 'outer; + } + } + } + } + } + } + None => (), + } + return true; + } + false + } +} diff --git a/src/app/errors.rs b/src/errors.rs similarity index 96% rename from src/app/errors.rs rename to src/errors.rs index aa7fa0bc..95a8d44d 100644 --- a/src/app/errors.rs +++ b/src/errors.rs @@ -11,6 +11,7 @@ pub type ClapResult = Result; #[doc(hidden)] #[allow(non_snake_case)] pub mod error_builder { + use suggestions; use super::ClapError; use super::ClapErrorType as cet; use fmt::Format; @@ -48,19 +49,29 @@ pub mod error_builder { Format::Warning(arg.as_ref()), usage.as_ref(), Format::Good("--help")), - error_type: cet::InvalidValue, + error_type: cet::EmptyValue, } } /// Error occurs when some possible values were set, but clap found unexpected value pub fn InvalidValue(bad_val: S, + good_vals: &[S], arg: S, - valid_vals: S, - did_you_mean: S, usage: S) -> ClapError where S: AsRef { + let suffix = suggestions::did_you_mean_suffix(arg.as_ref(), + good_vals.iter(), + suggestions::DidYouMeanMessageStyle::EnumValue); + + let mut sorted = vec![]; + for v in good_vals { + sorted.push(v.as_ref()); + } + sorted.sort(); + let valid_values = sorted.iter() + .fold(String::new(), |a, name| a + &format!( " {}", name)[..]); ClapError { error: format!("{} '{}' isn't a valid value for '{}'\n\t\ [valid values:{}]\n\ @@ -70,8 +81,8 @@ pub mod error_builder { Format::Error("error:"), Format::Warning(bad_val.as_ref()), Format::Warning(arg.as_ref()), - valid_vals.as_ref(), - did_you_mean.as_ref(), + valid_values, + suffix.0, usage.as_ref(), Format::Good("--help")), error_type: cet::InvalidValue, @@ -280,7 +291,6 @@ pub mod error_builder { error_type: cet::UnexpectedMultipleUsage, } } - } /// Command line argument parser error types diff --git a/src/lib.rs b/src/lib.rs index ed2abc57..6ff963ae 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -671,8 +671,9 @@ extern crate vec_map; #[cfg(feature = "yaml")] pub use yaml_rust::YamlLoader; pub use args::{Arg, ArgGroup, ArgMatches, SubCommand}; -pub use app::{App, AppSettings, ClapError, ClapErrorType}; +pub use app::{App, AppSettings}; pub use fmt::Format; +pub use errors::{ClapError, ClapErrorType}; #[macro_use] mod macros; @@ -680,3 +681,5 @@ mod app; mod args; mod usageparser; mod fmt; +mod suggestions; +mod errors; diff --git a/src/app/suggestions.rs b/src/suggestions.rs similarity index 60% rename from src/app/suggestions.rs rename to src/suggestions.rs index 00a7e6a8..6ac37d80 100644 --- a/src/app/suggestions.rs +++ b/src/suggestions.rs @@ -1,6 +1,8 @@ #[cfg(feature = "suggestions")] use strsim; +use fmt::Format; + /// Produces a string from a given list of possible values which is similar to /// the passed in value `v` with a certain confidence. /// Thus in a list of possible values like ["foo", "bar"], the value "fop" will yield @@ -34,6 +36,34 @@ pub fn did_you_mean<'a, T, I>(_: &str, _: I) -> Option<&'a str> None } +/// Returns a suffix that can be empty, or is the standard 'did you mean phrase +#[cfg_attr(feature = "lints", allow(needless_lifetimes))] +pub fn did_you_mean_suffix<'z, T, I>(arg: &str, + values: I, + style: DidYouMeanMessageStyle) + -> (String, Option<&'z str>) + where T: AsRef + 'z, + I: IntoIterator +{ + match did_you_mean(arg, values) { + Some(candidate) => { + let mut suffix = "\n\tDid you mean ".to_owned(); + match style { + DidYouMeanMessageStyle::LongFlag => + suffix.push_str(&*format!("{}", Format::Good("--"))), + DidYouMeanMessageStyle::EnumValue => suffix.push('\''), + } + suffix.push_str(&Format::Good(candidate).to_string()[..]); + if let DidYouMeanMessageStyle::EnumValue = style { + suffix.push('\''); + } + suffix.push_str(" ?"); + (suffix, Some(candidate)) + } + None => (String::new(), None), + } +} + /// A helper to determine message formatting pub enum DidYouMeanMessageStyle { /// Suggested value is a long flag