This repository has been archived on 2024-10-16. You can view files and clone it, but cannot push or open issues or pull requests.
hedu/src/main.rs

140 lines
4.0 KiB
Rust
Raw Normal View History

//! # A simple hexdumper with a somewhat fancy format
2024-02-01 22:22:16 +01:00
//!
//! Dump data from any readable source, such as the stdin or a file
2024-02-01 22:22:16 +01:00
#![warn(clippy::pedantic)]
use std::{fs::File, io::IsTerminal, path::PathBuf};
2024-02-01 22:22:16 +01:00
2024-02-16 16:39:04 +01:00
use libpt::log::{error, trace, warn, Level, Logger};
2024-02-01 22:22:16 +01:00
use clap::Parser;
use clap_verbosity_flag::{InfoLevel, Verbosity};
mod dumper;
2024-02-16 16:39:04 +01:00
use dumper::{DataSource, Hedu};
2024-02-01 22:22:16 +01:00
#[derive(Debug, Clone, Parser)]
#[command(
author,
version,
about,
long_about,
help_template = r#"{about-section}
2024-02-01 22:22:16 +01:00
{usage-heading} {usage}
{all-args}{tab}
2024-02-01 23:29:50 +01:00
{name}: {version}
2024-02-01 22:22:16 +01:00
Author: {author-with-newline}
"#
)]
/// Hexdumper written in Rust
2024-02-01 22:22:16 +01:00
pub struct Cli {
// clap_verbosity_flag seems to make this a global option implicitly
/// set a verbosity, multiple allowed (f.e. -vvv)
#[command(flatten)]
pub verbose: Verbosity<InfoLevel>,
/// show additional logging meta data
#[arg(long)]
pub meta: bool,
/// show character representation
#[arg(short, long)]
pub chars: bool,
/// skip first N bytes
#[arg(short, long, default_value_t = 0)]
pub skip: usize,
/// only interpret N bytes (end after N)
#[arg(short, long, default_value_t = 0)]
pub limit: usize,
/// show identical lines
#[arg(short = 'i', long)]
pub show_identical: bool,
/// a data source, probably a file.
///
/// If left empty or set as "-", the program will read from stdin.
pub data_source: Vec<String>,
}
2024-02-01 22:02:03 +01:00
fn main() {
2024-02-01 22:22:16 +01:00
let mut cli = cli_parse();
let mut sources: Vec<Box<dyn DataSource>> = Vec::new();
2024-02-16 16:39:04 +01:00
if !cli.data_source.is_empty() && cli.data_source[0] != "-" {
2024-02-01 22:22:16 +01:00
for data_source in &cli.data_source {
let data_source: PathBuf = PathBuf::from(data_source);
if data_source.is_dir() {
warn!("Not a file {:?}, skipping", data_source);
// std::process::exit(1);
continue;
}
trace!("Trying to open '{:?}'", data_source);
match File::open(&data_source) {
Ok(file) => sources.push(Box::new(file)),
Err(err) => {
error!("Could not open '{:?}': {err}", data_source);
std::process::exit(1);
}
};
}
} else {
trace!("Trying to open stdin");
let stdin = std::io::stdin();
if stdin.is_terminal() {
warn!("Refusing to dump from interactive terminal");
std::process::exit(2)
}
// just for the little header
cli.data_source = Vec::new();
2024-02-16 16:39:04 +01:00
cli.data_source.push("stdin".to_string());
2024-02-01 22:22:16 +01:00
sources.push(Box::new(stdin));
}
for (i, source) in sources.iter_mut().enumerate() {
let mut config = Hedu::new(cli.chars, cli.skip, cli.show_identical, cli.limit);
// FIXME: find a better way to get the file name
// Currently, skipped sources make an extra newline here.
match config.chars {
false => {
println!("{:─^59}", format!(" {} ", cli.data_source[i]));
}
true => {
println!("{:─^80}", format!(" {} ", cli.data_source[i]));
}
}
match config.dump(&mut **source) {
2024-02-16 16:39:04 +01:00
Ok(()) => (),
2024-02-01 22:22:16 +01:00
Err(err) => {
error!("Could not dump data of file: {err}");
std::process::exit(3);
}
}
if i < cli.data_source.len() - 1 {
config.newline();
}
}
}
2024-02-01 22:22:16 +01:00
fn cli_parse() -> Cli {
let cli = Cli::parse();
let ll: Level = match cli.verbose.log_level().unwrap().as_str() {
"TRACE" => Level::TRACE,
"DEBUG" => Level::DEBUG,
"INFO" => Level::INFO,
"WARN" => Level::WARN,
"ERROR" => Level::ERROR,
_ => {
unreachable!();
}
};
if cli.meta {
let _ = Logger::builder().max_level(ll).build();
2024-02-01 22:22:16 +01:00
} else {
// less verbose version
let _ = Logger::builder().build();
2024-02-01 22:22:16 +01:00
}
2024-02-16 16:39:04 +01:00
cli
2024-02-01 22:02:03 +01:00
}