clap/tests/builder/default_missing_vals.rs
Ed Page c72f03e53f refactor(parser): Be more explicit in default_missing_values
I wrote these tests expecting to highlight a bug but it turns out things
were structured just right to not exhibit it.  The fact that the code
looks like its broken is a problem, so I restructured it (put it first,
changed the source) so it doesn't look suspicious anymore.
2022-05-27 07:04:24 -05:00

273 lines
7.6 KiB
Rust

use clap::{arg, Arg, Command};
#[test]
fn opt_missing() {
let r = Command::new("df")
.arg(
Arg::new("color")
.long("color")
.default_value("auto")
.min_values(0)
.require_equals(true)
.default_missing_value("always"),
)
.try_get_matches_from(vec![""]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("color"));
assert_eq!(
m.get_one::<String>("color").map(|v| v.as_str()).unwrap(),
"auto"
);
assert_eq!(m.occurrences_of("color"), 0);
assert_eq!(
m.value_source("color").unwrap(),
clap::ValueSource::DefaultValue
);
}
#[test]
fn opt_present_with_missing_value() {
let r = Command::new("df")
.arg(
Arg::new("color")
.long("color")
.default_value("auto")
.min_values(0)
.require_equals(true)
.default_missing_value("always"),
)
.try_get_matches_from(vec!["", "--color"]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("color"));
assert_eq!(
m.get_one::<String>("color").map(|v| v.as_str()).unwrap(),
"always"
);
assert_eq!(m.occurrences_of("color"), 1);
assert_eq!(
m.value_source("color").unwrap(),
clap::ValueSource::CommandLine
);
}
#[test]
fn opt_present_with_value() {
let r = Command::new("df")
.arg(
Arg::new("color")
.long("color")
.default_value("auto")
.min_values(0)
.require_equals(true)
.default_missing_value("always"),
)
.try_get_matches_from(vec!["", "--color=never"]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("color"));
assert_eq!(
m.get_one::<String>("color").map(|v| v.as_str()).unwrap(),
"never"
);
assert_eq!(m.occurrences_of("color"), 1);
assert_eq!(
m.value_source("color").unwrap(),
clap::ValueSource::CommandLine
);
}
#[test]
fn opt_present_with_empty_value() {
let r = Command::new("df")
.arg(
Arg::new("color")
.long("color")
.default_value("auto")
.require_equals(true)
.default_missing_value("always"),
)
.try_get_matches_from(vec!["", "--color="]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("color"));
assert_eq!(
m.get_one::<String>("color").map(|v| v.as_str()).unwrap(),
""
);
assert_eq!(m.occurrences_of("color"), 1);
assert_eq!(
m.value_source("color").unwrap(),
clap::ValueSource::CommandLine
);
}
//## `default_value`/`default_missing_value` non-interaction checks
#[test]
fn opt_default() {
// assert no change to usual argument handling when adding default_missing_value()
let r = Command::new("cmd")
.arg(
arg!(o: -o [opt] "some opt")
.default_value("default")
.default_missing_value("default_missing"),
)
.try_get_matches_from(vec![""]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("o"));
assert_eq!(
m.get_one::<String>("o").map(|v| v.as_str()).unwrap(),
"default"
);
}
#[test]
fn opt_default_user_override() {
// assert no change to usual argument handling when adding default_missing_value()
let r = Command::new("cmd")
.arg(
arg!(o: -o [opt] "some opt")
.default_value("default")
.default_missing_value("default_missing"),
)
.try_get_matches_from(vec!["", "-o=value"]);
assert!(r.is_ok(), "{}", r.unwrap_err());
let m = r.unwrap();
assert!(m.is_present("o"));
assert_eq!(
m.get_one::<String>("o").map(|v| v.as_str()).unwrap(),
"value"
);
}
#[test]
#[allow(clippy::bool_assert_comparison)]
fn default_missing_value_flag_value() {
let cmd = Command::new("test").arg(
Arg::new("flag")
.long("flag")
.takes_value(true)
.default_value("false")
.default_missing_value("true"),
);
let m = cmd.clone().try_get_matches_from(&["test"]).unwrap();
assert!(m.is_present("flag"));
assert_eq!(
m.get_one::<String>("flag").map(|v| v.as_str()),
Some("false")
);
assert_eq!(m.occurrences_of("flag"), 0);
assert_eq!(
m.value_source("flag").unwrap(),
clap::ValueSource::DefaultValue
);
let m = cmd
.clone()
.try_get_matches_from(&["test", "--flag"])
.unwrap();
assert!(m.is_present("flag"));
assert_eq!(
m.get_one::<String>("flag").map(|v| v.as_str()),
Some("true")
);
assert_eq!(m.occurrences_of("flag"), 1);
assert_eq!(
m.value_source("flag").unwrap(),
clap::ValueSource::CommandLine
);
let m = cmd
.clone()
.try_get_matches_from(&["test", "--flag=true"])
.unwrap();
assert!(m.is_present("flag"));
assert_eq!(
m.get_one::<String>("flag").map(|v| v.as_str()),
Some("true")
);
assert_eq!(m.occurrences_of("flag"), 1);
assert_eq!(
m.value_source("flag").unwrap(),
clap::ValueSource::CommandLine
);
let m = cmd.try_get_matches_from(&["test", "--flag=false"]).unwrap();
assert!(m.is_present("flag"));
assert_eq!(
m.get_one::<String>("flag").map(|v| v.as_str()),
Some("false")
);
assert_eq!(m.occurrences_of("flag"), 1);
assert_eq!(
m.value_source("flag").unwrap(),
clap::ValueSource::CommandLine
);
}
#[test]
fn delimited_missing_value() {
let cmd = Command::new("test").arg(
Arg::new("flag")
.long("flag")
.default_value("one,two")
.default_missing_value("three,four")
.min_values(0)
.value_delimiter(',')
.require_equals(true),
);
let m = cmd.clone().try_get_matches_from(["test"]).unwrap();
assert_eq!(
m.get_many::<String>("flag")
.unwrap()
.map(|s| s.as_str())
.collect::<Vec<_>>(),
vec!["one", "two"]
);
assert_eq!(m.occurrences_of("flag"), 0);
let m = cmd.try_get_matches_from(["test", "--flag"]).unwrap();
assert_eq!(
m.get_many::<String>("flag")
.unwrap()
.map(|s| s.as_str())
.collect::<Vec<_>>(),
vec!["three", "four"]
);
assert_eq!(m.occurrences_of("flag"), 1);
}
#[cfg(debug_assertions)]
#[test]
#[should_panic = "Argument `arg`'s default_missing_value=\"value\" failed validation: error: \"value\" isn't a valid value for '<arg>'"]
fn default_missing_values_are_possible_values() {
use clap::{Arg, Command};
let _ = Command::new("test")
.arg(
Arg::new("arg")
.value_parser(["one", "two"])
.default_missing_value("value"),
)
.try_get_matches();
}
#[cfg(debug_assertions)]
#[test]
#[should_panic = "Argument `arg`'s default_missing_value=\"value\" failed validation: error: Invalid value \"value\" for '<arg>"]
fn default_missing_values_are_valid() {
use clap::{Arg, Command};
let _ = Command::new("test")
.arg(
Arg::new("arg")
.value_parser(clap::value_parser!(u32))
.default_missing_value("value"),
)
.try_get_matches();
}