use std::iter::Peekable;
use git_features::hash;
use crate::data::input;
pub struct EntriesToBytesIter<I: Iterator, W> {
pub input: Peekable<I>,
output: W,
trailer: Option<git_hash::ObjectId>,
data_version: crate::data::Version,
num_entries: u32,
is_done: bool,
object_hash: git_hash::Kind,
}
impl<I, W> EntriesToBytesIter<I, W>
where
I: Iterator<Item = Result<input::Entry, input::Error>>,
W: std::io::Read + std::io::Write + std::io::Seek,
{
pub fn new(input: I, output: W, version: crate::data::Version, object_hash: git_hash::Kind) -> Self {
assert!(
matches!(version, crate::data::Version::V2),
"currently only pack version 2 can be written",
);
assert!(
matches!(object_hash, git_hash::Kind::Sha1),
"currently only Sha1 is supported, right now we don't know how other hashes are encoded",
);
EntriesToBytesIter {
input: input.peekable(),
output,
object_hash,
num_entries: 0,
trailer: None,
data_version: version,
is_done: false,
}
}
pub fn digest(&self) -> Option<git_hash::ObjectId> {
self.trailer
}
fn next_inner(&mut self, entry: input::Entry) -> Result<input::Entry, input::Error> {
if self.num_entries == 0 {
let header_bytes = crate::data::header::encode(self.data_version, 0);
self.output.write_all(&header_bytes[..])?;
}
self.num_entries += 1;
entry.header.write_to(entry.decompressed_size, &mut self.output)?;
std::io::copy(
&mut entry
.compressed
.as_deref()
.expect("caller must configure generator to keep compressed bytes"),
&mut self.output,
)?;
Ok(entry)
}
fn write_header_and_digest(&mut self, last_entry: Option<&mut input::Entry>) -> Result<(), input::Error> {
let header_bytes = crate::data::header::encode(self.data_version, self.num_entries);
let num_bytes_written = if last_entry.is_some() {
self.output.stream_position()?
} else {
header_bytes.len() as u64
};
self.output.rewind()?;
self.output.write_all(&header_bytes[..])?;
self.output.flush()?;
self.output.rewind()?;
let interrupt_never = std::sync::atomic::AtomicBool::new(false);
let digest = hash::bytes(
&mut self.output,
num_bytes_written as usize,
self.object_hash,
&mut git_features::progress::Discard,
&interrupt_never,
)?;
self.output.write_all(digest.as_slice())?;
self.output.flush()?;
self.is_done = true;
if let Some(last_entry) = last_entry {
last_entry.trailer = Some(digest);
}
self.trailer = Some(digest);
Ok(())
}
}
impl<I, W> Iterator for EntriesToBytesIter<I, W>
where
I: Iterator<Item = Result<input::Entry, input::Error>>,
W: std::io::Read + std::io::Write + std::io::Seek,
{
type Item = Result<input::Entry, input::Error>;
fn next(&mut self) -> Option<Self::Item> {
if self.is_done {
return None;
}
match self.input.next() {
Some(res) => Some(match res {
Ok(entry) => self.next_inner(entry).and_then(|mut entry| {
if self.input.peek().is_none() {
self.write_header_and_digest(Some(&mut entry)).map(|_| entry)
} else {
Ok(entry)
}
}),
Err(err) => {
self.is_done = true;
Err(err)
}
}),
None => match self.write_header_and_digest(None) {
Ok(_) => None,
Err(err) => Some(Err(err)),
},
}
}
fn size_hint(&self) -> (usize, Option<usize>) {
self.input.size_hint()
}
}