use std::io;
use bstr::{BString, ByteSlice};
use crate::{
encode::SPACE,
tree::{Entry, EntryRef},
Kind, Tree, TreeRef,
};
#[derive(Debug, thiserror::Error)]
#[allow(missing_docs)]
pub enum Error {
#[error("Newlines are invalid in file paths: {name:?}")]
NewlineInFilename { name: BString },
}
impl From<Error> for io::Error {
fn from(err: Error) -> Self {
io::Error::new(io::ErrorKind::Other, err)
}
}
impl crate::WriteTo for Tree {
fn write_to(&self, out: &mut dyn io::Write) -> io::Result<()> {
debug_assert_eq!(
&{
let mut entries_sorted = self.entries.clone();
entries_sorted.sort();
entries_sorted
},
&self.entries,
"entries for serialization must be sorted by filename"
);
let mut buf = Default::default();
for Entry { mode, filename, oid } in &self.entries {
out.write_all(mode.as_bytes(&mut buf))?;
out.write_all(SPACE)?;
if filename.find_byte(b'\n').is_some() {
return Err(Error::NewlineInFilename {
name: (*filename).to_owned(),
}
.into());
}
out.write_all(filename)?;
out.write_all(b"\0")?;
out.write_all(oid.as_bytes())?;
}
Ok(())
}
fn kind(&self) -> Kind {
Kind::Tree
}
fn size(&self) -> u64 {
let mut buf = Default::default();
self.entries
.iter()
.map(|Entry { mode, filename, oid }| {
(mode.as_bytes(&mut buf).len() + 1 + filename.len() + 1 + oid.as_bytes().len()) as u64
})
.sum()
}
}
impl<'a> crate::WriteTo for TreeRef<'a> {
fn write_to(&self, out: &mut dyn io::Write) -> io::Result<()> {
debug_assert_eq!(
&{
let mut entries_sorted = self.entries.clone();
entries_sorted.sort();
entries_sorted
},
&self.entries,
"entries for serialization must be sorted by filename"
);
let mut buf = Default::default();
for EntryRef { mode, filename, oid } in &self.entries {
out.write_all(mode.as_bytes(&mut buf))?;
out.write_all(SPACE)?;
if filename.find_byte(b'\n').is_some() {
return Err(Error::NewlineInFilename {
name: (*filename).to_owned(),
}
.into());
}
out.write_all(filename)?;
out.write_all(b"\0")?;
out.write_all(oid.as_bytes())?;
}
Ok(())
}
fn kind(&self) -> Kind {
Kind::Tree
}
fn size(&self) -> u64 {
let mut buf = Default::default();
self.entries
.iter()
.map(|EntryRef { mode, filename, oid }| {
(mode.as_bytes(&mut buf).len() + 1 + filename.len() + 1 + oid.as_bytes().len()) as u64
})
.sum()
}
}