Client Cleanup (#159)

# Noteworthy Changes
- Simplified the `client` module and update procedure. Sending packets
is not deferred unless necessary.
- Client events are no longer buffered in `Client` before reaching the
user.
- Expanded `ClientEvent` to account for most packets.
- Most types containing custom `state` now implement `Deref` and
`DerefMut`. This means you don't have to write `.state` all over the
place. `Server` was excluded from this because it does not play well
with the borrow checker.
- Fixed bugs related to entity visibility.
- Client now correctly holds the semaphore permit from the initial
connection.
- Other miscellaneous API changes throughout the project. 

# Known Issues
- Inventory stuff is still incomplete. The inventory examples have been
temporarily disabled.
This commit is contained in:
Ryan Johnson 2022-11-29 03:37:32 -08:00 committed by GitHub
parent 6437381339
commit 58f8197913
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
34 changed files with 2162 additions and 1946 deletions

View file

@ -40,6 +40,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![Dimension { vec![Dimension {
@ -152,14 +153,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, entity)) => {
entity.set_world(world_id);
client.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport(spawn_pos, 0.0, 0.0); client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
@ -178,9 +182,11 @@ impl Config for Game {
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
} }
while client.next_event().is_some() {}
if client.is_disconnected() { if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst); self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id); server.entities.remove(client.entity_id);
if let Some(id) = &server.state.player_list { if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid()); server.player_lists.get_mut(id).remove(client.uuid());
} }

View file

@ -1,8 +1,6 @@
use std::net::SocketAddr; use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
use num::Integer;
use valence::client::DiggingStatus;
use valence::prelude::*; use valence::prelude::*;
pub fn main() -> ShutdownResult { pub fn main() -> ShutdownResult {
@ -31,8 +29,8 @@ struct ClientState {
const MAX_PLAYERS: usize = 10; const MAX_PLAYERS: usize = 10;
const SIZE_X: usize = 100; const SIZE_X: i32 = 100;
const SIZE_Z: usize = 100; const SIZE_Z: i32 = 100;
#[async_trait] #[async_trait]
impl Config for Game { impl Config for Game {
@ -42,6 +40,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![Dimension { vec![Dimension {
@ -70,33 +69,11 @@ impl Config for Game {
server.state.player_list = Some(server.player_lists.insert(()).0); server.state.player_list = Some(server.player_lists.insert(()).0);
// initialize chunks // initialize chunks
for chunk_z in -2..Integer::div_ceil(&(SIZE_Z as i32), &16) + 2 { for z in 0..SIZE_Z {
for chunk_x in -2..Integer::div_ceil(&(SIZE_X as i32), &16) + 2 { for x in 0..SIZE_X {
world.chunks.insert( world
[chunk_x as i32, chunk_z as i32],
UnloadedChunk::default(),
(),
);
}
}
// initialize blocks in the chunks
for chunk_x in 0..Integer::div_ceil(&SIZE_X, &16) {
for chunk_z in 0..Integer::div_ceil(&SIZE_Z, &16) {
let chunk = world
.chunks .chunks
.get_mut((chunk_x as i32, chunk_z as i32)) .set_block_state([x, 0, z], BlockState::GRASS_BLOCK);
.unwrap();
for x in 0..16 {
for z in 0..16 {
let cell_x = chunk_x * 16 + x;
let cell_z = chunk_z * 16 + z;
if cell_x < SIZE_X && cell_z < SIZE_Z {
chunk.set_block_state(x, 63, z, BlockState::GRASS_BLOCK);
}
}
}
} }
} }
} }
@ -123,14 +100,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, entity)) => {
entity.set_world(world_id);
client.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport(spawn_pos, 0.0, 0.0); client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
@ -150,46 +130,24 @@ impl Config for Game {
client.send_message("Welcome to Valence! Build something cool.".italic()); client.send_message("Welcome to Valence! Build something cool.".italic());
} }
if client.is_disconnected() { let player = server.entities.get_mut(client.entity_id).unwrap();
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
let player = server.entities.get_mut(client.state.entity_id).unwrap(); while let Some(event) = client.next_event() {
event.handle_default(client, player);
if client.position().y <= -20.0 {
client.teleport(spawn_pos, client.yaw(), client.pitch());
}
while let Some(event) = handle_event_default(client, player) {
match event { match event {
ClientEvent::Digging { ClientEvent::StartDigging { position, .. } => {
position, status, ..
} => {
match status {
DiggingStatus::Start => {
// Allows clients in creative mode to break blocks. // Allows clients in creative mode to break blocks.
if client.game_mode() == GameMode::Creative { if client.game_mode() == GameMode::Creative {
world.chunks.set_block_state(position, BlockState::AIR); world.chunks.set_block_state(position, BlockState::AIR);
} }
} }
DiggingStatus::Finish => { ClientEvent::FinishDigging { position, .. } => {
// Allows clients in survival mode to break blocks. // Allows clients in survival mode to break blocks.
world.chunks.set_block_state(position, BlockState::AIR); world.chunks.set_block_state(position, BlockState::AIR);
} }
_ => {} ClientEvent::UseItemOnBlock { .. } => {
} // TODO: reimplement when inventories are re-added.
} /*
ClientEvent::InteractWithBlock {
hand,
location,
face,
..
} => {
if hand == Hand::Main { if hand == Hand::Main {
if let Some(stack) = client.held_item() { if let Some(stack) = client.held_item() {
if let Some(held_block_kind) = stack.item.to_block_kind() { if let Some(held_block_kind) = stack.item.to_block_kind() {
@ -200,24 +158,38 @@ impl Config for Game {
{ {
if world if world
.chunks .chunks
.block_state(location) .block_state(position)
.map(|s| s.is_replaceable()) .map(|s| s.is_replaceable())
.unwrap_or(false) .unwrap_or(false)
{ {
world.chunks.set_block_state(location, block_to_place); world.chunks.set_block_state(position, block_to_place);
} else { } else {
let place_at = location.get_in_direction(face); let place_at = position.get_in_direction(face);
world.chunks.set_block_state(place_at, block_to_place); world.chunks.set_block_state(place_at, block_to_place);
} }
} }
} }
} }
} }
*/
} }
_ => {} _ => {}
} }
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
if client.position().y <= -20.0 {
client.teleport(spawn_pos, client.yaw(), client.pitch());
}
true true
}); });
} }

View file

@ -1,3 +1,8 @@
pub fn main() {
todo!("reimplement when inventories are re-added");
}
/*
use std::net::SocketAddr; use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
@ -49,6 +54,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![Dimension { vec![Dimension {
@ -101,7 +107,11 @@ impl Config for Game {
// create chest inventory // create chest inventory
let inv = ConfigurableInventory::new(27, VarInt(2), None); let inv = ConfigurableInventory::new(27, VarInt(2), None);
let (id, _inv) = server.inventories.insert(inv); let title = "Extra".italic()
+ " Chesty".not_italic().bold().color(Color::RED)
+ " Chest".not_italic();
let (id, _inv) = server.inventories.insert(inv, title, ());
server.state.chest = id; server.state.chest = id;
} }
@ -137,14 +147,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, entity)) => {
entity.set_world(world_id);
client.state.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport(spawn_pos, 0.0, 0.0); client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
@ -163,35 +176,17 @@ impl Config for Game {
client.send_message("Welcome to Valence! Sneak to give yourself an item.".italic()); client.send_message("Welcome to Valence! Sneak to give yourself an item.".italic());
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
let player = server.entities.get_mut(client.state.entity_id).unwrap(); let player = server.entities.get_mut(client.state.entity_id).unwrap();
if client.position().y <= -20.0 { while let Some(event) = client.next_event() {
client.teleport(spawn_pos, client.yaw(), client.pitch()); event.handle_default(client, player);
}
while let Some(event) = handle_event_default(client, player) {
match event { match event {
ClientEvent::InteractWithBlock { hand, location, .. } => { ClientEvent::UseItemOnBlock { hand, position, .. } => {
if hand == Hand::Main if hand == Hand::Main
&& world.chunks.block_state(location) == Some(BlockState::CHEST) && world.chunks.block_state(position) == Some(BlockState::CHEST)
{ {
client.send_message("Opening chest!"); client.send_message("Opening chest!");
client.open_inventory( client.open_inventory(server.state.chest);
&server.inventories,
server.state.chest,
"Extra".italic()
+ " Chesty".not_italic().bold().color(Color::RED)
+ " Chest".not_italic(),
);
} }
} }
ClientEvent::CloseScreen { window_id } => { ClientEvent::CloseScreen { window_id } => {
@ -207,6 +202,7 @@ impl Config for Game {
mode, mode,
slot_changes, slot_changes,
carried_item, carried_item,
..
} => { } => {
println!( println!(
"window_id: {:?}, state_id: {:?}, slot_id: {:?}, mode: {:?}, \ "window_id: {:?}, state_id: {:?}, slot_id: {:?}, mode: {:?}, \
@ -244,6 +240,19 @@ impl Config for Game {
} }
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
if client.position().y <= -20.0 {
client.teleport(spawn_pos, client.yaw(), client.pitch());
}
true true
}); });
} }
@ -256,3 +265,4 @@ fn rotate_items(inv: &mut ConfigurableInventory) {
inv.set_slot((i - 1) as SlotId, b); inv.set_slot((i - 1) as SlotId, b);
} }
} }
*/

View file

@ -47,6 +47,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -128,12 +129,12 @@ impl Config for Game {
} }
}; };
player.state.client = client_id; player.set_world(world_id);
player.client = client_id;
client.state.player = player_id; client.player = player_id;
client.state.extra_knockback = true;
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.set_game_mode(GameMode::Survival); client.set_game_mode(GameMode::Survival);
client.teleport( client.teleport(
@ -164,9 +165,42 @@ impl Config for Game {
} }
} }
while let Some(event) = client.next_event() {
let player = server
.entities
.get_mut(client.player)
.expect("missing player entity");
event.handle_default(client, player);
match event {
ClientEvent::StartSprinting => {
client.extra_knockback = true;
}
ClientEvent::StopSprinting => {
client.extra_knockback = false;
}
ClientEvent::InteractWithEntity { entity_id, .. } => {
if let Some((id, target)) = server.entities.get_with_raw_id_mut(entity_id) {
if !target.attacked
&& current_tick - target.last_attack_time >= 10
&& id != client.player
{
target.attacked = true;
target.attacker_pos = client.position();
target.extra_knockback = client.extra_knockback;
target.last_attack_time = current_tick;
client.extra_knockback = false;
}
}
}
_ => {}
}
}
if client.is_disconnected() { if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst); self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.player); server.entities.remove(client.player);
if let Some(id) = &server.state { if let Some(id) = &server.state {
server.player_lists.get_mut(id).remove(client.uuid()); server.player_lists.get_mut(id).remove(client.uuid());
} }
@ -185,70 +219,28 @@ impl Config for Game {
); );
} }
loop {
let player = server
.entities
.get_mut(client.state.player)
.expect("missing player entity");
match handle_event_default(client, player) {
Some(ClientEvent::StartSprinting) => {
client.state.extra_knockback = true;
}
Some(ClientEvent::StopSprinting) => {
client.state.extra_knockback = false;
}
Some(ClientEvent::InteractWithEntity { id, .. }) => {
if let Some(target) = server.entities.get_mut(id) {
if !target.state.attacked
&& current_tick - target.state.last_attack_time >= 10
&& id != client.state.player
{
target.state.attacked = true;
target.state.attacker_pos = client.position();
target.state.extra_knockback = client.state.extra_knockback;
target.state.last_attack_time = current_tick;
client.state.extra_knockback = false;
}
}
}
Some(_) => {}
None => break,
}
}
true true
}); });
for (_, entity) in server.entities.iter_mut() { for (_, entity) in server.entities.iter_mut() {
if entity.state.attacked { if entity.attacked {
entity.state.attacked = false; entity.attacked = false;
if let Some(victim) = server.clients.get_mut(entity.state.client) { if let Some(victim) = server.clients.get_mut(entity.client) {
let victim_pos = Vec2::new(victim.position().x, victim.position().z); let victim_pos = Vec2::new(victim.position().x, victim.position().z);
let attacker_pos = let attacker_pos = Vec2::new(entity.attacker_pos.x, entity.attacker_pos.z);
Vec2::new(entity.state.attacker_pos.x, entity.state.attacker_pos.z);
let dir = (victim_pos - attacker_pos).normalized(); let dir = (victim_pos - attacker_pos).normalized();
let knockback_xz = if entity.state.extra_knockback { let knockback_xz = if entity.extra_knockback { 18.0 } else { 8.0 };
18.0 let knockback_y = if entity.extra_knockback { 8.432 } else { 6.432 };
} else {
8.0
};
let knockback_y = if entity.state.extra_knockback {
8.432
} else {
6.432
};
let vel = Vec3::new(dir.x * knockback_xz, knockback_y, dir.y * knockback_xz); let vel = Vec3::new(dir.x * knockback_xz, knockback_y, dir.y * knockback_xz);
victim.set_velocity(vel.as_()); victim.set_velocity(vel.as_());
entity.push_event(EntityEvent::DamageFromGenericSource); entity.push_event(EntityEvent::DamageFromGenericSource);
entity.push_event(EntityEvent::Damage); entity.push_event(EntityEvent::Damage);
victim.push_entity_event(EntityEvent::DamageFromGenericSource); victim.send_entity_event(EntityEvent::DamageFromGenericSource);
victim.push_entity_event(EntityEvent::Damage); victim.send_entity_event(EntityEvent::Damage);
} }
} }
} }

View file

@ -1,5 +1,5 @@
use std::mem; use std::mem;
use std::net::{Ipv4Addr, SocketAddr, SocketAddrV4}; use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
use num::Integer; use num::Integer;
@ -52,10 +52,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn address(&self) -> SocketAddr {
SocketAddrV4::new(Ipv4Addr::new(0, 0, 0, 0), 25565).into() // TODO remove
}
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![Dimension { vec![Dimension {
@ -128,14 +125,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, entity)) => {
entity.set_world(world_id);
client.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport(spawn_pos, 0.0, 0.0); client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
@ -157,25 +157,12 @@ impl Config for Game {
); );
} }
if client.is_disconnected() { let player = server.entities.get_mut(client.entity_id).unwrap();
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
let player = server.entities.get_mut(client.state.entity_id).unwrap(); while let Some(event) = client.next_event() {
event.handle_default(client, player);
if client.position().y <= 0.0 {
client.teleport(spawn_pos, client.yaw(), client.pitch());
server.state.board.fill(false);
}
while let Some(event) = handle_event_default(client, player) {
match event { match event {
ClientEvent::Digging { position, .. } => { ClientEvent::StartDigging { position, .. } => {
if (0..SIZE_X as i32).contains(&position.x) if (0..SIZE_X as i32).contains(&position.x)
&& (0..SIZE_Z as i32).contains(&position.z) && (0..SIZE_Z as i32).contains(&position.z)
&& position.y == BOARD_Y && position.y == BOARD_Y
@ -195,7 +182,7 @@ impl Config for Game {
server.state.board[index] = true; server.state.board[index] = true;
} }
} }
ClientEvent::InteractWithBlock { hand, .. } => { ClientEvent::UseItemOnBlock { hand, .. } => {
if hand == Hand::Main { if hand == Hand::Main {
client.send_message("I said left click, not right click!".italic()); client.send_message("I said left click, not right click!".italic());
} }
@ -204,6 +191,20 @@ impl Config for Game {
} }
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
return false;
}
if client.position().y <= 0.0 {
client.teleport(spawn_pos, client.yaw(), client.pitch());
server.state.board.fill(false);
}
if let TrackedData::Player(data) = player.data() { if let TrackedData::Player(data) = player.data() {
let sneaking = data.get_pose() == Pose::Sneaking; let sneaking = data.get_pose() == Pose::Sneaking;
if sneaking != server.state.paused { if sneaking != server.state.paused {
@ -212,8 +213,8 @@ impl Config for Game {
Ident::new("block.note_block.pling").unwrap(), Ident::new("block.note_block.pling").unwrap(),
SoundCategory::Block, SoundCategory::Block,
client.position(), client.position(),
0.5f32, 0.5,
if sneaking { 0.5f32 } else { 1f32 }, if sneaking { 0.5 } else { 1.0 },
); );
} }
} }

View file

@ -3,7 +3,6 @@ use std::f64::consts::TAU;
use std::net::SocketAddr; use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
use uuid::Uuid;
use valence::prelude::*; use valence::prelude::*;
pub fn main() -> ShutdownResult { pub fn main() -> ShutdownResult {
@ -29,6 +28,11 @@ struct ServerState {
cows: Vec<EntityId>, cows: Vec<EntityId>,
} }
#[derive(Default)]
struct ClientState {
entity_id: EntityId,
}
const MAX_PLAYERS: usize = 10; const MAX_PLAYERS: usize = 10;
const SPAWN_POS: BlockPos = BlockPos::new(0, 100, -25); const SPAWN_POS: BlockPos = BlockPos::new(0, 100, -25);
@ -36,11 +40,12 @@ const SPAWN_POS: BlockPos = BlockPos::new(0, 100, -25);
#[async_trait] #[async_trait]
impl Config for Game { impl Config for Game {
type ServerState = ServerState; type ServerState = ServerState;
type ClientState = EntityId; type ClientState = ClientState;
type EntityState = (); type EntityState = ();
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -108,14 +113,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state = id, Some((id, entity)) => {
entity.set_world(world_id);
client.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
client.teleport( client.teleport(
@ -146,17 +154,19 @@ impl Config for Game {
if let Some(id) = &server.state.player_list { if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid()); server.player_lists.get_mut(id).remove(client.uuid());
} }
server.entities.remove(client.state); server.entities.remove(client.entity_id);
return false; return false;
} }
let entity = server let entity = server
.entities .entities
.get_mut(client.state) .get_mut(client.entity_id)
.expect("missing player entity"); .expect("missing player entity");
while handle_event_default(client, entity).is_some() {} while let Some(event) = client.next_event() {
event.handle_default(client, entity);
}
true true
}); });
@ -204,9 +214,9 @@ impl Config for Game {
/// Distributes N points on the surface of a unit sphere. /// Distributes N points on the surface of a unit sphere.
fn fibonacci_spiral(n: usize) -> impl Iterator<Item = Vec3<f64>> { fn fibonacci_spiral(n: usize) -> impl Iterator<Item = Vec3<f64>> {
(0..n).map(move |i| {
let golden_ratio = (1.0 + 5_f64.sqrt()) / 2.0; let golden_ratio = (1.0 + 5_f64.sqrt()) / 2.0;
(0..n).map(move |i| {
// Map to unit square // Map to unit square
let x = i as f64 / golden_ratio % 1.0; let x = i as f64 / golden_ratio % 1.0;
let y = i as f64 / n as f64; let y = i as f64 / n as f64;

View file

@ -27,12 +27,9 @@ struct ClientState {
can_respawn: bool, can_respawn: bool,
} }
struct WorldState {
player_list: PlayerListId,
}
#[derive(Default)] #[derive(Default)]
struct ServerState { struct ServerState {
player_list: Option<PlayerListId>,
first_world: WorldId, first_world: WorldId,
second_world: WorldId, second_world: WorldId,
third_world: WorldId, third_world: WorldId,
@ -71,9 +68,10 @@ impl Config for Game {
type ServerState = ServerState; type ServerState = ServerState;
type ClientState = ClientState; type ClientState = ClientState;
type EntityState = (); type EntityState = ();
type WorldState = WorldState; type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![ vec![
@ -107,6 +105,7 @@ impl Config for Game {
// We created server with meaningless default state. // We created server with meaningless default state.
// Let's create three worlds and create new ServerState. // Let's create three worlds and create new ServerState.
server.state = ServerState { server.state = ServerState {
player_list: Some(server.player_lists.insert(()).0),
first_world: create_world(server, FIRST_WORLD_SPAWN_BLOCK, WhichWorld::First), first_world: create_world(server, FIRST_WORLD_SPAWN_BLOCK, WhichWorld::First),
second_world: create_world(server, SECOND_WORLD_SPAWN_BLOCK, WhichWorld::Second), second_world: create_world(server, SECOND_WORLD_SPAWN_BLOCK, WhichWorld::Second),
third_world: create_world(server, THIRD_WORLD_SPAWN_BLOCK, WhichWorld::Third), third_world: create_world(server, THIRD_WORLD_SPAWN_BLOCK, WhichWorld::Third),
@ -131,17 +130,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, entity)) => {
entity.set_world(server.state.first_world);
client.entity_id = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
let first_world_id = server.state.first_world; client.respawn_location = (
let first_world = server.worlds.get(first_world_id).unwrap();
client.state.respawn_location = (
server.state.first_world, server.state.first_world,
block_pos_to_vec(FIRST_WORLD_SPAWN_BLOCK), block_pos_to_vec(FIRST_WORLD_SPAWN_BLOCK),
); );
@ -150,14 +149,14 @@ impl Config for Game {
client.set_spawn_position(FIRST_WORLD_SPAWN_BLOCK, 0.0); client.set_spawn_position(FIRST_WORLD_SPAWN_BLOCK, 0.0);
client.set_flat(true); client.set_flat(true);
client.spawn(first_world_id); client.respawn(server.state.first_world);
client.teleport(client.state.respawn_location.1, 0.0, 0.0); client.teleport(client.respawn_location.1, 0.0, 0.0);
client.set_player_list(first_world.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
server server
.player_lists .player_lists
.get_mut(&first_world.state.player_list) .get_mut(server.state.player_list.as_ref().unwrap())
.insert( .insert(
client.uuid(), client.uuid(),
client.username(), client.username(),
@ -178,28 +177,17 @@ impl Config for Game {
// TODO after inventory support is added, show interaction with compass. // TODO after inventory support is added, show interaction with compass.
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(list) = client.player_list() {
server.player_lists.get_mut(list).remove(client.uuid());
}
return false;
}
// Handling respawn locations // Handling respawn locations
if !client.state.can_respawn { if !client.can_respawn {
if client.position().y < 0.0 { if client.position().y < 0.0 {
client.state.can_respawn = true; client.can_respawn = true;
client.kill(None, "You fell"); client.kill(None, "You fell");
// You could have also killed the player with `Client::set_health_and_food`, // You could have also killed the player with `Client::set_health_and_food`,
// however you cannot send a message to the death screen // however you cannot send a message to the death screen
// that way // that way
if client.world() == server.state.third_world { if client.world() == server.state.third_world {
// Falling in third world gets you back to the first world // Falling in third world gets you back to the first world
client.state.respawn_location = ( client.respawn_location = (
server.state.first_world, server.state.first_world,
block_pos_to_vec(FIRST_WORLD_SPAWN_BLOCK), block_pos_to_vec(FIRST_WORLD_SPAWN_BLOCK),
); );
@ -207,7 +195,7 @@ impl Config for Game {
} else { } else {
// falling in first and second world will cause player to spawn in third // falling in first and second world will cause player to spawn in third
// world // world
client.state.respawn_location = ( client.respawn_location = (
server.state.third_world, server.state.third_world,
block_pos_to_vec(THIRD_WORLD_SPAWN_BLOCK), block_pos_to_vec(THIRD_WORLD_SPAWN_BLOCK),
); );
@ -222,15 +210,15 @@ impl Config for Game {
if client.position().x >= LEFT_DEATH_LINE as f64 { if client.position().x >= LEFT_DEATH_LINE as f64 {
// Client went to the left, he dies // Client went to the left, he dies
client.state.can_respawn = true; client.can_respawn = true;
client.kill(None, death_msg); client.kill(None, death_msg);
} }
if client.position().x <= RIGHT_DEATH_LINE as f64 { if client.position().x <= RIGHT_DEATH_LINE as f64 {
// Client went to the right, he dies and spawns in world2 // Client went to the right, he dies and spawns in world2
client.state.can_respawn = true; client.can_respawn = true;
client.kill(None, death_msg); client.kill(None, death_msg);
client.state.respawn_location = ( client.respawn_location = (
server.state.second_world, server.state.second_world,
block_pos_to_vec(SECOND_WORLD_SPAWN_BLOCK), block_pos_to_vec(SECOND_WORLD_SPAWN_BLOCK),
); );
@ -239,33 +227,46 @@ impl Config for Game {
} }
} }
let player = server.entities.get_mut(client.state.entity_id).unwrap(); let player = server.entities.get_mut(client.entity_id).unwrap();
while let Some(event) = handle_event_default(client, player) { while let Some(event) = client.next_event() {
event.handle_default(client, player);
match event { match event {
ClientEvent::RespawnRequest => { ClientEvent::PerformRespawn => {
if !client.state.can_respawn { if !client.can_respawn {
client.disconnect("Unexpected RespawnRequest"); client.disconnect("Unexpected PerformRespawn");
return false; return false;
} }
// Let's respawn our player. `spawn` will load the world, but we are // Let's respawn our player. `spawn` will load the world, but we are
// responsible for teleporting the player. // responsible for teleporting the player.
// You can store respawn however you want, for example in `Client`'s state. // You can store respawn however you want, for example in `Client`'s state.
let spawn = client.state.respawn_location; let spawn = client.respawn_location;
client.spawn(spawn.0); client.respawn(spawn.0);
player.set_world(spawn.0);
client.teleport(spawn.1, 0.0, 0.0); client.teleport(spawn.1, 0.0, 0.0);
client.state.can_respawn = false; client.can_respawn = false;
} }
ClientEvent::StartSneaking => { ClientEvent::StartSneaking => {
// Roll the credits, respawn after // Roll the credits, respawn after
client.state.can_respawn = true; client.can_respawn = true;
client.win_game(true); client.win_game(true);
} }
_ => {} _ => {}
} }
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.entity_id);
if let Some(list) = client.player_list() {
server.player_lists.get_mut(list).remove(client.uuid());
}
return false;
}
true true
}); });
} }
@ -279,10 +280,7 @@ fn create_world(server: &mut Server<Game>, spawn_pos: BlockPos, world_type: Whic
WhichWorld::Third => server.shared.dimensions().nth(1).unwrap(), WhichWorld::Third => server.shared.dimensions().nth(1).unwrap(),
}; };
let player_list = server.player_lists.insert(()).0; let (world_id, world) = server.worlds.insert(dimension.0, ());
let (world_id, world) = server
.worlds
.insert(dimension.0, WorldState { player_list });
// Create chunks // Create chunks
for chunk_z in -3..3 { for chunk_z in -3..3 {

View file

@ -42,6 +42,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -305,14 +306,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.player = id, Some((id, entity)) => {
entity.set_world(world_id);
client.player = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
client.teleport( client.teleport(
@ -344,12 +348,17 @@ impl Config for Game {
); );
} }
let entity = server.entities.get_mut(client.player).unwrap();
while let Some(event) = client.next_event() {
event.handle_default(client, entity);
}
if client.is_disconnected() { if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst); self.player_count.fetch_sub(1, Ordering::SeqCst);
if let Some(id) = &server.state { if let Some(id) = &server.state {
server.player_lists.get_mut(id).remove(client.uuid()); server.player_lists.get_mut(id).remove(client.uuid());
} }
server.entities.remove(client.state.player); server.entities.remove(client.player);
return false; return false;
} }
@ -359,19 +368,18 @@ impl Config for Game {
let origin = Vec3::new(client_pos.x, client_pos.y + PLAYER_EYE_HEIGHT, client_pos.z); let origin = Vec3::new(client_pos.x, client_pos.y + PLAYER_EYE_HEIGHT, client_pos.z);
let direction = from_yaw_and_pitch(client.yaw() as f64, client.pitch() as f64); let direction = from_yaw_and_pitch(client.yaw() as f64, client.pitch() as f64);
let not_self_or_bullet = |hit: &RaycastHit| { let not_self_or_bullet = |hit: &RaycastHit| {
hit.entity != client.state.player && hit.entity != client.state.shulker_bullet hit.entity != client.player && hit.entity != client.shulker_bullet
}; };
if let Some(hit) = world if let Some(hit) = world
.spatial_index .spatial_index
.raycast(origin, direction, not_self_or_bullet) .raycast(origin, direction, not_self_or_bullet)
{ {
let bullet = let bullet = if let Some(bullet) = server.entities.get_mut(client.shulker_bullet) {
if let Some(bullet) = server.entities.get_mut(client.state.shulker_bullet) {
bullet bullet
} else { } else {
let (id, bullet) = server.entities.insert(EntityKind::ShulkerBullet, ()); let (id, bullet) = server.entities.insert(EntityKind::ShulkerBullet, ());
client.state.shulker_bullet = id; client.shulker_bullet = id;
bullet.set_world(world_id); bullet.set_world(world_id);
bullet bullet
}; };
@ -385,17 +393,10 @@ impl Config for Game {
client.set_action_bar("Intersection".color(Color::GREEN)); client.set_action_bar("Intersection".color(Color::GREEN));
} else { } else {
server.entities.remove(client.state.shulker_bullet); server.entities.remove(client.shulker_bullet);
client.set_action_bar("No Intersection".color(Color::RED)); client.set_action_bar("No Intersection".color(Color::RED));
} }
while handle_event_default(
client,
server.entities.get_mut(client.state.player).unwrap(),
)
.is_some()
{}
true true
}); });
} }

View file

@ -1,3 +1,8 @@
pub fn main() {
todo!("reimplement when inventories are re-added");
}
/*
use std::net::SocketAddr; use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
@ -47,6 +52,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn dimensions(&self) -> Vec<Dimension> { fn dimensions(&self) -> Vec<Dimension> {
vec![Dimension { vec![Dimension {
@ -135,7 +141,7 @@ impl Config for Game {
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport(spawn_pos, 0.0, 0.0); client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
@ -227,3 +233,4 @@ fn play_note(client: &mut Client<Game>, player: &mut Entity<Game>, clicked_slot:
}); });
} }
} }
*/

View file

@ -26,6 +26,11 @@ struct ServerState {
player_list: Option<PlayerListId>, player_list: Option<PlayerListId>,
} }
#[derive(Default)]
struct ChunkState {
keep_loaded: bool,
}
#[derive(Default)] #[derive(Default)]
struct ClientState { struct ClientState {
entity_id: EntityId, entity_id: EntityId,
@ -56,9 +61,9 @@ impl Config for Game {
type ClientState = ClientState; type ClientState = ClientState;
type EntityState = (); type EntityState = ();
type WorldState = (); type WorldState = ();
/// If the chunk should stay loaded at the end of the tick. type ChunkState = ChunkState;
type ChunkState = bool;
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -80,8 +85,6 @@ impl Config for Game {
} }
fn update(&self, server: &mut Server<Self>) { fn update(&self, server: &mut Server<Self>) {
//let (world_id, world) = server.worlds.iter_mut().next().unwrap();
server.clients.retain(|_, client| { server.clients.retain(|_, client| {
if client.created_this_tick() { if client.created_this_tick() {
if self if self
@ -95,11 +98,15 @@ impl Config for Game {
return false; return false;
} }
let (world_id, world) = server.worlds.insert(DimensionId::default(), ());
match server match server
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => { Some((id, entity)) => {
entity.set_world(world_id);
// create client state // create client state
client.state = ClientState { client.state = ClientState {
entity_id: id, entity_id: id,
@ -108,32 +115,28 @@ impl Config for Game {
combo: 0, combo: 0,
last_block_timestamp: 0, last_block_timestamp: 0,
target_y: 0, target_y: 0,
world_id: WorldId::NULL, world_id,
}; };
} }
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
server.worlds.remove(world_id);
return false; return false;
} }
} }
let (world_id, world) = server.worlds.insert(DimensionId::default(), ());
client.state.world_id = world_id;
for chunk_z in -1..3 { for chunk_z in -1..3 {
for chunk_x in -2..2 { for chunk_x in -2..2 {
world.chunks.insert( world.chunks.insert(
(chunk_x as i32, chunk_z as i32), (chunk_x as i32, chunk_z as i32),
UnloadedChunk::default(), UnloadedChunk::default(),
true, ChunkState { keep_loaded: true },
); );
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
// client.teleport(spawn_pos, 0.0, 0.0);
client.set_player_list(server.state.player_list.clone()); client.set_player_list(server.state.player_list.clone());
if let Some(id) = &server.state.player_list { if let Some(id) = &server.state.player_list {
@ -152,34 +155,19 @@ impl Config for Game {
reset(client, world); reset(client, world);
} }
let (world_id, world) = server let world_id = client.world_id;
.worlds let world = server.worlds.get_mut(world_id).unwrap();
.iter_mut()
.find(|w| w.0 == client.state.world_id)
.unwrap();
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.state.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
for block in &client.state.blocks {
world.chunks.set_block_state(*block, BlockState::AIR);
}
client.state.blocks.clear();
client.state.score = 0;
server.worlds.remove(world_id);
return false;
}
let p = client.position(); let p = client.position();
for pos in chunks_in_view_distance(ChunkPos::at(p.x, p.z), 3) { for pos in chunks_in_view_distance(ChunkPos::at(p.x, p.z), 3) {
if let Some(chunk) = world.chunks.get_mut(pos) { if let Some(chunk) = world.chunks.get_mut(pos) {
chunk.state = true; chunk.keep_loaded = true;
} else { } else {
world.chunks.insert(pos, UnloadedChunk::default(), true); world.chunks.insert(
pos,
UnloadedChunk::default(),
ChunkState { keep_loaded: true },
);
} }
} }
@ -187,7 +175,6 @@ impl Config for Game {
client.send_message( client.send_message(
"Your score was ".italic() "Your score was ".italic()
+ client + client
.state
.score .score
.to_string() .to_string()
.color(Color::GOLD) .color(Color::GOLD)
@ -199,32 +186,31 @@ impl Config for Game {
} }
let pos_under_player = BlockPos::new( let pos_under_player = BlockPos::new(
(client.position().x - 0.5f64).round() as i32, (client.position().x - 0.5).round() as i32,
client.position().y as i32 - 1, client.position().y as i32 - 1,
(client.position().z - 0.5f64).round() as i32, (client.position().z - 0.5).round() as i32,
); );
if let Some(index) = client if let Some(index) = client
.state
.blocks .blocks
.iter() .iter()
.position(|block| *block == pos_under_player) .position(|block| *block == pos_under_player)
{ {
if index > 0 { if index > 0 {
let power_result = 2.0f32.powf((client.state.combo as f32) / 45.0); let power_result = 2.0f32.powf((client.combo as f32) / 45.0);
let max_time_taken = (1000.0f32 * (index as f32) / power_result) as u128; let max_time_taken = (1000.0f32 * (index as f32) / power_result) as u128;
let current_time_millis = SystemTime::now() let current_time_millis = SystemTime::now()
.duration_since(UNIX_EPOCH) .duration_since(UNIX_EPOCH)
.unwrap() .unwrap()
.as_millis(); .as_millis();
if current_time_millis - client.state.last_block_timestamp < max_time_taken { if current_time_millis - client.last_block_timestamp < max_time_taken {
client.state.combo += index as u32 client.combo += index as u32
} else { } else {
client.state.combo = 0 client.combo = 0
} }
let pitch = 0.9 + ((client.state.combo as f32) - 1.0) * 0.05; let pitch = 0.9 + ((client.combo as f32) - 1.0) * 0.05;
for _ in 0..index { for _ in 0..index {
generate_next_block(client, world, true) generate_next_block(client, world, true)
@ -239,12 +225,7 @@ impl Config for Game {
); );
client.set_title( client.set_title(
"", "",
client client.score.to_string().color(Color::LIGHT_PURPLE).bold(),
.state
.score
.to_string()
.color(Color::LIGHT_PURPLE)
.bold(),
SetTitleAnimationTimes { SetTitleAnimationTimes {
fade_in: 0, fade_in: 0,
stay: 7, stay: 7,
@ -254,23 +235,33 @@ impl Config for Game {
} }
} }
while handle_event_default( let player = server.entities.get_mut(client.entity_id).unwrap();
client,
server.entities.get_mut(client.state.entity_id).unwrap(), while let Some(event) = client.next_event() {
) event.handle_default(client, player);
.is_some() }
{}
// Remove chunks outside the view distance of players. // Remove chunks outside the view distance of players.
world.chunks.retain(|_, chunk| { world.chunks.retain(|_, chunk| {
if chunk.state { if chunk.keep_loaded {
chunk.state = false; chunk.keep_loaded = false;
true true
} else { } else {
false false
} }
}); });
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
server.entities.remove(client.entity_id);
if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid());
}
server.worlds.remove(world_id);
return false;
}
true true
}); });
} }
@ -283,19 +274,19 @@ fn reset(client: &mut Client<Game>, world: &mut World<Game>) {
world.chunks.insert( world.chunks.insert(
(chunk_x as i32, chunk_z as i32), (chunk_x as i32, chunk_z as i32),
UnloadedChunk::default(), UnloadedChunk::default(),
true, ChunkState { keep_loaded: true },
); );
} }
} }
client.state.score = 0; client.score = 0;
client.state.combo = 0; client.combo = 0;
for block in &client.state.blocks { for block in &client.blocks {
world.chunks.set_block_state(*block, BlockState::AIR); world.chunks.set_block_state(*block, BlockState::AIR);
} }
client.state.blocks.clear(); client.blocks.clear();
client.state.blocks.push_back(START_POS); client.blocks.push_back(START_POS);
world.chunks.set_block_state(START_POS, BlockState::STONE); world.chunks.set_block_state(START_POS, BlockState::STONE);
for _ in 0..10 { for _ in 0..10 {
@ -315,19 +306,19 @@ fn reset(client: &mut Client<Game>, world: &mut World<Game>) {
fn generate_next_block(client: &mut Client<Game>, world: &mut World<Game>, in_game: bool) { fn generate_next_block(client: &mut Client<Game>, world: &mut World<Game>, in_game: bool) {
if in_game { if in_game {
let removed_block = client.state.blocks.pop_front().unwrap(); let removed_block = client.blocks.pop_front().unwrap();
world.chunks.set_block_state(removed_block, BlockState::AIR); world.chunks.set_block_state(removed_block, BlockState::AIR);
client.state.score += 1 client.score += 1
} }
let last_pos = *client.state.blocks.back().unwrap(); let last_pos = *client.blocks.back().unwrap();
let block_pos = generate_random_block(last_pos, client.state.target_y); let block_pos = generate_random_block(last_pos, client.target_y);
if last_pos.y == START_POS.y { if last_pos.y == START_POS.y {
client.state.target_y = 0 client.target_y = 0
} else if last_pos.y < START_POS.y - 30 || last_pos.y > START_POS.y + 30 { } else if last_pos.y < START_POS.y - 30 || last_pos.y > START_POS.y + 30 {
client.state.target_y = START_POS.y; client.target_y = START_POS.y;
} }
let mut rng = rand::thread_rng(); let mut rng = rand::thread_rng();
@ -335,10 +326,10 @@ fn generate_next_block(client: &mut Client<Game>, world: &mut World<Game>, in_ga
world world
.chunks .chunks
.set_block_state(block_pos, *BLOCK_TYPES.choose(&mut rng).unwrap()); .set_block_state(block_pos, *BLOCK_TYPES.choose(&mut rng).unwrap());
client.state.blocks.push_back(block_pos); client.blocks.push_back(block_pos);
// Combo System // Combo System
client.state.last_block_timestamp = SystemTime::now() client.last_block_timestamp = SystemTime::now()
.duration_since(UNIX_EPOCH) .duration_since(UNIX_EPOCH)
.unwrap() .unwrap()
.as_millis(); .as_millis();

View file

@ -2,7 +2,6 @@ use std::net::SocketAddr;
use std::sync::atomic::{AtomicUsize, Ordering}; use std::sync::atomic::{AtomicUsize, Ordering};
use valence::prelude::*; use valence::prelude::*;
use valence_protocol::packets::c2s::play::ResourcePackC2s;
use valence_protocol::types::EntityInteraction; use valence_protocol::types::EntityInteraction;
pub fn main() -> ShutdownResult { pub fn main() -> ShutdownResult {
@ -14,7 +13,7 @@ pub fn main() -> ShutdownResult {
}, },
ServerState { ServerState {
player_list: None, player_list: None,
sheep_id: None, sheep_id: EntityId::NULL,
}, },
) )
} }
@ -25,7 +24,7 @@ struct Game {
struct ServerState { struct ServerState {
player_list: Option<PlayerListId>, player_list: Option<PlayerListId>,
sheep_id: Option<EntityId>, sheep_id: EntityId,
} }
#[derive(Default)] #[derive(Default)]
@ -45,6 +44,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -73,7 +73,7 @@ impl Config for Game {
} }
let (sheep_id, sheep) = server.entities.insert(EntityKind::Sheep, ()); let (sheep_id, sheep) = server.entities.insert(EntityKind::Sheep, ());
server.state.sheep_id = Some(sheep_id); server.state.sheep_id = sheep_id;
sheep.set_world(world_id); sheep.set_world(world_id);
sheep.set_position([ sheep.set_position([
SPAWN_POS.x as f64 + 0.5, SPAWN_POS.x as f64 + 0.5,
@ -108,14 +108,14 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, _)) => client.entity_id = id,
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
client.teleport( client.teleport(
@ -140,6 +140,10 @@ impl Config for Game {
); );
} }
client.send_message(
"Hit the sheep above you to prompt for the resource pack again.".italic(),
);
set_example_pack(client); set_example_pack(client);
} }
@ -148,44 +152,37 @@ impl Config for Game {
if let Some(id) = &server.state.player_list { if let Some(id) = &server.state.player_list {
server.player_lists.get_mut(id).remove(client.uuid()); server.player_lists.get_mut(id).remove(client.uuid());
} }
server.entities.remove(client.state.entity_id); server.entities.remove(client.entity_id);
return false; return false;
} }
let player = server.entities.get_mut(client.state.entity_id).unwrap(); let player = server.entities.get_mut(client.entity_id).unwrap();
while let Some(event) = handle_event_default(client, player) { while let Some(event) = client.next_event() {
event.handle_default(client, player);
match event { match event {
ClientEvent::InteractWithEntity { id, interact, .. } => { ClientEvent::InteractWithEntity {
entity_id,
interact,
..
} => {
if interact == EntityInteraction::Attack if interact == EntityInteraction::Attack
&& Some(id) == server.state.sheep_id && entity_id == server.state.sheep_id.to_raw()
{ {
set_example_pack(client); set_example_pack(client);
} }
} }
ClientEvent::ResourcePackStatusChanged(s) => { ClientEvent::ResourcePackLoaded => {
let message = match s { client.send_message("Resource pack loaded!".color(Color::GREEN));
ResourcePackC2s::SuccessfullyLoaded => {
"The resource pack was successfully loaded!".color(Color::GREEN)
} }
ResourcePackC2s::Declined => { ClientEvent::ResourcePackDeclined => {
"You declined the resource pack :(".color(Color::RED) client.send_message("Resource pack declined.".color(Color::RED));
} }
ResourcePackC2s::FailedDownload => { ClientEvent::ResourcePackFailedDownload => {
"The resource pack download failed.".color(Color::RED) client.send_message("Resource pack download failed.".color(Color::RED));
} }
_ => continue, _ => {}
};
client.send_message(message.italic());
client.send_message(
"Hit the sheep above you to prompt the resource pack again."
.color(Color::GRAY)
.italic(),
);
}
_ => (),
} }
} }

View file

@ -50,6 +50,7 @@ impl Config for Game {
/// If the chunk should stay loaded at the end of the tick. /// If the chunk should stay loaded at the end of the tick.
type ChunkState = bool; type ChunkState = bool;
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -91,14 +92,17 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state = id, Some((id, entity)) => {
entity.set_world(world_id);
client.state = id
}
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
} }
} }
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
client.teleport([0.0, 200.0, 0.0], 0.0, 0.0); client.teleport([0.0, 200.0, 0.0], 0.0, 0.0);
@ -118,18 +122,9 @@ impl Config for Game {
client.send_message("Welcome to the terrain example!".italic()); client.send_message("Welcome to the terrain example!".italic());
} }
if client.is_disconnected() { let player = server.entities.get_mut(client.state).unwrap();
self.player_count.fetch_sub(1, Ordering::SeqCst); while let Some(event) = client.next_event() {
if let Some(id) = &server.state { event.handle_default(client, player);
server.player_lists.get_mut(id).remove(client.uuid());
}
server.entities.remove(client.state);
return false;
}
if let Some(entity) = server.entities.get_mut(client.state) {
while handle_event_default(client, entity).is_some() {}
} }
let dist = client.view_distance(); let dist = client.view_distance();
@ -143,6 +138,16 @@ impl Config for Game {
} }
} }
if client.is_disconnected() {
self.player_count.fetch_sub(1, Ordering::SeqCst);
if let Some(id) = &server.state {
server.player_lists.get_mut(id).remove(client.uuid());
}
server.entities.remove(client.state);
return false;
}
true true
}); });

View file

@ -34,10 +34,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn max_connections(&self) -> usize {
64
}
async fn server_list_ping( async fn server_list_ping(
&self, &self,
@ -68,7 +65,7 @@ impl Config for Game {
.entities .entities
.insert_with_uuid(EntityKind::Player, client.uuid(), ()) .insert_with_uuid(EntityKind::Player, client.uuid(), ())
{ {
Some((id, _)) => client.state.entity_id = id, Some((id, _)) => client.entity_id = id,
None => { None => {
client.disconnect("Conflicting UUID"); client.disconnect("Conflicting UUID");
return false; return false;
@ -78,7 +75,7 @@ impl Config for Game {
let world_id = server.state.world; let world_id = server.state.world;
client.set_flat(true); client.set_flat(true);
client.spawn(world_id); client.respawn(world_id);
client.teleport(SPAWN_POS, -90.0, 0.0); client.teleport(SPAWN_POS, -90.0, 0.0);
client.set_game_mode(GameMode::Creative); client.set_game_mode(GameMode::Creative);
@ -164,18 +161,20 @@ impl Config for Game {
); );
} }
if client.is_disconnected() {
server.entities.remove(client.state.entity_id);
return false;
}
if client.position().y < 0.0 { if client.position().y < 0.0 {
client.teleport(SPAWN_POS, 0.0, 0.0); client.teleport(SPAWN_POS, 0.0, 0.0);
} }
let player = server.entities.get_mut(client.state.entity_id).unwrap(); let player = server.entities.get_mut(client.entity_id).unwrap();
while handle_event_default(client, player).is_some() {} while let Some(event) = client.next_event() {
event.handle_default(client, player);
}
if client.is_disconnected() {
server.entities.remove(client.entity_id);
return false;
}
true true
}); });

View file

@ -35,6 +35,7 @@ impl Config for Game {
type WorldState = (); type WorldState = ();
type ChunkState = (); type ChunkState = ();
type PlayerListState = (); type PlayerListState = ();
type InventoryState = ();
fn max_connections(&self) -> usize { fn max_connections(&self) -> usize {
MAX_PLAYERS + 64 MAX_PLAYERS + 64
@ -115,7 +116,7 @@ impl Config for Game {
server.clients.retain(|_, client| { server.clients.retain(|_, client| {
if client.created_this_tick() { if client.created_this_tick() {
client.spawn(world_id); client.respawn(world_id);
client.set_flat(true); client.set_flat(true);
client.teleport([0.0, 1.0, 0.0], 0.0, 0.0); client.teleport([0.0, 1.0, 0.0], 0.0, 0.0);
@ -157,12 +158,14 @@ impl Config for Game {
} }
if WITH_PLAYER_ENTITIES { if WITH_PLAYER_ENTITIES {
if let Some(entity) = server.entities.get_mut(client.state) { if let Some(player) = server.entities.get_mut(client.state) {
while handle_event_default(client, entity).is_some() {} while let Some(event) = client.next_event() {
event.handle_default(client, player);
}
} }
} else { } else {
while let Some(event) = client.pop_event() { while let Some(event) = client.next_event() {
if let ClientEvent::SettingsChanged { view_distance, .. } = event { if let ClientEvent::UpdateSettings { view_distance, .. } = event {
client.set_view_distance(view_distance); client.set_view_distance(view_distance);
} }
} }

View file

@ -10,6 +10,7 @@ use std::collections::hash_map::Entry;
use std::collections::HashMap; use std::collections::HashMap;
use std::io::Write; use std::io::Write;
use std::iter::FusedIterator; use std::iter::FusedIterator;
use std::ops::{Deref, DerefMut};
use paletted_container::PalettedContainer; use paletted_container::PalettedContainer;
use rayon::iter::{IntoParallelRefIterator, IntoParallelRefMutIterator, ParallelIterator}; use rayon::iter::{IntoParallelRefIterator, IntoParallelRefMutIterator, ParallelIterator};
@ -22,7 +23,7 @@ use valence_protocol::{BlockPos, BlockState, Encode, VarInt, VarLong};
use crate::biome::BiomeId; use crate::biome::BiomeId;
pub use crate::chunk_pos::ChunkPos; pub use crate::chunk_pos::ChunkPos;
use crate::config::Config; use crate::config::Config;
use crate::server::PlayPacketController; use crate::server::PlayPacketSender;
use crate::util::bits_needed; use crate::util::bits_needed;
mod paletted_container; mod paletted_container;
@ -54,7 +55,7 @@ impl<C: Config> Chunks<C> {
/// ///
/// **Note**: For the vanilla Minecraft client to see a chunk, all chunks /// **Note**: For the vanilla Minecraft client to see a chunk, all chunks
/// adjacent to it must also be loaded. Clients should not be spawned within /// adjacent to it must also be loaded. Clients should not be spawned within
/// unloaded chunks via [`spawn`](crate::client::Client::spawn). /// unloaded chunks via [`respawn`](crate::client::Client::respawn).
pub fn insert( pub fn insert(
&mut self, &mut self,
pos: impl Into<ChunkPos>, pos: impl Into<ChunkPos>,
@ -200,38 +201,46 @@ impl<C: Config> Chunks<C> {
} }
} }
/// Sets the block state at an absolute block position in world space. /// Sets the block state at an absolute block position in world space. The
/// previous block state at the position is returned.
/// ///
/// If the position is inside of a chunk, then `true` is returned and the /// If the given position is not inside of a loaded chunk, then a new chunk
/// block is set. Otherwise, `false` is returned and the function has no /// is created at the position before the block is set.
/// effect.
/// ///
/// **Note**: if you need to set a large number of blocks, it may be more /// If the position is completely out of bounds, then no new chunk is
/// efficient write to the chunks directly with /// created and [`BlockState::AIR`] is returned.
/// [`Chunk::set_block_state`]. pub fn set_block_state(&mut self, pos: impl Into<BlockPos>, block: BlockState) -> BlockState
pub fn set_block_state(&mut self, pos: impl Into<BlockPos>, block: BlockState) -> bool { where
let pos = pos.into(); C::ChunkState: Default,
let chunk_pos = ChunkPos::from(pos);
if let Some(chunk) = self.chunks.get_mut(&chunk_pos) {
if let Some(y) = pos
.y
.checked_sub(self.dimension_min_y)
.and_then(|y| y.try_into().ok())
{ {
if y < chunk.height() { let pos = pos.into();
let Some(y) = pos.y.checked_sub(self.dimension_min_y).and_then(|y| y.try_into().ok()) else {
return BlockState::AIR;
};
if y >= self.dimension_height as usize {
return BlockState::AIR;
}
let chunk = match self.chunks.entry(ChunkPos::from(pos)) {
Entry::Occupied(oe) => oe.into_mut(),
Entry::Vacant(ve) => {
let dimension_section_count = (self.dimension_height / 16) as usize;
ve.insert(LoadedChunk::new(
UnloadedChunk::default(),
dimension_section_count,
Default::default(),
))
}
};
chunk.set_block_state( chunk.set_block_state(
pos.x.rem_euclid(16) as usize, pos.x.rem_euclid(16) as usize,
y, y,
pos.z.rem_euclid(16) as usize, pos.z.rem_euclid(16) as usize,
block, block,
); )
return true;
}
}
}
false
} }
pub(crate) fn update(&mut self) { pub(crate) fn update(&mut self) {
@ -472,6 +481,20 @@ pub struct LoadedChunk<C: Config> {
created_this_tick: bool, created_this_tick: bool,
} }
impl<C: Config> Deref for LoadedChunk<C> {
type Target = C::ChunkState;
fn deref(&self) -> &Self::Target {
&self.state
}
}
impl<C: Config> DerefMut for LoadedChunk<C> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.state
}
}
/// A 16x16x16 meter volume of blocks, biomes, and light in a chunk. /// A 16x16x16 meter volume of blocks, biomes, and light in a chunk.
#[derive(Clone)] #[derive(Clone)]
struct ChunkSection { struct ChunkSection {
@ -540,7 +563,7 @@ impl<C: Config> LoadedChunk<C> {
/// Queues the chunk data packet for this chunk with the given position. /// Queues the chunk data packet for this chunk with the given position.
pub(crate) fn chunk_data_packet( pub(crate) fn chunk_data_packet(
&self, &self,
ctrl: &mut PlayPacketController, send: &mut PlayPacketSender,
scratch: &mut Vec<u8>, scratch: &mut Vec<u8>,
pos: ChunkPos, pos: ChunkPos,
biome_registry_len: usize, biome_registry_len: usize,
@ -567,7 +590,7 @@ impl<C: Config> LoadedChunk<C> {
)?; )?;
} }
ctrl.append_packet(&ChunkDataAndUpdateLight { send.append_packet(&ChunkDataAndUpdateLight {
chunk_x: pos.x, chunk_x: pos.x,
chunk_z: pos.z, chunk_z: pos.z,
heightmaps: compound! { heightmaps: compound! {
@ -590,7 +613,7 @@ impl<C: Config> LoadedChunk<C> {
&self, &self,
pos: ChunkPos, pos: ChunkPos,
min_y: i32, min_y: i32,
ctrl: &mut PlayPacketController, send: &mut PlayPacketSender,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
for (sect_y, sect) in self.sections.iter().enumerate() { for (sect_y, sect) in self.sections.iter().enumerate() {
if sect.modified_blocks_count == 1 { if sect.modified_blocks_count == 1 {
@ -611,8 +634,8 @@ impl<C: Config> LoadedChunk<C> {
let global_y = sect_y as i32 * 16 + (idx / (16 * 16)) as i32 + min_y; let global_y = sect_y as i32 * 16 + (idx / (16 * 16)) as i32 + min_y;
let global_z = pos.z * 16 + (idx / 16 % 16) as i32; let global_z = pos.z * 16 + (idx / 16 % 16) as i32;
ctrl.append_packet(&BlockUpdate { send.append_packet(&BlockUpdate {
location: BlockPos::new(global_x, global_y, global_z), position: BlockPos::new(global_x, global_y, global_z),
block_id: VarInt(block.to_raw() as _), block_id: VarInt(block.to_raw() as _),
})?; })?;
} else if sect.modified_blocks_count > 1 { } else if sect.modified_blocks_count > 1 {
@ -637,7 +660,7 @@ impl<C: Config> LoadedChunk<C> {
| (pos.z as i64 & 0x3fffff) << 20 | (pos.z as i64 & 0x3fffff) << 20
| (sect_y as i64 + min_y.div_euclid(16) as i64) & 0xfffff; | (sect_y as i64 + min_y.div_euclid(16) as i64) & 0xfffff;
ctrl.append_packet(&UpdateSectionBlocks { send.append_packet(&UpdateSectionBlocks {
chunk_section_position, chunk_section_position,
invert_trust_edges: false, invert_trust_edges: false,
blocks, blocks,

File diff suppressed because it is too large Load diff

View file

@ -1,47 +1,54 @@
use std::time::Duration; use std::cmp;
use anyhow::bail;
use uuid::Uuid;
use valence_protocol::entity_meta::Pose; use valence_protocol::entity_meta::Pose;
use valence_protocol::packets::c2s::play::ResourcePackC2s; use valence_protocol::packets::c2s::play::{
use valence_protocol::types::{ ClientCommand, PlayerAbilitiesC2s, ResourcePackC2s, SeenAdvancements,
ChatMode, ClickContainerMode, DisplayedSkinParts, EntityInteraction, Hand, MainHand,
}; };
use valence_protocol::{BlockFace, BlockPos, Ident, ItemStack, VarInt}; use valence_protocol::packets::C2sPlayPacket;
use vek::Vec3; use valence_protocol::types::{
Action, ChatMode, ClickContainerMode, CommandBlockMode, Difficulty, DiggingStatus,
DisplayedSkinParts, EntityInteraction, Hand, MainHand, RecipeBookId, StructureBlockAction,
StructureBlockFlags, StructureBlockMirror, StructureBlockMode, StructureBlockRotation,
};
use valence_protocol::{BlockFace, BlockPos, Ident, ItemStack, VarLong};
use super::Client; use crate::client::Client;
use crate::config::Config; use crate::config::Config;
use crate::entity::{Entity, EntityEvent, EntityId, TrackedData}; use crate::entity::{Entity, EntityEvent, TrackedData};
use crate::inventory::{Inventory, InventoryDirtyable, SlotId};
/// Represents an action performed by a client. /// A discrete action performed by a client.
/// ///
/// Client events can be obtained from /// Client events are a more convenient representation of the data contained in
/// [`pop_event`](super::Client::pop_event). /// a [`C2sPlayPacket`].
/// ///
/// # Event Validation /// [`C2sPlayPacket`]: crate::protocol::packets::C2sPlayPacket
/// #[derive(Clone, Debug)]
/// [`Client`](super::Client) makes no attempt to validate events against the
/// expected rules for players. Malicious clients can teleport through walls,
/// interact with distant entities, sneak and sprint backwards, break
/// bedrock in survival mode, etc.
///
/// It is best to think of events from clients as _requests_ to interact with
/// the server. It is then your responsibility to decide if the request should
/// be honored.
#[derive(Debug)]
pub enum ClientEvent { pub enum ClientEvent {
/// A regular message was sent to the chat. QueryBlockEntity {
ChatMessage { position: BlockPos,
/// The content of the message transaction_id: i32,
message: String,
/// The time the message was sent.
timestamp: Duration,
}, },
/// Settings were changed. This is always sent once after joining by the ChangeDifficulty(Difficulty),
/// vanilla client. MessageAcknowledgment {
SettingsChanged { last_seen: Vec<(Uuid, Box<[u8]>)>,
last_received: Option<(Uuid, Box<[u8]>)>,
},
ChatCommand {
command: Box<str>,
timestamp: u64,
},
ChatMessage {
message: Box<str>,
timestamp: u64,
},
ChatPreview,
PerformRespawn,
RequestStats,
UpdateSettings {
/// e.g. en_US /// e.g. en_US
locale: String, locale: Box<str>,
/// The client side render distance, in chunks. /// The client side render distance, in chunks.
/// ///
/// The value is always in `2..=32`. /// The value is always in `2..=32`.
@ -49,184 +56,629 @@ pub enum ClientEvent {
chat_mode: ChatMode, chat_mode: ChatMode,
/// `true` if the client has chat colors enabled, `false` otherwise. /// `true` if the client has chat colors enabled, `false` otherwise.
chat_colors: bool, chat_colors: bool,
main_hand: MainHand,
displayed_skin_parts: DisplayedSkinParts, displayed_skin_parts: DisplayedSkinParts,
main_hand: MainHand,
enable_text_filtering: bool,
allow_server_listings: bool, allow_server_listings: bool,
}, },
MovePosition { CommandSuggestionsRequest {
position: Vec3<f64>, transaction_id: i32,
on_ground: bool, text: Box<str>,
}, },
MovePositionAndRotation { ClickContainerButton {
position: Vec3<f64>, window_id: i8,
yaw: f32, button_id: i8,
pitch: f32,
on_ground: bool,
}, },
MoveRotation { ClickContainer {
yaw: f32, window_id: u8,
pitch: f32, state_id: i32,
on_ground: bool, slot_id: i16,
button: i8,
mode: ClickContainerMode,
slot_changes: Vec<(i16, Option<ItemStack>)>,
carried_item: Option<ItemStack>,
}, },
MoveOnGround { CloseContainer {
on_ground: bool, window_id: i8,
}, },
MoveVehicle { PluginMessage {
position: Vec3<f64>, channel: Ident<Box<str>>,
yaw: f32, data: Box<[u8]>,
pitch: f32,
}, },
StartSneaking, EditBook {
StopSneaking, slot: i32,
StartSprinting, entries: Vec<Box<str>>,
StopSprinting, title: Option<Box<str>>,
/// A jump while on a horse started. },
StartJumpWithHorse { QueryEntity {
/// The power of the horse jump. transaction_id: i32,
jump_boost: u8, entity_id: i32,
}, },
/// A jump while on a horse stopped.
StopJumpWithHorse,
/// The client left a bed.
LeaveBed,
/// The inventory was opened while on a horse.
OpenHorseInventory,
StartFlyingWithElytra,
ArmSwing(Hand),
/// Left or right click interaction with an entity's hitbox. /// Left or right click interaction with an entity's hitbox.
InteractWithEntity { InteractWithEntity {
/// The ID of the entity being interacted with. /// The raw ID of the entity being interacted with.
id: EntityId, entity_id: i32,
/// If the client was sneaking during the interaction. /// If the client was sneaking during the interaction.
sneaking: bool, sneaking: bool,
/// The kind of interaction that occurred. /// The kind of interaction that occurred.
interact: EntityInteraction, interact: EntityInteraction,
}, },
SteerBoat { JigsawGenerate {
position: BlockPos,
levels: i32,
keep_jigsaws: bool,
},
LockDifficulty(bool),
// TODO: combine movement events?
SetPlayerPosition {
position: [f64; 3],
on_ground: bool,
},
SetPlayerPositionAndRotation {
position: [f64; 3],
yaw: f32,
pitch: f32,
on_ground: bool,
},
SetPlayerRotation {
yaw: f32,
pitch: f32,
on_ground: bool,
},
SetPlayerOnGround(bool),
MoveVehicle {
position: [f64; 3],
yaw: f32,
pitch: f32,
},
StartSneaking,
StopSneaking,
LeaveBed,
StartSprinting,
StopSprinting,
StartJumpWithHorse {
/// The power of the horse jump in `0..=100`.
jump_boost: u8,
},
/// A jump while on a horse stopped.
StopJumpWithHorse,
/// The inventory was opened while on a horse.
OpenHorseInventory,
StartFlyingWithElytra,
PaddleBoat {
left_paddle_turning: bool, left_paddle_turning: bool,
right_paddle_turning: bool, right_paddle_turning: bool,
}, },
Digging { PickItem {
/// The kind of digging event this is. slot_to_use: i32,
status: DiggingStatus,
/// The position of the block being broken.
position: BlockPos,
/// The face of the block being broken.
face: BlockFace,
}, },
InteractWithBlock { PlaceRecipe {
window_id: i8,
recipe: Ident<Box<str>>,
make_all: bool,
},
StopFlying,
StartFlying,
StartDigging {
position: BlockPos,
face: BlockFace,
sequence: i32,
},
CancelDigging {
position: BlockPos,
face: BlockFace,
sequence: i32,
},
FinishDigging {
position: BlockPos,
face: BlockFace,
sequence: i32,
},
DropItem,
DropItemStack,
/// Eating food, pulling back bows, using buckets, etc.
UpdateHeldItemState,
SwapItemInHand,
PlayerInput {
sideways: f32,
forward: f32,
jump: bool,
unmount: bool,
},
Pong {
id: i32,
},
ChangeRecipeBookSettings {
book_id: RecipeBookId,
book_open: bool,
filter_active: bool,
},
SetSeenRecipe {
recipe_id: Ident<Box<str>>,
},
RenameItem {
name: Box<str>,
},
ResourcePackLoaded,
ResourcePackDeclined,
ResourcePackFailedDownload,
ResourcePackAccepted,
OpenAdvancementTab {
tab_id: Ident<Box<str>>,
},
CloseAdvancementScreen,
SelectTrade {
slot: i32,
},
SetBeaconEffect {
primary_effect: Option<i32>,
secondary_effect: Option<i32>,
},
SetHeldItem {
slot: i16,
},
ProgramCommandBlock {
position: BlockPos,
command: Box<str>,
mode: CommandBlockMode,
track_output: bool,
conditional: bool,
automatic: bool,
},
ProgramCommandBlockMinecart {
entity_id: i32,
command: Box<str>,
track_output: bool,
},
SetCreativeModeSlot {
slot: i16,
clicked_item: Option<ItemStack>,
},
ProgramJigsawBlock {
position: BlockPos,
name: Ident<Box<str>>,
target: Ident<Box<str>>,
pool: Ident<Box<str>>,
final_state: Box<str>,
joint_type: Box<str>,
},
ProgramStructureBlock {
position: BlockPos,
action: StructureBlockAction,
mode: StructureBlockMode,
name: Box<str>,
offset_xyz: [i8; 3],
size_xyz: [i8; 3],
mirror: StructureBlockMirror,
rotation: StructureBlockRotation,
metadata: Box<str>,
integrity: f32,
seed: VarLong,
flags: StructureBlockFlags,
},
UpdateSign {
position: BlockPos,
lines: [Box<str>; 4],
},
SwingArm(Hand),
TeleportToEntity {
target: Uuid,
},
UseItemOnBlock {
/// The hand that was used /// The hand that was used
hand: Hand, hand: Hand,
/// The location of the block that was interacted with /// The location of the block that was interacted with
location: BlockPos, position: BlockPos,
/// The face of the block that was clicked /// The face of the block that was clicked
face: BlockFace, face: BlockFace,
/// The pos inside of the block that was clicked on /// The position inside of the block that was clicked on
cursor_pos: Vec3<f32>, cursor_pos: [f32; 3],
/// Whether or not the player's head is inside a block /// Whether or not the player's head is inside a block
head_inside_block: bool, head_inside_block: bool,
/// Sequence number /// Sequence number for synchronization
sequence: VarInt, sequence: i32,
}, },
PluginMessageReceived { UseItem {
channel: Ident<String>, hand: Hand,
data: Vec<u8>, sequence: i32,
}, },
ResourcePackStatusChanged(ResourcePackC2s),
/// The client closed a screen. This occurs when the client closes their
/// inventory, closes a chest inventory, etc.
CloseScreen {
window_id: u8,
},
/// The client is attempting to drop 1 of the currently held item.
DropItem,
/// The client is attempting to drop a stack of items.
///
/// If the client is in creative mode, the items come from the void, so it
/// is safe to trust the contents of this event. Otherwise, you may need to
/// do some validation to make sure items are actually coming from the
/// user's inventory.
DropItemStack {
// TODO: maybe we could add `from_slot_id` to make validation easier
stack: ItemStack,
},
/// The client is in creative mode, and is trying to set it's inventory slot
/// to a value.
SetSlotCreative {
/// The slot number that the client is trying to set.
slot_id: SlotId,
/// The contents of the slot.
slot: Option<ItemStack>,
},
/// The client is in survival mode, and is trying to modify an inventory.
ClickContainer {
window_id: u8,
state_id: VarInt,
/// The slot that was clicked
slot_id: SlotId,
/// The type of click that the user performed
mode: ClickContainerMode,
/// A list of slot ids and what their contents should be set to.
///
/// It's not safe to blindly trust the contents of this. Servers need to
/// validate it if they want to prevent item duping.
slot_changes: Vec<(SlotId, Option<ItemStack>)>,
/// The item that is now being carried by the user's cursor
carried_item: Option<ItemStack>,
},
RespawnRequest,
} }
#[derive(Clone, PartialEq, Debug)] pub(super) fn next_event_fallible<C: Config>(
pub struct Settings { client: &mut Client<C>,
/// e.g. en_US ) -> anyhow::Result<Option<ClientEvent>> {
pub locale: String, loop {
/// The client side render distance, in chunks. let Some(pkt) = client.recv.try_next_packet::<C2sPlayPacket>()? else {
return Ok(None)
};
return Ok(Some(match pkt {
C2sPlayPacket::ConfirmTeleport(p) => {
if client.pending_teleports == 0 {
bail!("unexpected teleport confirmation");
}
let got = p.teleport_id.0 as u32;
let expected = client
.teleport_id_counter
.wrapping_sub(client.pending_teleports);
if got == expected {
client.pending_teleports -= 1;
} else {
bail!("unexpected teleport ID (expected {expected}, got {got}");
}
continue;
}
C2sPlayPacket::QueryBlockEntityTag(p) => ClientEvent::QueryBlockEntity {
position: p.position,
transaction_id: p.transaction_id.0,
},
C2sPlayPacket::ChangeDifficulty(p) => ClientEvent::ChangeDifficulty(p.0),
C2sPlayPacket::MessageAcknowledgmentC2s(p) => ClientEvent::MessageAcknowledgment {
last_seen: p
.0
.last_seen
.into_iter()
.map(|entry| (entry.profile_id, entry.signature.into()))
.collect(),
last_received: p
.0
.last_received
.map(|entry| (entry.profile_id, entry.signature.into())),
},
C2sPlayPacket::ChatCommand(p) => ClientEvent::ChatCommand {
command: p.command.into(),
timestamp: p.timestamp,
},
C2sPlayPacket::ChatMessage(p) => ClientEvent::ChatMessage {
message: p.message.into(),
timestamp: p.timestamp,
},
C2sPlayPacket::ChatPreviewC2s(_) => ClientEvent::ChatPreview,
C2sPlayPacket::ClientCommand(p) => match p {
ClientCommand::PerformRespawn => ClientEvent::PerformRespawn,
ClientCommand::RequestStats => ClientEvent::RequestStats,
},
C2sPlayPacket::ClientInformation(p) => ClientEvent::UpdateSettings {
locale: p.locale.into(),
view_distance: p.view_distance,
chat_mode: p.chat_mode,
chat_colors: p.chat_colors,
displayed_skin_parts: p.displayed_skin_parts,
main_hand: p.main_hand,
enable_text_filtering: p.enable_text_filtering,
allow_server_listings: p.allow_server_listings,
},
C2sPlayPacket::CommandSuggestionsRequest(p) => ClientEvent::CommandSuggestionsRequest {
transaction_id: p.transaction_id.0,
text: p.text.into(),
},
C2sPlayPacket::ClickContainerButton(p) => ClientEvent::ClickContainerButton {
window_id: p.window_id,
button_id: p.button_id,
},
C2sPlayPacket::ClickContainer(p) => {
// TODO: check that the slot modifications are legal.
// TODO: update cursor item.
for (idx, item) in &p.slots {
// TODO: check bounds on indices.
client.slots[*idx as usize] = item.clone();
}
ClientEvent::ClickContainer {
window_id: p.window_id,
state_id: p.state_id.0,
slot_id: p.slot_idx,
button: p.button,
mode: p.mode,
slot_changes: p.slots,
carried_item: p.carried_item,
}
}
C2sPlayPacket::CloseContainerC2s(p) => ClientEvent::CloseContainer {
window_id: p.window_id,
},
C2sPlayPacket::PluginMessageC2s(p) => ClientEvent::PluginMessage {
channel: p.channel.into(),
data: p.data.0.into(),
},
C2sPlayPacket::EditBook(p) => ClientEvent::EditBook {
slot: p.slot.0,
entries: p.entries.into_iter().map(From::from).collect(),
title: p.title.map(From::from),
},
C2sPlayPacket::QueryEntityTag(p) => ClientEvent::QueryEntity {
transaction_id: p.transaction_id.0,
entity_id: p.entity_id.0,
},
C2sPlayPacket::Interact(p) => ClientEvent::InteractWithEntity {
entity_id: p.entity_id.0,
sneaking: p.sneaking,
interact: p.interact,
},
C2sPlayPacket::JigsawGenerate(p) => ClientEvent::JigsawGenerate {
position: p.position,
levels: p.levels.0,
keep_jigsaws: p.keep_jigsaws,
},
C2sPlayPacket::KeepAliveC2s(p) => {
if client.bits.got_keepalive() {
bail!("unexpected keepalive");
} else if p.id != client.last_keepalive_id {
bail!(
"keepalive IDs don't match (expected {}, got {})",
client.last_keepalive_id,
p.id
);
} else {
client.bits.set_got_keepalive(true);
}
continue;
}
C2sPlayPacket::LockDifficulty(p) => ClientEvent::LockDifficulty(p.0),
C2sPlayPacket::SetPlayerPosition(p) => {
if client.pending_teleports != 0 {
continue;
}
client.position = p.position.into();
ClientEvent::SetPlayerPosition {
position: p.position,
on_ground: p.on_ground,
}
}
C2sPlayPacket::SetPlayerPositionAndRotation(p) => {
if client.pending_teleports != 0 {
continue;
}
client.position = p.position.into();
client.yaw = p.yaw;
client.pitch = p.pitch;
ClientEvent::SetPlayerPositionAndRotation {
position: p.position,
yaw: p.yaw,
pitch: p.pitch,
on_ground: p.on_ground,
}
}
C2sPlayPacket::SetPlayerRotation(p) => {
if client.pending_teleports != 0 {
continue;
}
client.yaw = p.yaw;
client.pitch = p.pitch;
ClientEvent::SetPlayerRotation {
yaw: p.yaw,
pitch: p.pitch,
on_ground: false,
}
}
C2sPlayPacket::SetPlayerOnGround(p) => {
if client.pending_teleports != 0 {
continue;
}
ClientEvent::SetPlayerOnGround(p.0)
}
C2sPlayPacket::MoveVehicleC2s(p) => {
if client.pending_teleports != 0 {
continue;
}
client.position = p.position.into();
client.yaw = p.yaw;
client.pitch = p.pitch;
ClientEvent::MoveVehicle {
position: p.position,
yaw: p.yaw,
pitch: p.pitch,
}
}
C2sPlayPacket::PlayerCommand(p) => match p.action_id {
Action::StartSneaking => ClientEvent::StartSneaking,
Action::StopSneaking => ClientEvent::StopSneaking,
Action::LeaveBed => ClientEvent::LeaveBed,
Action::StartSprinting => ClientEvent::StartSprinting,
Action::StopSprinting => ClientEvent::StopSprinting,
Action::StartJumpWithHorse => ClientEvent::StartJumpWithHorse {
jump_boost: p.jump_boost.0.clamp(0, 100) as u8,
},
Action::StopJumpWithHorse => ClientEvent::StopJumpWithHorse,
Action::OpenHorseInventory => ClientEvent::OpenHorseInventory,
Action::StartFlyingWithElytra => ClientEvent::StartFlyingWithElytra,
},
C2sPlayPacket::PaddleBoat(p) => ClientEvent::PaddleBoat {
left_paddle_turning: p.left_paddle_turning,
right_paddle_turning: p.right_paddle_turning,
},
C2sPlayPacket::PickItem(p) => ClientEvent::PickItem {
slot_to_use: p.slot_to_use.0,
},
C2sPlayPacket::PlaceRecipe(p) => ClientEvent::PlaceRecipe {
window_id: p.window_id,
recipe: p.recipe.into(),
make_all: p.make_all,
},
C2sPlayPacket::PlayerAbilitiesC2s(p) => match p {
PlayerAbilitiesC2s::StopFlying => ClientEvent::StopFlying,
PlayerAbilitiesC2s::StartFlying => ClientEvent::StartFlying,
},
C2sPlayPacket::PlayerAction(p) => {
if p.sequence.0 != 0 {
client.block_change_sequence =
cmp::max(p.sequence.0, client.block_change_sequence);
}
match p.status {
DiggingStatus::StartedDigging => ClientEvent::StartDigging {
position: p.position,
face: p.face,
sequence: p.sequence.0,
},
DiggingStatus::CancelledDigging => ClientEvent::CancelDigging {
position: p.position,
face: p.face,
sequence: p.sequence.0,
},
DiggingStatus::FinishedDigging => ClientEvent::FinishDigging {
position: p.position,
face: p.face,
sequence: p.sequence.0,
},
DiggingStatus::DropItemStack => ClientEvent::DropItemStack,
DiggingStatus::DropItem => ClientEvent::DropItem,
DiggingStatus::UpdateHeldItemState => ClientEvent::UpdateHeldItemState,
DiggingStatus::SwapItemInHand => ClientEvent::SwapItemInHand,
}
}
C2sPlayPacket::PlayerInput(p) => ClientEvent::PlayerInput {
sideways: p.sideways,
forward: p.forward,
jump: p.flags.jump(),
unmount: p.flags.unmount(),
},
C2sPlayPacket::PongPlay(p) => ClientEvent::Pong { id: p.id },
C2sPlayPacket::ChangeRecipeBookSettings(p) => ClientEvent::ChangeRecipeBookSettings {
book_id: p.book_id,
book_open: p.book_open,
filter_active: p.filter_active,
},
C2sPlayPacket::SetSeenRecipe(p) => ClientEvent::SetSeenRecipe {
recipe_id: p.recipe_id.into(),
},
C2sPlayPacket::RenameItem(p) => ClientEvent::RenameItem {
name: p.item_name.into(),
},
C2sPlayPacket::ResourcePackC2s(p) => match p {
ResourcePackC2s::SuccessfullyLoaded => ClientEvent::ResourcePackLoaded,
ResourcePackC2s::Declined => ClientEvent::ResourcePackDeclined,
ResourcePackC2s::FailedDownload => ClientEvent::ResourcePackFailedDownload,
ResourcePackC2s::Accepted => ClientEvent::ResourcePackAccepted,
},
C2sPlayPacket::SeenAdvancements(p) => match p {
SeenAdvancements::OpenedTab { tab_id } => ClientEvent::OpenAdvancementTab {
tab_id: tab_id.into(),
},
SeenAdvancements::ClosedScreen => ClientEvent::CloseAdvancementScreen,
},
C2sPlayPacket::SelectTrade(p) => ClientEvent::SelectTrade {
slot: p.selected_slot.0,
},
C2sPlayPacket::SetBeaconEffect(p) => ClientEvent::SetBeaconEffect {
primary_effect: p.primary_effect.map(|i| i.0),
secondary_effect: p.secondary_effect.map(|i| i.0),
},
C2sPlayPacket::SetHeldItemC2s(p) => ClientEvent::SetHeldItem { slot: p.slot },
C2sPlayPacket::ProgramCommandBlock(p) => ClientEvent::ProgramCommandBlock {
position: p.position,
command: p.command.into(),
mode: p.mode,
track_output: p.flags.track_output(),
conditional: p.flags.conditional(),
automatic: p.flags.automatic(),
},
C2sPlayPacket::ProgramCommandBlockMinecart(p) => {
ClientEvent::ProgramCommandBlockMinecart {
entity_id: p.entity_id.0,
command: p.command.into(),
track_output: p.track_output,
}
}
C2sPlayPacket::SetCreativeModeSlot(p) => ClientEvent::SetCreativeModeSlot {
slot: p.slot,
clicked_item: p.clicked_item,
},
C2sPlayPacket::ProgramJigsawBlock(p) => ClientEvent::ProgramJigsawBlock {
position: p.position,
name: p.name.into(),
target: p.target.into(),
pool: p.pool.into(),
final_state: p.final_state.into(),
joint_type: p.joint_type.into(),
},
C2sPlayPacket::ProgramStructureBlock(p) => ClientEvent::ProgramStructureBlock {
position: p.position,
action: p.action,
mode: p.mode,
name: p.name.into(),
offset_xyz: p.offset_xyz,
size_xyz: p.size_xyz,
mirror: p.mirror,
rotation: p.rotation,
metadata: p.metadata.into(),
integrity: p.integrity,
seed: p.seed,
flags: p.flags,
},
C2sPlayPacket::UpdateSign(p) => ClientEvent::UpdateSign {
position: p.position,
lines: p.lines.map(From::from),
},
C2sPlayPacket::SwingArm(p) => ClientEvent::SwingArm(p.0),
C2sPlayPacket::TeleportToEntity(p) => {
ClientEvent::TeleportToEntity { target: p.target }
}
C2sPlayPacket::UseItemOn(p) => {
if p.sequence.0 != 0 {
client.block_change_sequence =
cmp::max(p.sequence.0, client.block_change_sequence);
}
ClientEvent::UseItemOnBlock {
hand: p.hand,
position: p.position,
face: p.face,
cursor_pos: p.cursor_pos,
head_inside_block: p.head_inside_block,
sequence: p.sequence.0,
}
}
C2sPlayPacket::UseItem(p) => {
if p.sequence.0 != 0 {
client.block_change_sequence =
cmp::max(p.sequence.0, client.block_change_sequence);
}
ClientEvent::UseItem {
hand: p.hand,
sequence: p.sequence.0,
}
}
}));
}
}
impl ClientEvent {
/// Takes a client event, a client, and an entity representing the client
/// and expresses the event in a reasonable way.
/// ///
/// The value is always in `2..=32`. /// For instance, movement events are expressed by changing the entity's
pub view_distance: u8, /// position/rotation to match the received movement, crouching makes the
pub chat_mode: ChatMode, /// entity crouch, etc.
/// `true` if the client has chat colors enabled, `false` otherwise.
pub chat_colors: bool,
pub main_hand: MainHand,
pub displayed_skin_parts: DisplayedSkinParts,
pub allow_server_listings: bool,
}
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
pub enum DiggingStatus {
/// The client started digging a block.
Start,
/// The client stopped digging a block before it was fully broken.
Cancel,
/// The client finished digging a block successfully.
Finish,
}
/// Pops one event from the event queue of `client` and expresses the event in a
/// reasonable way using `entity`. For instance, movement events are expressed
/// by changing the entity's position to match the received position. Rotation
/// events rotate the entity. etc.
/// ///
/// This function's primary purpose is to reduce boilerplate code in the /// This function's primary purpose is to reduce boilerplate code in the
/// examples, but it can be used as a quick way to get started in your own code. /// examples, but it can be used as a quick way to get started in your own
/// The precise behavior of this function is left unspecified and is subject to /// code. The precise behavior of this function is left unspecified and
/// change. /// is subject to change.
/// pub fn handle_default<C: Config>(&self, client: &mut Client<C>, entity: &mut Entity<C>) {
/// The popped event is returned unmodified. `None` is returned if there are no match self {
/// more events in `client`. ClientEvent::RequestStats => {
pub fn handle_event_default<C: Config>( // TODO: award empty statistics
client: &mut Client<C>, }
entity: &mut Entity<C>, ClientEvent::UpdateSettings {
) -> Option<ClientEvent> {
let event = client.pop_event()?;
match &event {
ClientEvent::ChatMessage { .. } => {}
ClientEvent::SettingsChanged {
view_distance, view_distance,
main_hand,
displayed_skin_parts, displayed_skin_parts,
main_hand,
.. ..
} => { } => {
client.set_view_distance(*view_distance); client.set_view_distance(*view_distance);
@ -253,14 +705,15 @@ pub fn handle_event_default<C: Config>(
player.set_main_arm(*main_hand as u8); player.set_main_arm(*main_hand as u8);
} }
} }
ClientEvent::MovePosition { ClientEvent::CommandSuggestionsRequest { .. } => {}
ClientEvent::SetPlayerPosition {
position, position,
on_ground, on_ground,
} => { } => {
entity.set_position(*position); entity.set_position(*position);
entity.set_on_ground(*on_ground); entity.set_on_ground(*on_ground);
} }
ClientEvent::MovePositionAndRotation { ClientEvent::SetPlayerPositionAndRotation {
position, position,
yaw, yaw,
pitch, pitch,
@ -272,7 +725,7 @@ pub fn handle_event_default<C: Config>(
entity.set_pitch(*pitch); entity.set_pitch(*pitch);
entity.set_on_ground(*on_ground); entity.set_on_ground(*on_ground);
} }
ClientEvent::MoveRotation { ClientEvent::SetPlayerRotation {
yaw, yaw,
pitch, pitch,
on_ground, on_ground,
@ -282,10 +735,16 @@ pub fn handle_event_default<C: Config>(
entity.set_pitch(*pitch); entity.set_pitch(*pitch);
entity.set_on_ground(*on_ground); entity.set_on_ground(*on_ground);
} }
ClientEvent::MoveOnGround { on_ground } => { ClientEvent::SetPlayerOnGround(on_ground) => entity.set_on_ground(*on_ground),
entity.set_on_ground(*on_ground); ClientEvent::MoveVehicle {
position,
yaw,
pitch,
} => {
entity.set_position(*position);
entity.set_yaw(*yaw);
entity.set_pitch(*pitch);
} }
ClientEvent::MoveVehicle { .. } => {}
ClientEvent::StartSneaking => { ClientEvent::StartSneaking => {
if let TrackedData::Player(player) = entity.data_mut() { if let TrackedData::Player(player) = entity.data_mut() {
if player.get_pose() == Pose::Standing { if player.get_pose() == Pose::Standing {
@ -310,44 +769,13 @@ pub fn handle_event_default<C: Config>(
player.set_sprinting(false); player.set_sprinting(false);
} }
} }
ClientEvent::StartJumpWithHorse { .. } => {} ClientEvent::SwingArm(hand) => {
ClientEvent::StopJumpWithHorse => {}
ClientEvent::LeaveBed => {}
ClientEvent::OpenHorseInventory => {}
ClientEvent::StartFlyingWithElytra => {}
ClientEvent::ArmSwing(hand) => {
entity.push_event(match hand { entity.push_event(match hand {
Hand::Main => EntityEvent::SwingMainHand, Hand::Main => EntityEvent::SwingMainHand,
Hand::Off => EntityEvent::SwingOffHand, Hand::Off => EntityEvent::SwingOffHand,
}); });
} }
ClientEvent::InteractWithEntity { .. } => {} _ => {}
ClientEvent::SteerBoat { .. } => {}
ClientEvent::Digging { .. } => {}
ClientEvent::InteractWithBlock { .. } => {}
ClientEvent::PluginMessageReceived { .. } => {}
ClientEvent::ResourcePackStatusChanged(_) => {}
ClientEvent::CloseScreen { window_id } => {
if let Some(window) = &client.open_inventory {
if window.window_id == *window_id {
client.open_inventory = None;
} }
} }
} }
ClientEvent::DropItem => {}
ClientEvent::DropItemStack { .. } => {}
ClientEvent::SetSlotCreative { slot_id, slot } => {
let previous_dirty = client.inventory.is_dirty();
client.inventory.set_slot(*slot_id, slot.clone());
// HACK: we don't need to mark the inventory as dirty because the
// client already knows what the updated state of the inventory is.
client.inventory.mark_dirty(previous_dirty);
}
ClientEvent::ClickContainer { .. } => {}
ClientEvent::RespawnRequest => {}
}
entity.set_world(client.world());
Some(event)
}

View file

@ -40,6 +40,9 @@ pub trait Config: Sized + Send + Sync + 'static {
/// Custom state to store with every /// Custom state to store with every
/// [`PlayerList`](crate::player_list::PlayerList). /// [`PlayerList`](crate::player_list::PlayerList).
type PlayerListState: Send + Sync; type PlayerListState: Send + Sync;
/// Custom state to store with every
/// [`Inventory`](crate::inventory::Inventory).
type InventoryState: Send + Sync;
/// Called once at startup to get the maximum number of simultaneous /// Called once at startup to get the maximum number of simultaneous
/// connections allowed to the server. This includes all /// connections allowed to the server. This includes all
@ -305,7 +308,6 @@ pub trait Config: Sized + Send + Sync + 'static {
} }
/// The result of the [`server_list_ping`](Config::server_list_ping) callback. /// The result of the [`server_list_ping`](Config::server_list_ping) callback.
#[allow(clippy::large_enum_variant)]
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub enum ServerListPing<'a> { pub enum ServerListPing<'a> {
/// Responds to the server list ping with the given information. /// Responds to the server list ping with the given information.
@ -331,6 +333,18 @@ pub enum ServerListPing<'a> {
Ignore, Ignore,
} }
/// Represents an individual entry in the player sample.
#[derive(Clone, Debug, Serialize)]
pub struct PlayerSampleEntry<'a> {
/// The name of the player.
///
/// This string can contain
/// [legacy formatting codes](https://minecraft.fandom.com/wiki/Formatting_codes).
pub name: Cow<'a, str>,
/// The player UUID.
pub id: Uuid,
}
/// Describes how new connections to the server are handled. /// Describes how new connections to the server are handled.
#[non_exhaustive] #[non_exhaustive]
#[derive(Clone, PartialEq, Default)] #[derive(Clone, PartialEq, Default)]
@ -386,26 +400,14 @@ pub enum ConnectionMode {
}, },
} }
/// Represents an individual entry in the player sample.
#[derive(Clone, Debug, Serialize)]
pub struct PlayerSampleEntry<'a> {
/// The name of the player.
///
/// This string can contain
/// [legacy formatting codes](https://minecraft.fandom.com/wiki/Formatting_codes).
pub name: Cow<'a, str>,
/// The player UUID.
pub id: Uuid,
}
/// A minimal `Config` implementation for testing purposes. /// A minimal `Config` implementation for testing purposes.
#[cfg(test)] #[cfg(test)]
pub(crate) struct MockConfig<S = (), Cl = (), E = (), W = (), Ch = (), P = ()> { pub(crate) struct MockConfig<S = (), Cl = (), E = (), W = (), Ch = (), P = (), I = ()> {
_marker: std::marker::PhantomData<(S, Cl, E, W, Ch, P)>, _marker: std::marker::PhantomData<(S, Cl, E, W, Ch, P, I)>,
} }
#[cfg(test)] #[cfg(test)]
impl<S, Cl, E, W, Ch, P> Config for MockConfig<S, Cl, E, W, Ch, P> impl<S, Cl, E, W, Ch, P, I> Config for MockConfig<S, Cl, E, W, Ch, P, I>
where where
S: Send + Sync + 'static, S: Send + Sync + 'static,
Cl: Default + Send + Sync + 'static, Cl: Default + Send + Sync + 'static,
@ -413,6 +415,7 @@ where
W: Send + Sync + 'static, W: Send + Sync + 'static,
Ch: Send + Sync + 'static, Ch: Send + Sync + 'static,
P: Send + Sync + 'static, P: Send + Sync + 'static,
I: Send + Sync + 'static,
{ {
type ServerState = S; type ServerState = S;
type ClientState = Cl; type ClientState = Cl;
@ -420,4 +423,5 @@ where
type WorldState = W; type WorldState = W;
type ChunkState = Ch; type ChunkState = Ch;
type PlayerListState = P; type PlayerListState = P;
type InventoryState = I;
} }

View file

@ -4,6 +4,7 @@ use std::collections::hash_map::Entry;
use std::collections::HashMap; use std::collections::HashMap;
use std::iter::FusedIterator; use std::iter::FusedIterator;
use std::num::NonZeroU32; use std::num::NonZeroU32;
use std::ops::{Deref, DerefMut};
use bitfield_struct::bitfield; use bitfield_struct::bitfield;
pub use data::{EntityKind, TrackedData}; pub use data::{EntityKind, TrackedData};
@ -17,7 +18,7 @@ use valence_protocol::{ByteAngle, RawBytes, VarInt};
use vek::{Aabb, Vec3}; use vek::{Aabb, Vec3};
use crate::config::Config; use crate::config::Config;
use crate::server::PlayPacketController; use crate::server::PlayPacketSender;
use crate::slab_versioned::{Key, VersionedSlab}; use crate::slab_versioned::{Key, VersionedSlab};
use crate::util::aabb_from_bottom_and_size; use crate::util::aabb_from_bottom_and_size;
use crate::world::WorldId; use crate::world::WorldId;
@ -40,7 +41,7 @@ include!(concat!(env!("OUT_DIR"), "/entity_event.rs"));
pub struct Entities<C: Config> { pub struct Entities<C: Config> {
slab: VersionedSlab<Entity<C>>, slab: VersionedSlab<Entity<C>>,
uuid_to_entity: HashMap<Uuid, EntityId>, uuid_to_entity: HashMap<Uuid, EntityId>,
network_id_to_entity: HashMap<NonZeroU32, u32>, raw_id_to_entity: HashMap<NonZeroU32, u32>,
} }
impl<C: Config> Entities<C> { impl<C: Config> Entities<C> {
@ -48,7 +49,7 @@ impl<C: Config> Entities<C> {
Self { Self {
slab: VersionedSlab::new(), slab: VersionedSlab::new(),
uuid_to_entity: HashMap::new(), uuid_to_entity: HashMap::new(),
network_id_to_entity: HashMap::new(), raw_id_to_entity: HashMap::new(),
} }
} }
@ -93,7 +94,7 @@ impl<C: Config> Entities<C> {
}); });
// TODO check for overflowing version? // TODO check for overflowing version?
self.network_id_to_entity.insert(k.version(), k.index()); self.raw_id_to_entity.insert(k.version(), k.index());
ve.insert(EntityId(k)); ve.insert(EntityId(k));
@ -113,7 +114,7 @@ impl<C: Config> Entities<C> {
.remove(&e.uuid) .remove(&e.uuid)
.expect("UUID should have been in UUID map"); .expect("UUID should have been in UUID map");
self.network_id_to_entity self.raw_id_to_entity
.remove(&entity.0.version()) .remove(&entity.0.version())
.expect("network ID should have been in the network ID map"); .expect("network ID should have been in the network ID map");
@ -133,7 +134,7 @@ impl<C: Config> Entities<C> {
.remove(&v.uuid) .remove(&v.uuid)
.expect("UUID should have been in UUID map"); .expect("UUID should have been in UUID map");
self.network_id_to_entity self.raw_id_to_entity
.remove(&k.version()) .remove(&k.version())
.expect("network ID should have been in the network ID map"); .expect("network ID should have been in the network ID map");
@ -174,10 +175,22 @@ impl<C: Config> Entities<C> {
self.slab.get_mut(entity.0) self.slab.get_mut(entity.0)
} }
pub(crate) fn get_with_network_id(&self, network_id: i32) -> Option<EntityId> { pub fn get_with_raw_id(&self, raw_id: i32) -> Option<(EntityId, &Entity<C>)> {
let version = NonZeroU32::new(network_id as u32)?; let version = NonZeroU32::new(raw_id as u32)?;
let index = *self.network_id_to_entity.get(&version)?; let index = *self.raw_id_to_entity.get(&version)?;
Some(EntityId(Key::new(index, version)))
let id = EntityId(Key::new(index, version));
let entity = self.get(id)?;
Some((id, entity))
}
pub fn get_with_raw_id_mut(&mut self, raw_id: i32) -> Option<(EntityId, &mut Entity<C>)> {
let version = NonZeroU32::new(raw_id as u32)?;
let index = *self.raw_id_to_entity.get(&version)?;
let id = EntityId(Key::new(index, version));
let entity = self.get_mut(id)?;
Some((id, entity))
} }
/// Returns an iterator over all entities on the server in an unspecified /// Returns an iterator over all entities on the server in an unspecified
@ -239,7 +252,7 @@ impl EntityId {
/// The value of the default entity ID which is always invalid. /// The value of the default entity ID which is always invalid.
pub const NULL: Self = Self(Key::NULL); pub const NULL: Self = Self(Key::NULL);
pub fn to_network_id(self) -> i32 { pub fn to_raw(self) -> i32 {
self.0.version().get() as i32 self.0.version().get() as i32
} }
} }
@ -279,6 +292,20 @@ pub(crate) struct EntityBits {
_pad: u8, _pad: u8,
} }
impl<C: Config> Deref for Entity<C> {
type Target = C::EntityState;
fn deref(&self) -> &Self::Target {
&self.state
}
}
impl<C: Config> DerefMut for Entity<C> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.state
}
}
impl<C: Config> Entity<C> { impl<C: Config> Entity<C> {
pub(crate) fn bits(&self) -> EntityBits { pub(crate) fn bits(&self) -> EntityBits {
self.bits self.bits
@ -709,13 +736,13 @@ impl<C: Config> Entity<C> {
/// been spawned. /// been spawned.
pub(crate) fn send_initial_tracked_data( pub(crate) fn send_initial_tracked_data(
&self, &self,
ctrl: &mut PlayPacketController, send: &mut PlayPacketSender,
this_id: EntityId, this_id: EntityId,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
// TODO: cache metadata buffer? // TODO: cache metadata buffer?
if let Some(metadata) = self.variants.initial_tracked_data() { if let Some(metadata) = self.variants.initial_tracked_data() {
ctrl.append_packet(&SetEntityMetadata { send.append_packet(&SetEntityMetadata {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
metadata: RawBytes(&metadata), metadata: RawBytes(&metadata),
})?; })?;
} }
@ -727,13 +754,13 @@ impl<C: Config> Entity<C> {
/// modified. /// modified.
pub(crate) fn send_updated_tracked_data( pub(crate) fn send_updated_tracked_data(
&self, &self,
ctrl: &mut PlayPacketController, send: &mut PlayPacketSender,
this_id: EntityId, this_id: EntityId,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
// TODO: cache metadata buffer? // TODO: cache metadata buffer?
if let Some(metadata) = self.variants.updated_tracked_data() { if let Some(metadata) = self.variants.updated_tracked_data() {
ctrl.append_packet(&SetEntityMetadata { send.append_packet(&SetEntityMetadata {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
metadata: RawBytes(&metadata), metadata: RawBytes(&metadata),
})?; })?;
} }
@ -745,10 +772,10 @@ impl<C: Config> Entity<C> {
pub(crate) fn send_spawn_packets( pub(crate) fn send_spawn_packets(
&self, &self,
this_id: EntityId, this_id: EntityId,
ctrl: &mut PlayPacketController, send: &mut PlayPacketSender,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let with_object_data = |data| SpawnEntity { let with_object_data = |data| SpawnEntity {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
object_uuid: self.uuid, object_uuid: self.uuid,
kind: VarInt(self.kind() as i32), kind: VarInt(self.kind() as i32),
position: self.new_position.into_array(), position: self.new_position.into_array(),
@ -761,14 +788,14 @@ impl<C: Config> Entity<C> {
match &self.variants { match &self.variants {
TrackedData::Marker(_) => {} TrackedData::Marker(_) => {}
TrackedData::ExperienceOrb(_) => ctrl.append_packet(&SpawnExperienceOrb { TrackedData::ExperienceOrb(_) => send.append_packet(&SpawnExperienceOrb {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
position: self.new_position.into_array(), position: self.new_position.into_array(),
count: 0, // TODO count: 0, // TODO
})?, })?,
TrackedData::Player(_) => { TrackedData::Player(_) => {
ctrl.append_packet(&SpawnPlayer { send.append_packet(&SpawnPlayer {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
player_uuid: self.uuid, player_uuid: self.uuid,
position: self.new_position.into_array(), position: self.new_position.into_array(),
yaw: ByteAngle::from_degrees(self.yaw), yaw: ByteAngle::from_degrees(self.yaw),
@ -776,17 +803,17 @@ impl<C: Config> Entity<C> {
})?; })?;
// Player spawn packet doesn't include head yaw for some reason. // Player spawn packet doesn't include head yaw for some reason.
ctrl.append_packet(&SetHeadRotation { send.append_packet(&SetHeadRotation {
entity_id: VarInt(this_id.to_network_id()), entity_id: VarInt(this_id.to_raw()),
head_yaw: ByteAngle::from_degrees(self.head_yaw), head_yaw: ByteAngle::from_degrees(self.head_yaw),
})?; })?;
} }
TrackedData::ItemFrame(e) => ctrl.append_packet(&with_object_data(e.get_rotation()))?, TrackedData::ItemFrame(e) => send.append_packet(&with_object_data(e.get_rotation()))?,
TrackedData::GlowItemFrame(e) => { TrackedData::GlowItemFrame(e) => {
ctrl.append_packet(&with_object_data(e.get_rotation()))? send.append_packet(&with_object_data(e.get_rotation()))?
} }
TrackedData::Painting(_) => ctrl.append_packet(&with_object_data( TrackedData::Painting(_) => send.append_packet(&with_object_data(
match ((self.yaw + 45.0).rem_euclid(360.0) / 90.0) as u8 { match ((self.yaw + 45.0).rem_euclid(360.0) / 90.0) as u8 {
0 => 3, 0 => 3,
1 => 4, 1 => 4,
@ -795,14 +822,14 @@ impl<C: Config> Entity<C> {
}, },
))?, ))?,
// TODO: set block state ID for falling block. // TODO: set block state ID for falling block.
TrackedData::FallingBlock(_) => ctrl.append_packet(&with_object_data(1))?, TrackedData::FallingBlock(_) => send.append_packet(&with_object_data(1))?,
TrackedData::FishingBobber(e) => { TrackedData::FishingBobber(e) => {
ctrl.append_packet(&with_object_data(e.get_hook_entity_id()))? send.append_packet(&with_object_data(e.get_hook_entity_id()))?
} }
TrackedData::Warden(e) => { TrackedData::Warden(e) => {
ctrl.append_packet(&with_object_data((e.get_pose() == Pose::Emerging).into()))? send.append_packet(&with_object_data((e.get_pose() == Pose::Emerging).into()))?
} }
_ => ctrl.append_packet(&with_object_data(0))?, _ => send.append_packet(&with_object_data(0))?,
} }
Ok(()) Ok(())
@ -828,17 +855,17 @@ mod tests {
#[test] #[test]
fn entities_has_valid_new_state() { fn entities_has_valid_new_state() {
let mut entities: Entities<MockConfig> = Entities::new(); let mut entities: Entities<MockConfig> = Entities::new();
let network_id: i32 = 8675309; let raw_id: i32 = 8675309;
let entity_id = EntityId(Key::new( let entity_id = EntityId(Key::new(
202298, 202298,
NonZeroU32::new(network_id as u32).expect("Value given should never be zero!"), NonZeroU32::new(raw_id as u32).expect("value given should never be zero!"),
)); ));
let uuid = Uuid::from_bytes([2; 16]); let uuid = Uuid::from_bytes([2; 16]);
assert!(entities.is_empty()); assert!(entities.is_empty());
assert!(entities.get(entity_id).is_none()); assert!(entities.get(entity_id).is_none());
assert!(entities.get_mut(entity_id).is_none()); assert!(entities.get_mut(entity_id).is_none());
assert!(entities.get_with_uuid(uuid).is_none()); assert!(entities.get_with_uuid(uuid).is_none());
assert!(entities.get_with_network_id(network_id).is_none()); assert!(entities.get_with_raw_id(raw_id).is_none());
} }
#[test] #[test]
@ -850,7 +877,7 @@ mod tests {
assert_eq!(entities.get(player_id).unwrap().state, 1); assert_eq!(entities.get(player_id).unwrap().state, 1);
let mut_player_entity = entities let mut_player_entity = entities
.get_mut(player_id) .get_mut(player_id)
.expect("Failed to get mutable reference"); .expect("failed to get mutable reference");
mut_player_entity.state = 100; mut_player_entity.state = 100;
assert_eq!(entities.get(player_id).unwrap().state, 100); assert_eq!(entities.get(player_id).unwrap().state, 100);
assert_eq!(entities.len(), 1); assert_eq!(entities.len(), 1);
@ -863,17 +890,17 @@ mod tests {
assert!(entities.is_empty()); assert!(entities.is_empty());
let (zombie_id, zombie_entity) = entities let (zombie_id, zombie_entity) = entities
.insert_with_uuid(EntityKind::Zombie, uuid, 1) .insert_with_uuid(EntityKind::Zombie, uuid, 1)
.expect("Unexpected Uuid collision when inserting to an empty collection"); .expect("unexpected Uuid collision when inserting to an empty collection");
assert_eq!(zombie_entity.state, 1); assert_eq!(zombie_entity.state, 1);
let maybe_zombie = entities let maybe_zombie = entities
.get_with_uuid(uuid) .get_with_uuid(uuid)
.expect("Uuid lookup failed on item already added to this collection"); .expect("UUID lookup failed on item already added to this collection");
assert_eq!(zombie_id, maybe_zombie); assert_eq!(zombie_id, maybe_zombie);
assert_eq!(entities.len(), 1); assert_eq!(entities.len(), 1);
} }
#[test] #[test]
fn entities_can_be_set_and_get_with_network_id() { fn entities_can_be_set_and_get_with_raw_id() {
let mut entities: Entities<MockConfig> = Entities::new(); let mut entities: Entities<MockConfig> = Entities::new();
assert!(entities.is_empty()); assert!(entities.is_empty());
let (boat_id, boat_entity) = entities.insert(EntityKind::Boat, 12); let (boat_id, boat_entity) = entities.insert(EntityKind::Boat, 12);
@ -881,18 +908,20 @@ mod tests {
let (cat_id, cat_entity) = entities.insert(EntityKind::Cat, 75); let (cat_id, cat_entity) = entities.insert(EntityKind::Cat, 75);
assert_eq!(cat_entity.state, 75); assert_eq!(cat_entity.state, 75);
let maybe_boat_id = entities let maybe_boat_id = entities
.get_with_network_id(boat_id.0.version.get() as i32) .get_with_raw_id(boat_id.0.version.get() as i32)
.expect("Network id lookup failed on item already added to this collection"); .expect("raw id lookup failed on item already added to this collection")
.0;
let maybe_boat = entities let maybe_boat = entities
.get(maybe_boat_id) .get(maybe_boat_id)
.expect("Failed to look up item already added to collection"); .expect("failed to look up item already added to collection");
assert_eq!(maybe_boat.state, 12); assert_eq!(maybe_boat.state, 12);
let maybe_cat_id = entities let maybe_cat_id = entities
.get_with_network_id(cat_id.0.version.get() as i32) .get_with_raw_id(cat_id.0.version.get() as i32)
.expect("Network id lookup failed on item already added to this collection"); .expect("raw id lookup failed on item already added to this collection")
.0;
let maybe_cat = entities let maybe_cat = entities
.get(maybe_cat_id) .get(maybe_cat_id)
.expect("Failed to look up item already added to collection"); .expect("failed to look up item already added to collection");
assert_eq!(maybe_cat.state, 75); assert_eq!(maybe_cat.state, 75);
assert_eq!(entities.len(), 2); assert_eq!(entities.len(), 2);
} }
@ -904,7 +933,7 @@ mod tests {
let (player_id, _) = entities.insert(EntityKind::Player, 1); let (player_id, _) = entities.insert(EntityKind::Player, 1);
let player_state = entities let player_state = entities
.remove(player_id) .remove(player_id)
.expect("Failed to remove an item from the collection"); .expect("failed to remove an item from the collection");
assert_eq!(player_state, 1); assert_eq!(player_state, 1);
} }

View file

@ -1,305 +1,183 @@
use std::ops::Range; use std::iter::FusedIterator;
use std::mem;
use std::num::Wrapping;
use std::ops::{Deref, DerefMut};
use thiserror::Error; use valence_protocol::packets::s2c::play::SetContainerSlotEncode;
use valence_protocol::{ItemStack, VarInt}; use valence_protocol::{InventoryKind, ItemStack, Text, VarInt};
use crate::config::Config;
use crate::server::PlayPacketSender;
use crate::slab_versioned::{Key, VersionedSlab}; use crate::slab_versioned::{Key, VersionedSlab};
pub type SlotId = i16; pub struct Inventories<C: Config> {
slab: VersionedSlab<Inventory<C>>,
pub trait Inventory {
fn slot(&self, slot_id: SlotId) -> Option<&ItemStack>;
/// Sets the slot to the desired contents. Returns the previous contents of
/// the slot.
fn set_slot(&mut self, slot_id: SlotId, slot: Option<ItemStack>) -> Option<ItemStack>;
fn slot_range(&self) -> Range<SlotId>;
fn slot_count(&self) -> usize {
self.slot_range().count()
} }
// TODO: `entry()` style api impl<C: Config> Inventories<C> {
fn slots(&self) -> Vec<Option<&ItemStack>> {
(0..self.slot_count())
.map(|s| self.slot(s as SlotId))
.collect()
}
/// Decreases the count for stack in the slot by amount. If there is not
/// enough items in the stack to perform the operation, then it will fail.
///
/// Returns `Ok` if the stack had enough items, and the operation was
/// carried out. Otherwise, it returns `Err` if `amount > stack.count()`,
/// and no changes were made to the inventory.
#[allow(clippy::unnecessary_unwrap)]
fn consume(&mut self, slot_id: SlotId, amount: impl Into<u8>) -> Result<(), InventoryError> {
let amount: u8 = amount.into();
let slot = self.slot(slot_id).cloned();
if slot.is_some() {
// Intentionally not using `if let` so stack can be moved out of the slot as mut
// to avoid another clone later.
let mut stack = slot.unwrap();
if amount > stack.count() {
return Err(InventoryError);
}
let slot = if amount == stack.count() {
None
} else {
stack.set_count(stack.count() - amount);
Some(stack)
};
self.set_slot(slot_id, slot);
}
Ok(())
}
}
pub(crate) trait InventoryDirtyable {
fn mark_dirty(&mut self, dirty: bool);
fn is_dirty(&self) -> bool;
}
/// Represents a player's Inventory.
#[derive(Debug, Clone)]
pub struct PlayerInventory {
pub(crate) slots: Box<[Option<ItemStack>; 46]>,
dirty: bool,
pub(crate) state_id: i32,
}
impl PlayerInventory {
/// General slots are the slots that can hold all items, including the
/// hotbar, excluding offhand. These slots are shown when the player is
/// looking at another inventory.
pub const GENERAL_SLOTS: Range<SlotId> = 9..45;
pub const HOTBAR_SLOTS: Range<SlotId> = 36..45;
pub fn hotbar_to_slot(hotbar_slot: i16) -> Option<SlotId> {
if !(0..=8).contains(&hotbar_slot) {
return None;
}
Some(Self::HOTBAR_SLOTS.start + hotbar_slot)
}
pub(crate) fn new() -> Self {
Self {
// Can't do the shorthand because Option<ItemStack> is not Copy.
slots: Box::new(std::array::from_fn(|_| None)),
dirty: true,
state_id: Default::default(),
}
}
}
impl Inventory for PlayerInventory {
fn slot(&self, slot_id: SlotId) -> Option<&ItemStack> {
if !self.slot_range().contains(&slot_id) {
return None;
}
self.slots[slot_id as usize].as_ref()
}
fn set_slot(&mut self, slot_id: SlotId, slot: Option<ItemStack>) -> Option<ItemStack> {
if !self.slot_range().contains(&slot_id) {
return None;
}
self.mark_dirty(true);
std::mem::replace(&mut self.slots[slot_id as usize], slot)
}
fn slot_range(&self) -> Range<SlotId> {
0..(self.slots.len() as SlotId)
}
}
impl InventoryDirtyable for PlayerInventory {
fn mark_dirty(&mut self, dirty: bool) {
self.dirty = dirty
}
fn is_dirty(&self) -> bool {
self.dirty
}
}
#[derive(Debug, Clone)]
pub struct ConfigurableInventory {
slots: Vec<Option<ItemStack>>,
/// The slots that the player can place items into for crafting. The
/// crafting result slot is always zero, and should not be included in this
/// range.
#[allow(dead_code)] // TODO: implement crafting
crafting_slots: Option<Range<SlotId>>,
/// The type of window that should be used to display this inventory.
pub window_type: VarInt,
dirty: bool,
}
impl ConfigurableInventory {
pub fn new(size: usize, window_type: VarInt, crafting_slots: Option<Range<SlotId>>) -> Self {
ConfigurableInventory {
slots: vec![None; size],
crafting_slots,
window_type,
dirty: false,
}
}
}
impl Inventory for ConfigurableInventory {
fn slot(&self, slot_id: SlotId) -> Option<&ItemStack> {
if !self.slot_range().contains(&slot_id) {
return None;
}
self.slots[slot_id as usize].as_ref()
}
fn set_slot(&mut self, slot_id: SlotId, slot: Option<ItemStack>) -> Option<ItemStack> {
if !self.slot_range().contains(&slot_id) {
return None;
}
self.mark_dirty(true);
std::mem::replace(&mut self.slots[slot_id as usize], slot)
}
fn slot_range(&self) -> Range<SlotId> {
0..(self.slots.len() as SlotId)
}
}
impl InventoryDirtyable for ConfigurableInventory {
fn mark_dirty(&mut self, dirty: bool) {
self.dirty = dirty
}
fn is_dirty(&self) -> bool {
self.dirty
}
}
/// Represents what the player sees when they open an object's Inventory.
///
/// This exists because when an object inventory screen is being shown to the
/// player, it also shows part of the player's inventory so they can move items
/// between the inventories.
pub struct WindowInventory {
pub window_id: u8,
pub object_inventory: InventoryId,
}
impl WindowInventory {
pub fn new(window_id: impl Into<u8>, object_inventory: InventoryId) -> Self {
WindowInventory {
window_id: window_id.into(),
object_inventory,
}
}
pub fn slots<'a>(
&self,
obj_inventory: &'a ConfigurableInventory,
player_inventory: &'a PlayerInventory,
) -> Vec<Option<&'a ItemStack>> {
let total_slots = obj_inventory.slots.len() + PlayerInventory::GENERAL_SLOTS.len();
(0..total_slots)
.map(|s| {
if s < obj_inventory.slot_count() {
return obj_inventory.slot(s as SlotId);
}
let offset = obj_inventory.slot_count();
player_inventory.slot((s - offset) as SlotId + PlayerInventory::GENERAL_SLOTS.start)
})
.collect()
}
}
#[derive(Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash, Default, Debug)]
pub struct InventoryId(Key);
/// Manages all inventories that are present in the server.
pub struct Inventories {
slab: VersionedSlab<ConfigurableInventory>,
}
impl Inventories {
pub(crate) fn new() -> Self { pub(crate) fn new() -> Self {
Self { Self {
slab: VersionedSlab::new(), slab: VersionedSlab::new(),
} }
} }
/// Creates a new inventory on a server.
pub fn insert( pub fn insert(
&mut self, &mut self,
inv: ConfigurableInventory, kind: InventoryKind,
) -> (InventoryId, &mut ConfigurableInventory) { title: impl Into<Text>,
let (key, value) = self.slab.insert(inv); state: C::InventoryState,
(InventoryId(key), value) ) -> (InventoryId, &mut Inventory<C>) {
let (id, inv) = self.slab.insert(Inventory {
state,
title: title.into(),
kind,
slots: vec![None; kind.slot_count()].into(),
modified: 0,
});
(InventoryId(id), inv)
} }
/// Removes an inventory from the server. pub fn remove(&mut self, id: InventoryId) -> Option<C::InventoryState> {
pub fn remove(&mut self, inv: InventoryId) -> Option<ConfigurableInventory> { self.slab.remove(id.0).map(|inv| inv.state)
self.slab.remove(inv.0)
} }
/// Returns the number of inventories in this container. pub fn get(&self, id: InventoryId) -> Option<&Inventory<C>> {
pub fn len(&self) -> usize { self.slab.get(id.0)
self.slab.len()
} }
/// Returns `true` if there are no inventories. pub fn get_mut(&mut self, id: InventoryId) -> Option<&mut Inventory<C>> {
pub fn is_empty(&self) -> bool { self.slab.get_mut(id.0)
self.slab.len() == 0
} }
pub fn get(&self, inv: InventoryId) -> Option<&ConfigurableInventory> { pub fn iter(
self.slab.get(inv.0) &self,
) -> impl ExactSizeIterator<Item = (InventoryId, &Inventory<C>)> + FusedIterator + Clone + '_
{
self.slab.iter().map(|(k, inv)| (InventoryId(k), inv))
} }
pub fn get_mut(&mut self, inv: InventoryId) -> Option<&mut ConfigurableInventory> { pub fn iter_mut(
self.slab.get_mut(inv.0) &mut self,
) -> impl ExactSizeIterator<Item = (InventoryId, &mut Inventory<C>)> + FusedIterator + '_ {
self.slab.iter_mut().map(|(k, inv)| (InventoryId(k), inv))
} }
pub(crate) fn update(&mut self) { pub(crate) fn update(&mut self) {
// now that we have synced all the dirty inventories, mark them as clean for (_, inv) in self.iter_mut() {
for (_, inv) in self.slab.iter_mut() { inv.modified = 0;
inv.mark_dirty(false);
} }
} }
} }
#[derive(Copy, Clone, Debug, Error)] #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash, Default, Debug)]
#[error("InventoryError")] pub struct InventoryId(Key);
pub struct InventoryError;
#[cfg(test)] impl InventoryId {
mod test { pub const NULL: Self = Self(Key::NULL);
use valence_protocol::{ItemKind, ItemStack};
use super::*;
#[test]
fn test_get_set_slots() {
let mut inv = PlayerInventory::new();
let slot = Some(ItemStack::new(ItemKind::Bone, 12, None));
let prev = inv.set_slot(9, slot.clone());
assert_eq!(inv.slot(9), slot.as_ref());
assert_eq!(prev, None);
} }
#[test] pub struct Inventory<C: Config> {
fn test_consume() { /// Custom state
let mut inv = PlayerInventory::new(); pub state: C::InventoryState,
let slot_id = 9; title: Text,
let slot = Some(ItemStack::new(ItemKind::Bone, 12, None)); kind: InventoryKind,
inv.set_slot(slot_id, slot); slots: Box<[Option<ItemStack>]>,
assert!(matches!(inv.consume(slot_id, 2), Ok(_))); /// Contains a set bit for each modified slot in `slots`.
assert_eq!(inv.slot(slot_id).unwrap().count(), 10); modified: u64,
assert!(matches!(inv.consume(slot_id, 20), Err(_))); }
assert_eq!(inv.slot(slot_id).unwrap().count(), 10);
assert!(matches!(inv.consume(slot_id, 10), Ok(_))); impl<C: Config> Deref for Inventory<C> {
assert_eq!(inv.slot(slot_id), None); type Target = C::InventoryState;
fn deref(&self) -> &Self::Target {
&self.state
}
}
impl<C: Config> DerefMut for Inventory<C> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.state
}
}
impl<C: Config> Inventory<C> {
pub fn slot(&self, idx: u16) -> Option<&ItemStack> {
self.slots
.get(idx as usize)
.expect("slot index out of range")
.as_ref()
}
pub fn replace_slot(
&mut self,
idx: u16,
item: impl Into<Option<ItemStack>>,
) -> Option<ItemStack> {
assert!(idx < self.slot_count(), "slot index out of range");
let new = item.into();
let old = &mut self.slots[idx as usize];
if new != *old {
self.modified |= 1 << idx;
}
mem::replace(old, new)
}
pub fn slot_count(&self) -> u16 {
self.slots.len() as u16
}
pub fn slots(
&self,
) -> impl ExactSizeIterator<Item = Option<&ItemStack>>
+ DoubleEndedIterator
+ FusedIterator
+ Clone
+ '_ {
self.slots.iter().map(|item| item.as_ref())
}
pub fn kind(&self) -> InventoryKind {
self.kind
}
pub fn title(&self) -> &Text {
&self.title
}
pub fn replace_title(&mut self, title: impl Into<Text>) -> Text {
// TODO: set title modified flag
mem::replace(&mut self.title, title.into())
}
pub(crate) fn slot_slice(&self) -> &[Option<ItemStack>] {
self.slots.as_ref()
}
pub(crate) fn send_update(
&self,
send: &mut PlayPacketSender,
window_id: u8,
state_id: &mut Wrapping<i32>,
) -> anyhow::Result<()> {
if self.modified != 0 {
for (idx, slot) in self.slots.iter().enumerate() {
if (self.modified >> idx) & 1 == 1 {
*state_id += 1;
send.append_packet(&SetContainerSlotEncode {
window_id: window_id as i8,
state_id: VarInt(state_id.0),
slot_idx: idx as i16,
slot_data: slot.as_ref(),
})?;
}
}
}
Ok(())
} }
} }

View file

@ -111,7 +111,6 @@ pub mod entity;
pub mod inventory; pub mod inventory;
pub mod player_list; pub mod player_list;
pub mod player_textures; pub mod player_textures;
#[doc(hidden)]
pub mod server; pub mod server;
mod slab; mod slab;
mod slab_rc; mod slab_rc;
@ -125,13 +124,11 @@ pub mod world;
pub mod prelude { pub mod prelude {
pub use biome::{Biome, BiomeId}; pub use biome::{Biome, BiomeId};
pub use chunk::{Chunk, ChunkPos, Chunks, LoadedChunk, UnloadedChunk}; pub use chunk::{Chunk, ChunkPos, Chunks, LoadedChunk, UnloadedChunk};
pub use client::{handle_event_default, Client, ClientEvent, ClientId, Clients}; pub use client::{Client, ClientEvent, ClientId, Clients};
pub use config::{Config, ConnectionMode, PlayerSampleEntry, ServerListPing}; pub use config::{Config, ConnectionMode, PlayerSampleEntry, ServerListPing};
pub use dimension::{Dimension, DimensionId}; pub use dimension::{Dimension, DimensionId};
pub use entity::{Entities, Entity, EntityEvent, EntityId, EntityKind, TrackedData}; pub use entity::{Entities, Entity, EntityEvent, EntityId, EntityKind, TrackedData};
pub use inventory::{ pub use inventory::{Inventories, Inventory, InventoryId};
ConfigurableInventory, Inventories, Inventory, InventoryId, PlayerInventory, SlotId,
};
pub use player_list::{PlayerList, PlayerListEntry, PlayerListId, PlayerLists}; pub use player_list::{PlayerList, PlayerListEntry, PlayerListId, PlayerLists};
pub use server::{NewClientData, Server, SharedServer, ShutdownResult}; pub use server::{NewClientData, Server, SharedServer, ShutdownResult};
pub use spatial_index::{RaycastHit, SpatialIndex}; pub use spatial_index::{RaycastHit, SpatialIndex};
@ -147,8 +144,8 @@ pub mod prelude {
pub use valence_protocol::text::Color; pub use valence_protocol::text::Color;
pub use valence_protocol::types::{GameMode, Hand, SoundCategory}; pub use valence_protocol::types::{GameMode, Hand, SoundCategory};
pub use valence_protocol::{ pub use valence_protocol::{
ident, translation_key, BlockKind, BlockPos, BlockState, Ident, ItemKind, ItemStack, Text, ident, translation_key, BlockKind, BlockPos, BlockState, Ident, InventoryKind, ItemKind,
TextFormat, Username, MINECRAFT_VERSION, PROTOCOL_VERSION, ItemStack, Text, TextFormat, Username, MINECRAFT_VERSION, PROTOCOL_VERSION,
}; };
pub use vek::{Aabb, Mat2, Mat3, Mat4, Vec2, Vec3, Vec4}; pub use vek::{Aabb, Mat2, Mat3, Mat4, Vec2, Vec3, Vec4};
pub use world::{World, WorldId, WorldMeta, Worlds}; pub use world::{World, WorldId, WorldMeta, Worlds};

View file

@ -2,6 +2,7 @@
use std::collections::hash_map::Entry; use std::collections::hash_map::Entry;
use std::collections::{HashMap, HashSet}; use std::collections::{HashMap, HashSet};
use std::ops::{Deref, DerefMut};
use bitfield_struct::bitfield; use bitfield_struct::bitfield;
use uuid::Uuid; use uuid::Uuid;
@ -11,12 +12,12 @@ use valence_protocol::{Text, VarInt};
use crate::config::Config; use crate::config::Config;
use crate::player_textures::SignedPlayerTextures; use crate::player_textures::SignedPlayerTextures;
use crate::server::PlayPacketController; use crate::server::PlayPacketSender;
use crate::slab_rc::{Key, SlabRc}; use crate::slab_rc::{Key, RcSlab};
/// A container for all [`PlayerList`]s on a server. /// A container for all [`PlayerList`]s on a server.
pub struct PlayerLists<C: Config> { pub struct PlayerLists<C: Config> {
slab: SlabRc<PlayerList<C>>, slab: RcSlab<PlayerList<C>>,
} }
/// An identifier for a [`PlayerList`] on the server. /// An identifier for a [`PlayerList`] on the server.
@ -33,7 +34,7 @@ pub struct PlayerListId(Key);
impl<C: Config> PlayerLists<C> { impl<C: Config> PlayerLists<C> {
pub(crate) fn new() -> Self { pub(crate) fn new() -> Self {
Self { Self {
slab: SlabRc::new(), slab: RcSlab::new(),
} }
} }
@ -110,6 +111,20 @@ pub struct PlayerList<C: Config> {
modified_header_or_footer: bool, modified_header_or_footer: bool,
} }
impl<C: Config> Deref for PlayerList<C> {
type Target = C::PlayerListState;
fn deref(&self) -> &Self::Target {
&self.state
}
}
impl<C: Config> DerefMut for PlayerList<C> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.state
}
}
impl<C: Config> PlayerList<C> { impl<C: Config> PlayerList<C> {
/// Inserts a player into the player list. /// Inserts a player into the player list.
/// ///
@ -243,10 +258,7 @@ impl<C: Config> PlayerList<C> {
self.entries.iter_mut().map(|(k, v)| (*k, v)) self.entries.iter_mut().map(|(k, v)| (*k, v))
} }
pub(crate) fn send_initial_packets( pub(crate) fn send_initial_packets(&self, send: &mut PlayPacketSender) -> anyhow::Result<()> {
&self,
ctrl: &mut PlayPacketController,
) -> anyhow::Result<()> {
let add_player: Vec<_> = self let add_player: Vec<_> = self
.entries .entries
.iter() .iter()
@ -272,11 +284,11 @@ impl<C: Config> PlayerList<C> {
.collect(); .collect();
if !add_player.is_empty() { if !add_player.is_empty() {
ctrl.append_packet(&PlayerInfo::AddPlayer(add_player))?; send.append_packet(&PlayerInfo::AddPlayer(add_player))?;
} }
if self.header != Text::default() || self.footer != Text::default() { if self.header != Text::default() || self.footer != Text::default() {
ctrl.append_packet(&SetTabListHeaderAndFooter { send.append_packet(&SetTabListHeaderAndFooter {
header: self.header.clone(), header: self.header.clone(),
footer: self.footer.clone(), footer: self.footer.clone(),
})?; })?;
@ -285,12 +297,9 @@ impl<C: Config> PlayerList<C> {
Ok(()) Ok(())
} }
pub(crate) fn send_update_packets( pub(crate) fn send_update_packets(&self, send: &mut PlayPacketSender) -> anyhow::Result<()> {
&self,
ctrl: &mut PlayPacketController,
) -> anyhow::Result<()> {
if !self.removed.is_empty() { if !self.removed.is_empty() {
ctrl.append_packet(&PlayerInfo::RemovePlayer( send.append_packet(&PlayerInfo::RemovePlayer(
self.removed.iter().cloned().collect(), self.removed.iter().cloned().collect(),
))?; ))?;
} }
@ -338,23 +347,23 @@ impl<C: Config> PlayerList<C> {
} }
if !add_player.is_empty() { if !add_player.is_empty() {
ctrl.append_packet(&PlayerInfo::AddPlayer(add_player))?; send.append_packet(&PlayerInfo::AddPlayer(add_player))?;
} }
if !game_mode.is_empty() { if !game_mode.is_empty() {
ctrl.append_packet(&PlayerInfo::UpdateGameMode(game_mode))?; send.append_packet(&PlayerInfo::UpdateGameMode(game_mode))?;
} }
if !ping.is_empty() { if !ping.is_empty() {
ctrl.append_packet(&PlayerInfo::UpdateLatency(ping))?; send.append_packet(&PlayerInfo::UpdateLatency(ping))?;
} }
if !display_name.is_empty() { if !display_name.is_empty() {
ctrl.append_packet(&PlayerInfo::UpdateDisplayName(display_name))?; send.append_packet(&PlayerInfo::UpdateDisplayName(display_name))?;
} }
if self.modified_header_or_footer { if self.modified_header_or_footer {
ctrl.append_packet(&SetTabListHeaderAndFooter { send.append_packet(&SetTabListHeaderAndFooter {
header: self.header.clone(), header: self.header.clone(),
footer: self.footer.clone(), footer: self.footer.clone(),
})?; })?;
@ -363,10 +372,7 @@ impl<C: Config> PlayerList<C> {
Ok(()) Ok(())
} }
pub(crate) fn queue_clear_packets( pub(crate) fn queue_clear_packets(&self, ctrl: &mut PlayPacketSender) -> anyhow::Result<()> {
&self,
ctrl: &mut PlayPacketController,
) -> anyhow::Result<()> {
ctrl.append_packet(&PlayerInfo::RemovePlayer( ctrl.append_packet(&PlayerInfo::RemovePlayer(
self.entries.keys().cloned().collect(), self.entries.keys().cloned().collect(),
)) ))

View file

@ -10,16 +10,16 @@ use std::{io, thread};
use anyhow::{ensure, Context}; use anyhow::{ensure, Context};
use flume::{Receiver, Sender}; use flume::{Receiver, Sender};
pub(crate) use packet_controller::PlayPacketController; pub(crate) use packet_manager::{PlayPacketReceiver, PlayPacketSender};
use rand::rngs::OsRng; use rand::rngs::OsRng;
use rayon::iter::ParallelIterator; use rayon::iter::ParallelIterator;
use reqwest::Client as HttpClient; use reqwest::Client as ReqwestClient;
use rsa::{PublicKeyParts, RsaPrivateKey}; use rsa::{PublicKeyParts, RsaPrivateKey};
use serde_json::{json, Value}; use serde_json::{json, Value};
use tokio::net::tcp::{OwnedReadHalf, OwnedWriteHalf}; use tokio::net::tcp::{OwnedReadHalf, OwnedWriteHalf};
use tokio::net::{TcpListener, TcpStream}; use tokio::net::{TcpListener, TcpStream};
use tokio::runtime::{Handle, Runtime}; use tokio::runtime::{Handle, Runtime};
use tokio::sync::Semaphore; use tokio::sync::{OwnedSemaphorePermit, Semaphore};
use tracing::{error, info, info_span, instrument, trace, warn}; use tracing::{error, info, info_span, instrument, trace, warn};
use uuid::Uuid; use uuid::Uuid;
use valence_nbt::{compound, Compound, List}; use valence_nbt::{compound, Compound, List};
@ -41,16 +41,20 @@ use crate::entity::Entities;
use crate::inventory::Inventories; use crate::inventory::Inventories;
use crate::player_list::PlayerLists; use crate::player_list::PlayerLists;
use crate::player_textures::SignedPlayerTextures; use crate::player_textures::SignedPlayerTextures;
use crate::server::packet_controller::InitialPacketController; use crate::server::packet_manager::InitialPacketManager;
use crate::world::Worlds; use crate::world::Worlds;
use crate::Ticks; use crate::Ticks;
mod byte_channel; mod byte_channel;
mod login; mod login;
mod packet_controller; mod packet_manager;
/// Contains the entire state of a running Minecraft server, accessible from /// Contains the entire state of a running Minecraft server, accessible from
/// within the [update](crate::config::Config::update) loop. /// within the [init] and [update] functions.
///
/// [init]: crate::config::Config::init
/// [update]: crate::config::Config::update
#[non_exhaustive]
pub struct Server<C: Config> { pub struct Server<C: Config> {
/// Custom state. /// Custom state.
pub state: C::ServerState, pub state: C::ServerState,
@ -65,7 +69,7 @@ pub struct Server<C: Config> {
/// All of the player lists on the server. /// All of the player lists on the server.
pub player_lists: PlayerLists<C>, pub player_lists: PlayerLists<C>,
/// All of the inventories on the server. /// All of the inventories on the server.
pub inventories: Inventories, pub inventories: Inventories<C>,
} }
/// A handle to a Minecraft server containing the subset of functionality which /// A handle to a Minecraft server containing the subset of functionality which
@ -119,26 +123,28 @@ struct SharedServerInner<C: Config> {
/// This is sent to clients during the authentication process. /// This is sent to clients during the authentication process.
public_key_der: Box<[u8]>, public_key_der: Box<[u8]>,
/// For session server requests. /// For session server requests.
http_client: HttpClient, http_client: ReqwestClient,
} }
/// Contains information about a new client. /// Contains information about a new client joining the server.
#[non_exhaustive] #[non_exhaustive]
pub struct NewClientData { pub struct NewClientData {
/// The UUID of the new client.
pub uuid: Uuid,
/// The username of the new client. /// The username of the new client.
pub username: Username<String>, pub username: Username<String>,
/// The UUID of the new client.
pub uuid: Uuid,
/// The remote address of the new client.
pub ip: IpAddr,
/// The new client's player textures. May be `None` if the client does not /// The new client's player textures. May be `None` if the client does not
/// have a skin or cape. /// have a skin or cape.
pub textures: Option<SignedPlayerTextures>, pub textures: Option<SignedPlayerTextures>,
/// The remote address of the new client.
pub remote_addr: IpAddr,
} }
struct NewClientMessage { struct NewClientMessage {
ncd: NewClientData, ncd: NewClientData,
ctrl: PlayPacketController, send: PlayPacketSender,
recv: PlayPacketReceiver,
permit: OwnedSemaphorePermit,
} }
/// The result type returned from [`start_server`]. /// The result type returned from [`start_server`].
@ -359,7 +365,7 @@ fn setup_server<C: Config>(cfg: C) -> anyhow::Result<SharedServer<C>> {
shutdown_result: Mutex::new(None), shutdown_result: Mutex::new(None),
rsa_key, rsa_key,
public_key_der, public_key_der,
http_client: HttpClient::new(), http_client: ReqwestClient::new(),
}; };
Ok(SharedServer(Arc::new(server))) Ok(SharedServer(Arc::new(server)))
@ -408,20 +414,23 @@ fn do_update_loop(server: &mut Server<impl Config>) -> ShutdownResult {
info!( info!(
username = %msg.ncd.username, username = %msg.ncd.username,
uuid = %msg.ncd.uuid, uuid = %msg.ncd.uuid,
ip = %msg.ncd.remote_addr, ip = %msg.ncd.ip,
"inserting client" "inserting client"
); );
server server.clients.insert(Client::new(
.clients msg.send,
.insert(Client::new(msg.ctrl, msg.ncd, Default::default())); msg.recv,
msg.permit,
msg.ncd,
Default::default(),
));
} }
// Get serverbound packets first so they are not dealt with a tick late. // Get serverbound packets first so they are not dealt with a tick late.
for (_, client) in server.clients.iter_mut() {
server.clients.par_iter_mut().for_each(|(_, client)| { client.prepare_c2s_packets();
client.handle_serverbound_packets(&server.entities); }
});
info_span!("configured_update").in_scope(|| shared.config().update(server)); info_span!("configured_update").in_scope(|| shared.config().update(server));
@ -472,11 +481,12 @@ async fn do_accept_loop(server: SharedServer<impl Config>) {
match server.0.connection_sema.clone().acquire_owned().await { match server.0.connection_sema.clone().acquire_owned().await {
Ok(permit) => match listener.accept().await { Ok(permit) => match listener.accept().await {
Ok((stream, remote_addr)) => { Ok((stream, remote_addr)) => {
let server = server.clone(); tokio::spawn(handle_connection(
tokio::spawn(async move { server.clone(),
handle_connection(server, stream, remote_addr).await; stream,
drop(permit); remote_addr,
}); permit,
));
} }
Err(e) => { Err(e) => {
error!("failed to accept incoming connection: {e}"); error!("failed to accept incoming connection: {e}");
@ -493,6 +503,7 @@ async fn handle_connection(
server: SharedServer<impl Config>, server: SharedServer<impl Config>,
stream: TcpStream, stream: TcpStream,
remote_addr: SocketAddr, remote_addr: SocketAddr,
permit: OwnedSemaphorePermit,
) { ) {
trace!("handling connection"); trace!("handling connection");
@ -502,17 +513,18 @@ async fn handle_connection(
let (read, write) = stream.into_split(); let (read, write) = stream.into_split();
let ctrl = InitialPacketController::new( let mngr = InitialPacketManager::new(
read, read,
write, write,
PacketEncoder::new(), PacketEncoder::new(),
PacketDecoder::new(), PacketDecoder::new(),
Duration::from_secs(5), Duration::from_secs(5),
permit,
); );
// TODO: peek stream for 0xFE legacy ping // TODO: peek stream for 0xFE legacy ping
if let Err(e) = handle_handshake(server, ctrl, remote_addr).await { if let Err(e) = handle_handshake(server, mngr, remote_addr).await {
// EOF can happen if the client disconnects while joining, which isn't // EOF can happen if the client disconnects while joining, which isn't
// very erroneous. // very erroneous.
if let Some(e) = e.downcast_ref::<io::Error>() { if let Some(e) = e.downcast_ref::<io::Error>() {
@ -526,10 +538,10 @@ async fn handle_connection(
async fn handle_handshake( async fn handle_handshake(
server: SharedServer<impl Config>, server: SharedServer<impl Config>,
mut ctrl: InitialPacketController<OwnedReadHalf, OwnedWriteHalf>, mut mngr: InitialPacketManager<OwnedReadHalf, OwnedWriteHalf>,
remote_addr: SocketAddr, remote_addr: SocketAddr,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
let handshake = ctrl.recv_packet::<HandshakeOwned>().await?; let handshake = mngr.recv_packet::<HandshakeOwned>().await?;
ensure!( ensure!(
matches!(server.connection_mode(), ConnectionMode::BungeeCord) matches!(server.connection_mode(), ConnectionMode::BungeeCord)
@ -538,21 +550,25 @@ async fn handle_handshake(
); );
match handshake.next_state { match handshake.next_state {
HandshakeNextState::Status => handle_status(server, ctrl, remote_addr, handshake) HandshakeNextState::Status => handle_status(server, mngr, remote_addr, handshake)
.await .await
.context("error handling status"), .context("error handling status"),
HandshakeNextState::Login => match handle_login(&server, &mut ctrl, remote_addr, handshake) HandshakeNextState::Login => match handle_login(&server, &mut mngr, remote_addr, handshake)
.await .await
.context("error handling login")? .context("error handling login")?
{ {
Some(ncd) => { Some(ncd) => {
let msg = NewClientMessage { let (send, recv, permit) = mngr.into_play(
ncd,
ctrl: ctrl.into_play_packet_controller(
server.0.incoming_capacity, server.0.incoming_capacity,
server.0.outgoing_capacity, server.0.outgoing_capacity,
server.tokio_handle().clone(), server.tokio_handle().clone(),
), );
let msg = NewClientMessage {
ncd,
send,
recv,
permit,
}; };
let _ = server.0.new_clients_tx.send_async(msg).await; let _ = server.0.new_clients_tx.send_async(msg).await;
@ -565,11 +581,11 @@ async fn handle_handshake(
async fn handle_status( async fn handle_status(
server: SharedServer<impl Config>, server: SharedServer<impl Config>,
mut ctrl: InitialPacketController<OwnedReadHalf, OwnedWriteHalf>, mut mngr: InitialPacketManager<OwnedReadHalf, OwnedWriteHalf>,
remote_addr: SocketAddr, remote_addr: SocketAddr,
handshake: HandshakeOwned, handshake: HandshakeOwned,
) -> anyhow::Result<()> { ) -> anyhow::Result<()> {
ctrl.recv_packet::<StatusRequest>().await?; mngr.recv_packet::<StatusRequest>().await?;
match server match server
.0 .0
@ -605,7 +621,7 @@ async fn handle_status(
.insert("favicon".to_owned(), Value::String(buf)); .insert("favicon".to_owned(), Value::String(buf));
} }
ctrl.send_packet(&StatusResponse { mngr.send_packet(&StatusResponse {
json: &json.to_string(), json: &json.to_string(),
}) })
.await?; .await?;
@ -613,9 +629,9 @@ async fn handle_status(
ServerListPing::Ignore => return Ok(()), ServerListPing::Ignore => return Ok(()),
} }
let PingRequest { payload } = ctrl.recv_packet().await?; let PingRequest { payload } = mngr.recv_packet().await?;
ctrl.send_packet(&PingResponse { payload }).await?; mngr.send_packet(&PingResponse { payload }).await?;
Ok(()) Ok(())
} }
@ -623,7 +639,7 @@ async fn handle_status(
/// Handle the login process and return the new client's data if successful. /// Handle the login process and return the new client's data if successful.
async fn handle_login( async fn handle_login(
server: &SharedServer<impl Config>, server: &SharedServer<impl Config>,
ctrl: &mut InitialPacketController<OwnedReadHalf, OwnedWriteHalf>, mngr: &mut InitialPacketManager<OwnedReadHalf, OwnedWriteHalf>,
remote_addr: SocketAddr, remote_addr: SocketAddr,
handshake: HandshakeOwned, handshake: HandshakeOwned,
) -> anyhow::Result<Option<NewClientData>> { ) -> anyhow::Result<Option<NewClientData>> {
@ -636,33 +652,33 @@ async fn handle_login(
username, username,
sig_data: _, // TODO sig_data: _, // TODO
profile_id: _, // TODO profile_id: _, // TODO
} = ctrl.recv_packet().await?; } = mngr.recv_packet().await?;
let username = username.to_owned_username(); let username = username.to_owned_username();
let ncd = match server.connection_mode() { let ncd = match server.connection_mode() {
ConnectionMode::Online => login::online(server, ctrl, remote_addr, username).await?, ConnectionMode::Online => login::online(server, mngr, remote_addr, username).await?,
ConnectionMode::Offline => login::offline(remote_addr, username)?, ConnectionMode::Offline => login::offline(remote_addr, username)?,
ConnectionMode::BungeeCord => login::bungeecord(&handshake.server_address, username)?, ConnectionMode::BungeeCord => login::bungeecord(&handshake.server_address, username)?,
ConnectionMode::Velocity { secret } => login::velocity(ctrl, username, secret).await?, ConnectionMode::Velocity { secret } => login::velocity(mngr, username, secret).await?,
}; };
if let Some(threshold) = server.0.cfg.compression_threshold() { if let Some(threshold) = server.0.cfg.compression_threshold() {
ctrl.send_packet(&SetCompression { mngr.send_packet(&SetCompression {
threshold: VarInt(threshold as i32), threshold: VarInt(threshold as i32),
}) })
.await?; .await?;
ctrl.set_compression(Some(threshold)); mngr.set_compression(Some(threshold));
} }
if let Err(reason) = server.0.cfg.login(server, &ncd).await { if let Err(reason) = server.0.cfg.login(server, &ncd).await {
info!("disconnect at login: \"{reason}\""); info!("disconnect at login: \"{reason}\"");
ctrl.send_packet(&DisconnectLogin { reason }).await?; mngr.send_packet(&DisconnectLogin { reason }).await?;
return Ok(None); return Ok(None);
} }
ctrl.send_packet(&LoginSuccess { mngr.send_packet(&LoginSuccess {
uuid: ncd.uuid, uuid: ncd.uuid,
username: ncd.username.as_str_username(), username: ncd.username.as_str_username(),
properties: Vec::new(), properties: Vec::new(),

View file

@ -24,14 +24,14 @@ use valence_protocol::{translation_key, Decode, Ident, RawBytes, Text, Username,
use crate::config::Config; use crate::config::Config;
use crate::player_textures::SignedPlayerTextures; use crate::player_textures::SignedPlayerTextures;
use crate::server::packet_controller::InitialPacketController; use crate::server::packet_manager::InitialPacketManager;
use crate::server::{NewClientData, SharedServer}; use crate::server::{NewClientData, SharedServer};
/// Login sequence for /// Login sequence for
/// [`ConnectionMode::Online`](crate::config::ConnectionMode). /// [`ConnectionMode::Online`](crate::config::ConnectionMode).
pub(super) async fn online( pub(super) async fn online(
server: &SharedServer<impl Config>, server: &SharedServer<impl Config>,
ctrl: &mut InitialPacketController<OwnedReadHalf, OwnedWriteHalf>, ctrl: &mut InitialPacketManager<OwnedReadHalf, OwnedWriteHalf>,
remote_addr: SocketAddr, remote_addr: SocketAddr,
username: Username<String>, username: Username<String>,
) -> anyhow::Result<NewClientData> { ) -> anyhow::Result<NewClientData> {
@ -132,7 +132,7 @@ pub(super) async fn online(
uuid, uuid,
username, username,
textures: Some(textures), textures: Some(textures),
remote_addr: remote_addr.ip(), ip: remote_addr.ip(),
}) })
} }
@ -147,7 +147,7 @@ pub(super) fn offline(
uuid: Uuid::from_slice(&Sha256::digest(username.as_str())[..16])?, uuid: Uuid::from_slice(&Sha256::digest(username.as_str())[..16])?,
username, username,
textures: None, textures: None,
remote_addr: remote_addr.ip(), ip: remote_addr.ip(),
}) })
} }
@ -188,7 +188,7 @@ pub(super) fn bungeecord(
uuid: uuid.parse()?, uuid: uuid.parse()?,
username, username,
textures, textures,
remote_addr: client_ip.parse()?, ip: client_ip.parse()?,
}) })
} }
@ -197,7 +197,7 @@ fn auth_digest(bytes: &[u8]) -> String {
} }
pub(super) async fn velocity( pub(super) async fn velocity(
ctrl: &mut InitialPacketController<OwnedReadHalf, OwnedWriteHalf>, ctrl: &mut InitialPacketManager<OwnedReadHalf, OwnedWriteHalf>,
username: Username<String>, username: Username<String>,
velocity_secret: &str, velocity_secret: &str,
) -> anyhow::Result<NewClientData> { ) -> anyhow::Result<NewClientData> {
@ -279,7 +279,7 @@ pub(super) async fn velocity(
uuid, uuid,
username, username,
textures, textures,
remote_addr, ip: remote_addr,
}) })
} }

View file

@ -5,6 +5,7 @@ use anyhow::Result;
use tokio::io; use tokio::io;
use tokio::io::{AsyncRead, AsyncReadExt, AsyncWrite, AsyncWriteExt}; use tokio::io::{AsyncRead, AsyncReadExt, AsyncWrite, AsyncWriteExt};
use tokio::runtime::Handle; use tokio::runtime::Handle;
use tokio::sync::OwnedSemaphorePermit;
use tokio::task::JoinHandle; use tokio::task::JoinHandle;
use tokio::time::timeout; use tokio::time::timeout;
use tracing::debug; use tracing::debug;
@ -12,17 +13,18 @@ use valence_protocol::{Decode, Encode, Packet, PacketDecoder, PacketEncoder};
use crate::server::byte_channel::{byte_channel, ByteReceiver, ByteSender, TryRecvError}; use crate::server::byte_channel::{byte_channel, ByteReceiver, ByteSender, TryRecvError};
pub struct InitialPacketController<R, W> { pub struct InitialPacketManager<R, W> {
reader: R, reader: R,
writer: W, writer: W,
enc: PacketEncoder, enc: PacketEncoder,
dec: PacketDecoder, dec: PacketDecoder,
timeout: Duration, timeout: Duration,
permit: OwnedSemaphorePermit,
} }
const READ_BUF_SIZE: usize = 4096; const READ_BUF_SIZE: usize = 4096;
impl<R, W> InitialPacketController<R, W> impl<R, W> InitialPacketManager<R, W>
where where
R: AsyncRead + Unpin, R: AsyncRead + Unpin,
W: AsyncWrite + Unpin, W: AsyncWrite + Unpin,
@ -33,6 +35,7 @@ where
enc: PacketEncoder, enc: PacketEncoder,
dec: PacketDecoder, dec: PacketDecoder,
timeout: Duration, timeout: Duration,
permit: OwnedSemaphorePermit,
) -> Self { ) -> Self {
Self { Self {
reader, reader,
@ -40,6 +43,7 @@ where
enc, enc,
dec, dec,
timeout, timeout,
permit,
} }
} }
@ -74,6 +78,7 @@ where
Ok(self Ok(self
.dec .dec
.try_next_packet()? .try_next_packet()?
// TODO: this panicked after a timeout.
.expect("decoder said it had another packet")) .expect("decoder said it had another packet"))
// The following is what I want to write but can't due to borrow // The following is what I want to write but can't due to borrow
@ -111,12 +116,12 @@ where
self.dec.enable_encryption(key); self.dec.enable_encryption(key);
} }
pub fn into_play_packet_controller( pub fn into_play(
mut self, mut self,
incoming_limit: usize, incoming_limit: usize,
outgoing_limit: usize, outgoing_limit: usize,
handle: Handle, handle: Handle,
) -> PlayPacketController ) -> (PlayPacketSender, PlayPacketReceiver, OwnedSemaphorePermit)
where where
R: Send + 'static, R: Send + 'static,
W: Send + 'static, W: Send + 'static,
@ -162,32 +167,33 @@ where
} }
}); });
PlayPacketController { (
PlayPacketSender {
enc: self.enc, enc: self.enc,
dec: self.dec,
send: outgoing_sender, send: outgoing_sender,
recv: incoming_receiver,
reader_task,
writer_task: Some(writer_task), writer_task: Some(writer_task),
handle, handle,
} },
PlayPacketReceiver {
dec: self.dec,
recv: incoming_receiver,
reader_task,
},
self.permit,
)
} }
} }
/// A convenience structure for managing a pair of packet encoder/decoders and /// Manages a packet encoder and a byte channel to send the encoded packets
/// the byte channels from which to send and receive the packet data during the /// through.
/// play state. pub struct PlayPacketSender {
pub struct PlayPacketController {
enc: PacketEncoder, enc: PacketEncoder,
dec: PacketDecoder,
send: ByteSender, send: ByteSender,
recv: ByteReceiver,
reader_task: JoinHandle<()>,
writer_task: Option<JoinHandle<()>>, writer_task: Option<JoinHandle<()>>,
handle: Handle, handle: Handle,
} }
impl PlayPacketController { impl PlayPacketSender {
pub fn append_packet<P>(&mut self, pkt: &P) -> Result<()> pub fn append_packet<P>(&mut self, pkt: &P) -> Result<()>
where where
P: Encode + Packet + ?Sized, P: Encode + Packet + ?Sized,
@ -202,6 +208,42 @@ impl PlayPacketController {
self.enc.prepend_packet(pkt) self.enc.prepend_packet(pkt)
} }
#[allow(dead_code)]
pub fn set_compression(&mut self, threshold: Option<u32>) {
self.enc.set_compression(threshold)
}
pub fn flush(&mut self) -> Result<()> {
let bytes = self.enc.take();
self.send.try_send(bytes)?;
Ok(())
}
}
impl Drop for PlayPacketSender {
fn drop(&mut self) {
let _ = self.flush();
if let Some(writer_task) = self.writer_task.take() {
if !writer_task.is_finished() {
let _guard = self.handle.enter();
// Give any unsent packets a moment to send before we cut the connection.
self.handle
.spawn(timeout(Duration::from_secs(1), writer_task));
}
}
}
}
/// Manages a packet decoder and a byte channel to receive the encoded packets.
pub struct PlayPacketReceiver {
dec: PacketDecoder,
recv: ByteReceiver,
reader_task: JoinHandle<()>,
}
impl PlayPacketReceiver {
pub fn try_next_packet<'a, P>(&'a mut self) -> Result<Option<P>> pub fn try_next_packet<'a, P>(&'a mut self) -> Result<Option<P>>
where where
P: Decode<'a> + Packet, P: Decode<'a> + Packet,
@ -220,33 +262,10 @@ impl PlayPacketController {
Err(TryRecvError::Disconnected) => false, Err(TryRecvError::Disconnected) => false,
} }
} }
#[allow(dead_code)]
pub fn set_compression(&mut self, threshold: Option<u32>) {
self.enc.set_compression(threshold)
} }
pub fn flush(&mut self) -> Result<()> { impl Drop for PlayPacketReceiver {
let bytes = self.enc.take();
self.send.try_send(bytes)?;
Ok(())
}
}
impl Drop for PlayPacketController {
fn drop(&mut self) { fn drop(&mut self) {
self.reader_task.abort(); self.reader_task.abort();
let _ = self.flush();
if let Some(writer_task) = self.writer_task.take() {
if !writer_task.is_finished() {
let _guard = self.handle.enter();
// Give any unsent packets a moment to send before we cut the connection.
self.handle
.spawn(timeout(Duration::from_secs(1), writer_task));
}
}
} }
} }

View file

@ -10,7 +10,7 @@ use rayon::iter::{IntoParallelRefIterator, IntoParallelRefMutIterator, ParallelI
use crate::slab::Slab; use crate::slab::Slab;
#[derive(Clone, Debug)] #[derive(Clone, Debug)]
pub struct SlabRc<T> { pub struct RcSlab<T> {
slab: Slab<Slot<T>>, slab: Slab<Slot<T>>,
} }
@ -56,7 +56,7 @@ impl Hash for Key {
} }
} }
impl<T> SlabRc<T> { impl<T> RcSlab<T> {
pub const fn new() -> Self { pub const fn new() -> Self {
Self { slab: Slab::new() } Self { slab: Slab::new() }
} }

View file

@ -1,6 +1,7 @@
//! A space on a server for objects to occupy. //! A space on a server for objects to occupy.
use std::iter::FusedIterator; use std::iter::FusedIterator;
use std::ops::{Deref, DerefMut};
use rayon::iter::ParallelIterator; use rayon::iter::ParallelIterator;
@ -66,11 +67,8 @@ impl<C: Config> Worlds<C> {
/// Note that any entities located in the world are not deleted. /// Note that any entities located in the world are not deleted.
/// Additionally, clients that are still in the deleted world at the end /// Additionally, clients that are still in the deleted world at the end
/// of the tick are disconnected. /// of the tick are disconnected.
/// pub fn remove(&mut self, world: WorldId) -> Option<C::WorldState> {
/// Returns `true` if the world was deleted. Otherwise, `false` is returned self.slab.remove(world.0).map(|w| w.state)
/// and the function has no effect.
pub fn remove(&mut self, world: WorldId) -> bool {
self.slab.remove(world.0).is_some()
} }
/// Removes all worlds from the server for which `f` returns `false`. /// Removes all worlds from the server for which `f` returns `false`.
@ -143,6 +141,20 @@ pub struct World<C: Config> {
pub meta: WorldMeta, pub meta: WorldMeta,
} }
impl<C: Config> Deref for World<C> {
type Target = C::WorldState;
fn deref(&self) -> &Self::Target {
&self.state
}
}
impl<C: Config> DerefMut for World<C> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.state
}
}
/// Contains miscellaneous data about the world. /// Contains miscellaneous data about the world.
pub struct WorldMeta { pub struct WorldMeta {
dimension: DimensionId, dimension: DimensionId,

View file

@ -1,5 +1,6 @@
//! Resource identifiers. //! Resource identifiers.
use std::borrow::Cow;
use std::cmp::Ordering; use std::cmp::Ordering;
use std::error::Error; use std::error::Error;
use std::fmt; use std::fmt;
@ -101,11 +102,6 @@ impl<S: AsRef<str>> Ident<S> {
pub fn into_inner(self) -> S { pub fn into_inner(self) -> S {
self.string self.string
} }
/// Consumes the identifier and returns the underlying string.
pub fn get(self) -> S {
self.string
}
} }
impl<'a, S: ?Sized> Ident<&'a S> { impl<'a, S: ?Sized> Ident<&'a S> {
@ -124,6 +120,39 @@ impl<'a, S: ?Sized> Ident<&'a S> {
} }
} }
impl<'a> From<Ident<&'a str>> for Ident<String> {
fn from(value: Ident<&'a str>) -> Self {
value.to_owned_ident()
}
}
impl<'a> From<Ident<&'a str>> for Ident<Box<str>> {
fn from(value: Ident<&'a str>) -> Self {
Ident {
string: value.string.into(),
path_start: value.path_start,
}
}
}
impl<'a> From<Ident<&'a str>> for Ident<Cow<'a, str>> {
fn from(value: Ident<&'a str>) -> Self {
Ident {
string: Cow::Borrowed(value.string),
path_start: value.path_start,
}
}
}
impl<'a> From<Ident<Cow<'a, str>>> for Ident<String> {
fn from(value: Ident<Cow<'a, str>>) -> Self {
Ident {
string: value.string.into_owned(),
path_start: value.path_start,
}
}
}
impl FromStr for Ident<String> { impl FromStr for Ident<String> {
type Err = IdentError<String>; type Err = IdentError<String>;

View file

@ -0,0 +1,62 @@
use valence_derive::{Decode, Encode};
#[derive(Copy, Clone, PartialEq, Eq, Debug, Encode, Decode)]
pub enum InventoryKind {
Generic9x1,
Generic9x2,
Generic9x3,
Generic9x4,
Generic9x5,
Generic9x6,
Generic3x3,
Anvil,
Beacon,
BlastFurnace,
BrewingStand,
Crafting,
Enchantment,
Furnace,
Grindstone,
Hopper,
Lectern,
Loom,
Merchant,
ShulkerBox,
Smithing,
Smoker,
Cartography,
Stonecutter,
}
impl InventoryKind {
/// The number of slots in this inventory, not counting the player's main
/// inventory slots.
pub const fn slot_count(self) -> usize {
match self {
InventoryKind::Generic9x1 => 9,
InventoryKind::Generic9x2 => 9 * 2,
InventoryKind::Generic9x3 => 9 * 3,
InventoryKind::Generic9x4 => 9 * 4,
InventoryKind::Generic9x5 => 9 * 5,
InventoryKind::Generic9x6 => 9 * 6,
InventoryKind::Generic3x3 => 3 * 3,
InventoryKind::Anvil => 4,
InventoryKind::Beacon => 1,
InventoryKind::BlastFurnace => 3,
InventoryKind::BrewingStand => 5,
InventoryKind::Crafting => 10,
InventoryKind::Enchantment => 2,
InventoryKind::Furnace => 3,
InventoryKind::Grindstone => 3,
InventoryKind::Hopper => 5,
InventoryKind::Lectern => 1,
InventoryKind::Loom => 4,
InventoryKind::Merchant => 3,
InventoryKind::ShulkerBox => 27,
InventoryKind::Smithing => 3,
InventoryKind::Smoker => 3,
InventoryKind::Cartography => 3,
InventoryKind::Stonecutter => 2,
}
}
}

View file

@ -39,8 +39,18 @@ impl ItemStack {
} }
impl Encode for Option<ItemStack> { impl Encode for Option<ItemStack> {
fn encode(&self, w: impl Write) -> Result<()> {
self.as_ref().encode(w)
}
fn encoded_len(&self) -> usize {
self.as_ref().encoded_len()
}
}
impl<'a> Encode for Option<&'a ItemStack> {
fn encode(&self, mut w: impl Write) -> Result<()> { fn encode(&self, mut w: impl Write) -> Result<()> {
match self { match *self {
None => false.encode(w), None => false.encode(w),
Some(s) => { Some(s) => {
true.encode(&mut w)?; true.encode(&mut w)?;
@ -55,7 +65,7 @@ impl Encode for Option<ItemStack> {
} }
fn encoded_len(&self) -> usize { fn encoded_len(&self) -> usize {
match self { match *self {
None => 1, None => 1,
Some(s) => { Some(s) => {
1 + s.item.encoded_len() 1 + s.item.encoded_len()

View file

@ -78,6 +78,7 @@ pub use byte_angle::ByteAngle;
pub use cache::{Cached, EncodedBuf}; pub use cache::{Cached, EncodedBuf};
pub use codec::{PacketDecoder, PacketEncoder}; pub use codec::{PacketDecoder, PacketEncoder};
pub use ident::Ident; pub use ident::Ident;
pub use inventory::InventoryKind;
pub use item::{ItemKind, ItemStack}; pub use item::{ItemKind, ItemStack};
pub use raw_bytes::RawBytes; pub use raw_bytes::RawBytes;
pub use text::{Text, TextFormat}; pub use text::{Text, TextFormat};
@ -108,6 +109,7 @@ pub mod enchant;
pub mod entity_meta; pub mod entity_meta;
pub mod ident; pub mod ident;
mod impls; mod impls;
mod inventory;
mod item; mod item;
pub mod packets; pub mod packets;
mod raw_bytes; mod raw_bytes;

View file

@ -7,7 +7,7 @@ use crate::item::ItemStack;
use crate::raw_bytes::RawBytes; use crate::raw_bytes::RawBytes;
use crate::types::{ use crate::types::{
Action, ChatMode, ClickContainerMode, CommandArgumentSignature, CommandBlockFlags, Action, ChatMode, ClickContainerMode, CommandArgumentSignature, CommandBlockFlags,
CommandBlockMode, DiggingStatus, DisplayedSkinParts, EntityInteraction, Hand, CommandBlockMode, Difficulty, DiggingStatus, DisplayedSkinParts, EntityInteraction, Hand,
HandshakeNextState, MainHand, MessageAcknowledgment, MsgSigOrVerifyToken, PlayerInputFlags, HandshakeNextState, MainHand, MessageAcknowledgment, MsgSigOrVerifyToken, PlayerInputFlags,
PublicKeyData, RecipeBookId, StructureBlockAction, StructureBlockFlags, StructureBlockMirror, PublicKeyData, RecipeBookId, StructureBlockAction, StructureBlockFlags, StructureBlockMirror,
StructureBlockMode, StructureBlockRotation, StructureBlockMode, StructureBlockRotation,
@ -116,17 +116,12 @@ pub mod play {
#[packet_id = 0x01] #[packet_id = 0x01]
pub struct QueryBlockEntityTag { pub struct QueryBlockEntityTag {
pub transaction_id: VarInt, pub transaction_id: VarInt,
pub location: BlockPos, pub position: BlockPos,
} }
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x02] #[packet_id = 0x02]
pub enum ChangeDifficulty { pub struct ChangeDifficulty(pub Difficulty);
Peaceful,
Easy,
Normal,
Hard,
}
#[derive(Clone, Debug, Encode, Decode, Packet)] #[derive(Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x03] #[packet_id = 0x03]
@ -164,7 +159,7 @@ pub mod play {
#[packet_id = 0x07] #[packet_id = 0x07]
pub enum ClientCommand { pub enum ClientCommand {
PerformRespawn, PerformRespawn,
RequestStatus, RequestStats,
} }
#[derive(Clone, Debug, Encode, Decode, Packet)] #[derive(Clone, Debug, Encode, Decode, Packet)]
@ -209,7 +204,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x0c] #[packet_id = 0x0c]
pub struct CloseContainerC2s { pub struct CloseContainerC2s {
pub window_id: u8, pub window_id: i8,
} }
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
@ -245,7 +240,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x11] #[packet_id = 0x11]
pub struct JigsawGenerate { pub struct JigsawGenerate {
pub location: BlockPos, pub position: BlockPos,
pub levels: VarInt, pub levels: VarInt,
pub keep_jigsaws: bool, pub keep_jigsaws: bool,
} }
@ -258,9 +253,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x13] #[packet_id = 0x13]
pub struct LockDifficulty { pub struct LockDifficulty(pub bool);
pub locked: bool,
}
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x14] #[packet_id = 0x14]
@ -332,7 +325,7 @@ pub mod play {
#[packet_id = 0x1d] #[packet_id = 0x1d]
pub struct PlayerAction { pub struct PlayerAction {
pub status: DiggingStatus, pub status: DiggingStatus,
pub location: BlockPos, pub position: BlockPos,
pub face: BlockFace, pub face: BlockFace,
pub sequence: VarInt, pub sequence: VarInt,
} }
@ -417,7 +410,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x29] #[packet_id = 0x29]
pub struct ProgramCommandBlock<'a> { pub struct ProgramCommandBlock<'a> {
pub location: BlockPos, pub position: BlockPos,
pub command: &'a str, pub command: &'a str,
pub mode: CommandBlockMode, pub mode: CommandBlockMode,
pub flags: CommandBlockFlags, pub flags: CommandBlockFlags,
@ -441,7 +434,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x2c] #[packet_id = 0x2c]
pub struct ProgramJigsawBlock<'a> { pub struct ProgramJigsawBlock<'a> {
pub location: BlockPos, pub position: BlockPos,
pub name: Ident<&'a str>, pub name: Ident<&'a str>,
pub target: Ident<&'a str>, pub target: Ident<&'a str>,
pub pool: Ident<&'a str>, pub pool: Ident<&'a str>,
@ -452,7 +445,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x2d] #[packet_id = 0x2d]
pub struct ProgramStructureBlock<'a> { pub struct ProgramStructureBlock<'a> {
pub location: BlockPos, pub position: BlockPos,
pub action: StructureBlockAction, pub action: StructureBlockAction,
pub mode: StructureBlockMode, pub mode: StructureBlockMode,
pub name: &'a str, pub name: &'a str,
@ -469,7 +462,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x2e] #[packet_id = 0x2e]
pub struct UpdateSign<'a> { pub struct UpdateSign<'a> {
pub location: BlockPos, pub position: BlockPos,
pub lines: [&'a str; 4], pub lines: [&'a str; 4],
} }
@ -487,7 +480,7 @@ pub mod play {
#[packet_id = 0x31] #[packet_id = 0x31]
pub struct UseItemOn { pub struct UseItemOn {
pub hand: Hand, pub hand: Hand,
pub location: BlockPos, pub position: BlockPos,
pub face: BlockFace, pub face: BlockFace,
pub cursor_pos: [f32; 3], pub cursor_pos: [f32; 3],
pub head_inside_block: bool, pub head_inside_block: bool,

View file

@ -145,14 +145,14 @@ pub mod play {
#[packet_id = 0x06] #[packet_id = 0x06]
pub struct SetBlockDestroyStage { pub struct SetBlockDestroyStage {
pub entity_id: VarInt, pub entity_id: VarInt,
pub location: BlockPos, pub position: BlockPos,
pub destroy_stage: u8, pub destroy_stage: u8,
} }
#[derive(Clone, Debug, Encode, Decode, Packet)] #[derive(Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x07] #[packet_id = 0x07]
pub struct BlockEntityData { pub struct BlockEntityData {
pub location: BlockPos, pub position: BlockPos,
// TODO: BlockEntityKind enum? // TODO: BlockEntityKind enum?
pub kind: VarInt, pub kind: VarInt,
pub data: Compound, pub data: Compound,
@ -161,7 +161,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x09] #[packet_id = 0x09]
pub struct BlockUpdate { pub struct BlockUpdate {
pub location: BlockPos, pub position: BlockPos,
pub block_id: VarInt, pub block_id: VarInt,
} }
@ -194,6 +194,15 @@ pub mod play {
pub carried_item: Option<ItemStack>, pub carried_item: Option<ItemStack>,
} }
#[derive(Copy, Clone, Debug, Encode, Packet)]
#[packet_id = 0x11]
pub struct SetContainerContentEncode<'a> {
pub window_id: u8,
pub state_id: VarInt,
pub slots: &'a [Option<ItemStack>],
pub carried_item: &'a Option<ItemStack>,
}
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x12] #[packet_id = 0x12]
pub struct SetContainerProperty { pub struct SetContainerProperty {
@ -211,6 +220,15 @@ pub mod play {
pub slot_data: Option<ItemStack>, pub slot_data: Option<ItemStack>,
} }
#[derive(Clone, Debug, Encode, Packet)]
#[packet_id = 0x13]
pub struct SetContainerSlotEncode<'a> {
pub window_id: i8,
pub state_id: VarInt,
pub slot_idx: i16,
pub slot_data: Option<&'a ItemStack>,
}
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x14] #[packet_id = 0x14]
pub struct SetCooldown { pub struct SetCooldown {
@ -512,7 +530,7 @@ pub mod play {
#[derive(Copy, Clone, Debug, Encode, Decode, Packet)] #[derive(Copy, Clone, Debug, Encode, Decode, Packet)]
#[packet_id = 0x4d] #[packet_id = 0x4d]
pub struct SetDefaultSpawnPosition { pub struct SetDefaultSpawnPosition {
pub location: BlockPos, pub position: BlockPos,
pub angle: f32, pub angle: f32,
} }

View file

@ -91,7 +91,7 @@ pub enum DiggingStatus {
FinishedDigging, FinishedDigging,
DropItemStack, DropItemStack,
DropItem, DropItem,
ShootArrowOrFinishEating, UpdateHeldItemState,
SwapItemInHand, SwapItemInHand,
} }