Files
Hippolyzer/pyogp/lib/base/message_template.py

226 lines
6.2 KiB
Python

#!/usr/bin/python
"""
@file packet.py
@author Linden Lab
@date 2008-06-13
@brief Iniitializes path directories
$LicenseInfo:firstyear=2008&license=apachev2$
Copyright 2008, Linden Research, Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
$/LicenseInfo$
"""
#standard libraries
import struct
import string
import re
import pprint
#local libraries
from makepacketdict import makepacketdict, makereversepacketdict
mypacketdictionary = makepacketdict()
myreversedictionary = makereversepacketdict()
""" can construct and deconstruct packet headers. Has nothing to
do with the packet payload, yet. """
LL_ZERO_CODE_FLAG = '\x80'
LL_RELIABLE_FLAG = '\x40'
LL_RESENT_FLAG = '\x20'
LL_ACK_FLAG = '\x10'
LL_NONE = '\x00'
FIXED_FREQUENCY_MESSAGE = '\xFF\xFF\xFF'
LOW_FREQUENCY_MESSAGE = '\xFF\xFF'
MEDIUM_FREQUENCY_MESSAGE = '\xFF'
HIGH_FREQUENCY_MESSAGE = ''
class MessageTemplateVariable():
def __init__(self, name, tp):
self.name = name
self.lltype = tp
def get_name(self):
return self.name
def get_type(self):
return self.lltype
class MessageTemplateBlock():
def __init__(self, header):
self.vars = {}
self.name = header[0]
self.blockType = header[1]
if self.blockType == 'Multiple':
self.number = int(header[2])
else:
self.number = 0
def get_block_type(self):
return self.blockType
def get_block_number(self):
return self.number
def get_name(self):
return self.name
def add_variable(self, var):
self.vars[var.get_name()] = var
def get_variables(self):
return self.vars.values()
def get_variable(self, name):
return self.vars[name]
class MessageTemplate():
def __init__(self, header):
self.blocks = {}
self.name = header[0]
self.frequency = header[1]
self.msg_num = string.atoi(header[2],0)
if self.frequency == 'Fixed':
#have to do this because Fixed messages are stored as a long in the template
binTemp = struct.pack('>L', string.atol(header[2],0))
self.msg_num_hex = binTemp
self.msg_num = struct.unpack('>h','\x00' + binTemp[3])[0]
elif self.frequency == 'Low':
self.msg_num_hex = struct.pack('>BBh',0xff,0xff, self.msg_num)
elif self.frequency == 'Medium':
self.msg_num_hex = struct.pack('>BB',0xff, self.msg_num)
elif self.frequency == 'High':
self.msg_num_hex = struct.pack('>B', self.msg_num)
self.msg_trust = header[3]
self.msg_encoding = header[4]
if len(header) > 5:
self.msg_deprecation = header[5]
else:
self.msg_deprecation = ''
def get_frequency(self):
return self.frequency
def get_message_number(self):
return self.msg_num
def get_message_hex_num(self):
return self.msg_num_hex
def get_trust(self):
return self.msg_trust
def get_encoding(self):
return self.msg_encoding
def get_deprecation(self):
return self.msg_deprecation
def get_name(self):
return self.name
def add_block(self, block):
self.blocks[block.get_name()] = block
def get_blocks(self):
return self.blocks.values()
def get_block(self, name):
return self.blocks[name]
#these remain unformatted (by standard) because they are going to be moved
def decodeHeaderPair(frequency, num):
return mypacketdictionary[(frequency, num)]
def decodeFrequency(header):
#if it is not a high
if header[0] == '\xFF':
#if it is not a medium frequency message
if header[1] == '\xFF':
#if it is a Fixed frequency message
if header[2] == '\xFF':
return 'Fixed'
#then it is low
else:
return 'Low'
#then it is medium
else:
return 'Medium'
#then it is high
else:
return 'High'
return None
def decodeNum(header):
frequency = decodeFrequency(header)
if frequency == 'Low':
return struct.unpack('B', header[2:4])[0] #int("0x"+ByteToHex(header[2:4]).replace(' ', ''),16)
elif frequency == 'Medium':
return struct.unpack('B', header[1:2])[0] #int("0x"+ByteToHex(header[1:2]).replace(' ', ''),16)
elif frequency == 'High':
return struct.unpack('B', header[0])[0] #int("0x"+ByteToHex(header[0]), 16)
elif frequency == 'Fixed':
return struct.unpack('B', header[0:4])[0] #int("0x"+ByteToHex(header[0:4]).replace(' ', ''), 16)
else:
return None
def decodeHeader(header):
frequency = decodeFrequency(header)
num = decodeNum(header)
return decodeHeaderPair(frequency, num)
def encodePacketID(frequency, num):
if frequency == 'Low':
frequencyData = LOW_FREQUENCY_MESSAGE
packedNum = struct.pack('>H', num)
elif frequency == 'Medium':
frequencyData = MEDIUM_FREQUENCY_MESSAGE
packedNum = struct.pack('>B', num)
elif frequency == 'High':
frequencyData = HIGH_FREQUENCY_MESSAGE
packedNum = struct.pack('>B', num)
elif frequency == 'Fixed':
frequencyData = FIXED_FREQUENCY_MESSAGE
packedNum = struct.pack('>B', num)
return frequencyData + packedNum
def encodeHeaderName(ack, sequenceNumber, packetName):
header_tuple = myreversedictionary[packetName]
frequency = header_tuple[0]
num = header_tuple[1]
return encodeHeader(ack, sequenceNumber, frequency, num)
def encodeHeader(ack, sequenceNumber, frequency, num):
packetID = encodePacketID(frequency, num)
return ack + struct.pack('>LB', sequenceNumber, 0) + packetID