2015-06-22 19:23:38 +00:00
import configparser
2015-06-14 10:16:57 +00:00
import os
2015-06-08 19:22:58 +00:00
import sqlite3
import tkinter . filedialog as filedialog
import xml . etree . ElementTree as ET
import zipfile
from collections import OrderedDict
from ctypes import c_float , c_int , c_int64 , c_ubyte , c_uint , c_ushort
2015-08-09 11:05:21 +00:00
from tkinter import BooleanVar , END , Menu
2015-06-08 19:22:58 +00:00
2015-08-09 11:05:21 +00:00
import viewer
2015-06-08 19:22:58 +00:00
import structparser
from pyraknet . bitstream import BitStream , c_bit
with open ( " packetdefinitions/replica/creation_header.structs " , encoding = " utf-8 " ) as file :
creation_header_parser = structparser . StructParser ( file . read ( ) )
with open ( " packetdefinitions/replica/serialization_header.structs " , encoding = " utf-8 " ) as file :
serialization_header_parser = structparser . StructParser ( file . read ( ) )
component_name = OrderedDict ( )
component_name [ 1 ] = " ControllablePhysics "
component_name [ 3 ] = " SimplePhysics "
component_name [ 40 ] = " PhantomPhysics "
component_name [ 7 ] = " Destructible "
component_name [ 49 ] = " Switch "
component_name [ 26 ] = " Pet "
component_name [ 4 ] = " Character "
component_name [ 17 ] = " Inventory "
component_name [ 5 ] = " Script "
component_name [ 9 ] = " Skill "
component_name [ 60 ] = " BaseCombatAI "
component_name [ 16 ] = " Vendor "
component_name [ 6 ] = " Bouncer "
component_name [ 39 ] = " ScriptedActivity "
2015-07-31 20:16:48 +00:00
component_name [ 12 ] = None
2015-06-08 19:22:58 +00:00
component_name [ 2 ] = " Render "
component_name [ 107 ] = " Index36 "
component_name [ 31 ] = None
component_name [ 35 ] = None
component_name [ 56 ] = None
component_name [ 64 ] = None
component_name [ 73 ] = None
2015-07-31 20:16:48 +00:00
component_name [ 114 ] = None
2015-06-08 19:22:58 +00:00
comp_ids = list ( component_name . keys ( ) )
2015-06-14 10:16:57 +00:00
comp_parser = { }
2015-06-08 19:22:58 +00:00
for key , value in component_name . items ( ) :
if value is not None :
with open ( " packetdefinitions/replica/components/ " + value + " .structs " ) as file :
2015-06-14 10:16:57 +00:00
comp_parser [ key ] = structparser . StructParser ( file . read ( ) )
norm_parser = { }
for rootdir , _ , files in os . walk ( " packetdefinitions " ) :
for filename in files :
with open ( rootdir + " / " + filename ) as file :
norm_parser [ filename [ : filename . rindex ( " . " ) ] ] = structparser . StructParser ( file . read ( ) )
break
class ParserOutput :
def __init__ ( self ) :
self . text = " "
2015-08-10 10:09:54 +00:00
self . tags = [ ]
2015-06-14 10:16:57 +00:00
def __enter__ ( self ) :
pass
2015-08-10 10:09:54 +00:00
def __exit__ ( self , exc_type , exc_value , tb ) :
2015-06-14 10:16:57 +00:00
if exc_type is not None :
if exc_type == AssertionError :
exc_name = " ASSERTION FAILED "
2015-08-10 10:09:54 +00:00
self . tags . append ( " assertfail " )
2015-06-14 10:16:57 +00:00
elif exc_type == IndexError :
exc_name = " READ ERROR "
2015-08-10 10:09:54 +00:00
self . tags . append ( " readerror " )
else :
exc_name = " ERROR "
self . tags . append ( " error " )
import traceback
traceback . print_tb ( tb )
2015-06-14 10:16:57 +00:00
self . text = exc_name + " " + str ( exc_value ) + " \n " + self . text
return True
def append ( self , structs ) :
for level , description , value , unexpected in structs :
if unexpected :
self . text + = " UNEXPECTED: "
2015-08-10 10:09:54 +00:00
self . tags . append ( " unexpected " )
2015-06-14 10:16:57 +00:00
self . text + = " \t " * level + description + " : " + str ( value ) + " \n "
2015-06-08 19:22:58 +00:00
class CaptureObject :
2015-06-14 10:16:57 +00:00
def __init__ ( self , network_id = None , object_id = None , lot = None ) :
2015-06-08 19:22:58 +00:00
self . network_id = network_id
self . object_id = object_id
2015-06-14 10:16:57 +00:00
self . lot = lot
2015-06-08 19:22:58 +00:00
self . entry = None
2015-08-09 11:05:21 +00:00
class CaptureViewer ( viewer . Viewer ) :
2015-06-22 19:23:38 +00:00
def __init__ ( self ) :
super ( ) . __init__ ( )
config = configparser . ConfigParser ( )
2015-08-09 11:05:21 +00:00
config . read ( " captureviewer.ini " )
self . db = sqlite3 . connect ( config [ " paths " ] [ " db_path " ] )
2015-08-09 19:16:23 +00:00
self . enable_game_messages = " gamemessages_path " in config [ " paths " ]
if self . enable_game_messages :
gamemsg_xml = ET . parse ( config [ " paths " ] [ " gamemessages_path " ] )
self . gamemsgs = gamemsg_xml . findall ( " message " )
self . gamemsg_global_enums = { }
for enum in gamemsg_xml . findall ( " enum " ) :
self . gamemsg_global_enums [ enum . get ( " name " ) ] = tuple ( value . get ( " name " ) for value in enum . findall ( " value " ) )
2015-06-08 19:22:58 +00:00
self . objects = [ ]
self . lot_data = { }
2015-06-22 19:23:38 +00:00
self . parse_creations = BooleanVar ( value = config [ " parse " ] . get ( " creations " , True ) )
self . parse_serializations = BooleanVar ( value = config [ " parse " ] . get ( " serializations " , True ) )
2015-08-09 19:16:23 +00:00
if self . enable_game_messages :
self . parse_game_messages = BooleanVar ( value = config [ " parse " ] . get ( " game_messages " , True ) )
else :
self . parse_game_messages = BooleanVar ( value = False )
2015-06-22 19:23:38 +00:00
self . parse_normal_packets = BooleanVar ( value = config [ " parse " ] . get ( " normal_packets " , True ) )
2015-06-08 19:22:58 +00:00
self . create_widgets ( )
def create_widgets ( self ) :
2015-08-09 11:05:21 +00:00
super ( ) . create_widgets ( )
2015-06-08 19:22:58 +00:00
menubar = Menu ( )
menubar . add_command ( label = " Open " , command = self . askopenfiles )
parse_menu = Menu ( menubar )
parse_menu . add_checkbutton ( label = " Parse Creations " , variable = self . parse_creations )
parse_menu . add_checkbutton ( label = " Parse Serializations " , variable = self . parse_serializations )
2015-08-09 19:16:23 +00:00
if self . enable_game_messages :
parse_menu . add_checkbutton ( label = " Parse Game Messages " , variable = self . parse_game_messages )
2015-06-14 10:16:57 +00:00
parse_menu . add_checkbutton ( label = " Parse Normal Packets " , variable = self . parse_normal_packets )
2015-06-08 19:22:58 +00:00
menubar . add_cascade ( label = " Parse " , menu = parse_menu )
self . master . config ( menu = menubar )
2015-08-10 10:09:54 +00:00
columns = " id " ,
2015-08-09 11:05:21 +00:00
self . tree . configure ( columns = columns )
2015-06-08 19:22:58 +00:00
for col in columns :
self . tree . heading ( col , text = col , command = ( lambda col : lambda : self . sort_column ( col , False ) ) ( col ) )
2015-06-14 10:16:57 +00:00
self . tree . tag_configure ( " unexpected " , foreground = " medium blue " )
self . tree . tag_configure ( " assertfail " , foreground = " orange " )
self . tree . tag_configure ( " readerror " , background = " medium purple " )
self . tree . tag_configure ( " error " , foreground = " red " )
2015-06-08 19:22:58 +00:00
def askopenfiles ( self ) :
2015-08-09 11:05:21 +00:00
paths = filedialog . askopenfilenames ( filetypes = [ ( " Zip " , " *.zip " ) ] )
if paths :
self . load_captures ( paths )
2015-06-08 19:22:58 +00:00
def load_captures ( self , captures ) :
self . tree . set_children ( " " )
self . objects = [ ]
print ( " Loading captures, this might take a while " )
for capture in captures :
print ( " Loading " , capture )
with zipfile . ZipFile ( capture ) as capture :
files = [ i for i in capture . namelist ( ) if " of " not in i ]
if self . parse_creations . get ( ) :
print ( " Parsing creations " )
creations = [ i for i in files if " [24] " in i ]
for packet_name in creations :
packet = BitStream ( capture . read ( packet_name ) )
2015-06-14 10:16:57 +00:00
self . parse_creation ( packet_name , packet )
2015-06-08 19:22:58 +00:00
if self . parse_serializations . get ( ) :
print ( " Parsing serializations " )
serializations = [ i for i in files if " [27] " in i ]
for packet_name in serializations :
packet = BitStream ( capture . read ( packet_name ) [ 1 : ] )
2015-06-14 10:16:57 +00:00
self . parse_serialization_packet ( packet_name , packet )
2015-06-08 19:22:58 +00:00
if self . parse_game_messages . get ( ) :
print ( " Parsing game messages " )
2015-06-14 10:16:57 +00:00
game_messages = [ i for i in files if " [53-05-00-0c] " in i or " [53-04-00-05] " in i ]
2015-06-08 19:22:58 +00:00
for packet_name in game_messages :
packet = BitStream ( capture . read ( packet_name ) [ 8 : ] )
self . parse_game_message ( packet_name , packet )
2015-06-14 10:16:57 +00:00
if self . parse_normal_packets . get ( ) :
print ( " Parsing normal packets " )
packets = [ i for i in files if " [24] " not in i and " [27] " not in i and " [53-05-00-0c] " not in i and " [53-04-00-05] " not in i ]
for packet_name in packets :
packet = BitStream ( capture . read ( packet_name ) )
self . parse_normal_packet ( packet_name , packet )
def parse_creation ( self , packet_name , packet ) :
2015-06-08 19:22:58 +00:00
packet . skip_read ( 1 )
has_network_id = packet . read ( c_bit )
assert has_network_id
network_id = packet . read ( c_ushort )
object_id = packet . read ( c_int64 )
for obj in self . objects :
if obj . object_id == object_id : # We've already parsed this object (can happen due to ghosting)
return
2015-06-14 10:16:57 +00:00
lot = packet . read ( c_int )
if lot not in self . lot_data :
try :
2015-08-09 11:05:21 +00:00
lot_name = self . db . execute ( " select name from Objects where id == " + str ( lot ) ) . fetchone ( ) [ 0 ]
2015-06-14 10:16:57 +00:00
except TypeError :
print ( " Name for lot " , lot , " not found " )
lot_name = str ( lot )
2015-08-09 11:05:21 +00:00
component_types = [ i [ 0 ] for i in self . db . execute ( " select component_type from ComponentsRegistry where id == " + str ( lot ) ) . fetchall ( ) ]
2015-06-14 10:16:57 +00:00
parsers = [ ]
try :
component_types . sort ( key = comp_ids . index )
for comp_type in component_types :
if component_name [ comp_type ] is not None :
parsers . append ( ( component_name [ comp_type ] , comp_parser [ comp_type ] ) )
except ValueError as e :
error = " ERROR: Unknown component " + str ( e . args [ 0 ] . split ( ) [ 0 ] ) + " " + str ( component_types )
else :
error = None
self . lot_data [ lot ] = lot_name , parsers , error
else :
lot_name , parsers , error = self . lot_data [ lot ]
2015-06-08 19:22:58 +00:00
id_ = packet . read ( str , length_type = c_ubyte ) + " " + lot_name
packet . _read_offset = 0
2015-06-14 10:16:57 +00:00
parser_output = ParserOutput ( )
with parser_output :
parser_output . append ( creation_header_parser . parse ( packet ) )
if error is not None :
parser_output . text = error + " \n " + parser_output . text
2015-08-10 10:09:54 +00:00
parser_output . tags . append ( " error " )
2015-06-08 19:22:58 +00:00
else :
2015-06-14 10:16:57 +00:00
self . parse_serialization ( packet , parser_output , parsers , is_creation = True )
2015-06-08 19:22:58 +00:00
2015-06-14 10:16:57 +00:00
obj = CaptureObject ( network_id = network_id , object_id = object_id , lot = lot )
self . objects . append ( obj )
2015-08-10 10:09:54 +00:00
obj . entry = self . tree . insert ( " " , END , text = packet_name , values = ( id_ , parser_output . text ) , tags = parser_output . tags )
2015-06-14 10:16:57 +00:00
@staticmethod
def parse_serialization ( packet , parser_output , parsers , is_creation = False ) :
parser_output . append ( serialization_header_parser . parse ( packet ) )
for name , parser in parsers :
parser_output . text + = " \n " + name + " \n \n "
parser_output . append ( parser . parse ( packet , { " creation " : is_creation } ) )
if not packet . all_read ( ) :
raise IndexError ( " Not completely read " )
def parse_serialization_packet ( self , packet_name , packet ) :
2015-06-08 19:22:58 +00:00
network_id = packet . read ( c_ushort )
obj = None
for j in self . objects :
if j . network_id == network_id :
obj = j
break
if obj is None :
obj = CaptureObject ( network_id = network_id )
self . objects . append ( obj )
2015-08-09 11:05:21 +00:00
obj . entry = self . tree . insert ( " " , END , text = " Unknown " , values = ( " network_id= " + str ( network_id ) , " " ) )
2015-06-08 19:22:58 +00:00
2015-06-14 10:16:57 +00:00
if obj . lot is None :
parsers = [ ]
error = " Unknown object "
else :
_ , parsers , error = self . lot_data [ obj . lot ]
parser_output = ParserOutput ( )
with parser_output :
self . parse_serialization ( packet , parser_output , parsers )
if error is not None :
2015-08-10 10:09:54 +00:00
parser_output . tags . append ( " error " )
2015-06-14 10:16:57 +00:00
else :
error = " "
2015-08-10 10:09:54 +00:00
self . tree . insert ( obj . entry , END , text = packet_name , values = ( error , parser_output . text ) , tags = parser_output . tags )
2015-06-08 19:22:58 +00:00
def parse_game_message ( self , packet_name , packet ) :
object_id = packet . read ( c_int64 )
for i in self . objects :
if i . object_id == object_id :
entry = i . entry
break
else :
obj = CaptureObject ( object_id = object_id )
self . objects . append ( obj )
2015-08-09 11:05:21 +00:00
obj . entry = entry = self . tree . insert ( " " , END , text = " Unknown " , values = ( " object_id= " + str ( object_id ) , " " ) )
2015-06-08 19:22:58 +00:00
msg_id = packet . read ( c_ushort )
2015-06-22 19:23:38 +00:00
if msg_id < = 0x80 :
2015-06-08 19:22:58 +00:00
msg_id - = 1
elif msg_id < = 0xf9 :
msg_id - = 2
2015-08-09 11:05:21 +00:00
elif msg_id < = 0x1c0 :
2015-06-08 19:22:58 +00:00
msg_id + = 1
elif msg_id < = 0x1fd :
msg_id - = 1
elif msg_id < = 0x208 :
msg_id - = 5
elif msg_id < = 0x231 :
msg_id - = 8
elif msg_id < = 0x30d :
msg_id - = 10
elif msg_id < = 0x353 :
msg_id - = 9
elif msg_id < = 0x37a :
msg_id - = 10
elif msg_id < = 0x3a6 :
msg_id - = 9
elif msg_id < = 0x430 :
msg_id - = 33
elif msg_id < = 0x4c7 :
msg_id - = 34
elif msg_id < = 0x510 :
msg_id - = 31
elif msg_id < = 0x58b :
msg_id - = 30
elif msg_id < = 0x5e7 :
msg_id - = 29
try :
message = self . gamemsgs [ msg_id ]
msg_name = message . get ( " name " )
network = message . get ( " network " )
if network is None or ( ( ( " [53-05-00-0c] " in packet_name and " client " not in network ) or ( " [53-04-00-05] " in packet_name and " server " not in network ) ) and network != " duplicated " ) :
raise ValueError
attrs = message . findall ( " attr " )
attrs . sort ( key = lambda x : x . get ( " name " ) )
2015-06-14 10:16:57 +00:00
attr_values = OrderedDict ( )
2015-06-08 19:22:58 +00:00
if message . find ( " freeze " ) is not None or message . find ( " thaw " ) is not None :
# Custom serializations
if msg_name == " NotifyMissionTask " :
2015-06-14 10:16:57 +00:00
attr_values [ " missionID " ] = packet . read ( c_int )
attr_values [ " taskMask " ] = packet . read ( c_int )
2015-06-08 19:22:58 +00:00
updates = [ ]
for _ in range ( packet . read ( c_ubyte ) ) :
updates . append ( packet . read ( c_float ) )
2015-06-14 10:16:57 +00:00
attr_values [ " updates " ] = updates
2015-08-09 11:05:21 +00:00
elif msg_name == " VendorStatusUpdate " :
attr_values [ " bUpdateOnly " ] = packet . read ( c_bit )
inv = { }
for _ in range ( packet . read ( c_uint ) ) :
inv [ packet . read ( c_int ) ] = packet . read ( c_int )
attr_values [ " inventoryList " ] = inv
2015-06-08 19:22:58 +00:00
elif msg_name == " RequestLinkedMission " :
2015-06-14 10:16:57 +00:00
attr_values [ " playerID " ] = packet . read ( c_int64 )
attr_values [ " missionID " ] = packet . read ( c_int )
attr_values [ " bMissionOffered " ] = packet . read ( c_bit )
2015-07-31 20:16:48 +00:00
elif msg_name == " ModularBuildFinish " :
lots = [ ]
for _ in range ( packet . read ( c_ubyte ) ) :
lots . append ( packet . read ( c_int ) )
attr_values [ " moduleTemplateIDs " ] = lots
2015-06-08 19:22:58 +00:00
else :
raise NotImplementedError ( " Custom serialization " )
2015-06-14 10:16:57 +00:00
values = " \n " . join ( [ " %s = %s " % ( a , b ) for a , b in attr_values . items ( ) ] )
2015-08-10 10:09:54 +00:00
tags = [ ]
2015-06-08 19:22:58 +00:00
else :
local_enums = { }
for enum in message . findall ( " enum " ) :
local_enums [ enum . get ( " name " ) ] = tuple ( value . get ( " name " ) for value in enum . findall ( " value " ) )
for attr in attrs :
if attr . get ( " returnValue " ) is not None :
raise NotImplementedError ( attr . get ( " name " ) , " returnValue " )
type_ = attr . get ( " type " )
default = attr . get ( " default " )
if type_ == " bool " : # bools don't have default-flags
2015-06-14 10:16:57 +00:00
attr_values [ attr . get ( " name " ) ] = packet . read ( c_bit )
2015-06-08 19:22:58 +00:00
continue
if default is not None :
is_not_default = packet . read ( c_bit )
if not is_not_default :
2015-06-14 10:16:57 +00:00
attr_values [ attr . get ( " name " ) ] = default
2015-06-08 19:22:58 +00:00
continue
if type_ == " unsigned char " :
value = packet . read ( c_ubyte )
elif type_ == " LWOMAPID " :
value = packet . read ( c_ushort )
elif type_ in ( " int " , " LOT " ) :
value = packet . read ( c_int )
elif type_ in ( " unsigned int " , " TSkillID " ) :
value = packet . read ( c_uint )
elif type_ == " __int64 " :
value = packet . read ( c_int64 )
elif type_ == " LWOOBJID " :
value = packet . read ( c_int64 )
2015-06-14 10:16:57 +00:00
if value == object_id :
value = str ( value ) + " <self> "
else :
for obj in self . objects :
if value == obj . object_id :
value = str ( value ) + " < " + self . tree . item ( obj . entry , " values " ) [ 0 ] + " > "
break
2015-06-08 19:22:58 +00:00
elif type_ == " float " :
value = packet . read ( c_float )
elif type_ == " std::string " :
length = packet . read ( c_uint )
if length > 255 : # in case this isn't the right message after all and we read a way too high value
raise ValueError
value = packet . read ( str , char_size = 1 , allocated_length = length )
elif type_ == " std::wstring " :
length = packet . read ( c_uint )
if length > 255 : # in case this isn't the right message after all and we read a way too high value
raise ValueError
value = packet . read ( str , char_size = 2 , allocated_length = length * 2 )
elif type_ == " NiPoint3 " :
value = packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float )
elif type_ == " NiQuaternion " :
value = packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float )
elif type_ == " LwoNameValue " :
value = packet . read ( str , length_type = c_uint )
2015-08-09 11:05:21 +00:00
if value :
2015-06-08 19:22:58 +00:00
assert packet . read ( c_ushort ) == 0 # for some reason has a null terminator
elif type_ in local_enums :
value = packet . read ( c_uint )
value = local_enums [ type_ ] [ value ] + " ( " + str ( value ) + " ) "
elif type_ in self . gamemsg_global_enums :
value = packet . read ( c_uint )
value = self . gamemsg_global_enums [ type_ ] [ value ] + " ( " + str ( value ) + " ) "
else :
raise NotImplementedError ( type_ )
2015-06-14 10:16:57 +00:00
attr_values [ attr . get ( " name " ) ] = value
2015-06-08 19:22:58 +00:00
if not packet . all_read ( ) :
raise ValueError
except NotImplementedError as e :
2015-06-14 10:16:57 +00:00
values = ( msg_name , str ( e ) + " \n len: " + str ( len ( packet ) - 10 ) + " \n " + " \n " . join ( [ " %s = %s " % ( a , b ) for a , b in attr_values . items ( ) ] ) )
2015-08-10 10:09:54 +00:00
tags = [ " error " ]
2015-06-08 19:22:58 +00:00
except ( IndexError , UnicodeDecodeError ) as e :
print ( packet_name , msg_name )
import traceback
traceback . print_exc ( )
2015-06-14 10:16:57 +00:00
values = ( " likely not " + msg_name , " Error while parsing, likely not this message! \n " + str ( e ) + " \n len: " + str ( len ( packet ) - 10 ) + " \n " + " \n " . join ( [ " %s = %s " % ( a , b ) for a , b in attr_values . items ( ) ] ) )
2015-08-10 10:09:54 +00:00
tags = [ " error " ]
2015-06-14 10:16:57 +00:00
except ValueError as e :
values = ( " likely not " + msg_name , " Error while parsing, likely not this message! \n " + str ( e ) + " \n len: " + str ( len ( packet ) - 10 ) )
2015-08-10 10:09:54 +00:00
tags = [ " error " ]
2015-06-08 19:22:58 +00:00
else :
2015-06-14 10:16:57 +00:00
values = ( msg_name , " \n " . join ( [ " %s = %s " % ( a , b ) for a , b in attr_values . items ( ) ] ) )
2015-08-10 10:09:54 +00:00
tags = [ ]
self . tree . insert ( entry , END , text = packet_name , values = values , tags = tags )
2015-06-08 19:22:58 +00:00
2015-06-14 10:16:57 +00:00
def parse_normal_packet ( self , packet_name , packet ) :
id_ = packet_name [ packet_name . index ( " [ " ) + 1 : packet_name . index ( " ] " ) ]
if id_ not in norm_parser :
2015-08-10 10:09:54 +00:00
self . tree . insert ( " " , END , text = packet_name , values = ( id_ , " Add the struct definition file packetdefinitions/ " + id_ + " .structs to enable parsing of this packet. " ) , tags = [ " error " ] )
2015-06-14 10:16:57 +00:00
return
if id_ . startswith ( " 53 " ) :
packet . skip_read ( 8 )
else :
packet . skip_read ( 1 )
parser_output = ParserOutput ( )
parser_output . append ( norm_parser [ id_ ] . parse ( packet ) )
2015-08-10 10:09:54 +00:00
self . tree . insert ( " " , END , text = packet_name , values = ( id_ , parser_output . text ) , tags = parser_output . tags )
2015-08-09 11:05:21 +00:00
def on_item_select ( self , event ) :
2015-06-08 19:22:58 +00:00
item = self . tree . selection ( ) [ 0 ]
self . item_inspector . delete ( 1.0 , END )
self . item_inspector . insert ( END , self . tree . item ( item , " values " ) [ 1 ] )
2015-06-14 10:16:57 +00:00
if __name__ == " __main__ " :
2015-08-09 11:05:21 +00:00
app = CaptureViewer ( )
app . mainloop ( )