use std::ops::Range;
use bstr::{BStr, BString, ByteSlice};
use winnow::prelude::*;
use crate::{Commit, CommitRef, TagRef};
pub const SIGNATURE_FIELD_NAME: &str = "gpgsig";
mod decode;
pub mod message;
#[derive(PartialEq, Eq, Debug, Hash, Ord, PartialOrd, Clone, Copy)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct MessageRef<'a> {
#[cfg_attr(feature = "serde", serde(borrow))]
pub title: &'a BStr,
pub body: Option<&'a BStr>,
}
#[derive(PartialEq, Eq, Debug, Hash, Clone)]
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
pub struct SignedData<'a> {
data: &'a [u8],
signature_range: Range<usize>,
}
impl SignedData<'_> {
pub fn to_bstring(&self) -> BString {
let mut buf = BString::from(&self.data[..self.signature_range.start]);
buf.extend_from_slice(&self.data[self.signature_range.end..]);
buf
}
}
impl From<SignedData<'_>> for BString {
fn from(value: SignedData<'_>) -> Self {
value.to_bstring()
}
}
pub mod ref_iter;
mod write;
impl<'a> CommitRef<'a> {
pub fn from_bytes(mut data: &'a [u8]) -> Result<CommitRef<'a>, crate::decode::Error> {
let input = &mut data;
match decode::commit.parse_next(input) {
Ok(tag) => Ok(tag),
Err(err) => Err(crate::decode::Error::with_err(err, input)),
}
}
}
impl<'a> CommitRef<'a> {
pub fn tree(&self) -> gix_hash::ObjectId {
gix_hash::ObjectId::from_hex(self.tree).expect("prior validation of tree hash during parsing")
}
pub fn parents(&self) -> impl Iterator<Item = gix_hash::ObjectId> + '_ {
self.parents
.iter()
.map(|hex_hash| gix_hash::ObjectId::from_hex(hex_hash).expect("prior validation of hashes during parsing"))
}
pub fn extra_headers(&self) -> ExtraHeaders<impl Iterator<Item = (&BStr, &BStr)>> {
ExtraHeaders::new(self.extra_headers.iter().map(|(k, v)| (*k, v.as_ref())))
}
pub fn author(&self) -> gix_actor::SignatureRef<'a> {
self.author.trim()
}
pub fn committer(&self) -> gix_actor::SignatureRef<'a> {
self.committer.trim()
}
pub fn message(&self) -> MessageRef<'a> {
MessageRef::from_bytes(self.message)
}
pub fn time(&self) -> gix_date::Time {
self.committer.time.parse().unwrap_or_default()
}
}
impl CommitRef<'_> {
pub fn into_owned(self) -> Commit {
self.into()
}
pub fn to_owned(self) -> Commit {
self.clone().into()
}
}
impl Commit {
pub fn extra_headers(&self) -> ExtraHeaders<impl Iterator<Item = (&BStr, &BStr)>> {
ExtraHeaders::new(self.extra_headers.iter().map(|(k, v)| (k.as_bstr(), v.as_bstr())))
}
}
pub struct ExtraHeaders<I> {
inner: I,
}
impl<'a, I> ExtraHeaders<I>
where
I: Iterator<Item = (&'a BStr, &'a BStr)>,
{
pub fn new(iter: I) -> Self {
ExtraHeaders { inner: iter }
}
pub fn find(mut self, name: &str) -> Option<&'a BStr> {
self.inner
.find_map(move |(k, v)| if k == name.as_bytes().as_bstr() { Some(v) } else { None })
}
pub fn find_pos(self, name: &str) -> Option<usize> {
self.inner
.enumerate()
.find_map(|(pos, (field, _value))| (field == name).then_some(pos))
}
pub fn find_all(self, name: &'a str) -> impl Iterator<Item = &'a BStr> {
self.inner
.filter_map(move |(k, v)| if k == name.as_bytes().as_bstr() { Some(v) } else { None })
}
pub fn mergetags(self) -> impl Iterator<Item = Result<TagRef<'a>, crate::decode::Error>> {
self.find_all("mergetag").map(|b| TagRef::from_bytes(b))
}
pub fn pgp_signature(self) -> Option<&'a BStr> {
self.find(SIGNATURE_FIELD_NAME)
}
}