// Copyright (c) 2017 Guillaume Pinot // // 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 } #[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 }, #[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::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"])); }