use anyhow::{Context, Result}; use std::os::unix::fs::MetadataExt; use std::path::PathBuf; use crate::kind::Kind; use crate::object::{write_blob, write_object, TreeObject}; pub fn write_tree(repo_path: &PathBuf, path: &PathBuf) -> Result<[u8; 20]> { let mut entries = Vec::new(); let files = std::fs::read_dir(path)?; for file in files { let file = file?; let file_type = file.file_type()?; let file_name = file.file_name(); let file_path = file.path(); let hash: [u8; 20]; let kind; if file_type.is_dir() { hash = write_tree(repo_path, &file_path).context("could not write_tree of subtree")?; kind = Kind::Tree; } else { hash = write_blob(repo_path, &file_path).context(format!( "could not write object {:?}", file_path.file_name() ))?; kind = Kind::Blob(file_path.metadata()?.mode() & 0o111 != 0); } entries.push(TreeObject { mode: kind.to_mode().to_string(), kind, name: file_name.into_string().unwrap(), hash, }) } entries.sort_by(|a, b| a.name.cmp(&b.name)); let mut out: Vec = Vec::new(); for entry in &entries { out.extend_from_slice(entry.mode.as_bytes()); out.push(b' '); out.extend_from_slice(entry.name.as_bytes()); out.push(0); out.extend_from_slice(&entry.hash); } write_object(repo_path, Kind::Tree, &out).context("Write") }