mirror of
https://github.com/prometheus/node_exporter.git
synced 2025-01-03 09:57:47 -08:00
commit
3b6b5dfb92
|
@ -55,6 +55,7 @@ runit | Exposes service status from [runit](http://smarden.org/runit/). | _any_
|
||||||
supervisord | Exposes service status from [supervisord](http://supervisord.org/). | _any_
|
supervisord | Exposes service status from [supervisord](http://supervisord.org/). | _any_
|
||||||
systemd | Exposes service and system status from [systemd](http://www.freedesktop.org/wiki/Software/systemd/). | Linux
|
systemd | Exposes service and system status from [systemd](http://www.freedesktop.org/wiki/Software/systemd/). | Linux
|
||||||
tcpstat | Exposes TCP connection status information from `/proc/net/tcp` and `/proc/net/tcp6`. (Warning: the current version has potential performance issues in high load situations.) | Linux
|
tcpstat | Exposes TCP connection status information from `/proc/net/tcp` and `/proc/net/tcp6`. (Warning: the current version has potential performance issues in high load situations.) | Linux
|
||||||
|
wifi | Exposes WiFi device and station statistics. | Linux
|
||||||
|
|
||||||
### Deprecated
|
### Deprecated
|
||||||
|
|
||||||
|
|
|
@ -1972,6 +1972,33 @@ node_sockstat_sockets_used 229
|
||||||
# HELP node_textfile_scrape_error 1 if there was an error opening or reading a file, 0 otherwise
|
# HELP node_textfile_scrape_error 1 if there was an error opening or reading a file, 0 otherwise
|
||||||
# TYPE node_textfile_scrape_error gauge
|
# TYPE node_textfile_scrape_error gauge
|
||||||
node_textfile_scrape_error 0
|
node_textfile_scrape_error 0
|
||||||
|
# HELP node_wifi_interface_frequency_hertz The current frequency a WiFi interface is operating at, in hertz.
|
||||||
|
# TYPE node_wifi_interface_frequency_hertz gauge
|
||||||
|
node_wifi_interface_frequency_hertz{device="wlan0"} 2.412e+09
|
||||||
|
# HELP node_wifi_station_beacon_loss_total The total number of times a station has detected a beacon loss.
|
||||||
|
# TYPE node_wifi_station_beacon_loss_total counter
|
||||||
|
node_wifi_station_beacon_loss_total{device="wlan0"} 1
|
||||||
|
# HELP node_wifi_station_connected_seconds_total The total number of seconds a station has been connected to an access point.
|
||||||
|
# TYPE node_wifi_station_connected_seconds_total counter
|
||||||
|
node_wifi_station_connected_seconds_total{device="wlan0"} 30
|
||||||
|
# HELP node_wifi_station_inactive_seconds The number of seconds since any wireless activity has occurred on a station.
|
||||||
|
# TYPE node_wifi_station_inactive_seconds gauge
|
||||||
|
node_wifi_station_inactive_seconds{device="wlan0"} 0.4
|
||||||
|
# HELP node_wifi_station_receive_bits_per_second The current WiFi receive bitrate of a station, in bits per second.
|
||||||
|
# TYPE node_wifi_station_receive_bits_per_second gauge
|
||||||
|
node_wifi_station_receive_bits_per_second{device="wlan0"} 1.28e+08
|
||||||
|
# HELP node_wifi_station_signal_dbm The current WiFi signal strength, in decibel-milliwatts (dBm).
|
||||||
|
# TYPE node_wifi_station_signal_dbm gauge
|
||||||
|
node_wifi_station_signal_dbm{device="wlan0"} -52
|
||||||
|
# HELP node_wifi_station_transmit_bits_per_second The current WiFi transmit bitrate of a station, in bits per second.
|
||||||
|
# TYPE node_wifi_station_transmit_bits_per_second gauge
|
||||||
|
node_wifi_station_transmit_bits_per_second{device="wlan0"} 1.64e+08
|
||||||
|
# HELP node_wifi_station_transmit_failed_total The total number of times a station has failed to send a packet.
|
||||||
|
# TYPE node_wifi_station_transmit_failed_total counter
|
||||||
|
node_wifi_station_transmit_failed_total{device="wlan0"} 2
|
||||||
|
# HELP node_wifi_station_transmit_retries_total The total number of times a station has had to retry while sending a packet.
|
||||||
|
# TYPE node_wifi_station_transmit_retries_total counter
|
||||||
|
node_wifi_station_transmit_retries_total{device="wlan0"} 10
|
||||||
# HELP node_zfsArc_anon_evictable_data kstat.zfs.misc.arcstats.anon_evictable_data
|
# HELP node_zfsArc_anon_evictable_data kstat.zfs.misc.arcstats.anon_evictable_data
|
||||||
# TYPE node_zfsArc_anon_evictable_data untyped
|
# TYPE node_zfsArc_anon_evictable_data untyped
|
||||||
node_zfsArc_anon_evictable_data 0
|
node_zfsArc_anon_evictable_data 0
|
||||||
|
|
10
collector/fixtures/wifi/interfaces.json
Normal file
10
collector/fixtures/wifi/interfaces.json
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"name": "wlan0",
|
||||||
|
"type": 2,
|
||||||
|
"frequency": 2412
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": 10
|
||||||
|
}
|
||||||
|
]
|
10
collector/fixtures/wifi/wlan0/stationinfo.json
Normal file
10
collector/fixtures/wifi/wlan0/stationinfo.json
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
{
|
||||||
|
"connected": 30000000000,
|
||||||
|
"inactive": 400000000,
|
||||||
|
"receivebitrate": 128000000,
|
||||||
|
"transmitbitrate": 164000000,
|
||||||
|
"signal": -52,
|
||||||
|
"transmitretries": 10,
|
||||||
|
"transmitfailed": 2,
|
||||||
|
"beaconloss": 1
|
||||||
|
}
|
281
collector/wifi_linux.go
Normal file
281
collector/wifi_linux.go
Normal file
|
@ -0,0 +1,281 @@
|
||||||
|
// Copyright 2017 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.
|
||||||
|
|
||||||
|
package collector
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"io/ioutil"
|
||||||
|
"path/filepath"
|
||||||
|
|
||||||
|
"github.com/mdlayher/wifi"
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
)
|
||||||
|
|
||||||
|
type wifiCollector struct {
|
||||||
|
InterfaceFrequencyHertz *prometheus.Desc
|
||||||
|
|
||||||
|
StationConnectedSecondsTotal *prometheus.Desc
|
||||||
|
StationInactiveSeconds *prometheus.Desc
|
||||||
|
StationReceiveBitsPerSecond *prometheus.Desc
|
||||||
|
StationTransmitBitsPerSecond *prometheus.Desc
|
||||||
|
StationSignalDBM *prometheus.Desc
|
||||||
|
StationTransmitRetriesTotal *prometheus.Desc
|
||||||
|
StationTransmitFailedTotal *prometheus.Desc
|
||||||
|
StationBeaconLossTotal *prometheus.Desc
|
||||||
|
|
||||||
|
stat wifiStater
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
collectorWifi = flag.String("collector.wifi", "", "test fixtures to use for wifi collector metrics")
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
Factories["wifi"] = NewWifiCollector
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ wifiStater = &wifi.Client{}
|
||||||
|
|
||||||
|
// wifiStater is an interface used to swap out a *wifi.Client for end to end tests.
|
||||||
|
type wifiStater interface {
|
||||||
|
Interfaces() ([]*wifi.Interface, error)
|
||||||
|
StationInfo(ifi *wifi.Interface) (*wifi.StationInfo, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWifiCollector() (Collector, error) {
|
||||||
|
stat, err := newWifiStater(*collectorWifi)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to access wifi data: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
const (
|
||||||
|
subsystem = "wifi"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
labels = []string{"device"}
|
||||||
|
)
|
||||||
|
|
||||||
|
return &wifiCollector{
|
||||||
|
InterfaceFrequencyHertz: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "interface_frequency_hertz"),
|
||||||
|
"The current frequency a WiFi interface is operating at, in hertz.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationConnectedSecondsTotal: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_connected_seconds_total"),
|
||||||
|
"The total number of seconds a station has been connected to an access point.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationInactiveSeconds: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_inactive_seconds"),
|
||||||
|
"The number of seconds since any wireless activity has occurred on a station.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationReceiveBitsPerSecond: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_receive_bits_per_second"),
|
||||||
|
"The current WiFi receive bitrate of a station, in bits per second.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationTransmitBitsPerSecond: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_transmit_bits_per_second"),
|
||||||
|
"The current WiFi transmit bitrate of a station, in bits per second.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationSignalDBM: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_signal_dbm"),
|
||||||
|
"The current WiFi signal strength, in decibel-milliwatts (dBm).",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationTransmitRetriesTotal: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_transmit_retries_total"),
|
||||||
|
"The total number of times a station has had to retry while sending a packet.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationTransmitFailedTotal: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_transmit_failed_total"),
|
||||||
|
"The total number of times a station has failed to send a packet.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
StationBeaconLossTotal: prometheus.NewDesc(
|
||||||
|
prometheus.BuildFQName(Namespace, subsystem, "station_beacon_loss_total"),
|
||||||
|
"The total number of times a station has detected a beacon loss.",
|
||||||
|
labels,
|
||||||
|
nil,
|
||||||
|
),
|
||||||
|
|
||||||
|
stat: stat,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *wifiCollector) Update(ch chan<- prometheus.Metric) error {
|
||||||
|
ifis, err := c.stat.Interfaces()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to retrieve wifi interfaces: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, ifi := range ifis {
|
||||||
|
// Only collect metrics on stations for now
|
||||||
|
if ifi.Type != wifi.InterfaceTypeStation {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
info, err := c.stat.StationInfo(ifi)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("failed to retrieve station info for device %s: %v",
|
||||||
|
ifi.Name, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.InterfaceFrequencyHertz,
|
||||||
|
prometheus.GaugeValue,
|
||||||
|
mHzToHz(ifi.Frequency),
|
||||||
|
ifi.Name,
|
||||||
|
)
|
||||||
|
|
||||||
|
c.updateStationStats(ch, ifi.Name, info)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *wifiCollector) updateStationStats(ch chan<- prometheus.Metric, device string, info *wifi.StationInfo) {
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationConnectedSecondsTotal,
|
||||||
|
prometheus.CounterValue,
|
||||||
|
info.Connected.Seconds(),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationInactiveSeconds,
|
||||||
|
prometheus.GaugeValue,
|
||||||
|
info.Inactive.Seconds(),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationReceiveBitsPerSecond,
|
||||||
|
prometheus.GaugeValue,
|
||||||
|
float64(info.ReceiveBitrate),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationTransmitBitsPerSecond,
|
||||||
|
prometheus.GaugeValue,
|
||||||
|
float64(info.TransmitBitrate),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationSignalDBM,
|
||||||
|
prometheus.GaugeValue,
|
||||||
|
float64(info.Signal),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationTransmitRetriesTotal,
|
||||||
|
prometheus.CounterValue,
|
||||||
|
float64(info.TransmitRetries),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationTransmitFailedTotal,
|
||||||
|
prometheus.CounterValue,
|
||||||
|
float64(info.TransmitFailed),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
|
||||||
|
ch <- prometheus.MustNewConstMetric(
|
||||||
|
c.StationBeaconLossTotal,
|
||||||
|
prometheus.CounterValue,
|
||||||
|
float64(info.BeaconLoss),
|
||||||
|
device,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func mHzToHz(mHz int) float64 {
|
||||||
|
return float64(mHz) * 1000 * 1000
|
||||||
|
}
|
||||||
|
|
||||||
|
// All code below this point is used to assist with end-to-end tests for
|
||||||
|
// the wifi collector, since wifi devices are not available in CI.
|
||||||
|
|
||||||
|
// newWifiStater determines if mocked test fixtures from files should be used for
|
||||||
|
// collecting wifi metrics, or if package wifi should be used.
|
||||||
|
func newWifiStater(fixtures string) (wifiStater, error) {
|
||||||
|
if fixtures != "" {
|
||||||
|
return &mockWifiStater{
|
||||||
|
fixtures: fixtures,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return wifi.New()
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ wifiStater = &mockWifiStater{}
|
||||||
|
|
||||||
|
type mockWifiStater struct {
|
||||||
|
fixtures string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *mockWifiStater) unmarshalJSONFile(filename string, v interface{}) error {
|
||||||
|
b, err := ioutil.ReadFile(filepath.Join(s.fixtures, filename))
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return json.Unmarshal(b, v)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *mockWifiStater) Interfaces() ([]*wifi.Interface, error) {
|
||||||
|
var ifis []*wifi.Interface
|
||||||
|
if err := s.unmarshalJSONFile("interfaces.json", &ifis); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return ifis, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *mockWifiStater) StationInfo(ifi *wifi.Interface) (*wifi.StationInfo, error) {
|
||||||
|
p := filepath.Join(ifi.Name, "stationinfo.json")
|
||||||
|
|
||||||
|
var info wifi.StationInfo
|
||||||
|
if err := s.unmarshalJSONFile(p, &info); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return &info, nil
|
||||||
|
}
|
|
@ -24,6 +24,7 @@ collectors=$(cat << COLLECTORS
|
||||||
textfile
|
textfile
|
||||||
bonding
|
bonding
|
||||||
megacli
|
megacli
|
||||||
|
wifi
|
||||||
zfs
|
zfs
|
||||||
COLLECTORS
|
COLLECTORS
|
||||||
)
|
)
|
||||||
|
@ -70,6 +71,7 @@ fi
|
||||||
-collectors.enabled="$(echo ${collectors} | tr ' ' ',')" \
|
-collectors.enabled="$(echo ${collectors} | tr ' ' ',')" \
|
||||||
-collector.textfile.directory="collector/fixtures/textfile/two_metric_files/" \
|
-collector.textfile.directory="collector/fixtures/textfile/two_metric_files/" \
|
||||||
-collector.megacli.command="collector/fixtures/megacli" \
|
-collector.megacli.command="collector/fixtures/megacli" \
|
||||||
|
-collector.wifi="collector/fixtures/wifi" \
|
||||||
-web.listen-address "127.0.0.1:${port}" \
|
-web.listen-address "127.0.0.1:${port}" \
|
||||||
-log.level="debug" > "${tmpdir}/node_exporter.log" 2>&1 &
|
-log.level="debug" > "${tmpdir}/node_exporter.log" 2>&1 &
|
||||||
|
|
||||||
|
|
5
vendor/github.com/mdlayher/netlink/conn.go
generated
vendored
5
vendor/github.com/mdlayher/netlink/conn.go
generated
vendored
|
@ -2,7 +2,6 @@ package netlink
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"math"
|
|
||||||
"os"
|
"os"
|
||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
)
|
)
|
||||||
|
@ -102,8 +101,8 @@ func (c *Conn) Execute(m Message) ([]Message, error) {
|
||||||
func (c *Conn) Send(m Message) (Message, error) {
|
func (c *Conn) Send(m Message) (Message, error) {
|
||||||
ml := nlmsgLength(len(m.Data))
|
ml := nlmsgLength(len(m.Data))
|
||||||
|
|
||||||
// TODO(mdlayher): fine-tune this limit. ~4GiB is a huge message.
|
// TODO(mdlayher): fine-tune this limit.
|
||||||
if ml > math.MaxUint32 {
|
if ml > (1024 * 32) {
|
||||||
return Message{}, errors.New("netlink message data too large")
|
return Message{}, errors.New("netlink message data too large")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
2
vendor/github.com/mdlayher/netlink/sockopt_linux_386.go
generated
vendored
2
vendor/github.com/mdlayher/netlink/sockopt_linux_386.go
generated
vendored
|
@ -4,7 +4,7 @@
|
||||||
|
|
||||||
// +build linux,386
|
// +build linux,386
|
||||||
|
|
||||||
package raw
|
package netlink
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"syscall"
|
"syscall"
|
||||||
|
|
6
vendor/vendor.json
vendored
6
vendor/vendor.json
vendored
|
@ -51,10 +51,10 @@
|
||||||
"revisionTime": "2016-04-24T11:30:07Z"
|
"revisionTime": "2016-04-24T11:30:07Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"checksumSHA1": "dG2VS6m8AS9wvpZh7aSx5AX001U=",
|
"checksumSHA1": "87nUxyFGVJFXB6MQpGCGUHi5NY0=",
|
||||||
"path": "github.com/mdlayher/netlink",
|
"path": "github.com/mdlayher/netlink",
|
||||||
"revision": "1291b75abe0cc0cb335f110466bf1f02590c916d",
|
"revision": "a65cbc3bb3f7a793b7d79ad7d19b16d471ddbd78",
|
||||||
"revisionTime": "2017-01-04T04:59:06Z"
|
"revisionTime": "2017-01-10T22:29:47Z"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"checksumSHA1": "+2roeIWCAjCC58tZcs12Vqgf1Io=",
|
"checksumSHA1": "+2roeIWCAjCC58tZcs12Vqgf1Io=",
|
||||||
|
|
Loading…
Reference in a new issue