node_exporter/collector/netdev_freebsd.go

111 lines
3.1 KiB
Go
Raw Normal View History

2015-09-26 08:36:40 -07:00
// Copyright 2015 The Prometheus Authors
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
2015-05-12 06:18:05 -07:00
// +build !nonetdev
package collector
import (
"errors"
"fmt"
"strconv"
"github.com/prometheus/client_golang/prometheus"
)
/*
2015-06-24 05:02:12 -07:00
#cgo CFLAGS: -D_IFI_OQDROPS
2015-05-12 06:18:05 -07:00
#include <stdio.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <ifaddrs.h>
#include <net/if.h>
*/
import "C"
type netDevCollector struct {
2015-09-16 07:06:59 -07:00
subsystem string
metricDescs map[string]*prometheus.Desc
2015-05-12 06:18:05 -07:00
}
func init() {
Factories["netdev"] = NewNetDevCollector
}
2015-06-24 05:02:12 -07:00
// Takes a prometheus registry and returns a new Collector exposing
// Network device stats.
2015-06-23 06:49:18 -07:00
func NewNetDevCollector() (Collector, error) {
2015-05-12 06:18:05 -07:00
return &netDevCollector{
2015-09-16 07:06:59 -07:00
subsystem: "network",
metricDescs: map[string]*prometheus.Desc{},
2015-05-12 06:18:05 -07:00
}, nil
}
func (c *netDevCollector) Update(ch chan<- prometheus.Metric) (err error) {
netDev, err := getNetDevStats()
if err != nil {
2015-06-24 05:02:12 -07:00
return fmt.Errorf("couldn't get netstats: %s", err)
2015-05-12 06:18:05 -07:00
}
for dev, devStats := range netDev {
for key, value := range devStats {
2015-09-16 07:06:59 -07:00
desc, ok := c.metricDescs[key]
if !ok {
desc = prometheus.NewDesc(
prometheus.BuildFQName(Namespace, c.subsystem, key),
fmt.Sprintf("%s from getifaddrs().", key),
[]string{"device"},
nil,
)
c.metricDescs[key] = desc
}
v, err := strconv.ParseFloat(value, 64)
if err != nil {
2015-09-16 07:06:59 -07:00
return fmt.Errorf("invalid value %s in netstats: %s", value, err)
2015-05-12 06:18:05 -07:00
}
2015-09-16 07:06:59 -07:00
ch <- prometheus.MustNewConstMetric(desc, prometheus.GaugeValue, v, dev)
2015-05-12 06:18:05 -07:00
}
}
return nil
2015-05-12 06:18:05 -07:00
}
func getNetDevStats() (map[string]map[string]string, error) {
netDev := map[string]map[string]string{}
2015-05-12 06:18:05 -07:00
var ifap, ifa *C.struct_ifaddrs
if C.getifaddrs(&ifap) == -1 {
2015-07-13 14:29:55 -07:00
return nil, errors.New("getifaddrs() failed")
2015-05-12 06:18:05 -07:00
}
2015-06-24 05:02:12 -07:00
defer C.freeifaddrs(ifap)
2015-05-12 06:18:05 -07:00
for ifa = ifap; ifa != nil; ifa = ifa.ifa_next {
if ifa.ifa_addr.sa_family == C.AF_LINK {
devStats := map[string]string{}
2015-05-12 06:18:05 -07:00
data := (*C.struct_if_data)(ifa.ifa_data)
devStats["receive_packets"] = strconv.Itoa(int(data.ifi_ipackets))
devStats["transmit_packets"] = strconv.Itoa(int(data.ifi_opackets))
devStats["receive_errs"] = strconv.Itoa(int(data.ifi_ierrors))
devStats["transmit_errs"] = strconv.Itoa(int(data.ifi_oerrors))
devStats["receive_bytes"] = strconv.Itoa(int(data.ifi_ibytes))
devStats["transmit_bytes"] = strconv.Itoa(int(data.ifi_obytes))
devStats["receive_multicast"] = strconv.Itoa(int(data.ifi_imcasts))
devStats["transmit_multicast"] = strconv.Itoa(int(data.ifi_omcasts))
devStats["receive_drop"] = strconv.Itoa(int(data.ifi_iqdrops))
devStats["transmit_drop"] = strconv.Itoa(int(data.ifi_oqdrops))
2015-09-16 07:06:59 -07:00
netDev[C.GoString(ifa.ifa_name)] = devStats
2015-05-12 06:18:05 -07:00
}
}
return netDev, nil
}