author | alfadur |
Sat, 22 Feb 2025 19:39:31 +0300 | |
changeset 16091 | 5febd2bc5372 |
parent 16002 | e915ed28726e |
permissions | -rw-r--r-- |
15163 | 1 |
use std::{ |
2 |
cmp::PartialEq, |
|
3 |
collections::HashMap, |
|
4 |
fmt::{Formatter, LowerHex}, |
|
5 |
iter::Iterator, |
|
6 |
}; |
|
12149 | 7 |
|
15074 | 8 |
use self::{ |
9 |
actions::{Destination, DestinationGroup, PendingMessage}, |
|
15075 | 10 |
inanteroom::LoginResult, |
15441 | 11 |
strings::*, |
14694 | 12 |
}; |
15804 | 13 |
use crate::handlers::actions::ToPendingMessage; |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
14 |
use crate::{ |
15074 | 15 |
core::{ |
15520 | 16 |
anteroom::HwAnteroom, |
15075 | 17 |
room::RoomSave, |
16002
e915ed28726e
thread server reference through dispatch to ensure rooms can be auto dropped
alfadur
parents:
15831
diff
changeset
|
18 |
server::{HwRoomOrServer, HwServer}, |
15804 | 19 |
types::{ClientId, Replay, RoomId}, |
15074 | 20 |
}, |
15804 | 21 |
utils, |
22 |
}; |
|
23 |
use hedgewars_network_protocol::{ |
|
24 |
messages::{ |
|
15075 | 25 |
global_chat, server_chat, HwProtocolMessage, HwProtocolMessage::EngineMessage, |
26 |
HwServerMessage, HwServerMessage::*, |
|
15074 | 27 |
}, |
15804 | 28 |
types::{GameCfg, TeamInfo}, |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
29 |
}; |
15804 | 30 |
|
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
31 |
use crate::core::digest::Sha1Digest; |
14693 | 32 |
use base64::encode; |
13805 | 33 |
use log::*; |
14693 | 34 |
use rand::{thread_rng, RngCore}; |
13666 | 35 |
|
15074 | 36 |
mod actions; |
14457 | 37 |
mod checker; |
38 |
mod common; |
|
15075 | 39 |
mod inanteroom; |
15074 | 40 |
mod inlobby; |
15075 | 41 |
mod inroom; |
15439 | 42 |
mod strings; |
12149 | 43 |
|
15520 | 44 |
pub struct ServerState { |
45 |
pub server: HwServer, |
|
46 |
pub anteroom: HwAnteroom, |
|
47 |
} |
|
48 |
||
49 |
impl ServerState { |
|
50 |
pub fn new(clients_limit: usize, rooms_limit: usize) -> Self { |
|
51 |
Self { |
|
52 |
server: HwServer::new(clients_limit, rooms_limit), |
|
53 |
anteroom: HwAnteroom::new(clients_limit), |
|
54 |
} |
|
55 |
} |
|
56 |
} |
|
57 |
||
15102 | 58 |
#[derive(Debug)] |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
59 |
pub struct AccountInfo { |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
60 |
pub is_registered: bool, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
61 |
pub is_admin: bool, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
62 |
pub is_contributor: bool, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
63 |
pub server_hash: Sha1Digest, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
64 |
} |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
65 |
|
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
66 |
pub enum IoTask { |
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
67 |
CheckRegistered { |
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
68 |
nick: String, |
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
69 |
}, |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
70 |
GetAccount { |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
71 |
nick: String, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
72 |
protocol: u16, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
73 |
password_hash: String, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
74 |
client_salt: String, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
75 |
server_salt: String, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
76 |
}, |
15532 | 77 |
GetCheckerAccount { |
78 |
nick: String, |
|
79 |
password: String, |
|
80 |
}, |
|
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
81 |
GetReplay { |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
82 |
id: u32, |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
83 |
}, |
14780 | 84 |
SaveRoom { |
85 |
room_id: RoomId, |
|
86 |
filename: String, |
|
87 |
contents: String, |
|
88 |
}, |
|
89 |
LoadRoom { |
|
90 |
room_id: RoomId, |
|
14781 | 91 |
filename: String, |
92 |
}, |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
93 |
} |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
94 |
|
15102 | 95 |
#[derive(Debug)] |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
96 |
pub enum IoResult { |
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
97 |
AccountRegistered(bool), |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
98 |
Account(Option<AccountInfo>), |
15532 | 99 |
CheckerAccount { is_registered: bool }, |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
100 |
Replay(Option<Replay>), |
14780 | 101 |
SaveRoom(RoomId, bool), |
14781 | 102 |
LoadRoom(RoomId, Option<String>), |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
103 |
} |
14693 | 104 |
|
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
105 |
pub struct Response { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
106 |
client_id: ClientId, |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
107 |
messages: Vec<PendingMessage>, |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
108 |
io_tasks: Vec<IoTask>, |
14696 | 109 |
removed_clients: Vec<ClientId>, |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
110 |
} |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
111 |
|
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
112 |
impl Response { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
113 |
pub fn new(client_id: ClientId) -> Self { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
114 |
Self { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
115 |
client_id, |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
116 |
messages: vec![], |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
117 |
io_tasks: vec![], |
14696 | 118 |
removed_clients: vec![], |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
119 |
} |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
120 |
} |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
121 |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
122 |
#[inline] |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
123 |
pub fn is_empty(&self) -> bool { |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
124 |
self.messages.is_empty() && self.removed_clients.is_empty() && self.io_tasks.is_empty() |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
125 |
} |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
126 |
|
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
127 |
#[inline] |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
128 |
pub fn len(&self) -> usize { |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
129 |
self.messages.len() |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
130 |
} |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
131 |
|
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
132 |
#[inline] |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
133 |
pub fn client_id(&self) -> ClientId { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
134 |
self.client_id |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
135 |
} |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
136 |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
137 |
#[inline] |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
138 |
pub fn add(&mut self, message: PendingMessage) { |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
139 |
self.messages.push(message) |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
140 |
} |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
141 |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
142 |
#[inline] |
15439 | 143 |
pub fn warn(&mut self, message: &str) { |
144 |
self.add(Warning(message.to_string()).send_self()); |
|
145 |
} |
|
146 |
||
147 |
#[inline] |
|
15441 | 148 |
pub fn error(&mut self, message: &str) { |
149 |
self.add(Error(message.to_string()).send_self()); |
|
150 |
} |
|
151 |
||
152 |
#[inline] |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
153 |
pub fn request_io(&mut self, task: IoTask) { |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
154 |
self.io_tasks.push(task) |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
155 |
} |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
156 |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
157 |
pub fn extract_messages<'a, 'b: 'a>( |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
158 |
&'b mut self, |
15075 | 159 |
server: &'a HwServer, |
160 |
) -> impl Iterator<Item = (Vec<ClientId>, HwServerMessage)> + 'a { |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
161 |
let client_id = self.client_id; |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
162 |
self.messages.drain(..).map(move |m| { |
14788 | 163 |
let ids = get_recipients(server, client_id, m.destination); |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
164 |
(ids, m.message) |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
165 |
}) |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
166 |
} |
14696 | 167 |
|
168 |
pub fn remove_client(&mut self, client_id: ClientId) { |
|
169 |
self.removed_clients.push(client_id); |
|
170 |
} |
|
171 |
||
172 |
pub fn extract_removed_clients(&mut self) -> impl Iterator<Item = ClientId> + '_ { |
|
173 |
self.removed_clients.drain(..) |
|
174 |
} |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
175 |
|
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
176 |
pub fn extract_io_tasks(&mut self) -> impl Iterator<Item = IoTask> + '_ { |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
177 |
self.io_tasks.drain(..) |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
178 |
} |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
179 |
} |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
180 |
|
14687
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
181 |
impl Extend<PendingMessage> for Response { |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
182 |
fn extend<T: IntoIterator<Item = PendingMessage>>(&mut self, iter: T) { |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
183 |
for msg in iter { |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
184 |
self.add(msg) |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
185 |
} |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
186 |
} |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
187 |
} |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14673
diff
changeset
|
188 |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
189 |
fn get_recipients( |
15075 | 190 |
server: &HwServer, |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
191 |
client_id: ClientId, |
14788 | 192 |
destination: Destination, |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
193 |
) -> Vec<ClientId> { |
14788 | 194 |
match destination { |
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
195 |
Destination::ToSelf => vec![client_id], |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
196 |
Destination::ToId(id) => vec![id], |
14788 | 197 |
Destination::ToIds(ids) => ids, |
198 |
Destination::ToAll { group, skip_self } => { |
|
14789 | 199 |
let mut ids: Vec<_> = match group { |
15526
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
200 |
DestinationGroup::All => server.iter_client_ids().collect(), |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
201 |
DestinationGroup::Lobby => server.lobby_client_ids().collect(), |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
202 |
DestinationGroup::Protocol(proto) => server.protocol_client_ids(proto).collect(), |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
203 |
DestinationGroup::Room(id) => server.room_client_ids(id).collect(), |
14788 | 204 |
}; |
205 |
||
206 |
if skip_self { |
|
207 |
if let Some(index) = ids.iter().position(|id| *id == client_id) { |
|
208 |
ids.remove(index); |
|
209 |
} |
|
210 |
} |
|
211 |
||
212 |
ids |
|
14672
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
213 |
} |
6e6632068a33
Server action refactoring part 3 of N
alfadur <mail@none>
parents:
14671
diff
changeset
|
214 |
} |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
215 |
} |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
216 |
|
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
217 |
pub fn handle( |
15520 | 218 |
state: &mut ServerState, |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
219 |
client_id: ClientId, |
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
220 |
response: &mut Response, |
15075 | 221 |
message: HwProtocolMessage, |
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
222 |
) { |
12149 | 223 |
match message { |
15075 | 224 |
HwProtocolMessage::Ping => response.add(Pong.send_self()), |
15113 | 225 |
HwProtocolMessage::Pong => (), |
14693 | 226 |
_ => { |
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
227 |
if state.anteroom.has_client(client_id) { |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
228 |
match inanteroom::handle(&mut state.anteroom, client_id, response, message) { |
14693 | 229 |
LoginResult::Unchanged => (), |
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
230 |
LoginResult::Complete(client) => { |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
231 |
let is_checker = client.is_checker; |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
232 |
state.server.add_client(client_id, client); |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
233 |
if !is_checker { |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
234 |
common::get_lobby_join_data(&state.server, response); |
14693 | 235 |
} |
236 |
} |
|
237 |
LoginResult::Exit => { |
|
15520 | 238 |
state.anteroom.remove_client(client_id); |
14696 | 239 |
response.remove_client(client_id); |
14693 | 240 |
} |
241 |
} |
|
15520 | 242 |
} else if state.server.has_client(client_id) { |
14693 | 243 |
match message { |
15075 | 244 |
HwProtocolMessage::Quit(Some(msg)) => { |
15520 | 245 |
common::remove_client( |
246 |
&mut state.server, |
|
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
247 |
&mut state.anteroom, |
15520 | 248 |
response, |
249 |
"User quit: ".to_string() + &msg, |
|
250 |
); |
|
14693 | 251 |
} |
15075 | 252 |
HwProtocolMessage::Quit(None) => { |
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
253 |
common::remove_client( |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
254 |
&mut state.server, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
255 |
&mut state.anteroom, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
256 |
response, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
257 |
"User quit".to_string(), |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
258 |
); |
14693 | 259 |
} |
15075 | 260 |
HwProtocolMessage::Info(nick) => { |
15520 | 261 |
if let Some(client) = state.server.find_client(&nick) { |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
262 |
let admin_sign = if client.is_admin() { "@" } else { "" }; |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
263 |
let master_sign = if client.is_master() { "+" } else { "" }; |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
264 |
let room_info = match client.room_id { |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
265 |
Some(room_id) => { |
15520 | 266 |
let room = state.server.room(room_id); |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
267 |
let status = match room.game_info { |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
268 |
Some(_) if client.teams_in_game == 0 => "(spectating)", |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
269 |
Some(_) => "(playing)", |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
270 |
None => "", |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
271 |
}; |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
272 |
format!( |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
273 |
"[{}{}room {}]{}", |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
274 |
admin_sign, master_sign, room.name, status |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
275 |
) |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
276 |
} |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
277 |
None => format!("[{}lobby]", admin_sign), |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
278 |
}; |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
279 |
|
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
280 |
let info = vec![ |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
281 |
client.nick.clone(), |
14802 | 282 |
"[]".to_string(), |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
283 |
utils::protocol_version_string(client.protocol_number).to_string(), |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
284 |
room_info, |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
285 |
]; |
14802 | 286 |
response.add(Info(info).send_self()) |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
287 |
} else { |
15441 | 288 |
response.add(server_chat(USER_OFFLINE.to_string()).send_self()) |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
289 |
} |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
290 |
} |
15075 | 291 |
HwProtocolMessage::ToggleServerRegisteredOnly => { |
15520 | 292 |
if !state.server.is_admin(client_id) { |
15441 | 293 |
response.warn(ACCESS_DENIED); |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
294 |
} else { |
15520 | 295 |
state |
296 |
.server |
|
297 |
.set_is_registered_only(!state.server.is_registered_only()); |
|
298 |
let msg = if state.server.is_registered_only() { |
|
15441 | 299 |
REGISTERED_ONLY_ENABLED |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
300 |
} else { |
15441 | 301 |
REGISTERED_ONLY_DISABLED |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
302 |
}; |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
303 |
response.add(server_chat(msg.to_string()).send_all()); |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
304 |
} |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
305 |
} |
15075 | 306 |
HwProtocolMessage::Global(msg) => { |
15520 | 307 |
if !state.server.is_admin(client_id) { |
15441 | 308 |
response.warn(ACCESS_DENIED); |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
309 |
} else { |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
310 |
response.add(global_chat(msg).send_all()) |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
311 |
} |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
312 |
} |
15075 | 313 |
HwProtocolMessage::SuperPower => { |
15520 | 314 |
if state.server.enable_super_power(client_id) { |
15518 | 315 |
response.add(server_chat(SUPER_POWER.to_string()).send_self()) |
316 |
} else { |
|
15441 | 317 |
response.warn(ACCESS_DENIED); |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
318 |
} |
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
319 |
} |
15831 | 320 |
#[allow(unused_variables)] |
15075 | 321 |
HwProtocolMessage::Watch(id) => { |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
322 |
#[cfg(feature = "official-server")] |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
323 |
{ |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
324 |
response.request_io(IoTask::GetReplay { id }) |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
325 |
} |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
326 |
|
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
327 |
#[cfg(not(feature = "official-server"))] |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
328 |
{ |
15441 | 329 |
response.warn(REPLAY_NOT_SUPPORTED); |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
330 |
} |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
331 |
} |
15523
f4f6060b536c
add a separate interface for modifying room state
alfadur <mail@none>
parents:
15522
diff
changeset
|
332 |
_ => match state.server.get_room_control(client_id) { |
16002
e915ed28726e
thread server reference through dispatch to ensure rooms can be auto dropped
alfadur
parents:
15831
diff
changeset
|
333 |
HwRoomOrServer::Room(control) => inroom::handle(control, response, message), |
e915ed28726e
thread server reference through dispatch to ensure rooms can be auto dropped
alfadur
parents:
15831
diff
changeset
|
334 |
HwRoomOrServer::Server(server) => { |
e915ed28726e
thread server reference through dispatch to ensure rooms can be auto dropped
alfadur
parents:
15831
diff
changeset
|
335 |
inlobby::handle(server, client_id, response, message) |
e915ed28726e
thread server reference through dispatch to ensure rooms can be auto dropped
alfadur
parents:
15831
diff
changeset
|
336 |
} |
14693 | 337 |
}, |
338 |
} |
|
14671
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14457
diff
changeset
|
339 |
} |
14693 | 340 |
} |
12149 | 341 |
} |
342 |
} |
|
14673
08a8605bafaf
Server action refactoring part 4 of N
alfadur <mail@none>
parents:
14672
diff
changeset
|
343 |
|
15176 | 344 |
pub fn handle_client_accept( |
15520 | 345 |
state: &mut ServerState, |
15176 | 346 |
client_id: ClientId, |
347 |
response: &mut Response, |
|
15517 | 348 |
addr: [u8; 4], |
15176 | 349 |
is_local: bool, |
15831 | 350 |
) -> bool { |
15517 | 351 |
let ban_reason = Some(addr) |
352 |
.filter(|_| !is_local) |
|
15520 | 353 |
.and_then(|a| state.anteroom.find_ip_ban(a)); |
15517 | 354 |
if let Some(reason) = ban_reason { |
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
355 |
response.add(Bye(reason).send_self()); |
15517 | 356 |
response.remove_client(client_id); |
15831 | 357 |
false |
15517 | 358 |
} else { |
359 |
let mut salt = [0u8; 18]; |
|
360 |
thread_rng().fill_bytes(&mut salt); |
|
14693 | 361 |
|
15520 | 362 |
state |
15517 | 363 |
.anteroom |
364 |
.add_client(client_id, encode(&salt), is_local); |
|
14693 | 365 |
|
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
366 |
response |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
367 |
.add(Connected(utils::SERVER_MESSAGE.to_owned(), utils::SERVER_VERSION).send_self()); |
15831 | 368 |
true |
15517 | 369 |
} |
14693 | 370 |
} |
371 |
||
15520 | 372 |
pub fn handle_client_loss(state: &mut ServerState, client_id: ClientId, response: &mut Response) { |
373 |
if state.anteroom.remove_client(client_id).is_none() { |
|
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
374 |
common::remove_client( |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
375 |
&mut state.server, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
376 |
&mut state.anteroom, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
377 |
response, |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
378 |
"Connection reset".to_string(), |
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
379 |
); |
14696 | 380 |
} |
14673
08a8605bafaf
Server action refactoring part 4 of N
alfadur <mail@none>
parents:
14672
diff
changeset
|
381 |
} |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
382 |
|
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
383 |
pub fn handle_io_result( |
15520 | 384 |
state: &mut ServerState, |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
385 |
client_id: ClientId, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
386 |
response: &mut Response, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
387 |
io_result: IoResult, |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
388 |
) { |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
389 |
match io_result { |
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
390 |
IoResult::AccountRegistered(is_registered) => { |
15520 | 391 |
if !is_registered && state.server.is_registered_only() { |
15441 | 392 |
response.add(Bye(REGISTRATION_REQUIRED.to_string()).send_self()); |
14786
8ecdb5c6bb2a
implement info, registered only & super power messages
alfadur
parents:
14785
diff
changeset
|
393 |
response.remove_client(client_id); |
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
394 |
} else if is_registered { |
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
395 |
let client = state.anteroom.get_client(client_id); |
15532 | 396 |
response.add(AskPassword(client.server_salt.clone()).send_self()); |
15520 | 397 |
} else if let Some(client) = state.anteroom.remove_client(client_id) { |
398 |
state.server.add_client(client_id, client); |
|
399 |
common::get_lobby_join_data(&state.server, response); |
|
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
400 |
} |
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
401 |
} |
15532 | 402 |
IoResult::Account(None) => { |
403 |
response.add(Bye(AUTHENTICATION_FAILED.to_string()).send_self()); |
|
404 |
response.remove_client(client_id); |
|
405 |
} |
|
15103
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
406 |
IoResult::Account(Some(info)) => { |
823052e66611
check for account existence before asking passwords
alfadur
parents:
15102
diff
changeset
|
407 |
response.add(ServerAuth(format!("{:x}", info.server_hash)).send_self()); |
15520 | 408 |
if let Some(mut client) = state.anteroom.remove_client(client_id) { |
15441 | 409 |
client.is_registered = info.is_registered; |
410 |
client.is_admin = info.is_admin; |
|
411 |
client.is_contributor = info.is_contributor; |
|
15520 | 412 |
state.server.add_client(client_id, client); |
413 |
common::get_lobby_join_data(&state.server, response); |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
414 |
} |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
415 |
} |
15532 | 416 |
IoResult::CheckerAccount { is_registered } => { |
417 |
if is_registered { |
|
418 |
if let Some(client) = state.anteroom.remove_client(client_id) { |
|
419 |
state.server.add_client(client_id, client); |
|
420 |
response.add(LogonPassed.send_self()); |
|
421 |
} |
|
422 |
} else { |
|
423 |
response.add(Bye(NO_CHECKER_RIGHTS.to_string()).send_self()); |
|
424 |
response.remove_client(client_id); |
|
425 |
} |
|
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
426 |
} |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
427 |
IoResult::Replay(Some(replay)) => { |
15520 | 428 |
let client = state.server.client(client_id); |
15441 | 429 |
let protocol = client.protocol_number; |
14907 | 430 |
let start_msg = if protocol < 58 { |
15441 | 431 |
RoomJoined(vec![client.nick.clone()]) |
14907 | 432 |
} else { |
433 |
ReplayStart |
|
434 |
}; |
|
435 |
response.add(start_msg.send_self()); |
|
436 |
||
15543 | 437 |
common::get_room_config_impl(&replay.config, Destination::ToSelf, response); |
438 |
common::get_teams(replay.teams.iter(), Destination::ToSelf, response); |
|
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
439 |
response.add(RunGame.send_self()); |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
440 |
response.add(ForwardEngineMessage(replay.message_log).send_self()); |
14907 | 441 |
|
442 |
if protocol < 58 { |
|
443 |
response.add(Kicked.send_self()); |
|
444 |
} |
|
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
445 |
} |
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
446 |
IoResult::Replay(None) => { |
15441 | 447 |
response.warn(REPLAY_LOAD_FAILED); |
14785
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14784
diff
changeset
|
448 |
} |
14780 | 449 |
IoResult::SaveRoom(_, true) => { |
15441 | 450 |
response.add(server_chat(ROOM_CONFIG_SAVED.to_string()).send_self()); |
14780 | 451 |
} |
452 |
IoResult::SaveRoom(_, false) => { |
|
15441 | 453 |
response.warn(ROOM_CONFIG_SAVE_FAILED); |
14780 | 454 |
} |
455 |
IoResult::LoadRoom(room_id, Some(contents)) => { |
|
15526
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
456 |
match state.server.set_room_saves(room_id, &contents) { |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
457 |
Ok(_) => response.add(server_chat(ROOM_CONFIG_LOADED.to_string()).send_self()), |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
458 |
Err(e) => { |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
459 |
warn!("Error while deserializing the room configs: {}", e); |
24f692e791d3
disallow mutable rooms outside the server
alfadur <mail@none>
parents:
15523
diff
changeset
|
460 |
response.warn(ROOM_CONFIG_DESERIALIZE_FAILED); |
14780 | 461 |
} |
462 |
} |
|
463 |
} |
|
14781 | 464 |
IoResult::LoadRoom(_, None) => { |
15441 | 465 |
response.warn(ROOM_CONFIG_LOAD_FAILED); |
14780 | 466 |
} |
14779
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
467 |
} |
f43ab2bd76ae
add a thread for internal server IO and implement account checking with it
alfadur
parents:
14696
diff
changeset
|
468 |
} |
15163 | 469 |
|
470 |
#[cfg(test)] |
|
471 |
mod test { |
|
16091
5febd2bc5372
Add server reconnection tokens and anteroom local list of used nicks
alfadur
parents:
16002
diff
changeset
|
472 |
use crate::core::digest::Sha1Digest; |
15163 | 473 |
|
474 |
#[test] |
|
475 |
fn hash_cmp_test() { |
|
476 |
let hash = Sha1Digest([ |
|
477 |
0x37, 0xC4, 0x9F, 0x5C, 0xC3, 0xC9, 0xDB, 0xFC, 0x54, 0xAC, 0x22, 0x04, 0xF6, 0x12, |
|
478 |
0x9A, 0xED, 0x69, 0xB1, 0xC4, 0x5C, |
|
479 |
]); |
|
480 |
||
481 |
assert_eq!(hash, &format!("{:x}", hash)[..]); |
|
482 |
} |
|
483 |
} |