wireshark/epan/wspython/wspy_dissector.py

461 lines
14 KiB
Python
Executable File

# wspy_dissector.py
#
# $Id$
#
# Wireshark Protocol Python Binding
#
# Copyright (c) 2009 by Sebastien Tandel <sebastien [AT] tandel [dot] be>
# Copyright (c) 2001 by Gerald Combs <gerald@wireshark.org>
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
import ctypes as ct
from wspy_libws import get_libws_handle
# From epan/proto.h
# ? STA ? : is there a better way to include/define these constants?
# (duplicating definition is not a good thing)
(BASE_NONE,
BASE_DEC,
BASE_HEX,
BASE_OCT,
BASE_DEC_HEX,
BASE_HEX_DEC,
BASE_CUSTOM) = map(int, range(7))
# field types, see epan/ftypes/ftypes.h
(FT_NONE,
FT_PROTOCOL,
FT_BOOLEAN,
FT_UINT8,
FT_UINT16,
FT_UINT24,
FT_UINT32,
FT_UINT64,
FT_INT8,
FT_INT16,
FT_INT24,
FT_INT32,
FT_INT64,
FT_FLOAT,
FT_DOUBLE,
FT_ABSOLUTE_TIME,
FT_RELATIVE_TIME,
FT_STRING,
FT_STRINGZ,
FT_EBCDIC,
FT_UINT_STRING,
FT_ETHER,
FT_BYTES,
FT_UINT_BYTES,
FT_IPv4,
FT_IPv6,
FT_IPXNET,
FT_FRAMENUM,
FT_PCRE,
FT_GUID,
FT_OID,
FT_REL_OID) = map(int, range(32))
# hf_register_info from usual dissectors
class register_info(object):
def __init__(self, wsl):
self.__protocol = None
self.__wsl = wsl
self.__hf_register = None
self.__registers = []
def add(self, name, short_desc, \
type=FT_UINT32, display=BASE_DEC, \
strings=None, bitmask=0x0, desc=None):
if not desc:
desc = name
self.__registers.append( (name, short_desc, \
type, display, strings, bitmask, desc) )
def register(self, protocol):
self.__protocol = protocol
hf = self.__registers
lr = len(hf)
if not lr:
return None
self.__hf_register = self.__wsl.hf_register_info_create(lr)
chf = self.__hf_register
if not self.__hf_register:
return None
for i in range(lr):
n, sd, t, d, st, bm, ld = hf[i]
sdn = sd.replace('.', '_')
self.__dict__[sdn] = ct.c_int(-1)
p_id = ct.pointer(self.__dict__[sdn])
self.__wsl.hf_register_info_add(chf, i, p_id, n , sd, t, d, st, bm, ld)
self.__wsl.proto_register_field_array(self.__protocol, chf, lr)
def display(self):
self.__wsl.hf_register_info_print(self.__hf_register, \
len(self.__registers))
def get(self):
return self.__hf_register, len(self.__registers)
def __del__(self):
self.__wsl.hf_register_info_destroy(self.__hf_register)
#Subtrees definition
#Every subtree added can be accesses as an attribute after having been
#registered
class Subtree(object):
def __init__(self, wsl, protocol):
self.__wsl = wsl
self.__protocol = protocol
self.__st = {}
self.__user_defined_protocol_tree = False
def add(self, name):
if name == self.__protocol:
self.__user_defined_protocol_tree = True
self.__st[name] = ct.c_int(-1)
def has_user_defined_protocol_tree(self):
return self.__user_defined_protocol_tree
def register(self):
if not self.__user_defined_protocol_tree:
self.__st[self.__protocol] = ct.c_int(-1)
ls = len(self.__st)
if not ls:
return
CSubtrees = ct.POINTER(ct.c_int) * ls
p_sts = CSubtrees()
k = self.__st.keys()
for i in range(ls):
p_sts[i] = ct.pointer(self.__st[k[i]])
self.__wsl.proto_register_subtree_array(p_sts, ls)
def __getattr__(self, name):
if self.__st.has_key(name):
return self.__st[name]
#raise KeyError
#Dissector class : base class to write a dissector in python
class Dissector(object):
def __init__(self, protocol_name, short_desc, short):
self.__protocol_name = protocol_name
self.__short_desc = short_desc
self.__short = short
self.__tvb = None
self.__pinfo = None
self.__tree = None
self.__Tree = None
self.__offset = 0
self.__wsl = get_libws_handle()
self.__hf = None
self.__subtree = None
def _fields(self):
'''hf property : hf_register_info fields. every defined field is available
as an attribute of this object'''
if not self.__hf:
self.__hf = register_info(self.__wsl)
return self.__hf
hf = property(_fields)
def _subtrees(self):
'''subtrees property : subtress definition. every subtree added is
accessible as an attribute of this object'''
if not self.__subtree:
self.__subtree = Subtree(self.__wsl, self.__short)
return self.__subtree
subtrees = property(_subtrees)
def _tree(self):
'''tree property : initial tree at the start of the dissection'''
if not self.__Tree:
self.__Tree = Tree(self.__tree, self)
return self.__Tree
tree = property(_tree)
def display(self):
print(self.__short)
def _libhandle(self):
'''libhandle property : return a handle to the libwireshark lib. You don't
want to use this in normal situation. Use it only if you know what you're
doing.'''
return self.__wsl
libhandle = property(_libhandle)
def _raw_tree(self):
'''raw_tree property : returns the raw tree pointer. You can use this with
libhandle. You don't want to use this in normal situation. Use it only if
you know what you're doing.'''
return self.__tree
raw_tree = property(_raw_tree)
def _raw_pinfo(self):
'''raw_pinfo property : return the raw pinfo pointer. You can use this with
libhandle. You don't want to use this in normal situation. Use it only if
you know what you're doing.'''
return self.__pinfo
raw_pinfo = property(_raw_pinfo)
def _raw_tvb(self):
'''raw_tvb property : returns the raw tvb pointer. You can use this with
libhandle. You don't want to use this in normal situation. Use it only if
you know what you're doing.'''
return self.__tvb
raw_tvb = property(_raw_tvb)
def __str__(self):
# STA TODO : keep with short_desc because used in the hash table of
# dissectors in C code. If it is modified, it won't work anymore
return self.__short_desc
def __unicode__(self):
return self.__short
def __hash__(self):
return hash(self.__short)
def protocol(self):
return self.__protocol
def register_protocol(self):
'''private function called by libwireshark when registering all
protocols'''
self.__protocol = \
self.__wsl.proto_register_protocol( \
self.__protocol_name, self.__short_desc, \
self.__short)
self.__hf.register(self.__protocol)
#self.__hf.display()
self.subtrees.register()
def dissect(self):
'''point of entry when starting dissecting a packet. This method must be
therefore overloaded by the object implementing the dissector of a specific
protocol.'''
raise AttributeError('Dissector.dissect must be overridden')
def pre_dissect(self):
'''private method executed right before dissect in order to retrieve some
internal information and enabling the possibility to add the base tree of
this protocol dissection to the tree without any user intervention'''
self.__tvb = ct.c_void_p()
self.__pinfo = ct.c_void_p()
self.__tree = ct.c_void_p()
self.__wsl.py_dissector_args(ct.byref(self.__tvb), ct.byref(self.__pinfo), ct.byref(self.__tree))
# print self.__tvb, self.__pinfo, self.__tree
#self.__wsl.print_current_proto(ct.py_object(pinfo))
subt = self.subtrees
try:
if not subt.has_user_defined_protocol_tree():
p_tree = self.tree.add_item(self.protocol())
self.__Tree = p_tree.add_subtree(self.subtrees)
except:
print('pre_dissect error',e)
self.dissect()
def protocol_ids(self):
'''defined a list of tuples containing three values. Each tuple is defining
the parameters of dissector_add(). This function MUST be defined when
implementing the dissector of a specific protocol.'''
return [ (None, 0, None) ]
def find_dissector(self, protocol):
'''find_dissector : see proto.h'''
return self.__wsl.find_dissector(protocol)
def register_handoff(self):
'''private method used during the registration of protocol dissectors'''
#TODO STA : think how we would use dissector_add in an easy way *and* with
#the possibility to add the same dissector for TCP and UDP (extend
#py_generic_dissector)
private_handle = None
try:
ids = self.protocol_ids()
for type, protocol_id, handle in self.protocol_ids():
if not type:
continue
if not handle:
if not private_handle:
handle = self.__wsl.py_create_dissector_handle(self.__protocol)
else:
handle = private_handle
ct_type = ct.create_string_buffer(type)
ct_protocol_id = ct.c_uint(protocol_id)
self.__wsl.dissector_add_uint(ct_type, ct_protocol_id, handle)
except Exception as e:
print("creating dissector \"%s\" failed %s" % (self.__protocol_name, e))
raise
def advance(self, step):
'''method used to change the value of the offset'''
self.__offset += step
def _offset(self):
'''offset property : if is the current offset computed from the
dissection.'''
return self.__offset
offset = property(_offset)
#Tree class implementation
#see proto.h
class Tree(object):
def __init__(self, tree, dissector):
self.__dissector = dissector
self.__tree = ct.c_void_p(tree)
self.__wsl = dissector.libhandle
self.__tvb = dissector.raw_tvb
def _raw_tree(self):
return self.__tree
raw_tree = property(_raw_tree)
def add_item(self, field, offset=0, length=-1, little_endian=False, adv=True):
'''add an item to the tree'''
try:
tree = self.__wsl.proto_tree_add_item(self.__tree,
field, self.__tvb, self.__dissector.offset, length,
little_endian)
except Exception as e:
print(e)
else:
if length > 0 and adv:
self.__dissector.advance(length)
return Tree(tree, self.__dissector)
def add_uint(self, field, value, offset=0, length=4, adv=True):
'''add unsigned integer to the tree'''
try:
tree = self.__wsl.proto_tree_add_uint(self.__tree, field, self.__tvb, self.__dissector.offset, length, value)
except Exception as e:
print(e)
else:
if adv:
self.__dissector.advance(length)
return Tree(tree, self.__dissector)
def add_text(self, string, offset=0, length=-1, adv=True):
'''add text to the tree'''
try:
tree = self.__wsl.proto_tree_add_text(self.__tree, self.__tvb, self.__dissector.offset, length, string)
except Exception as e:
print(e)
else:
if length > 0 and adv:
self.__dissector.advance(length)
return Tree(tree, self.__dissector)
def add_subtree(self, subtree):
'''add a subtree to the tree'''
try:
tree = self.__wsl.proto_item_add_subtree(self.__tree, subtree)
except Exception as e:
print(e)
else:
return Tree(tree, self.__dissector)
#tvb class implementation
#see proto.h
class TVB(object):
def __init__(self, wsl, tvb, dissector):
self.__tvb = tvb
self.__wsl = wsl
self.__dissector = dissector
def length(self):
return self.__wsl.length(self.__wsl)
def length_remaining(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_length_remaining(self.__tvb, offset)
def reported_length(self):
return self.__wsl.tvb_reported_length(self.__tvb)
def reported_length_remaining(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_length_remaining(self.__tvb, offset)
def get_guint8(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_guint8(self.__tvb)
def get_ntohs(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_ntohs(self.__tvb, offset)
def get_ntohl(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_ntohl(self.__tvb, offset)
def get_letohl(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_letohl(self.__tvb, offset)
def get_letohs(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_letohs(self.__tvb, offset)
#STA TODO : check that we can do that
def get_ptr(self, offset=-1):
if offset < 0:
offset = self.__dissector.offset
return self.__wsl.tvb_get_ptr(self.__tvb, offset)
#how to get this working ??? check how application uses this!
#def new_subset(self, offset=0):
# return self.__wsl.tvb_get_new_subset(self.tvb, offset)
if False:
import linecache
import sys
# Start tracing when import has finished
def tracer(frame, event, arg):
if event == "line":
lineno = frame.f_lineno
filename = frame.f_globals["__file__"]
if (filename.endswith(".pyc") or
filename.endswith(".pyo")):
filename = filename[:-1]
name = frame.f_globals["__name__"]
line = linecache.getline(filename, lineno)
print("%s:%s: %s" % (name, lineno, line.rstrip()))
if event == "exception":
print("exception", arg)
return tracer
sys.settrace(tracer)