1use std::error;
2use std::io::prelude::*;
3use std::net::{TcpStream, ToSocketAddrs};
4use std::time::Duration;
5
6mod server_object;
7use server_object::ServerStatus;
8
9const TIMEOUT: Duration = Duration::from_secs(5);
10const MAX_PACKET_SIZE: u32 = 1024 * 1024 * 50; fn var_int_encode(num: i32) -> Vec<u8> {
13 let mut var_int = vec![];
15 let mut value = num;
16
17 while value >= 0x80 {
18 var_int.push(0x80 | (value as u8));
19 value >>= 7;
20 }
21
22 var_int.push(value as u8);
23 var_int
24}
25
26fn var_int_read(stream: &mut TcpStream) -> Result<i32, Box<dyn error::Error>> {
27 let mut value: i32 = 0;
29 let mut length = 0;
30 let mut current_byte = vec![0];
31
32 loop {
33 stream.read_exact(&mut current_byte)?;
34 value |= (current_byte[0] as i32 & 0x7F)
35 .checked_shl(length * 7)
36 .unwrap_or(0);
37 length += 1;
38 if length > 5 {
39 return Err("Server's reponse had invaild VarInt".into());
40 }
41 if (current_byte[0] & 0x80) != 0x80 {
42 break;
43 }
44 }
45 Ok(value)
46}
47
48fn var_int_pack(data: Vec<u8>) -> Vec<u8> {
49 let mut packed = var_int_encode(data.len() as i32);
51 packed.extend(data); packed
53}
54
55fn status_packet_builder(hostname: &str, port: u16) -> Vec<u8> {
56 vec![
58 var_int_pack(
59 [
60 vec![0x00, 0x00],
61 var_int_pack(hostname.as_bytes().to_vec()),
62 port.to_be_bytes().to_vec(),
63 vec![0x01],
64 ]
65 .into_iter()
66 .flatten()
67 .collect(),
68 ),
69 var_int_pack(vec![0x00]),
70 ]
71 .into_iter()
72 .flatten()
73 .collect()
74}
75
76pub fn get_server_json(hostname: &str, port: u16) -> Result<String, Box<dyn error::Error>> {
77 let socket_addr = match format!("{}:{}", hostname, port).to_socket_addrs()?.next() {
78 Some(socket) => socket,
79 None => return Err("Failed to parse hostname".into()),
80 };
81
82 let mut stream = TcpStream::connect_timeout(&socket_addr, TIMEOUT)?; stream.write_all(&status_packet_builder(hostname, port))?; let _length = var_int_read(&mut stream)?; let _id = var_int_read(&mut stream)?; let string_length = var_int_read(&mut stream)?; if string_length as u32 > MAX_PACKET_SIZE {
91 return Err("Response too large".into());
92 }
93
94 let mut buffer = vec![0; string_length as usize]; stream.read_exact(&mut buffer)?; let json: serde_json::Value = serde_json::from_str(&String::from_utf8(buffer)?)?;
99 Ok(json.to_string())
100}
101
102fn parse_json(json: &str) -> Result<ServerStatus, Box<dyn error::Error>> {
103 Ok(serde_json::from_str(json)?)
104 }
106
107pub fn server_status(hostname: &str, port: u16) -> Result<ServerStatus, Box<dyn error::Error>> {
108 let raw_json = get_server_json(hostname, port)?;
109 parse_json(&raw_json)
110}
111
112#[cfg(test)]
113mod tests {
114 use super::*;
115 #[test]
116 fn parse() {
117 let server_response = parse_json("{\"version\":{\"protocol\":758,\"name\":\"Velocity 1.7.2-1.18.2\"},\"players\":{\"online\":196,\"max\":150,\"sample\":[]},\"description\":{\"extra\":[{\"bold\":true,\"extra\":[{\"color\":\"aqua\",\"text\":\"E\"},{\"color\":\"aqua\",\"text\":\"a\"},{\"color\":\"aqua\",\"text\":\"r\"},{\"color\":\"aqua\",\"text\":\"t\"},{\"color\":\"aqua\",\"text\":\"h\"},{\"color\":\"green\",\"text\":\"M\"},{\"color\":\"green\",\"text\":\"C\"}],\"text\":\"\"},{\"text\":\"\\n\"},{\"color\":\"gray\",\"text\":\"Slava Ukraini!\"}],\"text\":\"\"},\"favicon\":\"\"}").unwrap();
118 assert_eq!(server_response.players.online, 196);
119 assert_eq!(server_response.players.max, 150);
120 assert_eq!(server_response.version.protocol, 758);
121 assert_eq!(server_response.version.name, "Velocity 1.7.2-1.18.2");
122 }
123}