//! This module contains traits that are usable with `#[derive(...)].` use crate::{App, ArgMatches, Error}; use std::ffi::OsString; /// This trait is just a convenience on top of FromArgMatches + IntoApp pub trait Clap: FromArgMatches + IntoApp + Sized { /// Parse from `std::env::args()`, exit on error fn parse() -> Self { let matches = ::into_app().get_matches(); ::from_arg_matches(&matches) } /// Parse from `std::env::args()`, return Err on error. fn try_parse() -> Result { let matches = ::into_app().try_get_matches()?; Ok(::from_arg_matches(&matches)) } /// Parse from iterator, exit on error fn parse_from(itr: I) -> Self where I: IntoIterator, // TODO (@CreepySkeleton): discover a way to avoid cloning here T: Into + Clone, { let matches = ::into_app().get_matches_from(itr); ::from_arg_matches(&matches) } /// Parse from iterator, return Err on error. fn try_parse_from(itr: I) -> Result where I: IntoIterator, // TODO (@CreepySkeleton): discover a way to avoid cloning here T: Into + Clone, { let matches = ::into_app().try_get_matches_from(itr)?; Ok(::from_arg_matches(&matches)) } } /// Build an App according to the struct /// /// Also serves for flattening pub trait IntoApp: Sized { /// @TODO @release @docs fn into_app<'b>() -> App<'b>; /// @TODO @release @docs fn augment_clap(app: App<'_>) -> App<'_>; } /// Extract values from ArgMatches into the struct. pub trait FromArgMatches: Sized { /// @TODO @release @docs fn from_arg_matches(matches: &ArgMatches) -> Self; } /// @TODO @release @docs pub trait Subcommand: Sized { /// @TODO @release @docs fn from_subcommand(name: &str, matches: Option<&ArgMatches>) -> Option; /// @TODO @release @docs fn augment_subcommands(app: App<'_>) -> App<'_>; } /// @TODO @release @docs pub trait ArgEnum {} impl Clap for Box { fn parse() -> Self { Box::new(::parse()) } fn try_parse() -> Result { ::try_parse().map(Box::new) } fn parse_from(itr: I) -> Self where I: IntoIterator, // TODO (@CreepySkeleton): discover a way to avoid cloning here It: Into + Clone, { Box::new(::parse_from(itr)) } fn try_parse_from(itr: I) -> Result where I: IntoIterator, // TODO (@CreepySkeleton): discover a way to avoid cloning here It: Into + Clone, { ::try_parse_from(itr).map(Box::new) } } impl IntoApp for Box { fn into_app<'b>() -> App<'b> { ::into_app() } fn augment_clap(app: App<'_>) -> App<'_> { ::augment_clap(app) } } impl FromArgMatches for Box { fn from_arg_matches(matches: &ArgMatches) -> Self { Box::new(::from_arg_matches(matches)) } } impl Subcommand for Box { fn from_subcommand(name: &str, matches: Option<&ArgMatches>) -> Option { ::from_subcommand(name, matches).map(Box::new) } fn augment_subcommands(app: App<'_>) -> App<'_> { ::augment_subcommands(app) } }