Add network metrics

This commit is contained in:
Johannes 'fish' Ziemke 2014-02-18 11:26:33 +01:00
parent 107f94d90a
commit 9f17cd31c5
1 changed files with 82 additions and 1 deletions

View File

@ -19,6 +19,7 @@ const (
procLoad = "/proc/loadavg" procLoad = "/proc/loadavg"
procMemInfo = "/proc/meminfo" procMemInfo = "/proc/meminfo"
procInterrupts = "/proc/interrupts" procInterrupts = "/proc/interrupts"
procNetDev = "/proc/net/dev"
) )
type nativeCollector struct { type nativeCollector struct {
@ -27,6 +28,7 @@ type nativeCollector struct {
lastSeen prometheus.Gauge lastSeen prometheus.Gauge
memInfo prometheus.Gauge memInfo prometheus.Gauge
interrupts prometheus.Counter interrupts prometheus.Counter
netStats prometheus.Counter
name string name string
config config config config
} }
@ -46,6 +48,7 @@ func NewNativeCollector(config config, registry prometheus.Registry) (Collector,
lastSeen: prometheus.NewGauge(), lastSeen: prometheus.NewGauge(),
memInfo: prometheus.NewGauge(), memInfo: prometheus.NewGauge(),
interrupts: prometheus.NewCounter(), interrupts: prometheus.NewCounter(),
netStats: prometheus.NewCounter(),
} }
registry.Register( registry.Register(
@ -83,6 +86,13 @@ func NewNativeCollector(config config, registry prometheus.Registry) (Collector,
c.interrupts, c.interrupts,
) )
registry.Register(
"node_net",
"node_exporter: network stats.",
prometheus.NilLabels,
c.netStats,
)
return &c, nil return &c, nil
} }
@ -131,7 +141,7 @@ func (c *nativeCollector) Update() (updates int, err error) {
updates++ updates++
fv, err := strconv.ParseFloat(value, 64) fv, err := strconv.ParseFloat(value, 64)
if err != nil { if err != nil {
return updates, fmt.Errorf("Invalid value in interrupts: %s", err) return updates, fmt.Errorf("Invalid value in interrupts: %s", fv, err)
} }
labels := map[string]string{ labels := map[string]string{
"CPU": strconv.Itoa(cpuNo), "CPU": strconv.Itoa(cpuNo),
@ -142,6 +152,28 @@ func (c *nativeCollector) Update() (updates int, err error) {
c.interrupts.Set(labels, fv) c.interrupts.Set(labels, fv)
} }
} }
netStats, err := getNetStats()
if err != nil {
return updates, fmt.Errorf("Couldn't get netstats: %s", err)
}
for direction, devStats := range netStats {
for dev, stats := range devStats {
for t, value := range stats {
updates++
v, err := strconv.ParseFloat(value, 64)
if err != nil {
return updates, fmt.Errorf("Invalid value %s in interrupts: %s", value, err)
}
labels := map[string]string{
"device": dev,
"direction": direction,
"type": t,
}
c.netStats.Set(labels, v)
}
}
}
return updates, err return updates, err
} }
@ -274,3 +306,52 @@ func getInterrupts() (map[string]interrupt, error) {
} }
return interrupts, nil return interrupts, nil
} }
func getNetStats() (map[string]map[string]map[string]string, error) {
netStats := map[string]map[string]map[string]string{}
netStats["transmit"] = map[string]map[string]string{}
netStats["receive"] = map[string]map[string]string{}
fh, err := os.Open(procNetDev)
if err != nil {
return nil, err
}
defer fh.Close()
scanner := bufio.NewScanner(fh)
scanner.Scan() // skip first header
scanner.Scan()
parts := strings.Split(string(scanner.Text()), "|")
if len(parts) != 3 { // interface + receive + transmit
return nil, fmt.Errorf("Invalid header line in %s: %s",
procNetDev, scanner.Text())
}
header := strings.Fields(parts[1])
for scanner.Scan() {
parts := strings.Fields(string(scanner.Text()))
if len(parts) != 2*len(header)+1 {
return nil, fmt.Errorf("Invalid line in %s: %s",
procNetDev, scanner.Text())
}
dev := parts[0][:len(parts[0])-1]
receive, err := parseNetDevLine(parts[1:len(header)+1], header)
if err != nil {
return nil, err
}
transmit, err := parseNetDevLine(parts[len(header)+1:], header)
if err != nil {
return nil, err
}
netStats["transmit"][dev] = transmit
netStats["receive"][dev] = receive
}
return netStats, nil
}
func parseNetDevLine(parts []string, header []string) (map[string]string, error) {
devStats := map[string]string{}
for i, v := range parts {
devStats[header[i]] = v
}
return devStats, nil
}