Implement RFC822.TEXT

This commit is contained in:
Quentin 2022-06-30 13:36:21 +02:00
parent 69428eaf25
commit 0f0a8d415e
Signed by: quentin
GPG key ID: E9602264D639FF68
6 changed files with 62 additions and 31 deletions

View file

@ -19,7 +19,10 @@ pub async fn dispatch<'a>(ctx: AuthenticatedContext<'a>) -> Result<(Response, fl
match &ctx.req.command.body { match &ctx.req.command.body {
CommandBody::Create { mailbox } => ctx.create(mailbox).await, CommandBody::Create { mailbox } => ctx.create(mailbox).await,
CommandBody::Delete { mailbox } => ctx.delete(mailbox).await, CommandBody::Delete { mailbox } => ctx.delete(mailbox).await,
CommandBody::Rename { mailbox, new_mailbox } => ctx.rename(mailbox, new_mailbox).await, CommandBody::Rename {
mailbox,
new_mailbox,
} => ctx.rename(mailbox, new_mailbox).await,
CommandBody::Lsub { CommandBody::Lsub {
reference, reference,
mailbox_wildcard, mailbox_wildcard,
@ -28,8 +31,10 @@ pub async fn dispatch<'a>(ctx: AuthenticatedContext<'a>) -> Result<(Response, fl
reference, reference,
mailbox_wildcard, mailbox_wildcard,
} => ctx.list(reference, mailbox_wildcard).await, } => ctx.list(reference, mailbox_wildcard).await,
CommandBody::Status { mailbox, attributes } => CommandBody::Status {
ctx.status(mailbox, attributes).await, mailbox,
attributes,
} => ctx.status(mailbox, attributes).await,
CommandBody::Select { mailbox } => ctx.select(mailbox).await, CommandBody::Select { mailbox } => ctx.select(mailbox).await,
CommandBody::Examine { mailbox } => ctx.examine(mailbox).await, CommandBody::Examine { mailbox } => ctx.examine(mailbox).await,
_ => { _ => {
@ -53,7 +58,11 @@ impl<'a> AuthenticatedContext<'a> {
Ok((Response::bad("Not implemented")?, flow::Transition::None)) Ok((Response::bad("Not implemented")?, flow::Transition::None))
} }
async fn rename(self, mailbox: &MailboxCodec, new_mailbox: &MailboxCodec) -> Result<(Response, flow::Transition)> { async fn rename(
self,
mailbox: &MailboxCodec,
new_mailbox: &MailboxCodec,
) -> Result<(Response, flow::Transition)> {
Ok((Response::bad("Not implemented")?, flow::Transition::None)) Ok((Response::bad("Not implemented")?, flow::Transition::None))
} }

View file

@ -48,9 +48,7 @@ pub async fn dispatch<'a>(ctx: ExaminedContext<'a>) -> Result<(Response, flow::T
// --- PRIVATE --- // --- PRIVATE ---
impl<'a> ExaminedContext<'a> { impl<'a> ExaminedContext<'a> {
async fn close( async fn close(self) -> Result<(Response, flow::Transition)> {
self,
) -> Result<(Response, flow::Transition)> {
Ok((Response::ok("CLOSE completed")?, flow::Transition::Unselect)) Ok((Response::ok("CLOSE completed")?, flow::Transition::Unselect))
} }

View file

@ -1,4 +1,4 @@
pub mod anonymous; pub mod anonymous;
pub mod authenticated; pub mod authenticated;
pub mod selected;
pub mod examined; pub mod examined;
pub mod selected;

View file

@ -2,9 +2,9 @@ use anyhow::Result;
use boitalettres::proto::Request; use boitalettres::proto::Request;
use boitalettres::proto::Response; use boitalettres::proto::Response;
use imap_codec::types::command::CommandBody; use imap_codec::types::command::CommandBody;
use imap_codec::types::mailbox::{Mailbox as MailboxCodec};
use imap_codec::types::flag::{Flag, StoreType, StoreResponse};
use imap_codec::types::fetch_attributes::MacroOrFetchAttributes; use imap_codec::types::fetch_attributes::MacroOrFetchAttributes;
use imap_codec::types::flag::{Flag, StoreResponse, StoreType};
use imap_codec::types::mailbox::Mailbox as MailboxCodec;
use imap_codec::types::sequence::SequenceSet; use imap_codec::types::sequence::SequenceSet;
@ -31,7 +31,7 @@ pub async fn dispatch<'a>(ctx: SelectedContext<'a>) -> Result<(Response, flow::T
kind, kind,
response, response,
flags, flags,
uid uid,
} => ctx.store(sequence_set, kind, response, flags, uid).await, } => ctx.store(sequence_set, kind, response, flags, uid).await,
CommandBody::Copy { CommandBody::Copy {
sequence_set, sequence_set,
@ -52,18 +52,14 @@ pub async fn dispatch<'a>(ctx: SelectedContext<'a>) -> Result<(Response, flow::T
// --- PRIVATE --- // --- PRIVATE ---
impl<'a> SelectedContext<'a> { impl<'a> SelectedContext<'a> {
async fn close( async fn close(self) -> Result<(Response, flow::Transition)> {
self,
) -> Result<(Response, flow::Transition)> {
// We expunge messages, // We expunge messages,
// but we don't send the untagged EXPUNGE responses // but we don't send the untagged EXPUNGE responses
self.expunge().await?; self.expunge().await?;
Ok((Response::ok("CLOSE completed")?, flow::Transition::Unselect)) Ok((Response::ok("CLOSE completed")?, flow::Transition::Unselect))
} }
async fn expunge( async fn expunge(self) -> Result<(Response, flow::Transition)> {
self,
) -> Result<(Response, flow::Transition)> {
Ok((Response::bad("Not implemented")?, flow::Transition::None)) Ok((Response::bad("Not implemented")?, flow::Transition::None))
} }

View file

@ -6,7 +6,7 @@ use anyhow::{anyhow, bail, Error, Result};
use boitalettres::proto::res::body::Data as Body; use boitalettres::proto::res::body::Data as Body;
use futures::stream::{FuturesOrdered, StreamExt}; use futures::stream::{FuturesOrdered, StreamExt};
use imap_codec::types::address::Address; use imap_codec::types::address::Address;
use imap_codec::types::core::{Atom, IString, NString}; use imap_codec::types::core::{Atom, IString, NString, NonZeroBytes};
use imap_codec::types::envelope::Envelope; use imap_codec::types::envelope::Envelope;
use imap_codec::types::fetch_attributes::{FetchAttribute, MacroOrFetchAttributes}; use imap_codec::types::fetch_attributes::{FetchAttribute, MacroOrFetchAttributes};
use imap_codec::types::flag::Flag; use imap_codec::types::flag::Flag;
@ -247,21 +247,49 @@ impl MailboxView {
FetchAttribute::Rfc822Size => { FetchAttribute::Rfc822Size => {
attributes.push(MessageAttribute::Rfc822Size(meta.rfc822_size as u32)) attributes.push(MessageAttribute::Rfc822Size(meta.rfc822_size as u32))
} }
FetchAttribute::Rfc822Header => { FetchAttribute::Rfc822Header => attributes.push(
attributes.push(MessageAttribute::Rfc822Header(NString(Some( MessageAttribute::Rfc822Header(NString(Some(IString::Literal(
IString::Literal(meta.headers.clone().try_into().unwrap()), meta.headers
.clone()
.try_into()
.or(Err(Error::msg("IString conversion error")))?,
)))),
),
FetchAttribute::Rfc822Text => {
let r = parsed
.raw_message.get(parsed.offset_body..parsed.offset_end)
.ok_or(Error::msg("Unable to extract email body, cursors out of bound. This is a bug."))?
.try_into()
.or(Err(Error::msg("IString conversion error")))?;
attributes.push(MessageAttribute::Rfc822Text(NString(Some(
IString::Literal(r),
)))) ))))
} }
FetchAttribute::Rfc822 => { FetchAttribute::Rfc822 => {
attributes.push(MessageAttribute::Rfc822(NString(Some( attributes.push(MessageAttribute::Rfc822(NString(Some(IString::Literal(
IString::Literal(body.as_ref().unwrap().clone().try_into().unwrap()), body.as_ref().unwrap().clone().try_into().unwrap(),
)))) )))))
} }
FetchAttribute::Envelope => { FetchAttribute::Envelope => {
attributes.push(MessageAttribute::Envelope(message_envelope(&parsed))) attributes.push(MessageAttribute::Envelope(message_envelope(&parsed)))
} }
// TODO FetchAttribute::Body => {
_ => (), todo!()
}
FetchAttribute::BodyExt {
section,
partial,
peek,
} => {
todo!()
}
FetchAttribute::BodyStructure => {
todo!()
}
FetchAttribute::InternalDate => {
todo!()
}
} }
} }

View file

@ -7,7 +7,7 @@ use futures::future::FutureExt;
use tokio::sync::mpsc::error::TrySendError; use tokio::sync::mpsc::error::TrySendError;
use tokio::sync::{mpsc, oneshot}; use tokio::sync::{mpsc, oneshot};
use crate::imap::command::{anonymous, authenticated, selected, examined}; use crate::imap::command::{anonymous, authenticated, examined, selected};
use crate::imap::flow; use crate::imap::flow;
use crate::login::ArcLoginProvider; use crate::login::ArcLoginProvider;