mirror of
https://github.com/clap-rs/clap
synced 2024-12-12 13:52:34 +00:00
db863de6c1
While we don't have a built-in mechanism, its relatively easy to support with the APIs we provide. Inspired by #3566
101 lines
3.4 KiB
Rust
101 lines
3.4 KiB
Rust
// Note: this requires the `cargo` feature
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use clap::{arg, Command};
|
|
|
|
fn cli() -> Command<'static> {
|
|
Command::new("git")
|
|
.about("A fictional versioning CLI")
|
|
.subcommand_required(true)
|
|
.arg_required_else_help(true)
|
|
.allow_external_subcommands(true)
|
|
.allow_invalid_utf8_for_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("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").allow_invalid_utf8(true)),
|
|
)
|
|
.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<'static>> {
|
|
vec![arg!(-m --message <MESSAGE>).required(false)]
|
|
}
|
|
|
|
fn main() {
|
|
let matches = cli().get_matches();
|
|
|
|
match matches.subcommand() {
|
|
Some(("clone", sub_matches)) => {
|
|
println!(
|
|
"Cloning {}",
|
|
sub_matches.value_of("REMOTE").expect("required")
|
|
);
|
|
}
|
|
Some(("push", sub_matches)) => {
|
|
println!(
|
|
"Pushing to {}",
|
|
sub_matches.value_of("REMOTE").expect("required")
|
|
);
|
|
}
|
|
Some(("add", sub_matches)) => {
|
|
let paths = sub_matches
|
|
.values_of_os("PATH")
|
|
.unwrap_or_default()
|
|
.map(PathBuf::from)
|
|
.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.value_of("STASH");
|
|
println!("Applying {:?}", stash);
|
|
}
|
|
("pop", sub_matches) => {
|
|
let stash = sub_matches.value_of("STASH");
|
|
println!("Popping {:?}", stash);
|
|
}
|
|
("push", sub_matches) => {
|
|
let message = sub_matches.value_of("message");
|
|
println!("Pushing {:?}", message);
|
|
}
|
|
(name, _) => {
|
|
unreachable!("Unsupported subcommand `{}`", name)
|
|
}
|
|
}
|
|
}
|
|
Some((ext, sub_matches)) => {
|
|
let args = sub_matches
|
|
.values_of_os("")
|
|
.unwrap_or_default()
|
|
.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...
|
|
}
|