1
0
Fork 0
mirror of https://github.com/de2tla2f/pymaster.git synced 2024-11-23 19:41:27 +00:00
pymaster/pymaster.py

196 lines
5.6 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
2016-01-19 21:32:16 +00:00
import socket
import random
import sys
import traceback
import logging
import os
from optparse import OptionParser
2023-12-26 16:58:55 +00:00
from struct import pack
from time import time
2016-01-19 21:32:16 +00:00
from server_entry import ServerEntry
from protocol import MasterProtocol
2016-01-19 21:32:16 +00:00
2023-12-26 16:58:55 +00:00
LOG_FILENAME = "pymaster.log"
2021-10-12 09:27:05 +00:00
MAX_SERVERS_FOR_IP = 14
2016-01-19 21:32:16 +00:00
2023-12-26 16:58:55 +00:00
class PyMaster:
def __init__(self, ip, port):
self.serverList = []
2023-12-26 17:01:26 +00:00
if ':' in ip:
self.sock = socket.socket(socket.AF_INET6, socket.SOCK_DGRAM)
else:
self.sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
2023-12-26 16:58:55 +00:00
self.sock.bind((ip, port))
logging.debug("Welcome to PyMaster!")
logging.debug("I ask you again, are you my master? @-@")
logging.debug("Running on %s:%d" % (ip, port))
def server_loop(self):
data, addr = self.sock.recvfrom(1024)
data = data.decode("latin_1")
match data[0]:
case MasterProtocol.clientQuery:
self.client_query(data, addr)
case MasterProtocol.challengeRequest:
self.send_challenge_to_server(data, addr)
case other:
logging.debug("Unknown message: {0} from {1}:{2}".format(data, addr[0], addr[1]))
def client_query(self, data, addr):
region = data[1] # UNUSED
data = data.strip("1" + region)
try:
query = data.split("\0")
except ValueError:
logging.debug(traceback.format_exc())
return
queryAddr = query[0] # UNUSED
rawFilter = query[1]
# Remove first \ character
rawFilter = rawFilter.strip("\\")
split = rawFilter.split("\\")
# Use NoneType as undefined
gamedir = "valve" # halflife, by default
clver = None
nat = 0
2023-12-26 17:07:15 +00:00
key = None
2023-12-26 16:58:55 +00:00
for i in range(0, len(split), 2):
try:
2023-12-26 17:07:15 +00:00
value = split[i + 1]
2023-12-26 16:58:55 +00:00
if split[i] == "gamedir":
2023-12-26 17:07:15 +00:00
gamedir = value.lower() # keep gamedir in lowercase
2023-12-26 16:58:55 +00:00
elif split[i] == "nat":
2023-12-26 17:07:15 +00:00
nat = int(value)
2023-12-26 16:58:55 +00:00
elif split[i] == "clver":
2023-12-26 17:07:15 +00:00
clver = value
elif split[i] == 'key':
key = int(value, 16)
2023-12-26 16:58:55 +00:00
else:
logging.debug(
"Unhandled info string entry: {0}/{1}. Infostring was: {2}".format(
2023-12-26 17:07:15 +00:00
split[i], value, split
2023-12-26 16:58:55 +00:00
)
)
except IndexError:
pass
packet = MasterProtocol.queryPacketHeader
2023-12-26 17:07:15 +00:00
if key != None: # Required in latest Xash3D version
packet += b'\x7F' + pack('<I', key) + b'\x00'
2023-12-26 16:58:55 +00:00
for i in self.serverList:
2024-01-09 02:10:47 +00:00
# Cleanup servers by timeout
if time() > i.die:
self.serverList.remove(i)
2024-01-09 02:39:46 +00:00
logging.debug("Server removed by timeout: {0}:{1}".format(addr[0], addr[1]))
2024-01-09 02:10:47 +00:00
continue
2023-12-26 16:58:55 +00:00
# Use pregenerated address string
packet += i.queryAddr
packet += b"\0\0\0\0\0\0" # Fill last IP:Port with \0
self.sock.sendto(packet, addr)
2023-12-26 16:58:55 +00:00
def send_challenge_to_server(self, data, addr):
2023-12-26 16:58:55 +00:00
2024-01-09 02:39:46 +00:00
logging.debug("Challenge request: {0}:{1}".format(addr[0], addr[1]))
2023-12-26 16:58:55 +00:00
# At first, remove old server- data from list
count = 0
for i in self.serverList:
if i.addr[0] == addr[0]:
if i.addr[1] == addr[1]:
self.serverList.remove(i)
else:
count += 1
if count > MAX_SERVERS_FOR_IP:
2024-01-08 19:10:13 +00:00
logging.debug("Reached MAX_SERVERS_FOR_IP: {0}".format(MAX_SERVERS_FOR_IP))
2023-12-26 16:58:55 +00:00
return
challenge = random.randint(0, 2**32 - 1)
# Add server to list
self.serverList.append(ServerEntry(addr, challenge))
# And send him a challenge
packet = MasterProtocol.challengePacketHeader
packet += pack("I", challenge)
self.sock.sendto(packet, addr)
def spawn_pymaster(verbose, ip, port):
if verbose:
logging.getLogger().addHandler(logging.StreamHandler())
logging.getLogger().addHandler(logging.FileHandler(LOG_FILENAME))
logging.getLogger().setLevel(logging.DEBUG)
2016-01-19 21:32:16 +00:00
2023-12-26 16:58:55 +00:00
masterMain = PyMaster(ip, port)
while True:
try:
masterMain.server_loop()
except Exception:
logging.debug(traceback.format_exc())
2016-01-19 21:32:16 +00:00
if __name__ == "__main__":
2023-12-26 16:58:55 +00:00
parser = OptionParser()
parser.add_option(
"-i",
"--ip",
action="store",
dest="ip",
default="0.0.0.0",
help="ip to listen [default: %default]",
)
parser.add_option(
"-p",
"--port",
action="store",
dest="port",
type="int",
default=27010,
help="port to listen [default: %default]",
)
parser.add_option(
"-d",
"--daemonize",
action="store_true",
dest="daemonize",
default=False,
help="run in background, argument is uid [default: %default]",
)
parser.add_option(
"-q",
"--quiet",
action="store_false",
dest="verbose",
default=True,
help="don't print to stdout [default: %default]",
)
(options, args) = parser.parse_args()
if options.daemonize != 0:
from daemon import DaemonContext
with DaemonContext(
stdout=sys.stdout, stderr=sys.stderr, working_directory=os.getcwd()
) as context:
spawn_pymaster(options.verbose, options.ip, options.port)
else:
sys.exit(spawn_pymaster(options.verbose, options.ip, options.port))