2023-07-20 14:26:59 +00:00
|
|
|
use nom::{
|
|
|
|
IResult,
|
2023-07-22 11:51:19 +00:00
|
|
|
branch::alt,
|
2023-07-20 14:26:59 +00:00
|
|
|
bytes::complete::{tag_no_case, tag, take_while1},
|
|
|
|
character::complete::space0,
|
2023-07-22 14:46:36 +00:00
|
|
|
combinator::{not, map},
|
2023-07-22 11:51:19 +00:00
|
|
|
multi::many0,
|
2023-07-22 14:46:36 +00:00
|
|
|
sequence::{pair, preceded, terminated, tuple},
|
2023-07-20 14:26:59 +00:00
|
|
|
};
|
2023-07-22 11:51:19 +00:00
|
|
|
use crate::text::whitespace::{foldable_line, obs_crlf};
|
2023-07-20 14:26:59 +00:00
|
|
|
use crate::text::misc_token::{Unstructured, unstructured};
|
2023-07-22 14:46:36 +00:00
|
|
|
use crate::text::boundary::boundary;
|
2023-07-20 14:26:59 +00:00
|
|
|
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
|
|
pub enum CompField<'a, T> {
|
|
|
|
Known(T),
|
|
|
|
Unknown(&'a [u8], Unstructured<'a>),
|
|
|
|
Bad(&'a [u8]),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
|
|
pub struct CompFieldList<'a, T>(pub Vec<CompField<'a, T>>);
|
|
|
|
impl<'a, T> CompFieldList<'a, T> {
|
|
|
|
pub fn known(self) -> Vec<T> {
|
|
|
|
self.0.into_iter().map(|v| match v {
|
|
|
|
CompField::Known(f) => Some(f),
|
|
|
|
_ => None,
|
|
|
|
}).flatten().collect()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-22 11:51:19 +00:00
|
|
|
pub fn header<'a, T>(fx: impl Fn(&'a [u8]) -> IResult<&[u8], T> + Copy)
|
|
|
|
-> impl Fn(&'a [u8]) -> IResult<&[u8], CompFieldList<T>>
|
|
|
|
{
|
|
|
|
move |input| map(terminated(many0(alt((
|
|
|
|
map(fx, CompField::Known),
|
|
|
|
map(opt_field, |(k,v)| CompField::Unknown(k,v)),
|
|
|
|
map(foldable_line, CompField::Bad),
|
|
|
|
))), obs_crlf), CompFieldList)(input)
|
|
|
|
}
|
|
|
|
|
2023-07-22 14:46:36 +00:00
|
|
|
pub fn header_in_boundaries<'a, T>(bound: &'a [u8], fx: impl Fn(&'a [u8]) -> IResult<&[u8], T> + Copy)
|
|
|
|
-> impl Fn(&'a [u8]) -> IResult<&[u8], CompFieldList<T>>
|
|
|
|
{
|
|
|
|
move |input| map(terminated(many0(preceded(
|
|
|
|
not(boundary(bound)),
|
|
|
|
alt((
|
|
|
|
map(fx, CompField::Known),
|
|
|
|
map(opt_field, |(k,v)| CompField::Unknown(k,v)),
|
|
|
|
map(foldable_line, CompField::Bad),
|
|
|
|
)))), obs_crlf), CompFieldList)(input)
|
|
|
|
}
|
|
|
|
|
2023-07-20 14:26:59 +00:00
|
|
|
pub fn field_name<'a>(name: &'static [u8]) -> impl Fn(&'a [u8]) -> IResult<&'a [u8], &'a [u8]> {
|
|
|
|
move |input| {
|
|
|
|
terminated(
|
|
|
|
tag_no_case(name),
|
|
|
|
tuple((space0, tag(b":"), space0)),
|
|
|
|
)(input)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Optional field
|
|
|
|
///
|
|
|
|
/// ```abnf
|
|
|
|
/// field = field-name ":" unstructured CRLF
|
|
|
|
/// field-name = 1*ftext
|
|
|
|
/// ftext = %d33-57 / ; Printable US-ASCII
|
|
|
|
/// %d59-126 ; characters not including
|
|
|
|
/// ; ":".
|
|
|
|
/// ```
|
|
|
|
pub fn opt_field(input: &[u8]) -> IResult<&[u8], (&[u8], Unstructured)> {
|
2023-07-22 11:51:19 +00:00
|
|
|
terminated(
|
|
|
|
pair(
|
|
|
|
terminated(
|
|
|
|
take_while1(|c| c >= 0x21 && c <= 0x7E && c != 0x3A),
|
|
|
|
tuple((space0, tag(b":"), space0)),
|
|
|
|
),
|
|
|
|
unstructured,
|
|
|
|
), obs_crlf)(input)
|
2023-07-20 14:26:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|