A library to quickly build full-featured REPLs supported by CLAP and readline (provided via rustyline)
value_parsers
(i.e. a list of valid values)ValueHint::FilePath
)value_parsers
at runtimeget_async_writer()
;
) for unconditional evaluation&&
) for chaining successful evaluations||
) for error handling evaluationstest-runner
featureA minimal example showing a basic REPL is as follows:
```rust use clapcmd::{ArgMatches, ClapCmd, ClapCmdResult, Command};
fn do_ping(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.output("pong"); Ok(()) }
fn main() { let mut cmd = ClapCmd::default(); cmd.addcommand( doping, Command::new("ping").about("do a ping") ); cmd.run_loop(); } ```
To pass state or persistent information to callbacks, provide a State
class like so.
The State
class must implement Clone
trait, and can be accessed via
the get_state()
and set_state()
methods on the ClapCmd
reference passed into the
callback.
```rust use clapcmd::{ArgMatches, ClapCmd, ClapCmdResult, Command};
struct State { counter: u32, }
fn docount(cmd: &mut ClapCmd
fn main() { let mut cmd = ClapCmd::withstate(State { counter: 0 }); cmd.addcommand(docount, Command::new("count").about("increment a counter")); cmd.runloop(); } ```
Groups can be used to logically separate sets of commands in the built-in help
menu.
They can also be used to quickly activate and deactivate commands via the add_group
and
remove_group
methods
```rust use clapcmd::{ArgMatches, ClapCmd, ClapCmdResult, Command, HandlerGroup}; use once_cell::sync::Lazy;
static LOADEDGROUP: Lazy
static UNLOADEDGROUP: Lazy
fn doload(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.addgroup(&LOADEDGROUP); cmd.removegroup(&UNLOADED_GROUP); cmd.info("loaded"); Ok(()) }
fn dounload(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.addgroup(&UNLOADEDGROUP); cmd.removegroup(&LOADED_GROUP); cmd.info("unloaded"); Ok(()) }
fn do_apple(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.output("apple"); Ok(()) }
fn do_banana(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.output("banana"); Ok(()) }
fn main() { let mut cmd = ClapCmd::default(); cmd.addgroup(&UNLOADEDGROUP); cmd.run_loop(); } ```
By enabling the test-runner
feature and using the built-in output
, success
, info
, warn
, and error
functions, it is easy to automate e2e tests of your CLI. See the tests/
folder for more examples.
```rust use clapcmd::{ArgMatches, ClapCmd, ClapCmdResult, Command};
fn do_hello(cmd: &mut ClapCmd, _: ArgMatches) -> ClapCmdResult { cmd.output("hello"); Ok(()) }
let mut cmd = ClapCmd::default(); cmd.addcommand( dohello, Command::new("hello").about("simple hello world") ); let _ = cmd.one_cmd("goodbye");
assert!( cmd.error.contains("unknown command"), "did not detect invalid command", ); let _ = cmd.one_cmd("hello");
assert!( cmd.output.contains("hello"), "did not run hello world command correctly", ); ```
Refer to the examples/
folder for more demonstrations of advanced use cases
This library is tested with Rust 1.65 along with the latest version of Rust