use bstr::{BStr, BString, ByteVec}; use winnow::{ combinator::{preceded, repeat, terminated}, error::{AddContext, ParserError, StrContext}, prelude::*, token::{take_till1, take_until0, take_while}, Parser, }; use crate::ByteSlice; pub(crate) const NL: &[u8] = b"\n"; pub(crate) const SPACE: &[u8] = b" "; const SPACE_OR_NL: &[u8] = b" \n"; pub(crate) fn any_header_field_multi_line<'a, E: ParserError<&'a [u8]> + AddContext<&'a [u8], StrContext>>( i: &mut &'a [u8], ) -> PResult<(&'a [u8], BString), E> { ( terminated(take_till1(SPACE_OR_NL), SPACE), ( take_till1(NL), NL, repeat(1.., terminated((SPACE, take_until0(NL)), NL)).map(|()| ()), ) .recognize() .map(|o: &[u8]| { let bytes = o.as_bstr(); let mut out = BString::from(Vec::with_capacity(bytes.len())); let mut lines = bytes.lines(); out.push_str(lines.next().expect("first line")); for line in lines { out.push(b'\n'); out.push_str(&line[1..]); // cut leading space } out }), ) .context(StrContext::Expected("name ".into())) .parse_next(i) } pub(crate) fn header_field<'a, T, E: ParserError<&'a [u8]>>( i: &mut &'a [u8], name: &'static [u8], parse_value: impl Parser<&'a [u8], T, E>, ) -> PResult { terminated(preceded(terminated(name, SPACE), parse_value), NL).parse_next(i) } pub(crate) fn any_header_field<'a, T, E: ParserError<&'a [u8]>>( i: &mut &'a [u8], parse_value: impl Parser<&'a [u8], T, E>, ) -> PResult<(&'a [u8], T), E> { terminated((terminated(take_till1(SPACE_OR_NL), SPACE), parse_value), NL).parse_next(i) } fn is_hex_digit_lc(b: u8) -> bool { matches!(b, b'0'..=b'9' | b'a'..=b'f') } pub fn hex_hash<'a, E: ParserError<&'a [u8]>>(i: &mut &'a [u8]) -> PResult<&'a BStr, E> { take_while( gix_hash::Kind::shortest().len_in_hex()..=gix_hash::Kind::longest().len_in_hex(), is_hex_digit_lc, ) .map(ByteSlice::as_bstr) .parse_next(i) } pub(crate) fn signature<'a, E: ParserError<&'a [u8]> + AddContext<&'a [u8], StrContext>>( i: &mut &'a [u8], ) -> PResult, E> { gix_actor::signature::decode(i) }