mirror of
https://github.com/clap-rs/clap
synced 2025-01-21 17:13:53 +00:00
im(ArgGroups): improves requirment and confliction support for groups
This commit is contained in:
parent
a29c3983c4
commit
c236dc5ff4
1 changed files with 40 additions and 14 deletions
54
src/app.rs
54
src/app.rs
|
@ -688,6 +688,30 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
return g_vec.iter().map(|g| self.get_group_members(g)).fold(vec![], |acc, v| acc + &v)
|
return g_vec.iter().map(|g| self.get_group_members(g)).fold(vec![], |acc, v| acc + &v)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_group_members_names(&self, group: &'ar str) -> Vec<&'ar str> {
|
||||||
|
let mut g_vec = HashSet::new();
|
||||||
|
let mut args = HashSet::new();
|
||||||
|
|
||||||
|
for n in self.groups.get(group).unwrap().args.iter() {
|
||||||
|
if self.flags.contains_key(n) {
|
||||||
|
args.insert(*n);
|
||||||
|
} else if self.opts.contains_key(n) {
|
||||||
|
args.insert(*n);
|
||||||
|
} else if self.groups.contains_key(n) {
|
||||||
|
g_vec.insert(*n);
|
||||||
|
} else {
|
||||||
|
if self.positionals_name.contains_key(n) {
|
||||||
|
args.insert(*n);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if g_vec.is_empty() {
|
||||||
|
return args.iter().map(|s| *s).collect::<Vec<_>>()
|
||||||
|
}
|
||||||
|
return g_vec.iter().map(|g| self.get_group_members_names(g)).fold(vec![], |acc, v| acc + &v)
|
||||||
|
}
|
||||||
|
|
||||||
fn get_required_from(&self, reqs: HashSet<&'ar str>) -> Vec<String> {
|
fn get_required_from(&self, reqs: HashSet<&'ar str>) -> Vec<String> {
|
||||||
let mut c_flags = HashSet::new();
|
let mut c_flags = HashSet::new();
|
||||||
let mut c_pos = HashSet::new();
|
let mut c_pos = HashSet::new();
|
||||||
|
@ -1172,7 +1196,10 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
if let Some(ref vals) = ma.values {
|
if let Some(ref vals) = ma.values {
|
||||||
if num == vals.len() as u8 {
|
if num == vals.len() as u8 {
|
||||||
self.report_error(format!("The argument \"{}\" was found, \
|
self.report_error(format!("The argument \"{}\" was found, \
|
||||||
but {} only expects {} values", arg, opt, vals.len()),
|
but '{}' only expects {} values",
|
||||||
|
arg,
|
||||||
|
opt,
|
||||||
|
vals.len()),
|
||||||
true,
|
true,
|
||||||
true,
|
true,
|
||||||
Some(
|
Some(
|
||||||
|
@ -1256,7 +1283,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
self.report_error(format!("The argument '{}' cannot be used with {}",
|
self.report_error(format!("The argument '{}' cannot be used with {}",
|
||||||
p,
|
p,
|
||||||
match self.blacklisted_from(p.name, &matches) {
|
match self.blacklisted_from(p.name, &matches) {
|
||||||
Some(name) => name,
|
Some(name) => format!("'{}'", name),
|
||||||
None => "one or more of the other specified \
|
None => "one or more of the other specified \
|
||||||
arguments".to_owned()
|
arguments".to_owned()
|
||||||
}),
|
}),
|
||||||
|
@ -1268,7 +1295,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
if let Some(ref p_vals) = p.possible_vals {
|
if let Some(ref p_vals) = p.possible_vals {
|
||||||
if !p_vals.is_empty() {
|
if !p_vals.is_empty() {
|
||||||
if !p_vals.contains(arg_slice) {
|
if !p_vals.contains(arg_slice) {
|
||||||
self.report_error(format!("\"{}\" isn't a valid value for {}{}",
|
self.report_error(format!("\"{}\" isn't a valid value for '{}'{}",
|
||||||
arg_slice,
|
arg_slice,
|
||||||
p,
|
p,
|
||||||
format!("\n\t[valid values:{}]",
|
format!("\n\t[valid values:{}]",
|
||||||
|
@ -1358,7 +1385,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
if o.multiple && self.required.is_empty() { () }
|
if o.multiple && self.required.is_empty() { () }
|
||||||
else if !o.multiple {
|
else if !o.multiple {
|
||||||
self.report_error(
|
self.report_error(
|
||||||
format!("Argument {} requires a value but none was supplied", o),
|
format!("Argument '{}' requires a value but none was supplied", o),
|
||||||
true,
|
true,
|
||||||
true,
|
true,
|
||||||
Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>() ) );
|
Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>() ) );
|
||||||
|
@ -1377,7 +1404,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
self.report_error(
|
self.report_error(
|
||||||
format!("Argument {} requires a value but none was supplied",
|
format!("Argument '{}' requires a value but none was supplied",
|
||||||
format!("{}", self.positionals_idx.get(
|
format!("{}", self.positionals_idx.get(
|
||||||
self.positionals_name.get(a).unwrap()).unwrap())),
|
self.positionals_name.get(a).unwrap()).unwrap())),
|
||||||
true,
|
true,
|
||||||
|
@ -1552,7 +1579,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
if let Some(ref p_vals) = v.possible_vals {
|
if let Some(ref p_vals) = v.possible_vals {
|
||||||
if let Some(ref av) = arg_val {
|
if let Some(ref av) = arg_val {
|
||||||
if !p_vals.contains(&av[..]) {
|
if !p_vals.contains(&av[..]) {
|
||||||
self.report_error(format!("\"{}\" isn't a valid value for {}{}",
|
self.report_error(format!("\"{}\" isn't a valid value for '{}'{}",
|
||||||
arg_val.clone().unwrap_or(arg.to_owned()),
|
arg_val.clone().unwrap_or(arg.to_owned()),
|
||||||
v,
|
v,
|
||||||
format!("\n\t[valid values:{}]",
|
format!("\n\t[valid values:{}]",
|
||||||
|
@ -1625,7 +1652,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
self.report_error(format!("The argument '{}' cannot be used with {}",
|
self.report_error(format!("The argument '{}' cannot be used with {}",
|
||||||
v,
|
v,
|
||||||
match self.blacklisted_from(v.name, matches) {
|
match self.blacklisted_from(v.name, matches) {
|
||||||
Some(name) => name,
|
Some(name) => format!("'{}'", name),
|
||||||
None => "one or more of the specified arguments".to_owned()
|
None => "one or more of the specified arguments".to_owned()
|
||||||
}),
|
}),
|
||||||
true,
|
true,
|
||||||
|
@ -1635,8 +1662,8 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
|
|
||||||
// Make sure this isn't one being added multiple times if it doesn't suppor it
|
// Make sure this isn't one being added multiple times if it doesn't suppor it
|
||||||
if matches.args.contains_key(v.name) && !v.multiple {
|
if matches.args.contains_key(v.name) && !v.multiple {
|
||||||
self.report_error(format!("Argument {} was supplied more than once, but does not \
|
self.report_error(format!("Argument '{}' was supplied more than once, but does \
|
||||||
support multiple values", v),
|
not support multiple values", v),
|
||||||
true,
|
true,
|
||||||
true,
|
true,
|
||||||
Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>()));
|
Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>()));
|
||||||
|
@ -1727,7 +1754,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
self.report_error(format!("The argument -{} cannot be used with {}",
|
self.report_error(format!("The argument -{} cannot be used with {}",
|
||||||
arg,
|
arg,
|
||||||
match self.blacklisted_from(v.name, matches) {
|
match self.blacklisted_from(v.name, matches) {
|
||||||
Some(name) => name,
|
Some(name) => format!("'{}'", name),
|
||||||
None => "one or more of the other specified arguments".to_owned()
|
None => "one or more of the other specified arguments".to_owned()
|
||||||
}),
|
}),
|
||||||
true,
|
true,
|
||||||
|
@ -1794,7 +1821,7 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
self.report_error(format!("The argument -{} cannot be used {}",
|
self.report_error(format!("The argument -{} cannot be used {}",
|
||||||
arg,
|
arg,
|
||||||
match self.blacklisted_from(v.name, matches) {
|
match self.blacklisted_from(v.name, matches) {
|
||||||
Some(name) => name,
|
Some(name) => format!("'{}'", name),
|
||||||
None => "with one or more of the other specified \
|
None => "with one or more of the other specified \
|
||||||
arguments".to_owned()
|
arguments".to_owned()
|
||||||
}),
|
}),
|
||||||
|
@ -1868,12 +1895,11 @@ impl<'a, 'v, 'ab, 'u, 'h, 'ar> App<'a, 'v, 'ab, 'u, 'h, 'ar>{
|
||||||
None => format!("\"{}\"", name)
|
None => format!("\"{}\"", name)
|
||||||
}
|
}
|
||||||
}, match self.blacklisted_from(name, matches) {
|
}, match self.blacklisted_from(name, matches) {
|
||||||
Some(name) => name,
|
Some(name) => format!("'{}'", name),
|
||||||
None => "one or more of the other specified arguments".to_owned()
|
None => "one or more of the other specified arguments".to_owned()
|
||||||
}), true, true, Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>()));
|
}), true, true, Some(matches.args.keys().map(|k| *k).collect::<Vec<_>>()));
|
||||||
} else if self.groups.contains_key(name) {
|
} else if self.groups.contains_key(name) {
|
||||||
let grp = self.groups.get(name).unwrap();
|
for n in self.get_group_members_names(name) {
|
||||||
for n in grp.args.iter() {
|
|
||||||
if matches.args.contains_key(n) {
|
if matches.args.contains_key(n) {
|
||||||
matches.args.remove(n);
|
matches.args.remove(n);
|
||||||
self.report_error(format!("The argument '{}' cannot be used with one or \
|
self.report_error(format!("The argument '{}' cannot be used with one or \
|
||||||
|
|
Loading…
Reference in a new issue