mirror of
https://github.com/clap-rs/clap
synced 2025-01-23 01:45:00 +00:00
d10938dd33
This is mostly targeted at reducing startup time for no-op commands within *very* large applications, like deno (see #4774). This comes at the cost of 1.1 KiB of binary size
153 lines
4.8 KiB
Rust
153 lines
4.8 KiB
Rust
use clap::{Arg, ArgAction, ArgMatches, Command};
|
|
|
|
fn get_app() -> Command {
|
|
Command::new("myprog")
|
|
.arg(
|
|
Arg::new("GLOBAL_ARG")
|
|
.long("global-arg")
|
|
.help("Specifies something needed by the subcommands")
|
|
.global(true)
|
|
.action(ArgAction::Set)
|
|
.default_value("default_value"),
|
|
)
|
|
.arg(
|
|
Arg::new("GLOBAL_FLAG")
|
|
.long("global-flag")
|
|
.help("Specifies something needed by the subcommands")
|
|
.global(true)
|
|
.action(ArgAction::Count),
|
|
)
|
|
.subcommand(Command::new("outer").defer(|cmd| cmd.subcommand(Command::new("inner"))))
|
|
}
|
|
|
|
fn get_matches(cmd: Command, argv: &'static str) -> ArgMatches {
|
|
cmd.try_get_matches_from(argv.split(' ').collect::<Vec<_>>())
|
|
.unwrap()
|
|
}
|
|
|
|
fn get_outer_matches(m: &ArgMatches) -> &ArgMatches {
|
|
m.subcommand_matches("outer")
|
|
.expect("could not access outer subcommand")
|
|
}
|
|
|
|
fn get_inner_matches(m: &ArgMatches) -> &ArgMatches {
|
|
get_outer_matches(m)
|
|
.subcommand_matches("inner")
|
|
.expect("could not access inner subcommand")
|
|
}
|
|
|
|
fn top_can_access_arg<T: Into<Option<&'static str>>>(m: &ArgMatches, val: T) -> bool {
|
|
m.get_one::<String>("GLOBAL_ARG").map(|v| v.as_str()) == val.into()
|
|
}
|
|
|
|
fn inner_can_access_arg<T: Into<Option<&'static str>>>(m: &ArgMatches, val: T) -> bool {
|
|
get_inner_matches(m)
|
|
.get_one::<String>("GLOBAL_ARG")
|
|
.map(|v| v.as_str())
|
|
== val.into()
|
|
}
|
|
|
|
fn outer_can_access_arg<T: Into<Option<&'static str>>>(m: &ArgMatches, val: T) -> bool {
|
|
get_outer_matches(m)
|
|
.get_one::<String>("GLOBAL_ARG")
|
|
.map(|v| v.as_str())
|
|
== val.into()
|
|
}
|
|
|
|
fn top_can_access_flag(m: &ArgMatches, present: bool, occurrences: u8) -> bool {
|
|
(m.contains_id("GLOBAL_FLAG") == present)
|
|
&& (m.get_one::<u8>("GLOBAL_FLAG").copied() == Some(occurrences))
|
|
}
|
|
|
|
fn inner_can_access_flag(m: &ArgMatches, present: bool, occurrences: u8) -> bool {
|
|
let m = get_inner_matches(m);
|
|
(m.contains_id("GLOBAL_FLAG") == present)
|
|
&& (m.get_one::<u8>("GLOBAL_FLAG").copied() == Some(occurrences))
|
|
}
|
|
|
|
fn outer_can_access_flag(m: &ArgMatches, present: bool, occurrences: u8) -> bool {
|
|
let m = get_outer_matches(m);
|
|
(m.contains_id("GLOBAL_FLAG") == present)
|
|
&& (m.get_one::<u8>("GLOBAL_FLAG").copied() == Some(occurrences))
|
|
}
|
|
|
|
#[test]
|
|
fn global_arg_used_top_level() {
|
|
let m = get_matches(get_app(), "myprog --global-arg=some_value outer inner");
|
|
|
|
assert!(top_can_access_arg(&m, "some_value"));
|
|
assert!(inner_can_access_arg(&m, "some_value"));
|
|
assert!(outer_can_access_arg(&m, "some_value"));
|
|
}
|
|
|
|
#[test]
|
|
fn global_arg_used_outer() {
|
|
let m = get_matches(get_app(), "myprog outer --global-arg=some_value inner");
|
|
|
|
assert!(top_can_access_arg(&m, "some_value"));
|
|
assert!(inner_can_access_arg(&m, "some_value"));
|
|
assert!(outer_can_access_arg(&m, "some_value"));
|
|
}
|
|
|
|
#[test]
|
|
fn global_arg_used_inner() {
|
|
let m = get_matches(get_app(), "myprog outer inner --global-arg=some_value");
|
|
|
|
assert!(top_can_access_arg(&m, "some_value"));
|
|
assert!(inner_can_access_arg(&m, "some_value"));
|
|
assert!(outer_can_access_arg(&m, "some_value"));
|
|
}
|
|
|
|
#[test]
|
|
fn global_arg_default_value() {
|
|
let m = get_matches(get_app(), "myprog outer inner");
|
|
|
|
assert!(top_can_access_arg(&m, "default_value"));
|
|
assert!(inner_can_access_arg(&m, "default_value"));
|
|
assert!(outer_can_access_arg(&m, "default_value"));
|
|
}
|
|
|
|
#[test]
|
|
fn global_flag_used_top_level() {
|
|
let m = get_matches(get_app(), "myprog --global-flag outer inner");
|
|
|
|
assert!(top_can_access_flag(&m, true, 1));
|
|
assert!(inner_can_access_flag(&m, true, 1));
|
|
assert!(outer_can_access_flag(&m, true, 1));
|
|
}
|
|
|
|
#[test]
|
|
fn global_flag_used_outer() {
|
|
let m = get_matches(get_app(), "myprog outer --global-flag inner");
|
|
|
|
assert!(top_can_access_flag(&m, true, 1));
|
|
assert!(inner_can_access_flag(&m, true, 1));
|
|
assert!(outer_can_access_flag(&m, true, 1));
|
|
}
|
|
|
|
#[test]
|
|
fn global_flag_used_inner() {
|
|
let m = get_matches(get_app(), "myprog outer inner --global-flag");
|
|
|
|
assert!(top_can_access_flag(&m, true, 1));
|
|
assert!(inner_can_access_flag(&m, true, 1));
|
|
assert!(outer_can_access_flag(&m, true, 1));
|
|
}
|
|
|
|
#[test]
|
|
fn global_flag_2x_used_top_level() {
|
|
let m = get_matches(get_app(), "myprog --global-flag --global-flag outer inner");
|
|
|
|
assert!(top_can_access_flag(&m, true, 2));
|
|
assert!(inner_can_access_flag(&m, true, 2));
|
|
assert!(outer_can_access_flag(&m, true, 2));
|
|
}
|
|
|
|
#[test]
|
|
fn global_flag_2x_used_inner() {
|
|
let m = get_matches(get_app(), "myprog outer inner --global-flag --global-flag");
|
|
|
|
assert!(top_can_access_flag(&m, true, 2));
|
|
assert!(inner_can_access_flag(&m, true, 2));
|
|
assert!(outer_can_access_flag(&m, true, 2));
|
|
}
|