From 8058a05876d84a79c282e6b9855feef10be03b03 Mon Sep 17 00:00:00 2001 From: MeexReay Date: Thu, 23 May 2024 19:15:38 +0300 Subject: [PATCH] fix compression mb --- Cargo.toml | 4 +- examples/status_server.rs | 29 +++++- examples/test_compression.rs | 38 ++++++++ src/lib.rs | 182 ++++++++++------------------------- 4 files changed, 119 insertions(+), 134 deletions(-) create mode 100644 examples/test_compression.rs diff --git a/Cargo.toml b/Cargo.toml index f09be0b..07d9999 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -7,10 +7,10 @@ license-file = "LICENSE" readme = "README.md" keywords = ["minecraft", "protocol", "packets", "lightweight"] -version = "0.1.6" +version = "0.1.7" edition = "2021" [dependencies] -flate2 = "1.0.30" +flate2 = { version = "1.0.30", features = ["zlib"]} bytebuffer = "2.2.0" uuid = "1.8.0" \ No newline at end of file diff --git a/examples/status_server.rs b/examples/status_server.rs index 52ec436..1101ed3 100644 --- a/examples/status_server.rs +++ b/examples/status_server.rs @@ -36,7 +36,9 @@ fn accept_client(mut conn: MCConnTcp, server: Arc>) -> Re loop { let mut packet = match conn.read_packet() { Ok(i) => i, - Err(_) => { break; }, + Err(_) => { + break; + }, }; if handshake { @@ -76,10 +78,31 @@ fn accept_client(mut conn: MCConnTcp, server: Arc>) -> Re fn main() { let server = MinecraftServer::new( - "localhost", + "127.0.0.1", 25565, 765, - "{\"version\":{\"protocol\":765,\"name\":\"Куриный ништяк\"},\"players\":{\"online\":0,\"max\":1},\"description\":{\"extra\":[{\"extra\":[{\"color\":\"aqua\",\"text\":\"☄\"},\" \",{\"bold\":true,\"extra\":[{\"color\":\"#00D982\",\"text\":\"S\"},{\"color\":\"#0DCB8B\",\"text\":\"l\"},{\"color\":\"#1BBC93\",\"text\":\"o\"},{\"color\":\"#28AE9C\",\"text\":\"g\"},{\"color\":\"#35A0A5\",\"text\":\"a\"},{\"color\":\"#4392AE\",\"text\":\"n\"},{\"color\":\"#5083B6\",\"text\":\"M\"},{\"color\":\"#5D75BF\",\"text\":\"C\"},{\"color\":\"#6A67C8\",\"text\":\".\"},{\"color\":\"#7858D0\",\"text\":\"r\"},{\"color\":\"#854AD9\",\"text\":\"u\"}],\"text\":\"\"},\" \",{\"color\":\"aqua\",\"text\":\"☄\"},\" \"],\"text\":\"\"},\"\\n\",{\"extra\":[{\"extra\":[{\"bold\":true,\"color\":\"#A1999E\",\"text\":\"░\"},\" \",{\"color\":\"#D1C7CD\",\"text\":\"прикол\"},\" \",{\"bold\":true,\"color\":\"#A1999E\",\"text\":\"░\"}],\"text\":\"\"}],\"text\":\" \"}],\"text\":\" \"},\"favicon\":\"\\u003d\"}" + "{ + \"version\":{ + \"protocol\":765, + \"name\":\"Version name\" + }, + \"players\":{ + \"online\":0, + \"max\":1, + \"sample\":[ + { + \"uuid\": \"\", + \"name\": \"Notch\" + } + ] + }, + \"description\": { + \"text\": \"Hello World!\", + \"color\": \"red\", + \"bold\": true + }, + \"favicon\": \"\" + }" ); let addr = server.server_ip.clone() + ":" + &server.server_port.to_string(); diff --git a/examples/test_compression.rs b/examples/test_compression.rs new file mode 100644 index 0000000..c85654b --- /dev/null +++ b/examples/test_compression.rs @@ -0,0 +1,38 @@ +use std::{net::TcpListener, sync::{atomic::AtomicBool, atomic::Ordering}, thread}; +use std::sync::mpsc::channel; +use rust_mc_proto::{DataBufferReader, DataBufferWriter, MCConn, MCConnTcp, MinecraftConnection, Packet, ProtocolError}; + +const LONG_TEXT: &str = "some_long_text_wow_123123123123123123123123"; + +fn main() { + let (tx, rx) = channel::<()>(); + + let server_tx = tx.clone(); + thread::spawn(move || { + let listener = TcpListener::bind("localhost:44447").unwrap(); + + server_tx.send(()).unwrap(); + + for stream in listener.incoming() { + let mut stream = MCConnTcp::new(stream.unwrap()); + stream.set_compression(2); + + let packet = stream.read_packet().unwrap(); + stream.write_packet(&packet).unwrap(); + } + }); + + rx.recv().unwrap(); + + let mut conn = MCConnTcp::connect("localhost:44447").unwrap(); + conn.set_compression(2); + + let mut packet = Packet::empty(0x12); + packet.write_string(LONG_TEXT).unwrap(); + conn.write_packet(&packet).unwrap(); + + let mut packet = conn.read_packet().unwrap(); + if packet.id == 0x12 && packet.read_string().unwrap() == LONG_TEXT { + println!("success"); + } +} diff --git a/src/lib.rs b/src/lib.rs index c89c48f..636b28a 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,7 +1,8 @@ use std::io::{Write, Read}; use std::net::{TcpStream, ToSocketAddrs}; -use flate2::bufread::ZlibDecoder; +use flate2::read::ZlibDecoder; +use flate2::write::ZlibEncoder; use flate2::{Compress, Compression, Decompress, FlushCompress, Status, FlushDecompress}; use bytebuffer::ByteBuffer; use uuid::Uuid; @@ -116,6 +117,15 @@ macro_rules! write_varint { }; } +macro_rules! return_error { + ($ex: expr, $error: expr) => { + match $ex { + Ok(i) => i, + Err(_) => { return Err($error) }, + } + }; +} + pub trait DataBufferReader { fn read_bytes(&mut self, size: usize) -> Result, ProtocolError>; fn read_byte(&mut self) -> Result; @@ -265,14 +275,8 @@ pub trait DataBufferWriter { impl DataBufferReader for R { fn read_bytes(&mut self, size: usize) -> Result, ProtocolError> { let mut buf = vec![0; size]; - match self.read(&mut buf) { - Ok(i) => { - if i < size { - Err(ProtocolError::ReadError) - } else { - Ok(buf) - } - }, + match self.read_exact(&mut buf) { + Ok(_) => Ok(buf), Err(_) => Err(ProtocolError::ReadError), } } @@ -391,14 +395,8 @@ impl MinecraftConnection { impl DataBufferReader for MinecraftConnection { fn read_bytes(&mut self, size: usize) -> Result, ProtocolError> { let mut buf = vec![0; size]; - match self.stream.read(&mut buf) { - Ok(i) => { - if i < size { - Err(ProtocolError::ReadError) - } else { - Ok(buf) - } - }, + match self.stream.read_exact(&mut buf) { + Ok(_) => Ok(buf), Err(_) => Err(ProtocolError::ReadError), } } @@ -436,155 +434,81 @@ impl MinecraftConnection { self.compress = true; self.compress_threashold = threashold; } + pub fn read_packet(&mut self) -> Result { if !self.compress { let length = self.read_usize_varint()?; let packet_id = self.read_u8_varint()?; - let mut data: Vec = vec![0; length - 1]; - match self.stream.read_exact(&mut data) { - Ok(i) => i, - Err(_) => { return Err(ProtocolError::ReadError) }, - }; + let data = self.read_bytes(length - 1)?; return Ok(Packet::from_bytes(packet_id, &data)) } - let packet_length = self.read_usize_varint_size()?; - let data_length = self.read_usize_varint_size()?; - if data_length.0 == 0 { + let packet_length = self.read_usize_varint()?; + let data_length = self.read_usize_varint()?; + + if data_length == 0 { let packet_id = self.read_u8_varint()?; - let mut data: Vec = vec![0; packet_length.0 - 2]; - match self.stream.read_exact(&mut data) { - Ok(i) => i, - Err(_) => { return Err(ProtocolError::ReadError) }, - }; + let data = self.read_bytes(packet_length - 2)?; return Ok(Packet::from_bytes(packet_id, &data)) } - let mut data: Vec = vec![0; packet_length.0 - packet_length.1 - data_length.1]; - match self.stream.read_exact(&mut data) { - Ok(i) => i, - Err(_) => { return Err(ProtocolError::ReadError) }, - }; + let data = self.read_bytes(packet_length - 2)?; + let mut data_buf = ByteBuffer::from_vec(decompress_zlib(&data, packet_length)?); - let mut data_buf = ByteBuffer::from_vec(decompress_zlib(&data, packet_length.0 - packet_length.1 - data_length.1)?); + let packet_id = return_error!(data_buf.read_u8_varint(), ProtocolError::VarIntError); + let mut packet_data = vec![0; data_length - 1]; + return_error!(data_buf.read_exact(&mut packet_data), ProtocolError::ReadError); - let packet_id = match data_buf.read_u8_varint() { - Ok(i) => i, - Err(_) => { return Err(ProtocolError::VarIntError) }, - }; - let mut data: Vec = vec![0; packet_length.0 - packet_length.1 - data_length.1 - 1]; - match data_buf.read_exact(&mut data) { - Ok(i) => i, - Err(_) => { return Err(ProtocolError::ReadError) }, - }; - - Ok(Packet::from_bytes(packet_id, &data)) + Ok(Packet::from_bytes(packet_id, &packet_data)) } - pub fn write_packet(&mut self, packet: &Packet) -> Result<(), ProtocolError> { + pub fn write_packet(&mut self, pack: &Packet) -> Result<(), ProtocolError> { let mut buf = ByteBuffer::new(); if !self.compress { - match buf.write_usize_varint(packet.buffer.len() + 1) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match buf.write_u8_varint(packet.id) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match buf.write_all(packet.buffer.as_bytes()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; + return_error!(buf.write_usize_varint(pack.buffer.len() + 1), ProtocolError::WriteError); + return_error!(buf.write_u8_varint(pack.id), ProtocolError::WriteError); + return_error!(buf.write_all(pack.buffer.as_bytes()), ProtocolError::WriteError); } else { - let mut pack = ByteBuffer::new(); + let mut packet = ByteBuffer::new(); - if packet.buffer.len() < self.compress_threashold { - match pack.write_usize_varint(0) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match pack.write_u8_varint(packet.id) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match pack.write_all(packet.buffer.as_bytes()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; + let mut data = ByteBuffer::new(); + return_error!(data.write_u8_varint(pack.id), ProtocolError::WriteError); + return_error!(data.write_all(pack.buffer.as_bytes()), ProtocolError::WriteError); + + if pack.buffer.len() < self.compress_threashold { + return_error!(packet.write_usize_varint(0), ProtocolError::WriteError); // data length + return_error!(packet.write_all(data.as_bytes()), ProtocolError::WriteError); } else { - let mut data = ByteBuffer::new(); - - match data.write_u8_varint(packet.id) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match data.write_all(packet.buffer.as_bytes()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - - let data = compress_zlib(data.as_bytes())?; - - match pack.write_usize_varint(packet.buffer.len() + 1) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match pack.write_all(&data) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; + return_error!(packet.write_usize_varint(data.len()), ProtocolError::WriteError); // data length + return_error!(packet.write_all(&compress_zlib(data.as_bytes())?), ProtocolError::WriteError); } - match buf.write_usize_varint(pack.len()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; - match buf.write_all(pack.as_bytes()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; + return_error!(buf.write_usize_varint(packet.len()), ProtocolError::WriteError); // packet length + return_error!(buf.write_all(packet.as_bytes()), ProtocolError::WriteError); } - match self.stream.write_all(buf.as_bytes()) { - Ok(_) => {}, - Err(_) => { return Err(ProtocolError::WriteError) }, - }; + return_error!(self.write_bytes(buf.as_bytes()), ProtocolError::WriteError); Ok(()) } } fn compress_zlib(bytes: &[u8]) -> Result, ProtocolError> { - let mut compresser = Compress::new(Compression::best(), true); - let mut output: Vec = Vec::new(); - match compresser.compress_vec(&bytes, &mut output, FlushCompress::Finish) { - Ok(i) => { - match i { - Status::Ok => Ok(output), - _ => Err(ProtocolError::ZlibError), - } - }, - Err(_) => Err(ProtocolError::ZlibError) - } + let mut encoder = ZlibEncoder::new(Vec::new(), Compression::fast()); + return_error!(encoder.write_all(bytes), ProtocolError::ZlibError); + let output = return_error!(encoder.finish(), ProtocolError::ZlibError); + Ok(output) } fn decompress_zlib(bytes: &[u8], packet_length: usize) -> Result, ProtocolError> { - let mut compresser = Decompress::new(true); - let mut output: Vec = Vec::with_capacity(packet_length); - match compresser.decompress_vec(&bytes, &mut output, FlushDecompress::Sync) { - Ok(i) => { - match i { - Status::Ok => Ok(output), - _ => Err(ProtocolError::ZlibError), - } - }, - Err(_) => Err(ProtocolError::ZlibError) - } + let mut decoder = ZlibDecoder::new(bytes); + let mut output = Vec::new(); + return_error!(decoder.read_to_end(&mut output), ProtocolError::ZlibError); + Ok(output) }