ping.go 10.0 KB

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