forked from facebookarchive/fbssdc
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathvarnum.py
49 lines (44 loc) · 998 Bytes
/
varnum.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
#!/usr/bin/env python3
import struct
# Implementation of vbytes
# This is a straight port of `varnum.rs` from binjs-ref.
def write(value, out):
while True:
byte = ((value & 0x7F) << 1)
if value > 0x7F:
byte |= 1
out.write(struct.pack('B', byte))
value >>= 7
if value == 0:
break
def read(inp):
'''Read a varnum
>>> import io
>>> def roundtrip(sample):
... buf = io.BytesIO()
... write(sample, buf)
... print(len(buf.getvalue()))
... buf.seek(0)
... return read(buf)
>>> roundtrip(0)
1
0
>>> roundtrip(1)
1
1
>>> roundtrip(10)
1
10
>>> roundtrip(777)
2
777
'''
result = 0
shift = 0
while True:
assert shift < 32 # FIXME: We need a better error message
byte = struct.unpack('B', inp.read(1))[0]
result |= (byte >> 1) << shift
if byte & 1 == 0:
return result
shift += 7