Reference

Deferred Expressions

Several fields accept others fields as parameter to define how many bytes to consume, how many times a fields must be there or if a field should be there or not.

>>> from bisturi.packet import Packet
>>> from bisturi.field  import Int, Data

>>> class DeferredValue(Packet):
...     number  = Int(1)
...     paylaod = Data(number)
...     elemens = Int(1).repeated(number)
...     msg     = Data(until_marker=b'\x00').when(number)
...     author  = Data(until_marker=b'\x00').when(msg)

>>> raw3 = b'\x03ABC\x01\x02\x03OK\x00joe\x00'
>>> pkt3  = DeferredValue.unpack(raw3)

>>> pkt3.number
3
>>> pkt3.paylaod
b'ABC'
>>> pkt3.elemens
[1, 2, 3]
>>> pkt3.msg
b'OK'
>>> pkt3.author
b'joe'

>>> raw0 = b'\x00ignored\x00'
>>> pkt0 = DeferredValue.unpack(raw0)

>>> pkt0.number
0
>>> pkt0.paylaod
b''
>>> pkt0.elemens
[]
>>> pkt0.msg is None
True
>>> pkt0.author is None
True

>>> pkt3.pack() == raw3
True
>>> pkt0.pack() == b'\x00'
True

But using only a field as parameter is just the begin.

You can use simple expressions as arguments that use one or more fields as operands:

>>> class ArithExpressions(Packet):
...     rows  = Int(1)
...     cols  = Int(1)
...
...     values  = Int(1).repeated(rows * cols)
...     padding = Data( 8 - ((rows * cols) % 8) )

>>> class SequenceExpressions(Packet):
...     items = Int(1).repeated(4)
...
...     extra_data  = Data(4).when(items[0] == 0xff)
...     hidden_data = Data(4).when(items[:2] == [0xbe, 0xef])

>>> raw_matrix = b'\x02\x03ABCDEF\xff\xff'
>>> matrix = ArithExpressions.unpack(raw_matrix)

>>> matrix.rows, matrix.cols
(2, 3)
>>> matrix.values
[65, 66, 67, 68, 69, 70]
>>> matrix.padding
b'\xff\xff'

>>> raw_extra_msg = b'\xffABCHELO'
>>> extra_msg = SequenceExpressions.unpack(raw_extra_msg)

>>> extra_msg.items
[255, 65, 66, 67]
>>> extra_msg.extra_data
b'HELO'
>>> extra_msg.hidden_data is None
True

>>> raw_hidden_msg = b'\xbe\xefABbeef'
>>> hidden_msg = SequenceExpressions.unpack(raw_hidden_msg)

>>> hidden_msg.items
[190, 239, 65, 66]
>>> hidden_msg.extra_data is None
True
>>> hidden_msg.hidden_data
b'beef'

>>> matrix.pack() == raw_matrix
True

>>> extra_msg.pack() == raw_extra_msg
True
>>> hidden_msg.pack() == raw_hidden_msg
True

As a special case, any field or expression of fields implement the chooses operator.

This one select an item based on the value of the field/expression who owns the operator.

This can be use to select one item from its parameters by position in the argument list ( chooses(a, b, ...) ) or by name from a keyword argument call ( chooses(k1=a, k2=b, ...) ).

If the chooses operator receives only one argument, this must be a list/tuple (to select by position like chooses([a, b, ...]) ) or a dictionary (to select by keyword like chooses({k1: a, k2: b, ...}) ).

Here are some examples of using chooses by position and by name:

>>> class ChooseExpressions(Packet):
...     a = Int(1)
...     b = Int(1)
...
...     # by position from the argument list:
...     # if a == 1 then the position 1 is looked up and True will be returned
...     # if a != 1 then the position 0 is looked up and False will be returned
...     extra     = Data(byte_count = 1).when( (a == 1).chooses(False, True) )
...
...     # by position but from a list
...     # if a < b then the position 1 is looked up and 'a' will be returned
...     # if a >= b then the position 0 is looked up and b' will be returned
...     mindata   = Data(byte_count = (a < b).chooses([b, a]))
...
...     # by name/keyword from a dictionary (this is the preferred way to
...     # code a 'if-then-else' expression. Equivalent to: 4 if a > 4 else a
...     truncated = Data(byte_count = (a > 4).chooses({True: 4, False: a}))

>>> raw_min = b'\x01\x02:AB'
>>> pkt_min = ChooseExpressions.unpack(raw_min)

>>> pkt_min.extra
b':'
>>> pkt_min.mindata
b'A'
>>> pkt_min.truncated
b'B'


>>> raw_trunc = b'\x06\x02AABBBBBBBBBBBBBBBB'
>>> pkt_trunc = ChooseExpressions.unpack(raw_trunc)

>>> pkt_trunc.extra is None
True
>>> pkt_trunc.mindata
b'AA'
>>> pkt_trunc.truncated
b'BBBB'

>>> pkt_min.pack() == raw_min
True
>>> pkt_trunc.pack() == raw_trunc[:8]
True

If the pool of names from where you want to choose one is a pool of valid Python names you can use keyword arguments as a shortcut instead of an explicit dictionary.

>>> class ChooseExpressions(Packet):
...     size_type = Data(until_marker=b'\x00')
...     data      = Data(byte_count = size_type.chooses(small=2, large=4, extra_large=8))

>>> raw_small = b'small\x00AB'
>>> pkt_small = ChooseExpressions.unpack(raw_small)

>>> pkt_small.size_type
b'small'
>>> pkt_small.data
b'AB'

>>> raw_large = b'large\x00ABCD'
>>> pkt_large = ChooseExpressions.unpack(raw_large)

>>> pkt_large.size_type
b'large'
>>> pkt_large.data
b'ABCD'

>>> pkt_small.pack() == raw_small
True
>>> pkt_large.pack() == raw_large
True