-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmultiping.go
156 lines (130 loc) · 4.3 KB
/
multiping.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package multiping
/**
* *** The motivation for this multi-ping fork ***
*
* There are quite a few Go pinger, but all of them have issues:
* * https://github.com/go-ping/ping works fine, but has problems when running
* several pingers in goroutines. When pinging ~300 hosts it looses ~1/3 packets.
* * https://github.com/caucy/batch_ping is umaintened for a long time and did not work for me at all.
* * https://github.com/rosenlo/go-MultiPing is a very young fork, has issues with logger, some parts
* of code are ineffective.
*
* Also need to mention that all these pingers are periodic pingers, they try to mimmic shell ping command.
* They run in internal loop, cancel that loop after timeout. They *can* be used, but you have to adjust your
* code to their style. Instead I wanted a pinger, that can ping multipple hosts at a time and be robust.
* I don't think its a problem for ping user to run it in a loop and don't want any hidden logic.
* So this ping is loosely based on above mentioned projects. It can ping multiple clients.
* And is cholesterol free.
**/
import (
"context"
"math/rand"
"sync"
"time"
"github.com/drgkaleda/go-multiping/pingdata"
"github.com/drgkaleda/go-multiping/pinger"
"golang.org/x/net/icmp"
"golang.org/x/net/ipv4"
"golang.org/x/net/ipv6"
)
var (
ipv4Proto = map[string]string{"icmp": "ip4:icmp", "udp": "udp4"}
ipv6Proto = map[string]string{"icmp": "ip6:ipv6-icmp", "udp": "udp6"}
)
type MultiPing struct {
// Locks MultiPing to protect internal members
sync.RWMutex
// Sync internal goroutines
wg sync.WaitGroup
// Timeout specifies a timeout before ping exits, regardless of how many
// packets have been received. Default is 1s.
Timeout time.Duration
// Tracker: Used to uniquely identify packet when non-priviledged
Tracker int64
ctx context.Context // context for timeouting
cancel context.CancelFunc // Do I need it ?
pinger *pinger.Pinger
pingData *pingdata.PingData
id uint16
sequence uint16 // ICMP seq number. Incremented on every ping
network string // one of "ip", "ip4", or "ip6"
protocol string // protocol is "icmp" or "udp".
conn4 *icmp.PacketConn
conn6 *icmp.PacketConn
rxChan chan *pinger.Packet
txChan chan *pinger.Packet
}
func New(privileged bool) (*MultiPing, error) {
protocol := "udp"
if privileged {
protocol = "icmp"
}
rand.Seed(time.Now().UnixNano())
mp := &MultiPing{
Timeout: time.Second,
id: uint16(rand.Intn(0xffff)),
network: "ip",
protocol: protocol,
Tracker: rand.Int63(),
}
mp.pinger = pinger.NewPinger(mp.network, mp.protocol, mp.id)
mp.pinger.SetPrivileged(privileged)
mp.pinger.Tracker = mp.Tracker
// try initialise connections to test that everything's working
err := mp.restart()
if err != nil {
mp.closeConnection()
return nil, err
}
// Sequence counter. It will be incremented in mp.restart on every ping
// Start with quite big initial value, so overwrap will occure fast (easier debugin)
mp.sequence = 0xfff0
return mp, nil
}
func (mp *MultiPing) restart() (err error) {
// ipv4
mp.conn4, err = icmp.ListenPacket(ipv4Proto[mp.protocol], "")
if err != nil {
return err
}
err = mp.conn4.IPv4PacketConn().SetControlMessage(ipv4.FlagTTL, true)
if err != nil {
return err
}
// ipv6 (note IPv6 may be disabled on OS and may fail)
mp.conn6, err = icmp.ListenPacket(ipv6Proto[mp.protocol], "")
if err == nil {
mp.conn6.IPv6PacketConn().SetControlMessage(ipv6.FlagHopLimit, true)
}
mp.pinger.SetConns(mp.conn4, mp.conn6)
mp.sequence++
// I use zero sequence number in statistics struct
// to detect duplicates, thus don't use it as valid sequence number
if mp.sequence == 0 {
mp.sequence++
}
mp.rxChan = make(chan *pinger.Packet)
mp.txChan = make(chan *pinger.Packet)
return nil
}
// closes active connections
func (mp *MultiPing) closeConnection() {
if mp.conn4 != nil {
mp.conn4.Close()
}
if mp.conn6 != nil {
mp.conn6.Close()
}
}
// cleanup cannot be done in close, because some goroutines may be using struct members
func (mp *MultiPing) cleanup() {
// Close rx channel.
// Tx channel is closed in batchPrepareIcmp()
close(mp.rxChan)
// invalidate connections
mp.conn4 = nil
mp.conn6 = nil
mp.pinger.SetConns(nil, nil)
// Invalidate pingData pointer (prevent from possible data corruption in future)
mp.pingData = nil
}