argwerk/
lib.rs

1//! [<img alt="github" src="https://img.shields.io/badge/github-udoprog/argwerk-8da0cb?style=for-the-badge&logo=github" height="20">](https://github.com/udoprog/argwerk)
2//! [<img alt="crates.io" src="https://img.shields.io/crates/v/argwerk.svg?style=for-the-badge&color=fc8d62&logo=rust" height="20">](https://crates.io/crates/argwerk)
3//! [<img alt="docs.rs" src="https://img.shields.io/badge/docs.rs-argwerk-66c2a5?style=for-the-badge&logoColor=white&logo=" height="20">](https://docs.rs/argwerk)
4//!
5//! Define a simple command-line parser through a declarative macro.
6//!
7//! This is **not** intended to be a complete command-line parser library.
8//! Instead this can be used as an alternative quick-and-dirty approach that can
9//! be cheaply incorporated into a tool.
10//!
11//! For a more complete command-line parsing library, use [clap].
12//!
13//! We provide:
14//! * A dependency-free command-line parsing framework using declarative macros.
15//! * A flexible mechanism for parsing.
16//! * Formatting of decent looking help messages.
17//!
18//! We *do not* provide:
19//! * As-close-to correct line wrapping with wide unicode characters as possible
20//!   (see [textwrap]).
21//! * Built-in complex command structures like subcommands (see the
22//!   [subcommands] example for how this can be accomplished).
23//!
24//! For how to use, see the documentation of [argwerk::define] and
25//! [argwerk::args].
26//!
27//! <br>
28//!
29//! ## Usage
30//!
31//! Initially when you're adding arguments to your program you can use
32//! [argwerk::args]. This allows for easily parsing out a handful of optional
33//! parameters.
34//!
35//! > This example is available as `simple`:
36//! > ```sh
37//! > cargo run --example simple -- --limit 20
38//! > ```
39//!
40//! ```rust
41//! # fn main() -> Result<(), argwerk::Error> {
42//! let args = argwerk::args! {
43//!     /// A simple tool.
44//!     "tool [-h]" {
45//!         help: bool,
46//!         limit: usize = 10,
47//!     }
48//!     /// The limit of the operation. (default: 10).
49//!     ["-l" | "--limit", int] => {
50//!         limit = str::parse(&int)?;
51//!     }
52//!     /// Print this help.
53//!     ["-h" | "--help"] => {
54//!         println!("{}", HELP);
55//!         help = true;
56//!     }
57//! }?;
58//!
59//! if args.help {
60//!     return Ok(());
61//! }
62//!
63//! dbg!(args);
64//! # Ok(()) }
65//! ```
66//!
67//! After a while you might want to graduate to defining a *named* struct
68//! containing the arguments. This can be useful if you want to pass the
69//! arguments around.
70//!
71//! > This example is available as `tour`:
72//! > ```sh
73//! > cargo run --example tour -- --help
74//! > ```
75//!
76//! ```rust
77//! use std::ffi::OsString;
78//!
79//! argwerk::define! {
80//!     /// A command touring the capabilities of argwerk.
81//!     #[derive(Default)]
82//!     #[usage = "tour [-h]"]
83//!     struct Args {
84//!         help: bool,
85//!         #[required = "--file must be specified"]
86//!         file: String,
87//!         input: Option<String>,
88//!         limit: usize = 10,
89//!         positional: Option<(String, Option<String>)>,
90//!         raw: Option<OsString>,
91//!         rest: Vec<String>,
92//!     }
93//!     /// Prints the help.
94//!     ///
95//!     /// This includes:
96//!     ///    * All the available switches.
97//!     ///    * All the available positional arguments.
98//!     ///    * Whatever else the developer decided to put in here! We even support wrapping comments which are overly long.
99//!     ["-h" | "--help"] => {
100//!         println!("{}", Args::help());
101//!         help = true;
102//!     }
103//!     /// Limit the number of things by <n> (default: 10).
104//!     ["--limit" | "-l", n] => {
105//!         limit = str::parse(&n)?;
106//!     }
107//!     /// Write to the file specified by <path>.
108//!     ["--file", path] if !file.is_some() => {
109//!         file = Some(path);
110//!     }
111//!     /// Read from the specified input.
112//!     ["--input", #[option] path] => {
113//!         input = path;
114//!     }
115//!     /// A really long argument that exceeds usage limit and forces the documentation to wrap around with newlines.
116//!     ["--really-really-really-long-argument", thing] => {
117//!     }
118//!     /// A raw argument that passes whatever was passed in from the operating system.
119//!     ["--raw", #[os] arg] => {
120//!         raw = Some(arg);
121//!     }
122//!     /// Takes argument at <foo> and <bar>.
123//!     ///
124//!     ///    * This is an indented message. The first alphanumeric character determines the indentation to use.
125//!     [foo, #[option] bar, #[rest] args] if positional.is_none() => {
126//!         positional = Some((foo, bar));
127//!         rest = args;
128//!     }
129//! }
130//!
131//! # fn main() -> anyhow::Result<()> {
132//! // Note: we're using `parse` here instead of `args` since it works better
133//! // with the example.
134//! let args = Args::parse(vec!["--file", "foo.txt", "--input", "-"])?;
135//!
136//! dbg!(args);
137//! # Ok(()) }
138//! ```
139//!
140//! <br>
141//!
142//! ## Time and size compared to other projects
143//!
144//! argwerk aims to be a lightweight dependency that is fast to compile. This is
145//! how it stacks up to other projects in that regard.
146//!
147//! The following summary was generated from the [projects found here].
148//!
149//! | project    | cold build (release) | rebuild* (release) | size (release) |
150//! |------------|----------------------|--------------------|----------------|
151//! | argh** | 5.142723s (4.849361s) | 416.9594ms (468.7003ms) | 297k (180k) |
152//! | argwerk | 1.443709s (1.2971457s) | 403.0641ms (514.036ms) | 265k (185k) |
153//! | clap*** | 11.9863223s (13.1338799s) | 551.407ms (807.8939ms) | 2188k (750k) |
154//! > *: rebuild was triggered by adding a single newline to `main.rs`.<br>
155//! > **: argh `0.1.4` including 11 dependencies.<br>
156//! > ***: clap `3.0.0-beta.2` including 32 dependencies.<br>
157//!
158//! You can try and build it yourself with:
159//!
160//! ```sh
161//! cargo run --manifest-path tools/builder/Cargo.toml
162//! ```
163//!
164//! [projects found here]: https://github.com/udoprog/argwerk/tree/main/projects
165//! [argwerk::define]: https://docs.rs/argwerk/0/argwerk/macro.define.html
166//! [argwerk::args]: https://docs.rs/argwerk/0/argwerk/macro.args.html
167//! [clap]: https://docs.rs/clap
168//! [ok_or_else]: https://doc.rust-lang.org/std/option/enum.Option.html#method.ok_or_else
169//! [OsString]: https://doc.rust-lang.org/std/ffi/struct.OsString.html
170//! [textwrap]: https://docs.rs/textwrap/0.13.2/textwrap/#displayed-width-vs-byte-size
171//! [subcommands]: https://github.com/udoprog/argwerk/blob/main/examples/subcommands.rs
172
173#![deny(missing_docs)]
174
175use std::fmt;
176
177#[doc(hidden)]
178/// Macro helpers. Not intended for public use!
179pub mod helpers;
180
181use std::error;
182
183pub use self::helpers::{Help, HelpFormat, InputError, Switch, TryIntoInput};
184
185/// An error raised by argwerk.
186#[derive(Debug)]
187pub struct Error {
188    kind: Box<ErrorKind>,
189}
190
191impl Error {
192    /// Construct a new error with the given kind.
193    pub fn new(kind: ErrorKind) -> Self {
194        Self {
195            kind: Box::new(kind),
196        }
197    }
198
199    /// Access the underlying error kind.
200    pub fn kind(&self) -> &ErrorKind {
201        &self.kind
202    }
203}
204
205impl fmt::Display for Error {
206    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
207        match self.kind.as_ref() {
208            ErrorKind::UnsupportedArgument { argument } => {
209                write!(f, "unsupported argument `{}`", argument)
210            }
211            ErrorKind::UnsupportedSwitch { switch } => {
212                write!(f, "unsupported switch `{}`", switch)
213            }
214            ErrorKind::MissingSwitchArgument { switch, argument } => {
215                write!(f, "switch `{}` missing argument `{}`", switch, argument,)
216            }
217            ErrorKind::MissingPositional { name } => {
218                write!(f, "missing argument `{}`", name)
219            }
220            ErrorKind::MissingRequired { name, reason } => match reason {
221                Some(reason) => write!(f, "missing required argument: {}", reason),
222                None => write!(f, "missing required argument `{}`", name),
223            },
224            ErrorKind::InputError { error } => {
225                write!(f, "{}", error)
226            }
227            ErrorKind::Error { name, error } => {
228                write!(f, "error in argument `{}`: {}", name, error)
229            }
230        }
231    }
232}
233
234impl error::Error for Error {
235    fn source(&self) -> Option<&(dyn error::Error + 'static)> {
236        match self.kind.as_ref() {
237            ErrorKind::Error { error, .. } => Some(error.as_ref()),
238            _ => None,
239        }
240    }
241}
242
243impl From<crate::helpers::InputError> for Error {
244    fn from(error: crate::helpers::InputError) -> Self {
245        Error::new(ErrorKind::InputError { error })
246    }
247}
248
249/// The kind of an error.
250#[derive(Debug)]
251pub enum ErrorKind {
252    /// Encountered an argument that was not supported.
253    ///
254    /// An unsupported argument is triggered when none of the branches in the
255    /// parser matches the current agument.
256    ///
257    /// # Examples
258    ///
259    /// ```rust
260    /// argwerk::define! {
261    ///     struct Args { }
262    ///     // This errors because `bar` is not a supported switch, nor do we
263    ///     // match any positional arguments.
264    ///     ["--file", arg] => {}
265    /// }
266    ///
267    /// # fn main() -> Result<(), argwerk::Error> {
268    /// let error = Args::parse(vec!["bar"]).unwrap_err();
269    ///
270    /// assert!(matches!(error.kind(), argwerk::ErrorKind::UnsupportedArgument { .. }));
271    /// # Ok(()) }
272    /// ```
273    UnsupportedArgument {
274        /// The name of the unsupported argument.
275        argument: Box<str>,
276    },
277    /// Encountered a switch that was not supported.
278    ///
279    /// An unsupported switch is caused by the same reason as an unsupported
280    /// argument, but it's prefixed with a hyphen `-`.
281    ///
282    /// # Examples
283    ///
284    /// ```rust
285    /// argwerk::define! {
286    ///     #[usage = "command [-h]"]
287    ///     struct Args { }
288    ///     // This errors because `--path` is not a supported switch. But
289    ///     // `"--file"` is.
290    ///     ["--file", arg] => {}
291    /// }
292    ///
293    /// # fn main() -> Result<(), argwerk::Error> {
294    /// let error = Args::parse(vec!["--path"]).unwrap_err();
295    ///
296    /// assert!(matches!(error.kind(), argwerk::ErrorKind::UnsupportedSwitch { .. }));
297    /// # Ok(()) }
298    /// ```
299    UnsupportedSwitch {
300        /// The name of the unsupported switch.
301        switch: Box<str>,
302    },
303    /// When a parameter to an argument is missing.
304    ///
305    /// # Examples
306    ///
307    /// ```rust
308    /// argwerk::define! {
309    ///     struct Args { }
310    ///     // This errors because `--file` requires an argument `path`, but
311    ///     // that is not provided.
312    ///     ["--file", path] => {}
313    /// }
314    ///
315    /// # fn main() -> Result<(), argwerk::Error> {
316    /// let error = Args::parse(vec!["--file"]).unwrap_err();
317    ///
318    /// assert!(matches!(error.kind(), argwerk::ErrorKind::MissingSwitchArgument { .. }));
319    /// # Ok(()) }
320    /// ```
321    MissingSwitchArgument {
322        /// The switch where the argument was missing, like `--file` in `--file
323        /// <path>`.
324        switch: Box<str>,
325        /// The argument that was missing, like `path` in `--file <path>`.
326        argument: &'static str,
327    },
328    /// When a positional argument is missing.
329    ///
330    /// # Examples
331    ///
332    /// ```rust
333    /// argwerk::define! {
334    ///     struct Args { }
335    ///     // This errors because `b` is a required argument, but we only have
336    ///     // one which matches `a`.
337    ///     [a, b] => {}
338    /// }
339    ///
340    /// # fn main() -> Result<(), argwerk::Error> {
341    /// let error = Args::parse(vec!["foo"]).unwrap_err();
342    ///
343    /// assert!(matches!(error.kind(), argwerk::ErrorKind::MissingPositional { .. }));
344    /// # Ok(()) }
345    /// ```
346    MissingPositional {
347        /// The name of the argument missing like `path` in `<path>`.
348        name: &'static str,
349    },
350    /// When a positional argument is missing.
351    ///
352    /// # Examples
353    ///
354    /// ```rust
355    /// argwerk::define! {
356    ///     struct Args {
357    ///         #[required = "--name must be used"]
358    ///         name: String,
359    ///     }
360    ///     ["--name", n] => {
361    ///         name = Some(n);
362    ///     }
363    ///     [rest] => {}
364    /// }
365    ///
366    /// # fn main() -> Result<(), argwerk::Error> {
367    /// let error = Args::parse(vec!["foo"]).unwrap_err();
368    ///
369    /// assert!(matches!(error.kind(), argwerk::ErrorKind::MissingRequired { name: "name", .. }));
370    /// # Ok(()) }
371    /// ```
372    MissingRequired {
373        /// The name of the required variable that is missing.
374        name: &'static str,
375        /// The reason that the required argument was missing.
376        reason: Option<&'static str>,
377    },
378    /// Failed to parse input as unicode string.
379    ///
380    /// This is raised in case argwerk needs to treat an input as a string, but
381    /// that is not possible.
382    ///
383    /// This is required if the string needs to be used in a [switch
384    /// branch][define#parsing-switches-likes---help].
385    InputError {
386        /// The underlying error.
387        error: crate::helpers::InputError,
388    },
389    /// When an error has been raised while processing an argument, typically
390    /// when something is being parsed.
391    ///
392    /// # Examples
393    ///
394    /// ```rust
395    /// argwerk::define! {
396    ///     #[usage = "command [-h]"]
397    ///     struct Args { }
398    ///     // This errors because we raise an error in the branch body.
399    ///     ["foo"] => {
400    ///         Err("something went wrong")
401    ///     }
402    /// }
403    ///
404    /// # fn main() -> Result<(), argwerk::Error> {
405    /// let error = Args::parse(vec!["foo"]).unwrap_err();
406    ///
407    /// assert!(matches!(error.kind(), argwerk::ErrorKind::Error { .. }));
408    /// # Ok(()) }
409    /// ```
410    Error {
411        /// The name of the switch or positional that couldn't be processed.
412        name: Box<str>,
413        /// The error that caused the parsing error.
414        error: Box<dyn error::Error + Send + Sync + 'static>,
415    },
416}
417
418/// Parse command-line arguments.
419///
420/// This will generate an anonymous structure containing the arguments defined
421/// which is returned by the macro.
422///
423/// Each branch is executed when an incoming argument matches and must return a
424/// [Result], like `Ok(())`. Error raised in the branch will cause a
425/// [ErrorKind::Error] error to be raised associated with that argument
426/// with the relevant error attached.
427///
428/// ## The generated arguments structure
429///
430/// The first part of the macro defines the state available to the parser. These
431/// are field-like declarations which can specify a default initialization
432/// value. Fields which do not specify a value will be initialized using
433/// [Default::default]. This is the only required component of the macro.
434///
435/// The macro produces an arguments struct with fields matching this
436/// declaration. This can be used to conveniently group and access data
437/// populated during argument parsing.
438///
439/// You can use arbitrary attributes for the struct.
440/// Note that [`std::fmt::Debug`] will be automatically derived.
441///
442/// ```rust
443/// argwerk::define! {
444///     /// A simple test command.
445///     #[usage = "command [-h]"]
446///     #[derive(Default)]
447///     struct Args {
448///         help: bool,
449///         limit: usize = 10,
450///     }
451///     /// Print this help.
452///     ["-h" | "--help"] => {
453///         help = true;
454///     }
455///     /// Specify a limit (default: 10).
456///     ["--limit", n] => {
457///         limit = str::parse(&n)?;
458///     }
459/// }
460///
461/// # fn main() -> Result<(), argwerk::Error> {
462/// let args = Args::parse(["--limit", "20"].iter().copied())?;
463///
464/// if args.help {
465///     println!("{}", Args::help());
466/// }
467///
468/// assert_eq!(args.help, false);
469/// assert_eq!(args.limit, 20);
470/// # Ok(()) }
471/// ```
472///
473/// This structure also has two associated functions which can be used to parse
474/// input:
475///
476/// * `args` - Which parses OS arguments using [std::env::args_os].
477/// * `parse` - Which can be provided with a custom iterator. This is what's
478///   used in almost all the examples.
479///
480/// When using the custom parse function each item produced by the passed in
481/// iterator must implement [TryIntoInput]. This is implemented by types such
482/// as: `&str`, `String`, `OsString` and `&OsStr`.
483///
484/// ```rust
485/// argwerk::define! {
486///     /// A simple test command.
487///     #[usage = "command [-h]"]
488///     struct Args {
489///         help: bool,
490///         limit: usize = 10,
491///         positional: Option<(String, String, String)>,
492///     }
493///     /// Print this help.
494///     ["-h" | "--help"] => {
495///         help = true;
496///     }
497///     [a, b, c] => {
498///         positional = Some((a, b, c));
499///     }
500/// }
501///
502/// # fn main() -> Result<(), argwerk::Error> {
503/// let args = Args::args()?;
504///
505/// if args.help {
506///     println!("{}", Args::help());
507/// }
508///
509/// let args = Args::parse(vec!["foo", "bar", "baz"])?;
510///
511/// assert_eq!(args.positional, Some((String::from("foo"), String::from("bar"), String::from("baz"))));
512/// # Ok(()) }
513/// ```
514///
515/// ## Parsing switches likes `--help`
516///
517/// The basic form of an argument branch one which matches on a string literal.
518/// The string literal (e.g. `"--help"`) will then be treated as the switch for
519/// the branch. You can specify multiple matches for each branch by separating
520/// them with a pipe (`|`).
521///
522/// It's not necessary that switches start with `-`, but this is assumed for
523/// convenience. In particular, argwerk will treat any arguments starting with a
524/// hyphen as "switch-like". This is used to determine whether an argument is
525/// present if its optional (see later section).
526///
527/// ```rust
528/// argwerk::define! {
529///     #[usage = "command [-h]"]
530///     struct Args {
531///         help: bool
532///     }
533///     ["-h" | "--help"] => {
534///         help = true;
535///     }
536/// }
537///
538/// # fn main() -> Result<(), argwerk::Error> {
539/// let args = Args::parse(vec!["-h"])?;
540///
541/// if args.help {
542///     println!("{}", Args::help());
543/// }
544///
545/// assert_eq!(args.help, true);
546/// # Ok(()) }
547/// ```
548///
549/// ## Parsing positional arguments
550///
551/// Positional arguments are parsed by specifying a vector of bindings in a
552/// branch. Like `[foo, bar]`.
553///
554/// The following is a basic example. Two arguments `foo` and `bar` are required
555/// if the branch matches. If there is no such input an
556/// [ErrorKind::MissingPositional] error will be raised.
557///
558/// ```rust
559/// argwerk::define! {
560///     #[usage = "command [-h]"]
561///     struct Args {
562///         positional: Option<(String, String)>,
563///     }
564///     [foo, bar] if positional.is_none() => {
565///         positional = Some((foo, bar));
566///     }
567/// }
568///
569/// # fn main() -> Result<(), argwerk::Error> {
570/// let args = Args::parse(["a", "b"].iter().copied())?;
571///
572/// assert_eq!(args.positional, Some((String::from("a"), String::from("b"))));
573/// # Ok(()) }
574/// ```
575///
576/// ## Help documentation
577///
578/// You specify documentation for switches and arguments using doc comments
579/// (e.g. `/// Hello World`). These are automatically wrapped to 80 characters.
580///
581/// Documentation can be formatted with the `help` associated function, which
582/// returns a static instance of [Help]. This is also available as the `HELP`
583/// static variable inside of match branches. Help formatting can be further
584/// customized using [Help::format].
585///
586/// ```rust
587/// argwerk::define! {
588///     /// A simple test command.
589///     #[usage = "command [-h]"]
590///     struct Args {
591///         help2: bool,
592///     }
593///     /// Prints the help.
594///     ///
595///     /// This includes:
596///     ///    * All the available switches.
597///     ///    * All the available positional arguments.
598///     ///    * Whatever else the developer decided to put in here! We even support wrapping comments which are overly long.
599///     ["-h" | "--help"] => {
600///         println!("{}", HELP.format().width(120));
601///     }
602///     ["--help2"] => {
603///         help2 = true;
604///     }
605/// }
606///
607/// # fn main() -> Result<(), argwerk::Error> {
608/// let args = Args::args()?;
609///
610/// // Another way to access and format help documentation.
611/// if args.help2 {
612///     println!("{}", Args::help().format().width(120));
613/// }
614///
615/// # Ok(()) }
616/// ```
617///
618/// Invoking this with `-h` would print:
619///
620/// ```text
621/// Usage: command [-h]
622/// A simple test command.
623///
624/// This is nice!
625///
626/// Options:
627///   -h, --help  Prints the help.
628///
629///               This includes:
630///                  * All the available switches.
631///                  * All the available positional arguments.
632///                  * Whatever else the developer decided to put in here! We even
633///                    support wrapping comments which are overly long.
634/// ```
635///
636/// We determine the initial indentation level from the first doc comment.
637/// Looking at the code above, this would be the line containing `Prints the
638/// help.`. We then wrap additional lines relative to this level of indentation.
639///
640/// We also determine the individual indentation level of a line by looking at
641/// all the non-alphanumerical character that prefixes that line. That's why the
642/// "overly long" markdown list bullet above wraps correctly. Instead of
643/// wrapping at the `*`, it wraps to the first alphanumeric character after it.
644///
645/// ## Required arguments using `#[required]`
646///
647/// You can specify required arguments using the `#[required]` attribute in the
648/// field specification. Fields which are marked as `#[required]` have the type
649/// [Option\<T\>][Option]. If the field is left as uninitialized (`None`) once
650/// all arguments have been parsed will cause an error to be raised. See
651/// [ErrorKind::MissingRequired].
652///
653/// A reason that the argument is required can be optionally provided by doing
654/// `#[required = "--name is required"]`.
655///
656/// # Examples
657///
658/// ```rust
659/// argwerk::define! {
660///     struct Args {
661///         #[required = "--name must be used"]
662///         name: String,
663///     }
664///     ["--name", n] => {
665///         name = Some(n);
666///     }
667/// }
668///
669/// # fn main() -> Result<(), argwerk::Error> {
670/// let args = Args::parse(vec!["--name", "John"])?;
671/// assert_eq!(args.name, "John");
672/// # Ok(()) }
673/// ```
674///
675/// ## Raw os arguments with `#[os]`
676///
677/// In argwerk you can specify that a branch takes a raw argument using the
678/// `#[os]` attribute. This value will then be an
679/// [OsString][::std::ffi::OsString] and represents exactly what was fed to your
680/// program from the operating system.
681///
682/// ```rust
683/// use std::ffi::OsString;
684///
685/// argwerk::define! {
686///     /// A simple test command.
687///     #[usage = "command [-h]"]
688///     struct Args {
689///         raw: Option<OsString>,
690///     }
691///     ["--raw", #[os] arg] => {
692///         raw = Some(arg);
693///     }
694/// }
695///
696/// # fn main() -> Result<(), argwerk::Error> {
697/// let args = Args::parse(vec![OsString::from("--raw"), OsString::from("baz")])?;
698///
699/// assert!(args.raw.is_some());
700/// # Ok(()) }
701/// ```
702///
703/// ## Capture all available arguments using `#[rest]`
704///
705/// You can write a branch that receives all available arguments using the
706/// `#[rest]` attribute. This can be done both with arguments to switches, and
707/// positional arguments.
708///
709/// You can get the rest of the arguments in their raw form using `#[rest(os)]`.
710///
711/// The following showcases capturing using a positional argument:
712///
713/// ```rust
714/// argwerk::define! {
715///     /// A simple test command.
716///     #[usage = "command [-h]"]
717///     struct Args {
718///         rest: Vec<String>,
719///     }
720///     [#[rest] args] => {
721///         rest = args;
722///     }
723/// }
724///
725/// # fn main() -> Result<(), argwerk::Error> {
726/// let args = Args::parse(["foo", "bar", "baz"].iter().copied())?;
727///
728/// assert_eq!(args.rest, &["foo", "bar", "baz"]);
729/// # Ok(()) }
730/// ```
731///
732/// And the following through a switch:
733///
734/// ```rust
735/// argwerk::define! {
736///     #[usage = "command [-h]"]
737///     struct Args {
738///         rest: Vec<String>,
739///     }
740///     ["--test", #[rest] args] => {
741///         rest = args;
742///     }
743/// }
744///
745/// # fn main() -> Result<(), argwerk::Error> {
746/// let args = Args::parse(["--test", "foo", "bar", "baz"].iter().copied())?;
747///
748/// assert_eq!(args.rest, &["foo", "bar", "baz"]);
749/// # Ok(()) }
750/// ```
751///
752/// This showcases getting raw os arguments using `#[rest(os)]`:
753///
754/// ```rust
755/// use std::ffi::{OsString, OsStr};
756///
757/// argwerk::define! {
758///     /// A simple test command.
759///     #[usage = "command [-h]"]
760///     struct Args {
761///         rest: Vec<OsString>,
762///     }
763///     [#[rest(os)] args] => {
764///         rest = args;
765///     }
766/// }
767///
768/// # fn main() -> Result<(), argwerk::Error> {
769/// let args = Args::parse(["foo", "bar", "baz"].iter().copied())?;
770///
771/// assert_eq!(args.rest, &[OsStr::new("foo"), OsStr::new("bar"), OsStr::new("baz")]);
772/// # Ok(()) }
773/// ```
774///
775/// ## Parsing optional arguments with `#[option]`
776///
777/// Switches and positional arguments can be marked with the `#[option]`
778/// attribute. This will cause the argument to take a value of type
779/// `Option<I::Item>` where `I` represents the iterator that is being parsed.
780///
781/// You can get an optional argument in its raw form using `#[option(os)]`.
782///
783/// An optional argument parses to `None` if:
784/// * There are no more arguments to parse.
785/// * The argument is "switch-like" (starts with `-`).
786///
787/// ```rust
788/// use std::ffi::{OsString, OsStr};
789///
790/// argwerk::define! {
791///     /// A simple test command.
792///     #[usage = "command [-h]"]
793///     struct Args {
794///         foo: Option<String>,
795///         bar: bool,
796///         baz: Option<OsString>,
797///     }
798///     /// A switch taking an optional argument.
799///     ["--foo", #[option] arg] => {
800///         foo = arg;
801///     }
802///     ["--bar"] => {
803///         bar = true;
804///     }
805///     /// A switch taking an optional raw argument.
806///     ["--baz", #[option(os)] arg] => {
807///         baz = arg;
808///     }
809/// }
810///
811/// # fn main() -> Result<(), argwerk::Error> {
812/// // Argument exists, but looks like a switch.
813/// let args = Args::parse(["--foo", "--bar"].iter().copied())?;
814/// assert_eq!(args.foo.as_deref(), None);
815/// assert!(args.bar);
816///
817/// // Argument does not exist.
818/// let args = Args::parse(["--foo"].iter().copied())?;
819/// assert_eq!(args.foo.as_deref(), None);
820/// assert!(!args.bar);
821///
822/// let args = Args::parse(["--foo", "bar"].iter().copied())?;
823/// assert_eq!(args.foo.as_deref(), Some("bar"));
824/// assert!(!args.bar);
825///
826/// let args = Args::parse(["--baz"].iter().copied())?;
827/// assert_eq!(args.baz.as_deref(), None);
828/// assert!(!args.bar);
829///
830/// let args = Args::parse(["--baz", "bar"].iter().copied())?;
831/// assert_eq!(args.baz.as_deref(), Some(OsStr::new("bar")));
832/// assert!(!args.bar);
833/// # Ok(()) }
834/// ```
835#[macro_export]
836macro_rules! define {
837    (
838        $(#$attr:tt)*
839        $vis:vis struct $name:ident { $($body:tt)* }
840        $($config:tt)*
841    ) => {
842        $crate::__impl! {
843            $(#$attr)*
844            $vis struct $name { $($body)* }
845            $($config)*
846        }
847
848        impl $name {
849            /// Return a formatter that formats to the help string at 80
850            /// characters witdth of this argument structure.
851            $vis fn help() -> &'static $crate::Help {
852                &Self::HELP
853            }
854        }
855    };
856}
857
858/// Works the same as [define], but immediately parses arguments from
859/// [std::env::args] in place.
860///
861/// # Examples
862///
863/// ```rust
864/// # fn main() -> Result<(), argwerk::Error> {
865/// let args = argwerk::args! {
866///     /// A simple test command.
867///     "command [-h]" {
868///         help: bool,
869///         limit: usize = 10,
870///     }
871///     /// Print this help.
872///     ["-h" | "--help"] => {
873///         help = true;
874///     }
875/// }?;
876///
877/// if args.help {
878///     println!("{}", args.help());
879/// }
880/// # Ok(()) }
881/// ```
882#[macro_export]
883macro_rules! args {
884    (
885        $(#[doc = $doc:literal])*
886        $usage:literal { $($body:tt)* }
887        $($config:tt)*
888    ) => {{
889        $crate::__impl! {
890            $(#[doc = $doc])*
891            #[usage = $usage]
892            struct Args { $($body)* }
893            $($config)*
894        };
895
896        impl Args {
897            /// Return a formatter that formats to the help string at 80
898            /// characters witdth of this argument structure.
899            fn help(&self) -> &'static $crate::Help {
900                &Self::HELP
901            }
902        }
903
904        Args::args()
905    }};
906}
907
908/// Filter for docstrings.
909#[doc(hidden)]
910#[macro_export]
911macro_rules! __filter_asg_doc {
912    (#[doc = $d:literal] $(#$other:tt)* $($l:literal)*) => {
913        $crate::__filter_asg_doc!($(#$other)* $($l)* $d)
914    };
915    (#$_:tt $(#$other:tt)* $($l:literal)*) => {
916        $crate::__filter_asg_doc!($(#$other)* $($l)*)
917    };
918    ($($l:literal)*) => {
919        &[$($l),*]
920    };
921}
922
923/// Filter for usage. XXX Stops at first found.
924#[doc(hidden)]
925#[macro_export]
926macro_rules! __filter_asg_usage {
927    ($name:ident, #[usage = $usage:literal] $(#$_:tt)*) => {
928        $usage
929    };
930    ($name:ident, #$_:tt $(#$other:tt)*) => {
931        $crate::__filter_asg_usage!($name, $(#$other)*)
932    };
933    ($name:ident,) => {
934        stringify!($name)
935    };
936}
937
938/// Filter out the stuff we don't want to process ourself.
939#[doc(hidden)]
940#[macro_export]
941macro_rules! __filter_asg_other {
942    (#[doc = $_:literal] $(#$other:tt)*, $(#$done:tt)* $v:vis struct $i:ident $body:tt) => {
943        $crate::__filter_asg_other!{
944            $(#$other)*,
945            $(#$done)*
946            $v struct $i $body
947        }
948    };
949    (#[usage = $_:literal] $(#$other:tt)*, $(#$done:tt)* $v:vis struct $i:ident $body:tt) => {
950        $crate::__filter_asg_other!{
951            $(#$other)*,
952            $(#$done)*
953            $v struct $i $body
954        }
955    };
956    (#$attr:tt $(#$other:tt)*, $(#$done:tt)* $v:vis struct $i:ident $body:tt) => {
957        $crate::__filter_asg_other!{
958            $(#$other)*,
959            $(#$done)*
960            #$attr
961            $v struct $i $body
962        }
963    };
964    (, $(#$done:tt)* $v:vis struct $i:ident $body:tt) => {
965        $(#$done)*
966        $v struct $i $body
967    }
968}
969
970/// Internal implementation details of the [args] macro.
971#[doc(hidden)]
972#[macro_export]
973macro_rules! __impl {
974    // The guts of the parser.
975    (
976        $(#$attr:tt)*
977        $vis:vis struct $name:ident {
978            $( $(#$field_m:tt)* $fvis:vis $field:ident : $ty:ty $(= $expr:expr)? ),* $(,)?
979        }
980        $($config:tt)*
981    ) => {
982        $crate::__filter_asg_other! {
983            $(#$attr)*
984            #[derive(Debug)],
985            $vis struct $name { $($fvis $field: $ty,)* }
986        }
987
988        impl $name {
989            pub const HELP: $crate::Help = $crate::Help {
990                usage: $crate::__filter_asg_usage!($name, $(#$attr)*),
991                docs: $crate::__filter_asg_doc!($(#$attr)*),
992                switches: $crate::__impl!(@switches $($config)*)
993            };
994
995            /// Parse [std::env::args_os].
996            // XXX Clippy gets unaccountably worked up here.
997            #[allow(clippy::self_named_constructors)]
998            $vis fn args() -> Result<Self, $crate::Error> {
999                let mut it = std::env::args_os();
1000                it.next();
1001                Self::parse(it)
1002            }
1003
1004            /// Parse a custom iterator.
1005            $vis fn parse<I>(it: I) -> Result<Self, $crate::Error>
1006            where
1007                I: IntoIterator,
1008                I::Item: $crate::TryIntoInput,
1009            {
1010                static HELP: &$crate::Help = &$name::HELP;
1011
1012                let mut it = $crate::helpers::Input::new(it.into_iter());
1013                $($crate::__impl!(@init $(#$field_m)* $field, $ty $(, $expr)*);)*
1014
1015                while let Some(__argwerk_item) = it.next()? {
1016                    $crate::__impl!(@branches __argwerk_item, it, $($config)*);
1017                }
1018
1019                Ok(Self {
1020                    $($field: $crate::__impl!(@assign $(#$field_m)* $field)),*
1021                })
1022            }
1023        }
1024    };
1025
1026    // Argument formatting.
1027    (@doc #[rest $($tt:tt)*] $argument:ident) => { concat!("<", stringify!($argument), "..>") };
1028    (@doc #[option $($tt:tt)*] $argument:ident) => { concat!("[", stringify!($argument), "]") };
1029    (@doc #[os] $argument:ident) => { concat!("<", stringify!($argument), ">") };
1030    (@doc $argument:ident) => { concat!("<", stringify!($argument), ">") };
1031
1032    (@init $field:ident, $ty:ty) => {
1033        let mut $field: $ty = Default::default();
1034    };
1035
1036    (@init #[required $(= $reason:literal)?] $field:ident, $ty:ty) => {
1037        let mut $field: Option<$ty> = None;
1038    };
1039
1040    (@init $field:ident, $ty:ty, $expr:expr) => {
1041        let mut $field: $ty = $expr;
1042    };
1043
1044    (@assign $field:ident) => {
1045        $field
1046    };
1047
1048    (@assign #[required $(= $reason:literal)?] $field:ident) => {
1049        match $field {
1050            Some($field) => $field,
1051            None => return Err($crate::Error::new($crate::ErrorKind::MissingRequired {
1052                name: stringify!($field),
1053                reason: $crate::__impl!(@required $($reason)*),
1054            })),
1055        }
1056    };
1057
1058    // The missing required argument.
1059    (@required) => { None };
1060    (@required $reason:literal) => { Some($reason) };
1061
1062    // Generate help for positional branches.
1063    (@switch-help
1064        $($doc:literal)*
1065        [ $(#$first_m:tt)* $first:ident $(, $(#$rest_m:tt)* $rest:ident)* ]
1066    ) => {
1067        $crate::Switch {
1068            usage: concat!(
1069                $crate::__impl!(@doc $(#$first_m)* $first),
1070                $(" ", $crate::__impl!(@doc $(#$rest_m)* $rest),)*
1071            ),
1072            docs: &[$($doc,)*]
1073        }
1074    };
1075
1076    // Generate help for matching branches.
1077    (@switch-help
1078        $($doc:literal)*
1079        [$first:literal $(| $rest:literal)* $(, $(#$arg_m:tt)* $arg:ident)*]
1080    ) => {
1081        $crate::Switch {
1082            usage: concat!(
1083                $first, $(", ", $rest,)*
1084                $(" ", $crate::__impl!(@doc $(#$arg_m)* $arg),)*
1085            ),
1086            docs: &[$($doc,)*]
1087        }
1088    };
1089
1090    // Generate switches help.
1091    (@switches $( $(#[doc = $doc:literal])* [$($branch:tt)*] $(if $cond:expr)? => $block:block)*) => {
1092        &[$($crate::__impl!(@switch-help $($doc)* [$($branch)*])),*]
1093    };
1094
1095    // Expansion for all branches.
1096    (@branches
1097        $switch:ident, $it:ident,
1098        $(#$_pfx_meta:tt)*
1099        $(
1100            [$sw_first_pat:literal $(| $sw_rest_pat:literal)* $(, $(#$sw_arg_m:tt)? $sw_arg:ident)*]
1101            $(if $sw_cond:expr)?
1102            => $sw_block:block
1103            $(#$_sw_meta:tt)*
1104        )*
1105        $(
1106            [$(#$pos_first_m:tt)? $pos_first:ident $(, $(#$pos_rest_m:tt)? $pos_rest:ident)*]
1107            $(if $pos_cond:expr)?
1108            => $pos_block:block
1109            $(#$_pos_meta:tt)*
1110        )*
1111    ) => {
1112        let __argwerk_name = $switch.as_str();
1113
1114        match __argwerk_name {
1115            $($sw_first_pat $(| $sw_rest_pat)* $(if $sw_cond)* => {
1116                $(let $sw_arg = $crate::__var!(switch $switch, $it, $(#$sw_arg_m)* $sw_arg);)*
1117
1118                if let Err(error) = (|| $crate::helpers::into_result($sw_block))() {
1119                    return Err(::argwerk::Error::new(::argwerk::ErrorKind::Error {
1120                        name: __argwerk_name.into(),
1121                        error
1122                    }));
1123                }
1124
1125                continue;
1126            })*
1127            _ => {
1128                $(if true $(&& $pos_cond)* {
1129                    let __argwerk_name: Box<str> = __argwerk_name.into();
1130
1131                    let $pos_first = $crate::__var!(first $it, $(#$pos_first_m)* $switch);
1132                    $(let $pos_rest = $crate::__var!(pos $it, $(#$pos_rest_m)* $pos_rest);)*
1133
1134                    if let Err(error) = (|| $crate::helpers::into_result($pos_block))() {
1135                        return Err(::argwerk::Error::new(::argwerk::ErrorKind::Error {
1136                            name: __argwerk_name,
1137                            error
1138                        }));
1139                    }
1140
1141                    continue;
1142                })*
1143            },
1144        }
1145
1146        if __argwerk_name.starts_with('-') {
1147            return Err(::argwerk::Error::new(::argwerk::ErrorKind::UnsupportedSwitch {
1148                switch: __argwerk_name.into()
1149            }));
1150        } else {
1151            return Err(::argwerk::Error::new(::argwerk::ErrorKind::UnsupportedArgument {
1152                argument: __argwerk_name.into()
1153            }));
1154        }
1155    };
1156}
1157
1158/// Helper to decode a variable.
1159#[doc(hidden)]
1160#[macro_export]
1161macro_rules! __var {
1162    (var $var:ident) => { $var };
1163    (var (os) $var:ident) => { ::std::ffi::OsString::from($var) };
1164
1165    (rest $it:ident) => { $it.rest()? };
1166    (rest (os) $it:ident) => { $it.rest_os() };
1167
1168    (next_unless_switch $it:ident) => { $it.next_unless_switch()? };
1169    (next_unless_switch (os) $it:ident) => { $it.next_unless_switch_os() };
1170
1171    // Various ways of parsing the first argument.
1172    (first $it:ident, #[rest $($tt:tt)*] $var:ident) => {
1173        Some($crate::__var!(var $($tt)* $var))
1174            .into_iter()
1175            .chain($crate::__var!(rest $($tt)* $it))
1176            .collect::<Vec<_>>();
1177    };
1178    (first $it:ident, #[option $($tt:tt)*] $var:ident) => {
1179        Some($crate::__var!(var $($tt)* $var))
1180    };
1181    (first $it:ident, #[os] $var:ident) => {
1182        Some(::std::ffi::OsString::from($var))
1183    };
1184    (first $it:ident, $var:ident) => {
1185        $var
1186    };
1187
1188    // Parse the rest of the available arguments.
1189    (pos $it:ident, #[rest $($tt:tt)*] $_:ident) => {
1190        $crate::__var!(rest $($tt)* $it)
1191    };
1192    // Parse an optional argument.
1193    (pos $it:ident, #[option $($tt:tt)*] $_:ident) => {
1194        $crate::__var!(next_unless_switch $($tt)* $it)
1195    };
1196    // Parse an os string argument.
1197    (pos $it:ident, #[os] $_:ident) => {
1198        match $it.next_os() {
1199            Some($var) => $var,
1200            None => {
1201                return Err(::argwerk::Error::new(
1202                    ::argwerk::ErrorKind::MissingPositional {
1203                        name: stringify!($var),
1204                    },
1205                ))
1206            }
1207        }
1208    };
1209
1210    // Parse the rest of the arguments.
1211    (pos $it:ident, $var:ident) => {
1212        match $it.next()? {
1213            Some($var) => $var,
1214            None => {
1215                return Err(::argwerk::Error::new(
1216                    ::argwerk::ErrorKind::MissingPositional {
1217                        name: stringify!($var),
1218                    },
1219                ))
1220            }
1221        }
1222    };
1223
1224    // Parse the rest of the available arguments.
1225    (switch $switch:ident, $it:ident, #[rest $($tt:tt)*] $arg:ident) => {
1226        $crate::__var!(rest $($tt)* $it)
1227    };
1228    // Parse an optional argument.
1229    (switch $switch:ident, $it:ident, #[option $($tt:tt)*] $arg:ident) => {
1230        $crate::__var!(next_unless_switch $($tt)* $it)
1231    };
1232
1233    // Parse next #[os] string argument.
1234    (switch $switch:ident, $it:ident, #[os] $var:ident) => {
1235        match $it.next_os() {
1236            Some($var) => $var,
1237            None => {
1238                return Err(::argwerk::Error::new(
1239                    ::argwerk::ErrorKind::MissingSwitchArgument {
1240                        switch: $switch.into(),
1241                        argument: stringify!($var),
1242                    },
1243                ))
1244            }
1245        }
1246    };
1247
1248    // Parse next argument.
1249    (switch $switch:ident, $it:ident, $var:ident) => {
1250        match $it.next()? {
1251            Some($var) => $var,
1252            None => {
1253                return Err(::argwerk::Error::new(
1254                    ::argwerk::ErrorKind::MissingSwitchArgument {
1255                        switch: $switch.into(),
1256                        argument: stringify!($var),
1257                    },
1258                ))
1259            }
1260        }
1261    };
1262}