Files
mongo/lang/python/packing.py
Don Anderson fd85797033 Added support to pack type 't', needed for 'fix' tables.
Added a simple test for fix tables.
refs #31
refs #16

--HG--
extra : rebase_source : 26f08d2b3715df318a0965e9c0d0f8b437cd8240
2011-10-06 07:06:08 -04:00

123 lines
2.7 KiB
Python

# WiredTiger variable-length packing and unpacking functions
from intpacking import pack_int, unpack_int
def __get_type(fmt):
if not fmt:
return None, fmt
# Variable-sized encoding is the default (and only supported format in v1)
if fmt[0] in '.@<>':
tfmt = fmt[0]
fmt = fmt[1:]
else:
tfmt = '.'
return tfmt, fmt
def unpack(fmt, s):
tfmt, fmt = __get_type(fmt)
if not fmt:
return ()
if tfmt != '.':
raise ValueError('Only variable-length encoding is currently supported')
result = []
havesize = size = 0
for offset, f in enumerate(fmt):
if f.isdigit():
size = (size * 10) + int(f)
havesize = 1
continue
elif f == 'x':
if not havesize:
size = 1
s = s[size:]
# Note: no value, don't increment i
elif f in 'Ssu':
if not havesize:
if f == 's':
size = 1
elif f == 'S':
size = s.find('\0')
elif f == 'u':
if offset == len(fmt) - 1:
size = len(s)
else:
size, s = unpack_int(s)
result.append(s[:size])
if f == 'S' and not havesize:
size += 1
s = s[size:]
elif f in 't':
# bit type, size is number of bits
if not havesize:
size = 1
result.append(ord(s[0:1]))
s = s[1:]
else:
# integral type
if not havesize:
size = 1
for j in xrange(size):
v, s = unpack_int(s)
result.append(v)
havesize = size = 0
return result
def pack(fmt, *values):
tfmt, fmt = __get_type(fmt)
if not fmt:
return ()
if tfmt != '.':
raise ValueError('Only variable-length encoding is currently supported')
result = ''
havesize = i = size = 0
for offset, f in enumerate(fmt):
if f.isdigit():
size = (size * 10) + int(f)
havesize = 1
continue
elif f == 'x':
if not havesize:
result += '\0'
else:
result += '\0' * size
# Note: no value, don't increment i
elif f in 'Ssu':
if f == 'S' and '\0' in values[i]:
l = values[i].find('\0')
else:
l = len(values[i])
if havesize:
if l > size:
l = size
elif f == 's':
havesize = size = 1
elif f == 'u' and offset != len(fmt) - 1:
result += pack_int(l)
result += values[i][:l]
if f == 'S' and not havesize:
result += '\0'
elif size > l:
result += '\0' * (size - l)
i += 1
elif f in 't':
# bit type, size is number of bits
if not havesize:
size = 1
if size > 8:
raise ValueError("bit count cannot be greater than 8 for 't' encoding")
mask = (1 << size) - 1
val = values[i]
if (mask & val) != val:
raise ValueError("value out of range for 't' encoding")
result += chr(val)
i += 1
else:
# integral type
if not havesize:
size = 1
for j in xrange(size):
result += pack_int(values[i])
i += 1
havesize = size = 0
return result