use crate::{
data,
pack::{self, data::output},
};
use git_hash::ObjectId;
use std::io::Write;
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde1", derive(serde::Serialize, serde::Deserialize))]
pub enum Kind {
Base,
DeltaRef {
nth_before: usize,
},
DeltaOid {
id: ObjectId,
},
}
#[allow(missing_docs)]
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("{0}")]
ZlibDeflate(#[from] std::io::Error),
}
impl output::Entry {
pub fn from_data(oid: impl Into<ObjectId>, obj: &data::Object<'_>) -> Result<Self, Error> {
Ok(output::Entry {
id: oid.into(),
object_kind: obj.kind,
entry_kind: Kind::Base,
decompressed_size: obj.data.len(),
compressed_data: {
let mut out = crate::zlib::stream::deflate::Write::new(Vec::new());
if let Err(err) = std::io::copy(&mut &*obj.data, &mut out) {
match err.kind() {
std::io::ErrorKind::Other => return Err(Error::ZlibDeflate(err)),
err => unreachable!("Should never see other errors than zlib, but got {:?}", err,),
}
};
out.flush()?;
out.into_inner()
},
})
}
pub fn to_entry_header(
&self,
version: pack::data::Version,
index_to_pack: impl FnOnce(usize) -> u64,
) -> pack::data::entry::Header {
use pack::data;
assert!(
matches!(version, data::Version::V2),
"we can only write V2 pack entries for now"
);
use Kind::*;
match self.entry_kind {
Base => {
use git_object::Kind::*;
match self.object_kind {
Tree => data::entry::Header::Tree,
Blob => data::entry::Header::Blob,
Commit => data::entry::Header::Commit,
Tag => data::entry::Header::Tag,
}
}
DeltaOid { id } => data::entry::Header::RefDelta { base_id: id.to_owned() },
DeltaRef { nth_before } => data::entry::Header::OfsDelta {
base_distance: index_to_pack(nth_before),
},
}
}
}