aboutsummaryrefslogtreecommitdiff
path: root/src/imap/command/anonymous.rs
blob: 6ba19cf880426a3b4655a15c0bdb5a4b556bb7d2 (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
82
83
84
85
86
87
88
89
90
91
92
93
94
95
use anyhow::{Error, Result};
use boitalettres::proto::{res::body::Data as Body, Request, Response};
use imap_codec::types::command::CommandBody;
use imap_codec::types::core::AString;
use imap_codec::types::response::{Capability, Data, Status};

use crate::imap::flow;
use crate::login::ArcLoginProvider;
use crate::mail::user::User;

//--- dispatching

pub struct AnonymousContext<'a> {
    pub req: &'a Request,
    pub login_provider: Option<&'a ArcLoginProvider>,
}

pub async fn dispatch(ctx: AnonymousContext<'_>) -> Result<(Response, flow::Transition)> {
    match &ctx.req.command.body {
        CommandBody::Noop => Ok((Response::ok("Noop completed.")?, flow::Transition::None)),
        CommandBody::Capability => ctx.capability().await,
        CommandBody::Logout => ctx.logout().await,
        CommandBody::Login { username, password } => ctx.login(username, password).await,
        cmd => {
            tracing::warn!("Unknown command {:?}", cmd);
            Ok((Response::no("Command unavailable")?, flow::Transition::None))
        }
    }
}

//--- Command controllers, private

impl<'a> AnonymousContext<'a> {
    async fn capability(self) -> Result<(Response, flow::Transition)> {
        let capabilities = vec![Capability::Imap4Rev1, Capability::Idle];
        let res = Response::ok("Server capabilities")?.with_body(Data::Capability(capabilities));
        Ok((res, flow::Transition::None))
    }

    async fn login(
        self,
        username: &AString,
        password: &AString,
    ) -> Result<(Response, flow::Transition)> {
        let (u, p) = (
            String::try_from(username.clone())?,
            String::try_from(password.clone())?,
        );
        tracing::info!(user = %u, "command.login");

        let login_provider = match &self.login_provider {
            Some(lp) => lp,
            None => {
                return Ok((
                    Response::no("Login command not available (already logged in)")?,
                    flow::Transition::None,
                ))
            }
        };

        let creds = match login_provider.login(&u, &p).await {
            Err(e) => {
                tracing::debug!(error=%e, "authentication failed");
                return Ok((
                    Response::no("Authentication failed")?,
                    flow::Transition::None,
                ));
            }
            Ok(c) => c,
        };

        let user = User::new(u.clone(), creds).await?;

        tracing::info!(username=%u, "connected");
        Ok((
            Response::ok("Completed")?,
            flow::Transition::Authenticate(user),
        ))
    }

    // C: 10 logout
    // S: * BYE Logging out
    // S: 10 OK Logout completed.
    async fn logout(self) -> Result<(Response, flow::Transition)> {
        // @FIXME we should implement  From<Vec<Status>> and From<Vec<ImapStatus>> in
        // boitalettres/src/proto/res/body.rs
        Ok((
            Response::ok("Logout completed")?.with_body(vec![Body::Status(
                Status::bye(None, "Logging out")
                    .map_err(|e| Error::msg(e).context("Unable to generate IMAP status"))?,
            )]),
            flow::Transition::Logout,
        ))
    }
}