2015-06-22 19:23:38 +00:00
import configparser
2015-10-12 17:33:34 +00:00
import math
2015-06-14 10:16:57 +00:00
import os
2016-07-23 09:39:00 +00:00
import pprint
2017-02-09 09:30:51 +00:00
import struct
2015-06-08 19:22:58 +00:00
import sqlite3
2015-11-19 20:25:16 +00:00
import sys
2015-06-08 19:22:58 +00:00
import tkinter . filedialog as filedialog
2015-11-19 20:25:16 +00:00
import tkinter . messagebox as messagebox
2015-06-08 19:22:58 +00:00
import xml . etree . ElementTree as ET
import zipfile
2016-07-23 09:39:29 +00:00
import zlib
2015-06-08 19:22:58 +00:00
from collections import OrderedDict
2015-08-09 11:05:21 +00:00
from tkinter import BooleanVar , END , Menu
2015-06-08 19:22:58 +00:00
2015-10-22 16:30:15 +00:00
import amf3
2015-06-08 19:22:58 +00:00
import structparser
2015-10-22 16:30:15 +00:00
import viewer
2016-07-23 09:39:29 +00:00
import ldf
from pyraknet . bitstream import BitStream , c_bit , c_bool , c_float , c_int , c_int64 , c_ubyte , c_uint , c_uint64 , c_ushort
def compressed_ldf_handler ( stream ) :
size = stream . read ( c_uint )
is_compressed = stream . read ( c_bool )
if is_compressed :
uncompressed_size = stream . read ( c_uint )
uncompressed = zlib . decompress ( stream . read ( bytes , length = stream . read ( c_uint ) ) )
assert len ( uncompressed ) == uncompressed_size
else :
uncompressed = stream . read ( bytes , length = size )
return ldf . from_ldf ( BitStream ( uncompressed ) )
type_handlers = { }
type_handlers [ " compressed_ldf " ] = compressed_ldf_handler
2015-06-08 19:22:58 +00:00
with open ( " packetdefinitions/replica/creation_header.structs " , encoding = " utf-8 " ) as file :
2016-07-23 09:39:29 +00:00
creation_header_parser = structparser . StructParser ( file . read ( ) , type_handlers )
2015-06-08 19:22:58 +00:00
with open ( " packetdefinitions/replica/serialization_header.structs " , encoding = " utf-8 " ) as file :
2016-07-23 09:39:29 +00:00
serialization_header_parser = structparser . StructParser ( file . read ( ) , type_handlers )
2015-06-08 19:22:58 +00:00
component_name = OrderedDict ( )
2015-12-15 17:46:30 +00:00
component_name [ 108 ] = " Component 108 " ,
component_name [ 61 ] = " ModuleAssembly " ,
component_name [ 1 ] = " ControllablePhysics " ,
component_name [ 3 ] = " SimplePhysics " ,
component_name [ 20 ] = " RigidBodyPhantomPhysics " ,
component_name [ 30 ] = " VehiclePhysics 30 " ,
component_name [ 40 ] = " PhantomPhysics " ,
component_name [ 7 ] = " Destructible " , " Stats "
component_name [ 23 ] = " Stats " , " Collectible "
component_name [ 26 ] = " Pet " ,
component_name [ 4 ] = " Character " ,
component_name [ 17 ] = " Inventory " ,
component_name [ 5 ] = " Script " ,
component_name [ 9 ] = " Skill " ,
2016-10-22 12:02:58 +00:00
component_name [ 11 ] = " Item " ,
2015-12-15 17:46:30 +00:00
component_name [ 60 ] = " BaseCombatAI " ,
component_name [ 48 ] = " Stats " , " Rebuild "
2016-06-21 19:19:50 +00:00
component_name [ 25 ] = " MovingPlatform " ,
2015-12-15 17:46:30 +00:00
component_name [ 49 ] = " Switch " ,
component_name [ 16 ] = " Vendor " ,
component_name [ 6 ] = " Bouncer " ,
component_name [ 39 ] = " ScriptedActivity " ,
component_name [ 71 ] = " RacingControl " ,
2016-10-22 12:02:58 +00:00
component_name [ 75 ] = " Exhibit " ,
2015-12-15 17:46:30 +00:00
component_name [ 2 ] = " Render " ,
component_name [ 107 ] = " Component 107 " ,
2016-10-22 12:04:59 +00:00
component_name [ 69 ] = " Trigger " ,
2015-10-12 17:33:34 +00:00
component_name [ 12 ] = None
2016-06-21 19:19:50 +00:00
component_name [ 27 ] = None
2015-06-08 19:22:58 +00:00
component_name [ 31 ] = None
component_name [ 35 ] = None
2015-10-12 17:33:34 +00:00
component_name [ 36 ] = None
component_name [ 45 ] = None
component_name [ 55 ] = None
2015-06-08 19:22:58 +00:00
component_name [ 56 ] = None
2016-10-22 12:02:58 +00:00
component_name [ 57 ] = None
2015-06-08 19:22:58 +00:00
component_name [ 64 ] = None
2015-10-12 17:33:34 +00:00
component_name [ 65 ] = None
component_name [ 68 ] = None
2015-06-08 19:22:58 +00:00
component_name [ 73 ] = None
2015-12-15 17:46:30 +00:00
component_name [ 104 ] = None
2015-10-12 17:33:34 +00:00
component_name [ 113 ] = 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-12-15 17:46:30 +00:00
for comp_id , indices in component_name . items ( ) :
if indices is not None :
comp_parser [ comp_id ] = [ ]
for index in indices :
with open ( " packetdefinitions/replica/components/ " + index + " .structs " ) as file :
2016-07-23 09:39:29 +00:00
comp_parser [ comp_id ] . append ( structparser . StructParser ( file . read ( ) , type_handlers ) )
2015-06-14 10:16:57 +00:00
norm_parser = { }
for rootdir , _ , files in os . walk ( " packetdefinitions " ) :
for filename in files :
with open ( rootdir + " / " + filename ) as file :
2016-07-23 09:39:29 +00:00
norm_parser [ filename [ : filename . rindex ( " . " ) ] ] = structparser . StructParser ( file . read ( ) , type_handlers )
2015-06-14 10:16:57 +00:00
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 )
2016-07-23 09:39:29 +00:00
self . text = exc_name + " " + str ( exc_type . __name__ ) + " : " + str ( exc_value ) + " \n " + self . text
2015-06-14 10:16:57 +00:00
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 " )
2015-11-19 20:25:16 +00:00
try :
self . db = sqlite3 . connect ( config [ " paths " ] [ " db_path " ] )
except :
messagebox . showerror ( " Can not open database " , " Make sure db_path in the INI is set correctly. " )
sys . exit ( )
2017-02-09 09:30:51 +00:00
gamemsg_xml = ET . parse ( config [ " paths " ] [ " gamemessages_path " ] )
self . gamemsgs = { }
for msg in gamemsg_xml . findall ( " message " ) :
self . gamemsgs [ int ( msg . get ( " id " ) ) ] = msg
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-11-19 20:25:16 +00:00
self . parse_creations = BooleanVar ( value = config [ " parse " ] [ " creations " ] )
self . parse_serializations = BooleanVar ( value = config [ " parse " ] [ " serializations " ] )
2017-02-09 09:30:51 +00:00
self . parse_game_messages = BooleanVar ( value = config [ " parse " ] [ " game_messages " ] )
2015-11-19 20:25:16 +00:00
self . parse_normal_packets = BooleanVar ( value = config [ " parse " ] [ " normal_packets " ] )
2016-08-30 08:32:09 +00:00
self . retry_with_script_component = BooleanVar ( value = config [ " parse " ] [ " retry_with_script_component " ] )
2016-10-22 12:04:59 +00:00
self . retry_with_trigger_component = BooleanVar ( value = config [ " parse " ] [ " retry_with_trigger_component " ] )
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 )
2017-02-09 09:30:51 +00:00
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 )
2016-08-30 08:32:09 +00:00
parse_menu . add_checkbutton ( label = " Retry parsing with script component if failed " , variable = self . retry_with_script_component )
2016-10-22 12:04:59 +00:00
parse_menu . add_checkbutton ( label = " Retry parsing with trigger component if failed " , variable = self . retry_with_trigger_component )
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 ( " " )
2015-12-15 17:46:30 +00:00
self . detached_items . clear ( )
2015-06-08 19:22:58 +00:00
self . objects = [ ]
print ( " Loading captures, this might take a while " )
2015-12-15 17:46:30 +00:00
for i , capture in enumerate ( captures ) :
print ( " Loading " , capture , " [ %i / %i ] " % ( i + 1 , len ( captures ) ) )
2015-06-08 19:22:58 +00:00
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 )
2016-10-22 12:04:59 +00:00
def parse_creation ( self , packet_name , packet , retry_with_components = [ ] ) :
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 ( ) ]
2016-10-22 12:04:59 +00:00
component_types . extend ( retry_with_components )
2015-12-15 17:46:30 +00:00
parsers = OrderedDict ( )
2015-06-14 10:16:57 +00:00
try :
component_types . sort ( key = comp_ids . index )
for comp_type in component_types :
if component_name [ comp_type ] is not None :
2015-12-15 17:46:30 +00:00
for name , parser in zip ( component_name [ comp_type ] , comp_parser [ comp_type ] ) :
if name not in parsers :
parsers [ name ] = parser
2015-06-14 10:16:57 +00:00
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 :
2016-08-30 08:32:09 +00:00
try :
self . parse_serialization ( packet , parser_output , parsers , is_creation = True )
2017-02-09 09:30:51 +00:00
except ( AssertionError , IndexError , struct . error ) :
2016-10-22 12:04:59 +00:00
if retry_with_components :
print ( " retry was not able to resolve parsing error " )
raise
retry_with_components = [ ]
if self . retry_with_script_component . get ( ) :
retry_with_components . append ( 5 )
elif self . retry_with_trigger_component . get ( ) :
retry_with_components . append ( 69 )
if retry_with_components :
print ( " retrying with " , retry_with_components , packet_name )
2016-08-30 08:32:09 +00:00
del self . lot_data [ lot ]
packet . _read_offset = 0
2016-10-22 12:04:59 +00:00
self . parse_creation ( packet_name , packet , retry_with_components )
2016-08-30 08:32:09 +00:00
return
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 )
2016-07-23 09:39:29 +00:00
obj . entry = self . tree . insert ( " " , END , text = packet_name , values = ( id_ , parser_output . text . replace ( " { " , " <crlbrktopen> " ) . replace ( " } " , " <crlbrktclose> " ) . replace ( " \\ " , " <backslash> " ) ) , 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 ) )
2015-12-15 17:46:30 +00:00
for name , parser in parsers . items ( ) :
2015-06-14 10:16:57 +00:00
parser_output . text + = " \n " + name + " \n \n "
parser_output . append ( parser . parse ( packet , { " creation " : is_creation } ) )
if not packet . all_read ( ) :
2015-10-12 17:33:34 +00:00
raise IndexError ( " Not completely read, %i bytes unread " % ( len ( packet ) - math . ceil ( packet . _read_offset / 8 ) ) )
2015-06-14 10:16:57 +00:00
def parse_serialization_packet ( self , packet_name , packet ) :
2015-06-08 19:22:58 +00:00
network_id = packet . read ( c_ushort )
obj = None
2015-12-15 17:46:30 +00:00
for i in self . objects :
if i . network_id == network_id :
obj = i
2015-06-08 19:22:58 +00:00
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 :
2015-12-15 17:46:30 +00:00
parsers = { }
2015-06-14 10:16:57 +00:00
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 = " "
2016-07-23 09:39:29 +00:00
self . tree . insert ( obj . entry , END , text = packet_name , values = ( error , parser_output . text . replace ( " { " , " <crlbrktopen> " ) . replace ( " } " , " <crlbrktclose> " ) . replace ( " \\ " , " <backslash> " ) ) , 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 )
2017-02-09 09:30:51 +00:00
tags = [ ]
2015-06-08 19:22:58 +00:00
try :
message = self . gamemsgs [ msg_id ]
msg_name = message . get ( " name " )
network = message . get ( " network " )
2015-10-12 17:33:34 +00:00
attr_values = OrderedDict ( )
2015-06-08 19:22:58 +00:00
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 " )
2015-10-12 17:33:34 +00:00
if msg_name == " Teleport " :
attrs = [ attr for attr in attrs if attr . get ( " name " ) != " NoGravTeleport " ]
2015-06-08 19:22:58 +00:00
attrs . sort ( key = lambda x : x . get ( " name " ) )
2015-10-12 17:33:34 +00:00
2017-02-09 09:30:51 +00:00
if message . get ( " custom " ) is not None :
2015-06-08 19:22:58 +00:00
# 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 ) )
2017-02-09 09:30:51 +00:00
if len ( updates ) != 1 :
tags . append ( " unexpected " )
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-10-12 17:33:34 +00:00
elif msg_name == " FetchModelMetadataResponse " :
attr_values [ " ugID " ] = packet . read ( c_int64 )
attr_values [ " objectID " ] = packet . read ( c_int64 )
attr_values [ " requestorID " ] = packet . read ( c_int64 )
attr_values [ " context " ] = packet . read ( c_int )
attr_values [ " bHasUGData " ] = packet . read ( c_bit )
attr_values [ " bHasBPData " ] = packet . read ( c_bit )
if attr_values [ " bHasUGData " ] :
attr_values [ " UGM_unknown1 " ] = packet . read ( c_int64 )
attr_values [ " UGM_unknown2 " ] = packet . read ( c_int64 )
attr_values [ " UGM_unknown_str_1 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " UGM_unknown_str_2 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " UGM_unknown3 " ] = packet . read ( c_int64 )
attr_values [ " UGM_unknown4 " ] = packet . read ( c_int64 )
attr_values [ " UGM_unknown_str_3 " ] = packet . read ( str , length_type = c_uint )
unknown_list = [ ]
for _ in range ( packet . read ( c_ubyte ) ) :
unknown_list . append ( packet . read ( c_int64 ) )
attr_values [ " UGM_unknown_list " ] = unknown_list
if attr_values [ " bHasBPData " ] :
attr_values [ " BPM_unknown1 " ] = packet . read ( c_int64 )
attr_values [ " BPM_some_timestamp " ] = packet . read ( c_uint64 )
attr_values [ " BPM_unknown2 " ] = packet . read ( c_uint )
attr_values [ " BPM_unknown3 " ] = packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float )
attr_values [ " BPM_unknown4 " ] = packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float )
attr_values [ " BPM_unknown_bool_1 " ] = packet . read ( c_bit )
attr_values [ " BPM_unknown_bool_2 " ] = packet . read ( c_bit )
attr_values [ " BPM_unknown_str_1 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " BPM_unknown_bool_3 " ] = packet . read ( c_bit )
attr_values [ " BPM_unknown5 " ] = packet . read ( c_uint )
elif msg_name == " NotifyPetTamingPuzzleSelected " :
bricks = [ ]
for _ in range ( packet . read ( c_uint ) ) :
bricks . append ( ( packet . read ( c_uint ) , packet . read ( c_uint ) ) )
attr_values [ " randBrickIDList " ] = bricks
elif msg_name == " DownloadPropertyData " :
attr_values [ " object_id " ] = packet . read ( c_int64 )
attr_values [ " component_id " ] = packet . read ( c_int )
attr_values [ " mapID " ] = packet . read ( c_ushort )
attr_values [ " vendorMapID " ] = packet . read ( c_ushort )
attr_values [ " unknown1 " ] = packet . read ( c_uint )
attr_values [ " property_name " ] = packet . read ( str , length_type = c_uint )
attr_values [ " property_description " ] = packet . read ( str , length_type = c_uint )
attr_values [ " owner_name " ] = packet . read ( str , length_type = c_uint )
attr_values [ " owner_object_id " ] = packet . read ( c_int64 )
attr_values [ " type " ] = packet . read ( c_uint )
attr_values [ " sizecode " ] = packet . read ( c_uint )
attr_values [ " minimumPrice " ] = packet . read ( c_uint )
attr_values [ " rentDuration " ] = packet . read ( c_uint )
attr_values [ " timestamp1 " ] = packet . read ( c_uint64 )
attr_values [ " unknown2 " ] = packet . read ( c_uint )
attr_values [ " unknown3 " ] = packet . read ( c_uint64 )
attr_values [ " spawnName " ] = packet . read ( str , length_type = c_uint )
attr_values [ " unknown_str_1 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " unknown_str_2 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " durationType " ] = packet . read ( c_uint )
attr_values [ " unknown4 " ] = packet . read ( c_uint )
attr_values [ " unknown5 " ] = packet . read ( c_uint )
attr_values [ " unknown6 " ] = packet . read ( c_ubyte )
attr_values [ " unknown7 " ] = packet . read ( c_uint64 )
attr_values [ " unknown8 " ] = packet . read ( c_uint )
attr_values [ " unknown_str_3 " ] = packet . read ( str , length_type = c_uint )
attr_values [ " unknown9 " ] = packet . read ( c_uint64 )
attr_values [ " unknown10 " ] = packet . read ( c_uint )
attr_values [ " unknown11 " ] = packet . read ( c_uint )
attr_values [ " zoneX " ] = packet . read ( c_float )
attr_values [ " zoneY " ] = packet . read ( c_float )
attr_values [ " zoneZ " ] = packet . read ( c_float )
attr_values [ " maxBuildHeight " ] = packet . read ( c_float )
attr_values [ " timestamp2 " ] = packet . read ( c_uint64 )
attr_values [ " unknown12 " ] = packet . read ( c_ubyte )
path = [ ]
for _ in range ( packet . read ( c_uint ) ) :
path . append ( ( packet . read ( c_float ) , packet . read ( c_float ) , packet . read ( c_float ) ) )
attr_values [ " path " ] = path
2016-07-23 09:39:00 +00:00
elif msg_name == " PropertySelectQuery " :
attr_values [ " navOffset " ] = packet . read ( c_int )
attr_values [ " bThereAreMore " ] = packet . read ( c_bit )
attr_values [ " myCloneID " ] = packet . read ( c_int )
attr_values [ " bHasFeaturedProperty " ] = packet . read ( c_bit )
attr_values [ " bWasFriends " ] = packet . read ( c_bit )
properties = [ ]
attr_values [ " properties " ] = properties
for _ in range ( packet . read ( c_uint ) ) :
property = OrderedDict ( )
property [ " cloneID " ] = packet . read ( c_int )
property [ " ownerName " ] = packet . read ( str , length_type = c_uint )
property [ " name " ] = packet . read ( str , length_type = c_uint )
property [ " description " ] = packet . read ( str , length_type = c_uint )
property [ " reputation " ] = packet . read ( c_uint )
property [ " isBff " ] = packet . read ( c_bit )
property [ " isFriend " ] = packet . read ( c_bit )
property [ " isModeratedApproved " ] = packet . read ( c_bit )
property [ " isAlt " ] = packet . read ( c_bit )
property [ " isOwned " ] = packet . read ( c_bit )
property [ " accessType " ] = packet . read ( c_uint )
property [ " dateLastPublished " ] = packet . read ( c_uint )
property [ " performanceCost " ] = packet . read ( c_uint64 )
properties . append ( property )
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-10-12 17:33:34 +00:00
elif msg_name == " PetTamingTryBuild " :
selections = [ ]
for _ in range ( packet . read ( c_uint ) ) :
selections . append ( ( packet . read ( c_uint ) , packet . read ( c_uint ) ) )
attr_values [ " currentSelections " ] = selections
attr_values [ " clientFailed " ] = packet . read ( c_bit )
2016-07-23 09:39:00 +00:00
elif msg_name == " GetModelsOnProperty " :
models = [ ]
for _ in range ( packet . read ( c_uint ) ) :
models . append ( ( packet . read ( c_int64 ) , packet . read ( c_int64 ) ) )
attr_values [ " models " ] = models
elif msg_name == " MatchRequest " :
attr_values [ " activator " ] = packet . read ( c_int64 )
choices = packet . read ( str , length_type = c_uint )
if choices :
assert packet . read ( c_ushort ) == 0 # for some reason has a null terminator
attr_values [ " playerChoices " ] = choices
attr_values [ " type " ] = packet . read ( c_int )
attr_values [ " value " ] = packet . read ( c_int )
elif msg_name == " TeamCreateLocal " :
team_members = [ ]
for _ in range ( packet . read ( c_uint ) ) :
team_members . append ( ( packet . read ( c_int64 ) , packet . read ( c_bit ) ) )
attr_values [ " team_members " ] = team_members
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-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 :
2016-07-23 09:39:00 +00:00
continue
2015-06-08 19:22:58 +00:00
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 )
2015-10-12 17:33:34 +00:00
elif type_ in ( " unsigned int " , " LWOCLONEID " , " TSkillID " ) :
2015-06-08 19:22:58 +00:00
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
2016-07-23 09:39:00 +00:00
elif type_ == " LWOZONEID " :
value = packet . read ( c_ushort ) , packet . read ( c_ushort ) , packet . read ( c_uint )
2015-06-08 19:22:58 +00:00
elif type_ == " float " :
value = packet . read ( c_float )
2015-10-12 17:33:34 +00:00
elif type_ == " BinaryBuffer " :
length = packet . read ( c_uint )
value = packet . read ( bytes , length = length )
2015-06-08 19:22:58 +00:00
elif type_ == " std::string " :
length = packet . read ( c_uint )
value = packet . read ( str , char_size = 1 , allocated_length = length )
elif type_ == " std::wstring " :
length = packet . read ( c_uint )
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
2015-10-22 16:30:15 +00:00
elif type_ == " NDGFxValue " :
value = amf3 . read ( packet )
2015-06-08 19:22:58 +00:00
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 :
2016-07-23 09:39:00 +00:00
raise NotImplementedError ( " Unknown type " , type_ )
2015-06-14 10:16:57 +00:00
attr_values [ attr . get ( " name " ) ] = value
2015-10-12 17:33:34 +00:00
if not packet . all_read ( ) :
raise ValueError
2015-06-08 19:22:58 +00:00
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 ( ) ] ) )
2017-02-09 09:30:51 +00:00
tags . append ( " error " )
2015-10-12 17:33:34 +00:00
except Exception as e :
2015-06-08 19:22:58 +00:00
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 ( ) ] ) )
2017-02-09 09:30:51 +00:00
tags . append ( " error " )
2015-06-08 19:22:58 +00:00
else :
2016-07-23 09:39:00 +00:00
values = ( msg_name , " \n " . join ( [ " %s = %s " % ( a , pprint . pformat ( b ) ) for a , b in attr_values . items ( ) ] ) )
2015-08-10 10:09:54 +00:00
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 ( )
2015-10-12 17:33:34 +00:00
with parser_output :
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
2015-10-12 17:33:34 +00:00
def on_item_select ( self , _ ) :
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 ( )