如何在Rust中的structopt中使用枚举?

H. *_*ane 2 enums command-line-arguments rust structopt

我想StructOpt使用枚举,这样每次用户传递-d sunday它时都会被解析为Day::Sunday:

#[macro_use]
extern crate structopt;

use std::path::PathBuf;
use structopt::StructOpt;

// My enum
enum Day {
    Sunday, Monday
}

#[derive(Debug, StructOpt)]
#[structopt(name = "example", about = "An example of StructOpt usage.")]
struct Opt {
    /// Set speed
    #[structopt(short = "s", long = "speed", default_value = "42")]
    speed: f64,
    /// Input file
    #[structopt(parse(from_os_str))]
    input: PathBuf,
    /// Day of the week
    #[structopt(short = "d", long = "day", default_value = Day::Monday)]
    day: Day,
}

fn main() {
    let opt = Opt::from_args();
    println!("{:?}", opt);
}
Run Code Online (Sandbox Code Playgroud)

我目前最好的解决方案是使用Option<String>作为类型并传递自定义parse_day():

fn parse_day(day: &str) -> Result<Day, ParseError> {
    match day {
        "sunday" => Ok(Day::Sunday),
        _ => Ok(Day::Monday)
    }
    Err("Could not parse a day")
}
Run Code Online (Sandbox Code Playgroud)

alm*_*mel 18

@kennytm 的方法有效,但arg_enum!宏是一种更简洁的方法,如以下示例所示structopt

arg_enum! {
    #[derive(Debug)]
    enum Day {
        Sunday,
        Monday
    }
}

#[derive(StructOpt, Debug)]
struct Opt {
    /// Important argument.
    #[structopt(possible_values = &Day::variants(), case_insensitive = true)]
    i: Day,
}

fn main() {
    let opt = Opt::from_args();
    println!("{:?}", opt);
}
Run Code Online (Sandbox Code Playgroud)

这将让您将工作日解析为Sundayor sunday


ken*_*ytm 12

Struct-opt接受任何实现的类型FromStr,它离你的parse_day函数不远:

use std::str::FromStr;

impl FromStr for Day {
    type Err = ParseError;
    fn from_str(day: &str) -> Result<Self, Self::Err> {
        match day {
            "sunday" => Ok(Day::Sunday),
            "monday" => Ok(Day::Monday),
            _ => Err("Could not parse a day"),
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

另外,default_value应该是一个字符串,它将被解释为一个Day使用from_str.

#[structopt(short = "d", long = "day", default_value = "monday")]
day: Day,
Run Code Online (Sandbox Code Playgroud)

  • 至少在 Rust 1.46 中,“ParseError”看起来不存在(?)。 (3认同)

Pet*_*all 5

错误消息是:

error[E0277]: the trait bound `Day: std::str::FromStr` is not satisfied
  --> src/main.rs:22:17
   |
22 | #[derive(Debug, StructOpt)]
   |                 ^^^^^^^^^ the trait `std::str::FromStr` is not implemented for `Day`
   |
   = note: required by `std::str::FromStr::from_str`
Run Code Online (Sandbox Code Playgroud)

您可以通过实现FromStrfor Day(参见kennytm的答案)来解决这个问题,如消息所示,或者通过定义解析函数Day:

fn parse_day(src: &str) -> Result<Day, String> {
    match src {
        "sunday" => Ok(Day::Sunday),
        "monday" => Ok(Day::Monday),
        _ => Err(format!("Invalid day: {}", src))
    }
}
Run Code Online (Sandbox Code Playgroud)

并使用try_from_str属性指定它:

/// Day of the week
#[structopt(short = "d", long = "day", parse(try_from_str = "parse_day"), default_value = "monday")]
day: Day,
Run Code Online (Sandbox Code Playgroud)