use std::hash;
use crate::{Kind, ObjectId};
#[cfg(feature = "sha1")]
use crate::{EMPTY_BLOB_SHA1, EMPTY_TREE_SHA1, SIZE_OF_SHA1_DIGEST};
#[cfg(feature = "sha256")]
use crate::{EMPTY_BLOB_SHA256, EMPTY_TREE_SHA256, SIZE_OF_SHA256_DIGEST};
#[derive(PartialEq, Eq, Ord, PartialOrd)]
#[repr(transparent)]
#[allow(non_camel_case_types)]
#[cfg_attr(feature = "serde", derive(serde::Serialize))]
pub struct oid {
bytes: [u8],
}
#[allow(clippy::derived_hash_with_manual_eq)]
impl hash::Hash for oid {
fn hash<H: hash::Hasher>(&self, state: &mut H) {
state.write(self.as_bytes());
}
}
#[derive(PartialEq, Eq, Hash, Ord, PartialOrd)]
pub struct HexDisplay<'a> {
inner: &'a oid,
hex_len: usize,
}
impl std::fmt::Display for HexDisplay<'_> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut hex = Kind::hex_buf();
let hex = self.inner.hex_to_buf(hex.as_mut());
let max_len = hex.len();
f.write_str(&hex[..self.hex_len.min(max_len)])
}
}
impl std::fmt::Debug for oid {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(
f,
"{}({})",
match self.kind() {
#[cfg(feature = "sha1")]
Kind::Sha1 => "Sha1",
#[cfg(feature = "sha256")]
Kind::Sha256 => "Sha256",
},
self.to_hex(),
)
}
}
#[allow(missing_docs)]
#[derive(Debug, thiserror::Error)]
pub enum Error {
#[error("Cannot instantiate git hash from a digest of length {0}")]
InvalidByteSliceLength(usize),
}
impl oid {
#[inline]
pub fn try_from_bytes(digest: &[u8]) -> Result<&Self, Error> {
match digest.len() {
#[cfg(feature = "sha1")]
SIZE_OF_SHA1_DIGEST => Ok(
#[allow(unsafe_code)]
unsafe {
&*(std::ptr::from_ref::<[u8]>(digest) as *const oid)
},
),
#[cfg(feature = "sha256")]
SIZE_OF_SHA256_DIGEST => Ok(
#[allow(unsafe_code)]
unsafe {
&*(std::ptr::from_ref::<[u8]>(digest) as *const oid)
},
),
len => Err(Error::InvalidByteSliceLength(len)),
}
}
pub fn from_bytes_unchecked(value: &[u8]) -> &Self {
Self::from_bytes(value)
}
pub(crate) fn from_bytes(value: &[u8]) -> &Self {
#[allow(unsafe_code)]
unsafe {
&*(std::ptr::from_ref::<[u8]>(value) as *const oid)
}
}
}
impl oid {
#[inline]
pub fn kind(&self) -> Kind {
Kind::from_len_in_bytes(self.bytes.len())
}
#[inline]
pub fn first_byte(&self) -> u8 {
self.bytes[0]
}
#[inline]
pub fn as_bytes(&self) -> &[u8] {
&self.bytes
}
#[inline]
pub fn to_hex_with_len(&self, len: usize) -> HexDisplay<'_> {
HexDisplay {
inner: self,
hex_len: len,
}
}
#[inline]
pub fn to_hex(&self) -> HexDisplay<'_> {
HexDisplay {
inner: self,
hex_len: self.bytes.len() * 2,
}
}
#[inline]
#[must_use]
pub fn hex_to_buf<'a>(&self, buf: &'a mut [u8]) -> &'a mut str {
let num_hex_bytes = self.bytes.len() * 2;
faster_hex::hex_encode(&self.bytes, &mut buf[..num_hex_bytes])
.expect("buffer size must be at least twice the hash digest size in bytes")
}
#[inline]
pub fn write_hex_to(&self, out: &mut dyn std::io::Write) -> std::io::Result<()> {
let mut hex = Kind::hex_buf();
let hex_len = self.hex_to_buf(&mut hex).len();
out.write_all(&hex[..hex_len])
}
#[inline]
#[doc(alias = "is_zero", alias = "git2")]
pub fn is_null(&self) -> bool {
match self.kind() {
#[cfg(feature = "sha1")]
Kind::Sha1 => &self.bytes == oid::null_sha1().as_bytes(),
#[cfg(feature = "sha256")]
Kind::Sha256 => &self.bytes == oid::null_sha256().as_bytes(),
}
}
#[inline]
pub fn is_empty_blob(&self) -> bool {
match self.kind() {
#[cfg(feature = "sha1")]
Kind::Sha1 => &self.bytes == oid::empty_blob_sha1().as_bytes(),
#[cfg(feature = "sha256")]
Kind::Sha256 => &self.bytes == oid::empty_blob_sha256().as_bytes(),
}
}
#[inline]
pub fn is_empty_tree(&self) -> bool {
match self.kind() {
#[cfg(feature = "sha1")]
Kind::Sha1 => &self.bytes == oid::empty_tree_sha1().as_bytes(),
#[cfg(feature = "sha256")]
Kind::Sha256 => &self.bytes == oid::empty_tree_sha256().as_bytes(),
}
}
}
impl oid {
#[inline]
#[cfg(feature = "sha1")]
pub(crate) fn null_sha1() -> &'static Self {
oid::from_bytes([0u8; SIZE_OF_SHA1_DIGEST].as_ref())
}
#[inline]
#[cfg(feature = "sha256")]
pub(crate) fn null_sha256() -> &'static Self {
oid::from_bytes([0u8; SIZE_OF_SHA256_DIGEST].as_ref())
}
#[inline]
#[cfg(feature = "sha1")]
pub(crate) fn empty_blob_sha1() -> &'static Self {
oid::from_bytes(EMPTY_BLOB_SHA1)
}
#[inline]
#[cfg(feature = "sha256")]
pub(crate) fn empty_blob_sha256() -> &'static Self {
oid::from_bytes(EMPTY_BLOB_SHA256)
}
#[inline]
#[cfg(feature = "sha1")]
pub(crate) fn empty_tree_sha1() -> &'static Self {
oid::from_bytes(EMPTY_TREE_SHA1)
}
#[inline]
#[cfg(feature = "sha256")]
pub(crate) fn empty_tree_sha256() -> &'static Self {
oid::from_bytes(EMPTY_TREE_SHA256)
}
}
impl AsRef<oid> for &oid {
fn as_ref(&self) -> &oid {
self
}
}
impl<'a> TryFrom<&'a [u8]> for &'a oid {
type Error = Error;
fn try_from(value: &'a [u8]) -> Result<Self, Self::Error> {
oid::try_from_bytes(value)
}
}
impl ToOwned for oid {
type Owned = ObjectId;
fn to_owned(&self) -> Self::Owned {
match self.kind() {
#[cfg(feature = "sha1")]
Kind::Sha1 => ObjectId::Sha1(self.bytes.try_into().expect("no bug in hash detection")),
#[cfg(feature = "sha256")]
Kind::Sha256 => ObjectId::Sha256(self.bytes.try_into().expect("no bug in hash detection")),
}
}
}
#[cfg(feature = "sha1")]
impl<'a> From<&'a [u8; SIZE_OF_SHA1_DIGEST]> for &'a oid {
fn from(v: &'a [u8; SIZE_OF_SHA1_DIGEST]) -> Self {
oid::from_bytes(v.as_ref())
}
}
#[cfg(feature = "sha256")]
impl<'a> From<&'a [u8; SIZE_OF_SHA256_DIGEST]> for &'a oid {
fn from(v: &'a [u8; SIZE_OF_SHA256_DIGEST]) -> Self {
oid::from_bytes(v.as_ref())
}
}
impl std::fmt::Display for &oid {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let mut buf = Kind::hex_buf();
f.write_str(self.hex_to_buf(&mut buf))
}
}
impl PartialEq<ObjectId> for &oid {
fn eq(&self, other: &ObjectId) -> bool {
*self == other.as_ref()
}
}
#[cfg(feature = "serde")]
impl<'de: 'a, 'a> serde::Deserialize<'de> for &'a oid {
fn deserialize<D>(deserializer: D) -> Result<Self, <D as serde::Deserializer<'de>>::Error>
where
D: serde::Deserializer<'de>,
{
struct __Visitor<'de: 'a, 'a> {
marker: std::marker::PhantomData<&'a oid>,
lifetime: std::marker::PhantomData<&'de ()>,
}
impl<'de: 'a, 'a> serde::de::Visitor<'de> for __Visitor<'de, 'a> {
type Value = &'a oid;
fn expecting(&self, __formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
std::fmt::Formatter::write_str(__formatter, "tuple struct Digest")
}
#[inline]
fn visit_newtype_struct<__E>(self, __e: __E) -> std::result::Result<Self::Value, __E::Error>
where
__E: serde::Deserializer<'de>,
{
let __field0: &'a [u8] = match <&'a [u8] as serde::Deserialize>::deserialize(__e) {
Ok(__val) => __val,
Err(__err) => {
return Err(__err);
}
};
Ok(oid::try_from_bytes(__field0).expect("hash of known length"))
}
#[inline]
fn visit_seq<__A>(self, mut __seq: __A) -> std::result::Result<Self::Value, __A::Error>
where
__A: serde::de::SeqAccess<'de>,
{
let __field0 = match match serde::de::SeqAccess::next_element::<&'a [u8]>(&mut __seq) {
Ok(__val) => __val,
Err(__err) => {
return Err(__err);
}
} {
Some(__value) => __value,
None => {
return Err(serde::de::Error::invalid_length(
0usize,
&"tuple struct Digest with 1 element",
));
}
};
Ok(oid::try_from_bytes(__field0).expect("hash of known length"))
}
}
serde::Deserializer::deserialize_newtype_struct(
deserializer,
"Digest",
__Visitor {
marker: std::marker::PhantomData::<&'a oid>,
lifetime: std::marker::PhantomData,
},
)
}
}