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
//! The StateChart for the server ...
//! StateMachine type/struct is created inside here by macros I suppose, so using it
//! needs to "reimport" this file/mod
use super::super::{data::ipc::IFCollectionOutputData, net::ui_data::UiData};
use libp2p::core::{Multiaddr, PeerId};
use smlang::statemachine;

#[derive(PartialEq)]
pub struct Go {}
#[derive(PartialEq)]
pub struct StartData {
    pub own_address: PeerId,
}
#[derive(PartialEq)]
pub struct NewPeerData {
    pub id: PeerId,
    pub addr: Multiaddr,
}

#[derive(PartialEq)]
pub struct UpdateData {
    pub id: PeerId,
    pub data: IFCollectionOutputData,
}

statemachine! {
    *Start + Go = WaitingForPeerAction,
    WaitingForPeerAction + GotANewPeer(NewPeerData) [ not_known ] / process_new_peer = WaitingForPeerAction,
    WaitingForPeerAction + HaveToRemovePeer(PeerId) [ known ] / remove_peer = WaitingForPeerAction,
    WaitingForPeerAction + UpdatePeer(UpdateData) [ is_allowed ] / update_peer = WaitingForPeerAction
}

pub struct AdbfStateChart {
    ui_data: UiData,
}
impl AdbfStateChart {
    pub fn new(ui_data: UiData) -> Self {
        Self { ui_data }
    }
}
impl AdbfStateChart {
    pub fn init(me: Self) -> StateMachine<AdbfStateChart> {
        let mut sm = StateMachine::new(me);
        // todo: is that ok, start here?
        if sm.process_event(Events::Go).is_err() {
            error!("No, no, re-check state chart to hold the Go initial state!");
        }
        sm
    }
}

impl StateMachineContext for AdbfStateChart {
    // 1) guards
    fn not_known(&mut self, event_data: &NewPeerData) -> bool {
        !self.ui_data.has_peer(&event_data.id)
    }
    fn known(&mut self, event_data: &PeerId) -> bool {
        self.ui_data.has_peer(&event_data)
    }
    fn is_allowed(&mut self, _event_data: &UpdateData) -> bool {
        // for now always true
        true
    }

    // 2) actions
    fn process_new_peer(&mut self, peer_data: &NewPeerData) {
        let ref peer_id = peer_data.id;
        let multiaddr = peer_data.addr.clone();
        self.ui_data.register_address(peer_id, &multiaddr);
    }

    fn remove_peer(&mut self, peer_id: &PeerId) {
        self.ui_data.unregister_address(&peer_id);
    }

    fn update_peer(&mut self, update_data: &UpdateData) {
        self.ui_data.update_peer_data(update_data);
    }
}