flexiargs is a lightweight and dependency-friendly Rust crate designed for rule-based
command-line argument parsing without relying on macros, derive systems, or bloated
abstractions.
Instead of hiding behavior behind procedural magic, flexiargs provides a clean and explicit API that allows developers to bind CLI arguments directly to variables while retaining full control over parsing logic and application flow.
Built with simplicity and flexibility in mind, it gives you low-level control when you need it, without sacrificing ergonomics. With flexiargs, you can easily manage:
-
βοΈ Parsing behavior and argument rules;
-
β Validation and constraint handling;
-
π¦ Unmatched or forwarded arguments;
-
π Execution flow and command dispatching;
-
π§© Custom CLI architectures and dynamic behaviors;
Unlike heavy CLI frameworks, flexiargs focuses on predictable behavior, small footprint, and straightforward integration, making it ideal for projects where control and portability matter. Perfect for:
-
π¦ Package managers;
-
π οΈ System utilities;
-
π₯ Installers and bootstrap tools;
-
π§© Embedded CLI environments;
-
π Custom shell-like applications;
-
π§ͺ Internal developer tooling;
-
β‘ Lightweight standalone binaries;
-
π Experimental runtime environments;
flexiargs is designed to keep CLI parsing simple, explicit, and predictable, without relying on macros or heavy abstractions. Instead of hiding behavior behind complex frameworks, it exposes clear rule-based control over how arguments are interpreted and processed.
-
π§© Simple rule-based parser API
Defines parsing rules in a declarative way, without DSLs or code generation. You have full control over how each argument is interpreted. -
π« No procedural macros
No dependency onderiveor procedural macros. This reduces compile time, avoids hidden behavior, and improves debugging clarity. -
π Supports short and long flags
Full support for both short flags (-v,-h) and long flags (--verbose,--help) for flexible CLI design. -
π Supports argument formats
--flag=value
Inline assignment for compact CLI usage.--flag value
Classic POSIX-style separation for readability in interactive usage.
-
π Typed parsing with
FromStr
Automatically converts string inputs into Rust types using theFromStrtrait, ensuring type safety and reducing manual parsing code. -
β οΈ Automatic error formatting
Parsing errors are automatically formatted in a clear and consistent way, improving end-user feedback. -
π Optional and required arguments
Explicit support for required and optional parameters, removing the need for manual validation logic. -
π Multi-value collection
Support for multiple values for the same flag (e.g.--file a b c), collected into typed containers. -
π₯ Positional argument passthrough
Positional arguments can be captured or forwarded directly to subprocesses or higher-level handlers. -
π Strict validation modes
Enables strict parsing mode to reject unknown or malformed arguments, ideal for robust tools and system utilities. -
π§΅ Thread-safe shared settings (
RwLock)
Safe shared state across threads usingRwLock, useful for concurrent CLI applications or embedded runtimes. -
β‘ Custom actions/callbacks
Allows execution of custom callbacks during parsing for dynamic or context-aware behavior. -
πͺΆ Minimal and dependency-light design
Keeps the core lightweight with minimal dependencies, focusing on portability and predictable behavior.
Add the crate to your Cargo.toml:
[dependencies]
flexiargs = "1.0"The crate intentionally exposes a very small API surface:
pub use messages::{invalid_arg, missing_arg};
pub use flexiargs::{Arg, parse_into_vars};| Item | Description |
|---|---|
Arg |
Defines parsing rules |
parse_into_vars |
Executes parsing |
invalid_arg |
Standardized invalid argument error |
missing_arg |
Standardized missing parameter error |
use flexiargs::{Arg, parse_into_vars};
use std::collections::VecDeque;
let mut sync_mode = false;
let mut packages = Vec::new();
let mut cache_dir = String::new();
let mut config_file: Option<String> = None;
let mut use_overlay = true;
let mut rules = [
Arg::bool(Some("-S"), "--sync", &mut sync_mode),
Arg::collect_list(None, "--pkgs", "packages", &mut packages),
Arg::value(None, "--cache-dir", "path", &mut cache_dir),
Arg::option(Some("-c"), "--config", "file", &mut config_file),
Arg::set(None, "--disable-overlay", false, &mut use_overlay),
];
let args = VecDeque::from(vec![
"-S".to_string(),
"wget".to_string(),
"curl".to_string(),
"--cache-dir=/tmp".to_string(),
"--disable-overlay".to_string()
]);
parse_into_vars("aports", &mut rules, args).ok();
drop(rules);
println!("Sync: {}", sync_mode);
println!("Packages: {:?}", packages);
println!("Cache dir: {}", cache_dir);
println!("Overlay enabled: {}", use_overlay);Every CLI behavior is defined using an Arg.
Each rule describes:
- π·οΈ Accepted flags;
- βοΈ Parsing behavior;
- π― Target variable;
- β Validation requirements;
Example:
Arg::bool(Some("-v"), "--verbose", &mut verbose)flexiargs is built around a small set of explicit rule types that define how command-line
input is interpreted, validated, and transformed. Instead of relying on implicit behavior
or hidden conventions, each rule type has a clear and predictable responsibility,
allowing you to compose CLI behavior in a controlled and deterministic way.
Sets a boolean to true when matched.
let mut verbose = false;
Arg::bool(Some("-v"), "--verbose", &mut verbose);-v--verbose
Parses values using FromStr.
let mut port: u16 = 0;
Arg::value(
Some("-p"),
"--port",
"port",
&mut port
);--port 8080--port=8080-p 8080
Stores values in Option<String>.
let mut config: Option<String> = None;
Arg::option(
Some("-c"),
"--config",
"file",
&mut config
);Assigns a predefined value when matched.
let mut overlay = true;
Arg::set(
None,
"--disable-overlay",
false,
&mut overlay
);Collects sequential positional values until another flag appears.
let mut packages = Vec::new();
Arg::collect_list(
None,
"--pkgs",
"packages",
&mut packages
);--pkgs wget curl git["wget", "curl", "git"]Executes arbitrary logic.
Arg::action(
Some("-V"),
"--version",
|| {
println!("myapp 1.0");
}
);- π Version output
- π οΈ Custom handlers
- πͺ Early exits
- π Dynamic state manipulation
Arguments can be marked as essential:
Arg::value(
None,
"--root",
"path",
&mut root
).essential();If no essential rule is matched:
myapp: setup: no essential parameter specified
flexiargs includes built-in support for shared application state using RwLock.
use std::sync::RwLock;
static DEBUG: RwLock<bool> = RwLock::new(false);
Arg::rw_bool(
Some("-d"),
"--debug",
&DEBUG
);use std::sync::RwLock;
static PORT: RwLock<u16> = RwLock::new(8080);
Arg::rw_value(
None,
"--port",
"port",
&PORT
);Arg::rw_set(
None,
"--production",
true,
&MODE
);Parsing in flexiargs is explicit, deterministic, and fully rule-driven. Each step of the parsing process is defined by clear rules that transform raw command-line input into structured, validated, and typed data. This makes behavior predictable, easier to debug, and consistent across different CLI designs.
parse_into_vars(
"server",
&mut rules,
args
).ok()?;The parser returns a ParseResult.
This provides:
- β Parsing success/failure;
- π¦ Unmatched arguments;
- π Positional handling;
- π Strict validation helpers;
Extracts the parsing result.
parse_into_vars("app", &mut rules, args)
.ok()?;Rejects any unmatched arguments.
parse_into_vars("app", &mut rules, args)
.strict()
.ok()?;Ensures the first argument matches a rule.
.strict_first()Rejects unmatched arguments up to a given depth.
.strict_level(2)Suppresses parsing errors.
Useful for optional parsing stages.
.passthrough()Fails if no arguments were supplied.
.require_args()?Collects unmatched positional arguments.
let mut remaining = Vec::new();
parse_into_vars("app", &mut rules, args)
.collect_rest(&mut remaining)?;The parser supports -- to stop option parsing.
myapp --verbose -- file1 file2Everything after -- becomes positional data.
flexiargs provides standardized and human-readable errors automatically.
myapp: invalid argument '--unknown'
Use 'myapp --help' to see available options.
myapp: setup: --port requires a <port>.
Usage: myapp setup --port <port>
Rules in flexiargs define how input tokens are interpreted and matched against declared CLI arguments. This matching system is flexible but explicit, allowing multiple naming styles and aliasing strategies while keeping behavior predictable and rule-based. Rules support:
- π€ Short flags;
- π·οΈ Long flags;
- π Aliases via
|; - π§· Inline assignment;
Arg::bool(
Some("-v"),
"--verbose|--debug",
&mut verbose
);-v--verbose--debug
Argument matching in flexiargs is designed to be both flexible and explicit. Instead of enforcing a single naming convention, it allows multiple identifiers and aliases per argument while keeping resolution deterministic and rule-based.
Contributions, improvements, and issue reports are welcome.
- π€ Auto-help generation
- π Environment variable integration
- π Shell completion generation
This repository has scripts created to be free software.
Therefore, they can be distributed and/or modified within the terms of the MIT License.
See the MIT License file for details.
- π§ Email: m10ferrari1200@gmail.com
- π§ Email: contatolinuxdicaspro@gmail.com
