# <a href="https://www.gaodaima.com/tag/ping" title="查看更多关于ping的文章" target="_blank">ping</a>的原理是发送一个ICMP请求包,然后根据目的地址的应答包来判断是否能够和这个主机进行通信。<br># 我们使用python实现,借助于<a href="https://www.gaodaima.com/tag/scapy" title="查看更多关于scapy的文章" target="_blank">scapy</a>来进行编写程序。<br># 导入scapy包。<br>from scapy.all import *<br>import <a href="https://www.gaodaima.com/tag/time" title="查看更多关于time的文章" target="_blank">time</a>,struct,random<br># 编写ping一个包的函数。<br>def ping_one(dst = "36.152.44.95",ttl_no = 64,id_no = 345,seq_no = 5):<br> start_time = time.time()<br> # 将时间转换为二进制序列。<br> time_to_bytes = struct.pack(">d",start_time)<br> # 进行发送ICMP包,发送出去一个,收回来一个。<br> ping_one_result = sr1(IP(dst = dst,ttl = ttl_no)/ICMP(seq = seq_no,id = id_no)/time_to_bytes, timeout = 1, verbose=False)<br> # print(ping_one_result.show())<br> # 判断收回来的包是不是ICMP的应答包,和序列号是否相同。<br> try:<br> if ping_one_result.getlayer("ICMP").type == 0 and ping_one_result.getlayer("ICMP").seq == seq_no:<br> # print("进行解析包")<br> # 提取IP头部中的源IP地址,也就是我们ping的IP地址。<br> reply_src_IP = ping_one_result.getlayer("IP").src<br> # 提取序列号。<br> reply_icmp_seq = ping_one_result.getlayer("ICMP").seq<br> # 提取ttl<br> reply_icmp_ttl = ping_one_result.getlayer("IP").ttl<br> # 数据长度等于 数据长度(Raw) + 垫片长度(Padding) + 8字节(ICMP头部长度)<br> if ping_one_result.getlayer(Raw) != None:<br> Raw_length = len(ping_one_result.getlayer(Raw).load)<br> else:<br> Raw_length = 0<br> if ping_one_result.getlayer(Padding) != None:<br> Padding_length = len(ping_one_result.getlayer(Padding).load)<br> else:<br> Padding_length = 0<br> # 计算数据长度。<br> reply_data_length = Raw_length + Padding_length + 8<br> # 取出数据部分,这里的数据部分是我们发送ICMP请求包的时候填入的时间。<br> reply_data = ping_one_result.getlayer(Raw).load<br> # 定义我们收包的时间。<br> end_time = time.time()<br> # 将数据时间部分进行转换。<br> reply_data_time = struct.unpack(">d",reply_data)<br> # 然后打印出转换后的类型。<br> # print(type(reply_data_time))<br> # print(reply_data_time)<br> time_to_pass_ms = (end_time - reply_data_time[0]) * 1000<br> # (接收时间 - 发送时间) * 1000为毫秒数为消耗时间的毫秒数<br> # print(time_to_pass_ms)<br> return reply_data_length,reply_src_IP,reply_icmp_seq,reply_icmp_ttl,time_to_pass_ms<br> except Exception as e:<br> # 打印出错误。<br> # print("e", e)<br> # 匹配错误是否为NoneType类型。<br> if re.match(".*NoneType.*", str(e)):<br> print("错误了")<br> # 如果没有回应,就返回None<br> return None<br>def ping(dst = "36.152.44.95"):<br> # 这里其实可以取进程号的,但是我们用随机生成一个数字模拟一下。<br> id_no = random.randint(0,65535)<br> # print(id_no)<br> # 然后进行发送5个数据包。<br> for i in range(1,6):<br> # 调用ping一个包函数,入参为目的需要ping的IP地址。ttl,id,和序列号。seq。<br> ping_result = ping_one(dst,64,id_no,i)<br> if ping_result != None:<br> print("%d bytes from %s: icmp_seq=%d ttl=%d time=%4.2f ms" % (ping_result[0], ping_result[1], ping_result[2], ping_result[3], ping_result[4]))<br> else:<br> print(".",end = "",flush = True)<br> # 这里我们暂停一秒。<br> time.sleep(1)<br><br>if __name__ == "__main__":<br> ping("36.152.44.95")<br><br>但现在为止,我们的ping小程序就用python实现了,接下来就可以用wireshark工具抓包来看一下,进行ping百度的地址。
www#gaodaima.com来源gaodai#ma#com搞@代~码网搞代码