#stdin #clap #value #arguments #wrapper #reading #arg

bin+lib clap-stdin

Provides a type for easily accepting Clap arguments from stdin

7 unstable releases (3 breaking)

0.4.0 Jan 8, 2024
0.3.0 Dec 10, 2023
0.2.2 Dec 6, 2023
0.2.1 Jul 30, 2023
0.1.1 Apr 26, 2023

#84 in Command-line interface

Download history 122/week @ 2024-01-03 389/week @ 2024-01-10 661/week @ 2024-01-17 692/week @ 2024-01-24 934/week @ 2024-01-31 1220/week @ 2024-02-07 996/week @ 2024-02-14 1101/week @ 2024-02-21 1161/week @ 2024-02-28 1171/week @ 2024-03-06 1166/week @ 2024-03-13 1779/week @ 2024-03-20 1305/week @ 2024-03-27 1233/week @ 2024-04-03 1611/week @ 2024-04-10 1337/week @ 2024-04-17

5,805 downloads per month
Used in 13 crates (12 directly)

MIT/Apache

21KB
289 lines

clap-stdin Build

This library offers two wrapper types for clap Args that help for cases where values may be passed in via stdin. When an Arg value is to be read from stdin, the user will pass the commonly used stdin alias: -

  • MaybeStdin: Used when a value can be passed in via args OR stdin
  • FileOrStdin: Used when a value can be read in from a file OR stdin

MaybeStdin

Example usage with clap's derive feature for a positional argument:

use clap::Parser;

use clap_stdin::MaybeStdin;

#[derive(Debug, Parser)]
struct Args {
    value: MaybeStdin<String>,
}

let args = Args::parse();
println!("value={}", args.value);

Calling this CLI:

# using stdin for positional arg value
$ echo "testing" | cargo run -- -
value=testing

Compatible Types

MaybeStdin can wrap any type that matches the trait bounds for Arg: FromStr and Clone

use std::path::PathBuf;
use clap::Parser;
use clap_stdin::MaybeStdin;

#[derive(Debug, Parser)]
struct Args {
    path: MaybeStdin<PathBuf>,
}
$ pwd | ./example -

FileOrStdin

Example usage with clap's derive feature for a positional argument:

use clap::Parser;

use clap_stdin::FileOrStdin;

#[derive(Debug, Parser)]
struct Args {
    input: FileOrStdin,
}

# fn main() -> anyhow::Result<()> {
let args = Args::parse();
println!("input={}", args.input.contents()?);
# Ok(())
# }

Calling this CLI:

# using stdin for positional arg value
$ echo "testing" | cargo run -- -
input=testing

# using filename for positional arg value
$ echo "testing" > input.txt
$ cargo run -- input.txt
input=testing

Compatible Types

FileOrStdin can wrap any type that matches the trait bounds for Arg: FromStr and Clone

use std::path::PathBuf;
use clap::Parser;
use clap_stdin::FileOrStdin;

#[derive(Debug, Parser)]
struct Args {
    path: FileOrStdin<u32>,
}
# Value from stdin
$ wc ~/myfile.txt -l | ./example -

# Value from file
$ cat myfile.txt
42
$ .example myfile.txt

Reading from Stdin without special characters

When using MaybeStdin or FileOrStdin, you can allow your users to omit the "-" character to read from stdin by providing a default_value to clap. This works with positional and optional args:

use clap::Parser;

use clap_stdin::FileOrStdin;

#[derive(Debug, Parser)]
struct Args {
    #[clap(default_value = "-")]
    input: FileOrStdin,
}

# fn main() -> anyhow::Result<()> {
let args = Args::parse();
println!("input={}", args.input.contents()?);
# Ok(())
# }

Calling this CLI:

# using stdin for positional arg value
$ echo "testing" | cargo run
input=testing

# using filename for positional arg value
$ echo "testing" > input.txt
$ cargo run -- input.txt
input=testing

Async Support

FileOrStdin can also be used with tokio::io::AsyncRead using the tokio feature. See FileOrStdin::contents_async and FileOrStdin::into_async_reader for examples.

Using MaybeStdin or FileOrStdin multiple times

Both MaybeStdin and FileOrStdin will check at runtime if stdin is being read from multiple times. You can use this as a feature if you have mutually exclusive args that should both be able to read from stdin, but know that the user will receive an error if 2+ MaybeStdin args receive the "-" value.

For example, this compiles:

use clap_stdin::{FileOrStdin, MaybeStdin};

#[derive(Debug, clap::Parser)]
struct Args {
    first: FileOrStdin,
    second: MaybeStdin<u32>,
}

and it will work fine if the stdin alias - is only passed for one of the arguments:

$ echo "2" | ./example FIRST -

But if stdin is attempted to be used for both arguments, there will be no value for the second arg

$ echo "2" | ./example - -
error: invalid value '-' for '<SECOND>': stdin argument used more than once

License

clap-stdin is both MIT and Apache License, Version 2.0 licensed, as found in the LICENSE-MIT and LICENSE-APACHE files.

Dependencies

~0.3–2.1MB
~41K SLoC