1、什么是fping
fping 是一个类似 ping 的程序,与 ping 不同的是你可以同时指定多个要 ping 的目标。
fping 发送完 ping 包后不等待回应,而是继续下一个目标
2、实现思路
1、获取IP段
2、写一个ping方法
3、将IP段分解成一个个的IP
4、用协程一个个去ping这些IP
3、具体实现
- 获取IP段
'1', '2020-11-26 17:11:39.515', '2020-11-26 17:11:39.515', '1.0.1.0', '1.0.3.255', '福建', '电信'
'2', '2020-11-26 17:11:43.553', '2020-11-26 17:11:43.553', '1.0.8.0', '1.0.15.255', '广东', '电信'
'3', '2020-11-26 17:11:47.563', '2020-11-26 17:11:47.563', '1.0.32.0', '1.0.63.255', '广东', '电信'
'4', '2020-11-26 17:11:53.621', '2020-11-26 17:11:53.621', '1.1.0.0', '1.1.0.255', '福建', '电信'
'5', '2020-11-26 17:11:57.635', '2020-11-26 17:11:57.635', '1.1.2.0', '1.1.7.255', '福建', '电信'
'6', '2020-11-26 17:11:59.642', '2020-11-26 17:11:59.642', '1.1.8.0', '1.1.63.255', '广东', '电信'
'7', '2020-11-26 17:12:05.671', '2020-11-26 17:12:05.671', '1.2.0.0', '1.2.1.255', '福建', '电信'
'8', '2020-11-26 17:12:07.674', '2020-11-26 17:12:07.674', '1.2.2.0', '1.2.2.255', '北京', '联通'
'9', '2020-11-26 17:12:13.690', '2020-11-26 17:12:13.690', '1.2.5.0', '1.2.7.255', '北京', '电信'
- ping方法(单个,多个已给出)
func IsAlivePing(target string) (bool, error) {
pinger, err := ping.NewPinger(target)
if err != nil {
return false, err
}
pinger.Count = 10
PINGTIME := time.Duration(1000)
pinger.Timeout = time.Duration(PINGTIME * time.Millisecond)
pinger.SetPrivileged(true)
pinger.Run()
stats := pinger.Statistics()
if stats.PacketsRecv >= 1 {
return true, nil
}
return false, nil
}
func FilterActiveIPs(ips []string) (inactiveIPs []string) {
var lock sync.Mutex
var wg sync.WaitGroup
wg.Add(len(ips))
for _, ip := range ips {
go func(ip string) {
defer wg.Done()
isAlive, err := IsAlivePing(ip)
if !isAlive && err == nil {
RandomSleep(2)
if isAlive, _ := IsAlivePing(ip); !isAlive {
RandomSleep(2)
if isAlive, _ := IsAlivePing(ip); !isAlive {
lock.Lock()
defer lock.Unlock()
inactiveIPs = append(inactiveIPs, ip)
}
}
}
}(ip)
}
wg.Wait()
return inactiveIPs
}
func FilterInactiveIPs(ips []string) (activeIPs []string) {
var lock sync.Mutex
var wg sync.WaitGroup
wg.Add(len(ips))
for _, ip := range ips {
go func(ip string) {
defer wg.Done()
if isAlive, err := IsAlivePing(ip); isAlive && err == nil {
lock.Lock()
defer lock.Unlock()
activeIPs = append(activeIPs, ip)
}
}(ip)
}
wg.Wait()
return activeIPs
}
func RandomSleep(t int) {
rand.Seed(time.Now().UnixNano())
r := time.Duration(rand.Intn(t) + 1)
time.Sleep(r * time.Second)
}
- 分解IP段
func (i *impl) Separation(minIP, maxIP string) []string {
shouldStrs := make([]string, 0)
min, _ := IPString2Long(minIP)
max, _ := IPString2Long(maxIP)
for min <= max {
minIPstr, _ := Long2IPString(min)
xxxip := net.ParseIP(minIPstr).To4()
if xxxip[3] != 0 && xxxip[3] != 255 {
shouldStrs = append(shouldStrs, minIPstr)
}
min++
}
fmt.Printf("IP段%s--%s不包括两端的长度: %d\n", minIP, maxIP, len(shouldStrs))
return shouldStrs
}
func Long2IPString(i uint) (string, error) {
if i > math.MaxUint32 {
return "", errors.New("beyond the scope of ipv4")
}
ip := make(net.IP, net.IPv4len)
ip[0] = byte(i >> 24)
ip[1] = byte(i >> 16)
ip[2] = byte(i >> 8)
ip[3] = byte(i)
return ip.String(), nil
}
func IPString2Long(ip string) (uint, error) {
b := net.ParseIP(ip).To4()
if b == nil {
return 0, errors.New("invalid ipv4 format")
}
return uint(b[3]) | uint(b[2])<<8 | uint(b[1])<<16 | uint(b[0])<<24, nil
}