mirror of
https://github.com/clap-rs/clap
synced 2024-12-13 14:22:34 +00:00
c9eef44213
This was ported over from the usage parser which modeled after docopt. We just never got around to implementing the rest of the syntax. However, when considering this as a standalone feature, an `arg!(--flag <value>)`, outside of other context, should be optional. This is how the help would display it. Fixes #4206
124 lines
4.3 KiB
Rust
124 lines
4.3 KiB
Rust
use std::ffi::OsString;
|
|
use std::path::PathBuf;
|
|
|
|
use clap::{arg, Command};
|
|
|
|
fn cli() -> Command {
|
|
Command::new("git")
|
|
.about("A fictional versioning CLI")
|
|
.subcommand_required(true)
|
|
.arg_required_else_help(true)
|
|
.allow_external_subcommands(true)
|
|
.subcommand(
|
|
Command::new("clone")
|
|
.about("Clones repos")
|
|
.arg(arg!(<REMOTE> "The remote to clone"))
|
|
.arg_required_else_help(true),
|
|
)
|
|
.subcommand(
|
|
Command::new("diff")
|
|
.about("Compare two commits")
|
|
.arg(arg!(base: [COMMIT]))
|
|
.arg(arg!(head: [COMMIT]))
|
|
.arg(arg!(path: [PATH]).last(true)),
|
|
)
|
|
.subcommand(
|
|
Command::new("push")
|
|
.about("pushes things")
|
|
.arg(arg!(<REMOTE> "The remote to target"))
|
|
.arg_required_else_help(true),
|
|
)
|
|
.subcommand(
|
|
Command::new("add")
|
|
.about("adds things")
|
|
.arg_required_else_help(true)
|
|
.arg(arg!(<PATH> ... "Stuff to add").value_parser(clap::value_parser!(PathBuf))),
|
|
)
|
|
.subcommand(
|
|
Command::new("stash")
|
|
.args_conflicts_with_subcommands(true)
|
|
.args(push_args())
|
|
.subcommand(Command::new("push").args(push_args()))
|
|
.subcommand(Command::new("pop").arg(arg!([STASH])))
|
|
.subcommand(Command::new("apply").arg(arg!([STASH]))),
|
|
)
|
|
}
|
|
|
|
fn push_args() -> Vec<clap::Arg> {
|
|
vec![arg!(-m --message <MESSAGE>)]
|
|
}
|
|
|
|
fn main() {
|
|
let matches = cli().get_matches();
|
|
|
|
match matches.subcommand() {
|
|
Some(("clone", sub_matches)) => {
|
|
println!(
|
|
"Cloning {}",
|
|
sub_matches.get_one::<String>("REMOTE").expect("required")
|
|
);
|
|
}
|
|
Some(("diff", sub_matches)) => {
|
|
let mut base = sub_matches.get_one::<String>("base").map(|s| s.as_str());
|
|
let mut head = sub_matches.get_one::<String>("head").map(|s| s.as_str());
|
|
let mut path = sub_matches.get_one::<String>("path").map(|s| s.as_str());
|
|
if path.is_none() {
|
|
path = head;
|
|
head = None;
|
|
if path.is_none() {
|
|
path = base;
|
|
base = None;
|
|
}
|
|
}
|
|
let base = base.unwrap_or("stage");
|
|
let head = head.unwrap_or("worktree");
|
|
let path = path.unwrap_or("");
|
|
println!("Diffing {}..{} {}", base, head, path);
|
|
}
|
|
Some(("push", sub_matches)) => {
|
|
println!(
|
|
"Pushing to {}",
|
|
sub_matches.get_one::<String>("REMOTE").expect("required")
|
|
);
|
|
}
|
|
Some(("add", sub_matches)) => {
|
|
let paths = sub_matches
|
|
.get_many::<PathBuf>("PATH")
|
|
.into_iter()
|
|
.flatten()
|
|
.collect::<Vec<_>>();
|
|
println!("Adding {:?}", paths);
|
|
}
|
|
Some(("stash", sub_matches)) => {
|
|
let stash_command = sub_matches.subcommand().unwrap_or(("push", sub_matches));
|
|
match stash_command {
|
|
("apply", sub_matches) => {
|
|
let stash = sub_matches.get_one::<String>("STASH");
|
|
println!("Applying {:?}", stash);
|
|
}
|
|
("pop", sub_matches) => {
|
|
let stash = sub_matches.get_one::<String>("STASH");
|
|
println!("Popping {:?}", stash);
|
|
}
|
|
("push", sub_matches) => {
|
|
let message = sub_matches.get_one::<String>("message");
|
|
println!("Pushing {:?}", message);
|
|
}
|
|
(name, _) => {
|
|
unreachable!("Unsupported subcommand `{}`", name)
|
|
}
|
|
}
|
|
}
|
|
Some((ext, sub_matches)) => {
|
|
let args = sub_matches
|
|
.get_many::<OsString>("")
|
|
.into_iter()
|
|
.flatten()
|
|
.collect::<Vec<_>>();
|
|
println!("Calling out to {:?} with {:?}", ext, args);
|
|
}
|
|
_ => unreachable!(), // If all subcommands are defined above, anything else is unreachabe!()
|
|
}
|
|
|
|
// Continued program logic goes here...
|
|
}
|