From 06aa418fa6b2148a625e5a9353cc672523624fb6 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Tue, 1 Feb 2022 19:04:53 -0600 Subject: [PATCH] fix(error): Be more accurate in smart usage For some errors, we use the unroll logic to get the list of required arguments. The usage then does the same, but without a matcher. This was causing the lists to not match. As a side effect, this fixed an ordering issue where we were putting the present arg after the not-present arg. I assume its because we ended up reporting the items twice but the first time is correctly ordered and gets precedence. This was split out of #3020 --- src/build/app/mod.rs | 22 +++++++--- src/output/usage.rs | 10 ++--- src/parse/validator.rs | 6 ++- tests/builder/double_require.rs | 4 +- tests/builder/require.rs | 75 +++++++++++++++++++++++++++++++++ 5 files changed, 101 insertions(+), 16 deletions(-) diff --git a/src/build/app/mod.rs b/src/build/app/mod.rs index 74b0f4e794f..03336801f45 100644 --- a/src/build/app/mod.rs +++ b/src/build/app/mod.rs @@ -3238,16 +3238,24 @@ impl<'help> App<'help> { args } - pub(crate) fn unroll_requirements_for_arg(&self, arg: &Id, matcher: &ArgMatcher) -> Vec { + pub(crate) fn unroll_requirements_for_arg( + &self, + arg: &Id, + matcher: Option<&ArgMatcher>, + ) -> Vec { let requires_if_or_not = |(val, req_arg): &(ArgPredicate<'_>, Id)| -> Option { match val { ArgPredicate::Equals(v) => { - if matcher - .get(arg) - .map(|ma| ma.contains_val_os(v)) - .unwrap_or(false) - { - Some(req_arg.clone()) + if let Some(matcher) = matcher { + if matcher + .get(arg) + .map(|ma| ma.contains_val_os(v)) + .unwrap_or(false) + { + Some(req_arg.clone()) + } else { + None + } } else { None } diff --git a/src/output/usage.rs b/src/output/usage.rs index 8eaf171b816..0ed3c39514a 100644 --- a/src/output/usage.rs +++ b/src/output/usage.rs @@ -360,12 +360,10 @@ impl<'help, 'app> Usage<'help, 'app> { let mut unrolled_reqs = IndexSet::new(); for a in self.required.iter() { - if let Some(m) = matcher { - for aa in self.app.unroll_requirements_for_arg(a, m) { - // if we don't check for duplicates here this causes duplicate error messages - // see https://github.com/clap-rs/clap/issues/2770 - unrolled_reqs.insert(aa); - } + for aa in self.app.unroll_requirements_for_arg(a, matcher) { + // if we don't check for duplicates here this causes duplicate error messages + // see https://github.com/clap-rs/clap/issues/2770 + unrolled_reqs.insert(aa); } // always include the required arg itself. it will not be enumerated // by unroll_requirements_for_arg. diff --git a/src/parse/validator.rs b/src/parse/validator.rs index 853be564799..35e81466946 100644 --- a/src/parse/validator.rs +++ b/src/parse/validator.rs @@ -272,7 +272,11 @@ impl<'help, 'app, 'parser> Validator<'help, 'app, 'parser> { for name in matcher.arg_names() { debug!("Validator::gather_requirements:iter:{:?}", name); if let Some(arg) = self.p.app.find(name) { - for req in self.p.app.unroll_requirements_for_arg(&arg.id, matcher) { + for req in self + .p + .app + .unroll_requirements_for_arg(&arg.id, Some(matcher)) + { self.p.required.insert(req); } } else if let Some(g) = self.p.app.find_group(name) { diff --git a/tests/builder/double_require.rs b/tests/builder/double_require.rs index 537e3d26707..c9f198b7948 100644 --- a/tests/builder/double_require.rs +++ b/tests/builder/double_require.rs @@ -16,7 +16,7 @@ static ONLY_B_ERROR: &str = "error: The following required arguments were not pr -c USAGE: - prog [OPTIONS] -c -b + prog [OPTIONS] -b -c For more information try --help "; @@ -25,7 +25,7 @@ static ONLY_C_ERROR: &str = "error: The following required arguments were not pr -b USAGE: - prog [OPTIONS] -b -c + prog [OPTIONS] -c -b For more information try --help "; diff --git a/tests/builder/require.rs b/tests/builder/require.rs index 48a8780faf6..072454b8d52 100644 --- a/tests/builder/require.rs +++ b/tests/builder/require.rs @@ -115,6 +115,81 @@ fn positional_required_2() { assert_eq!(m.value_of("flag").unwrap(), "someval"); } +#[test] +fn positional_required_with_requires() { + let app = App::new("positional_required") + .arg(Arg::new("flag").required(true).requires("opt")) + .arg(Arg::new("opt")) + .arg(Arg::new("bar")); + + assert!(utils::compare_output( + app, + "clap-test", + POSITIONAL_REQ, + true + )); +} + +static POSITIONAL_REQ: &str = "error: The following required arguments were not provided: + + + +USAGE: + clap-test [ARGS] + +For more information try --help +"; + +#[test] +fn positional_required_with_requires_if_no_value() { + let app = App::new("positional_required") + .arg(Arg::new("flag").required(true).requires_if("val", "opt")) + .arg(Arg::new("opt")) + .arg(Arg::new("bar")); + + assert!(utils::compare_output( + app, + "clap-test", + POSITIONAL_REQ_IF_NO_VAL, + true + )); +} + +static POSITIONAL_REQ_IF_NO_VAL: &str = "error: The following required arguments were not provided: + + +USAGE: + clap-test [ARGS] + +For more information try --help +"; + +#[test] +fn positional_required_with_requires_if_value() { + let app = App::new("positional_required") + .arg(Arg::new("flag").required(true).requires_if("val", "opt")) + .arg(Arg::new("foo").required(true)) + .arg(Arg::new("opt")) + .arg(Arg::new("bar")); + + assert!(utils::compare_output( + app, + "clap-test val", + POSITIONAL_REQ_IF_VAL, + true + )); +} + +static POSITIONAL_REQ_IF_VAL: &str = "error: The following required arguments were not provided: + + + +USAGE: + clap-test [ARGS] + +For more information try --help +"; + #[test] fn group_required() { let result = App::new("group_required")