be more tolerant on field names
This commit is contained in:
parent
9f512ddc0e
commit
0ff9466b5f
2 changed files with 23 additions and 15 deletions
|
@ -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)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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é
|
||||||
|
|
Loading…
Reference in a new issue