ping.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. package main
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "os"
  6. "runtime"
  7. "sort"
  8. "strings"
  9. "sync"
  10. "sync/atomic"
  11. "time"
  12. "trial/ping/probing"
  13. "trial/ping/utils"
  14. "git.wecise.com/wecise/common/logger"
  15. "git.wecise.com/wecise/common/matrix/cfg"
  16. "git.wecise.com/wecise/common/matrix/util"
  17. "github.com/scylladb/go-set/strset"
  18. )
  19. type task struct {
  20. Server string `json:"server"`
  21. Timeout int `json:"timeout"`
  22. NumberOfPings int `json:"numberofpings"`
  23. PacketInterval int `json:"packetinterval"`
  24. PacketSize int `json:"packetsize"`
  25. TypeOfService int `json:"typeofservice"` // Not used
  26. Retries int `json:"retries"` // Not used
  27. Poll int `json:"poll"`
  28. FailureRetests int `json:"failureretests"` // Not used
  29. RetestInterval int `json:"retestinterval"` // Not used
  30. HostNameLookupPreference string `json:"hostnamelookuppreference"` // Not used
  31. Description string `json:"description"` // Not used
  32. Rule string `json:"rule"`
  33. PerfRule string `json:"perfrule"`
  34. TaskTime int64 `json:"tasktime"`
  35. Indicator string `json:"indicator"`
  36. }
  37. type InputConfig struct {
  38. Poolsize int `toml:"poolsize"`
  39. Domain string `toml:"domain"`
  40. StatInterval int `toml:"stat_interval"`
  41. }
  42. type StatInfo struct {
  43. MinRtt time.Duration
  44. MaxRtt time.Duration
  45. AvgRtt time.Duration
  46. LossCount int
  47. Count int
  48. }
  49. type Input struct {
  50. *InputConfig
  51. ips []string
  52. statinfomutex sync.Mutex
  53. statinfo map[string]*StatInfo
  54. subSize int
  55. workChan chan *task
  56. stopChan chan bool
  57. ds *utils.DataStat
  58. }
  59. var mcfg = cfg.MConfig()
  60. func main() {
  61. input := &Input{}
  62. input.statinfo = map[string]*StatInfo{}
  63. inputcfg := &InputConfig{
  64. Poolsize: 1000,
  65. StatInterval: 600,
  66. }
  67. inputcfg.Poolsize = mcfg.GetInt("poolsize", inputcfg.Poolsize)
  68. fips := func() []string {
  69. xips := mcfg.GetStrings("ip|ping.ip", "")
  70. sips := strset.New()
  71. for _, aips := range xips {
  72. sips.Add(strings.Split(aips, ",")...)
  73. }
  74. bs, _ := util.ReadFile("./hosts.txt")
  75. if len(bs) > 0 {
  76. xips := strings.Split(string(bs), "\n")
  77. for _, aips := range xips {
  78. sips.Add(strings.Split(aips, ",")...)
  79. }
  80. }
  81. return sips.List()
  82. }
  83. go func() {
  84. t := time.NewTicker(5 * time.Second)
  85. for {
  86. <-t.C
  87. input.ips = fips()
  88. }
  89. }()
  90. input.ips = fips()
  91. mcfg.OnChanged(func(cfg cfg.Configure) {
  92. input.ips = fips()
  93. })
  94. input.Init(inputcfg)
  95. go input.Run()
  96. go func() {
  97. last_count := 0
  98. t := time.NewTicker(1 * time.Second)
  99. for {
  100. select {
  101. case <-t.C:
  102. s := ""
  103. input.statinfomutex.Lock()
  104. ks := []string{}
  105. for k := range input.statinfo {
  106. ks = append(ks, k)
  107. }
  108. sort.Strings(ks)
  109. for i, k := range ks {
  110. v := input.statinfo[k]
  111. s += fmt.Sprintf("%-3d %-20s: %-12s [%12s ~ %-12s] loss %d/%d\n", i, k, v.AvgRtt, v.MinRtt, v.MaxRtt, v.LossCount, v.Count)
  112. }
  113. input.statinfomutex.Unlock()
  114. logger.Info("统计信息更新:", fmt.Sprint("\n", s))
  115. logger.Info(util.FormatDuration(time.Since(starttime)), "已经完成", pingcount, "次Ping操作,",
  116. "平均每秒", (int64(pingcount+1) * int64(time.Second) / int64(time.Since(starttime))), "次",
  117. "最近一秒", (pingcount - int32(last_count)), "次",
  118. )
  119. last_count = int(pingcount)
  120. }
  121. }
  122. }()
  123. n := 0
  124. for {
  125. if n >= len(input.ips) {
  126. n = 0
  127. }
  128. ip := input.ips[n]
  129. n++
  130. //
  131. ip = strings.TrimSpace(ip)
  132. if ip != "" {
  133. input.workChan <- &task{
  134. Server: ip,
  135. Timeout: mcfg.GetInt("timeout", 1),
  136. NumberOfPings: mcfg.GetInt("count", 5), // 至少为2,小于2会导致不能正常结束
  137. PacketInterval: mcfg.GetInt("interval", 1000),
  138. PacketSize: mcfg.GetInt("size", 32),
  139. }
  140. }
  141. // time.Sleep(1 * time.Millisecond)
  142. }
  143. }
  144. func (input *Input) Init(config interface{}) error {
  145. input.InputConfig = config.(*InputConfig)
  146. input.workChan = make(chan *task, input.Poolsize)
  147. input.stopChan = make(chan bool)
  148. input.subSize = runtime.GOMAXPROCS(0)
  149. return nil
  150. }
  151. func (input *Input) Run() (err error) {
  152. input.ds = utils.NewDataStat("service", "", input.StatInterval, input.Poolsize*2)
  153. defer input.ds.Close()
  154. go input.ds.Start()
  155. defer close(input.workChan)
  156. var wg sync.WaitGroup
  157. // Start worker
  158. // count := int32(0)
  159. // lpcmutex := sync.Mutex{}
  160. // lastprintcount := time.Now()
  161. for i := 0; i < input.Poolsize; i++ {
  162. go func(n int) {
  163. for t := range input.workChan {
  164. wg.Add(1)
  165. // Run task
  166. if e := input.send(t, n); e != nil {
  167. logger.Error("Send error:", e)
  168. input.ds.Send(1)
  169. } else {
  170. input.ds.Send(0)
  171. }
  172. wg.Done()
  173. // x := atomic.AddInt32(&count, 1)
  174. // lpcmutex.Lock()
  175. // printmsg := time.Since(lastprintcount) >= 1*time.Second
  176. // if printmsg {
  177. // lastprintcount = time.Now()
  178. // }
  179. // lpcmutex.Unlock()
  180. // if printmsg {
  181. // logger.Info("已经完成", x, "个目标的Ping操作")
  182. // }
  183. }
  184. }(i)
  185. }
  186. <-input.stopChan
  187. wg.Wait()
  188. return nil
  189. }
  190. var pingcount int32
  191. var starttime = time.Now()
  192. var printpingcountmutex sync.Mutex
  193. var printpingcounttime = time.Now()
  194. func (input *Input) send(t *task, workerNum int) error {
  195. hostname, _ := os.Hostname()
  196. var (
  197. status = "success"
  198. message = "Pings Complete"
  199. m = map[string]interface{}{
  200. "service": "icmp",
  201. "monitorHost": hostname,
  202. "host": t.Server,
  203. "pollInterval": t.Poll,
  204. "taskTime": t.TaskTime,
  205. "indicator": t.Indicator,
  206. }
  207. )
  208. pinger, err := probing.NewPinger(t.Server)
  209. if err != nil {
  210. status = "failure"
  211. message = err.Error()
  212. sips := strset.New(input.ips...)
  213. sips.Remove(t.Server)
  214. input.ips = sips.List()
  215. logger.Error(err)
  216. } else {
  217. /*
  218. https://stackoverflow.com/questions/41423637/go-ping-library-for-unprivileged-icmp-ping-in-golang/41425527#41425527
  219. This library attempts to send an "unprivileged" ping via UDP. On linux, this must be enabled by setting
  220. sudo sysctl -w net.ipv4.ping_group_range="0 2147483647"
  221. If you do not wish to do this, you can set pinger.SetPrivileged(true) and use setcap to allow your binary using go-ping to bind to raw sockets (or just run as super-user):
  222. setcap cap_net_raw=+ep /bin/goping-binary
  223. getcap /bin/goping-binary to validate
  224. */
  225. pinger.SetPrivileged(false)
  226. pinger.TTL = 64
  227. if t.NumberOfPings > 0 {
  228. pinger.Count = t.NumberOfPings
  229. } else {
  230. pinger.Count = 1
  231. }
  232. if t.PacketInterval > 0 {
  233. pinger.Interval = time.Millisecond * time.Duration(t.PacketInterval)
  234. }
  235. if t.PacketSize > 0 {
  236. pinger.Size = t.PacketSize
  237. }
  238. if t.Timeout > 0 {
  239. pinger.Timeout = time.Second * time.Duration(t.Timeout)
  240. } else {
  241. pinger.Timeout = time.Second * time.Duration(pinger.Count)
  242. }
  243. var (
  244. consecutiveFailures int
  245. )
  246. var recvMutex sync.Mutex
  247. var recvList = map[int]int{}
  248. pinger.OnSend = func(pkt *probing.Packet) {
  249. recvMutex.Lock()
  250. recvList[pkt.Seq]++
  251. recvMutex.Unlock()
  252. input.statinfomutex.Lock()
  253. si := input.statinfo[t.Server]
  254. if si == nil {
  255. si = &StatInfo{}
  256. input.statinfo[t.Server] = si
  257. }
  258. si.Count++
  259. input.statinfomutex.Unlock()
  260. }
  261. pingMsg := fmt.Sprintf("\nWoker %d PING %s (%s):\n", workerNum, pinger.Addr(), pinger.IPAddr())
  262. pinger.OnRecv = func(pkt *probing.Packet) {
  263. recvMutex.Lock()
  264. recvList[pkt.Seq]++
  265. recvMutex.Unlock()
  266. s := fmt.Sprintf("%s %d bytes from %s: icmp_seq=%d time=%v\n",
  267. time.Now().Format("15:04:05.000"), pkt.Nbytes, pkt.IPAddr, pkt.Seq, pkt.Rtt)
  268. pingMsg += s
  269. // fmt.Print(s)
  270. // printpingcount
  271. atomic.AddInt32(&pingcount, 1)
  272. }
  273. pinger.OnDuplicateRecv = func(pkt *probing.Packet) {
  274. recvMutex.Lock()
  275. recvList[pkt.Seq]++
  276. recvMutex.Unlock()
  277. s := fmt.Sprintf("%s %d bytes from %s: icmp_seq=%d time=%v (DUP!)\n",
  278. time.Now().Format("15:04:05.000"), pkt.Nbytes, pkt.IPAddr, pkt.Seq, pkt.Rtt)
  279. pingMsg += s
  280. // fmt.Print(s)
  281. }
  282. pinger.OnFinish = func(stats *probing.Statistics) {
  283. m["numberPackets"] = stats.PacketsSent
  284. m["averageRTT"] = stats.AvgRtt.Milliseconds()
  285. if stats.PacketsSent == 0 {
  286. m["respondPercent"] = 1
  287. } else {
  288. m["respondPercent"] = stats.PacketsRecv / stats.PacketsSent * 100
  289. }
  290. //m["failureRetests"] = ?
  291. s := fmt.Sprintf("--- %s ping statistics ---\n", stats.Addr)
  292. s += fmt.Sprintf("%d packets transmitted, %d packets received, %v%% packet loss\n",
  293. stats.PacketsSent, stats.PacketsRecv, stats.PacketLoss)
  294. s += fmt.Sprintf("round-trip min/avg/max/stddev = %v/%v/%v/%v\n",
  295. stats.MinRtt, stats.AvgRtt, stats.MaxRtt, stats.StdDevRtt)
  296. pingMsg += s
  297. // fmt.Print(s)
  298. input.statinfomutex.Lock()
  299. si := input.statinfo[t.Server]
  300. if si == nil {
  301. si = &StatInfo{
  302. MinRtt: stats.MinRtt,
  303. MaxRtt: stats.MaxRtt,
  304. AvgRtt: stats.AvgRtt,
  305. LossCount: 0,
  306. Count: 0,
  307. }
  308. input.statinfo[t.Server] = si
  309. }
  310. if stats.MinRtt > 0 && (si.MinRtt == 0 || stats.MinRtt < si.MinRtt) {
  311. si.MinRtt = stats.MinRtt
  312. }
  313. if stats.MaxRtt > si.MaxRtt {
  314. si.MaxRtt = stats.MaxRtt
  315. }
  316. si.AvgRtt = stats.AvgRtt
  317. si.LossCount += stats.PacketsSent - stats.PacketsRecv
  318. input.statinfomutex.Unlock()
  319. }
  320. m["requestTime"] = time.Now().UnixNano() / int64(time.Millisecond)
  321. if err = pinger.Run(); err != nil {
  322. status = "failure"
  323. message = err.Error()
  324. logger.Errorf("Ping error: %v", err)
  325. } else {
  326. }
  327. m["responseTime"] = time.Now().UnixNano() / int64(time.Millisecond)
  328. var failCount, totalFailCount int
  329. recvMutex.Lock()
  330. for i := range recvList {
  331. if recvList[i] < 2 {
  332. failCount++
  333. message = "Packet loss"
  334. totalFailCount++
  335. } else {
  336. failCount = 0
  337. }
  338. if failCount > consecutiveFailures {
  339. consecutiveFailures = failCount
  340. }
  341. pingMsg += fmt.Sprintf("icmp_seq:%d %t\n", i, (recvList[i] >= 2))
  342. }
  343. // logger.Debug(pingMsg)
  344. m["consecutiveFailures"] = consecutiveFailures
  345. if totalFailCount == len(recvList) {
  346. message = "ICMP echo failed"
  347. }
  348. recvMutex.Unlock()
  349. }
  350. // Special
  351. m["returnStatus"] = status
  352. m["message"] = message
  353. b, err := json.Marshal(m)
  354. if err != nil {
  355. logger.Fatal(err)
  356. }
  357. protocolMsg := map[string]interface{}{
  358. "protocol": true,
  359. "output": b,
  360. "attr": map[string]string{"previous_key": input.Domain + "_" + t.Server},
  361. }
  362. b, err = json.Marshal(protocolMsg)
  363. if err != nil {
  364. logger.Fatal(err)
  365. }
  366. return nil
  367. }