4

有时我的 DSL 路由器会以这种奇怪的方式出现故障:

luis@balanceador:~$ sudo ping 8.8.8.8 -I eth9
[sudo] password for luis:
PING 8.8.8.8 (8.8.8.8) from 192.168.3.100 eth9: 56(84) bytes of data.
64 bytes from 8.8.8.8: icmp_seq=1 ttl=47 time=69.3 ms
ping: sendmsg: Operation not permitted
64 bytes from 8.8.8.8: icmp_seq=3 ttl=47 time=68.0 ms
ping: sendmsg: Operation not permitted
64 bytes from 8.8.8.8: icmp_seq=5 ttl=47 time=68.9 ms
64 bytes from 8.8.8.8: icmp_seq=6 ttl=47 time=67.2 ms
ping: sendmsg: Operation not permitted
64 bytes from 8.8.8.8: icmp_seq=8 ttl=47 time=67.2 ms
^C
--- 8.8.8.8 ping statistics ---
8 packets transmitted, 5 received, 37% packet loss, time 7012ms
rtt min/avg/max/mdev = 67.254/68.183/69.391/0.906 ms
luis@balanceador:~$ echo $?
0

可以看出,错误代码$?0. 所以我不能简单地检测命令是否失败,因为任何脚本的输出都不会产生错误

检测数据包丢失的正确方法是什么?
我需要解析输出grep还是有一些更简单的方法?

4

1 回答 1

4

根据手册页,默认情况下(在 Linux 上),如果 ping 根本没有收到任何回复数据包,它将以代码 1 退出。但是如果数据包计数(-c)和截止时间超时(-w,秒)是两者都指定,并且在超时之前收到的数据包较少,它也会以代码 1 退出。在其他错误时,它会以代码 2 退出。

ping 8.8.8.8 -I eth9 -c 3 -w 3

因此,如果在 3 秒内没有收到 3 个数据包,则会设置错误代码。

正如@mklement0 所指出的,在 BSD 上执行 ping 操作的方式有点不同:

ping 实用程序以下列值之一退出:

0 - 至少从指定主机听到一个响应。

2 - 传输成功,但未收到响应。

因此,在这种情况下,应该尝试通过在循环中逐一发送来解决它

ip=8.8.8.8
count=3
for i in $(seq ${count}); do
   ping ${ip} -I eth9 -c 1
   if [ $? -eq 2 ]; then
      ## break and retransmit exit code
      exit 2
   fi
done

当然,如果您需要完整的统计信息,只需对某些变量计数代码“2”和“0”,并在需要时在for循环后打印结果/设置错误代码。

于 2015-11-29T00:13:10.920 回答