From 2bedad2bfa3a8d197887f82717785db8491653e9 Mon Sep 17 00:00:00 2001 From: ldm0 Date: Tue, 23 Feb 2021 17:44:31 +0000 Subject: [PATCH 1/4] Split debug_asserts from Arg --- src/build/app/debug_asserts.rs | 4 +-- src/build/arg/debug_asserts.rs | 45 +++++++++++++++++++++++++++++++ src/build/arg/mod.rs | 48 ++-------------------------------- 3 files changed, 49 insertions(+), 48 deletions(-) create mode 100644 src/build/arg/debug_asserts.rs diff --git a/src/build/app/debug_asserts.rs b/src/build/app/debug_asserts.rs index 5fb9c3166c3..e67ec3f92f1 100644 --- a/src/build/app/debug_asserts.rs +++ b/src/build/app/debug_asserts.rs @@ -1,4 +1,4 @@ -use crate::{App, AppSettings, ArgSettings, ValueHint}; +use crate::{build::arg::debug_asserts::assert_arg, App, AppSettings, ArgSettings, ValueHint}; use std::cmp::Ordering; #[derive(Eq)] @@ -63,7 +63,7 @@ pub(crate) fn assert_app(app: &App) { } for arg in app.args.args() { - arg._debug_asserts(); + assert_arg(arg); if let Some(s) = arg.short.as_ref() { short_flags.push(Flag::Arg(format!("-{}", s), &*arg.name)); diff --git a/src/build/arg/debug_asserts.rs b/src/build/arg/debug_asserts.rs new file mode 100644 index 00000000000..4a03f4e1e1a --- /dev/null +++ b/src/build/arg/debug_asserts.rs @@ -0,0 +1,45 @@ +use crate::{Arg, ArgSettings, ValueHint}; + +pub(crate) fn assert_arg(arg: &Arg) { + debug!("Arg::_debug_asserts:{}", arg.name); + + // Self conflict + // TODO: this check should be recursive + assert!( + !arg.blacklist.iter().any(|x| *x == arg.id), + "Argument '{}' cannot conflict with itarg", + arg.name, + ); + + if arg.value_hint != ValueHint::Unknown { + assert!( + arg.is_set(ArgSettings::TakesValue), + "Argument '{}' has value hint but takes no value", + arg.name + ); + + if arg.value_hint == ValueHint::CommandWithArguments { + assert!( + arg.is_set(ArgSettings::MultipleValues), + "Argument '{}' uses hint CommandWithArguments and must accept multiple values", + arg.name + ) + } + } + + if arg.index.is_some() { + assert!( + arg.short.is_none() && arg.long.is_none(), + "Argument '{}' is a positional argument and can't have short or long name versions", + arg.name + ); + } + + if arg.is_set(ArgSettings::Required) { + assert!( + arg.default_vals.is_empty(), + "Argument '{}' is required and can't have a default value", + arg.name + ); + } +} diff --git a/src/build/arg/mod.rs b/src/build/arg/mod.rs index 42e10a836f6..61aa8b2935b 100644 --- a/src/build/arg/mod.rs +++ b/src/build/arg/mod.rs @@ -1,3 +1,5 @@ +#[cfg(debug_assertions)] +pub mod debug_asserts; mod settings; #[cfg(test)] mod tests; @@ -4585,52 +4587,6 @@ impl<'help> Arg<'help> { } } -impl Arg<'_> { - pub(crate) fn _debug_asserts(&self) { - debug!("Arg::_debug_asserts:{}", self.name); - - // Self conflict - // TODO: this check should be recursive - assert!( - !self.blacklist.iter().any(|x| *x == self.id), - "Argument '{}' cannot conflict with itself", - self.name, - ); - - if self.value_hint != ValueHint::Unknown { - assert!( - self.is_set(ArgSettings::TakesValue), - "Argument '{}' has value hint but takes no value", - self.name - ); - - if self.value_hint == ValueHint::CommandWithArguments { - assert!( - self.is_set(ArgSettings::MultipleValues), - "Argument '{}' uses hint CommandWithArguments and must accept multiple values", - self.name - ) - } - } - - if self.index.is_some() { - assert!( - self.short.is_none() && self.long.is_none(), - "Argument '{}' is a positional argument and can't have short or long name versions", - self.name - ); - } - - if self.is_set(ArgSettings::Required) { - assert!( - self.default_vals.is_empty(), - "Argument '{}' is required and can't have a default value", - self.name - ); - } - } -} - #[cfg(feature = "yaml")] impl<'help> From<&'help Yaml> for Arg<'help> { /// Creates a new instance of [`Arg`] from a .yaml (YAML) file. From 886b8737094342cf27bf083829d2343175dc0f48 Mon Sep 17 00:00:00 2001 From: ldm0 Date: Wed, 24 Feb 2021 15:07:57 +0000 Subject: [PATCH 2/4] Demangle interlinking flags --- README.md | 1 + benches/03_complex.rs | 6 ++ benches/04_new_help.rs | 2 + benches/05_ripgrep.rs | 2 + benches/06_rustup.rs | 1 + clap_generate/examples/value_hints.rs | 1 + clap_generate/tests/value_hints.rs | 1 + examples/05_flag_args.rs | 1 + examples/17_yaml.yaml | 1 + examples/22_stop_parsing_with_--.rs | 2 +- examples/23_flag_subcommands_pacman.rs | 2 + src/build/app/settings.rs | 4 +- src/build/arg/debug_asserts.rs | 42 ++++++++++++- src/build/arg/mod.rs | 78 ++++++++++++++++------- src/build/arg/settings.rs | 18 +++--- src/build/usage_parser.rs | 1 + src/macros.rs | 2 +- src/parse/errors.rs | 1 + src/parse/matches/arg_matches.rs | 1 + src/parse/parser.rs | 1 + tests/app_settings.rs | 2 + tests/delimiters.rs | 5 +- tests/fixtures/app.yaml | 4 ++ tests/fixtures/app_2space.yaml | 1 + tests/flag_subcommands.rs | 6 ++ tests/global_args.rs | 1 + tests/grouped_values.rs | 23 ++++++- tests/help.rs | 39 ++++++++++-- tests/multiple_values.rs | 87 +++++++++++++++++++++----- tests/opts.rs | 19 +++++- tests/positionals.rs | 10 ++- tests/posix_compatible.rs | 2 + tests/possible_values.rs | 2 + tests/subcommands.rs | 2 +- 34 files changed, 306 insertions(+), 65 deletions(-) diff --git a/README.md b/README.md index f13c74a23fd..822c03ecf2e 100644 --- a/README.md +++ b/README.md @@ -229,6 +229,7 @@ fn main() { .arg(Arg::new("v") .short('v') .multiple(true) + .takes_value(true) .about("Sets the level of verbosity")) .subcommand(App::new("test") .about("controls testing features") diff --git a/benches/03_complex.rs b/benches/03_complex.rs index 4e152dd37fa..d489e769daf 100644 --- a/benches/03_complex.rs +++ b/benches/03_complex.rs @@ -57,6 +57,7 @@ pub fn build_from_builder(c: &mut Criterion) { .about("tests options") .short('o') .long("option") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences), ) @@ -99,6 +100,7 @@ pub fn build_from_builder(c: &mut Criterion) { ) .arg( Arg::new("positional3") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .about("tests positionals with specific values") @@ -114,6 +116,7 @@ pub fn build_from_builder(c: &mut Criterion) { .arg( Arg::new("multvalsmo") .long("multvalsmo") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .about("Tests multiple values, not mult occs") @@ -122,6 +125,7 @@ pub fn build_from_builder(c: &mut Criterion) { .arg( Arg::new("minvals") .long("minvals2") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .about("Tests 2 min vals") @@ -130,6 +134,7 @@ pub fn build_from_builder(c: &mut Criterion) { .arg( Arg::new("maxvals") .long("maxvals3") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .about("Tests 3 max vals") @@ -144,6 +149,7 @@ pub fn build_from_builder(c: &mut Criterion) { Arg::new("scoption") .short('o') .long("option") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .about("tests options"), diff --git a/benches/04_new_help.rs b/benches/04_new_help.rs index 3b2f7149135..83c9e001c38 100644 --- a/benches/04_new_help.rs +++ b/benches/04_new_help.rs @@ -112,6 +112,7 @@ fn app_example7<'c>() -> App<'c> { .arg( Arg::new("input") .about("the input file to use") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .setting(ArgSettings::Required) @@ -129,6 +130,7 @@ fn app_example8<'c>() -> App<'c> { .arg( Arg::new("input") .about("the input file to use") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .setting(ArgSettings::Required) diff --git a/benches/05_ripgrep.rs b/benches/05_ripgrep.rs index 866e8e95d6b..98a6d73ec13 100644 --- a/benches/05_ripgrep.rs +++ b/benches/05_ripgrep.rs @@ -326,6 +326,7 @@ where ])) .arg( arg("path") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences), ) @@ -350,6 +351,7 @@ where .arg( flag("color") .value_name("WHEN") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::HidePossibleValues) .possible_values(&["never", "auto", "always", "ansi"]), ) diff --git a/benches/06_rustup.rs b/benches/06_rustup.rs index e056aa26ff6..bf48af2fe0e 100644 --- a/benches/06_rustup.rs +++ b/benches/06_rustup.rs @@ -265,6 +265,7 @@ fn build_cli() -> App<'static> { .arg( Arg::new("command") .setting(ArgSettings::Required) + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences), ), diff --git a/clap_generate/examples/value_hints.rs b/clap_generate/examples/value_hints.rs index 91162f1045f..82219b2a997 100644 --- a/clap_generate/examples/value_hints.rs +++ b/clap_generate/examples/value_hints.rs @@ -72,6 +72,7 @@ fn build_cli() -> App<'static> { ) .arg( Arg::new("command_with_args") + .takes_value(true) .multiple_values(true) .value_hint(ValueHint::CommandWithArguments), ) diff --git a/clap_generate/tests/value_hints.rs b/clap_generate/tests/value_hints.rs index 2cf678ff01a..6148dda7007 100644 --- a/clap_generate/tests/value_hints.rs +++ b/clap_generate/tests/value_hints.rs @@ -56,6 +56,7 @@ pub fn build_app_with_value_hints() -> App<'static> { ) .arg( Arg::new("command_with_args") + .takes_value(true) .multiple_values(true) .value_hint(ValueHint::CommandWithArguments), ) diff --git a/examples/05_flag_args.rs b/examples/05_flag_args.rs index 28e771d3ee1..ded32752e67 100644 --- a/examples/05_flag_args.rs +++ b/examples/05_flag_args.rs @@ -18,6 +18,7 @@ fn main() { .about("turns up the awesome") // Displayed when showing help info .short('a') // Trigger this arg with "-a" .long("awesome") // Trigger this arg with "--awesome" + .takes_value(true) .multiple(true) // This flag should allow multiple // occurrences such as "-aaa" or "-a -a" .requires("config") // Says, "If the user uses -a, they MUST diff --git a/examples/17_yaml.yaml b/examples/17_yaml.yaml index 9d08a1bc6e6..0e9dd794af4 100644 --- a/examples/17_yaml.yaml +++ b/examples/17_yaml.yaml @@ -29,6 +29,7 @@ args: help: demo flag argument short: F multiple: true + takes_value: true global: true # Conflicts, mutual overrides, and requirements can all be defined as a # list, where the key is the name of the other argument diff --git a/examples/22_stop_parsing_with_--.rs b/examples/22_stop_parsing_with_--.rs index 463715939ef..2ff376f7ba8 100644 --- a/examples/22_stop_parsing_with_--.rs +++ b/examples/22_stop_parsing_with_--.rs @@ -5,7 +5,7 @@ fn main() { let matches = App::new("myprog") .arg(Arg::new("eff").short('f')) .arg(Arg::new("pea").short('p').takes_value(true)) - .arg(Arg::new("slop").multiple(true).last(true)) + .arg(Arg::new("slop").takes_value(true).multiple(true).last(true)) .get_matches(); println!("-f used: {:?}", matches.is_present("eff")); diff --git a/examples/23_flag_subcommands_pacman.rs b/examples/23_flag_subcommands_pacman.rs index 7f2ed61f876..49f337efbd4 100644 --- a/examples/23_flag_subcommands_pacman.rs +++ b/examples/23_flag_subcommands_pacman.rs @@ -40,6 +40,7 @@ fn main() { .long("search") .about("search locally installed packages for matching strings") .conflicts_with("info") + .takes_value(true) .multiple_values(true), ) .arg( @@ -48,6 +49,7 @@ fn main() { .short('i') .conflicts_with("search") .about("view package information") + .takes_value(true) .multiple_values(true), ), ) diff --git a/src/build/app/settings.rs b/src/build/app/settings.rs index 249d2b57d41..e10d4d0bef4 100644 --- a/src/build/app/settings.rs +++ b/src/build/app/settings.rs @@ -329,7 +329,7 @@ pub enum AppSettings { /// .setting(AppSettings::AllowMissingPositional) /// .arg(Arg::new("foo")) /// .arg(Arg::new("bar")) - /// .arg(Arg::new("baz").multiple(true)) + /// .arg(Arg::new("baz").takes_value(true).multiple(true)) /// .get_matches_from(vec![ /// "prog", "foo", "bar", "baz1", "baz2", "baz3" /// ]); @@ -348,7 +348,7 @@ pub enum AppSettings { /// .setting(AppSettings::AllowMissingPositional) /// .arg(Arg::new("foo")) /// .arg(Arg::new("bar")) - /// .arg(Arg::new("baz").multiple(true)) + /// .arg(Arg::new("baz").takes_value(true).multiple(true)) /// .get_matches_from(vec![ /// "prog", "--", "baz1", "baz2", "baz3" /// ]); diff --git a/src/build/arg/debug_asserts.rs b/src/build/arg/debug_asserts.rs index 4a03f4e1e1a..19a6c26c7e4 100644 --- a/src/build/arg/debug_asserts.rs +++ b/src/build/arg/debug_asserts.rs @@ -7,7 +7,7 @@ pub(crate) fn assert_arg(arg: &Arg) { // TODO: this check should be recursive assert!( !arg.blacklist.iter().any(|x| *x == arg.id), - "Argument '{}' cannot conflict with itarg", + "Argument '{}' cannot conflict with itself", arg.name, ); @@ -42,4 +42,44 @@ pub(crate) fn assert_arg(arg: &Arg) { arg.name ); } + + assert_app_flag(arg); +} + +fn assert_app_flag(arg: &Arg) { + use ArgSettings::*; + if arg.is_set(AllowEmptyValues) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(RequireDelimiter) { + assert!(arg.is_set(TakesValue)); + assert!(arg.is_set(UseValueDelimiter)); + } + if arg.is_set(HidePossibleValues) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(AllowHyphenValues) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(RequireEquals) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(Last) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(HideDefaultValue) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(MultipleValues) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(HideEnv) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(HideEnvValues) { + assert!(arg.is_set(TakesValue)); + } + if arg.is_set(IgnoreCase) { + assert!(arg.is_set(TakesValue)); + } } diff --git a/src/build/arg/mod.rs b/src/build/arg/mod.rs index 61aa8b2935b..46154018096 100644 --- a/src/build/arg/mod.rs +++ b/src/build/arg/mod.rs @@ -1799,6 +1799,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg}; /// let m = App::new("prog") /// .arg(Arg::new("cmds") + /// .takes_value(true) /// .multiple(true) /// .allow_hyphen_values(true) /// .value_terminator(";")) @@ -3137,7 +3138,7 @@ impl<'help> Arg<'help> { /// **NOTE**: This setting only applies to positional arguments, and has no affect on FLAGS / /// OPTIONS /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// **CAUTION:** Using this setting *and* having child subcommands is not /// recommended with the exception of *also* using [`AppSettings::ArgsNegateSubcommands`] @@ -3149,6 +3150,7 @@ impl<'help> Arg<'help> { /// ```rust /// # use clap::{Arg, ArgSettings}; /// Arg::new("args") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::Last) /// # ; /// ``` @@ -3162,6 +3164,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("first")) /// .arg(Arg::new("second")) /// .arg(Arg::new("third") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::Last)) /// .try_get_matches_from(vec![ /// "prog", "one", "--", "three" @@ -3182,6 +3185,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("first")) /// .arg(Arg::new("second")) /// .arg(Arg::new("third") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::Last)) /// .try_get_matches_from(vec![ /// "prog", "one", "two", "three" @@ -3320,7 +3324,7 @@ impl<'help> Arg<'help> { /// Allows values which start with a leading hyphen (`-`) /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// **WARNING**: Take caution when using this setting combined with /// [`ArgSettings::MultipleValues`], as this becomes ambiguous `$ prog --arg -- -- val`. All @@ -3340,6 +3344,7 @@ impl<'help> Arg<'help> { /// ```rust /// # use clap::{Arg, ArgSettings}; /// Arg::new("pattern") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::AllowHyphenValues) /// # ; /// ``` @@ -3348,6 +3353,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// let m = App::new("prog") /// .arg(Arg::new("pat") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::AllowHyphenValues) /// .long("pattern")) /// .get_matches_from(vec![ @@ -3389,7 +3395,7 @@ impl<'help> Arg<'help> { /// Requires that options use the `--option=val` syntax (i.e. an equals between the option and /// associated value) **Default:** `false` /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// @@ -3397,6 +3403,7 @@ impl<'help> Arg<'help> { /// # use clap::{Arg, ArgSettings}; /// Arg::new("config") /// .long("config") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::RequireEquals) /// # ; /// ``` @@ -3408,6 +3415,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// let res = App::new("prog") /// .arg(Arg::new("cfg") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::RequireEquals) /// .long("config")) /// .try_get_matches_from(vec![ @@ -3424,6 +3432,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ErrorKind, ArgSettings}; /// let res = App::new("prog") /// .arg(Arg::new("cfg") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::RequireEquals) /// .long("config")) /// .try_get_matches_from(vec![ @@ -3439,8 +3448,7 @@ impl<'help> Arg<'help> { #[inline] pub fn require_equals(self, r: bool) -> Self { if r { - self.unset_setting(ArgSettings::AllowEmptyValues) - .setting(ArgSettings::RequireEquals) + self.setting(ArgSettings::RequireEquals) } else { self.unset_setting(ArgSettings::RequireEquals) } @@ -3502,7 +3510,7 @@ impl<'help> Arg<'help> { /// /// **NOTE:** The default is `false`. /// - /// **NOTE:** Setting this implies [`ArgSettings::UseValueDelimiter`] and + /// **NOTE:** Setting this requires [`ArgSettings::UseValueDelimiter`] and /// [`ArgSettings::TakesValue`] /// /// **NOTE:** It's a good idea to inform the user that use of a delimiter is required, either @@ -3518,6 +3526,8 @@ impl<'help> Arg<'help> { /// let delims = App::new("prog") /// .arg(Arg::new("opt") /// .short('o') + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::UseValueDelimiter) /// .setting(ArgSettings::RequireDelimiter) /// .setting(ArgSettings::MultipleValues)) /// .get_matches_from(vec![ @@ -3534,6 +3544,8 @@ impl<'help> Arg<'help> { /// let res = App::new("prog") /// .arg(Arg::new("opt") /// .short('o') + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::UseValueDelimiter) /// .setting(ArgSettings::RequireDelimiter)) /// .try_get_matches_from(vec![ /// "prog", "-o", "val1", "val2", "val3", @@ -3555,6 +3567,7 @@ impl<'help> Arg<'help> { /// let delims = App::new("prog") /// .arg(Arg::new("opt") /// .short('o') + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::MultipleValues)) /// .get_matches_from(vec![ /// "prog", "-o", "val1", "val2", "val3", @@ -3568,9 +3581,7 @@ impl<'help> Arg<'help> { #[inline] pub fn require_delimiter(self, d: bool) -> Self { if d { - self.takes_value(true) - .setting(ArgSettings::UseValueDelimiter) - .setting(ArgSettings::RequireDelimiter) + self.setting(ArgSettings::RequireDelimiter) } else { self.unset_setting(ArgSettings::RequireDelimiter) } @@ -3582,13 +3593,14 @@ impl<'help> Arg<'help> { /// This is useful for args with many values, or ones which are explained elsewhere in the /// help text. /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// /// ```rust /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("config") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HidePossibleValues) /// # ; /// ``` @@ -3599,6 +3611,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("mode") /// .long("mode") /// .possible_values(&["fast", "slow"]) + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HidePossibleValues)); /// ``` /// If we were to run the above program with `--help` the `[values: fast, slow]` portion of @@ -3616,13 +3629,14 @@ impl<'help> Arg<'help> { /// /// This is useful when default behavior of an arg is explained elsewhere in the help text. /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// /// ```rust /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("config") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HideDefaultValue) /// # ; /// ``` @@ -3633,6 +3647,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("host") /// .long("host") /// .default_value("localhost") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HideDefaultValue)); /// /// ``` @@ -3698,7 +3713,7 @@ impl<'help> Arg<'help> { /// When used with [`Arg::possible_values`] it allows the argument value to pass validation even /// if the case differs from that of the specified `possible_value`. /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// @@ -3707,6 +3722,7 @@ impl<'help> Arg<'help> { /// let m = App::new("pv") /// .arg(Arg::new("option") /// .long("--option") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::IgnoreCase) /// .possible_value("test123")) /// .get_matches_from(vec![ @@ -3725,6 +3741,7 @@ impl<'help> Arg<'help> { /// .short('o') /// .long("--option") /// .setting(ArgSettings::IgnoreCase) + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::MultipleValues) /// .possible_value("test123") /// .possible_value("test321")) @@ -3809,13 +3826,14 @@ impl<'help> Arg<'help> { /// /// This is useful when the variable option is explained elsewhere in the help text. /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// /// ```rust /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("config") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HideEnv) /// # ; /// ``` @@ -3826,6 +3844,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("mode") /// .long("mode") /// .env("MODE") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HideEnv)); /// /// ``` @@ -3846,14 +3865,15 @@ impl<'help> Arg<'help> { /// /// This is useful when ENV vars contain sensitive values. /// - /// **NOTE:** Setting this implies [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`] /// /// # Examples /// /// ```rust /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("config") - /// .setting(ArgSettings::HideDefaultValue) + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::HideEnvValues) /// # ; /// ``` /// @@ -3863,6 +3883,7 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("host") /// .long("host") /// .env("CONNECT") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::HideEnvValues)); /// /// ``` @@ -3987,6 +4008,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("debug") /// .short('d') + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::MultipleValues) /// # ; /// ``` @@ -4012,7 +4034,8 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// let m = App::new("prog") /// .arg(Arg::new("file") - /// .setting(ArgSettings::MultipleValues) // implies TakesValue + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::MultipleValues) /// .short('F')) /// .get_matches_from(vec![ /// "prog", "-F", "file1", "file2", "file3" @@ -4029,7 +4052,8 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ErrorKind, ArgSettings}; /// let res = App::new("prog") /// .arg(Arg::new("file") - /// .setting(ArgSettings::MultipleValues) // implies TakesValue + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::MultipleValues) /// .short('F')) /// .try_get_matches_from(vec![ /// "prog", "-F", "file1", "-F", "file2", "-F", "file3" @@ -4045,7 +4069,8 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// let m = App::new("prog") /// .arg(Arg::new("file") - /// .setting(ArgSettings::MultipleValues) // implies TakesValue + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::MultipleValues) /// .short('F')) /// .arg(Arg::new("word") /// .index(1)) @@ -4123,7 +4148,7 @@ impl<'help> Arg<'help> { /// /// **NOTE:** By default empty values are *not* allowed /// - /// **NOTE:** Implicitly sets [`ArgSettings::TakesValue`] + /// **NOTE:** Setting this requires [`ArgSettings::TakesValue`]. /// /// # Examples /// @@ -4131,6 +4156,7 @@ impl<'help> Arg<'help> { /// # use clap::{App, Arg, ArgSettings}; /// Arg::new("file") /// .long("file") + /// .setting(ArgSettings::TakesValue) /// .setting(ArgSettings::AllowEmptyValues) /// # ; /// ``` @@ -4158,7 +4184,8 @@ impl<'help> Arg<'help> { /// .arg(Arg::new("cfg") /// .long("config") /// .short('v') - /// .setting(ArgSettings::AllowEmptyValues)) // implies TakesValue + /// .setting(ArgSettings::TakesValue) + /// .setting(ArgSettings::AllowEmptyValues)) /// .try_get_matches_from(vec![ /// "prog", "--config=" /// ]); @@ -4259,16 +4286,20 @@ impl<'help> Arg<'help> { /// may not be exactly what you are expecting and using [`AppSettings::TrailingVarArg`] /// may be more appropriate. /// - /// **NOTE:** Implicitly sets [`Arg::multiple(true)`], [`Arg::allow_hyphen_values(true)`], and + /// **NOTE:** Implicitly sets [`Arg::takes_value(true)`] [`Arg::multiple(true)`], [`Arg::allow_hyphen_values(true)`], and /// [`Arg::last(true)`] when set to `true` /// + /// [`Arg::takes_value(true)`]: ./struct.Arg.html#method.takes_value /// [`Arg::multiple(true)`]: ./struct.Arg.html#method.multiple /// [`Arg::allow_hyphen_values(true)`]: ./struct.Arg.html#method.allow_hyphen_values /// [`Arg::last(true)`]: ./struct.Arg.html#method.last /// [`AppSettings::TrailingVarArg`]: ./enum.AppSettings.html#variant.TrailingVarArg #[inline] pub fn raw(self, raw: bool) -> Self { - self.multiple(raw).allow_hyphen_values(raw).last(raw) + self.takes_value(true) + .multiple(raw) + .allow_hyphen_values(raw) + .last(raw) } /// Hides an argument from short help message output. @@ -4503,6 +4534,7 @@ impl<'help> Arg<'help> { /// .setting(AppSettings::TrailingVarArg) /// .arg( /// Arg::new("command") + /// .takes_value(true) /// .multiple(true) /// .value_hint(ValueHint::CommandWithArguments) /// ) @@ -4954,6 +4986,7 @@ mod test { fn option_display3() { let o2 = Arg::new("opt") .short('o') + .takes_value(true) .multiple(true) .value_names(&["file", "name"]); @@ -5008,6 +5041,7 @@ mod test { fn positiona_display_mult() { let p = Arg::new("pos") .index(1) + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues); assert_eq!(&*format!("{}", p), "..."); diff --git a/src/build/arg/settings.rs b/src/build/arg/settings.rs index 6b722b880c4..14b0279d82a 100644 --- a/src/build/arg/settings.rs +++ b/src/build/arg/settings.rs @@ -8,26 +8,26 @@ bitflags! { struct Flags: u32 { const REQUIRED = 1; const MULTIPLE_OCC = 1 << 1; - const EMPTY_VALS = 1 << 2 | Self::TAKES_VAL.bits; + const EMPTY_VALS = 1 << 2; const GLOBAL = 1 << 3; const HIDDEN = 1 << 4; const TAKES_VAL = 1 << 5; const USE_DELIM = 1 << 6; const NEXT_LINE_HELP = 1 << 7; const R_UNLESS_ALL = 1 << 8; - const REQ_DELIM = 1 << 9 | Self::TAKES_VAL.bits | Self::USE_DELIM.bits; + const REQ_DELIM = 1 << 9; const DELIM_NOT_SET = 1 << 10; - const HIDE_POS_VALS = 1 << 11 | Self::TAKES_VAL.bits; - const ALLOW_TAC_VALS = 1 << 12 | Self::TAKES_VAL.bits; - const REQUIRE_EQUALS = 1 << 13 | Self::TAKES_VAL.bits; - const LAST = 1 << 14 | Self::TAKES_VAL.bits; - const HIDE_DEFAULT_VAL = 1 << 15 | Self::TAKES_VAL.bits; + const HIDE_POS_VALS = 1 << 11; + const ALLOW_TAC_VALS = 1 << 12; + const REQUIRE_EQUALS = 1 << 13; + const LAST = 1 << 14; + const HIDE_DEFAULT_VAL = 1 << 15; const CASE_INSENSITIVE = 1 << 16; const HIDE_ENV_VALS = 1 << 17; const HIDDEN_SHORT_H = 1 << 18; const HIDDEN_LONG_H = 1 << 19; - const MULTIPLE_VALS = 1 << 20 | Self::TAKES_VAL.bits; - const HIDE_ENV = 1 << 21 | Self::TAKES_VAL.bits; + const MULTIPLE_VALS = 1 << 20; + const HIDE_ENV = 1 << 21; } } diff --git a/src/build/usage_parser.rs b/src/build/usage_parser.rs index 10ab059c360..9ac5fff1bd9 100644 --- a/src/build/usage_parser.rs +++ b/src/build/usage_parser.rs @@ -71,6 +71,7 @@ impl<'help> UsageParser<'help> { }; if !arg.has_switch() && arg.is_set(ArgSettings::MultipleOccurrences) { // We had a positional and need to set mult vals too + arg.settings.set(ArgSettings::TakesValue); arg.settings.set(ArgSettings::MultipleValues); } debug!("UsageParser::parse: vals...{:?}", arg.val_names); diff --git a/src/macros.rs b/src/macros.rs index 533bcdb528a..7c270779090 100644 --- a/src/macros.rs +++ b/src/macros.rs @@ -461,7 +461,7 @@ macro_rules! clap_app { $crate::clap_app!{ @arg ($arg.value_name(stringify!($var))) (+) $($tail)* } }; (@arg ($arg:expr) $modes:tt ... $($tail:tt)*) => { - $crate::clap_app!{ @arg ($arg) $modes +multiple $($tail)* } + $crate::clap_app!{ @arg ($arg) $modes +multiple +takes_value $($tail)* } }; // Shorthand magic (@arg ($arg:expr) $modes:tt #{$n:expr, $m:expr} $($tail:tt)*) => { diff --git a/src/parse/errors.rs b/src/parse/errors.rs index 26da6929f91..41a2be51ba7 100644 --- a/src/parse/errors.rs +++ b/src/parse/errors.rs @@ -155,6 +155,7 @@ pub enum ErrorKind { /// # use clap::{App, Arg, ErrorKind}; /// let result = App::new("prog") /// .arg(Arg::new("arg") + /// .takes_value(true) /// .multiple(true) /// .max_values(2)) /// .try_get_matches_from(vec!["prog", "too", "many", "values"]); diff --git a/src/parse/matches/arg_matches.rs b/src/parse/matches/arg_matches.rs index 5382cb010a9..e1bc6f439b2 100644 --- a/src/parse/matches/arg_matches.rs +++ b/src/parse/matches/arg_matches.rs @@ -40,6 +40,7 @@ pub(crate) struct SubCommand { /// .takes_value(true)) /// .arg(Arg::new("debug") /// .short('d') +/// .takes_value(true) /// .multiple(true)) /// .arg(Arg::new("cfg") /// .short('c') diff --git a/src/parse/parser.rs b/src/parse/parser.rs index efb5faf3566..e448a4a9a8c 100644 --- a/src/parse/parser.rs +++ b/src/parse/parser.rs @@ -794,6 +794,7 @@ impl<'help, 'app> Parser<'help, 'app> { if help_help { let mut pb = Arg::new("subcommand") .index(1) + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .about("The subcommand whose help message to display"); diff --git a/tests/app_settings.rs b/tests/app_settings.rs index 5d01d168540..3988ce60fcc 100644 --- a/tests/app_settings.rs +++ b/tests/app_settings.rs @@ -951,6 +951,8 @@ fn aaos_opts_mult() { .arg( Arg::from("--opt [val]... 'some option'") .number_of_values(1) + .takes_value(true) + .use_delimiter(true) .require_delimiter(true), ) .try_get_matches_from(vec!["", "--opt=some", "--opt=other", "--opt=one,two"]); diff --git a/tests/delimiters.rs b/tests/delimiters.rs index 0e01506ffdb..4873e7d43e4 100644 --- a/tests/delimiters.rs +++ b/tests/delimiters.rs @@ -96,6 +96,7 @@ fn opt_s_no_space_mult_no_delim() { .arg( Arg::new("option") .short('o') + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues), ) .try_get_matches_from(vec!["", "-o", "val1,val2,val3"]); @@ -114,9 +115,9 @@ fn opt_eq_mult_def_delim() { .arg( Arg::new("option") .long("opt") + .takes_value(true) .multiple(true) - .use_delimiter(true) - .setting(ArgSettings::TakesValue), + .use_delimiter(true), ) .try_get_matches_from(vec!["", "--opt=val1,val2,val3"]); diff --git a/tests/fixtures/app.yaml b/tests/fixtures/app.yaml index b6660ee07c6..4d70bd437c9 100644 --- a/tests/fixtures/app.yaml +++ b/tests/fixtures/app.yaml @@ -12,6 +12,7 @@ args: - option: short: o long: option + takes_value: true multiple: true about: tests options - positional: @@ -26,6 +27,7 @@ args: - flag: short: f long: flag + takes_value: true multiple: true about: tests flags global: true @@ -71,7 +73,9 @@ args: - multvalsdelim: long: multvalsdelim about: Tests multiple values with required delimiter + takes_value: true multiple: true + use_delimiter: true require_delimiter: true - singlealias: long: singlealias diff --git a/tests/fixtures/app_2space.yaml b/tests/fixtures/app_2space.yaml index 0c9e3877576..83318bbe946 100644 --- a/tests/fixtures/app_2space.yaml +++ b/tests/fixtures/app_2space.yaml @@ -12,5 +12,6 @@ args: - opt: short: o long: option + takes_value: true multiple: true about: tests options diff --git a/tests/flag_subcommands.rs b/tests/flag_subcommands.rs index 2e0e0d4ceed..fd521b58fbb 100644 --- a/tests/flag_subcommands.rs +++ b/tests/flag_subcommands.rs @@ -443,6 +443,7 @@ fn flag_subcommand_long_short_normal_usage_string() { .long("search") .about("search locally installed packages for matching strings") .conflicts_with("info") + .takes_value(true) .multiple_values(true), ) .arg( @@ -451,6 +452,7 @@ fn flag_subcommand_long_short_normal_usage_string() { .short('i') .conflicts_with("search") .about("view package information") + .takes_value(true) .multiple_values(true), ), ); @@ -497,6 +499,7 @@ fn flag_subcommand_long_normal_usage_string() { .long("search") .about("search locally installed packages for matching strings") .conflicts_with("info") + .takes_value(true) .multiple_values(true), ) .arg( @@ -505,6 +508,7 @@ fn flag_subcommand_long_normal_usage_string() { .short('i') .conflicts_with("search") .about("view package information") + .takes_value(true) .multiple_values(true), ), ); @@ -551,6 +555,7 @@ fn flag_subcommand_short_normal_usage_string() { .long("search") .about("search locally installed packages for matching strings") .conflicts_with("info") + .takes_value(true) .multiple_values(true), ) .arg( @@ -559,6 +564,7 @@ fn flag_subcommand_short_normal_usage_string() { .short('i') .conflicts_with("search") .about("view package information") + .takes_value(true) .multiple_values(true), ), ); diff --git a/tests/global_args.rs b/tests/global_args.rs index def291b3bc1..4917aa8e849 100644 --- a/tests/global_args.rs +++ b/tests/global_args.rs @@ -17,6 +17,7 @@ fn issue_1076() { Arg::new("GLOBAL_FLAG") .long("global-flag") .about("Specifies something needed by the subcommands") + .takes_value(true) .multiple(true) .global(true), ) diff --git a/tests/grouped_values.rs b/tests/grouped_values.rs index 4a7984f2c1a..0e1848e509c 100644 --- a/tests/grouped_values.rs +++ b/tests/grouped_values.rs @@ -5,7 +5,12 @@ use clap::{App, AppSettings, Arg}; #[test] fn grouped_value_works() { let m = App::new("cli") - .arg(Arg::new("option").long("option").multiple(true)) + .arg( + Arg::new("option") + .long("option") + .takes_value(true) + .multiple(true), + ) .get_matches_from(&[ "cli", "--option", @@ -30,7 +35,12 @@ fn issue_1026() { let m = App::new("cli") .arg(Arg::new("server").short('s').takes_value(true)) .arg(Arg::new("user").short('u').takes_value(true)) - .arg(Arg::new("target").long("target").multiple(true)) + .arg( + Arg::new("target") + .long("target") + .takes_value(true) + .multiple(true), + ) .get_matches_from(&[ "backup", "-s", "server", "-u", "user", "--target", "target1", "file1", "file2", "file3", "--target", "target2", "file4", "file5", "file6", "file7", "--target", @@ -96,7 +106,12 @@ fn grouped_value_short_flag_delimiter() { #[test] fn grouped_value_positional_arg() { let m = App::new("multiple_values") - .arg(Arg::new("pos").about("multiple positionals").multiple(true)) + .arg( + Arg::new("pos") + .about("multiple positionals") + .takes_value(true) + .multiple(true), + ) .get_matches_from(vec![ "myprog", "val1", "val2", "val3", "val4", "val5", "val6", ]); @@ -114,6 +129,7 @@ fn grouped_value_multiple_positional_arg() { .arg( Arg::new("pos2") .about("multiple positionals") + .takes_value(true) .multiple(true), ) .get_matches_from(vec![ @@ -133,6 +149,7 @@ fn grouped_value_multiple_positional_arg_last_multiple() { .arg( Arg::new("pos2") .about("multiple positionals") + .takes_value(true) .multiple(true) .last(true), ) diff --git a/tests/help.rs b/tests/help.rs index fb03a0f529d..1fad729d8e2 100644 --- a/tests/help.rs +++ b/tests/help.rs @@ -758,6 +758,7 @@ fn args_with_last_usage() { .arg( Arg::new("pass through args") .about("Any arguments you wish to pass to the being profiled.") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::MultipleValues) .setting(ArgSettings::MultipleOccurrences) .setting(ArgSettings::Last) @@ -1209,7 +1210,12 @@ fn issue_702_multiple_values() { .author("foo") .about("bar") .arg(Arg::new("arg1").about("some option")) - .arg(Arg::new("arg2").multiple(true).about("some option")) + .arg( + Arg::new("arg2") + .takes_value(true) + .multiple(true) + .about("some option"), + ) .arg( Arg::new("some") .about("some option") @@ -1511,7 +1517,13 @@ fn last_arg_mult_usage() { .version("0.1") .arg(Arg::new("TARGET").required(true).about("some")) .arg(Arg::new("CORPUS").about("some")) - .arg(Arg::new("ARGS").multiple(true).last(true).about("some")); + .arg( + Arg::new("ARGS") + .takes_value(true) + .multiple(true) + .last(true) + .about("some"), + ); assert!(utils::compare_output(app, "last --help", LAST_ARG, false)); } @@ -1523,6 +1535,7 @@ fn last_arg_mult_usage_req() { .arg(Arg::new("CORPUS").about("some")) .arg( Arg::new("ARGS") + .takes_value(true) .multiple(true) .last(true) .required(true) @@ -1545,6 +1558,7 @@ fn last_arg_mult_usage_req_with_sc() { .arg(Arg::new("CORPUS").about("some")) .arg( Arg::new("ARGS") + .takes_value(true) .multiple(true) .last(true) .required(true) @@ -1566,7 +1580,13 @@ fn last_arg_mult_usage_with_sc() { .setting(AppSettings::ArgsNegateSubcommands) .arg(Arg::new("TARGET").required(true).about("some")) .arg(Arg::new("CORPUS").about("some")) - .arg(Arg::new("ARGS").multiple(true).last(true).about("some")) + .arg( + Arg::new("ARGS") + .takes_value(true) + .multiple(true) + .last(true) + .about("some"), + ) .subcommand(App::new("test").about("some")); assert!(utils::compare_output( app, @@ -1680,6 +1700,8 @@ fn issue_1052_require_delim_help() { .version("1.3") .arg( Arg::from("-f, --fake 'some help'") + .takes_value(true) + .use_delimiter(true) .require_delimiter(true) .value_delimiter(":"), ); @@ -1804,6 +1826,8 @@ fn custom_headers_headers() { .version("1.4") .arg( Arg::from("-f, --fake 'some help'") + .takes_value(true) + .use_delimiter(true) .require_delimiter(true) .value_delimiter(":"), ) @@ -1856,6 +1880,8 @@ fn multiple_custom_help_headers() { .version("1.4") .arg( Arg::from("-f, --fake 'some help'") + .takes_value(true) + .use_delimiter(true) .require_delimiter(true) .value_delimiter(":"), ) @@ -1970,7 +1996,12 @@ fn issue_1364_no_short_options() { .value_name("BAZ") .hidden_short_help(true), ) - .arg(Arg::new("files").value_name("FILES").multiple(true)); + .arg( + Arg::new("files") + .value_name("FILES") + .takes_value(true) + .multiple(true), + ); assert!(utils::compare_output(app, "demo -h", ISSUE_1364, false)); } diff --git a/tests/multiple_values.rs b/tests/multiple_values.rs index f1f0f54bf54..0352ccb2adb 100644 --- a/tests/multiple_values.rs +++ b/tests/multiple_values.rs @@ -350,7 +350,12 @@ fn option_max_more() { #[test] fn positional() { let m = App::new("multiple_values") - .arg(Arg::new("pos").about("multiple positionals").multiple(true)) + .arg( + Arg::new("pos") + .about("multiple positionals") + .takes_value(true) + .multiple(true), + ) .try_get_matches_from(vec!["myprog", "val1", "val2", "val3"]); assert!(m.is_ok()); @@ -627,6 +632,7 @@ fn sep_positional() { .arg( Arg::new("option") .about("multiple options") + .takes_value(true) .use_delimiter(true) .multiple(true), ) @@ -736,7 +742,7 @@ fn req_delimiter_long() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec!["", "--option", "val1", "val2", "val3"]); assert!(m.is_ok()); @@ -765,7 +771,7 @@ fn req_delimiter_long_with_equal() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec!["", "--option=val1", "val2", "val3"]); assert!(m.is_ok()); @@ -794,7 +800,7 @@ fn req_delimiter_short_with_space() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec!["", "-o", "val1", "val2", "val3"]); assert!(m.is_ok()); @@ -823,7 +829,7 @@ fn req_delimiter_short_with_no_space() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec!["", "-oval1", "val2", "val3"]); assert!(m.is_ok()); @@ -852,7 +858,7 @@ fn req_delimiter_short_with_equal() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec!["", "-o=val1", "val2", "val3"]); assert!(m.is_ok()); @@ -882,7 +888,7 @@ fn req_delimiter_complex() { .require_delimiter(true) .takes_value(true), ) - .arg(Arg::new("args").multiple(true).index(1)) + .arg(Arg::new("args").takes_value(true).multiple(true).index(1)) .try_get_matches_from(vec![ "", "val1", @@ -942,7 +948,13 @@ positional argument (i.e. the one with the highest index) *must* have \ .required(true) or .last(true) set."] fn low_index_positional_not_required() { let _ = App::new("lip") - .arg(Arg::new("files").index(1).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) .arg(Arg::new("target").index(2)) .try_get_matches_from(vec![""]); } @@ -954,8 +966,20 @@ fn low_index_positional_not_required() { set is allowed per command, unless the second one also has .last(true) set"] fn low_index_positional_last_multiple_too() { let _ = App::new("lip") - .arg(Arg::new("files").index(1).required(true).multiple(true)) - .arg(Arg::new("target").index(2).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) + .arg( + Arg::new("target") + .index(2) + .takes_value(true) + .required(true) + .multiple(true), + ) .try_get_matches_from(vec![""]); } @@ -966,7 +990,13 @@ fn low_index_positional_last_multiple_too() { last positional argument may be set to .multiple(true)"] fn low_index_positional_too_far_back() { let _ = App::new("lip") - .arg(Arg::new("files").index(1).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) .arg(Arg::new("target").required(true).index(2)) .arg(Arg::new("target2").required(true).index(3)) .try_get_matches_from(vec![""]); @@ -975,7 +1005,13 @@ fn low_index_positional_too_far_back() { #[test] fn low_index_positional() { let m = App::new("lip") - .arg(Arg::new("files").index(1).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) .arg(Arg::new("target").index(2).required(true)) .try_get_matches_from(vec!["lip", "file1", "file2", "file3", "target"]); @@ -998,7 +1034,13 @@ fn low_index_positional_in_subcmd() { let m = App::new("lip") .subcommand( App::new("test") - .arg(Arg::new("files").index(1).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) .arg(Arg::new("target").index(2).required(true)), ) .try_get_matches_from(vec!["lip", "test", "file1", "file2", "file3", "target"]); @@ -1021,7 +1063,13 @@ fn low_index_positional_in_subcmd() { #[test] fn low_index_positional_with_option() { let m = App::new("lip") - .arg(Arg::new("files").required(true).index(1).multiple(true)) + .arg( + Arg::new("files") + .required(true) + .index(1) + .takes_value(true) + .multiple(true), + ) .arg(Arg::new("target").index(2).required(true)) .arg(Arg::new("opt").long("option").takes_value(true)) .try_get_matches_from(vec![ @@ -1046,7 +1094,13 @@ fn low_index_positional_with_option() { #[test] fn low_index_positional_with_flag() { let m = App::new("lip") - .arg(Arg::new("files").index(1).required(true).multiple(true)) + .arg( + Arg::new("files") + .index(1) + .takes_value(true) + .required(true) + .multiple(true), + ) .arg(Arg::new("target").index(2).required(true)) .arg(Arg::new("flg").long("flag")) .try_get_matches_from(vec!["lip", "file1", "file2", "file3", "target", "--flag"]); @@ -1073,6 +1127,7 @@ fn multiple_value_terminator_option() { Arg::new("files") .short('f') .value_terminator(";") + .takes_value(true) .multiple(true), ) .arg(Arg::new("other")) @@ -1098,6 +1153,7 @@ fn multiple_value_terminator_option_other_arg() { Arg::new("files") .short('f') .value_terminator(";") + .takes_value(true) .multiple(true), ) .arg(Arg::new("other")) @@ -1122,6 +1178,7 @@ fn multiple_vals_with_hyphen() { let res = App::new("do") .arg( Arg::new("cmds") + .takes_value(true) .multiple(true) .allow_hyphen_values(true) .value_terminator(";"), diff --git a/tests/opts.rs b/tests/opts.rs index 0218f93feb1..906a6c5ceae 100644 --- a/tests/opts.rs +++ b/tests/opts.rs @@ -47,6 +47,7 @@ fn require_equals_min_values_zero() { let res = App::new("prog") .arg( Arg::new("cfg") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::RequireEquals) .min_values(0) .long("config"), @@ -64,6 +65,7 @@ fn double_hyphen_as_value() { let res = App::new("prog") .arg( Arg::new("cfg") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::AllowHyphenValues) .long("config"), ) @@ -77,6 +79,7 @@ fn require_equals_no_empty_values_fail() { let res = App::new("prog") .arg( Arg::new("cfg") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::RequireEquals) .long("config"), ) @@ -91,6 +94,7 @@ fn require_equals_empty_vals_pass() { let res = App::new("prog") .arg( Arg::new("cfg") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::RequireEquals) .setting(ArgSettings::AllowEmptyValues) .long("config"), @@ -104,6 +108,7 @@ fn require_equals_pass() { let res = App::new("prog") .arg( Arg::new("cfg") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::RequireEquals) .long("config"), ) @@ -284,7 +289,11 @@ fn multiple_vals_pos_arg_delim() { #[test] fn require_delims_no_delim() { let r = App::new("mvae") - .arg(Arg::from("-o [opt]... 'some opt'").setting(ArgSettings::RequireDelimiter)) + .arg( + Arg::from("-o [opt]... 'some opt'") + .setting(ArgSettings::UseValueDelimiter) + .setting(ArgSettings::RequireDelimiter), + ) .arg(Arg::from("[file] 'some file'")) .try_get_matches_from(vec!["mvae", "-o", "1", "2", "some"]); assert!(r.is_err()); @@ -295,7 +304,11 @@ fn require_delims_no_delim() { #[test] fn require_delims() { let r = App::new("mvae") - .arg(Arg::from("-o [opt]... 'some opt'").setting(ArgSettings::RequireDelimiter)) + .arg( + Arg::from("-o [opt]... 'some opt'") + .setting(ArgSettings::UseValueDelimiter) + .setting(ArgSettings::RequireDelimiter), + ) .arg(Arg::from("[file] 'some file'")) .try_get_matches_from(vec!["", "-o", "1,2", "some"]); assert!(r.is_ok()); @@ -359,6 +372,7 @@ fn leading_hyphen_with_only_pos_follows() { .arg( Arg::from("-o [opt]... 'some opt'") .number_of_values(1) + .setting(ArgSettings::TakesValue) .setting(ArgSettings::AllowHyphenValues), ) .arg("[arg] 'some arg'") @@ -400,6 +414,7 @@ fn issue_1047_min_zero_vals_default_val() { Arg::new("del") .short('d') .long("del") + .setting(ArgSettings::TakesValue) .setting(ArgSettings::RequireEquals) .min_values(0) .default_missing_value("default"), diff --git a/tests/positionals.rs b/tests/positionals.rs index 305a79901e4..2dcb68e4210 100644 --- a/tests/positionals.rs +++ b/tests/positionals.rs @@ -104,7 +104,10 @@ fn positional_multiple() { let r = App::new("positional_multiple") .args(&[ Arg::from("-f, --flag 'some flag'"), - Arg::new("positional").index(1).multiple(true), + Arg::new("positional") + .index(1) + .takes_value(true) + .multiple(true), ]) .try_get_matches_from(vec!["", "-f", "test1", "test2", "test3"]); assert!(r.is_ok(), "{:#?}", r); @@ -122,7 +125,10 @@ fn positional_multiple_3() { let r = App::new("positional_multiple") .args(&[ Arg::from("-f, --flag 'some flag'"), - Arg::new("positional").index(1).multiple(true), + Arg::new("positional") + .index(1) + .takes_value(true) + .multiple(true), ]) .try_get_matches_from(vec!["", "test1", "test2", "test3", "--flag"]); assert!(r.is_ok(), "{:#?}", r); diff --git a/tests/posix_compatible.rs b/tests/posix_compatible.rs index 8d56ce6fc1e..744b9622f0c 100644 --- a/tests/posix_compatible.rs +++ b/tests/posix_compatible.rs @@ -41,6 +41,8 @@ fn mult_option_require_delim_overrides_itself() { Arg::from("--opt [val]... 'some option'") .overrides_with("opt") .number_of_values(1) + .takes_value(true) + .use_delimiter(true) .require_delimiter(true), ) .try_get_matches_from(vec!["", "--opt=some", "--opt=other", "--opt=one,two"]); diff --git a/tests/possible_values.rs b/tests/possible_values.rs index b5d2eb3d15e..53ddeff5757 100644 --- a/tests/possible_values.rs +++ b/tests/possible_values.rs @@ -71,6 +71,7 @@ fn possible_values_of_positional_multiple() { .arg( Arg::new("positional") .index(1) + .takes_value(true) .possible_value("test123") .possible_value("test321") .multiple(true), @@ -93,6 +94,7 @@ fn possible_values_of_positional_multiple_fail() { .arg( Arg::new("positional") .index(1) + .takes_value(true) .possible_value("test123") .possible_value("test321") .multiple(true), diff --git a/tests/subcommands.rs b/tests/subcommands.rs index 23bf8e8f6e5..855f7606e6a 100644 --- a/tests/subcommands.rs +++ b/tests/subcommands.rs @@ -362,7 +362,7 @@ fn issue_1161_multiple_hyphen_hyphen() { let res = App::new("myprog") .arg(Arg::new("eff").short('f')) .arg(Arg::new("pea").short('p').takes_value(true)) - .arg(Arg::new("slop").multiple(true).last(true)) + .arg(Arg::new("slop").takes_value(true).multiple(true).last(true)) .try_get_matches_from(vec![ "-f", "-p=bob", From 6df56ad289b16b7cd7c409fbf6965ea27bf249e8 Mon Sep 17 00:00:00 2001 From: ldm0 Date: Wed, 24 Feb 2021 15:43:23 +0000 Subject: [PATCH 3/4] Move checker to macro(better panic message) --- src/build/arg/debug_asserts.rs | 58 +++++++++++++++------------------- 1 file changed, 25 insertions(+), 33 deletions(-) diff --git a/src/build/arg/debug_asserts.rs b/src/build/arg/debug_asserts.rs index 19a6c26c7e4..be045234917 100644 --- a/src/build/arg/debug_asserts.rs +++ b/src/build/arg/debug_asserts.rs @@ -48,38 +48,30 @@ pub(crate) fn assert_arg(arg: &Arg) { fn assert_app_flag(arg: &Arg) { use ArgSettings::*; - if arg.is_set(AllowEmptyValues) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(RequireDelimiter) { - assert!(arg.is_set(TakesValue)); - assert!(arg.is_set(UseValueDelimiter)); - } - if arg.is_set(HidePossibleValues) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(AllowHyphenValues) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(RequireEquals) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(Last) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(HideDefaultValue) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(MultipleValues) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(HideEnv) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(HideEnvValues) { - assert!(arg.is_set(TakesValue)); - } - if arg.is_set(IgnoreCase) { - assert!(arg.is_set(TakesValue)); + macro_rules! checker { + ($a:ident requires $($b:ident)|+) => { + if arg.is_set($a) { + let mut s = String::new(); + $( + if !arg.is_set($b) { + s.push_str(&format!("\nArgSettings::{} is required when ArgSettings::{} is on.\n", std::stringify!($b), std::stringify!($a))); + } + )+ + if !s.is_empty() { + panic!("{}", s) + } + } + } } + checker!(AllowEmptyValues requires TakesValue); + checker!(RequireDelimiter requires TakesValue | UseValueDelimiter); + checker!(HidePossibleValues requires TakesValue); + checker!(AllowHyphenValues requires TakesValue); + checker!(RequireEquals requires TakesValue); + checker!(Last requires TakesValue); + checker!(HideDefaultValue requires TakesValue); + checker!(MultipleValues requires TakesValue); + checker!(HideEnv requires TakesValue); + checker!(HideEnvValues requires TakesValue); + checker!(IgnoreCase requires TakesValue); } From 0b1d137ee94e3f27b582389ec82b8b3fd3148f27 Mon Sep 17 00:00:00 2001 From: ldm0 Date: Tue, 9 Mar 2021 13:52:12 +0000 Subject: [PATCH 4/4] Apply suggestions --- src/build/arg/debug_asserts.rs | 10 +++++++--- src/build/arg/mod.rs | 2 +- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/src/build/arg/debug_asserts.rs b/src/build/arg/debug_asserts.rs index be045234917..9ca7a9c740d 100644 --- a/src/build/arg/debug_asserts.rs +++ b/src/build/arg/debug_asserts.rs @@ -43,26 +43,30 @@ pub(crate) fn assert_arg(arg: &Arg) { ); } - assert_app_flag(arg); + assert_app_flags(arg); } -fn assert_app_flag(arg: &Arg) { +fn assert_app_flags(arg: &Arg) { use ArgSettings::*; + macro_rules! checker { ($a:ident requires $($b:ident)|+) => { if arg.is_set($a) { let mut s = String::new(); + $( if !arg.is_set($b) { - s.push_str(&format!("\nArgSettings::{} is required when ArgSettings::{} is on.\n", std::stringify!($b), std::stringify!($a))); + s.push_str(&format!("\nArgSettings::{} is required when ArgSettings::{} is set.\n", std::stringify!($b), std::stringify!($a))); } )+ + if !s.is_empty() { panic!("{}", s) } } } } + checker!(AllowEmptyValues requires TakesValue); checker!(RequireDelimiter requires TakesValue | UseValueDelimiter); checker!(HidePossibleValues requires TakesValue); diff --git a/src/build/arg/mod.rs b/src/build/arg/mod.rs index 46154018096..a5ed40c990c 100644 --- a/src/build/arg/mod.rs +++ b/src/build/arg/mod.rs @@ -4296,7 +4296,7 @@ impl<'help> Arg<'help> { /// [`AppSettings::TrailingVarArg`]: ./enum.AppSettings.html#variant.TrailingVarArg #[inline] pub fn raw(self, raw: bool) -> Self { - self.takes_value(true) + self.takes_value(raw) .multiple(raw) .allow_hyphen_values(raw) .last(raw)