From 130b2ff997bb852fa6a3f179cb8523982e58e0f3 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Fri, 19 May 2023 10:36:43 -0500 Subject: [PATCH 1/3] test(parser): Show multiple positional multiple values behavior --- tests/builder/multiple_values.rs | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/tests/builder/multiple_values.rs b/tests/builder/multiple_values.rs index 78dc54914dc..30fb9a9d16d 100644 --- a/tests/builder/multiple_values.rs +++ b/tests/builder/multiple_values.rs @@ -1411,6 +1411,37 @@ fn multiple_vals_with_hyphen() { ); } +#[test] +#[should_panic] +fn multiple_positional_multiple_values() { + let _ = Command::new("do") + .arg( + Arg::new("cmd1") + .action(ArgAction::Set) + .num_args(1..) + .allow_hyphen_values(true) + .value_terminator(";"), + ) + .arg( + Arg::new("cmd2") + .action(ArgAction::Set) + .num_args(1..) + .allow_hyphen_values(true) + .value_terminator(";"), + ) + .try_get_matches_from(vec![ + "do", + "find", + "-type", + "f", + "-name", + "special", + ";", + "/home/clap", + "foo", + ]); +} + #[test] fn issue_1480_max_values_consumes_extra_arg_1() { let res = Command::new("prog") From 1ee2e95e96f95892f60dac095dc0fc095ae303e3 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Fri, 19 May 2023 10:39:05 -0500 Subject: [PATCH 2/3] fix(builder): Allow value terminated multiple positional values --- clap_builder/src/builder/debug_asserts.rs | 1 + tests/builder/multiple_values.rs | 21 +++++++++++++++++++-- 2 files changed, 20 insertions(+), 2 deletions(-) diff --git a/clap_builder/src/builder/debug_asserts.rs b/clap_builder/src/builder/debug_asserts.rs index 83b1dea84b3..ef970cdaad0 100644 --- a/clap_builder/src/builder/debug_asserts.rs +++ b/clap_builder/src/builder/debug_asserts.rs @@ -603,6 +603,7 @@ fn _verify_positionals(cmd: &Command) -> bool { .get_positionals() .filter(|p| { p.is_multiple_values_set() + && p.get_value_terminator().is_none() && !p.get_num_args().expect(INTERNAL_ERROR_MSG).is_fixed() }) .count(); diff --git a/tests/builder/multiple_values.rs b/tests/builder/multiple_values.rs index 30fb9a9d16d..38b72f5dba2 100644 --- a/tests/builder/multiple_values.rs +++ b/tests/builder/multiple_values.rs @@ -1412,9 +1412,8 @@ fn multiple_vals_with_hyphen() { } #[test] -#[should_panic] fn multiple_positional_multiple_values() { - let _ = Command::new("do") + let res = Command::new("do") .arg( Arg::new("cmd1") .action(ArgAction::Set) @@ -1440,6 +1439,24 @@ fn multiple_positional_multiple_values() { "/home/clap", "foo", ]); + assert!(res.is_ok(), "{:?}", res.unwrap_err().kind()); + + let m = res.unwrap(); + let cmd1: Vec<_> = m + .get_many::("cmd1") + .unwrap() + .map(|v| v.as_str()) + .collect(); + assert_eq!( + &cmd1, + &["find", "-type", "f", "-name", "special", "/home/clap"] + ); + let cmd2: Vec<_> = m + .get_many::("cmd2") + .unwrap() + .map(|v| v.as_str()) + .collect(); + assert_eq!(&cmd2, &["foo"]); } #[test] From e1db168d69cd1dcde5e31715f983400509f81af4 Mon Sep 17 00:00:00 2001 From: Ed Page Date: Fri, 19 May 2023 10:41:44 -0500 Subject: [PATCH 3/3] fix(parser): Ensure terminated positionals are assigned correctly --- clap_builder/src/parser/parser.rs | 39 ++++++++++++++++--------------- tests/builder/multiple_values.rs | 7 ++---- 2 files changed, 22 insertions(+), 24 deletions(-) diff --git a/clap_builder/src/parser/parser.rs b/clap_builder/src/parser/parser.rs index a96c0c77de3..09e8984391e 100644 --- a/clap_builder/src/parser/parser.rs +++ b/clap_builder/src/parser/parser.rs @@ -372,26 +372,27 @@ impl<'cmd> Parser<'cmd> { if matcher.pending_arg_id() != Some(arg.get_id()) || !arg.is_multiple_values_set() { ok!(self.resolve_pending(matcher)); } - if let Some(_parse_result) = self.check_terminator(arg, arg_os.to_value_os()) { - debug!( - "Parser::get_matches_with: ignoring terminator result {_parse_result:?}" - ); - } else { - let arg_values = matcher.pending_values_mut( - arg.get_id(), - Some(Identifier::Index), - trailing_values, - ); - arg_values.push(arg_os.to_value_os().to_owned()); - } + parse_state = + if let Some(parse_result) = self.check_terminator(arg, arg_os.to_value_os()) { + debug_assert_eq!(parse_result, ParseResult::ValuesDone); + pos_counter += 1; + ParseState::ValuesDone + } else { + let arg_values = matcher.pending_values_mut( + arg.get_id(), + Some(Identifier::Index), + trailing_values, + ); + arg_values.push(arg_os.to_value_os().to_owned()); - // Only increment the positional counter if it doesn't allow multiples - if !arg.is_multiple() { - pos_counter += 1; - parse_state = ParseState::ValuesDone; - } else { - parse_state = ParseState::Pos(arg.get_id().clone()); - } + // Only increment the positional counter if it doesn't allow multiples + if !arg.is_multiple() { + pos_counter += 1; + ParseState::ValuesDone + } else { + ParseState::Pos(arg.get_id().clone()) + } + }; valid_arg_found = true; } else if let Some(external_parser) = self.cmd.get_external_subcommand_value_parser().cloned() diff --git a/tests/builder/multiple_values.rs b/tests/builder/multiple_values.rs index 38b72f5dba2..5f3333c211d 100644 --- a/tests/builder/multiple_values.rs +++ b/tests/builder/multiple_values.rs @@ -1447,16 +1447,13 @@ fn multiple_positional_multiple_values() { .unwrap() .map(|v| v.as_str()) .collect(); - assert_eq!( - &cmd1, - &["find", "-type", "f", "-name", "special", "/home/clap"] - ); + assert_eq!(&cmd1, &["find", "-type", "f", "-name", "special"]); let cmd2: Vec<_> = m .get_many::("cmd2") .unwrap() .map(|v| v.as_str()) .collect(); - assert_eq!(&cmd2, &["foo"]); + assert_eq!(&cmd2, &["/home/clap", "foo"]); } #[test]