author | alfadur |
Fri, 06 Jul 2018 21:03:03 +0300 | |
changeset 13447 | f748a72432f2 |
parent 13445 | d3c86ade3d4d |
child 13450 | d79795acaa73 |
permissions | -rw-r--r-- |
12147 | 1 |
use mio; |
2 |
||
13416 | 3 |
use protocol::messages::{ |
4 |
HWProtocolMessage, |
|
5 |
HWServerMessage::* |
|
6 |
}; |
|
13419 | 7 |
use server::{ |
8 |
server::HWServer, |
|
9 |
client::ClientId, |
|
10 |
room::HWRoom, |
|
11 |
actions::{Action, Action::*} |
|
12 |
}; |
|
13416 | 13 |
use utils::is_name_illegal; |
14 |
use std::mem::swap; |
|
13423 | 15 |
use base64::{encode, decode}; |
13445
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
16 |
use super::common::rnd_action; |
13423 | 17 |
|
18 |
#[derive(Clone)] |
|
19 |
struct ByMsg<'a> { |
|
20 |
messages: &'a[u8] |
|
21 |
} |
|
22 |
||
23 |
impl <'a> Iterator for ByMsg<'a> { |
|
24 |
type Item = &'a[u8]; |
|
25 |
||
26 |
fn next(&mut self) -> Option<<Self as Iterator>::Item> { |
|
27 |
if let Some(size) = self.messages.get(0) { |
|
28 |
let (msg, next) = self.messages.split_at(*size as usize + 1); |
|
29 |
self.messages = next; |
|
30 |
Some(msg) |
|
31 |
} else { |
|
32 |
None |
|
33 |
} |
|
34 |
} |
|
35 |
} |
|
36 |
||
37 |
fn by_msg(source: &Vec<u8>) -> ByMsg { |
|
38 |
ByMsg {messages: &source[..]} |
|
39 |
} |
|
40 |
||
41 |
const VALID_MESSAGES: &[u8] = |
|
42 |
b"M#+LlRrUuDdZzAaSjJ,NpPwtgfhbc12345\x80\x81\x82\x83\x84\x85\x86\x87\x88\x89\x8A"; |
|
43 |
const NON_TIMED_MESSAGES: &[u8] = b"M#hb"; |
|
44 |
||
13429 | 45 |
#[cfg(canhazslicepatterns)] |
13423 | 46 |
fn is_msg_valid(msg: &[u8], team_indices: &[u8]) -> bool { |
13424 | 47 |
match msg { |
48 |
[size, typ, body..] => VALID_MESSAGES.contains(typ) |
|
49 |
&& match body { |
|
13423 | 50 |
[1...8, team, ..] if *typ == b'h' => team_indices.contains(team), |
51 |
_ => *typ != b'h' |
|
13424 | 52 |
}, |
53 |
_ => false |
|
13423 | 54 |
} |
55 |
} |
|
56 |
||
13429 | 57 |
fn is_msg_valid(msg: &[u8], team_indices: &[u8]) -> bool { |
58 |
if let Some(typ) = msg.get(1) { |
|
59 |
VALID_MESSAGES.contains(typ) |
|
60 |
} else { |
|
61 |
false |
|
62 |
} |
|
63 |
} |
|
64 |
||
13423 | 65 |
fn is_msg_empty(msg: &[u8]) -> bool { |
13429 | 66 |
msg.get(1).filter(|t| **t == b'+').is_some() |
13423 | 67 |
} |
12147 | 68 |
|
13443 | 69 |
fn is_msg_timed(msg: &[u8]) -> bool { |
70 |
msg.get(1).filter(|t| !NON_TIMED_MESSAGES.contains(t)).is_some() |
|
71 |
} |
|
72 |
||
13419 | 73 |
pub fn handle(server: &mut HWServer, client_id: ClientId, message: HWProtocolMessage) { |
13416 | 74 |
use protocol::messages::HWProtocolMessage::*; |
12147 | 75 |
match message { |
13419 | 76 |
Part(None) => server.react(client_id, vec![ |
13416 | 77 |
MoveToLobby("part".to_string())]), |
13419 | 78 |
Part(Some(msg)) => server.react(client_id, vec![ |
13416 | 79 |
MoveToLobby(format!("part: {}", msg))]), |
80 |
Chat(msg) => { |
|
13419 | 81 |
let actions = { |
82 |
let c = &mut server.clients[client_id]; |
|
13444
914f9b970f4d
Implement server-side logic for Rnd
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13443
diff
changeset
|
83 |
let chat_msg = ChatMsg {nick: c.nick.clone(), msg: msg}; |
13419 | 84 |
if let Some(room_id) = c.room_id { |
85 |
vec![chat_msg.send_all().in_room(room_id).but_self().action()] |
|
86 |
} else { |
|
87 |
Vec::new() |
|
88 |
} |
|
89 |
}; |
|
90 |
server.react(client_id, actions); |
|
13416 | 91 |
}, |
13447 | 92 |
Fix => { |
93 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
94 |
if c.is_admin { r.is_fixed = true } |
|
95 |
} |
|
96 |
} |
|
97 |
Unfix => { |
|
98 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
99 |
if c.is_admin { r.is_fixed = false } |
|
100 |
} |
|
101 |
} |
|
102 |
Greeting(text) => { |
|
103 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
104 |
if c.is_admin || c.is_master && !r.is_fixed { |
|
105 |
r.greeting = text |
|
106 |
} |
|
107 |
} |
|
108 |
} |
|
13416 | 109 |
RoomName(new_name) => { |
110 |
let actions = |
|
111 |
if is_name_illegal(&new_name) { |
|
112 |
vec![Warn("Illegal room name! A room name must be between 1-40 characters long, must not have a trailing or leading space and must not have any of these characters: $()*+?[]^{|}".to_string())] |
|
13447 | 113 |
} else if server.room(client_id).map(|r| r.is_fixed).unwrap_or(false) { |
114 |
vec![Warn("Access denied.".to_string())] |
|
13416 | 115 |
} else if server.has_room(&new_name) { |
116 |
vec![Warn("A room with the same name already exists.".to_string())] |
|
117 |
} else { |
|
118 |
let mut old_name = new_name.clone(); |
|
13422 | 119 |
if let (_, Some(r)) = server.client_and_room(client_id) { |
13416 | 120 |
swap(&mut r.name, &mut old_name); |
121 |
vec![SendRoomUpdate(Some(old_name))] |
|
122 |
} else { |
|
123 |
Vec::new() |
|
124 |
} |
|
125 |
}; |
|
13419 | 126 |
server.react(client_id, actions); |
127 |
}, |
|
128 |
ToggleReady => { |
|
129 |
let actions = if let (c, Some(r)) = server.client_and_room(client_id) { |
|
130 |
let flags = if c.is_ready { |
|
131 |
r.ready_players_number -= 1; |
|
132 |
"-r" |
|
133 |
} else { |
|
134 |
r.ready_players_number += 1; |
|
135 |
"+r" |
|
136 |
}; |
|
137 |
c.is_ready = !c.is_ready; |
|
13447 | 138 |
let mut v = |
139 |
vec![ClientFlags(flags.to_string(), vec![c.nick.clone()]) |
|
140 |
.send_all().in_room(r.id).action()]; |
|
141 |
if r.is_fixed && r.ready_players_number as u32 == r.players_number { |
|
142 |
v.push(StartRoomGame(r.id)) |
|
143 |
} |
|
144 |
v |
|
13419 | 145 |
} else { |
146 |
Vec::new() |
|
147 |
}; |
|
148 |
server.react(client_id, actions); |
|
13416 | 149 |
} |
13422 | 150 |
AddTeam(info) => { |
13419 | 151 |
let mut actions = Vec::new(); |
152 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
153 |
let room_id = r.id; |
|
154 |
if r.teams.len() >= r.team_limit as usize { |
|
155 |
actions.push(Warn("Too many teams!".to_string())) |
|
156 |
} else if r.addable_hedgehogs() == 0 { |
|
157 |
actions.push(Warn("Too many hedgehogs!".to_string())) |
|
158 |
} else if r.find_team(|t| t.name == info.name) != None { |
|
159 |
actions.push(Warn("There's already a team with same name in the list.".to_string())) |
|
13423 | 160 |
} else if r.game_info.is_some() { |
13419 | 161 |
actions.push(Warn("Joining not possible: Round is in progress.".to_string())) |
162 |
} else { |
|
163 |
let team = r.add_team(c.id, info); |
|
164 |
c.teams_in_game += 1; |
|
165 |
c.clan = Some(team.color); |
|
166 |
actions.push(TeamAccepted(team.name.clone()) |
|
167 |
.send_self().action()); |
|
168 |
actions.push(TeamAdd(HWRoom::team_info(&c, team)) |
|
169 |
.send_all().in_room(room_id).but_self().action()); |
|
170 |
actions.push(TeamColor(team.name.clone(), team.color) |
|
171 |
.send_all().in_room(room_id).action()); |
|
172 |
actions.push(HedgehogsNumber(team.name.clone(), team.hedgehogs_number) |
|
173 |
.send_all().in_room(room_id).action()); |
|
174 |
actions.push(SendRoomUpdate(None)); |
|
175 |
} |
|
176 |
} |
|
177 |
server.react(client_id, actions); |
|
178 |
}, |
|
179 |
RemoveTeam(name) => { |
|
180 |
let mut actions = Vec::new(); |
|
181 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
182 |
match r.find_team_owner(&name) { |
|
183 |
None => |
|
184 |
actions.push(Warn("Error: The team you tried to remove does not exist.".to_string())), |
|
185 |
Some((id, _)) if id != client_id => |
|
186 |
actions.push(Warn("You can't remove a team you don't own.".to_string())), |
|
187 |
Some((_, name)) => { |
|
188 |
c.teams_in_game -= 1; |
|
189 |
c.clan = r.find_team_color(c.id); |
|
190 |
actions.push(Action::RemoveTeam(name.to_string())); |
|
191 |
} |
|
192 |
} |
|
193 |
}; |
|
194 |
server.react(client_id, actions); |
|
195 |
}, |
|
196 |
SetHedgehogsNumber(team_name, number) => { |
|
197 |
let actions = if let (c, Some(r)) = server.client_and_room(client_id) { |
|
198 |
let room_id = r.id; |
|
199 |
let addable_hedgehogs = r.addable_hedgehogs(); |
|
200 |
if let Some((_, mut team)) = r.find_team_and_owner_mut(|t| t.name == team_name) { |
|
201 |
if !c.is_master { |
|
202 |
vec![ProtocolError("You're not the room master!".to_string())] |
|
203 |
} else if number < 1 || number > 8 |
|
204 |
|| number > addable_hedgehogs + team.hedgehogs_number { |
|
205 |
vec![HedgehogsNumber(team.name.clone(), team.hedgehogs_number) |
|
206 |
.send_self().action()] |
|
207 |
} else { |
|
208 |
team.hedgehogs_number = number; |
|
209 |
vec![HedgehogsNumber(team.name.clone(), number) |
|
210 |
.send_all().in_room(room_id).but_self().action()] |
|
211 |
} |
|
212 |
} else { |
|
213 |
vec![(Warn("No such team.".to_string()))] |
|
214 |
} |
|
215 |
} else { |
|
216 |
Vec::new() |
|
217 |
}; |
|
218 |
server.react(client_id, actions); |
|
219 |
}, |
|
220 |
SetTeamColor(team_name, color) => { |
|
221 |
let mut owner_id = None; |
|
222 |
let actions = if let (c, Some(r)) = server.client_and_room(client_id) { |
|
223 |
let room_id = r.id; |
|
224 |
if let Some((owner, mut team)) = r.find_team_and_owner_mut(|t| t.name == team_name) { |
|
225 |
if !c.is_master { |
|
226 |
vec![ProtocolError("You're not the room master!".to_string())] |
|
227 |
} else if false { |
|
228 |
Vec::new() |
|
229 |
} else { |
|
230 |
owner_id = Some(owner); |
|
231 |
team.color = color; |
|
232 |
vec![TeamColor(team.name.clone(), color) |
|
233 |
.send_all().in_room(room_id).but_self().action()] |
|
234 |
} |
|
235 |
} else { |
|
236 |
vec![(Warn("No such team.".to_string()))] |
|
237 |
} |
|
238 |
} else { |
|
239 |
Vec::new() |
|
240 |
}; |
|
241 |
||
242 |
if let Some(id) = owner_id { |
|
243 |
server.clients[id].clan = Some(color); |
|
244 |
} |
|
245 |
||
246 |
server.react(client_id, actions); |
|
13422 | 247 |
}, |
248 |
Cfg(cfg) => { |
|
249 |
let actions = if let (c, Some(r)) = server.client_and_room(client_id) { |
|
13447 | 250 |
if r.is_fixed { |
251 |
vec![Warn("Access denied.".to_string())] |
|
252 |
} else if !c.is_master { |
|
13422 | 253 |
vec![ProtocolError("You're not the room master!".to_string())] |
254 |
} else { |
|
13439 | 255 |
let v = vec![cfg.to_server_msg() |
256 |
.send_all().in_room(r.id).but_self().action()]; |
|
257 |
r.set_config(cfg); |
|
258 |
v |
|
13422 | 259 |
} |
260 |
} else { |
|
261 |
Vec::new() |
|
262 |
}; |
|
263 |
server.react(client_id, actions); |
|
13419 | 264 |
} |
13423 | 265 |
StartGame => { |
266 |
let actions = if let (_, Some(r)) = server.client_and_room(client_id) { |
|
267 |
vec![StartRoomGame(r.id)] |
|
268 |
} else { |
|
269 |
Vec::new() |
|
270 |
}; |
|
271 |
server.react(client_id, actions); |
|
272 |
} |
|
273 |
EngineMessage(em) => { |
|
274 |
let mut actions = Vec::new(); |
|
275 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
276 |
if c.teams_in_game > 0 { |
|
277 |
let decoding = decode(&em[..]).unwrap(); |
|
278 |
let messages = by_msg(&decoding); |
|
13443 | 279 |
let valid = messages.filter(|m| is_msg_valid(m, &c.team_indices)); |
280 |
let non_empty = valid.clone().filter(|m| !is_msg_empty(m)); |
|
281 |
let sync_msg = valid.clone().filter(|m| is_msg_timed(m)) |
|
282 |
.last().map(|m| if is_msg_empty(m) {Some(encode(m))} else {None}); |
|
13423 | 283 |
|
284 |
let em_response = encode(&valid.flat_map(|msg| msg).cloned().collect::<Vec<_>>()); |
|
285 |
if !em_response.is_empty() { |
|
13428 | 286 |
actions.push(ForwardEngineMessage(vec![em_response]) |
13423 | 287 |
.send_all().in_room(r.id).but_self().action()); |
288 |
} |
|
13427 | 289 |
let em_log = encode(&non_empty.flat_map(|msg| msg).cloned().collect::<Vec<_>>()); |
290 |
if let Some(ref mut info) = r.game_info { |
|
13429 | 291 |
if !em_log.is_empty() { |
13428 | 292 |
info.msg_log.push(em_log); |
293 |
} |
|
13443 | 294 |
if let Some(msg) = sync_msg { |
295 |
info.sync_msg = msg; |
|
13427 | 296 |
} |
297 |
} |
|
13423 | 298 |
} |
299 |
} |
|
300 |
server.react(client_id, actions) |
|
301 |
} |
|
302 |
RoundFinished => { |
|
303 |
let mut actions = Vec::new(); |
|
304 |
if let (c, Some(r)) = server.client_and_room(client_id) { |
|
305 |
if c.is_in_game { |
|
306 |
c.is_in_game = false; |
|
307 |
actions.push(ClientFlags("-g".to_string(), vec![c.nick.clone()]). |
|
308 |
send_all().in_room(r.id).action()); |
|
13426 | 309 |
if r.game_info.is_some() { |
310 |
for team in r.client_teams(c.id) { |
|
311 |
actions.push(SendTeamRemovalMessage(team.name.clone())); |
|
312 |
} |
|
13423 | 313 |
} |
314 |
} |
|
315 |
} |
|
316 |
server.react(client_id, actions) |
|
13444
914f9b970f4d
Implement server-side logic for Rnd
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13443
diff
changeset
|
317 |
}, |
13445
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
318 |
Rnd(v) => { |
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
319 |
let actions = rnd_action(v, server.room(client_id)); |
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
320 |
server.react(client_id, actions) |
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
321 |
}, |
13419 | 322 |
_ => warn!("Unimplemented!") |
12147 | 323 |
} |
324 |
} |