aboutsummaryrefslogtreecommitdiff
path: root/src/command.rs
blob: 9b2f12def22e29b2b37a11d67cf7c6d29f42ce49 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
use std::sync::{Arc, Mutex};

use boitalettres::errors::Error as BalError;
use boitalettres::proto::{Request, Response};
use imap_codec::types::core::{Tag, AString};
use imap_codec::types::response::{Capability, Data};
use imap_codec::types::mailbox::{Mailbox as MailboxCodec, ListMailbox};
use imap_codec::types::sequence::SequenceSet;
use imap_codec::types::fetch_attributes::MacroOrFetchAttributes;

use crate::mailstore::Mailstore;
use crate::mailbox::Mailbox;
use crate::service::Session;

pub struct Command {
    tag: Tag,
    mailstore: Arc<Mailstore>,
    session: Arc<Mutex<Session>>,
}

impl Command {
    pub fn new(tag: Tag, mailstore: Arc<Mailstore>, session: Arc<Mutex<Session>>) -> Self {
        Self { tag, mailstore, session }
    }

    pub async fn capability(&self) -> Result<Response, BalError> {
        let capabilities = vec![Capability::Imap4Rev1, Capability::Idle];
        let body = vec![Data::Capability(capabilities)];
        let r = Response::ok("Pre-login capabilities listed, post-login capabilities have more.")?
            .with_body(body);
        Ok(r)
    }

    pub async fn login(&self, username: AString, password: AString) -> Result<Response, BalError> {
        let (u, p) = match (String::try_from(username), String::try_from(password)) {
            (Ok(u), Ok(p)) => (u, p),
            _ => return Response::bad("Invalid characters"),
        };

        tracing::debug!(user = %u, "command.login");
        let creds = match self.mailstore.login_provider.login(&u, &p).await {
            Err(_) => return Response::no("[AUTHENTICATIONFAILED] Authentication failed."),
            Ok(c) => c,
        };

        let mut session = match self.session.lock() {
          Err(_) => return Response::bad("[AUTHENTICATIONFAILED] Unable to acquire lock on session."),
          Ok(s) => s,
        };
        session.creds = Some(creds);
        drop(session);

        Response::ok("Logged in")
    }

    pub async fn lsub(&self, reference: MailboxCodec, mailbox_wildcard: ListMailbox) -> Result<Response, BalError> {
        Response::bad("Not implemented")
    }

    pub async fn list(&self, reference: MailboxCodec, mailbox_wildcard: ListMailbox) -> Result<Response, BalError> {
        Response::bad("Not implemented")
    }

    pub async fn select(&self, mailbox: MailboxCodec) -> Result<Response, BalError> {

        let mut session = match self.session.lock() {
            Err(_) => return Response::no("[SELECTFAILED] Unable to acquire lock on session."),
            Ok(s) => s,
        };

        let mb = Mailbox::new(session.creds.as_ref().unwrap(), "TestMailbox".to_string()).unwrap();
        session.selected = Some(mb);
        drop(session);

        Response::bad("Not implemented")
    }

    pub async fn fetch(&self, sequence_set: SequenceSet, attributes: MacroOrFetchAttributes, uid: bool) -> Result<Response, BalError> {
        Response::bad("Not implemented")
    }
}