Don't allow more than 7 servers from one IP

This commit is contained in:
a1batross 2016-01-30 20:40:48 +03:00
parent def879f91c
commit 6eb0a2ae83
1 changed files with 35 additions and 27 deletions

View File

@ -34,18 +34,18 @@ def logPrint( msg ):
class PyMaster: class PyMaster:
serverList = [] serverList = []
sock = socket.socket( socket.AF_INET, socket.SOCK_DGRAM ) sock = socket.socket( socket.AF_INET, socket.SOCK_DGRAM )
def __init__(self): def __init__(self):
self.sock.bind( (UDP_IP, UDP_PORT) ) self.sock.bind( (UDP_IP, UDP_PORT) )
logPrint("Welcome to PyMaster!") logPrint("Welcome to PyMaster!")
logPrint("I ask you again, are you my master?") logPrint("I ask you again, are you my master?")
logPrint("Running on {0}:{1}".format( UDP_IP, UDP_PORT)) logPrint("Running on {0}:{1}".format( UDP_IP, UDP_PORT))
def serverLoop(self): def serverLoop(self):
data, addr = self.sock.recvfrom(1024) data, addr = self.sock.recvfrom(1024)
data = data.decode('latin_1') data = data.decode('latin_1')
if( data[0] == MasterProtocol.clientQuery ): if( data[0] == MasterProtocol.clientQuery ):
self.clientQuery(data, addr) self.clientQuery(data, addr)
elif( data[0] == MasterProtocol.challengeRequest ): elif( data[0] == MasterProtocol.challengeRequest ):
@ -67,18 +67,18 @@ class PyMaster:
except ValueError: except ValueError:
logPrint(traceback.format_exc()) logPrint(traceback.format_exc())
return return
queryAddr = query[0] # UNUSED queryAddr = query[0] # UNUSED
rawFilter = query[1] rawFilter = query[1]
# Remove first \ character # Remove first \ character
rawFilter = rawFilter.strip('\\') rawFilter = rawFilter.strip('\\')
split = rawFilter.split('\\') split = rawFilter.split('\\')
# Use NoneType as undefined # Use NoneType as undefined
gamedir = None gamedir = None
gamemap = None # UNUSED: until Xash3D will not support full filter gamemap = None # UNUSED: until Xash3D will not support full filter
for i in range( 0, len(split), 2 ): for i in range( 0, len(split), 2 ):
try: try:
key = split[i + 1] key = split[i + 1]
@ -90,43 +90,51 @@ class PyMaster:
logPrint('Unhandled info string entry: {0}/{1}'.format(split[i], key)) logPrint('Unhandled info string entry: {0}/{1}'.format(split[i], key))
except IndexError: except IndexError:
pass pass
packet = MasterProtocol.queryPacketHeader packet = MasterProtocol.queryPacketHeader
for i in self.serverList: for i in self.serverList:
if( time() > i.die ): if( time() > i.die ):
self.serverList.remove(i) self.serverList.remove(i)
continue continue
if( not i.check ): if( not i.check ):
continue continue
if( gamedir != None ): if( gamedir != None ):
if( gamedir != i.gamedir): if( gamedir != i.gamedir):
continue continue
# Use pregenerated address string # Use pregenerated address string
packet += i.queryAddr packet += i.queryAddr
packet += b'\0\0\0\0\0\0' # Fill last IP:Port with \0 packet += b'\0\0\0\0\0\0' # Fill last IP:Port with \0
self.sock.sendto(packet, addr) self.sock.sendto(packet, addr)
def removeServerFromList(self, data, addr): def removeServerFromList(self, data, addr):
logPrint("Remove Server: from {0}:{1}".format(addr[0], addr[1]))
for i in self.serverList: for i in self.serverList:
if (i.addr == addr): if (i.addr == addr):
logPrint("Remove Server: from {0}:{1}".format(addr[0], addr[1]))
self.serverList.remove(i) self.serverList.remove(i)
def sendChallengeToServer(self, data, addr): def sendChallengeToServer(self, data, addr):
logPrint("Challenge Request: from {0}:{1}".format(addr[0], addr[1])) logPrint("Challenge Request: from {0}:{1}".format(addr[0], addr[1]))
# At first, remove old server- data from list # At first, remove old server- data from list
self.removeServerFromList(None, addr) #self.removeServerFromList(None, addr)
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 > 7 ):
return
# Generate a 32 bit challenge number # Generate a 32 bit challenge number
challenge = random.randint(0, 2**32-1) challenge = random.randint(0, 2**32-1)
# Add server to list # Add server to list
self.serverList.append(ServerEntry(addr, challenge)) self.serverList.append(ServerEntry(addr, challenge))
# And send him a challenge # And send him a challenge
packet = MasterProtocol.challengePacketHeader packet = MasterProtocol.challengePacketHeader
packet += pack('I', challenge) packet += pack('I', challenge)
@ -136,21 +144,21 @@ class PyMaster:
logPrint("Add Server: from {0}:{1}".format(addr[0], addr[1])) logPrint("Add Server: from {0}:{1}".format(addr[0], addr[1]))
# Remove the header. Just for better parsing. # Remove the header. Just for better parsing.
serverInfo = data.strip('\x30\x0a\x5c') serverInfo = data.strip('\x30\x0a\x5c')
# Find a server with same address # Find a server with same address
for serverEntry in self.serverList: for serverEntry in self.serverList:
if( serverEntry.addr == addr ): if( serverEntry.addr == addr ):
break break
serverEntry.setInfoString( serverInfo ) serverEntry.setInfoString( serverInfo )
def sendStatus( self, data, addr ): def sendStatus( self, data, addr ):
logPrint("Status Request: from {0}:{1}".format(addr[0], addr[1])) logPrint("Status Request: from {0}:{1}".format(addr[0], addr[1]))
count = len(self.serverList) count = len(self.serverList)
packet = b'Server\t\t\tGame\tMap\t\tPlayers\tVersion\tChallenge\tCheck\n' packet = b'Server\t\t\tGame\tMap\t\tPlayers\tVersion\tChallenge\tCheck\n'
for i in self.serverList: for i in self.serverList:
line = '{0}:{1}\t{2}\t{3}\t\t{4}/{5}\t{6}\t{7}\t{8}n'.format(i.addr[0], i.addr[1], line = '{0}:{1}\t{2}\t{3}\t{4}/{5}\t{6}\t{7}\t{8}\n'.format(i.addr[0], i.addr[1],
i.gamedir, i.gamemap, i.players, i.gamedir, i.gamemap, i.players,
i.maxplayers, i.version, i.challenge, i.check) i.maxplayers, i.version, i.challenge, i.check)
packet += line.encode('latin_1') packet += line.encode('latin_1')
@ -160,9 +168,9 @@ class PyMaster:
def main( argv = None ): def main( argv = None ):
if argv is None: if argv is None:
argv = sys.argv argv = sys.argv
masterMain = PyMaster() masterMain = PyMaster()
while True: while True:
try: try:
masterMain.serverLoop() masterMain.serverLoop()
except Exception: except Exception:
@ -170,4 +178,4 @@ def main( argv = None ):
pass pass
if __name__ == "__main__": if __name__ == "__main__":
sys.exit( main( ) ) sys.exit( main( ) )