2023-06-13 13:12:16 +00:00
|
|
|
use nom::{
|
|
|
|
IResult,
|
|
|
|
branch::alt,
|
|
|
|
bytes::complete::{take_while, tag},
|
|
|
|
combinator::opt,
|
2023-06-20 13:56:06 +00:00
|
|
|
multi::many1,
|
2023-06-13 13:12:16 +00:00
|
|
|
sequence::{delimited, pair, tuple},
|
|
|
|
};
|
|
|
|
|
2023-06-20 13:56:06 +00:00
|
|
|
use crate::fragments::lazy;
|
2023-06-19 15:25:16 +00:00
|
|
|
use crate::fragments::whitespace::cfws;
|
|
|
|
use crate::fragments::words::dot_atom_text;
|
|
|
|
use crate::fragments::mailbox::is_dtext;
|
2023-06-20 13:56:06 +00:00
|
|
|
use crate::fragments::model::{MessageId, MessageIdList};
|
|
|
|
use crate::error::IMFError;
|
|
|
|
|
2023-06-22 08:48:07 +00:00
|
|
|
impl<'a> TryFrom<&'a lazy::Identifier<'a>> for MessageId<'a> {
|
2023-06-20 13:56:06 +00:00
|
|
|
type Error = IMFError<'a>;
|
|
|
|
|
2023-06-22 08:48:07 +00:00
|
|
|
fn try_from(id: &'a lazy::Identifier<'a>) -> Result<Self, Self::Error> {
|
2023-06-20 13:56:06 +00:00
|
|
|
msg_id(id.0)
|
|
|
|
.map(|(_, i)| i)
|
|
|
|
.map_err(|e| IMFError::MessageID(e))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-22 08:48:07 +00:00
|
|
|
impl<'a> TryFrom<&'a lazy::IdentifierList<'a>> for MessageIdList<'a> {
|
2023-06-20 13:56:06 +00:00
|
|
|
type Error = IMFError<'a>;
|
|
|
|
|
2023-06-22 08:48:07 +00:00
|
|
|
fn try_from(id: &'a lazy::IdentifierList<'a>) -> Result<Self, Self::Error> {
|
2023-06-20 13:56:06 +00:00
|
|
|
many1(msg_id)(id.0)
|
|
|
|
.map(|(_, i)| i)
|
|
|
|
.map_err(|e| IMFError::MessageIDList(e))
|
|
|
|
}
|
|
|
|
}
|
2023-06-13 13:12:16 +00:00
|
|
|
|
|
|
|
/// Message identifier
|
|
|
|
///
|
|
|
|
/// ```abnf
|
|
|
|
/// msg-id = [CFWS] "<" id-left "@" id-right ">" [CFWS]
|
|
|
|
/// ```
|
|
|
|
pub fn msg_id(input: &str) -> IResult<&str, MessageId> {
|
|
|
|
let (input, (left, _, right)) = delimited(
|
|
|
|
pair(opt(cfws), tag("<")),
|
|
|
|
tuple((id_left, tag("@"), id_right)),
|
|
|
|
pair(tag(">"), opt(cfws)),
|
|
|
|
)(input)?;
|
|
|
|
Ok((input, MessageId{ left, right }))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Missing obsolete
|
|
|
|
fn id_left(input: &str) -> IResult<&str, &str> {
|
|
|
|
dot_atom_text(input)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Missing obsolete
|
|
|
|
fn id_right(input: &str) -> IResult<&str, &str> {
|
|
|
|
alt((dot_atom_text, no_fold_litteral))(input)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn no_fold_litteral(input: &str) -> IResult<&str, &str> {
|
|
|
|
delimited(tag("["), take_while(is_dtext), tag("]"))(input)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_msg_id() {
|
|
|
|
assert_eq!(
|
|
|
|
msg_id("<5678.21-Nov-1997@example.com>"),
|
|
|
|
Ok(("", MessageId{left: "5678.21-Nov-1997", right: "example.com"})),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|