diff options
Diffstat (limited to 'third_party/rust/clap/src/parse/arg_matcher.rs')
-rw-r--r-- | third_party/rust/clap/src/parse/arg_matcher.rs | 234 |
1 files changed, 234 insertions, 0 deletions
diff --git a/third_party/rust/clap/src/parse/arg_matcher.rs b/third_party/rust/clap/src/parse/arg_matcher.rs new file mode 100644 index 0000000000..31804a46fc --- /dev/null +++ b/third_party/rust/clap/src/parse/arg_matcher.rs @@ -0,0 +1,234 @@ +// Std +use std::{collections::HashMap, ffi::OsString, mem, ops::Deref}; + +// Internal +use crate::{ + build::{Arg, ArgPredicate, Command}, + parse::{ArgMatches, MatchedArg, SubCommand, ValueSource}, + util::Id, +}; + +// Third party +use indexmap::map::Entry; + +#[derive(Debug, Default)] +pub(crate) struct ArgMatcher(ArgMatches); + +impl ArgMatcher { + pub(crate) fn new(_cmd: &Command) -> Self { + ArgMatcher(ArgMatches { + #[cfg(debug_assertions)] + valid_args: { + let args = _cmd.get_arguments().map(|a| a.id.clone()); + let groups = _cmd.get_groups().map(|g| g.id.clone()); + args.chain(groups).collect() + }, + #[cfg(debug_assertions)] + valid_subcommands: _cmd.get_subcommands().map(|sc| sc.get_id()).collect(), + // HACK: Allow an external subcommand's ArgMatches be a stand-in for any ArgMatches + // since users can't detect it and avoid the asserts. + // + // See clap-rs/clap#3263 + #[cfg(debug_assertions)] + #[cfg(not(feature = "unstable-v4"))] + disable_asserts: _cmd.is_allow_external_subcommands_set(), + #[cfg(debug_assertions)] + #[cfg(feature = "unstable-v4")] + disable_asserts: false, + ..Default::default() + }) + } + + pub(crate) fn into_inner(self) -> ArgMatches { + self.0 + } + + pub(crate) fn propagate_globals(&mut self, global_arg_vec: &[Id]) { + debug!( + "ArgMatcher::get_global_values: global_arg_vec={:?}", + global_arg_vec + ); + let mut vals_map = HashMap::new(); + self.fill_in_global_values(global_arg_vec, &mut vals_map); + } + + fn fill_in_global_values( + &mut self, + global_arg_vec: &[Id], + vals_map: &mut HashMap<Id, MatchedArg>, + ) { + for global_arg in global_arg_vec { + if let Some(ma) = self.get(global_arg) { + // We have to check if the parent's global arg wasn't used but still exists + // such as from a default value. + // + // For example, `myprog subcommand --global-arg=value` where --global-arg defines + // a default value of `other` myprog would have an existing MatchedArg for + // --global-arg where the value is `other`, however the occurs will be 0. + let to_update = if let Some(parent_ma) = vals_map.get(global_arg) { + if parent_ma.get_occurrences() > 0 && ma.get_occurrences() == 0 { + parent_ma + } else { + ma + } + } else { + ma + } + .clone(); + vals_map.insert(global_arg.clone(), to_update); + } + } + if let Some(ref mut sc) = self.0.subcommand { + let mut am = ArgMatcher(mem::take(&mut sc.matches)); + am.fill_in_global_values(global_arg_vec, vals_map); + mem::swap(&mut am.0, &mut sc.matches); + } + + for (name, matched_arg) in vals_map.iter_mut() { + self.0.args.insert(name.clone(), matched_arg.clone()); + } + } + + #[cfg(not(feature = "unstable-v4"))] + pub(crate) fn get_mut(&mut self, arg: &Id) -> Option<&mut MatchedArg> { + self.0.args.get_mut(arg) + } + + pub(crate) fn get(&self, arg: &Id) -> Option<&MatchedArg> { + self.0.args.get(arg) + } + + pub(crate) fn remove(&mut self, arg: &Id) { + self.0.args.swap_remove(arg); + } + + pub(crate) fn contains(&self, arg: &Id) -> bool { + self.0.args.contains_key(arg) + } + + pub(crate) fn arg_names(&self) -> indexmap::map::Keys<Id, MatchedArg> { + self.0.args.keys() + } + + pub(crate) fn entry(&mut self, arg: &Id) -> indexmap::map::Entry<Id, MatchedArg> { + self.0.args.entry(arg.clone()) + } + + pub(crate) fn subcommand(&mut self, sc: SubCommand) { + self.0.subcommand = Some(Box::new(sc)); + } + + pub(crate) fn subcommand_name(&self) -> Option<&str> { + self.0.subcommand_name() + } + + pub(crate) fn iter(&self) -> indexmap::map::Iter<Id, MatchedArg> { + self.0.args.iter() + } + + pub(crate) fn check_explicit<'a>(&self, arg: &Id, predicate: ArgPredicate<'a>) -> bool { + self.get(arg).map_or(false, |a| a.check_explicit(predicate)) + } + + pub(crate) fn inc_occurrence_of_arg(&mut self, arg: &Arg) { + let id = &arg.id; + debug!("ArgMatcher::inc_occurrence_of_arg: id={:?}", id); + let ma = self.entry(id).or_insert(MatchedArg::new()); + ma.update_ty(ValueSource::CommandLine); + ma.set_ignore_case(arg.is_ignore_case_set()); + ma.invalid_utf8_allowed(arg.is_allow_invalid_utf8_set()); + ma.inc_occurrences(); + } + + pub(crate) fn inc_occurrence_of_group(&mut self, id: &Id) { + debug!("ArgMatcher::inc_occurrence_of_group: id={:?}", id); + let ma = self.entry(id).or_insert(MatchedArg::new()); + ma.update_ty(ValueSource::CommandLine); + ma.inc_occurrences(); + } + + #[cfg(feature = "unstable-v4")] + pub(crate) fn inc_occurrence_of_external(&mut self, allow_invalid_utf8: bool) { + let id = &Id::empty_hash(); + debug!( + "ArgMatcher::inc_occurrence_of_external: id={:?}, allow_invalid_utf8={}", + id, allow_invalid_utf8 + ); + let ma = self.entry(id).or_insert(MatchedArg::new()); + ma.update_ty(ValueSource::CommandLine); + ma.invalid_utf8_allowed(allow_invalid_utf8); + ma.inc_occurrences(); + } + + pub(crate) fn add_val_to(&mut self, arg: &Id, val: OsString, ty: ValueSource, append: bool) { + if append { + self.append_val_to(arg, val, ty); + } else { + self.push_val_to(arg, val, ty); + } + } + + fn push_val_to(&mut self, arg: &Id, val: OsString, ty: ValueSource) { + // We will manually inc occurrences later(for flexibility under + // specific circumstances, like only add one occurrence for flag + // when we met: `--flag=one,two`). + let ma = self.entry(arg).or_default(); + ma.update_ty(ty); + ma.push_val(val); + } + + fn append_val_to(&mut self, arg: &Id, val: OsString, ty: ValueSource) { + let ma = self.entry(arg).or_default(); + ma.update_ty(ty); + ma.append_val(val); + } + + pub(crate) fn new_val_group(&mut self, arg: &Id) { + let ma = self.entry(arg).or_default(); + ma.new_val_group(); + } + + pub(crate) fn add_index_to(&mut self, arg: &Id, idx: usize, ty: ValueSource) { + let ma = self.entry(arg).or_default(); + ma.update_ty(ty); + ma.push_index(idx); + } + + pub(crate) fn has_val_groups(&mut self, arg: &Id) -> bool { + match self.entry(arg) { + Entry::Occupied(e) => e.get().has_val_groups(), + Entry::Vacant(_) => false, + } + } + + pub(crate) fn needs_more_vals(&self, o: &Arg) -> bool { + debug!("ArgMatcher::needs_more_vals: o={}", o.name); + if let Some(ma) = self.get(&o.id) { + let current_num = ma.num_vals(); + if let Some(num) = o.num_vals { + debug!("ArgMatcher::needs_more_vals: num_vals...{}", num); + return if o.is_multiple_occurrences_set() { + (current_num % num) != 0 + } else { + num != current_num + }; + } else if let Some(num) = o.max_vals { + debug!("ArgMatcher::needs_more_vals: max_vals...{}", num); + return current_num < num; + } else if o.min_vals.is_some() { + debug!("ArgMatcher::needs_more_vals: min_vals...true"); + return true; + } + return o.is_multiple_values_set(); + } + true + } +} + +impl Deref for ArgMatcher { + type Target = ArgMatches; + + fn deref(&self) -> &Self::Target { + &self.0 + } +} |