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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
use futures::{
channel::mpsc::{self, Receiver, Sender},
sink::SinkExt,
stream::StreamExt,
};
use log::error;
use koibumi_core::{message::UserAgent, net::SocketAddrExt};
use crate::config::Config;
use crate::connection_loop::ConnectionManager;
#[derive(Clone, Debug)]
pub enum Event {
ConnectionCounts {
incoming_initiated: usize,
incoming_connected: usize,
incoming_established: usize,
outgoing_initiated: usize,
outgoing_connected: usize,
outgoing_established: usize,
},
AddrCount(usize),
Established {
addr: SocketAddrExt,
user_agent: UserAgent,
},
Disconnected { addr: SocketAddrExt },
Objects {
missing: usize,
loaded: usize,
uploaded: usize,
},
Stopped,
}
pub enum Command {
Start(Config),
Stop,
Connect(SocketAddrExt),
}
pub enum Response {
Started(Receiver<Event>),
}
pub async fn run(receiver: Receiver<Command>, sender: Sender<Response>) {
let mut receiver = receiver;
let mut sender = sender;
let mut conn_mngr: Option<ConnectionManager> = None;
while let Some(event) = receiver.next().await {
match event {
Command::Start(config) => {
if conn_mngr.is_none() {
let (bm_event_sender, bm_event_receiver) =
mpsc::channel(config.channel_buffer());
let cm = ConnectionManager::new(config, bm_event_sender);
sender
.send(Response::Started(bm_event_receiver))
.await
.unwrap();
conn_mngr = Some(cm);
}
}
Command::Stop => {
if let Some(cm) = conn_mngr {
cm.close_and_join().await;
conn_mngr = None;
}
}
Command::Connect(addr) => {
if let Some(cm) = &mut conn_mngr {
if let Err(err) = cm.connect(addr).await {
error!(target: "koibumi", "{}", err);
}
}
}
}
}
}