summaryrefslogtreecommitdiffstats
path: root/third_party/rust/clap/src/parse/arg_matcher.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/clap/src/parse/arg_matcher.rs')
-rw-r--r--third_party/rust/clap/src/parse/arg_matcher.rs234
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
+ }
+}