Skip to content

Commit

Permalink
feat(clap_complete): Support multi-values of positional argument with…
Browse files Browse the repository at this point in the history
… `num_arg(N)`
  • Loading branch information
shannmu committed Jul 30, 2024
1 parent 5613e14 commit e4d7151
Show file tree
Hide file tree
Showing 2 changed files with 79 additions and 22 deletions.
68 changes: 60 additions & 8 deletions clap_complete/src/dynamic/completer.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
use core::num;
use std::ffi::OsStr;
use std::ffi::OsString;

Expand Down Expand Up @@ -71,8 +72,7 @@ pub fn complete(
}

if is_escaped {
pos_index += 1;
state = ParseState::Pos(pos_index);
(state, pos_index) = parse_positional(current_cmd, pos_index, is_escaped, state);
} else if arg.is_escape() {
is_escaped = true;
state = ParseState::ValueDone;
Expand Down Expand Up @@ -124,10 +124,11 @@ pub fn complete(
}
} else {
match state {
ParseState::ValueDone | ParseState::Pos(_) => {
pos_index += 1;
state = ParseState::ValueDone;
ParseState::ValueDone | ParseState::Pos(..) => {
(state, pos_index) =
parse_positional(current_cmd, pos_index, is_escaped, state);
}

ParseState::Opt((ref opt, count)) => match opt.get_num_args() {
Some(range) => {
let max = range.max_values();
Expand Down Expand Up @@ -156,8 +157,8 @@ enum ParseState<'a> {
/// Parsing a value done, there is no state to record.
ValueDone,

/// Parsing a positional argument after `--`
Pos(usize),
/// Parsing a positional argument after `--`. Pos(pos_index, takes_num_args)
Pos((usize, usize)),

/// Parsing a optional flag argument
Opt((&'a clap::Arg, usize)),
Expand Down Expand Up @@ -300,7 +301,7 @@ fn complete_arg(
completions.extend(complete_subcommand(value, cmd));
}
}
ParseState::Pos(_) => {
ParseState::Pos(..) => {
if let Some(positional) = cmd
.get_positionals()
.find(|p| p.get_index() == Some(pos_index))
Expand Down Expand Up @@ -603,6 +604,57 @@ fn parse_shortflags<'c, 's>(
(leading_flags, takes_value_opt, short)
}

/// Parse the positional arguments. Return the new state and the new positional index.
fn parse_positional<'a>(
cmd: &clap::Command,
pos_index: usize,
is_escaped: bool,
state: ParseState<'a>,
) -> (ParseState<'a>, usize) {
let pos_arg = cmd
.get_positionals()
.find(|p| p.get_index() == Some(pos_index));
let num_args = pos_arg
.and_then(|a| a.get_num_args().and_then(|r| Some(r.max_values())))
.unwrap_or(1);

let update_state_with_new_positional = |pos_index| -> (ParseState<'a>, usize) {
if num_args > 1 {
(ParseState::Pos((pos_index, 1)), pos_index)
} else {
if is_escaped {
(ParseState::Pos((pos_index, 1)), pos_index + 1)
} else {
(ParseState::ValueDone, pos_index + 1)
}
}
};
match state {
ParseState::ValueDone => {
update_state_with_new_positional(pos_index)
},
ParseState::Pos((prev_pos_index, num_arg)) => {
if prev_pos_index == pos_index {
if num_arg + 1 < num_args {
(ParseState::Pos((pos_index, num_arg + 1)), pos_index)
} else {
if is_escaped {
(ParseState::Pos((pos_index, 1)), pos_index + 1)
} else {
(ParseState::ValueDone, pos_index + 1)
}
}
} else {
update_state_with_new_positional(pos_index)
}
}
ParseState::Opt(..) => unreachable!(
"This branch won't be hit,
because ParseState::Opt should not be seen as a positional argument and passed to this function."
),
}
}

/// A completion candidate definition
///
/// This makes it easier to add more fields to completion candidate,
Expand Down
33 changes: 19 additions & 14 deletions clap_complete/tests/testsuite/dynamic.rs
Original file line number Diff line number Diff line change
Expand Up @@ -609,20 +609,18 @@ fn suggest_multi_positional() {
assert_data_eq!(
complete!(cmd, "pos_a [TAB]"),
snapbox::str![
"--format
--help\tPrint help
-F
-h\tPrint help"
"pos_a
pos_b
pos_c"
]
);

assert_data_eq!(
complete!(cmd, "pos_a pos_b [TAB]"),
snapbox::str![
"--format
--help\tPrint help
-F
-h\tPrint help"
"pos_a
pos_b
pos_c"
]
);

Expand All @@ -642,10 +640,9 @@ pos_c"
assert_data_eq!(
complete!(cmd, "--format json pos_a [TAB]"),
snapbox::str![
"--format
--help\tPrint help
-F
-h\tPrint help"
"pos_a
pos_b
pos_c"
]
);

Expand All @@ -661,12 +658,20 @@ pos_c"

assert_data_eq!(
complete!(cmd, "--format json -- pos_a [TAB]"),
snapbox::str![""]
snapbox::str![
"pos_a
pos_b
pos_c"
]
);

assert_data_eq!(
complete!(cmd, "--format json -- pos_a pos_b [TAB]"),
snapbox::str![""]
snapbox::str![
"pos_a
pos_b
pos_c"
]
);

assert_data_eq!(
Expand Down

0 comments on commit e4d7151

Please sign in to comment.