Argp is a Derive-based argument parser optimized for code size and flexibility.

The public API of this library consists primarily of the FromArgs derive and the parse_args_or_exit function, which can be used to produce a top-level FromArgs type from the current program’s command-line arguments.

Features

Origins

Argp originally started as a fork of argh to make it less opinionated, more UNIXy and flexible.

Notable changes from argh:

Basic Example

``` rust use argp::FromArgs;

/// Reach new heights.

[derive(FromArgs)]

struct GoUp { /// Whether or not to jump. #[argp(switch, short = 'j')] jump: bool,

/// How high to go.
#[argp(option, arg_name = "meters")]
height: usize,

/// An optional nickname for the pilot.
#[argp(option, arg_name = "name")]
pilot_nickname: Option<String>,

}

fn main() { let up: GoUp = argp::parseargsor_exit(argp::DEFAULT); } ```

./some_bin --help will then output the following:

Usage: cmdname [-j] --height <meters> [--pilot-nickname <name>]

Reach new heights.

Options:
  -j, --jump                   Whether or not to jump.
      --height <meters>        How high to go.
      --pilot-nickname <name>  An optional nickname for the pilot.
  -h, --help                   Show this help message and exit.

The resulting program can then be used in any of these ways:

Switches, like jump, are optional and will be set to true if provided.

Options, like height and pilot_nickname, can be either required, optional, or repeating, depending on whether they are contained in an Option or a Vec. Default values can be provided using the #[argp(default = "<your_code_here>")] attribute, and in this case an option is treated as optional.

``` rust use argp::FromArgs;

fn default_height() -> usize { 5 }

/// Reach new heights.

[derive(FromArgs)]

struct GoUp { /// An optional nickname for the pilot. #[argp(option)] pilot_nickname: Option,

/// An optional height.
#[argp(option, default = "default_height()")]
height: usize,

/// An optional direction which is "up" by default.
#[argp(option, default = "String::from(\"only up\")")]
direction: String,

}

fn main() { let up: GoUp = argp::parseargsor_exit(argp::DEFAULT); } ```

Custom option types can be deserialized so long as they implement the FromArgValue trait (already implemented for most types in std for which the FromStr trait is implemented). If more customized parsing is required, you can supply a custom fn(&str) → Result<T, E> using the from_str_fn attribute, or fn(&OsStr) → Result<T, E> using the from_os_str_fn attribute, where E implements ToString:

``` rust use argp::FromArgs; use std::ffi::OsStr; use std::path::PathBuf;

/// Goofy thing.

[derive(FromArgs)]

struct FineStruct { /// Always five. #[argp(option, fromstrfn(always_five))] five: usize,

/// File path.
#[argp(option, from_os_str_fn(convert_path))]
path: PathBuf,

}

fn alwaysfive(value: &str) -> Result { Ok(5) }

fn convert_path(value: &OsStr) -> Result { Ok(PathBuf::from("/tmp").join(value)) } ```

Positional arguments can be declared using #[argp(positional)]. These arguments will be parsed in order of their declaration in the structure:

``` rust use argp::FromArgs;

/// A command with positional arguments.

[derive(FromArgs, PartialEq, Debug)]

struct WithPositional { #[argp(positional)] first: String, } ```

The last positional argument may include a default, or be wrapped in Option or Vec to indicate an optional or repeating positional argument.

Subcommands are also supported. To use a subcommand, declare a separate FromArgs type for each subcommand as well as an enum that cases over each command:

``` rust use argp::FromArgs;

/// Top-level command.

[derive(FromArgs, PartialEq, Debug)]

struct TopLevel { /// Be verbose. #[argp(switch, short = 'v', global)] verbose: bool,

#[argp(subcommand)]
nested: MySubCommandEnum,

}

[derive(FromArgs, PartialEq, Debug)]

[argp(subcommand)]

enum MySubCommandEnum { One(SubCommandOne), Two(SubCommandTwo), }

/// First subcommand.

[derive(FromArgs, PartialEq, Debug)]

[argp(subcommand, name = "one")]

struct SubCommandOne { /// How many x. #[argp(option)] x: usize, }

/// Second subcommand.

[derive(FromArgs, PartialEq, Debug)]

[argp(subcommand, name = "two")]

struct SubCommandTwo { /// Whether to fooey. #[argp(switch)] fooey: bool, } ```

For more information, refer to the argp documentation.

How to debug the expanded derive macro for argp

The argp::FromArgs derive macro can be debugged with the cargo-expand crate.

Expand the derive macro in examples/simple_example.rs

See argp/examples/simple_example.rs for the example struct we wish to expand.

First, install cargo-expand by running cargo install cargo-expand. Note this requires the nightly build of Rust.

Once installed, run cargo expand with in the argp package and you can see the expanded code.

License

This project is licensed under BSD-3-Clause license. For the full text of the license, see the LICENSE file.

  1. Measured on a release build with strip = true and panic = "abort". The exact size depends on several factors, including the number of options and subcommands.