Nathan Fisher
fb389fd309
- one match statement to return a `Box<dyn Cmd>` - one array containing all command names Only two places to register new commands (besides their module), both in crate::cmd::mod.rs. Also removes `once_cell` crate dependency. Replace `base64` crate dependency with `data_encoding::BASE64` so that both base32 and base64 commands use the same crate.
159 lines
5.1 KiB
Rust
159 lines
5.1 KiB
Rust
use super::Cmd;
|
|
use clap::{value_parser, Arg, ArgAction, Command};
|
|
use data_encoding::BASE32;
|
|
use std::{
|
|
error::Error,
|
|
fs,
|
|
io::{self, Read, Write},
|
|
};
|
|
use termcolor::{Color, ColorChoice, ColorSpec, StandardStream, WriteColor};
|
|
|
|
#[derive(Debug)]
|
|
pub struct Base32 {
|
|
name: &'static str,
|
|
path: Option<crate::Path>,
|
|
}
|
|
|
|
impl Default for Base32 {
|
|
fn default() -> Self {
|
|
Self {
|
|
name: "base32",
|
|
path: Some(crate::Path::UsrBin),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl Cmd for Base32 {
|
|
fn name(&self) -> &str {
|
|
self.name
|
|
}
|
|
|
|
fn cli(&self) -> clap::Command {
|
|
Command::new("base32")
|
|
.author("Nathan Fisher")
|
|
.about("Base32 encode/decode data and print to standard output")
|
|
.args([
|
|
Arg::new("INPUT")
|
|
.help("The input file to use")
|
|
.num_args(1..),
|
|
Arg::new("DECODE")
|
|
.help("Decode rather than encode")
|
|
.short('d')
|
|
.long("decode")
|
|
.action(ArgAction::SetTrue),
|
|
Arg::new("IGNORE")
|
|
.help("Ignore whitespace when decoding")
|
|
.short('i')
|
|
.long("ignore-space")
|
|
.action(ArgAction::SetTrue),
|
|
Arg::new("WRAP")
|
|
.help("Wrap encoded lines after n characters")
|
|
.short('w')
|
|
.long("wrap")
|
|
.value_parser(value_parser!(usize))
|
|
.default_value("76"),
|
|
Arg::new("color")
|
|
.short('c')
|
|
.long("color")
|
|
.value_parser(["always", "ansi", "auto", "never"]),
|
|
Arg::new("VERBOSE")
|
|
.help("Display a header naming each file")
|
|
.short('v')
|
|
.long("verbose")
|
|
.action(ArgAction::SetTrue),
|
|
Arg::new("QUIET")
|
|
.help("Do not display header, even with multiple files")
|
|
.short('q')
|
|
.long("quiet")
|
|
.action(ArgAction::SetTrue),
|
|
])
|
|
}
|
|
|
|
fn run(&self, matches: Option<&clap::ArgMatches>) -> Result<(), Box<dyn std::error::Error>> {
|
|
let Some(matches) = matches else {
|
|
return Err(io::Error::new(io::ErrorKind::Other, "No input").into());
|
|
};
|
|
let files: Vec<_> = match matches.get_many::<String>("INPUT") {
|
|
Some(c) => c.cloned().collect(),
|
|
None => vec![String::from("-")],
|
|
};
|
|
let len = files.len();
|
|
let color = match matches.get_one::<String>("color").map(String::as_str) {
|
|
Some("always") => ColorChoice::Always,
|
|
Some("ansi") => ColorChoice::AlwaysAnsi,
|
|
Some("auto") => {
|
|
if atty::is(atty::Stream::Stdout) {
|
|
ColorChoice::Auto
|
|
} else {
|
|
ColorChoice::Never
|
|
}
|
|
}
|
|
_ => ColorChoice::Never,
|
|
};
|
|
for (index, file) in files.into_iter().enumerate() {
|
|
if { len > 1 || matches.get_flag("VERBOSE") } && !matches.get_flag("QUIET") {
|
|
let mut stdout = StandardStream::stdout(color);
|
|
stdout.set_color(ColorSpec::new().set_fg(Some(Color::Green)))?;
|
|
match index {
|
|
0 => writeln!(stdout, "===> {file} <==="),
|
|
_ => writeln!(stdout, "\n===> {file} <==="),
|
|
}?;
|
|
stdout.reset()?;
|
|
} else if index > 0 {
|
|
println!();
|
|
}
|
|
let contents = get_contents(&file)?;
|
|
if matches.get_flag("DECODE") {
|
|
decode_base32(contents, matches.get_flag("IGNORE"))?;
|
|
} else {
|
|
encode_base32(
|
|
&contents,
|
|
match matches.get_one::<usize>("WRAP") {
|
|
Some(c) => *c,
|
|
None => {
|
|
return Err(io::Error::new(io::ErrorKind::Other, "Invalid wrap").into())
|
|
}
|
|
},
|
|
);
|
|
}
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn path(&self) -> Option<crate::Path> {
|
|
self.path
|
|
}
|
|
}
|
|
|
|
fn decode_base32(mut contents: String, ignore: bool) -> Result<(), Box<dyn Error>> {
|
|
if ignore {
|
|
contents.retain(|c| !c.is_whitespace());
|
|
} else {
|
|
contents = contents.replace('\n', "");
|
|
}
|
|
let decoded = BASE32.decode(contents.as_bytes())?;
|
|
let output = String::from_utf8(decoded)?;
|
|
println!("{}\n", output.trim_end());
|
|
Ok(())
|
|
}
|
|
|
|
fn encode_base32(contents: &str, wrap: usize) {
|
|
BASE32
|
|
.encode(contents.as_bytes())
|
|
.chars()
|
|
.collect::<Vec<char>>()
|
|
.chunks(wrap)
|
|
.map(|c| c.iter().collect::<String>())
|
|
.for_each(|line| println!("{line}"));
|
|
}
|
|
|
|
fn get_contents(file: &str) -> Result<String, Box<dyn Error>> {
|
|
let mut contents = String::new();
|
|
if file == "-" {
|
|
io::stdin().read_to_string(&mut contents)?;
|
|
} else {
|
|
contents = fs::read_to_string(file)?;
|
|
}
|
|
Ok(contents)
|
|
}
|