我正在使用 scapy,我想创建一个数据包并计算它的校验和而不发送它。有没有办法做到这一点?
谢谢。
我也试图避免 show2() 因为它打印数据包。我在源代码中找到了更好的解决方案:
del packet.chksum
packet = packet.__class__(bytes(packet))
此代码使用正确的校验和重新生成数据包,无需任何打印,实际上是 show2() 在打印前在后台运行的内容。
创建数据包后,您需要.chksum
从数据包中删除该值;然后打电话.show2()
>>> from scapy.layers.inet import IP
>>> from scapy.layers.inet import ICMP
>>> from scapy.layers.inet import TCP
>>> target = "10.9.8.7"
>>> ttl = 64
>>> id = 32711
>>> sport = 2927
>>> dport = 80
>>> pak = IP(dst=target, src = "100.99.98.97", ttl=ttl, flags="DF", id=id, len=1200, chksum = 0)/TCP(flags="S", sport=sport, dport=int(dport), options=[('Timestamp',(0,0))], chksum = 0)
>>> del pak[IP].chksum
>>> del pak[TCP].chksum
>>> pak.show2()
###[ IP ]###
version = 4L
ihl = 5L
tos = 0x0
len = 1200
id = 32711
flags = DF
frag = 0L
ttl = 64
proto = tcp
chksum = 0x9afd
src = 100.99.98.97
dst = 10.9.8.7
\options \
###[ TCP ]###
sport = 2927
dport = www
seq = 0
ack = 0
dataofs = 8L
reserved = 0L
flags = S
window = 8192
chksum = 0x2c0e
urgptr = 0
options = [('Timestamp', (0, 0)), ('EOL', None)]
>>>
将此补丁添加到 scapy/packet.py:
+ def checksum_silent(self):
+ """
+ Internal method that recalcs checksum without the annoying prints
+ **AFTER old checksums are deleted.**
+ """
+
+ for f in self.fields_desc:
+ if isinstance(f, ConditionalField) and not f._evalcond(self):
+ continue
+ fvalue = self.getfieldval(f.name)
+ if isinstance(fvalue, Packet) or (f.islist and f.holds_packets and type(fvalue) is list):
+ fvalue_gen = SetGen(fvalue,_iterpacket=0)
+ for fvalue in fvalue_gen:
+ fvalue.checksum_silent()
+ if self.payload:
+ self.payload.checksum_silent()
然后不用调用pkt.show2()
,只需调用这个函数
pkt.checksum_silent()
。 (记得先做del pkt[IP].chksum
anddel pkt[UDP].chksum
等)如上一个答案所示。
此功能应该更快且无声。(可能还有其他需要修剪的东西;我将这段代码一起破解,只测试以确保它在校验和正确的情况下保持沉默。)
实际上,该show2()
函数会为您计算校验和,但它也会在完成工作后打印数据包的内容。但是,show2()
有一个有用的小参数,名为dump
. 消息来源是这样描述的:
:param dump: 判断是打印还是返回字符串值
所以通过设置dump=True
,你可以避免函数默认提供的讨厌的输出,并且仍然得到你想要的计算。
您还可以使用packet.build()
which 返回具有正确校验和的原始字节。然后将字节转换为数据包。
>>> import scapy.all as sp
>>> packet = sp.IP(src='127.0.0.1', dst='8.8.8.8')
>>> packet
<IP src=127.0.0.1 dst=8.8.8.8 |>
>>> sp.IP(packet.build())
<IP version=4 ihl=5 tos=0x0 len=20 id=1 flags= frag=0 ttl=64
proto=hopopt chksum=0xebd8 src=127.0.0.1 dst=8.8.8.8 |>