pyCraft/networking/NetworkManager.py

351 lines
16 KiB
Python
Raw Normal View History

2012-04-07 16:02:46 +02:00
import socket
2012-04-15 18:49:47 +02:00
import PacketListenerManager
2012-04-07 16:02:46 +02:00
import urllib2
import traceback
2012-04-10 21:14:32 +02:00
import threading
import hashlib
import string
import unicodedata
import Utils
import sys
from networking import PacketSenderManager
from Crypto.Random import _UserFriendlyRNG
from Crypto.Util import asn1
from Crypto.PublicKey import RSA
from Crypto.Cipher import AES
from Crypto.Cipher import PKCS1_v1_5
2012-04-07 16:02:46 +02:00
EntityID = 0
class ServerConnection(threading.Thread):
def __init__(self, pluginLoader, username, sessionID, server, port, options=None):
threading.Thread.__init__(self)
self.pluginLoader = pluginLoader
self.options = options
self.isConnected = False
2012-04-07 16:02:46 +02:00
self.username = username
self.sessionID = sessionID
self.server = server
self.port = port
2012-12-25 21:38:02 +01:00
def disconnect(self, reason="Disconnected by user"):
PacketSenderManager.sendFF(self.socket, reason)
2012-09-12 02:11:58 +02:00
self.listener.kill = True
self.socket.close()
2012-12-25 21:38:02 +01:00
2012-07-17 23:42:42 +02:00
def setWindow(self, window):
self.window = window
2012-12-25 21:38:02 +01:00
2012-04-15 18:49:47 +02:00
def grabSocket(self):
return self.socket
2012-12-25 21:38:02 +01:00
def run(self):
2012-04-07 16:02:46 +02:00
try:
#Create the socket and fileobject
2012-12-25 21:38:02 +01:00
self.socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
self.socket.connect(( self.server, self.port ))
2012-04-15 16:51:00 +02:00
self.FileObject = self.socket.makefile()
2012-12-25 21:38:02 +01:00
#Send out the handshake packet
PacketSenderManager.sendHandshake(self.socket, self.username, self.server, self.port)
2012-12-25 21:38:02 +01:00
#Receive the encryption packet id
packetid = self.socket.recv(1)
2012-12-25 21:38:02 +01:00
#Sanity check the packet id
assert packetid == "\xFD", "Server didn't respond back to handshake with proper packet!"
2012-12-25 21:38:02 +01:00
#Parse the packet
packetFD = PacketListenerManager.handleFD(self.FileObject)
2012-12-25 21:38:02 +01:00
#Import the server's public key
self.pubkey = RSA.importKey(packetFD['Public Key'])
2012-12-25 21:38:02 +01:00
#Generate a 16 byte (128 bit) shared secret
self.sharedSecret = _UserFriendlyRNG.get_random_bytes(16)
2012-12-25 21:38:02 +01:00
#Authenticate the server from sessions.minecraft.net
2012-12-25 21:38:02 +01:00
if (packetFD['ServerID'] != '-'):
2012-07-17 23:42:42 +02:00
try:
2012-12-04 10:48:49 +01:00
#Grab the server id
sha1 = hashlib.sha1()
sha1.update(packetFD['ServerID'])
sha1.update(self.sharedSecret)
sha1.update(packetFD['Public Key'])
#lovely java style hex digest by barneygale
serverid = Utils.javaHexDigest(sha1)
#Open up the url with the appropriate get parameters
2012-07-17 23:42:42 +02:00
url = "http://session.minecraft.net/game/joinserver.jsp?user=" + self.username + "&sessionId=" + self.sessionID + "&serverId=" + serverid
response = urllib2.urlopen(url).read()
2012-12-25 21:38:02 +01:00
if (response != "OK"):
print "Response from sessions.minecraft.net wasn't OK, it was " + response
return False
2012-12-25 21:38:02 +01:00
2012-12-04 10:48:49 +01:00
#Success \o/ We can now begin sending our serverAddress to the server
2012-12-25 21:38:02 +01:00
#Instantiate our main packet listener
self.listener = PacketListener(self, self.socket, self.FileObject)
self.listener.setDaemon(True)
self.listener.start()
2012-12-25 21:38:02 +01:00
#Encrypt the verification token from earlier along with our shared secret with the server's rsa key
self.RSACipher = PKCS1_v1_5.new(self.pubkey)
encryptedSanityToken = self.RSACipher.encrypt(str(packetFD['Token']))
encryptedSharedSecret = self.RSACipher.encrypt(str(self.sharedSecret))
2012-12-25 21:38:02 +01:00
#Send out a a packet FC to the server
PacketSenderManager.sendFC(self.socket, encryptedSharedSecret, encryptedSanityToken)
2012-12-26 16:23:47 +01:00
self.pluginLoader.notify("onConnect")
2012-12-25 21:38:02 +01:00
2012-07-17 23:42:42 +02:00
except Exception, e:
traceback.print_exc()
else:
print "Server is in offline mode"
2012-12-04 10:48:49 +01:00
#Instantiate our main packet listener
self.listener = PacketListener(self, self.socket, self.FileObject)
self.listener.setDaemon(True)
2012-12-04 10:48:49 +01:00
self.listener.start()
2012-12-25 21:38:02 +01:00
2012-12-04 10:48:49 +01:00
#Encrypt the verification token from earlier along with our shared secret with the server's rsa key
self.RSACipher = PKCS1_v1_5.new(self.pubkey)
encryptedSanityToken = self.RSACipher.encrypt(str(packetFD['Token']))
encryptedSharedSecret = self.RSACipher.encrypt(str(self.sharedSecret))
2012-12-25 21:38:02 +01:00
2012-12-04 10:48:49 +01:00
#Send out a a packet FC to the server
PacketSenderManager.sendFC(self.socket, encryptedSharedSecret, encryptedSanityToken)
2012-12-26 16:23:47 +01:00
self.pluginLoader.notify("onConnect")
2012-04-07 16:02:46 +02:00
except Exception, e:
print "Connection to server failed"
2012-04-07 16:02:46 +02:00
traceback.print_exc()
sys.exit(1)
2012-12-25 21:38:02 +01:00
class EncryptedFileObjectHandler():
def __init__(self, fileobject, cipher):
self.fileobject = fileobject
self.cipher = cipher
self.length = 0
2012-12-25 21:38:02 +01:00
def read(self, length):
rawData = self.fileobject.read(length)
self.length += length
unencryptedData = self.cipher.decrypt(rawData)
return unencryptedData
2012-12-25 21:38:02 +01:00
def tell(self):
return self.length
2012-12-25 21:38:02 +01:00
class EncryptedSocketObjectHandler():
def __init__(self, socket, cipher):
self.socket = socket
self.cipher = cipher
2012-12-25 21:38:02 +01:00
2012-12-04 10:48:49 +01:00
def send(self, serverAddress):
self.socket.send(self.cipher.encrypt(serverAddress))
2012-12-25 21:38:02 +01:00
def close(self):
self.socket.close()
2012-12-25 21:38:02 +01:00
2012-04-10 21:14:32 +02:00
class PacketListener(threading.Thread):
def __init__(self, connection, socket, FileObject):
2012-04-10 21:14:32 +02:00
threading.Thread.__init__(self)
self.connection = connection
2012-04-10 21:14:32 +02:00
self.socket = socket
2012-04-15 16:51:00 +02:00
self.FileObject = FileObject
self.encryptedConnection = False
2012-09-12 02:11:58 +02:00
self.kill = False
2012-12-25 21:38:02 +01:00
def enableEncryption(self):
#Create an AES cipher from the previously obtained public key
self.cipher = AES.new(self.connection.sharedSecret, AES.MODE_CFB, IV=self.connection.sharedSecret)
self.decipher = AES.new(self.connection.sharedSecret, AES.MODE_CFB, IV=self.connection.sharedSecret)
2012-12-25 21:38:02 +01:00
self.rawsocket = self.socket
self.connection.rawsocket = self.connection.socket
self.socket = EncryptedSocketObjectHandler(self.rawsocket, self.cipher)
self.connection.socket = self.socket
2012-12-25 21:38:02 +01:00
self.rawFileObject = self.FileObject
self.connection.rawFileObject = self.connection.FileObject
self.FileObject = EncryptedFileObjectHandler(self.rawFileObject, self.decipher)
self.connection.FileObject = self.FileObject
2012-12-25 21:38:02 +01:00
self.encryptedConnection = True
2012-12-25 21:38:02 +01:00
2012-04-10 21:14:32 +02:00
def run(self):
while True:
2012-09-12 02:11:58 +02:00
if (self.kill):
break
2012-04-10 21:14:32 +02:00
try:
response = self.FileObject.read(1)
if (response == ""):
continue
2012-07-17 23:42:42 +02:00
except Exception, e:
print "Ping timeout"
sys.exit()
2012-04-10 21:14:32 +02:00
break
2012-12-25 21:38:02 +01:00
if (response == "\x00"):
packet = PacketListenerManager.handle00(self.FileObject, self.socket)
2012-12-25 21:38:02 +01:00
elif (response == "\x01"):
packet = PacketListenerManager.handle01(self.FileObject)
print "Logged in \o/ Received an entity id of " + str(packet['EntityID'])
2012-12-25 21:38:02 +01:00
elif (response == "\x03"):
packet = PacketListenerManager.handle03(self.FileObject)
if not self.connection.options.disableAnsiColours:
filtered_string = Utils.translate_escapes(packet['Message'])
else:
filtered_string = packet['Message']
print filtered_string
2012-12-25 21:38:02 +01:00
elif (response == "\x04"):
packet = PacketListenerManager.handle04(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x05"):
packet = PacketListenerManager.handle05(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x06"):
packet = PacketListenerManager.handle06(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x07"):
packet = PacketListenerManager.handle07(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x08"):
packet = PacketListenerManager.handle08(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x09"):
packet = PacketListenerManager.handle09(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x0D"):
packet = PacketListenerManager.handle0D(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x10"):
2012-12-20 23:28:38 +01:00
packet = PacketListenerManager.handle10(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x11"):
packet = PacketListenerManager.handle11(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x12"):
packet = PacketListenerManager.handle12(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x14"):
packet = PacketListenerManager.handle14(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x15"):
packet = PacketListenerManager.handle15(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x16"):
packet = PacketListenerManager.handle16(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x17"):
packet = PacketListenerManager.handle17(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x18"):
packet = PacketListenerManager.handle18(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x19"):
packet = PacketListenerManager.handle19(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x1A"):
packet = PacketListenerManager.handle1A(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x1C"):
packet = PacketListenerManager.handle1C(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x1D"):
packet = PacketListenerManager.handle1D(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x1E"):
packet = PacketListenerManager.handle1E(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x1F"):
packet = PacketListenerManager.handle1F(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x20"):
packet = PacketListenerManager.handle20(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x21"):
packet = PacketListenerManager.handle21(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x22"):
packet = PacketListenerManager.handle22(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x23"):
packet = PacketListenerManager.handle23(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x26"):
packet = PacketListenerManager.handle26(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x27"):
packet = PacketListenerManager.handle27(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x28"):
packet = PacketListenerManager.handle28(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x29"):
packet = PacketListenerManager.handle29(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x2A"):
packet = PacketListenerManager.handle2A(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x2B"):
packet = PacketListenerManager.handle2B(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x33"):
packet = PacketListenerManager.handle33(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x34"):
packet = PacketListenerManager.handle34(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x35"):
packet = PacketListenerManager.handle35(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x36"):
packet = PacketListenerManager.handle36(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x37"):
packet = PacketListenerManager.handle37(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x38"):
packet = PacketListenerManager.handle38(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x3C"):
packet = PacketListenerManager.handle3C(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x3D"):
packet = PacketListenerManager.handle3D(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x3E"):
packet = PacketListenerManager.handle3E(self.FileObject)
2013-03-16 23:07:41 +01:00
elif (response == "\x3F"):
packet = PacketListenerManager.handle3F(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x46"):
packet = PacketListenerManager.handle46(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x47"):
packet = PacketListenerManager.handle47(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x64"):
packet = PacketListenerManager.handle64(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x65"):
packet = PacketListenerManager.handle65(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x67"):
packet = PacketListenerManager.handle67(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x68"):
packet = PacketListenerManager.handle68(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x69"):
packet = PacketListenerManager.handle69(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x6A"):
packet = PacketListenerManager.handle6A(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x6B"):
packet = PacketListenerManager.handle6B(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x82"):
packet = PacketListenerManager.handle82(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x83"):
packet = PacketListenerManager.handle83(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\x84"):
packet = PacketListenerManager.handle84(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xC8"):
packet = PacketListenerManager.handleC8(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xC9"):
packet = PacketListenerManager.handleC9(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xCA"):
packet = PacketListenerManager.handleCA(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xCB"):
packet = PacketListenerManager.handleCB(self.FileObject)
2013-03-16 23:07:41 +01:00
elif (response == "\xCE"):
packet = PacketListenerManager.handleCE(self.FileObject)
elif (response == "\xCF"):
packet = PacketListenerManager.handleCF(self.FileObject)
elif (response == "\xD0"):
packet = PacketListenerManager.handleD0(self.FileObject)
elif (response == "\xD1"):
packet = PacketListenerManager.handleD1(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xFA"):
packet = PacketListenerManager.handleFA(self.FileObject)
2012-12-25 21:38:02 +01:00
elif (response == "\xFC"):
packet = PacketListenerManager.handleFC(self.FileObject)
if (not self.encryptedConnection):
self.enableEncryption()
self.connection.isConnected = True
PacketSenderManager.sendCD(self.socket, 0)
2012-12-25 21:38:02 +01:00
elif (response == "\xFF"):
packet = PacketListenerManager.handleFF(self.FileObject)
print "Disconnected: " + packet['Reason']
self.connection.disconnect()
self.connection.pluginLoader.disablePlugins()
sys.exit(1)
2012-04-14 17:29:04 +02:00
break
else:
print "Protocol error: " + hex(ord(response))
self.connection.disconnect("Protocol error, invalid packet: " + hex(ord(response)))
self.connection.pluginLoader.disablePlugins()
sys.exit(1)
break
2012-12-25 21:38:02 +01:00
# Invoke plugin listeners
for listener in self.connection.pluginLoader.getPacketListeners():
listener(response, packet)