summaryrefslogtreecommitdiff
path: root/src/login/snapshot.rs
blob: 13790059993008f5c463bc36dc122601cba65abb (plain)
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
use super::{
    event::{Created, Event},
    Id,
};

// This also implements FromRequestParts (see `./extract.rs`). As a result, it
// can be used as an extractor for endpoints that want to require login, or for
// endpoints that need to behave differently depending on whether the client is
// or is not logged in.
#[derive(Clone, Debug, Eq, PartialEq, serde::Serialize)]
pub struct Login {
    pub id: Id,
    pub name: String,
    // The omission of the hashed password is deliberate, to minimize the
    // chance that it ends up tangled up in debug output or in some other chunk
    // of logic elsewhere.
}

impl Login {
    // Two reasons for this allow:
    //
    // 1. This is used to collect streams using a fold, below, which requires a type consistent with the fold, and
    // 2. It's also consistent with the other history state machine types.
    #[allow(clippy::unnecessary_wraps)]
    fn apply(state: Option<Self>, event: Event) -> Option<Self> {
        match (state, event) {
            (None, Event::Created(event)) => Some(event.into()),
            (state, event) => panic!("invalid message event {event:#?} for state {state:#?}"),
        }
    }
}

impl FromIterator<Event> for Option<Login> {
    fn from_iter<I: IntoIterator<Item = Event>>(events: I) -> Self {
        events.into_iter().fold(None, Login::apply)
    }
}

impl From<&Created> for Login {
    fn from(event: &Created) -> Self {
        event.login.clone()
    }
}

impl From<Created> for Login {
    fn from(event: Created) -> Self {
        event.login
    }
}