Scapy-图层的长度字段

时间:2019-01-10 09:37:12

标签: python scapy variable-length

我正在尝试使用Scapy构建PTPv2协议。 该协议中的消息类型很少,因此我使用ConditionalField来描述不同的字段选项:

class PTPv2(Packet):
    name = "Precision Time Protocol V2"
    fields_desc = [
        # Header
        BitField('transportSpecific', 1, 4),
        BitEnumField('messageType', 0, 4, Message_Types),
        ByteField('versionPTP', 2),
        LenField('messageLength', None),
        ByteField('subdomainNumber', 0),
        ByteField('empty1', 0),
        XShortField('flags', 0),
        LongField('correction', 0),
        IntField('empty2', 0),
        XLongField('ClockIdentity', 0),
        XShortField('SourcePortId', 0),
        XShortField('sequenceId', 0),
        ByteField('control', 0),
        SignedByteField('logMessagePeriod', 0),

    # SYNC message, messageType=0
        ConditionalField(XBitField('TimestampSec', 0, 48),lambda pkt: pkt.messageType==0),
        ConditionalField(IntField('TimestampNanoSec', 0), lambda pkt: pkt.messageType == 0),

    # Follow up message, messageType=8
        ConditionalField(XBitField('preciseOriginTimestampSec', 0, 48), lambda pkt: pkt.messageType == 8),
        ConditionalField(IntField('preciseOriginTimestampNanoSec', 0), lambda pkt: pkt.messageType == 8)

     # Path delay resp follow up message, messageType=0xA
        ConditionalField(XBitField('responseOriginTimestampSec', 0, 48), lambda pkt: pkt.messageType == 0xA),
        ConditionalField(IntField('responseOriginTimestampNanoSec', 0), lambda pkt: pkt.messageType == 0xA),
        ConditionalField(XLongField('requestingSourcePortIdentity', 0), lambda pkt: pkt.messageType == 0xA),
        ConditionalField(XShortField('requestingSourcePortId', 0), lambda pkt: pkt.messageType == 0xA)

现在,我希望messageLength字段描述图层的长度,并根据现有字段自动进行计算。 它应该描述PTPv2层中所有字段的长度,从第一个字段(transportSpecific开始。

我了解了PacketLenFieldFieldLenField,但似乎它们都描述了一个字段的长度,而不是一组字段的长度(据我了解)。

我也读过LenField(在代码中写为类型),但是它计算的是下一层的长度,而不是当前层的长度(因此,在这种情况下始终为0)。

任何想法我该如何解决?

谢谢!

1 个答案:

答案 0 :(得分:1)

通常使用post_build回调在后台进行操作,因此:(您需要将其添加到数据包中)

(取自inet6.py)

def post_build(self, p, pay):
    # p += pay  # if you also want the payload to be taken into account
    if self.messageLength is None:
        tmp_len = len(p) # edit as you want
        p = p[:2] + struct.pack("!H", tmp_len) + p[4:]  # Adds length as short on bytes 3-4
    return p + pay # edit if previous is changed

然后可以相应地使用ByteField / ShortField ...代替LenField。

PacketLenField是您使用PacketListField,而FieldLenFieldFieldListField