Remove OK [UNSEEN x] responses to SELECT and EXAMINE, and fix STATUS UNSEEN

This commit is contained in:
Alex 2022-07-13 15:39:52 +02:00
parent 956a92377e
commit 3b256de3dc
Signed by: lx
GPG key ID: 0E496D15096376BE
3 changed files with 18 additions and 32 deletions

View file

@ -218,9 +218,7 @@ impl<'a> AuthenticatedContext<'a> {
for attr in attributes.iter() { for attr in attributes.iter() {
ret_attrs.push(match attr { ret_attrs.push(match attr {
StatusAttribute::Messages => StatusAttributeValue::Messages(view.exists()?), StatusAttribute::Messages => StatusAttributeValue::Messages(view.exists()?),
StatusAttribute::Unseen => { StatusAttribute::Unseen => StatusAttributeValue::Unseen(view.unseen_count() as u32),
StatusAttributeValue::Unseen(view.unseen().map(|x| x.get()).unwrap_or(0))
}
StatusAttribute::Recent => StatusAttributeValue::Recent(view.recent()?), StatusAttribute::Recent => StatusAttributeValue::Recent(view.recent()?),
StatusAttribute::UidNext => StatusAttributeValue::UidNext(view.uidnext()), StatusAttribute::UidNext => StatusAttributeValue::UidNext(view.uidnext()),
StatusAttribute::UidValidity => { StatusAttribute::UidValidity => {
@ -287,6 +285,11 @@ impl<'a> AuthenticatedContext<'a> {
S: A142 OK [READ-WRITE] SELECT completed S: A142 OK [READ-WRITE] SELECT completed
--- a mailbox with no unseen message -> no unseen entry --- a mailbox with no unseen message -> no unseen entry
NOTES:
RFC3501 (imap4rev1) says if there is no OK [UNSEEN] response, client must make no assumption,
it is therefore correct to not return it even if there are unseen messages
RFC9051 (imap4rev2) says that OK [UNSEEN] responses are deprecated after SELECT and EXAMINE
For Aerogramme, we just don't send the OK [UNSEEN], it's correct to do in both specifications.
20 select "INBOX.achats" 20 select "INBOX.achats"
* FLAGS (\Answered \Flagged \Deleted \Seen \Draft $Forwarded JUNK $label1) * FLAGS (\Answered \Flagged \Deleted \Seen \Draft $Forwarded JUNK $label1)

View file

@ -61,9 +61,6 @@ impl MailboxView {
data.extend(new_view.flags_status()?.into_iter()); data.extend(new_view.flags_status()?.into_iter());
data.push(new_view.uidvalidity_status()?); data.push(new_view.uidvalidity_status()?);
data.push(new_view.uidnext_status()?); data.push(new_view.uidnext_status()?);
if let Some(unseen) = new_view.unseen_status()? {
data.push(unseen);
}
Ok((new_view, data)) Ok((new_view, data))
} }
@ -432,27 +429,6 @@ impl MailboxView {
self.known_state.uidnext self.known_state.uidnext
} }
/// Produces an UNSEEN message (if relevant) corresponding to the
/// first unseen message id in `known_state`
fn unseen_status(&self) -> Result<Option<Body>> {
if let Some(unseen) = self.unseen() {
let status_unseen =
Status::ok(None, Some(Code::Unseen(unseen.clone())), "First unseen UID")
.map_err(Error::msg)?;
Ok(Some(Body::Status(status_unseen)))
} else {
Ok(None)
}
}
pub(crate) fn unseen(&self) -> Option<ImapUid> {
self.known_state
.idx_by_flag
.get(&"$unseen".to_string())
.and_then(|os| os.get_min())
.cloned()
}
/// Produce an EXISTS message corresponding to the number of mails /// Produce an EXISTS message corresponding to the number of mails
/// in `known_state` /// in `known_state`
fn exists_status(&self) -> Result<Body> { fn exists_status(&self) -> Result<Body> {
@ -504,6 +480,17 @@ impl MailboxView {
Ok(ret) Ok(ret)
} }
pub(crate) fn unseen_count(&self) -> usize {
let total = self.known_state.table.len();
let seen = self
.known_state
.idx_by_flag
.get(&Flag::Seen.to_string())
.map(|x| x.len())
.unwrap_or(0);
total - seen
}
} }
fn string_to_flag(f: &str) -> Option<Flag> { fn string_to_flag(f: &str) -> Option<Flag> {
@ -523,7 +510,6 @@ fn string_to_flag(f: &str) -> Option<Flag> {
Ok(a) => Some(Flag::Extension(a)), Ok(a) => Some(Flag::Extension(a)),
}, },
}, },
Some('$') if f == "$unseen" => None,
Some(_) => match Atom::try_from(f.clone()) { Some(_) => match Atom::try_from(f.clone()) {
Err(_) => { Err(_) => {
tracing::error!(flag=%f, "Unable to encode flag as IMAP atom"); tracing::error!(flag=%f, "Unable to encode flag as IMAP atom");

View file

@ -374,10 +374,7 @@ impl MailboxInternal {
)?; )?;
// Add mail to Bayou mail index // Add mail to Bayou mail index
let add_mail_op = self let add_mail_op = self.uid_index.state().op_mail_add(ident, vec![]);
.uid_index
.state()
.op_mail_add(ident, vec!["\\Unseen".into()]);
self.uid_index.push(add_mail_op).await?; self.uid_index.push(add_mail_op).await?;
Ok(()) Ok(())