eml-codec/src/model.rs

119 lines
2.8 KiB
Rust
Raw Normal View History

2023-06-08 19:43:32 +00:00
use std::collections::HashMap;
2023-06-12 09:54:15 +00:00
use chrono::{DateTime,FixedOffset,ParseError};
2023-06-13 11:31:43 +00:00
#[derive(Debug, PartialEq, Default)]
2023-06-12 09:54:15 +00:00
pub enum HeaderDate {
Parsed(DateTime<FixedOffset>),
Unknown(String, ParseError),
#[default]
None,
}
2023-06-08 19:43:32 +00:00
2023-06-13 07:00:40 +00:00
#[derive(Debug, PartialEq)]
pub struct AddrSpec {
pub local_part: String,
pub domain: String,
}
impl AddrSpec {
pub fn fully_qualified(&self) -> String {
format!("{}@{}", self.local_part, self.domain)
}
}
2023-06-13 07:18:36 +00:00
#[derive(Debug, PartialEq)]
2023-06-13 07:00:40 +00:00
pub struct MailboxRef {
2023-06-12 14:05:06 +00:00
// The actual "email address" like hello@example.com
2023-06-13 07:00:40 +00:00
pub addrspec: AddrSpec,
pub name: Option<String>,
}
impl From<AddrSpec> for MailboxRef {
fn from(addr: AddrSpec) -> Self {
MailboxRef {
name: None,
addrspec: addr,
}
}
2023-06-12 14:05:06 +00:00
}
2023-06-13 11:21:00 +00:00
#[derive(Debug, PartialEq)]
2023-06-13 07:00:40 +00:00
pub struct GroupRef {
pub name: String,
2023-06-13 10:25:00 +00:00
pub participants: Vec<MailboxRef>,
2023-06-12 14:05:06 +00:00
}
2023-06-13 11:21:00 +00:00
#[derive(Debug, PartialEq)]
2023-06-13 07:00:40 +00:00
pub enum AddressRef {
Single(MailboxRef),
Many(GroupRef),
2023-06-12 14:05:06 +00:00
}
2023-06-13 10:25:00 +00:00
impl From<MailboxRef> for AddressRef {
fn from(mx: MailboxRef) -> Self {
AddressRef::Single(mx)
}
}
impl From<GroupRef> for AddressRef {
fn from(grp: GroupRef) -> Self {
AddressRef::Many(grp)
}
}
2023-06-12 14:05:06 +00:00
2023-06-13 13:12:16 +00:00
#[derive(Debug, PartialEq)]
pub struct MessageId<'a> {
pub left: &'a str,
pub right: &'a str,
}
2023-06-12 14:05:06 +00:00
/// Permissive Header Section
///
/// This is a structure intended for parsing/decoding,
/// hence it's support cases where the email is considered
/// as invalid according to RFC5322 but for which we can
/// still extract some data.
2023-06-08 19:43:32 +00:00
#[derive(Debug, Default)]
2023-06-12 14:05:06 +00:00
pub struct PermissiveHeaderSection<'a> {
2023-06-13 12:44:41 +00:00
// 3.6.1. The Origination Date Field
pub date: HeaderDate,
// 3.6.2. Originator Fields
2023-06-13 07:00:40 +00:00
pub from: Vec<MailboxRef>,
pub sender: Option<MailboxRef>,
pub reply_to: Vec<AddressRef>,
2023-06-13 12:44:41 +00:00
// 3.6.3. Destination Address Fields
pub to: Vec<AddressRef>,
pub cc: Vec<AddressRef>,
pub bcc: Vec<AddressRef>,
2023-06-13 13:43:14 +00:00
// 3.6.4. Identification Fields
pub msg_id: Option<MessageId<'a>>,
pub in_reply_to: Vec<MessageId<'a>>,
pub references: Vec<MessageId<'a>>,
// Rest
2023-06-13 12:44:41 +00:00
pub subject: Option<String>,
2023-06-08 19:43:32 +00:00
pub optional: HashMap<&'a str, String>,
}
2023-06-12 09:54:15 +00:00
enum InvalidEmailErr {
NoUsableDate,
}
2023-06-12 14:05:06 +00:00
impl<'a> PermissiveHeaderSection<'a> {
/// Check validity of the email
///
/// Especially check that there is no missing fields,
/// or no unique fields declared multiple times.
///
/// See: https://www.rfc-editor.org/rfc/rfc5322#section-3.6
//@FIXME could be changed to a to_StrictHeaderSection call. All fixed errors would be returned in
// a vec of errors.
2023-06-12 09:54:15 +00:00
fn is_valid(&self) -> Result<(), InvalidEmailErr> {
match self.date {
HeaderDate::Parsed(_) => (),
_ => return Err(InvalidEmailErr::NoUsableDate),
};
Ok(())
}
}