0

我希望能够在我的命令行工具中使用“通用”标志,该工具使用StructOpt. 也就是说,如果我有一个标志(例如--debug),无论该标志在输入中的哪个位置,我都希望它的行为相同:

$ mycli --debug alpha
$ mycli alpha --debug

alpha并且--debug在此示例中只是有用的占位符;实际的子命令和标志会有所不同。)

实际行为:

$ cargo run -- --debug alpha                                                                                               nford 14:36:32
    Finished dev [unoptimized + debuginfo] target(s) in 0.01s
     Running `target/debug/scratch-rust --debug alpha`
Debug mode enabled!
Alpha subcommand selected.
$ cargo run -- alpha --debug                                                                                               nford 14:36:27
    Finished dev [unoptimized + debuginfo] target(s) in 0.01s
     Running `target/debug/scratch-rust alpha --debug`
error: Found argument '--debug' which wasn't expected, or isn't valid in this context

USAGE:
    scratch-rust alpha

预期行为:

$ cargo run -- --debug alpha                                                                                               nford 14:36:32
    Finished dev [unoptimized + debuginfo] target(s) in 0.01s
     Running `target/debug/scratch-rust --debug alpha`
Debug mode enabled!
Alpha subcommand selected.
$ cargo run -- alpha --debug       
    Finished dev [unoptimized + debuginfo] target(s) in 0.01s
     Running `target/debug/scratch-rust --debug alpha`
Debug mode enabled!
Alpha subcommand selected.

这是我的main.rs代码:

use std::str::FromStr;
use structopt::StructOpt;
use std::io::{Error, ErrorKind};

#[derive(Debug, StructOpt)]
/// Specific task being executed; one task might imply others, depending on flags.
pub enum Subcommand {
    Alpha,
    Omega,
}

impl FromStr for Subcommand {
    type Err = Error;
    fn from_str(subcommand_input: &str) -> Result<Self, Self::Err> {
        match subcommand_input {
            "alpha" => Ok(Subcommand::Alpha),
            "omega" => Ok(Subcommand::Omega),
            _ => Err(Error::new(ErrorKind::Other, "Unrecognized subcommand.")),
        }
    }
}

#[derive(Debug, StructOpt)]
#[structopt(
    name = "Minimal Example CLI",
    about = "A minimal example of multi-location command line inputs.",
)]
pub struct CLIOpts {
    /// Set logging to verbose mode.
    // short and long flags (-d, --debug) will be deduced from the field's name
    #[structopt(short, long)]
    pub debug: bool,

    #[structopt(subcommand)]
    pub cmd: Subcommand,
}


fn main() {
    let args = CLIOpts::from_args();
    if args.debug {
        println!("Debug mode enabled!");
    }
    match args.cmd {
        Subcommand::Alpha => println!("Alpha subcommand selected."),
        Subcommand::Omega => println!("Omega subcommand selected."),
    }
}

这是我的文件(上面的示例在没有依赖Cargo.toml项的情况下不起作用):structopt

[package]
name = "scratch-rust"
version = "0.1.0"
edition = "2018"

[dependencies]
structopt = "*"

有没有办法在不复制子命令每个级别的“通用”标志的情况下做到这一点?

4

1 回答 1

1

Clap 有一个global()参数方法,可以满足你的要求。您可以使用原始属性通过 Structopt 访问此方法:

pub struct CLIOpts {
    /// Set logging to verbose mode.
    // short and long flags (-d, --debug) will be deduced from the field's name
    #[structopt(short, long, global = true)]
    pub debug: bool,

    #[structopt(subcommand)]
    pub cmd: Subcommand,
}

(注意为 增加的global属性debug

该应用程序现在可以按照您希望的方式工作:

$ cargo run -- --debug alpha
    Finished dev [unoptimized + debuginfo] target(s) in 0.02s
     Running `target/debug/scratch-rust --debug alpha`
Debug mode enabled!
Alpha subcommand selected.
$ cargo run -- alpha --debug
    Finished dev [unoptimized + debuginfo] target(s) in 0.02s
     Running `target/debug/scratch-rust alpha --debug`
Debug mode enabled!
Alpha subcommand selected.
于 2021-07-05T12:30:45.990 回答