use crate::{config, log}; use std::net::{TcpListener, TcpStream}; use crate::mctypes::*; use crate::protocol::*; pub fn start_listening() { let server_address: &str = &format!("127.0.0.1:{}", config.port); let listener = TcpListener::bind(server_address); if listener.is_err() { log.error("Could not start listener"); } else { log.important(&format!("Started server on {}", server_address)); for stream in listener.unwrap().incoming() { if stream.is_err() { log.error("Could not connect to client"); } else { std::thread::spawn(move || { if let Err(e) = handle_client(stream.unwrap()) { log.error(&format!("Error when handling client: {}", e)); } }); } } } } fn handle_client(t: TcpStream) -> std::io::Result<()> { log.info("Got a client!"); let mut gc = GameConnection { stream: t, state: GameState::Handshake, protocol_version: 0, }; 'main: loop { match gc.state { GameState::Handshake => { // Read the handshake packet. let (_packet_len, packet_id) = read_packet_header(&mut gc.stream)?; let handshake = Handshake::read(&mut gc.stream)?; log.info(&format!("{:?}", handshake)); gc.state = if handshake.protocol_version.value != config.protocol_version as i32 { GameState::Closed } else { match handshake.next_state.value { 1 => GameState::Status, 2 => GameState::Login, _ => GameState::Closed, } }; log.info(&format!("Next state: {:?}", gc.state)); gc.protocol_version = handshake.protocol_version.value as u16; } GameState::Status => { // Read the request packet. let (_request_packet_len, _request_packet_id) = read_packet_header(&mut gc.stream)?; // Send the response packet. let response = MCString::from( r#"{ "version": { "name": "1.15.2", "protocol": 578 }, "players": { "max": 2147483647, "online": 69, "sample": [ { "name": "thinkofdeath", "id": "4566e69f-c907-48ee-8d71-d7ba5aa00d20" } ] }, "description": { "text": "ligma balls lol" }, "favicon": "" }"#, ); let packet_id = MCVarInt::from(0x00); let packet_len = MCVarInt::from( packet_id.to_bytes().len() as i32 + response.to_bytes().len() as i32, ); for b in packet_len.to_bytes() { write_byte(&mut gc.stream, b)?; } for b in packet_id.to_bytes() { write_byte(&mut gc.stream, b)?; } for b in response.to_bytes() { write_byte(&mut gc.stream, b)?; } // Read the ping packet. let (_ping_packet_len, _ping_packet_id) = read_packet_header(&mut gc.stream)?; let num = MCLong::from_stream(&mut gc.stream)?; log.info(&format!("Ping number: {:?}", num)); // Send the pong packet. let packet_id = MCVarInt::from(0x01); let packet_len = MCVarInt::from(packet_id.to_bytes().len() as i32 + 8i32); for b in packet_len.to_bytes() { write_byte(&mut gc.stream, b)?; } for b in packet_id.to_bytes() { write_byte(&mut gc.stream, b)?; } for b in num.to_bytes() { write_byte(&mut gc.stream, b)?; } gc.state = GameState::Closed; } GameState::Login => { // Read the login start packet. let (_packet_len, packet_id) = read_packet_header(&mut gc.stream)?; let login = LoginStart::read(&mut gc.stream)?; log.info(&format!("{:?}", login)); } GameState::Play => {} GameState::Closed => { log.info(&format!( "Client at {} closed connection", gc.stream.peer_addr().unwrap() )); break 'main; } } } Ok(()) } // pub struct MCPacket { // pub id: MCVarInt, // pub data: Vec, // } // #[allow(dead_code)] // impl MCPacket { // pub fn read_header(t: &mut TcpStream) -> std::io::Result<(MCVarInt, MCVarInt)> { // let length = MCVarInt::from_stream(t)?; // let id = MCVarInt::from_stream(t)?; // Ok((length, id)) // } // pub fn new(id: u8) -> MCPacket { // MCPacket { // id: MCVarInt::new(id as i32), // data: Vec::new(), // } // } // pub fn write(&mut self, v: Vec) { // for b in v { // self.data.push(b); // } // } // pub fn to_bytes(&self) -> Vec { // let mut bytes = Vec::new(); // for b in MCVarInt::new((self.id.to_bytes().len() + self.data.len()) as i32).to_bytes() { // bytes.push(b); // } // for b in self.id.to_bytes() { // bytes.push(b); // } // for b in &self.data { // bytes.push(*b); // } // bytes // } // } #[allow(dead_code)] #[derive(PartialEq, Debug)] pub enum GameState { Handshake, Status, Login, Play, Closed, } #[allow(dead_code)] pub struct GameConnection { pub stream: TcpStream, pub state: GameState, pub protocol_version: u16, }