Skeleton for incoming mail processor
This commit is contained in:
parent
3277fb16a8
commit
484ad97b65
3 changed files with 36 additions and 13 deletions
16
src/mail/incoming.rs
Normal file
16
src/mail/incoming.rs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
use std::sync::{Arc, Weak};
|
||||||
|
use std::time::Duration;
|
||||||
|
|
||||||
|
use tokio::sync::watch;
|
||||||
|
|
||||||
|
use crate::mail::unique_ident::UniqueIdent;
|
||||||
|
use crate::mail::user::User;
|
||||||
|
use crate::mail::uidindex::ImapUidvalidity;
|
||||||
|
|
||||||
|
pub async fn incoming_mail_watch_process(user: Weak<User>, rx_inbox_id: watch::Receiver<Option<(UniqueIdent, ImapUidvalidity)>>) {
|
||||||
|
while Weak::upgrade(&user).is_some() {
|
||||||
|
eprintln!("User still available");
|
||||||
|
tokio::time::sleep(Duration::from_secs(10)).await;
|
||||||
|
}
|
||||||
|
drop(rx_inbox_id);
|
||||||
|
}
|
|
@ -4,6 +4,7 @@ pub mod mailbox;
|
||||||
pub mod uidindex;
|
pub mod uidindex;
|
||||||
pub mod unique_ident;
|
pub mod unique_ident;
|
||||||
pub mod user;
|
pub mod user;
|
||||||
|
pub mod incoming;
|
||||||
|
|
||||||
// Internet Message Format
|
// Internet Message Format
|
||||||
// aka RFC 822 - RFC 2822 - RFC 5322
|
// aka RFC 822 - RFC 2822 - RFC 5322
|
||||||
|
|
|
@ -12,6 +12,7 @@ use crate::login::{Credentials, StorageCredentials};
|
||||||
use crate::mail::mailbox::Mailbox;
|
use crate::mail::mailbox::Mailbox;
|
||||||
use crate::mail::uidindex::ImapUidvalidity;
|
use crate::mail::uidindex::ImapUidvalidity;
|
||||||
use crate::mail::unique_ident::{gen_ident, UniqueIdent};
|
use crate::mail::unique_ident::{gen_ident, UniqueIdent};
|
||||||
|
use crate::mail::incoming::incoming_mail_watch_process;
|
||||||
use crate::time::now_msec;
|
use crate::time::now_msec;
|
||||||
|
|
||||||
const MAILBOX_HIERARCHY_DELIMITER: &str = "/";
|
const MAILBOX_HIERARCHY_DELIMITER: &str = "/";
|
||||||
|
@ -70,6 +71,7 @@ impl User {
|
||||||
/// Opens an existing mailbox given its IMAP name.
|
/// Opens an existing mailbox given its IMAP name.
|
||||||
pub async fn open_mailbox(&self, name: &str) -> Result<Option<Arc<Mailbox>>> {
|
pub async fn open_mailbox(&self, name: &str) -> Result<Option<Arc<Mailbox>>> {
|
||||||
let (mut list, ct) = self.load_mailbox_list().await?;
|
let (mut list, ct) = self.load_mailbox_list().await?;
|
||||||
|
eprintln!("List of mailboxes: {:?}", list);
|
||||||
match list.get_mut(name) {
|
match list.get_mut(name) {
|
||||||
Some(MailboxListEntry {
|
Some(MailboxListEntry {
|
||||||
id_lww: (_, Some(mbid)),
|
id_lww: (_, Some(mbid)),
|
||||||
|
@ -126,6 +128,8 @@ impl User {
|
||||||
// Ensure INBOX exists (done inside load_mailbox_list)
|
// Ensure INBOX exists (done inside load_mailbox_list)
|
||||||
user.load_mailbox_list().await?;
|
user.load_mailbox_list().await?;
|
||||||
|
|
||||||
|
tokio::spawn(incoming_mail_watch_process(Arc::downgrade(&user), rx_inbox_id));
|
||||||
|
|
||||||
Ok(user)
|
Ok(user)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,12 +160,10 @@ impl User {
|
||||||
// ---- Mailbox list management ----
|
// ---- Mailbox list management ----
|
||||||
|
|
||||||
async fn load_mailbox_list(&self) -> Result<(MailboxList, Option<CausalityToken>)> {
|
async fn load_mailbox_list(&self) -> Result<(MailboxList, Option<CausalityToken>)> {
|
||||||
let cv = match self.k2v.read_item(MAILBOX_LIST_PK, MAILBOX_LIST_SK).await {
|
let (mut list, ct) = match self.k2v.read_item(MAILBOX_LIST_PK, MAILBOX_LIST_SK).await {
|
||||||
Err(k2v_client::Error::NotFound) => return Ok((BTreeMap::new(), None)),
|
Err(k2v_client::Error::NotFound) => (BTreeMap::new(), None),
|
||||||
Err(e) => return Err(e.into()),
|
Err(e) => return Err(e.into()),
|
||||||
Ok(cv) => cv,
|
Ok(cv) => {
|
||||||
};
|
|
||||||
|
|
||||||
let mut list = BTreeMap::new();
|
let mut list = BTreeMap::new();
|
||||||
for v in cv.value {
|
for v in cv.value {
|
||||||
if let K2vValue::Value(vbytes) = v {
|
if let K2vValue::Value(vbytes) = v {
|
||||||
|
@ -169,13 +171,17 @@ impl User {
|
||||||
list = merge_mailbox_lists(list, list2);
|
list = merge_mailbox_lists(list, list2);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
(list, Some(cv.causality))
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
// If INBOX doesn't exist, create a new mailbox with that name
|
// If INBOX doesn't exist, create a new mailbox with that name
|
||||||
// and save new mailbox list.
|
// and save new mailbox list.
|
||||||
// Also, ensure that the mpsc::watch that keeps track of the
|
// Also, ensure that the mpsc::watch that keeps track of the
|
||||||
// inbox id is up-to-date.
|
// inbox id is up-to-date.
|
||||||
let (inbox_id, inbox_uidvalidity) = match self
|
let (inbox_id, inbox_uidvalidity) = match self
|
||||||
.create_mailbox_internal(&mut list, Some(cv.causality.clone()), INBOX)
|
.create_mailbox_internal(&mut list, ct.clone(), INBOX)
|
||||||
.await?
|
.await?
|
||||||
{
|
{
|
||||||
CreatedMailbox::Created(i, v) => (i, v),
|
CreatedMailbox::Created(i, v) => (i, v),
|
||||||
|
@ -185,7 +191,7 @@ impl User {
|
||||||
.send(Some((inbox_id, inbox_uidvalidity)))
|
.send(Some((inbox_id, inbox_uidvalidity)))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
Ok((list, Some(cv.causality)))
|
Ok((list, ct))
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn save_mailbox_list(
|
async fn save_mailbox_list(
|
||||||
|
@ -243,7 +249,7 @@ impl User {
|
||||||
|
|
||||||
type MailboxList = BTreeMap<String, MailboxListEntry>;
|
type MailboxList = BTreeMap<String, MailboxListEntry>;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Clone, Copy)]
|
#[derive(Serialize, Deserialize, Clone, Copy, Debug)]
|
||||||
struct MailboxListEntry {
|
struct MailboxListEntry {
|
||||||
id_lww: (u64, Option<UniqueIdent>),
|
id_lww: (u64, Option<UniqueIdent>),
|
||||||
uidvalidity: ImapUidvalidity,
|
uidvalidity: ImapUidvalidity,
|
||||||
|
|
Loading…
Reference in a new issue