aboutsummaryrefslogtreecommitdiffstats
path: root/teleterm-web/src/lib.rs
blob: 2ff15146f5b6dfad94f4fab409000dc3893dcf03 (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
mod model;
mod prelude;
mod protocol;
mod views;
mod ws;

use crate::prelude::*;

#[allow(clippy::large_enum_variant)]
#[derive(Clone)]
enum Msg {
    List(seed::fetch::ResponseDataResult<Vec<crate::model::Session>>),
    Refresh,
    StartWatching(String),
    Watch(String, ws::WebSocketEvent),
}

fn init(_: Url, orders: &mut impl Orders<Msg>) -> Init<crate::model::Model> {
    log::trace!("init");
    let model = crate::model::Model::default();
    orders.perform_cmd(model.list());
    Init::new(model)
}

fn update(
    msg: Msg,
    model: &mut crate::model::Model,
    orders: &mut impl Orders<Msg>,
) {
    log::trace!("update");
    match msg {
        Msg::List(sessions) => match sessions {
            Ok(sessions) => {
                log::debug!("got sessions");
                model.update_sessions(sessions);
            }
            Err(e) => {
                log::error!("error getting sessions: {:?}", e);
            }
        },
        Msg::Refresh => {
            log::debug!("refreshing");
            orders.perform_cmd(model.list());
        }
        Msg::StartWatching(id) => {
            log::debug!("watching {}", id);
            model.watch(&id, orders);
        }
        Msg::Watch(id, event) => match event {
            ws::WebSocketEvent::Connected(_) => {
                log::info!("{}: connected", id);
            }
            ws::WebSocketEvent::Disconnected(_) => {
                log::info!("{}: disconnected", id);
            }
            ws::WebSocketEvent::Message(msg) => {
                log::info!("{}: message: {:?}", id, msg);
                let json = msg.data().as_string().unwrap();
                let msg: crate::protocol::Message =
                    serde_json::from_str(&json).unwrap();
                match msg {
                    crate::protocol::Message::TerminalOutput { data } => {
                        model.process(&data);
                    }
                    crate::protocol::Message::Disconnected => {
                        model.disconnect_watch();
                        orders.perform_cmd(model.list());
                    }
                }
            }
            ws::WebSocketEvent::Error(e) => {
                log::error!("{}: error: {:?}", id, e);
            }
        },
    }
}

fn view(model: &crate::model::Model) -> impl View<Msg> {
    log::trace!("view");
    crate::views::page::render(model)
}

#[wasm_bindgen(start)]
pub fn start() {
    console_log::init_with_level(log::Level::Debug).unwrap();
    log::debug!("start");
    seed::App::build(init, update, view).build_and_start();
}