crafty-4/app/classes/minecraft/raknet_ping.py

153 lines
5.7 KiB
Python
Raw Normal View History

from contextlib import redirect_stderr
import os
import socket
import time
from app.classes.shared.null_writer import NullWriter
with redirect_stderr(NullWriter()):
import psutil
class RaknetPing:
magic = b"\x00\xff\xff\x00\xfe\xfe\xfe\xfe\xfd\xfd\xfd\xfd\x12\x34\x56\x78"
fields = { # (len, signed)
2022-03-01 06:14:26 +00:00
"byte": (1, False),
"long": (8, True),
"ulong": (8, False),
"magic": (16, False),
"short": (2, True),
"ushort": (2, False), # unsigned short
"string": (2, False), # strlen is ushort
2022-03-01 06:14:26 +00:00
"bool": (1, False),
"address": (7, False),
"uint24le": (3, False),
2022-03-01 06:14:26 +00:00
}
byte_order = "big"
2022-03-01 06:14:26 +00:00
def __init__(self, server_addr, server_port, client_guid=0, timeout=5):
self.addr = server_addr
self.port = server_port
2022-03-01 06:14:26 +00:00
self.sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
self.sock.settimeout(timeout)
self.proc = psutil.Process(os.getpid())
self.guid = client_guid
self.guid_bytes = self.guid.to_bytes(8, RaknetPing.byte_order)
2022-03-01 06:14:26 +00:00
@staticmethod
def __byter(in_val, to_type):
f = RaknetPing.fields[to_type]
return in_val.to_bytes(f[0], RaknetPing.byte_order, signed=f[1])
2022-03-01 06:14:26 +00:00
@staticmethod
def __slice(in_bytes, pattern):
ret = []
bytes_index = 0
pattern_index = 0
while bytes_index < len(in_bytes):
2022-03-01 06:14:26 +00:00
try:
field = RaknetPing.fields[pattern[pattern_index]]
2022-03-01 06:14:26 +00:00
except IndexError as index_error:
raise IndexError(
"Ran out of pattern with additional bytes remaining"
) from index_error
if pattern[pattern_index] == "string":
string_header_length = field[0]
string_length = int.from_bytes(
in_bytes[bytes_index : bytes_index + string_header_length],
RaknetPing.byte_order,
signed=field[1],
)
length = string_header_length + string_length
ret.append(
in_bytes[
bytes_index
+ string_header_length : bytes_index
+ string_header_length
+ string_length
].decode("ascii")
)
elif pattern[pattern_index] == "magic":
length = field[0]
ret.append(in_bytes[bytes_index : bytes_index + length])
2022-03-01 06:14:26 +00:00
else:
length = field[0]
ret.append(
int.from_bytes(
in_bytes[bytes_index : bytes_index + length],
RaknetPing.byte_order,
signed=field[1],
)
)
bytes_index += length
pattern_index += 1
2022-03-01 06:14:26 +00:00
return ret
2022-03-01 06:14:26 +00:00
@staticmethod
def __get_time():
# return time.time_ns() // 1000000
2022-03-01 06:14:26 +00:00
return time.perf_counter_ns() // 1000000
2022-03-01 06:14:26 +00:00
def __sendping(self):
pack_id = RaknetPing.__byter(0x01, "byte")
now = RaknetPing.__byter(RaknetPing.__get_time(), "ulong")
2022-03-01 06:14:26 +00:00
guid = self.guid_bytes
d2s = pack_id + now + RaknetPing.magic + guid
# print("S:", d2s)
2022-03-01 06:14:26 +00:00
self.sock.sendto(d2s, (self.addr, self.port))
2022-03-01 06:14:26 +00:00
def __recvpong(self):
data = self.sock.recv(4096)
if data[0] == 0x1C:
2022-03-01 06:14:26 +00:00
ret = {}
sliced = RaknetPing.__slice(
data, ["byte", "ulong", "ulong", "magic", "string"]
)
if sliced[3] != RaknetPing.magic:
2022-03-01 06:14:26 +00:00
raise ValueError(f"Incorrect magic received ({sliced[3]})")
ret["server_guid"] = sliced[2]
try:
if len(sliced) >= 5:
ret["server_string_raw"] = sliced[4]
server_info = sliced[4].split(";")
ret["server_edition"] = server_info[0]
ret["server_motd"] = (server_info[1], server_info[7])
ret["server_protocol_version"] = server_info[2]
ret["server_version_name"] = server_info[3]
ret["server_player_count"] = server_info[4]
ret["server_player_max"] = server_info[5]
ret["server_uuid"] = server_info[6]
ret["server_game_mode"] = server_info[8]
ret["server_game_mode_num"] = server_info[9]
ret["server_port_ipv4"] = server_info[10]
ret["server_port_ipv6"] = server_info[11]
else:
ret["server_string_raw"] = ""
ret["server_edition"] = "Generic Raknet"
ret["server_motd"] = (None, None)
ret["server_protocol_version"] = None
ret["server_version_name"] = "Generic Raknet"
ret["server_player_count"] = -1
ret["server_player_max"] = -1
ret["server_uuid"] = None
ret["server_game_mode"] = None
ret["server_game_mode_num"] = 0
ret["server_port_ipv4"] = ""
ret["server_port_ipv6"] = ""
except:
raise ValueError(f"Received unexpected Raknet response: {sliced}")
2022-03-01 06:14:26 +00:00
return ret
2022-06-14 12:40:57 +00:00
raise ValueError(f"Incorrect packet type ({data[0]} detected")
2022-03-01 06:14:26 +00:00
def ping(self, retries=3):
rtr = retries
while rtr > 0:
try:
self.__sendping()
return self.__recvpong()
except ValueError as e:
print(
f"E: {e}, checking next packet. Retries remaining: {rtr}/{retries}"
)
2022-03-01 06:14:26 +00:00
rtr -= 1