aerogramme/aero-proto/src/imap/mail_view.rs

307 lines
9.8 KiB
Rust
Raw Normal View History

2024-01-04 19:54:21 +00:00
use std::num::NonZeroU32;
2024-01-06 10:07:53 +00:00
use anyhow::{anyhow, bail, Result};
2024-01-06 22:35:23 +00:00
use chrono::{naive::NaiveDate, DateTime as ChronoDateTime, Local, Offset, TimeZone, Utc};
2024-01-04 19:54:21 +00:00
2024-01-06 10:07:53 +00:00
use imap_codec::imap_types::core::NString;
2024-01-04 19:54:21 +00:00
use imap_codec::imap_types::datetime::DateTime;
use imap_codec::imap_types::fetch::{
MessageDataItem, MessageDataItemName, Section as FetchSection,
};
use imap_codec::imap_types::flag::Flag;
use imap_codec::imap_types::response::Data;
use eml_codec::{
imf,
part::{composite::Message, AnyPart},
};
2024-03-08 08:55:33 +00:00
use aero_collections::mail::query::QueryResult;
2024-01-06 10:07:53 +00:00
2024-01-04 19:54:21 +00:00
use crate::imap::attributes::AttributesProxy;
2024-01-05 09:05:30 +00:00
use crate::imap::flags;
use crate::imap::imf_view::ImfView;
2024-01-06 10:33:56 +00:00
use crate::imap::index::MailIndex;
2024-01-05 09:05:30 +00:00
use crate::imap::mime_view;
use crate::imap::response::Body;
2024-01-04 19:54:21 +00:00
pub struct MailView<'a> {
2024-01-08 10:13:13 +00:00
pub in_idx: &'a MailIndex<'a>,
2024-01-08 20:18:45 +00:00
pub query_result: &'a QueryResult,
2024-01-04 19:54:21 +00:00
pub content: FetchedMail<'a>,
}
impl<'a> MailView<'a> {
pub fn new(query_result: &'a QueryResult, in_idx: &'a MailIndex<'a>) -> Result<MailView<'a>> {
2024-01-05 17:59:19 +00:00
Ok(Self {
2024-01-06 10:07:53 +00:00
in_idx,
2024-01-05 17:59:19 +00:00
query_result,
content: match query_result {
QueryResult::FullResult { content, .. } => {
2024-01-06 10:33:56 +00:00
let (_, parsed) =
eml_codec::parse_message(&content).or(Err(anyhow!("Invalid mail body")))?;
2024-01-06 21:53:41 +00:00
FetchedMail::full_from_message(parsed)
2024-01-06 10:33:56 +00:00
}
2024-01-05 17:59:19 +00:00
QueryResult::PartialResult { metadata, .. } => {
2024-01-06 21:53:41 +00:00
let (_, parsed) = eml_codec::parse_message(&metadata.headers)
2024-01-06 10:33:56 +00:00
.or(Err(anyhow!("unable to parse email headers")))?;
2024-01-06 21:53:41 +00:00
FetchedMail::partial_from_message(parsed)
2024-01-05 17:59:19 +00:00
}
2024-01-06 10:07:53 +00:00
QueryResult::IndexResult { .. } => FetchedMail::IndexOnly,
2024-01-06 10:33:56 +00:00
},
2024-01-05 17:59:19 +00:00
})
}
2024-01-06 10:33:56 +00:00
pub fn imf(&self) -> Option<ImfView> {
2024-01-06 21:53:41 +00:00
self.content.as_imf().map(ImfView)
}
pub fn selected_mime(&'a self) -> Option<mime_view::SelectedMime<'a>> {
self.content.as_anypart().ok().map(mime_view::SelectedMime)
}
2024-01-06 17:01:44 +00:00
pub fn filter(&self, ap: &AttributesProxy) -> Result<(Body<'static>, SeenFlag)> {
let mut seen = SeenFlag::DoNothing;
let res_attrs = ap
.attrs
.iter()
.map(|attr| match attr {
MessageDataItemName::Uid => Ok(self.uid()),
MessageDataItemName::Flags => Ok(self.flags()),
MessageDataItemName::Rfc822Size => self.rfc_822_size(),
MessageDataItemName::Rfc822Header => self.rfc_822_header(),
MessageDataItemName::Rfc822Text => self.rfc_822_text(),
2024-01-08 14:07:02 +00:00
MessageDataItemName::Rfc822 => {
if self.is_not_yet_seen() {
seen = SeenFlag::MustAdd;
}
self.rfc822()
}
2024-01-06 17:01:44 +00:00
MessageDataItemName::Envelope => Ok(self.envelope()),
MessageDataItemName::Body => self.body(),
MessageDataItemName::BodyStructure => self.body_structure(),
MessageDataItemName::BodyExt {
section,
partial,
peek,
} => {
let (body, has_seen) = self.body_ext(section, partial, peek)?;
seen = has_seen;
Ok(body)
}
MessageDataItemName::InternalDate => self.internal_date(),
2024-01-10 17:08:44 +00:00
MessageDataItemName::ModSeq => Ok(self.modseq()),
2024-01-06 17:01:44 +00:00
})
.collect::<Result<Vec<_>, _>>()?;
Ok((
Body::Data(Data::Fetch {
seq: self.in_idx.i,
items: res_attrs.try_into()?,
}),
seen,
))
}
pub fn stored_naive_date(&self) -> Result<NaiveDate> {
let mail_meta = self.query_result.metadata().expect("metadata were fetched");
let mail_ts: i64 = mail_meta.internaldate.try_into()?;
let msg_date: ChronoDateTime<Local> = ChronoDateTime::from_timestamp(mail_ts, 0)
.ok_or(anyhow!("unable to parse timestamp"))?
.with_timezone(&Local);
Ok(msg_date.date_naive())
}
2024-01-06 21:53:41 +00:00
pub fn is_header_contains_pattern(&self, hdr: &[u8], pattern: &[u8]) -> bool {
let mime = match self.selected_mime() {
None => return false,
Some(x) => x,
};
let val = match mime.header_value(hdr) {
None => return false,
Some(x) => x,
};
val.windows(pattern.len()).any(|win| win == pattern)
}
2024-01-06 17:01:44 +00:00
// Private function, mainly for filter!
2024-01-04 19:54:21 +00:00
fn uid(&self) -> MessageDataItem<'static> {
2024-01-06 10:07:53 +00:00
MessageDataItem::Uid(self.in_idx.uid.clone())
2024-01-04 19:54:21 +00:00
}
fn flags(&self) -> MessageDataItem<'static> {
MessageDataItem::Flags(
2024-01-06 10:07:53 +00:00
self.in_idx
.flags
2024-01-04 19:54:21 +00:00
.iter()
.filter_map(|f| flags::from_str(f))
.collect(),
)
}
2024-01-06 10:07:53 +00:00
fn rfc_822_size(&self) -> Result<MessageDataItem<'static>> {
2024-01-06 10:33:56 +00:00
let sz = self
.query_result
.metadata()
.ok_or(anyhow!("mail metadata are required"))?
.rfc822_size;
2024-01-06 10:07:53 +00:00
Ok(MessageDataItem::Rfc822Size(sz as u32))
2024-01-04 19:54:21 +00:00
}
2024-01-06 10:07:53 +00:00
fn rfc_822_header(&self) -> Result<MessageDataItem<'static>> {
2024-01-06 10:33:56 +00:00
let hdrs: NString = self
.query_result
.metadata()
.ok_or(anyhow!("mail metadata are required"))?
.headers
.to_vec()
.try_into()?;
2024-01-06 10:07:53 +00:00
Ok(MessageDataItem::Rfc822Header(hdrs))
2024-01-04 19:54:21 +00:00
}
fn rfc_822_text(&self) -> Result<MessageDataItem<'static>> {
2024-01-06 21:53:41 +00:00
let txt: NString = self.content.as_msg()?.raw_body.to_vec().try_into()?;
2024-01-06 10:07:53 +00:00
Ok(MessageDataItem::Rfc822Text(txt))
2024-01-04 19:54:21 +00:00
}
fn rfc822(&self) -> Result<MessageDataItem<'static>> {
2024-01-06 21:53:41 +00:00
let full: NString = self.content.as_msg()?.raw_part.to_vec().try_into()?;
2024-01-06 10:07:53 +00:00
Ok(MessageDataItem::Rfc822(full))
2024-01-04 19:54:21 +00:00
}
fn envelope(&self) -> MessageDataItem<'static> {
2024-01-06 22:35:23 +00:00
MessageDataItem::Envelope(
self.imf()
.expect("an imf object is derivable from fetchedmail")
.message_envelope(),
)
2024-01-04 19:54:21 +00:00
}
fn body(&self) -> Result<MessageDataItem<'static>> {
Ok(MessageDataItem::Body(mime_view::bodystructure(
2024-01-06 21:53:41 +00:00
self.content.as_msg()?.child.as_ref(),
false,
2024-01-04 19:54:21 +00:00
)?))
}
fn body_structure(&self) -> Result<MessageDataItem<'static>> {
2024-01-08 15:03:42 +00:00
Ok(MessageDataItem::BodyStructure(mime_view::bodystructure(
2024-01-06 21:53:41 +00:00
self.content.as_msg()?.child.as_ref(),
true,
2024-01-04 19:54:21 +00:00
)?))
}
2024-01-08 14:07:02 +00:00
fn is_not_yet_seen(&self) -> bool {
let seen_flag = Flag::Seen.to_string();
!self.in_idx.flags.iter().any(|x| *x == seen_flag)
}
2024-01-04 19:54:21 +00:00
/// maps to BODY[<section>]<<partial>> and BODY.PEEK[<section>]<<partial>>
/// peek does not implicitly set the \Seen flag
/// eg. BODY[HEADER.FIELDS (DATE FROM)]
/// eg. BODY[]<0.2048>
2024-01-06 10:07:53 +00:00
fn body_ext(
2024-01-04 19:54:21 +00:00
&self,
2024-01-06 10:07:53 +00:00
section: &Option<FetchSection<'static>>,
2024-01-04 19:54:21 +00:00
partial: &Option<(u32, NonZeroU32)>,
peek: &bool,
2024-01-06 10:07:53 +00:00
) -> Result<(MessageDataItem<'static>, SeenFlag)> {
2024-01-04 19:54:21 +00:00
// Manage Seen flag
let mut seen = SeenFlag::DoNothing;
2024-01-08 14:07:02 +00:00
if !peek && self.is_not_yet_seen() {
2024-01-04 19:54:21 +00:00
// Add \Seen flag
//self.mailbox.add_flags(uuid, &[seen_flag]).await?;
seen = SeenFlag::MustAdd;
}
// Process message
2024-01-05 09:05:30 +00:00
let (text, origin) =
match mime_view::body_ext(self.content.as_anypart()?, section, partial)? {
mime_view::BodySection::Full(body) => (body, None),
mime_view::BodySection::Slice { body, origin_octet } => (body, Some(origin_octet)),
};
2024-01-04 19:54:21 +00:00
2024-01-06 10:07:53 +00:00
let data: NString = text.to_vec().try_into()?;
2024-01-04 19:54:21 +00:00
return Ok((
MessageDataItem::BodyExt {
section: section.as_ref().map(|fs| fs.clone()),
origin,
data,
},
seen,
));
}
fn internal_date(&self) -> Result<MessageDataItem<'static>> {
let dt = Utc
.fix()
2024-01-06 10:33:56 +00:00
.timestamp_opt(
i64::try_from(
self.query_result
.metadata()
.ok_or(anyhow!("mail metadata were not fetched"))?
.internaldate
/ 1000,
)?,
0,
)
2024-01-04 19:54:21 +00:00
.earliest()
.ok_or(anyhow!("Unable to parse internal date"))?;
Ok(MessageDataItem::InternalDate(DateTime::unvalidated(dt)))
}
2024-01-10 17:08:44 +00:00
fn modseq(&self) -> MessageDataItem<'static> {
MessageDataItem::ModSeq(self.in_idx.modseq)
}
2024-01-04 19:54:21 +00:00
}
pub enum SeenFlag {
DoNothing,
MustAdd,
}
2024-01-05 09:05:30 +00:00
// -------------------
2024-01-04 19:54:21 +00:00
pub enum FetchedMail<'a> {
2024-01-06 10:07:53 +00:00
IndexOnly,
2024-01-06 21:53:41 +00:00
Partial(AnyPart<'a>),
2024-01-04 19:54:21 +00:00
Full(AnyPart<'a>),
}
impl<'a> FetchedMail<'a> {
2024-01-06 21:53:41 +00:00
pub fn full_from_message(msg: Message<'a>) -> Self {
2024-01-06 10:07:53 +00:00
Self::Full(AnyPart::Msg(msg))
2024-01-04 19:54:21 +00:00
}
2024-01-06 21:53:41 +00:00
pub fn partial_from_message(msg: Message<'a>) -> Self {
Self::Partial(AnyPart::Msg(msg))
}
2024-01-06 22:24:44 +00:00
pub fn as_anypart(&self) -> Result<&AnyPart<'a>> {
2024-01-04 19:54:21 +00:00
match self {
FetchedMail::Full(x) => Ok(&x),
2024-01-06 21:53:41 +00:00
FetchedMail::Partial(x) => Ok(&x),
2024-01-04 19:54:21 +00:00
_ => bail!("The full message must be fetched, not only its headers"),
}
}
2024-01-06 22:24:44 +00:00
pub fn as_msg(&self) -> Result<&Message<'a>> {
2024-01-04 19:54:21 +00:00
match self {
FetchedMail::Full(AnyPart::Msg(x)) => Ok(&x),
2024-01-06 21:53:41 +00:00
FetchedMail::Partial(AnyPart::Msg(x)) => Ok(&x),
2024-01-04 19:54:21 +00:00
_ => bail!("The full message must be fetched, not only its headers AND it must be an AnyPart::Msg."),
}
}
2024-01-06 22:24:44 +00:00
pub fn as_imf(&self) -> Option<&imf::Imf<'a>> {
2024-01-04 19:54:21 +00:00
match self {
FetchedMail::Full(AnyPart::Msg(x)) => Some(&x.imf),
2024-01-06 21:53:41 +00:00
FetchedMail::Partial(AnyPart::Msg(x)) => Some(&x.imf),
_ => None,
2024-01-04 19:54:21 +00:00
}
}
}