Share UniqueIdent between collections
This commit is contained in:
parent
22e4f29555
commit
ed47855ef1
15 changed files with 90 additions and 17 deletions
|
@ -1 +1,5 @@
|
||||||
//@FIXME Event Index
|
pub mod namespace;
|
||||||
|
|
||||||
|
pub struct Calendar {
|
||||||
|
a: u64,
|
||||||
|
}
|
||||||
|
|
47
aero-collections/src/calendar/namespace.rs
Normal file
47
aero-collections/src/calendar/namespace.rs
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
use anyhow::Result;
|
||||||
|
use std::collections::{HashMap, BTreeMap};
|
||||||
|
use std::sync::{Weak, Arc};
|
||||||
|
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
|
use aero_user::storage;
|
||||||
|
|
||||||
|
use crate::unique_ident::UniqueIdent;
|
||||||
|
use crate::user::User;
|
||||||
|
use super::Calendar;
|
||||||
|
|
||||||
|
pub(crate) const CAL_LIST_PK: &str = "calendars";
|
||||||
|
pub(crate) const CAL_LIST_SK: &str = "list";
|
||||||
|
|
||||||
|
pub(crate) struct CalendarNs(std::sync::Mutex<HashMap<UniqueIdent, Weak<Calendar>>>);
|
||||||
|
impl CalendarNs {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self(std::sync::Mutex::new(HashMap::new()))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn list(&self) {
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
pub(crate) struct CalendarList(BTreeMap<String, CalendarListEntry>);
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize, Clone, Copy, Debug)]
|
||||||
|
pub(crate) struct CalendarListEntry {
|
||||||
|
id_lww: (u64, Option<UniqueIdent>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl CalendarList {
|
||||||
|
pub(crate) async fn load(user: &Arc<User>) -> Result<(Self, Option<storage::RowRef>)> {
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) async fn save(user: &Arc<User>, ct: Option<storage::RowRef>) -> Result<()> {
|
||||||
|
todo!();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn new() -> Self {
|
||||||
|
Self(BTreeMap::new())
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,3 +1,4 @@
|
||||||
|
pub mod unique_ident;
|
||||||
pub mod user;
|
pub mod user;
|
||||||
pub mod mail;
|
pub mod mail;
|
||||||
pub mod calendar;
|
pub mod calendar;
|
||||||
|
|
|
@ -15,7 +15,7 @@ use aero_bayou::timestamp::now_msec;
|
||||||
|
|
||||||
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::*;
|
use crate::unique_ident::*;
|
||||||
use crate::user::User;
|
use crate::user::User;
|
||||||
use crate::mail::IMF;
|
use crate::mail::IMF;
|
||||||
|
|
||||||
|
|
|
@ -9,7 +9,7 @@ use aero_bayou::Bayou;
|
||||||
use aero_bayou::timestamp::now_msec;
|
use aero_bayou::timestamp::now_msec;
|
||||||
|
|
||||||
use crate::mail::uidindex::*;
|
use crate::mail::uidindex::*;
|
||||||
use crate::mail::unique_ident::*;
|
use crate::unique_ident::*;
|
||||||
use crate::mail::IMF;
|
use crate::mail::IMF;
|
||||||
|
|
||||||
pub struct Mailbox {
|
pub struct Mailbox {
|
||||||
|
|
|
@ -3,7 +3,6 @@ pub mod mailbox;
|
||||||
pub mod query;
|
pub mod query;
|
||||||
pub mod snapshot;
|
pub mod snapshot;
|
||||||
pub mod uidindex;
|
pub mod uidindex;
|
||||||
pub mod unique_ident;
|
|
||||||
pub mod namespace;
|
pub mod namespace;
|
||||||
|
|
||||||
// Internet Message Format
|
// Internet Message Format
|
||||||
|
|
|
@ -6,7 +6,7 @@ use serde::{Deserialize, Serialize};
|
||||||
use aero_bayou::timestamp::now_msec;
|
use aero_bayou::timestamp::now_msec;
|
||||||
|
|
||||||
use crate::mail::uidindex::ImapUidvalidity;
|
use crate::mail::uidindex::ImapUidvalidity;
|
||||||
use crate::mail::unique_ident::{gen_ident, UniqueIdent};
|
use crate::unique_ident::{gen_ident, UniqueIdent};
|
||||||
|
|
||||||
pub const MAILBOX_HIERARCHY_DELIMITER: char = '.';
|
pub const MAILBOX_HIERARCHY_DELIMITER: char = '.';
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use super::mailbox::MailMeta;
|
use super::mailbox::MailMeta;
|
||||||
use super::snapshot::FrozenMailbox;
|
use super::snapshot::FrozenMailbox;
|
||||||
use super::unique_ident::UniqueIdent;
|
use crate::unique_ident::UniqueIdent;
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use futures::future::FutureExt;
|
use futures::future::FutureExt;
|
||||||
use futures::stream::{BoxStream, Stream, StreamExt};
|
use futures::stream::{BoxStream, Stream, StreamExt};
|
||||||
|
|
|
@ -2,10 +2,10 @@ use std::sync::Arc;
|
||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
|
use crate::unique_ident::UniqueIdent;
|
||||||
use super::mailbox::Mailbox;
|
use super::mailbox::Mailbox;
|
||||||
use super::query::{Query, QueryScope};
|
use super::query::{Query, QueryScope};
|
||||||
use super::uidindex::UidIndex;
|
use super::uidindex::UidIndex;
|
||||||
use super::unique_ident::UniqueIdent;
|
|
||||||
|
|
||||||
/// A Frozen Mailbox has a snapshot of the current mailbox
|
/// A Frozen Mailbox has a snapshot of the current mailbox
|
||||||
/// state that is desynchronized with the real mailbox state.
|
/// state that is desynchronized with the real mailbox state.
|
||||||
|
|
|
@ -4,7 +4,7 @@ use im::{HashMap, OrdMap, OrdSet};
|
||||||
use serde::{Deserialize, Deserializer, Serialize, Serializer};
|
use serde::{Deserialize, Deserializer, Serialize, Serializer};
|
||||||
|
|
||||||
use aero_bayou::*;
|
use aero_bayou::*;
|
||||||
use crate::mail::unique_ident::UniqueIdent;
|
use crate::unique_ident::UniqueIdent;
|
||||||
|
|
||||||
pub type ModSeq = NonZeroU64;
|
pub type ModSeq = NonZeroU64;
|
||||||
pub type ImapUid = NonZeroU32;
|
pub type ImapUid = NonZeroU32;
|
||||||
|
|
|
@ -7,7 +7,7 @@ use serde::{de::Error, Deserialize, Deserializer, Serialize, Serializer};
|
||||||
|
|
||||||
use aero_bayou::timestamp::now_msec;
|
use aero_bayou::timestamp::now_msec;
|
||||||
|
|
||||||
/// An internal Mail Identifier is composed of two components:
|
/// An internal Aerogramme identifier is composed of two components:
|
||||||
/// - a process identifier, 128 bits, itself composed of:
|
/// - a process identifier, 128 bits, itself composed of:
|
||||||
/// - the timestamp of when the process started, 64 bits
|
/// - the timestamp of when the process started, 64 bits
|
||||||
/// - a 64-bit random number
|
/// - a 64-bit random number
|
||||||
|
@ -15,7 +15,7 @@ use aero_bayou::timestamp::now_msec;
|
||||||
/// They are not part of the protocol but an internal representation
|
/// They are not part of the protocol but an internal representation
|
||||||
/// required by Aerogramme.
|
/// required by Aerogramme.
|
||||||
/// Their main property is to be unique without having to rely
|
/// Their main property is to be unique without having to rely
|
||||||
/// on synchronization between IMAP processes.
|
/// on synchronization between (IMAP) processes.
|
||||||
#[derive(Clone, Copy, PartialOrd, Ord, PartialEq, Eq, Hash)]
|
#[derive(Clone, Copy, PartialOrd, Ord, PartialEq, Eq, Hash)]
|
||||||
pub struct UniqueIdent(pub [u8; 24]);
|
pub struct UniqueIdent(pub [u8; 24]);
|
||||||
|
|
|
@ -12,19 +12,27 @@ use aero_user::storage;
|
||||||
use crate::mail::incoming::incoming_mail_watch_process;
|
use crate::mail::incoming::incoming_mail_watch_process;
|
||||||
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::UniqueIdent;
|
use crate::unique_ident::UniqueIdent;
|
||||||
use crate::mail::namespace::{MAILBOX_HIERARCHY_DELIMITER, INBOX, DRAFTS, ARCHIVE, SENT, TRASH, MAILBOX_LIST_PK, MAILBOX_LIST_SK,MailboxList,CreatedMailbox};
|
use crate::mail::namespace::{MAILBOX_HIERARCHY_DELIMITER, INBOX, DRAFTS, ARCHIVE, SENT, TRASH, MAILBOX_LIST_PK, MAILBOX_LIST_SK,MailboxList,CreatedMailbox};
|
||||||
|
use crate::calendar::Calendar;
|
||||||
|
|
||||||
//@FIXME User should be totally rewriten
|
//@FIXME User should be totally rewriten
|
||||||
// to extract the local mailbox list
|
// to extract the local mailbox list
|
||||||
// to the mail/namespace.rs file (and mailbox list should be reworded as mail namespace)
|
// to the mail/namespace.rs file (and mailbox list should be reworded as mail namespace)
|
||||||
|
|
||||||
|
//@FIXME User should be run in a LocalSet
|
||||||
|
// to remove most - if not all - synchronizations types.
|
||||||
|
// Especially RwLock & co.
|
||||||
|
|
||||||
pub struct User {
|
pub struct User {
|
||||||
pub username: String,
|
pub username: String,
|
||||||
pub creds: Credentials,
|
pub creds: Credentials,
|
||||||
pub storage: storage::Store,
|
pub storage: storage::Store,
|
||||||
pub mailboxes: std::sync::Mutex<HashMap<UniqueIdent, Weak<Mailbox>>>,
|
pub mailboxes: std::sync::Mutex<HashMap<UniqueIdent, Weak<Mailbox>>>,
|
||||||
|
pub calendars: std::sync::Mutex<HashMap<UniqueIdent, Weak<Calendar>>>,
|
||||||
|
|
||||||
|
// Handle on worker processing received email
|
||||||
|
// (moving emails from the mailqueue to the user's INBOX)
|
||||||
tx_inbox_id: watch::Sender<Option<(UniqueIdent, ImapUidvalidity)>>,
|
tx_inbox_id: watch::Sender<Option<(UniqueIdent, ImapUidvalidity)>>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -178,6 +186,7 @@ impl User {
|
||||||
storage,
|
storage,
|
||||||
tx_inbox_id,
|
tx_inbox_id,
|
||||||
mailboxes: std::sync::Mutex::new(HashMap::new()),
|
mailboxes: std::sync::Mutex::new(HashMap::new()),
|
||||||
|
calendars: std::sync::Mutex::new(HashMap::new()),
|
||||||
});
|
});
|
||||||
|
|
||||||
// Ensure INBOX exists (done inside load_mailbox_list)
|
// Ensure INBOX exists (done inside load_mailbox_list)
|
||||||
|
@ -204,6 +213,10 @@ impl User {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The idea here is that:
|
||||||
|
// 1. Opening a mailbox that is not already opened takes a significant amount of time
|
||||||
|
// 2. We don't want to lock the whole HashMap that contain the mailboxes during this
|
||||||
|
// operation which is why we droppped the lock above but take it again below.
|
||||||
let mb = Arc::new(Mailbox::open(&self.creds, id, min_uidvalidity).await?);
|
let mb = Arc::new(Mailbox::open(&self.creds, id, min_uidvalidity).await?);
|
||||||
|
|
||||||
let mut cache = self.mailboxes.lock().unwrap();
|
let mut cache = self.mailboxes.lock().unwrap();
|
||||||
|
|
|
@ -27,6 +27,8 @@ use aero_dav::acltypes as acl;
|
||||||
use aero_dav::realization::{All, self as all};
|
use aero_dav::realization::{All, self as all};
|
||||||
use aero_dav::xml as dxml;
|
use aero_dav::xml as dxml;
|
||||||
|
|
||||||
|
type ArcUser = std::sync::Arc<User>;
|
||||||
|
|
||||||
pub struct Server {
|
pub struct Server {
|
||||||
bind_addr: SocketAddr,
|
bind_addr: SocketAddr,
|
||||||
login_provider: ArcLoginProvider,
|
login_provider: ArcLoginProvider,
|
||||||
|
@ -359,7 +361,15 @@ async fn propfind(user: std::sync::Arc<User>, req: Request<Incoming>, base_node:
|
||||||
async fn report(user: std::sync::Arc<User>, req: Request<Incoming>, node: Box<dyn DavNode>) -> Result<Response<BoxBody<Bytes, std::io::Error>>> {
|
async fn report(user: std::sync::Arc<User>, req: Request<Incoming>, node: Box<dyn DavNode>) -> Result<Response<BoxBody<Bytes, std::io::Error>>> {
|
||||||
let status = hyper::StatusCode::from_u16(207)?;
|
let status = hyper::StatusCode::from_u16(207)?;
|
||||||
|
|
||||||
let report = deserialize::<cal::Report<All>>(req).await?;
|
let report = match deserialize::<cal::Report<All>>(req).await {
|
||||||
|
Ok(v) => v,
|
||||||
|
Err(e) => {
|
||||||
|
tracing::error!(err=?e, "unable to decode REPORT body");
|
||||||
|
return Ok(Response::builder()
|
||||||
|
.status(400)
|
||||||
|
.body(text_body("Bad request"))?)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// Multiget is really like a propfind where Depth: 0|1|Infinity is replaced by an arbitrary
|
// Multiget is really like a propfind where Depth: 0|1|Infinity is replaced by an arbitrary
|
||||||
// list of URLs
|
// list of URLs
|
||||||
|
@ -492,7 +502,6 @@ async fn deserialize<T: dxml::Node<T>>(req: Request<Incoming>) -> Result<T> {
|
||||||
|
|
||||||
//---
|
//---
|
||||||
|
|
||||||
type ArcUser = std::sync::Arc<User>;
|
|
||||||
trait DavNode: Send {
|
trait DavNode: Send {
|
||||||
// ------- specialized logic
|
// ------- specialized logic
|
||||||
|
|
||||||
|
|
|
@ -4,7 +4,7 @@ use anyhow::{anyhow, Result};
|
||||||
use imap_codec::imap_types::sequence::{SeqOrUid, Sequence, SequenceSet};
|
use imap_codec::imap_types::sequence::{SeqOrUid, Sequence, SequenceSet};
|
||||||
|
|
||||||
use aero_collections::mail::uidindex::{ImapUid, ModSeq, UidIndex};
|
use aero_collections::mail::uidindex::{ImapUid, ModSeq, UidIndex};
|
||||||
use aero_collections::mail::unique_ident::UniqueIdent;
|
use aero_collections::unique_ident::UniqueIdent;
|
||||||
|
|
||||||
pub struct Index<'a> {
|
pub struct Index<'a> {
|
||||||
pub imap_index: Vec<MailIndex<'a>>,
|
pub imap_index: Vec<MailIndex<'a>>,
|
||||||
|
|
|
@ -17,7 +17,7 @@ use aero_collections::mail::mailbox::Mailbox;
|
||||||
use aero_collections::mail::query::QueryScope;
|
use aero_collections::mail::query::QueryScope;
|
||||||
use aero_collections::mail::snapshot::FrozenMailbox;
|
use aero_collections::mail::snapshot::FrozenMailbox;
|
||||||
use aero_collections::mail::uidindex::{ImapUid, ImapUidvalidity, ModSeq};
|
use aero_collections::mail::uidindex::{ImapUid, ImapUidvalidity, ModSeq};
|
||||||
use aero_collections::mail::unique_ident::UniqueIdent;
|
use aero_collections::unique_ident::UniqueIdent;
|
||||||
|
|
||||||
use crate::imap::attributes::AttributesProxy;
|
use crate::imap::attributes::AttributesProxy;
|
||||||
use crate::imap::flags;
|
use crate::imap::flags;
|
||||||
|
|
Loading…
Reference in a new issue