use std::io::{Read, Write};
use std::time::Instant;
use anyhow::Result;
use crate::checksum;
use crate::cli::{GlobalArgs, HashArgs};
use crate::ndjson_types::HashOutput;
use crate::output::NdjsonWriter;
#[tracing::instrument(skip_all, fields(command = "hash"))]
pub fn cmd_hash(
args: &HashArgs,
global: &GlobalArgs,
stdin: impl Read,
writer: &mut NdjsonWriter<impl Write>,
) -> Result<()> {
let start = Instant::now();
let workspace = global.resolve_workspace()?;
if args.stdin {
let mut reader = std::io::BufReader::with_capacity(crate::constants::BUF_CAPACITY, stdin);
let hash = checksum::hash_reader(&mut reader)?;
writer.write_event(&HashOutput {
r#type: "hash",
path: None,
source: Some("stdin"),
algorithm: "blake3",
value: hash,
bytes: None,
verified: None,
elapsed_ms: start.elapsed().as_millis() as u64,
})?;
return Ok(());
}
for path in &args.paths {
let path = crate::path_safety::validate_path(path, &workspace)?;
if path.is_dir() && !args.recursive {
continue;
}
if path.is_file() {
let path_str = path.display().to_string();
let hash = checksum::hash_file(&path, global.effective_max_filesize())?;
let bytes = std::fs::metadata(&path)?.len();
if let Some(ref expected) = args.verify {
let verified = &hash == expected;
writer.write_event(&HashOutput {
r#type: "hash",
path: Some(path_str.clone()),
source: None,
algorithm: "blake3",
value: hash,
bytes: Some(bytes),
verified: Some(verified),
elapsed_ms: start.elapsed().as_millis() as u64,
})?;
if !verified {
return Err(crate::error::AtomwriteError::ChecksumVerifyFailed {
path: path.clone(),
expected: expected.clone(),
}
.into());
}
} else {
writer.write_event(&HashOutput {
r#type: "hash",
path: Some(path_str),
source: None,
algorithm: "blake3",
value: hash,
bytes: Some(bytes),
verified: None,
elapsed_ms: start.elapsed().as_millis() as u64,
})?;
}
}
}
Ok(())
}