mirror of
https://github.com/clap-rs/clap
synced 2025-01-21 17:13:53 +00:00
be60de036b
As using structopt without structopt_derive is quite meaningless, reexport structopt_derive, symplifying the usage. Inspired by the failure crate. fix #45
158 lines
4.4 KiB
Rust
158 lines
4.4 KiB
Rust
// Copyright (c) 2017 Guillaume Pinot <texitoi(a)texitoi.eu>
|
|
//
|
|
// This work is free. You can redistribute it and/or modify it under
|
|
// the terms of the Do What The Fuck You Want To Public License,
|
|
// Version 2, as published by Sam Hocevar. See the COPYING file for
|
|
// more details.
|
|
|
|
#[macro_use] extern crate structopt;
|
|
|
|
use structopt::StructOpt;
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
struct Opt {
|
|
#[structopt(short = "f", long = "force")]
|
|
force: bool,
|
|
#[structopt(short = "v", long = "verbose", parse(from_occurrences))]
|
|
verbose: u64,
|
|
#[structopt(subcommand)]
|
|
cmd: Sub
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum Sub {
|
|
#[structopt(name = "fetch")]
|
|
Fetch {},
|
|
#[structopt(name = "add")]
|
|
Add {}
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
struct Opt2 {
|
|
#[structopt(short = "f", long = "force")]
|
|
force: bool,
|
|
#[structopt(short = "v", long = "verbose", parse(from_occurrences))]
|
|
verbose: u64,
|
|
#[structopt(subcommand)]
|
|
cmd: Option<Sub>
|
|
}
|
|
|
|
#[test]
|
|
fn test_no_cmd() {
|
|
let result = Opt::clap().get_matches_from_safe(&["test"]);
|
|
assert!(result.is_err());
|
|
|
|
assert_eq!(Opt2 { force: false, verbose: 0, cmd: None },
|
|
Opt2::from_clap(&Opt2::clap().get_matches_from(&["test"])));
|
|
}
|
|
|
|
#[test]
|
|
fn test_fetch() {
|
|
assert_eq!(Opt { force: false, verbose: 3, cmd: Sub::Fetch {} },
|
|
Opt::from_clap(&Opt::clap().get_matches_from(&["test", "-vvv", "fetch"])));
|
|
assert_eq!(Opt { force: true, verbose: 0, cmd: Sub::Fetch {} },
|
|
Opt::from_clap(&Opt::clap().get_matches_from(&["test", "--force", "fetch"])));
|
|
}
|
|
|
|
#[test]
|
|
fn test_add() {
|
|
assert_eq!(Opt { force: false, verbose: 0, cmd: Sub::Add {} },
|
|
Opt::from_clap(&Opt::clap().get_matches_from(&["test", "add"])));
|
|
assert_eq!(Opt { force: false, verbose: 2, cmd: Sub::Add {} },
|
|
Opt::from_clap(&Opt::clap().get_matches_from(&["test", "-vv", "add"])));
|
|
}
|
|
|
|
#[test]
|
|
fn test_badinput() {
|
|
let result = Opt::clap().get_matches_from_safe(&["test", "badcmd"]);
|
|
assert!(result.is_err());
|
|
let result = Opt::clap().get_matches_from_safe(&["test", "add", "--verbose"]);
|
|
assert!(result.is_err());
|
|
let result = Opt::clap().get_matches_from_safe(&["test", "--badopt", "add"]);
|
|
assert!(result.is_err());
|
|
let result = Opt::clap().get_matches_from_safe(&["test", "add", "--badopt"]);
|
|
assert!(result.is_err());
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
struct Opt3 {
|
|
#[structopt(short = "a", long = "all")]
|
|
all: bool,
|
|
#[structopt(subcommand)]
|
|
cmd: Sub2
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum Sub2 {
|
|
#[structopt(name = "foo")]
|
|
Foo {
|
|
file: String,
|
|
#[structopt(subcommand)]
|
|
cmd: Sub3
|
|
},
|
|
#[structopt(name = "bar")]
|
|
Bar {
|
|
}
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum Sub3 {
|
|
#[structopt(name = "baz")]
|
|
Baz {},
|
|
#[structopt(name = "quux")]
|
|
Quux {}
|
|
}
|
|
|
|
#[test]
|
|
fn test_subsubcommand() {
|
|
assert_eq!(
|
|
Opt3 {
|
|
all: true,
|
|
cmd: Sub2::Foo { file: "lib.rs".to_string(), cmd: Sub3::Quux {} }
|
|
},
|
|
Opt3::from_clap(&Opt3::clap().get_matches_from(&["test", "--all", "foo", "lib.rs", "quux"]))
|
|
);
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum SubSubCmdWithOption {
|
|
#[structopt(name = "remote")]
|
|
Remote {
|
|
#[structopt(subcommand)]
|
|
cmd: Option<Remote>
|
|
},
|
|
#[structopt(name = "stash")]
|
|
Stash {
|
|
#[structopt(subcommand)]
|
|
cmd: Stash
|
|
},
|
|
}
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum Remote {
|
|
#[structopt(name = "add")]
|
|
Add { name: String, url: String },
|
|
#[structopt(name = "remove")]
|
|
Remove { name: String },
|
|
}
|
|
|
|
#[derive(StructOpt, PartialEq, Debug)]
|
|
enum Stash {
|
|
#[structopt(name = "save")]
|
|
Save,
|
|
#[structopt(name = "pop")]
|
|
Pop,
|
|
}
|
|
|
|
#[test]
|
|
fn sub_sub_cmd_with_option() {
|
|
fn make(args: &[&str]) -> Option<SubSubCmdWithOption> {
|
|
SubSubCmdWithOption::clap().get_matches_from_safe(args).ok().map(|m| SubSubCmdWithOption::from_clap(&m))
|
|
}
|
|
assert_eq!(Some(SubSubCmdWithOption::Remote { cmd: None }), make(&["", "remote"]));
|
|
assert_eq!(
|
|
Some(SubSubCmdWithOption::Remote { cmd: Some(Remote::Add { name: "origin".into(), url: "http".into() }) }),
|
|
make(&["", "remote", "add", "origin", "http"])
|
|
);
|
|
assert_eq!(Some(SubSubCmdWithOption::Stash { cmd: Stash::Save }), make(&["", "stash", "save"]));
|
|
assert_eq!(None, make(&["", "stash"]));
|
|
}
|