diff --git a/pumpkin/src/client/client_packet.rs b/pumpkin/src/client/client_packet.rs index cba93abc..92199817 100644 --- a/pumpkin/src/client/client_packet.rs +++ b/pumpkin/src/client/client_packet.rs @@ -151,6 +151,7 @@ impl Client { }; if BASIC_CONFIG.online_mode { + // Online mode auth match self .authenticate(server, &shared_secret, &profile.name) .await @@ -163,6 +164,21 @@ impl Client { } } + // Don't allow duplicate UUIDs + if let Some(online_player) = &server.get_player_by_uuid(profile.id).await { + log::debug!("Player (IP '{}', username '{}') tried to log in with the same UUID ('{}') as an online player (IP '{}', username '{}')", &self.address.lock().await.to_string(), &profile.name, &profile.id.to_string(), &online_player.client.address.lock().await.to_string(), &online_player.gameprofile.name); + self.kick("You are already connected to this server").await; + return; + } + + // Don't allow a duplicate username + if let Some(online_player) = &server.get_player_by_name(&profile.name).await { + log::debug!("A player (IP '{}', attempted username '{}') tried to log in with the same username as an online player (UUID '{}', IP '{}', username '{}')", &self.address.lock().await.to_string(), &profile.name, &profile.id.to_string(), &online_player.client.address.lock().await.to_string(), &online_player.gameprofile.name); + self.kick("A player with this username is already connected") + .await; + return; + } + if ADVANCED_CONFIG.packet_compression.enabled { self.enable_compression().await; } @@ -190,8 +206,8 @@ impl Client { if let Some(auth_client) = &server.auth_client { let hash = server.digest_secret(shared_secret); let ip = self.address.lock().await.ip(); - let profile = authentication::authenticate(username, &hash, &ip, auth_client).await?; + // Check if player should join if let Some(actions) = &profile.profile_actions { if ADVANCED_CONFIG diff --git a/pumpkin/src/commands/arg_player.rs b/pumpkin/src/commands/arg_player.rs index 15d26ace..ef59c4da 100644 --- a/pumpkin/src/commands/arg_player.rs +++ b/pumpkin/src/commands/arg_player.rs @@ -26,7 +26,7 @@ pub fn consume_arg_player( name => { // todo: implement any other player than sender for world in &server.worlds { - if world.get_player_by_name(name).is_some() { + if world.get_player_by_name_blocking(name).is_some() { return Ok(name.into()); } } @@ -56,7 +56,7 @@ pub fn parse_arg_player( "@a" | "@e" => todo!(), // todo: implement all players target selector name => { for world in &server.worlds { - if let Some(player) = world.get_player_by_name(name) { + if let Some(player) = world.get_player_by_name_blocking(name) { return Ok(player); } } diff --git a/pumpkin/src/server/mod.rs b/pumpkin/src/server/mod.rs index 14f4b892..4e5feb50 100644 --- a/pumpkin/src/server/mod.rs +++ b/pumpkin/src/server/mod.rs @@ -143,10 +143,20 @@ impl Server { } } - /// Searches every world for a player by name - pub fn get_player_by_name(&self, name: &str) -> Option> { + /// Searches every world for a player by username + pub async fn get_player_by_name(&self, name: &str) -> Option> { for world in &self.worlds { - if let Some(player) = world.get_player_by_name(name) { + if let Some(player) = world.get_player_by_name(name).await { + return Some(player); + } + } + None + } + + /// Searches every world for a player by UUID + pub async fn get_player_by_uuid(&self, id: uuid::Uuid) -> Option> { + for world in &self.worlds { + if let Some(player) = world.get_player_by_uuid(id).await { return Some(player); } } diff --git a/pumpkin/src/world/mod.rs b/pumpkin/src/world/mod.rs index 4cccac5d..a20a009a 100644 --- a/pumpkin/src/world/mod.rs +++ b/pumpkin/src/world/mod.rs @@ -361,9 +361,19 @@ impl World { None } - /// Gets a Player by name - pub fn get_player_by_name(&self, name: &str) -> Option> { - // not sure of blocking lock + /// Gets a Player by username + pub async fn get_player_by_name(&self, name: &str) -> Option> { + for player in self.current_players.lock().await.values() { + if player.gameprofile.name == name { + return Some(player.clone()); + } + } + None + } + + /// Gets a Player by username (Blocking - Legacy use only) + pub fn get_player_by_name_blocking(&self, name: &str) -> Option> { + // TODO: Remove this blocking functions when commands are async. for player in self.current_players.blocking_lock().values() { if player.gameprofile.name == name { return Some(player.clone()); @@ -372,6 +382,11 @@ impl World { None } + /// Gets a Player by UUID + pub async fn get_player_by_uuid(&self, id: uuid::Uuid) -> Option> { + return self.current_players.lock().await.get(&id).cloned(); + } + pub async fn add_player(&self, uuid: uuid::Uuid, player: Arc) { self.current_players.lock().await.insert(uuid, player); }