mirror of
https://github.com/clap-rs/clap
synced 2024-12-15 15:22:30 +00:00
bd25b5f615
Because we gradually build the `ArgGroup` as we parse the YAML, we don't use `ArgGroup::new`. Clap3 introduced an internal `id` in addition to the public `name` and it appears that this custom initialization code was not updated. This shows the problem with publically exposing `impl Default`. Choices: - Remove `impl Default` - Always valid - Requires spreading invariants - Callers can't implement code the same way we do - Add `ArgGroup::name` - Can be constructed in an invalid state - Centralizes invariants - A caller could implement code like the yaml logic I decided to go with `ArgGroup::name`. Fixes #2719
266 lines
6.9 KiB
Rust
266 lines
6.9 KiB
Rust
#![cfg(feature = "yaml")]
|
|
|
|
use clap::{load_yaml, App, Arg, ErrorKind, ValueHint};
|
|
|
|
#[test]
|
|
fn create_app_from_yaml() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
App::from(yaml);
|
|
}
|
|
|
|
// TODO: Uncomment to test yaml with 2 spaces https://github.com/chyh1990/yaml-rust/issues/101
|
|
// #[test]
|
|
// fn create_app_from_yaml_2spaces() {
|
|
// let yaml = load_yaml!("fixtures/app_2space.yaml");
|
|
// App::from(yaml);
|
|
// }
|
|
|
|
#[test]
|
|
fn help_message() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
let mut app = App::from(yaml);
|
|
// Generate the full help message!
|
|
let _ = app.try_get_matches_from_mut(Vec::<String>::new());
|
|
|
|
let mut help_buffer = Vec::new();
|
|
app.write_help(&mut help_buffer).unwrap();
|
|
let help_string = String::from_utf8(help_buffer).unwrap();
|
|
assert!(help_string
|
|
.contains("-h, --help prints help with a nonstandard description\n"));
|
|
}
|
|
|
|
#[test]
|
|
fn author() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
let mut app = App::from(yaml);
|
|
// Generate the full help message!
|
|
let _ = app.try_get_matches_from_mut(Vec::<String>::new());
|
|
|
|
let mut help_buffer = Vec::new();
|
|
app.write_help(&mut help_buffer).unwrap();
|
|
let help_string = String::from_utf8(help_buffer).unwrap();
|
|
assert!(help_string.contains("Kevin K. <kbknapp@gmail.com>"));
|
|
}
|
|
|
|
#[test]
|
|
fn app_settings() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yaml);
|
|
|
|
let m = app.try_get_matches_from(vec!["prog"]);
|
|
|
|
assert!(m.is_err());
|
|
assert_eq!(
|
|
m.unwrap_err().kind,
|
|
ErrorKind::DisplayHelpOnMissingArgumentOrSubcommand
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Unknown AppSetting 'random' found in YAML file for app"]
|
|
fn app_setting_invalid() {
|
|
let yaml = load_yaml!("fixtures/app_setting_invalid.yaml");
|
|
App::from(yaml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Unknown ArgSetting 'random' found in YAML file for arg 'option'"]
|
|
fn arg_setting_invalid() {
|
|
let yaml = load_yaml!("fixtures/arg_setting_invalid.yaml");
|
|
App::from(yaml);
|
|
}
|
|
|
|
// ValueHint must be parsed correctly from Yaml
|
|
#[test]
|
|
fn value_hint() {
|
|
let yml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yml);
|
|
|
|
let arg = app
|
|
.get_arguments()
|
|
.find(|a| a.get_name() == "value_hint")
|
|
.unwrap();
|
|
assert_eq!(arg.get_value_hint(), ValueHint::FilePath);
|
|
}
|
|
|
|
#[test]
|
|
fn default_value_if_not_triggered_by_argument() {
|
|
let yml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yml);
|
|
|
|
// Fixtures use "other" as value
|
|
let matches = app.try_get_matches_from(vec!["prog", "wrong"]).unwrap();
|
|
|
|
assert!(matches.value_of("positional2").is_none());
|
|
}
|
|
|
|
#[test]
|
|
fn default_value_if_triggered_by_matching_argument() {
|
|
let yml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yml);
|
|
|
|
let matches = app.try_get_matches_from(vec!["prog", "other"]).unwrap();
|
|
assert_eq!(matches.value_of("positional2").unwrap(), "something");
|
|
}
|
|
|
|
#[test]
|
|
fn default_value_if_triggered_by_flag() {
|
|
let yml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yml);
|
|
|
|
let matches = app
|
|
.try_get_matches_from(vec!["prog", "--flag", "flagvalue"])
|
|
.unwrap();
|
|
|
|
assert_eq!(matches.value_of("positional2").unwrap(), "some");
|
|
}
|
|
|
|
#[test]
|
|
fn default_value_if_triggered_by_flag_and_argument() {
|
|
let yml = load_yaml!("fixtures/app.yaml");
|
|
let app = App::from(yml);
|
|
|
|
let matches = app
|
|
.try_get_matches_from(vec!["prog", "--flag", "flagvalue", "other"])
|
|
.unwrap();
|
|
|
|
// First condition triggers, therefore "some"
|
|
assert_eq!(matches.value_of("positional2").unwrap(), "some");
|
|
}
|
|
|
|
#[test]
|
|
fn yaml_multiple_occurrences() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
let matches = App::from(yaml)
|
|
.try_get_matches_from(vec!["prog", "-vvv"])
|
|
.unwrap();
|
|
assert_eq!(matches.occurrences_of("verbose"), 3);
|
|
}
|
|
|
|
#[test]
|
|
fn yaml_multiple_values() {
|
|
let yaml = load_yaml!("fixtures/app.yaml");
|
|
let matches = App::from(yaml)
|
|
.try_get_matches_from(vec!["prog", "-s", "aaa", "bbb"])
|
|
.unwrap();
|
|
assert_eq!(
|
|
matches
|
|
.values_of("settings")
|
|
.unwrap()
|
|
.collect::<Vec<&str>>(),
|
|
vec!["aaa", "bbb"]
|
|
);
|
|
}
|
|
|
|
#[cfg(feature = "regex")]
|
|
#[test]
|
|
fn regex_with_invalid_string() {
|
|
let yml = load_yaml!("fixtures/app_regex.yaml");
|
|
let app = App::from(yml);
|
|
let res = app.try_get_matches_from(vec!["prog", "not a proper filter"]);
|
|
|
|
assert!(res.is_err());
|
|
}
|
|
|
|
#[cfg(feature = "regex")]
|
|
#[test]
|
|
fn regex_with_valid_string() {
|
|
let yml = load_yaml!("fixtures/app_regex.yaml");
|
|
let app = App::from(yml);
|
|
|
|
let matches = app.try_get_matches_from(vec!["prog", "*.txt"]).unwrap();
|
|
|
|
assert_eq!(matches.value_of("filter").unwrap(), "*.txt");
|
|
}
|
|
|
|
#[cfg(feature = "regex")]
|
|
#[test]
|
|
#[should_panic]
|
|
fn regex_with_invalid_yaml() {
|
|
let yml = load_yaml!("fixtures/app_regex_invalid.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
fn extra_fields() {
|
|
let yml = load_yaml!("fixtures/extra_fields.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Unknown setting 'random' in YAML file for arg 'option'"]
|
|
fn extra_fields_invalid_arg() {
|
|
let yml = load_yaml!("fixtures/extra_fields_invalid_arg.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Unknown setting 'random' in YAML file for subcommand 'info'"]
|
|
fn extra_fields_invalid_app() {
|
|
let yml = load_yaml!("fixtures/extra_fields_invalid_app.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "YAML file must be a hash"]
|
|
fn app_not_hash() {
|
|
let yml = load_yaml!("fixtures/not_hash.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "YAML file must be a hash"]
|
|
fn arg_file_not_hash() {
|
|
let yml = load_yaml!("fixtures/not_hash.yaml");
|
|
Arg::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Subcommand must be a hash"]
|
|
fn subcommand_not_hash() {
|
|
let yml = load_yaml!("fixtures/field_not_hash.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Arg must be a hash"]
|
|
fn arg_not_hash() {
|
|
let yml = load_yaml!("fixtures/arg_not_hash.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Subcommand name must be a string"]
|
|
fn subcommand_name_not_string() {
|
|
let yml = load_yaml!("fixtures/name_not_string.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Arg name must be a string"]
|
|
fn arg_name_not_string() {
|
|
let yml = load_yaml!("fixtures/name_not_string.yaml");
|
|
Arg::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "App fields must be strings"]
|
|
fn app_field_not_string() {
|
|
let yml = load_yaml!("fixtures/app_field_not_string.yaml");
|
|
App::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Arg fields must be strings"]
|
|
fn arg_field_not_string() {
|
|
let yml = load_yaml!("fixtures/arg_field_not_string.yaml");
|
|
Arg::from(yml);
|
|
}
|
|
|
|
#[test]
|
|
fn multiple_groups() {
|
|
let yml = load_yaml!("fixtures/multiple_groups.yaml");
|
|
let matches = App::from(yml).try_get_matches_from(&["app", "-a", "-c"]);
|
|
eprintln!("{:?}", matches);
|
|
assert!(matches.is_ok());
|
|
}
|