Skip to content

Commit

Permalink
Add tests, fix grouped_values_of()
Browse files Browse the repository at this point in the history
Fix clippy
  • Loading branch information
ldm0 committed Jan 20, 2021
1 parent f634573 commit d60981b
Show file tree
Hide file tree
Showing 3 changed files with 201 additions and 13 deletions.
65 changes: 53 additions & 12 deletions src/parse/matches/arg_matches.rs
Original file line number Diff line number Diff line change
Expand Up @@ -242,8 +242,21 @@ impl ArgMatches {
}

/// Placeholder documentation.
pub fn grouped_values_of<T: Key>(&self, id: T) -> Option<Iter<Vec<OsString>>> {
self.args.get(&Id::from(id)).map(|arg| arg.vals())
pub fn grouped_values_of<T: Key>(&self, id: T) -> Option<GroupedValues> {
#[allow(clippy::type_complexity)]
let arg_values: for<'a> fn(
&'a MatchedArg,
) -> Map<
std::slice::Iter<'a, Vec<OsString>>,
fn(&Vec<OsString>) -> Vec<&str>,
> = |arg| {
arg.vals()
.map(|g| g.iter().map(|x| x.to_str().expect(INVALID_UTF8)).collect())
};
self.args
.get(&Id::from(id))
.map(arg_values)
.map(|iter| GroupedValues { iter })
}

/// Gets the lossy values of a specific argument. If the option wasn't present at runtime
Expand Down Expand Up @@ -1017,12 +1030,44 @@ impl<'a> ExactSizeIterator for Values<'a> {}
impl<'a> Default for Values<'a> {
fn default() -> Self {
static EMPTY: [Vec<OsString>; 0] = [];
// This is never called because the iterator is empty:
fn to_str_slice(_: &OsString) -> &str {
unreachable!()
}
Values {
iter: EMPTY[..].iter().flatten().map(to_str_slice),
iter: EMPTY[..].iter().flatten().map(|_| unreachable!()),
}
}
}

#[derive(Clone)]
#[allow(missing_debug_implementations)]
pub struct GroupedValues<'a> {
#[allow(clippy::type_complexity)]
iter: Map<std::slice::Iter<'a, Vec<OsString>>, fn(&Vec<OsString>) -> Vec<&str>>,
}

impl<'a> Iterator for GroupedValues<'a> {
type Item = Vec<&'a str>;

fn next(&mut self) -> Option<Self::Item> {
self.iter.next()
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.iter.size_hint()
}
}

impl<'a> DoubleEndedIterator for GroupedValues<'a> {
fn next_back(&mut self) -> Option<Self::Item> {
self.iter.next_back()
}
}

impl<'a> ExactSizeIterator for GroupedValues<'a> {}

/// Creates an empty iterator. Used for `unwrap_or_default()`.
impl<'a> Default for GroupedValues<'a> {
fn default() -> Self {
static EMPTY: [Vec<OsString>; 0] = [];
GroupedValues {
iter: EMPTY[..].iter().map(|_| unreachable!()),
}
}
}
Expand Down Expand Up @@ -1078,12 +1123,8 @@ impl<'a> ExactSizeIterator for OsValues<'a> {}
impl Default for OsValues<'_> {
fn default() -> Self {
static EMPTY: [Vec<OsString>; 0] = [];
// This is never called because the iterator is empty:
fn to_str_slice(_: &OsString) -> &OsStr {
unreachable!()
}
OsValues {
iter: EMPTY[..].iter().flatten().map(to_str_slice),
iter: EMPTY[..].iter().flatten().map(|_| unreachable!()),
}
}
}
Expand Down
2 changes: 1 addition & 1 deletion src/parse/parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1252,7 +1252,7 @@ impl<'help, 'app> Parser<'help, 'app> {
}
vals
} else {
arg_split.into_iter().collect()
arg_split.collect()
};
let vals = vals.into_iter().map(|x| x.to_os_string()).collect();
self.add_multiple_vals_to_arg(arg, vals, matcher, ty, append);
Expand Down
147 changes: 147 additions & 0 deletions tests/grouped_values.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,147 @@
mod utils;

use clap::{App, Arg};

#[test]
fn value_sets_works() {
let m = App::new("cli")
.arg(Arg::new("option").long("option").multiple(true))
.get_matches_from(&[
"cli",
"--option",
"fr_FR:mon option 1",
"en_US:my option 1",
"--option",
"fr_FR:mon option 2",
"en_US:my option 2",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("option").unwrap().collect();
assert_eq!(
grouped_vals,
vec![
vec!["fr_FR:mon option 1", "en_US:my option 1",],
vec!["fr_FR:mon option 2", "en_US:my option 2",],
]
);
}

#[test]
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))
.get_matches_from(&[
"backup", "-s", "server", "-u", "user", "--target", "target1", "file1", "file2",
"file3", "--target", "target2", "file4", "file5", "file6", "file7", "--target",
"target3", "file8",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("target").unwrap().collect();
assert_eq!(
grouped_vals,
vec![
vec!["target1", "file1", "file2", "file3"],
vec!["target2", "file4", "file5", "file6", "file7",],
vec!["target3", "file8"]
]
);
}

#[test]
fn value_sets_long_flag_delimiter() {
let m = App::new("myapp")
.arg(
Arg::new("option")
.long("option")
.takes_value(true)
.use_delimiter(true)
.multiple(true),
)
.get_matches_from(vec![
"myapp",
"--option=hmm",
"--option=val1,val2,val3",
"--option",
"alice,bob",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("option").unwrap().collect();
assert_eq!(
grouped_vals,
vec![
vec!["hmm"],
vec!["val1", "val2", "val3"],
vec!["alice", "bob"]
]
);
}

#[test]
fn value_sets_short_flag_delimiter() {
let m = App::new("myapp")
.arg(
Arg::new("option")
.short('o')
.takes_value(true)
.use_delimiter(true)
.multiple(true),
)
.get_matches_from(vec!["myapp", "-o=foo", "-o=val1,val2,val3", "-o=bar"]);
let grouped_vals: Vec<_> = m.grouped_values_of("option").unwrap().collect();
assert_eq!(
grouped_vals,
vec![vec!["foo"], vec!["val1", "val2", "val3"], vec!["bar"]]
);
}

#[test]
fn value_sets_positional_arg() {
let m = App::new("multiple_values")
.arg(Arg::new("pos").about("multiple positionals").multiple(true))
.get_matches_from(vec![
"myprog", "val1", "val2", "val3", "val4", "val5", "val6",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("pos").unwrap().collect();
assert_eq!(
grouped_vals,
vec![vec!["val1", "val2", "val3", "val4", "val5", "val6"]]
);
}

#[test]
fn value_sets_multiple_positional_arg() {
let m = App::new("multiple_values")
.arg(Arg::new("pos1").about("multiple positionals"))
.arg(
Arg::new("pos2")
.about("multiple positionals")
.multiple(true),
)
.get_matches_from(vec![
"myprog", "val1", "val2", "val3", "val4", "val5", "val6",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("pos2").unwrap().collect();
assert_eq!(
grouped_vals,
vec![vec!["val2", "val3", "val4", "val5", "val6"]]
);
}

#[test]
fn value_sets_multiple_positional_arg_last_multiple() {
let m = App::new("multiple_values")
.arg(Arg::new("pos1").about("multiple positionals"))
.arg(
Arg::new("pos2")
.about("multiple positionals")
.multiple(true)
.last(true),
)
.get_matches_from(vec![
"myprog", "val1", "--", "val2", "val3", "val4", "val5", "val6",
]);
let grouped_vals: Vec<_> = m.grouped_values_of("pos2").unwrap().collect();
assert_eq!(
grouped_vals,
vec![vec!["val2", "val3", "val4", "val5", "val6"]]
);
}

0 comments on commit d60981b

Please sign in to comment.