use std::{ffi::OsString, fmt, str::FromStr}; use crate::{Error, Result}; macro_rules! format_err { ($($tt:tt)*) => { Error { msg: format!($($tt)*) } }; } macro_rules! bail { ($($tt:tt)*) => { return Err(format_err!($($tt)*)) }; } pub struct Parser { rargs: Vec, } impl Parser { pub fn new(mut args: Vec) -> Self { args.reverse(); Self { rargs: args } } pub fn new_from_env() -> Self { let mut args = std::env::args_os().collect::>(); args.reverse(); args.pop(); Self { rargs: args } } pub fn is_empty(&self) -> bool { self.rargs.is_empty() } pub fn peek_flag(&self) -> Option<&str> { self.rargs.last().and_then(|it| it.to_str()).filter(|it| it.starts_with('-')) } pub fn pop_flag(&mut self) -> Option> { if self.peek_flag().is_some() { self.next().map(|it| it.into_string()) } else { self.next().map(Err) } } pub fn push_back(&mut self, arg: Result) { let arg = match arg { Ok(it) => it.into(), Err(it) => it, }; self.rargs.push(arg) } pub fn next(&mut self) -> Option { self.rargs.pop() } pub fn next_value(&mut self, flag: &str) -> Result { if self.peek_flag().is_some() { bail!("expected a value for `{}`", flag) } self.next().ok_or_else(|| format_err!("expected a value for `{}`", flag)) } pub fn next_value_from_str(&mut self, flag: &str) -> Result where T::Err: fmt::Display, { let value = self.next_value(flag)?; self.value_from_str(flag, value) } pub fn value_from_str(&mut self, flag: &str, value: OsString) -> Result where T::Err: fmt::Display, { match value.into_string() { Ok(str) => { str.parse::().map_err(|err| format_err!("can't parse `{}`, {}", flag, err)) } Err(it) => { bail!("can't parse `{}`, invalid utf8: {:?}", flag, it) } } } pub fn unexpected_flag(&self, flag: &str) -> Error { format_err!("unexpected flag: `{}`", flag) } pub fn unexpected_arg(&self, arg: OsString) -> Error { format_err!("unexpected argument: {:?}", arg) } pub fn optional(&self, flag: &str, mut vals: Vec) -> Result> { if vals.len() > 1 { bail!("flag specified more than once: `{}`", flag) } Ok(vals.pop()) } pub fn required(&self, flag: &str, mut vals: Vec) -> Result { if vals.len() > 1 { bail!("flag specified more than once: `{}`", flag) } vals.pop().ok_or_else(|| format_err!("flag is required: `{}`", flag)) } pub fn subcommand(&self, cmd: Option) -> Result { cmd.ok_or_else(|| format_err!("subcommand is required")) } }