mirror of
https://github.com/clap-rs/clap
synced 2025-01-25 02:45:00 +00:00
126 lines
4.1 KiB
Rust
126 lines
4.1 KiB
Rust
use std::collections::BTreeMap;
|
|
|
|
use clap::{command, value_parser, Arg, ArgAction, ArgGroup, ArgMatches, Command};
|
|
|
|
fn main() {
|
|
let matches = cli().get_matches();
|
|
let values = Value::from_matches(&matches);
|
|
println!("{values:#?}");
|
|
}
|
|
|
|
fn cli() -> Command {
|
|
command!()
|
|
.group(ArgGroup::new("tests").multiple(true))
|
|
.next_help_heading("TESTS")
|
|
.args([
|
|
position_sensitive_flag(Arg::new("empty"))
|
|
.long("empty")
|
|
.action(ArgAction::Append)
|
|
.help("File is empty and is either a regular file or a directory")
|
|
.group("tests"),
|
|
Arg::new("name")
|
|
.long("name")
|
|
.action(ArgAction::Append)
|
|
.help("Base of file name (the path with the leading directories removed) matches shell pattern pattern")
|
|
.group("tests")
|
|
])
|
|
.group(ArgGroup::new("operators").multiple(true))
|
|
.next_help_heading("OPERATORS")
|
|
.args([
|
|
position_sensitive_flag(Arg::new("or"))
|
|
.short('o')
|
|
.long("or")
|
|
.action(ArgAction::Append)
|
|
.help("expr2 is not evaluate if exp1 is true")
|
|
.group("operators"),
|
|
position_sensitive_flag(Arg::new("and"))
|
|
.short('a')
|
|
.long("and")
|
|
.action(ArgAction::Append)
|
|
.help("Same as `expr1 expr1`")
|
|
.group("operators"),
|
|
])
|
|
}
|
|
|
|
fn position_sensitive_flag(arg: Arg) -> Arg {
|
|
// Flags don't track the position of each occurrence, so we need to emulate flags with
|
|
// value-less options to get the same result
|
|
arg.num_args(0)
|
|
.value_parser(value_parser!(bool))
|
|
.default_missing_value("true")
|
|
.default_value("false")
|
|
}
|
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, Debug)]
|
|
pub enum Value {
|
|
Bool(bool),
|
|
String(String),
|
|
}
|
|
|
|
impl Value {
|
|
pub fn from_matches(matches: &ArgMatches) -> Vec<(clap::Id, Self)> {
|
|
let mut values = BTreeMap::new();
|
|
for id in matches.ids() {
|
|
if matches.try_get_many::<clap::Id>(id.as_str()).is_ok() {
|
|
// ignore groups
|
|
continue;
|
|
}
|
|
let value_source = matches
|
|
.value_source(id.as_str())
|
|
.expect("id came from matches");
|
|
if value_source != clap::parser::ValueSource::CommandLine {
|
|
// Any other source just gets tacked on at the end (like default values)
|
|
continue;
|
|
}
|
|
if Self::extract::<String>(matches, id, &mut values) {
|
|
continue;
|
|
}
|
|
if Self::extract::<bool>(matches, id, &mut values) {
|
|
continue;
|
|
}
|
|
unimplemented!("unknown type for {id}: {matches:?}");
|
|
}
|
|
values.into_values().collect::<Vec<_>>()
|
|
}
|
|
|
|
fn extract<T: Clone + Into<Value> + Send + Sync + 'static>(
|
|
matches: &ArgMatches,
|
|
id: &clap::Id,
|
|
output: &mut BTreeMap<usize, (clap::Id, Self)>,
|
|
) -> bool {
|
|
match matches.try_get_many::<T>(id.as_str()) {
|
|
Ok(Some(values)) => {
|
|
for (value, index) in values.zip(
|
|
matches
|
|
.indices_of(id.as_str())
|
|
.expect("id came from matches"),
|
|
) {
|
|
output.insert(index, (id.clone(), value.clone().into()));
|
|
}
|
|
true
|
|
}
|
|
Ok(None) => {
|
|
unreachable!("`ids` only reports what is present")
|
|
}
|
|
Err(clap::parser::MatchesError::UnknownArgument { .. }) => {
|
|
unreachable!("id came from matches")
|
|
}
|
|
Err(clap::parser::MatchesError::Downcast { .. }) => false,
|
|
Err(_) => {
|
|
unreachable!("id came from matches")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl From<String> for Value {
|
|
fn from(other: String) -> Self {
|
|
Self::String(other)
|
|
}
|
|
}
|
|
|
|
impl From<bool> for Value {
|
|
fn from(other: bool) -> Self {
|
|
Self::Bool(other)
|
|
}
|
|
}
|