use clap::{Arg, ArgAction, ArgMatches, Command}; fn get_app() -> Command<'static> { Command::new("myprog") .arg( Arg::new("GLOBAL_ARG") .long("global-arg") .help("Specifies something needed by the subcommands") .global(true) .takes_value(true) .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").subcommand(Command::new("inner"))) } fn get_matches(cmd: Command<'static>, argv: &'static str) -> ArgMatches { cmd.try_get_matches_from(argv.split(' ').collect::>()) .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>>(m: &ArgMatches, val: T) -> bool { m.get_one::("GLOBAL_ARG").map(|v| v.as_str()) == val.into() } fn inner_can_access_arg>>(m: &ArgMatches, val: T) -> bool { get_inner_matches(m) .get_one::("GLOBAL_ARG") .map(|v| v.as_str()) == val.into() } fn outer_can_access_arg>>(m: &ArgMatches, val: T) -> bool { get_outer_matches(m) .get_one::("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::("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::("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::("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)); }