use shipyard::{Unique, UniqueView, UniqueViewMut, Workload, IntoWorkload, AllStoragesView, View, Get, NonSendSync, IntoIter}; use glam::IVec3; use hashbrown::HashMap; use kubi_shared::{ chunk::CHUNK_SIZE, queue::QueuedBlock, networking::{ channels::Channel, client::{Client, ClientId}, messages::{ClientToServerMessage, ClientToServerMessageType, ServerToClientMessage} }, }; use uflow::{server::RemoteClient, SendMode}; use lz4_flex::compress_prepend_size as lz4_compress; use anyhow::Result; use std::{rc::Rc, cell::RefCell}; use kubi_shared::networking::client::ClientIdMap; use crate::{ server::{UdpServer, ServerEvents}, config::ConfigTable, client::{ClientAddress, ClientAddressMap}, util::check_message_auth, }; pub mod chunk; pub mod tasks; use chunk::Chunk; use self::{ tasks::{ChunkTaskManager, ChunkTask, ChunkTaskResponse, init_chunk_task_manager}, chunk::ChunkState }; #[derive(Unique, Default)] pub struct LocalBlockQueue { pub queue: Vec, } #[derive(Unique, Default)] pub struct ChunkManager { pub chunks: HashMap } impl ChunkManager { pub fn unsubscribe_all(&mut self, client_id: ClientId) { for chunk in self.chunks.values_mut() { chunk.subscriptions.remove(&client_id); } } pub fn new() -> Self { Self::default() } } ///Sends a compressed chunk packet pub fn send_chunk_compressed( client: &Rc>, message: &ServerToClientMessage ) -> Result<()> { let mut ser_message = postcard::to_allocvec(&message)?; let mut compressed = lz4_compress(&ser_message[1..]); ser_message.truncate(1); ser_message.append(&mut compressed); let ser_message = ser_message.into_boxed_slice(); client.borrow_mut().send( ser_message, Channel::World as usize, SendMode::Reliable ); Ok(()) } fn process_chunk_requests( server: NonSendSync>, events: UniqueView, mut chunk_manager: UniqueViewMut, task_manager: UniqueView, config: UniqueView, addr_map: UniqueView, clients: View ) { for event in &events.0 { let Some(message) = check_message_auth ::<{ClientToServerMessageType::ChunkSubRequest as u8}> (&server, event, &clients, &addr_map) else { continue }; let ClientToServerMessage::ChunkSubRequest { chunk: chunk_position } = message.message else { unreachable!() }; if let Some(chunk) = chunk_manager.chunks.get_mut(&chunk_position) { chunk.subscriptions.insert(message.client_id); //TODO Start task here if status is "Nothing" if let Some(blocks) = &chunk.blocks { send_chunk_compressed( message.client, &ServerToClientMessage::ChunkResponse { chunk: chunk_position, data: blocks.clone(), queued: Vec::with_capacity(0) } ).unwrap(); } } else { let mut chunk = Chunk::new(chunk_position); chunk.state = ChunkState::Loading; chunk.subscriptions.insert(message.client_id); chunk_manager.chunks.insert(chunk_position, chunk); task_manager.spawn_task(ChunkTask::LoadChunk { position: chunk_position, seed: config.world.seed, }); } } } fn process_finished_tasks( server: NonSendSync>, task_manager: UniqueView, mut chunk_manager: UniqueViewMut, id_map: UniqueView, client_addr: View, mut local_queue: UniqueViewMut, ) { 'outer: while let Some(res) = task_manager.receive() { let ChunkTaskResponse::ChunkLoaded { chunk_position, blocks, queue } = res; let Some(chunk) = chunk_manager.chunks.get_mut(&chunk_position) else { log::warn!("Chunk discarded: Doesn't exist"); continue }; if chunk.state != ChunkState::Loading { log::warn!("Chunk discarded: Not Loading"); continue } chunk.state = ChunkState::Loaded; chunk.blocks = Some(blocks.clone()); local_queue.queue.extend_from_slice(&queue); log::debug!("Chunk {chunk_position} loaded, {} subs", chunk.subscriptions.len()); let chunk_packet = &ServerToClientMessage::ChunkResponse { chunk: chunk_position, data: blocks, queued: queue //should this be here? }; for &subscriber in &chunk.subscriptions { let Some(&entity_id) = id_map.0.get(&subscriber) else { log::error!("Invalid subscriber client id"); continue 'outer; }; let Ok(&ClientAddress(client_addr)) = (&client_addr).get(entity_id) else { log::error!("Invalid subscriber entity id"); continue 'outer; }; let Some(client) = server.0.client(&client_addr) else { log::error!("Client not connected"); continue 'outer; }; send_chunk_compressed(client, chunk_packet).unwrap(); // client.borrow_mut().send( // chunk_packet.clone(), // CHANNEL_WORLD, // SendMode::Reliable, // ); } } } fn process_block_queue_messages( server: NonSendSync>, events: UniqueView, addr_map: UniqueView, clients: View, addrs: View, mut queue: UniqueViewMut, ) { for event in &events.0 { let Some(message) = check_message_auth ::<{ClientToServerMessageType::QueueBlock as u8}> (&server, event, &clients, &addr_map) else { continue }; let ClientToServerMessage::QueueBlock { item } = message.message else { unreachable!() }; //place in our local world queue.queue.push(item); log::info!("Placed block {:?} at {}", item.block_type, item.position); for (other_client, other_client_address) in (&clients, &addrs).iter() { //No need to send the event back if message.client_id == other_client.0 { continue } //Get client let Some(client) = server.0.client(&other_client_address.0) else { log::error!("Client with address not found"); continue }; //Send the message client.borrow_mut().send( postcard::to_allocvec( &ServerToClientMessage::QueueBlock { item } ).unwrap().into_boxed_slice(), Channel::Block as usize, SendMode::Reliable, ); } } } fn process_block_queue( mut chunk_manager: UniqueViewMut, mut queue: UniqueViewMut, ) { let initial_len = queue.queue.len(); queue.queue.retain(|item| { let chunk_position = item.position.div_euclid(IVec3::splat(CHUNK_SIZE as i32)); let block_position = item.position.rem_euclid(IVec3::splat(CHUNK_SIZE as i32)); let Some(chunk) = chunk_manager.chunks.get_mut(&chunk_position) else { return true }; let Some(blocks) = &mut chunk.blocks else { return true }; blocks[block_position.x as usize][block_position.y as usize][block_position.z as usize] = item.block_type; false }); if initial_len != queue.queue.len() { log::debug!("queue processed {}/{} items", initial_len - queue.queue.len(), initial_len); } } /// init local block queue and chunk manager fn init_chunk_manager_and_block_queue( storages: AllStoragesView ) { storages.add_unique(ChunkManager::new()); storages.add_unique(LocalBlockQueue::default()); } pub fn init_world() -> Workload { ( init_chunk_manager_and_block_queue, init_chunk_task_manager, ).into_workload() } pub fn update_world() -> Workload { ( process_finished_tasks, process_block_queue_messages, process_block_queue, process_chunk_requests, ).into_sequential_workload() }