Skip to content

Commit

Permalink
Change the way parser do self override, Fixes clap-rs#1374
Browse files Browse the repository at this point in the history
  • Loading branch information
ldm0 committed Jan 27, 2021
1 parent 4b59174 commit 622da8e
Show file tree
Hide file tree
Showing 4 changed files with 91 additions and 31 deletions.
2 changes: 1 addition & 1 deletion src/build/app/settings.rs
Original file line number Diff line number Diff line change
Expand Up @@ -225,7 +225,7 @@ pub enum AppSettings {
/// [`Arg::allow_hyphen_values`]: ./struct.Arg.html#method.allow_hyphen_values
AllowLeadingHyphen,

/// Specifies that all arguments override themselves. This is the equivolent to saying the `foo`
/// Specifies that all arguments override themselves. This is the equivalent to saying the `foo`
/// arg using [`Arg::overrides_with("foo")`] for all defined arguments.
///
/// [`Arg::overrides_with("foo")`]: ./struct.Arg.html#method.overrides_with
Expand Down
50 changes: 50 additions & 0 deletions src/parse/matches/matched_arg.rs
Original file line number Diff line number Diff line change
Expand Up @@ -89,6 +89,8 @@ impl MatchedArg {
self.vals.iter().flatten().count() == 0
}

// Will be used later
#[allow(dead_code)]
pub(crate) fn remove_vals(&mut self, len: usize) {
let mut want_remove = len;
let mut remove_group = None;
Expand All @@ -110,6 +112,13 @@ impl MatchedArg {
}
}

pub(crate) fn override_vals(&mut self) {
let len = self.vals.len();
if len > 1 {
self.vals.drain(0..len - 1);
}
}

pub(crate) fn contains_val(&self, val: &str) -> bool {
self.vals_flatten()
.any(|v| OsString::as_os_str(v) == OsStr::new(val))
Expand All @@ -124,6 +133,47 @@ impl MatchedArg {
mod tests {
use super::*;

#[test]
fn test_vals_override() {
let mut m = MatchedArg::new();
m.push_val("aaa".into());
m.new_val_group();
m.append_val("bbb".into());
m.append_val("ccc".into());
m.new_val_group();
m.append_val("ddd".into());
m.push_val("eee".into());
m.new_val_group();
m.append_val("fff".into());
m.append_val("ggg".into());
m.append_val("hhh".into());
m.append_val("iii".into());

m.override_vals();
let vals: Vec<&Vec<OsString>> = m.vals().collect();
assert_eq!(
vals,
vec![&vec![
OsString::from("fff"),
OsString::from("ggg"),
OsString::from("hhh"),
OsString::from("iii"),
]]
);
m.override_vals();

let vals: Vec<&Vec<OsString>> = m.vals().collect();
assert_eq!(
vals,
vec![&vec![
OsString::from("fff"),
OsString::from("ggg"),
OsString::from("hhh"),
OsString::from("iii"),
]]
);
}

#[test]
fn test_grouped_vals_push_and_append() {
let mut m = MatchedArg::new();
Expand Down
49 changes: 19 additions & 30 deletions src/parse/parser.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1347,34 +1347,28 @@ impl<'help, 'app> Parser<'help, 'app> {
let mut arg_overrides = Vec::new();
for name in matcher.arg_names() {
debug!("Parser::remove_overrides:iter:{:?}", name);
if let Some(arg) = self.app.find(name) {
let mut handle_self_override = |o: &Id| {
if (arg.is_set(ArgSettings::MultipleValues)
|| arg.is_set(ArgSettings::MultipleOccurrences))
|| !arg.has_switch()
{
return true;
}
debug!(
"Parser::remove_overrides:iter:{:?}:iter:{:?}: self override",
name, o
);
self_override.push(o.clone());
false
};
for o in &arg.overrides {
if let Some(overrider) = self.app.find(name) {
let mut override_self = false;
for overridee in &overrider.overrides {
debug!("Parser::remove_overrides:iter:{:?} => {:?}", name, o);
if *o == arg.id {
if handle_self_override(o) {
continue;
}
if *overridee == overrider.id {
override_self = true;
} else {
arg_overrides.push((arg.id.clone(), o));
arg_overrides.push((o.clone(), &arg.id));
arg_overrides.push((overrider.id.clone(), overridee));
arg_overrides.push((overridee.clone(), &overrider.id));
}
}
if self.is_set(AS::AllArgsOverrideSelf) {
let _ = handle_self_override(&arg.id);
if self.is_set(AS::AllArgsOverrideSelf) || override_self {
if (!overrider.is_set(ArgSettings::MultipleValues)
&& !overrider.is_set(ArgSettings::MultipleOccurrences))
&& overrider.has_switch()
{
debug!(
"Parser::remove_overrides:iter:{:?}:iter:{:?}: self override",
name, o
);
self_override.push(overrider.id.clone());
}
}
}
}
Expand All @@ -1392,13 +1386,8 @@ impl<'help, 'app> Parser<'help, 'app> {
for name in &self_override {
debug!("Parser::remove_overrides:iter:self:{:?}: resetting", name);
if let Some(ma) = matcher.get_mut(name) {
if ma.occurs < 2 {
continue;
}
ma.occurs = 1;

let len = ma.num_vals().saturating_sub(1);
ma.remove_vals(len);
ma.override_vals();
}
}

Expand Down
21 changes: 21 additions & 0 deletions tests/grouped_values.rs
Original file line number Diff line number Diff line change
Expand Up @@ -145,3 +145,24 @@ fn value_sets_multiple_positional_arg_last_multiple() {
vec![vec!["val2", "val3", "val4", "val5", "val6"]]
);
}

#[test]
fn issue_1374() {
let app = App::new("MyApp").arg(
Arg::new("input")
.takes_value(true)
.long("input")
.overrides_with("input")
.min_values(0),
);
let matches = app
.clone()
.get_matches_from(&["MyApp", "--input", "a", "b", "c", "--input", "d"]);
let vs = matches.values_of("input").unwrap();
assert_eq!(vs.collect::<Vec<_>>(), vec!["d"]);
let matches = app
.clone()
.get_matches_from(&["MyApp", "--input", "a", "b", "--input", "c", "d"]);
let vs = matches.values_of("input").unwrap();
assert_eq!(vs.collect::<Vec<_>>(), vec!["c", "d"]);
}

0 comments on commit 622da8e

Please sign in to comment.