summaryrefslogtreecommitdiffstats
path: root/src/go/collectors/go.d.plugin/modules/portcheck/collect.go
blob: dab45ec41ad06b5f15d2ef41dc5720ae9126ea1a (plain)
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
// SPDX-License-Identifier: GPL-3.0-or-later

package portcheck

import (
	"fmt"
	"sync"
	"time"
)

type checkState string

const (
	checkStateSuccess checkState = "success"
	checkStateTimeout checkState = "timeout"
	checkStateFailed  checkState = "failed"
)

func (pc *PortCheck) collect() (map[string]int64, error) {
	wg := &sync.WaitGroup{}

	for _, p := range pc.ports {
		wg.Add(1)
		go func(p *port) { pc.checkPort(p); wg.Done() }(p)
	}
	wg.Wait()

	mx := make(map[string]int64)

	for _, p := range pc.ports {
		mx[fmt.Sprintf("port_%d_current_state_duration", p.number)] = int64(p.inState)
		mx[fmt.Sprintf("port_%d_latency", p.number)] = int64(p.latency)
		mx[fmt.Sprintf("port_%d_%s", p.number, checkStateSuccess)] = 0
		mx[fmt.Sprintf("port_%d_%s", p.number, checkStateTimeout)] = 0
		mx[fmt.Sprintf("port_%d_%s", p.number, checkStateFailed)] = 0
		mx[fmt.Sprintf("port_%d_%s", p.number, p.state)] = 1
	}

	return mx, nil
}

func (pc *PortCheck) checkPort(p *port) {
	start := time.Now()
	conn, err := pc.dial("tcp", fmt.Sprintf("%s:%d", pc.Host, p.number), pc.Timeout.Duration())
	dur := time.Since(start)

	defer func() {
		if conn != nil {
			_ = conn.Close()
		}
	}()

	if err != nil {
		v, ok := err.(interface{ Timeout() bool })
		if ok && v.Timeout() {
			pc.setPortState(p, checkStateTimeout)
		} else {
			pc.setPortState(p, checkStateFailed)
		}
		return
	}
	pc.setPortState(p, checkStateSuccess)
	p.latency = durationToMs(dur)
}

func (pc *PortCheck) setPortState(p *port, s checkState) {
	if p.state != s {
		p.inState = pc.UpdateEvery
		p.state = s
	} else {
		p.inState += pc.UpdateEvery
	}
}

func durationToMs(duration time.Duration) int {
	return int(duration) / (int(time.Millisecond) / int(time.Nanosecond))
}