be more tolerant on field names

This commit is contained in:
Quentin 2023-06-16 11:41:42 +02:00
parent 9f512ddc0e
commit 0ff9466b5f
Signed by: quentin
GPG key ID: E9602264D639FF68
2 changed files with 23 additions and 15 deletions

View file

@ -1,7 +1,7 @@
use nom::{ use nom::{
IResult, IResult,
branch::alt, branch::alt,
bytes::complete::{is_not, take_while1, take_while, tag}, bytes::complete::{is_not, take_while1, take_while, tag, tag_no_case},
character::complete::space0, character::complete::space0,
combinator::{map, opt, recognize}, combinator::{map, opt, recognize},
multi::{many0, many1, fold_many0, separated_list1}, multi::{many0, many1, fold_many0, separated_list1},
@ -185,36 +185,36 @@ pub fn header_field(input: &str) -> IResult<&str, HeaderField> {
// 3.6.1. The Origination Date Field // 3.6.1. The Origination Date Field
fn date(input: &str) -> IResult<&str, HeaderField> { fn date(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Date:"), space0), datetime::section)(input)?; let (input, body) = preceded(field_name_tag("Date"), datetime::section)(input)?;
Ok((input, HeaderField::Date(body))) Ok((input, HeaderField::Date(body)))
} }
// 3.6.2. Originator Fields // 3.6.2. Originator Fields
fn from(input: &str) -> IResult<&str, HeaderField> { fn from(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("From:"), space0), mailbox_list)(input)?; let (input, body) = preceded(field_name_tag("From"), mailbox_list)(input)?;
Ok((input, HeaderField::From(body))) Ok((input, HeaderField::From(body)))
} }
fn sender(input: &str) -> IResult<&str, HeaderField> { fn sender(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Sender:"), space0), mailbox)(input)?; let (input, body) = preceded(field_name_tag("Sender"), mailbox)(input)?;
Ok((input, HeaderField::Sender(body))) Ok((input, HeaderField::Sender(body)))
} }
fn reply_to(input: &str) -> IResult<&str, HeaderField> { fn reply_to(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Reply-To:"), space0), address_list)(input)?; let (input, body) = preceded(field_name_tag("Reply-To"), address_list)(input)?;
Ok((input, HeaderField::ReplyTo(body))) Ok((input, HeaderField::ReplyTo(body)))
} }
// 3.6.3. Destination Address Fields // 3.6.3. Destination Address Fields
fn to(input: &str) -> IResult<&str, HeaderField> { fn to(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("To:"), space0), address_list)(input)?; let (input, body) = preceded(field_name_tag("To"), address_list)(input)?;
Ok((input, HeaderField::To(body))) Ok((input, HeaderField::To(body)))
} }
fn cc(input: &str) -> IResult<&str, HeaderField> { fn cc(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Cc:"), space0), address_list)(input)?; let (input, body) = preceded(field_name_tag("Cc"), address_list)(input)?;
Ok((input, HeaderField::Cc(body))) Ok((input, HeaderField::Cc(body)))
} }
fn bcc(input: &str) -> IResult<&str, HeaderField> { fn bcc(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded( let (input, body) = preceded(
pair(tag("Bcc:"), space0), field_name_tag("Bcc"),
opt(alt((address_list, address_list_cfws))), opt(alt((address_list, address_list_cfws))),
)(input)?; )(input)?;
@ -223,35 +223,41 @@ fn bcc(input: &str) -> IResult<&str, HeaderField> {
// 3.6.4. Identification Fields // 3.6.4. Identification Fields
fn msg_id_field(input: &str) -> IResult<&str, HeaderField> { fn msg_id_field(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Message-ID:"), space0), msg_id)(input)?; let (input, body) = preceded(field_name_tag("Message-ID"), msg_id)(input)?;
Ok((input, HeaderField::MessageID(body))) Ok((input, HeaderField::MessageID(body)))
} }
fn in_reply_to(input: &str) -> IResult<&str, HeaderField> { fn in_reply_to(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("In-Reply-To:"), space0), many1(msg_id))(input)?; let (input, body) = preceded(field_name_tag("In-Reply-To"), many1(msg_id))(input)?;
Ok((input, HeaderField::InReplyTo(body))) Ok((input, HeaderField::InReplyTo(body)))
} }
fn references(input: &str) -> IResult<&str, HeaderField> { fn references(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("References:"), space0), many1(msg_id))(input)?; let (input, body) = preceded(field_name_tag("References"), many1(msg_id))(input)?;
Ok((input, HeaderField::References(body))) Ok((input, HeaderField::References(body)))
} }
// 3.6.5. Informational Fields // 3.6.5. Informational Fields
fn subject(input: &str) -> IResult<&str, HeaderField> { fn subject(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Subject:"), space0), unstructured)(input)?; let (input, body) = preceded(field_name_tag("Subject"), unstructured)(input)?;
Ok((input, HeaderField::Subject(body))) Ok((input, HeaderField::Subject(body)))
} }
fn comments(input: &str) -> IResult<&str, HeaderField> { fn comments(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded(pair(tag("Comments:"), space0), unstructured)(input)?; let (input, body) = preceded(field_name_tag("Comments"), unstructured)(input)?;
Ok((input, HeaderField::Comments(body))) Ok((input, HeaderField::Comments(body)))
} }
fn keywords(input: &str) -> IResult<&str, HeaderField> { fn keywords(input: &str) -> IResult<&str, HeaderField> {
let (input, body) = preceded( let (input, body) = preceded(
pair(tag("Keywords:"), space0), field_name_tag("Keywords"),
separated_list1(tag(","), phrase), separated_list1(tag(","), phrase),
)(input)?; )(input)?;
Ok((input, HeaderField::Keywords(body))) Ok((input, HeaderField::Keywords(body)))
} }
fn field_name_tag(field_name: &str) -> impl FnMut(&str) -> IResult<&str, &str> + '_ {
move |input: &str| {
recognize(tuple((tag_no_case(field_name), space0, tag(":"), space0)))(input)
}
}
// 3.6.6 Resent fields // 3.6.6 Resent fields
// Not implemented // Not implemented
@ -288,7 +294,7 @@ fn unknown_field(input: &str) -> IResult<&str, HeaderField> {
fn field_name(input: &str) -> IResult<&str, &str> { fn field_name(input: &str) -> IResult<&str, &str> {
terminated( terminated(
take_while1(|c| c >= '\x21' && c <= '\x7E' && c != '\x3A'), take_while1(|c| c >= '\x21' && c <= '\x7E' && c != '\x3A'),
pair(tag(":"), space0) tuple((space0, tag(":"), space0))
)(input) )(input)
} }

View file

@ -19,6 +19,8 @@ Bcc: (hidden)
Subject: Re: Saying Hello Subject: Re: Saying Hello
Comments: A simple message Comments: A simple message
Comments: Not that complicated Comments: Not that complicated
comments : not valid but should be accepted
by the parser.
Keywords: hello, world Keywords: hello, world
Héron: Raté Héron: Raté
Raté raté Raté raté