use super::{ClientType, EcsCompPacket, PingMsg}; use crate::sync; use authc::AuthClientError; use common::{ character::{self, CharacterItem}, comp::{self, group::InviteKind}, outcome::Outcome, recipe::RecipeBook, resources::TimeOfDay, trade::PendingTrade, terrain::{Block, TerrainChunk}, uid::Uid, }; use hashbrown::HashMap; use serde::{Deserialize, Serialize}; use std::time::Duration; use vek::*; ///This struct contains all messages the server might send (on different /// streams though) #[derive(Debug, Clone)] pub enum ServerMsg { /// Basic info about server, send ONCE, clients need it to Register Info(ServerInfo), /// Initial data package, send BEFORE Register ONCE. Not Register relevant Init(Box), /// Result to `ClientMsg::Register`. send ONCE RegisterAnswer(ServerRegisterAnswer), ///Msg that can be send ALWAYS as soon as client is registered, e.g. `Chat` General(ServerGeneral), Ping(PingMsg), } /* 2nd Level Enums */ #[derive(Debug, Clone, Serialize, Deserialize)] pub struct ServerInfo { pub name: String, pub description: String, pub git_hash: String, pub git_date: String, pub auth_provider: Option, } /// Reponse To ClientType #[derive(Debug, Clone, Serialize, Deserialize)] #[allow(clippy::clippy::large_enum_variant)] pub enum ServerInit { TooManyPlayers, GameSync { entity_package: sync::EntityPackage, time_of_day: TimeOfDay, max_group_size: u32, client_timeout: Duration, world_map: crate::msg::world_msg::WorldMapMsg, recipe_book: RecipeBook, ability_map: comp::item::tool::AbilityMap, }, } pub type ServerRegisterAnswer = Result<(), RegisterError>; /// Messages sent from the server to the client #[derive(Debug, Clone, Serialize, Deserialize)] pub enum ServerGeneral { //Character Screen related /// An error occurred while loading character data CharacterDataLoadError(String), /// A list of characters belonging to the a authenticated player was sent CharacterListUpdate(Vec), /// An error occurred while creating or deleting a character CharacterActionError(String), /// A new character was created CharacterCreated(character::CharacterId), CharacterSuccess, //Ingame related GroupUpdate(comp::group::ChangeNotification), /// Indicate to the client that they are invited to join a group GroupInvite { inviter: sync::Uid, timeout: std::time::Duration, kind: InviteKind, }, /// Indicate to the client that their sent invite was not invalid and is /// currently pending InvitePending(sync::Uid), /// Note: this could potentially include all the failure cases such as /// inviting yourself in which case the `InvitePending` message could be /// removed and the client could consider their invite pending until /// they receive this message Indicate to the client the result of their /// invite InviteComplete { target: sync::Uid, answer: InviteAnswer, kind: InviteKind, }, /// Trigger cleanup for when the client goes back to the `Registered` state /// from an ingame state ExitInGameSuccess, InventoryUpdate(comp::Inventory, comp::InventoryUpdateEvent), TerrainChunkUpdate { key: Vec2, chunk: Result, ()>, }, TerrainBlockUpdates(HashMap, Block>), SetViewDistance(u32), Outcomes(Vec), Knockback(Vec3), // Always possible PlayerListUpdate(PlayerListUpdate), /// A message to go into the client chat box. The client is responsible for /// formatting the message and turning it into a speech bubble. ChatMsg(comp::ChatMsg), ChatMode(comp::ChatMode), SetPlayerEntity(Uid), TimeOfDay(TimeOfDay), EntitySync(sync::EntitySyncPackage), CompSync(sync::CompSyncPackage), CreateEntity(sync::EntityPackage), DeleteEntity(Uid), Disconnect(DisconnectReason), /// Send a popup notification such as "Waypoint Saved" Notification(Notification), UpdatePendingTrade(usize, PendingTrade), } impl ServerGeneral { pub fn server_msg(chat_type: comp::ChatType, msg: S) -> Self where S: Into, { ServerGeneral::ChatMsg(chat_type.chat_msg(msg)) } } /* end of 2nd level Enums */ /// Inform the client of updates to the player list. #[derive(Debug, Clone, Serialize, Deserialize)] pub enum PlayerListUpdate { Init(HashMap), Add(Uid, PlayerInfo), SelectedCharacter(Uid, CharacterInfo), LevelChange(Uid, u32), Admin(Uid, bool), Remove(Uid), Alias(Uid, String), } #[derive(Debug, Clone, Serialize, Deserialize)] pub struct PlayerInfo { pub is_admin: bool, pub is_online: bool, pub player_alias: String, pub character: Option, } #[derive(Debug, Clone, Serialize, Deserialize)] pub struct CharacterInfo { pub name: String, } #[derive(Debug, Clone, Serialize, Deserialize)] pub enum InviteAnswer { Accepted, Declined, TimedOut, } #[derive(Debug, Clone, Serialize, Deserialize)] pub enum Notification { WaypointSaved, } #[derive(Debug, Clone, Serialize, Deserialize)] pub enum DisconnectReason { /// Server shut down Shutdown, /// Client was kicked Kicked(String), } #[derive(Debug, Clone, Serialize, Deserialize, PartialEq)] pub enum RegisterError { AlreadyLoggedIn, AuthError(String), Banned(String), InvalidCharacter, NotOnWhitelist, //TODO: InvalidAlias, } impl ServerMsg { pub fn verify( &self, c_type: ClientType, registered: bool, presence: Option, ) -> bool { match self { ServerMsg::Info(_) | ServerMsg::Init(_) | ServerMsg::RegisterAnswer(_) => { !registered && presence.is_none() }, ServerMsg::General(g) => { registered && match g { //Character Screen related ServerGeneral::CharacterDataLoadError(_) | ServerGeneral::CharacterListUpdate(_) | ServerGeneral::CharacterActionError(_) | ServerGeneral::CharacterCreated(_) => { c_type != ClientType::ChatOnly && presence.is_none() }, ServerGeneral::CharacterSuccess => { c_type == ClientType::Game && presence.is_none() }, //Ingame related ServerGeneral::GroupUpdate(_) | ServerGeneral::GroupInvite { .. } | ServerGeneral::InvitePending(_) | ServerGeneral::InviteComplete { .. } | ServerGeneral::ExitInGameSuccess | ServerGeneral::InventoryUpdate(_, _) | ServerGeneral::TerrainChunkUpdate { .. } | ServerGeneral::TerrainBlockUpdates(_) | ServerGeneral::SetViewDistance(_) | ServerGeneral::Outcomes(_) | ServerGeneral::Knockback(_) | ServerGeneral::UpdatePendingTrade(_, _) => { c_type == ClientType::Game && presence.is_some() }, // Always possible ServerGeneral::PlayerListUpdate(_) | ServerGeneral::ChatMsg(_) | ServerGeneral::ChatMode(_) | ServerGeneral::SetPlayerEntity(_) | ServerGeneral::TimeOfDay(_) | ServerGeneral::EntitySync(_) | ServerGeneral::CompSync(_) | ServerGeneral::CreateEntity(_) | ServerGeneral::DeleteEntity(_) | ServerGeneral::Disconnect(_) | ServerGeneral::Notification(_) => true, } }, ServerMsg::Ping(_) => true, } } } impl From for RegisterError { fn from(err: AuthClientError) -> Self { Self::AuthError(err.to_string()) } } impl From for ServerGeneral { fn from(v: comp::ChatMsg) -> Self { ServerGeneral::ChatMsg(v) } } impl Into for ServerInfo { fn into(self) -> ServerMsg { ServerMsg::Info(self) } } impl Into for ServerInit { fn into(self) -> ServerMsg { ServerMsg::Init(Box::new(self)) } } impl Into for ServerRegisterAnswer { fn into(self) -> ServerMsg { ServerMsg::RegisterAnswer(self) } } impl Into for ServerGeneral { fn into(self) -> ServerMsg { ServerMsg::General(self) } } impl Into for PingMsg { fn into(self) -> ServerMsg { ServerMsg::Ping(self) } }