author | Wuzzy <Wuzzy2@mail.ru> |
Thu, 14 Nov 2019 17:12:31 +0100 | |
changeset 15526 | 266ff128a65a |
parent 15514 | 395be40faa51 |
child 15538 | b907b9071ec5 |
permissions | -rw-r--r-- |
15504 | 1 |
use super::{common::rnd_reply, strings::*}; |
13666 | 2 |
use crate::{ |
15095 | 3 |
core::{ |
15096 | 4 |
room::{HwRoom, RoomFlags, MAX_TEAMS_IN_ROOM}, |
15514 | 5 |
server::{ |
6 |
ChangeMasterError, ChangeMasterResult, HwServer, LeaveRoomResult, StartGameError, |
|
7 |
}, |
|
15095 | 8 |
types, |
9 |
types::{ClientId, GameCfg, RoomId, VoteType, Voting, MAX_HEDGEHOGS_PER_TEAM}, |
|
15096 | 10 |
}, |
11 |
protocol::messages::{ |
|
12 |
add_flags, remove_flags, server_chat, HwProtocolMessage, HwServerMessage::*, |
|
13 |
ProtocolFlags as Flags, |
|
13666 | 14 |
}, |
15504 | 15 |
utils::{is_name_illegal, to_engine_msg}, |
13416 | 16 |
}; |
14478 | 17 |
use base64::{decode, encode}; |
13810 | 18 |
use log::*; |
15047 | 19 |
use std::{cmp::min, iter::once, mem::swap}; |
13423 | 20 |
|
21 |
#[derive(Clone)] |
|
22 |
struct ByMsg<'a> { |
|
14478 | 23 |
messages: &'a [u8], |
13423 | 24 |
} |
25 |
||
14478 | 26 |
impl<'a> Iterator for ByMsg<'a> { |
27 |
type Item = &'a [u8]; |
|
13423 | 28 |
|
29 |
fn next(&mut self) -> Option<<Self as Iterator>::Item> { |
|
30 |
if let Some(size) = self.messages.get(0) { |
|
31 |
let (msg, next) = self.messages.split_at(*size as usize + 1); |
|
32 |
self.messages = next; |
|
33 |
Some(msg) |
|
34 |
} else { |
|
35 |
None |
|
36 |
} |
|
37 |
} |
|
38 |
} |
|
39 |
||
13500 | 40 |
fn by_msg(source: &[u8]) -> ByMsg { |
14478 | 41 |
ByMsg { messages: source } |
13423 | 42 |
} |
43 |
||
44 |
const VALID_MESSAGES: &[u8] = |
|
45 |
b"M#+LlRrUuDdZzAaSjJ,NpPwtgfhbc12345\x80\x81\x82\x83\x84\x85\x86\x87\x88\x89\x8A"; |
|
46 |
const NON_TIMED_MESSAGES: &[u8] = b"M#hb"; |
|
47 |
||
15504 | 48 |
/*#[cfg(canhazslicepatterns)] |
13423 | 49 |
fn is_msg_valid(msg: &[u8], team_indices: &[u8]) -> bool { |
13424 | 50 |
match msg { |
15347 | 51 |
[size, typ, body..MAX] => { |
14478 | 52 |
VALID_MESSAGES.contains(typ) |
53 |
&& match body { |
|
15463 | 54 |
[1..=MAX_HEDGEHOGS_PER_TEAM, team, ..] if *typ == b'h' => { |
14478 | 55 |
team_indices.contains(team) |
56 |
} |
|
57 |
_ => *typ != b'h', |
|
58 |
} |
|
59 |
} |
|
60 |
_ => false, |
|
13423 | 61 |
} |
15504 | 62 |
}*/ |
13423 | 63 |
|
13666 | 64 |
fn is_msg_valid(msg: &[u8], _team_indices: &[u8]) -> bool { |
13429 | 65 |
if let Some(typ) = msg.get(1) { |
66 |
VALID_MESSAGES.contains(typ) |
|
67 |
} else { |
|
68 |
false |
|
69 |
} |
|
70 |
} |
|
71 |
||
13423 | 72 |
fn is_msg_empty(msg: &[u8]) -> bool { |
13429 | 73 |
msg.get(1).filter(|t| **t == b'+').is_some() |
13423 | 74 |
} |
12147 | 75 |
|
13443 | 76 |
fn is_msg_timed(msg: &[u8]) -> bool { |
14478 | 77 |
msg.get(1) |
78 |
.filter(|t| !NON_TIMED_MESSAGES.contains(t)) |
|
79 |
.is_some() |
|
13443 | 80 |
} |
81 |
||
13460 | 82 |
fn voting_description(kind: &VoteType) -> String { |
14478 | 83 |
format!( |
84 |
"New voting started: {}", |
|
85 |
match kind { |
|
86 |
VoteType::Kick(nick) => format!("kick {}", nick), |
|
87 |
VoteType::Map(name) => format!("map {}", name.as_ref().unwrap()), |
|
88 |
VoteType::Pause => "pause".to_string(), |
|
89 |
VoteType::NewSeed => "new seed".to_string(), |
|
90 |
VoteType::HedgehogsPerTeam(number) => format!("hedgehogs per team: {}", number), |
|
91 |
} |
|
92 |
) |
|
13460 | 93 |
} |
94 |
||
15096 | 95 |
fn room_message_flag(msg: &HwProtocolMessage) -> RoomFlags { |
96 |
use crate::protocol::messages::HwProtocolMessage::*; |
|
13494 | 97 |
match msg { |
98 |
ToggleRestrictJoin => RoomFlags::RESTRICTED_JOIN, |
|
99 |
ToggleRestrictTeams => RoomFlags::RESTRICTED_TEAM_ADD, |
|
100 |
ToggleRegisteredOnly => RoomFlags::RESTRICTED_UNREGISTERED_PLAYERS, |
|
14478 | 101 |
_ => RoomFlags::empty(), |
13494 | 102 |
} |
103 |
} |
|
104 |
||
14478 | 105 |
pub fn handle( |
15096 | 106 |
server: &mut HwServer, |
14478 | 107 |
client_id: ClientId, |
14692
455865ccd36c
Server action refactoring part 2 of N
alfadur <mail@none>
parents:
14478
diff
changeset
|
108 |
response: &mut super::Response, |
14478 | 109 |
room_id: RoomId, |
15096 | 110 |
message: HwProtocolMessage, |
14478 | 111 |
) { |
15504 | 112 |
let (client, room) = server.client_and_room_mut(client_id, room_id); |
14718 | 113 |
|
15096 | 114 |
use crate::protocol::messages::HwProtocolMessage::*; |
12147 | 115 |
match message { |
14696
dfe652c53470
Server action refactoring part 6 of N
alfadur <mail@none>
parents:
14692
diff
changeset
|
116 |
Part(msg) => { |
14718 | 117 |
let msg = match msg { |
118 |
Some(s) => format!("part: {}", s), |
|
119 |
None => "part".to_string(), |
|
120 |
}; |
|
15504 | 121 |
|
15514 | 122 |
let result = server.leave_room(client_id); |
123 |
super::common::get_room_leave_data(server, room_id, &msg, result, response); |
|
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
124 |
} |
13416 | 125 |
Chat(msg) => { |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
126 |
response.add( |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
127 |
ChatMsg { |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
128 |
nick: client.nick.clone(), |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
129 |
msg, |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
130 |
} |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
131 |
.send_all() |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
132 |
.in_room(room_id), |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
133 |
); |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
134 |
} |
14809 | 135 |
TeamChat(msg) => { |
136 |
if let Some(ref info) = room.game_info { |
|
137 |
if let Some(clan_color) = room.find_team_color(client_id) { |
|
138 |
let engine_msg = |
|
139 |
to_engine_msg(format!("b{}]{}\x20\x20", client.nick, msg).bytes()); |
|
140 |
let team = room.clan_team_owners(clan_color).collect(); |
|
141 |
response.add(ForwardEngineMessage(vec![engine_msg]).send_many(team)) |
|
142 |
} |
|
143 |
} |
|
144 |
} |
|
13447 | 145 |
Fix => { |
14718 | 146 |
if client.is_admin() { |
147 |
room.set_is_fixed(true); |
|
148 |
room.set_join_restriction(false); |
|
149 |
room.set_team_add_restriction(false); |
|
150 |
room.set_unregistered_players_restriction(true); |
|
15514 | 151 |
} else { |
152 |
response.warn(ACCESS_DENIED) |
|
13447 | 153 |
} |
154 |
} |
|
155 |
Unfix => { |
|
14718 | 156 |
if client.is_admin() { |
157 |
room.set_is_fixed(false); |
|
15514 | 158 |
} else { |
159 |
response.warn(ACCESS_DENIED) |
|
13447 | 160 |
} |
161 |
} |
|
162 |
Greeting(text) => { |
|
14718 | 163 |
if client.is_admin() || client.is_master() && !room.is_fixed() { |
15132 | 164 |
room.greeting = text.unwrap_or(String::new()); |
13447 | 165 |
} |
166 |
} |
|
14809 | 167 |
MaxTeams(count) => { |
168 |
if !client.is_master() { |
|
15504 | 169 |
response.warn(NOT_MASTER); |
15047 | 170 |
} else if !(2..=MAX_TEAMS_IN_ROOM).contains(&count) { |
15504 | 171 |
response.warn("/maxteams: specify number from 2 to 8"); |
14809 | 172 |
} else { |
15504 | 173 |
room.max_teams = count; |
14809 | 174 |
} |
175 |
} |
|
13416 | 176 |
RoomName(new_name) => { |
14696
dfe652c53470
Server action refactoring part 6 of N
alfadur <mail@none>
parents:
14692
diff
changeset
|
177 |
if is_name_illegal(&new_name) { |
15504 | 178 |
response.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: $()*+?[]^{|}"); |
14696
dfe652c53470
Server action refactoring part 6 of N
alfadur <mail@none>
parents:
14692
diff
changeset
|
179 |
} else if server.has_room(&new_name) { |
15504 | 180 |
response.warn("A room with the same name already exists."); |
14696
dfe652c53470
Server action refactoring part 6 of N
alfadur <mail@none>
parents:
14692
diff
changeset
|
181 |
} else { |
15504 | 182 |
let (client, room) = server.client_and_room_mut(client_id, room_id); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
183 |
if room.is_fixed() || room.master_id != Some(client_id) { |
15504 | 184 |
response.warn(ACCESS_DENIED); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
185 |
} else { |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
186 |
let mut old_name = new_name.clone(); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
187 |
swap(&mut room.name, &mut old_name); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
188 |
super::common::get_room_update(Some(old_name), room, Some(&client), response); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
189 |
} |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
190 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
191 |
} |
13419 | 192 |
ToggleReady => { |
14718 | 193 |
let flags = if client.is_ready() { |
194 |
room.ready_players_number -= 1; |
|
14803 | 195 |
remove_flags(&[Flags::Ready]) |
14718 | 196 |
} else { |
197 |
room.ready_players_number += 1; |
|
14803 | 198 |
add_flags(&[Flags::Ready]) |
14718 | 199 |
}; |
13775 | 200 |
|
14718 | 201 |
let msg = if client.protocol_number < 38 { |
202 |
LegacyReady(client.is_ready(), vec![client.nick.clone()]) |
|
203 |
} else { |
|
14803 | 204 |
ClientFlags(flags, vec![client.nick.clone()]) |
14718 | 205 |
}; |
206 |
response.add(msg.send_all().in_room(room.id)); |
|
207 |
client.set_is_ready(!client.is_ready()); |
|
14712 | 208 |
|
14718 | 209 |
if room.is_fixed() && room.ready_players_number == room.players_number { |
15514 | 210 |
let result = server.start_game(room_id); |
211 |
super::common::get_start_game_data(server, room_id, result, response); |
|
13666 | 212 |
} |
13416 | 213 |
} |
14806
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14805
diff
changeset
|
214 |
AddTeam(mut info) => { |
14809 | 215 |
if room.teams.len() >= room.max_teams as usize { |
15504 | 216 |
response.warn("Too many teams!"); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
217 |
} else if room.addable_hedgehogs() == 0 { |
15504 | 218 |
response.warn("Too many hedgehogs!"); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
219 |
} else if room.find_team(|t| t.name == info.name) != None { |
15504 | 220 |
response.warn("There's already a team with same name in the list."); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
221 |
} else if room.game_info.is_some() { |
15504 | 222 |
response.warn("Joining not possible: Round is in progress."); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
223 |
} else if room.is_team_add_restricted() { |
15504 | 224 |
response.warn("This room currently does not allow adding new teams."); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
225 |
} else { |
14806
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14805
diff
changeset
|
226 |
info.owner = client.nick.clone(); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
227 |
let team = room.add_team(client.id, *info, client.protocol_number < 42); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
228 |
client.teams_in_game += 1; |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
229 |
client.clan = Some(team.color); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
230 |
response.add(TeamAccepted(team.name.clone()).send_self()); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
231 |
response.add( |
14806
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14805
diff
changeset
|
232 |
TeamAdd(team.to_protocol()) |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
233 |
.send_all() |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
234 |
.in_room(room_id) |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
235 |
.but_self(), |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
236 |
); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
237 |
response.add( |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
238 |
TeamColor(team.name.clone(), team.color) |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
239 |
.send_all() |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
240 |
.in_room(room_id), |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
241 |
); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
242 |
response.add( |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
243 |
HedgehogsNumber(team.name.clone(), team.hedgehogs_number) |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
244 |
.send_all() |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
245 |
.in_room(room_id), |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
246 |
); |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
247 |
|
15504 | 248 |
let room = server.room(room_id); |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
249 |
let room_master = if let Some(id) = room.master_id { |
15504 | 250 |
Some(server.client(id)) |
13419 | 251 |
} else { |
14709
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
252 |
None |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
253 |
}; |
4569d8d50286
Server action refactoring part B of N
alfadur <mail@none>
parents:
14708
diff
changeset
|
254 |
super::common::get_room_update(None, room, room_master, response); |
13419 | 255 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
256 |
} |
14718 | 257 |
RemoveTeam(name) => match room.find_team_owner(&name) { |
15504 | 258 |
None => response.warn("Error: The team you tried to remove does not exist."), |
259 |
Some((id, _)) if id != client_id => { |
|
260 |
response.warn("You can't remove a team you don't own.") |
|
261 |
} |
|
14718 | 262 |
Some((_, name)) => { |
15504 | 263 |
let name = name.to_string(); |
14718 | 264 |
client.teams_in_game -= 1; |
265 |
client.clan = room.find_team_color(client.id); |
|
15504 | 266 |
room.remove_team(&name); |
267 |
let removed_teams = vec![name]; |
|
268 |
super::common::get_remove_teams_data( |
|
269 |
room_id, |
|
270 |
client.is_in_game(), |
|
271 |
removed_teams, |
|
272 |
response, |
|
273 |
); |
|
14712 | 274 |
|
14718 | 275 |
match room.game_info { |
276 |
Some(ref info) if info.teams_in_game == 0 => { |
|
277 |
super::common::end_game(server, room_id, response) |
|
13419 | 278 |
} |
14718 | 279 |
_ => (), |
13419 | 280 |
} |
14696
dfe652c53470
Server action refactoring part 6 of N
alfadur <mail@none>
parents:
14692
diff
changeset
|
281 |
} |
14718 | 282 |
}, |
13419 | 283 |
SetHedgehogsNumber(team_name, number) => { |
14718 | 284 |
let addable_hedgehogs = room.addable_hedgehogs(); |
285 |
if let Some((_, team)) = room.find_team_and_owner_mut(|t| t.name == team_name) { |
|
15047 | 286 |
let max_hedgehogs = min( |
287 |
MAX_HEDGEHOGS_PER_TEAM, |
|
288 |
addable_hedgehogs + team.hedgehogs_number, |
|
289 |
); |
|
14718 | 290 |
if !client.is_master() { |
15504 | 291 |
response.error(NOT_MASTER); |
15047 | 292 |
} else if !(1..=max_hedgehogs).contains(&number) { |
14718 | 293 |
response |
294 |
.add(HedgehogsNumber(team.name.clone(), team.hedgehogs_number).send_self()); |
|
13419 | 295 |
} else { |
14718 | 296 |
team.hedgehogs_number = number; |
297 |
response.add( |
|
298 |
HedgehogsNumber(team.name.clone(), number) |
|
299 |
.send_all() |
|
300 |
.in_room(room_id) |
|
301 |
.but_self(), |
|
302 |
); |
|
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
303 |
} |
14718 | 304 |
} else { |
15504 | 305 |
response.warn(NO_TEAM); |
13666 | 306 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
307 |
} |
13419 | 308 |
SetTeamColor(team_name, color) => { |
14718 | 309 |
if let Some((owner, team)) = room.find_team_and_owner_mut(|t| t.name == team_name) { |
310 |
if !client.is_master() { |
|
15504 | 311 |
response.error(NOT_MASTER); |
13419 | 312 |
} else { |
14718 | 313 |
team.color = color; |
314 |
response.add( |
|
315 |
TeamColor(team.name.clone(), color) |
|
316 |
.send_all() |
|
317 |
.in_room(room_id) |
|
318 |
.but_self(), |
|
319 |
); |
|
15509 | 320 |
server.client_mut(owner).clan = Some(color); |
13666 | 321 |
} |
14718 | 322 |
} else { |
15504 | 323 |
response.warn(NO_TEAM); |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
324 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
325 |
} |
13422 | 326 |
Cfg(cfg) => { |
14718 | 327 |
if room.is_fixed() { |
15504 | 328 |
response.warn(ACCESS_DENIED); |
14718 | 329 |
} else if !client.is_master() { |
15504 | 330 |
response.error(NOT_MASTER); |
14718 | 331 |
} else { |
332 |
let cfg = match cfg { |
|
333 |
GameCfg::Scheme(name, mut values) => { |
|
334 |
if client.protocol_number == 49 && values.len() >= 2 { |
|
335 |
let mut s = "X".repeat(50); |
|
336 |
s.push_str(&values.pop().unwrap()); |
|
337 |
values.push(s); |
|
13775 | 338 |
} |
14718 | 339 |
GameCfg::Scheme(name, values) |
340 |
} |
|
341 |
cfg => cfg, |
|
342 |
}; |
|
13775 | 343 |
|
14718 | 344 |
response.add(cfg.to_server_msg().send_all().in_room(room.id).but_self()); |
345 |
room.set_config(cfg); |
|
13666 | 346 |
} |
13419 | 347 |
} |
13528 | 348 |
Save(name, location) => { |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
349 |
response.add( |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
350 |
server_chat(format!("Room config saved as {}", name)) |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
351 |
.send_all() |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
352 |
.in_room(room_id), |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
353 |
); |
14718 | 354 |
room.save_config(name, location); |
13528 | 355 |
} |
14806
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14805
diff
changeset
|
356 |
#[cfg(feature = "official-server")] |
13529 | 357 |
SaveRoom(filename) => { |
14718 | 358 |
if client.is_admin() { |
359 |
match room.get_saves() { |
|
14802 | 360 |
Ok(contents) => response.request_io(super::IoTask::SaveRoom { |
361 |
room_id, |
|
362 |
filename, |
|
363 |
contents, |
|
364 |
}), |
|
13529 | 365 |
Err(e) => { |
366 |
warn!("Error while serializing the room configs: {}", e); |
|
15504 | 367 |
response.warn("Unable to serialize the room configs.") |
13529 | 368 |
} |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
369 |
} |
13666 | 370 |
} |
13529 | 371 |
} |
14806
a1077e8d26f4
implement watch message apart from replay deserializing
alfadur
parents:
14805
diff
changeset
|
372 |
#[cfg(feature = "official-server")] |
13529 | 373 |
LoadRoom(filename) => { |
14718 | 374 |
if client.is_admin() { |
14802 | 375 |
response.request_io(super::IoTask::LoadRoom { room_id, filename }); |
13666 | 376 |
} |
13529 | 377 |
} |
13528 | 378 |
Delete(name) => { |
14718 | 379 |
if !room.delete_config(&name) { |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
380 |
response.add(Warning(format!("Save doesn't exist: {}", name)).send_self()); |
13528 | 381 |
} else { |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
382 |
response.add( |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
383 |
server_chat(format!("Room config {} has been deleted", name)) |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
384 |
.send_all() |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
385 |
.in_room(room_id), |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
386 |
); |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
387 |
} |
13528 | 388 |
} |
13450 | 389 |
CallVote(None) => { |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
390 |
response.add(server_chat("Available callvote commands: kick <nickname>, map <name>, pause, newseed, hedgehogs <number>".to_string()) |
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
391 |
.send_self()); |
13450 | 392 |
} |
393 |
CallVote(Some(kind)) => { |
|
14718 | 394 |
let is_in_game = room.game_info.is_some(); |
13450 | 395 |
let error = match &kind { |
396 |
VoteType::Kick(nick) => { |
|
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
397 |
if server |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
398 |
.find_client(&nick) |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
399 |
.filter(|c| c.room_id == Some(room_id)) |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
400 |
.is_some() |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
401 |
{ |
13450 | 402 |
None |
403 |
} else { |
|
13521 | 404 |
Some("/callvote kick: No such user!".to_string()) |
13450 | 405 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
406 |
} |
13450 | 407 |
VoteType::Map(None) => { |
13521 | 408 |
let names: Vec<_> = server.rooms[room_id].saves.keys().cloned().collect(); |
409 |
if names.is_empty() { |
|
410 |
Some("/callvote map: No maps saved in this room!".to_string()) |
|
411 |
} else { |
|
412 |
Some(format!("Available maps: {}", names.join(", "))) |
|
413 |
} |
|
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
414 |
} |
13450 | 415 |
VoteType::Map(Some(name)) => { |
14718 | 416 |
if room.saves.get(&name[..]).is_some() { |
13530 | 417 |
None |
13521 | 418 |
} else { |
13530 | 419 |
Some("/callvote map: No such map!".to_string()) |
13521 | 420 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
421 |
} |
13450 | 422 |
VoteType::Pause => { |
423 |
if is_in_game { |
|
424 |
None |
|
425 |
} else { |
|
13521 | 426 |
Some("/callvote pause: No game in progress!".to_string()) |
13450 | 427 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
428 |
} |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
429 |
VoteType::NewSeed => None, |
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
430 |
VoteType::HedgehogsPerTeam(number) => match number { |
15463 | 431 |
1..=MAX_HEDGEHOGS_PER_TEAM => None, |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
432 |
_ => Some("/callvote hedgehogs: Specify number from 1 to 8.".to_string()), |
13450 | 433 |
}, |
434 |
}; |
|
14718 | 435 |
|
13450 | 436 |
match error { |
437 |
None => { |
|
13460 | 438 |
let msg = voting_description(&kind); |
14810 | 439 |
let voting = Voting::new(kind, server.room_clients(client_id).collect()); |
14707
9f98086de1b6
Server action refactoring part 9 of N
alfadur <mail@none>
parents:
14704
diff
changeset
|
440 |
let room = &mut server.rooms[room_id]; |
9f98086de1b6
Server action refactoring part 9 of N
alfadur <mail@none>
parents:
14704
diff
changeset
|
441 |
room.voting = Some(voting); |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
442 |
response.add(server_chat(msg).send_all().in_room(room_id)); |
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
443 |
super::common::submit_vote( |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
444 |
server, |
15095 | 445 |
types::Vote { |
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
446 |
is_pro: true, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
447 |
is_forced: false, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
448 |
}, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
449 |
response, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
450 |
); |
13450 | 451 |
} |
452 |
Some(msg) => { |
|
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
453 |
response.add(server_chat(msg).send_self()); |
13450 | 454 |
} |
455 |
} |
|
456 |
} |
|
457 |
Vote(vote) => { |
|
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
458 |
super::common::submit_vote( |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
459 |
server, |
15095 | 460 |
types::Vote { |
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
461 |
is_pro: vote, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
462 |
is_forced: false, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
463 |
}, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
464 |
response, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
465 |
); |
13450 | 466 |
} |
467 |
ForceVote(vote) => { |
|
14718 | 468 |
let is_forced = client.is_admin(); |
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
469 |
super::common::submit_vote( |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
470 |
server, |
15095 | 471 |
types::Vote { |
14708
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
472 |
is_pro: vote, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
473 |
is_forced, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
474 |
}, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
475 |
response, |
5122c584804e
Server action refactoring part A of N
alfadur <mail@none>
parents:
14707
diff
changeset
|
476 |
); |
13450 | 477 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
478 |
ToggleRestrictJoin | ToggleRestrictTeams | ToggleRegisteredOnly => { |
14707
9f98086de1b6
Server action refactoring part 9 of N
alfadur <mail@none>
parents:
14704
diff
changeset
|
479 |
if client.is_master() { |
9f98086de1b6
Server action refactoring part 9 of N
alfadur <mail@none>
parents:
14704
diff
changeset
|
480 |
room.flags.toggle(room_message_flag(&message)); |
9f98086de1b6
Server action refactoring part 9 of N
alfadur <mail@none>
parents:
14704
diff
changeset
|
481 |
super::common::get_room_update(None, room, Some(&client), response); |
13494 | 482 |
} |
483 |
} |
|
13423 | 484 |
StartGame => { |
15514 | 485 |
let result = server.start_game(room_id); |
486 |
super::common::get_start_game_data(server, room_id, result, response); |
|
13423 | 487 |
} |
488 |
EngineMessage(em) => { |
|
14718 | 489 |
if client.teams_in_game > 0 { |
490 |
let decoding = decode(&em[..]).unwrap(); |
|
491 |
let messages = by_msg(&decoding); |
|
492 |
let valid = messages.filter(|m| is_msg_valid(m, &client.team_indices)); |
|
493 |
let non_empty = valid.clone().filter(|m| !is_msg_empty(m)); |
|
494 |
let sync_msg = valid.clone().filter(|m| is_msg_timed(m)).last().map(|m| { |
|
495 |
if is_msg_empty(m) { |
|
496 |
Some(encode(m)) |
|
497 |
} else { |
|
498 |
None |
|
499 |
} |
|
500 |
}); |
|
13423 | 501 |
|
14718 | 502 |
let em_response = encode(&valid.flat_map(|msg| msg).cloned().collect::<Vec<_>>()); |
503 |
if !em_response.is_empty() { |
|
504 |
response.add( |
|
505 |
ForwardEngineMessage(vec![em_response]) |
|
506 |
.send_all() |
|
507 |
.in_room(room.id) |
|
508 |
.but_self(), |
|
509 |
); |
|
510 |
} |
|
511 |
let em_log = encode(&non_empty.flat_map(|msg| msg).cloned().collect::<Vec<_>>()); |
|
512 |
if let Some(ref mut info) = room.game_info { |
|
513 |
if !em_log.is_empty() { |
|
514 |
info.msg_log.push(em_log); |
|
13423 | 515 |
} |
14718 | 516 |
if let Some(msg) = sync_msg { |
517 |
info.sync_msg = msg; |
|
13427 | 518 |
} |
13423 | 519 |
} |
520 |
} |
|
521 |
} |
|
522 |
RoundFinished => { |
|
14710
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
523 |
let mut game_ended = false; |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
524 |
if client.is_in_game() { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
525 |
client.set_is_in_game(false); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
526 |
response.add( |
14803 | 527 |
ClientFlags(remove_flags(&[Flags::InGame]), vec![client.nick.clone()]) |
14710
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
528 |
.send_all() |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
529 |
.in_room(room.id), |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
530 |
); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
531 |
let team_names: Vec<_> = room |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
532 |
.client_teams(client_id) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
533 |
.map(|t| t.name.clone()) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
534 |
.collect(); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
535 |
|
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
536 |
if let Some(ref mut info) = room.game_info { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
537 |
info.teams_in_game -= team_names.len() as u8; |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
538 |
if info.teams_in_game == 0 { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
539 |
game_ended = true; |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
540 |
} |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
541 |
|
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
542 |
for team_name in team_names { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
543 |
let msg = once(b'F').chain(team_name.bytes()); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
544 |
response.add( |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
545 |
ForwardEngineMessage(vec![to_engine_msg(msg)]) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
546 |
.send_all() |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
547 |
.in_room(room_id) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
548 |
.but_self(), |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
549 |
); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
550 |
|
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
551 |
let remove_msg = to_engine_msg(once(b'F').chain(team_name.bytes())); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
552 |
if let Some(m) = &info.sync_msg { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
553 |
info.msg_log.push(m.clone()); |
13426 | 554 |
} |
14710
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
555 |
if info.sync_msg.is_some() { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
556 |
info.sync_msg = None |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
557 |
} |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
558 |
info.msg_log.push(remove_msg.clone()); |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
559 |
response.add( |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
560 |
ForwardEngineMessage(vec![remove_msg]) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
561 |
.send_all() |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
562 |
.in_room(room_id) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
563 |
.but_self(), |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
564 |
); |
13423 | 565 |
} |
566 |
} |
|
567 |
} |
|
14710
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
568 |
if game_ended { |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
569 |
super::common::end_game(server, room_id, response) |
aae29ba56aec
Server action refactoring part C of N
alfadur <mail@none>
parents:
14709
diff
changeset
|
570 |
} |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
571 |
} |
13445
d3c86ade3d4d
Send the rnd reply to the room only.
Marcin Mielniczuk <marmistrz.dev@zoho.eu>
parents:
13444
diff
changeset
|
572 |
Rnd(v) => { |
13492 | 573 |
let result = rnd_reply(&v); |
574 |
let mut echo = vec!["/rnd".to_string()]; |
|
575 |
echo.extend(v.into_iter()); |
|
576 |
let chat_msg = ChatMsg { |
|
15509 | 577 |
nick: server.client(client_id).nick.clone(), |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
578 |
msg: echo.join(" "), |
13492 | 579 |
}; |
14697
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
580 |
response.add(chat_msg.send_all().in_room(room_id)); |
9377ee00f1f1
Server action refactoring part 7 of N
alfadur <mail@none>
parents:
14696
diff
changeset
|
581 |
response.add(result.send_all().in_room(room_id)); |
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
582 |
} |
15509 | 583 |
Delegate(nick) => match server.change_master(client_id, room_id, nick) { |
584 |
Ok(ChangeMasterResult { |
|
585 |
old_master_id, |
|
586 |
new_master_id, |
|
587 |
}) => { |
|
588 |
if let Some(master_id) = old_master_id { |
|
589 |
response.add( |
|
590 |
ClientFlags( |
|
591 |
remove_flags(&[Flags::RoomMaster]), |
|
592 |
vec![server.client(master_id).nick.clone()], |
|
593 |
) |
|
594 |
.send_all() |
|
595 |
.in_room(room_id), |
|
596 |
); |
|
597 |
} |
|
598 |
response.add( |
|
599 |
ClientFlags( |
|
600 |
add_flags(&[Flags::RoomMaster]), |
|
601 |
vec![server.client(new_master_id).nick.clone()], |
|
602 |
) |
|
603 |
.send_all() |
|
604 |
.in_room(room_id), |
|
605 |
); |
|
606 |
} |
|
607 |
Err(ChangeMasterError::NoAccess) => { |
|
15504 | 608 |
response.warn("You're not the room master or a server admin!") |
15509 | 609 |
} |
610 |
Err(ChangeMasterError::AlreadyMaster) => { |
|
611 |
response.warn("You're already the room master.") |
|
14805 | 612 |
} |
15509 | 613 |
Err(ChangeMasterError::NoClient) => response.warn("Player is not online."), |
614 |
Err(ChangeMasterError::ClientNotInRoom) => { |
|
615 |
response.warn("The player is not in your room.") |
|
616 |
} |
|
617 |
}, |
|
14704
932ff7683653
Server action refactoring part 8 of N
alfadur <mail@none>
parents:
14697
diff
changeset
|
618 |
_ => warn!("Unimplemented!"), |
12147 | 619 |
} |
620 |
} |