Sop*_*rez 5 shell ping packet-loss
有时我的 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
Run Code Online (Sandbox Code Playgroud)
可以看出,错误代码$?为0。因此,我不能简单地检测命令是否失败,因为任何脚本的输出都不会产生错误。
检测是否存在丢包的正确方法是什么?
我是否需要使用或有一些更简单的方法来解析输出?grep
根据手册页,默认情况下(在 Linux 上),如果 ping 根本没有收到任何回复数据包,它将以代码 1 退出。但是如果数据包计数( -c)和截止时间超时(-w,秒)是都指定了,并且在超时之前收到的数据包较少,它也会以代码 1 退出。如果出现其他错误,它会以代码 2 退出。
ping 8.8.8.8 -I eth9 -c 3 -w 3
Run Code Online (Sandbox Code Playgroud)
因此,如果 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
Run Code Online (Sandbox Code Playgroud)
当然,如果您需要完整的统计数据,只需对某些变量计数代码“2”和“0”,并在for循环后打印结果/设置错误代码(如果需要)。