shitbox/src/cmd/head/mod.rs

144 lines
4.1 KiB
Rust
Raw Normal View History

2022-12-25 18:29:09 -05:00
use super::Cmd;
2022-12-20 12:05:21 -05:00
use crate::Path;
2022-12-20 18:35:45 -05:00
use clap::{value_parser, Arg, ArgAction, ArgMatches, Command};
2022-12-25 18:29:09 -05:00
use std::{
error::Error,
fs,
io::{self, stdin, Read},
process,
};
2022-12-20 12:05:21 -05:00
#[derive(Debug)]
2022-12-25 18:29:09 -05:00
pub struct Head {
name: &'static str,
path: Option<Path>,
}
pub const HEAD: Head = Head {
name: "head",
path: Some(Path::Bin),
};
2022-12-20 12:05:21 -05:00
2022-12-25 18:29:09 -05:00
impl Cmd for Head {
fn name(&self) -> &str {
self.name
}
fn cli(&self) -> Command {
Command::new(self.name)
2022-12-20 12:05:21 -05:00
.version(env!("CARGO_PKG_VERSION"))
.author("Nathan Fisher")
.about("Display first lines of a file")
.long_about(
"Print the first 10 lines of each FILE to standard output.\n\
With more than one FILE, precede each with a header giving the file name.\n\
With no FILE, or when FILE is -, read standard input.\n\
Mandatory arguments to long options are mandatory for short options too."
)
.args([
Arg::new("FILES")
.help("The input file to use")
.num_args(1..),
Arg::new("BYTES")
.help("Count bytes instead of lines")
.short('c')
.long("bytes")
.action(ArgAction::SetTrue),
Arg::new("QUIET")
.help("Disable printing a header. Overrides -c")
.short('q')
.long("quiet")
.action(ArgAction::SetTrue),
Arg::new("HEADER")
.help("Each file is preceded by a header consisting of the string \"==> XXX <==\" where \"XXX\" is the name of the file.")
.short('v')
.long("verbose")
.action(ArgAction::SetTrue),
Arg::new("LINES")
.help("Count n number of lines (or bytes if -c is specified).")
.short('n')
.long("lines")
.default_value("10")
2022-12-21 20:19:38 -05:00
.allow_negative_numbers(false)
2022-12-20 12:05:21 -05:00
.value_parser(value_parser!(usize))
])
2022-12-25 18:29:09 -05:00
}
fn run(&self, matches: Option<&ArgMatches>) -> Result<(), Box<dyn Error>> {
let matches = if let Some(m) = matches {
m
} else {
return Err(io::Error::new(io::ErrorKind::Other, "No input").into());
};
let files = match matches.get_many::<String>("FILES") {
Some(c) => c.map(std::string::ToString::to_string).collect(),
None => vec!["-".to_string()],
};
let header =
!matches.get_flag("QUIET") && { files.len() > 1 || matches.get_flag("HEADER") };
for (index, file) in files.into_iter().enumerate() {
if index == 1 && header {
println!();
}
head(
&file,
match matches.get_one("LINES") {
Some(c) => *c,
None => 10,
},
header,
matches.get_flag("BYTES"),
);
}
Ok(())
}
fn path(&self) -> Option<Path> {
self.path
}
2022-12-20 12:05:21 -05:00
}
fn head(file: &str, count: usize, header: bool, bytes: bool) {
let mut contents = String::new();
if file == "-" {
match stdin().read_to_string(&mut contents) {
Ok(_) => true,
Err(e) => {
2022-12-20 18:35:45 -05:00
eprintln!("head: {e}");
2022-12-20 12:05:21 -05:00
process::exit(1);
}
};
} else {
let buf = fs::read_to_string(file);
contents = match buf {
Ok(c) => c,
Err(e) => {
2022-12-20 18:35:45 -05:00
eprintln!("head: {e}");
2022-12-20 12:05:21 -05:00
process::exit(1);
}
};
}
if header {
2022-12-20 18:35:45 -05:00
println!("==> {file} <==");
2022-12-20 12:05:21 -05:00
}
if bytes {
for (index, char) in contents.chars().into_iter().enumerate() {
if index < count {
2022-12-20 18:35:45 -05:00
print!("{char}");
2022-12-20 12:05:21 -05:00
} else {
println!();
return;
}
}
println!();
} else {
for (index, line) in contents.lines().into_iter().enumerate() {
if index < count {
2022-12-20 18:35:45 -05:00
println!("{line}");
2022-12-20 12:05:21 -05:00
} else {
return;
}
}
}
}