6 releases

new 0.3.1 Jan 11, 2025
0.3.0 Aug 27, 2024
0.2.0 May 23, 2024
0.1.2 May 22, 2024
0.1.0 Jun 22, 2023

#675 in Command-line interface

Download history 141/week @ 2024-09-24 135/week @ 2024-10-01 108/week @ 2024-10-08 75/week @ 2024-10-15 25/week @ 2024-10-22 38/week @ 2024-10-29 46/week @ 2024-11-05 27/week @ 2024-11-12 41/week @ 2024-11-19 90/week @ 2024-11-26 43/week @ 2024-12-03 78/week @ 2024-12-10 105/week @ 2024-12-17 19/week @ 2024-12-24 19/week @ 2024-12-31 162/week @ 2025-01-07

314 downloads per month
Used in 3 crates (2 directly)

MIT/Apache

18KB
180 lines

clap-repl

Rust crates.io

One of the typical user interfaces for prompting commands is the repl (read eval print loop). One of the best ways of representing commands in a repl is using space separated arguments, which is what terminal shells do. And the way to parse such commands in Rust is the clap crate. This crate uses clap and reedline to provide such user interface in a way that you only focus on your app logic.

Features

Thanks to clap and reedline this crate handles:

  • Parsing the space separated commands into your data structure.
  • Help flag for each command.
  • Verifying the command is valid, generating useful errors and suggestions otherwise.
  • Auto complete and hint for the commands.

Example

use std::path::PathBuf;

use clap::{Parser, ValueEnum};
use clap_repl::reedline::{
    DefaultPrompt, DefaultPromptSegment, FileBackedHistory, Reedline, Signal,
};
use clap_repl::ClapEditor;

#[derive(Debug, Parser)]
#[command(name = "")] // This name will show up in clap's error messages, so it is important to set it to "".
enum SampleCommand {
    Download {
        path: PathBuf,
        /// Check the integrity of the downloaded object
        ///
        /// Uses SHA256
        #[arg(long)]
        check_sha: bool,
    },
    /// A command to upload things.
    Upload,
    /// Login into the system.
    Login {
        /// Optional. You will be prompted if you don't provide it.
        #[arg(short, long)]
        username: Option<String>,
        #[arg(short, long, value_enum, default_value_t = Mode::Secure)]
        mode: Mode,
    },
}

#[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, ValueEnum)]
enum Mode {
    /// Encrypt the password
    Secure,
    /// Send the password plain
    ///
    /// This paragraph is ignored because there is no long help text for possible values in clap.
    Insecure,
}

fn main() {
    let prompt = DefaultPrompt {
        left_prompt: DefaultPromptSegment::Basic("simple-example".to_owned()),
        ..DefaultPrompt::default()
    };
    let rl = ClapEditor::<SampleCommand>::builder()
        .with_prompt(Box::new(prompt))
        .with_editor_hook(|reed| {
            // Do custom things with `Reedline` instance here
            reed.with_history(Box::new(
                FileBackedHistory::with_file(10000, "/tmp/clap-repl-simple-example-history".into())
                    .unwrap(),
            ))
        })
        .build();
    rl.repl(|command| {
        match command {
            SampleCommand::Download { path, check_sha } => {
                println!("Downloaded {path:?} with checking = {check_sha}");
            }
            SampleCommand::Upload => {
                println!("Uploaded");
            }
            SampleCommand::Login { username, mode } => {
                // You can use another `reedline::Reedline` inside the loop.
                let mut rl = Reedline::create();
                let username = username
                    .unwrap_or_else(|| read_line_with_reedline(&mut rl, "What is your username? "));
                let password = read_line_with_reedline(&mut rl, "What is your password? ");
                println!("Logged in with {username} and {password} in mode {mode:?}");
            }
        }
    });
}

fn read_line_with_reedline(rl: &mut Reedline, prompt: &str) -> String {
    let Signal::Success(x) = rl
        .read_line(&DefaultPrompt::new(
            DefaultPromptSegment::Basic(prompt.to_owned()),
            DefaultPromptSegment::Empty,
        ))
        .unwrap()
    else {
        panic!();
    };
    x
}

Screenshot from 2023-06-22 11-32-58 Screenshot from 2023-06-22 11-35-33 image

Dependencies

~8–19MB
~265K SLoC