mirror of
				https://github.com/prometheus/node_exporter.git
				synced 2025-08-20 18:33:52 -07:00 
			
		
		
		
	Merge 4bb3f0315d into be19d537cd
				
					
				
			This commit is contained in:
		
						commit
						78d2bd63c9
					
				| 
						 | 
					@ -17,6 +17,7 @@
 | 
				
			||||||
package collector
 | 
					package collector
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
 | 
						"encoding/binary"
 | 
				
			||||||
	"errors"
 | 
						"errors"
 | 
				
			||||||
	"fmt"
 | 
						"fmt"
 | 
				
			||||||
	"log/slog"
 | 
						"log/slog"
 | 
				
			||||||
| 
						 | 
					@ -33,23 +34,249 @@ import (
 | 
				
			||||||
#include <netinet/tcp.h>
 | 
					#include <netinet/tcp.h>
 | 
				
			||||||
#include <netinet/tcp_var.h>
 | 
					#include <netinet/tcp_var.h>
 | 
				
			||||||
#include <netinet/udp.h>
 | 
					#include <netinet/udp.h>
 | 
				
			||||||
 | 
					#include <netinet/udp_var.h>
 | 
				
			||||||
 | 
					#include <netinet/ip_var.h>
 | 
				
			||||||
 | 
					#include <netinet6/ip6_var.h>
 | 
				
			||||||
*/
 | 
					*/
 | 
				
			||||||
import "C"
 | 
					import "C"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
var (
 | 
					var (
 | 
				
			||||||
	bsdNetstatTcpSendPacketsTotal = prometheus.NewDesc(
 | 
						sysctlRaw              = unix.SysctlRaw
 | 
				
			||||||
		prometheus.BuildFQName(namespace, "netstat", "tcp_transmit_packets_total"),
 | 
						tcpSendTotal           = "bsdNetstatTcpSendPacketsTotal"
 | 
				
			||||||
		"TCP packets sent",
 | 
						tcpRecvTotal           = "bsdNetstatTcpRecvPacketsTotal"
 | 
				
			||||||
		nil, nil,
 | 
						udpSendTotal           = "bsdNetstatUdpSendPacketsTotal"
 | 
				
			||||||
	)
 | 
						udpRecvTotal           = "bsdNetstatUdpRecvPacketsTotal"
 | 
				
			||||||
 | 
						ipv4SendTotal          = "bsdNetstatIPv4SendPacketsTotal"
 | 
				
			||||||
 | 
						ipv4RawSendTotal       = "bsdNetstatIPv4RawSendPacketsTotal"
 | 
				
			||||||
 | 
						ipv4RecvTotal          = "bsdNetstatIPv4RecvPacketsTotal"
 | 
				
			||||||
 | 
						ipv4RecvFragmentsTotal = "bsdNetstatIPv4RecvFragmentsTotal"
 | 
				
			||||||
 | 
						ipv4ForwardTotal       = "bsdNetstatIPv4ForwardTotal"
 | 
				
			||||||
 | 
						ipv4FastForwardTotal   = "bsdNetstatIPv4FastForwardTotal"
 | 
				
			||||||
 | 
						ipv4DeliveredTotal     = "bsdNetstatIPv4DeliveredTotal"
 | 
				
			||||||
 | 
						ipv6SendTotal          = "bsdNetstatIPv6SendPacketsTotal"
 | 
				
			||||||
 | 
						ipv6RawSendTotal       = "bsdNetstatIPv6RawSendPacketsTotal"
 | 
				
			||||||
 | 
						ipv6RecvTotal          = "bsdNetstatIPv6RecvPacketsTotal"
 | 
				
			||||||
 | 
						ipv6RecvFragmentsTotal = "bsdNetstatIPv6RecvFragmentsTotal"
 | 
				
			||||||
 | 
						ipv6ForwardTotal       = "bsdNetstatIPv6ForwardTotal"
 | 
				
			||||||
 | 
						ipv6DeliveredTotal     = "bsdNetstatIPv6DeliveredTotal"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	bsdNetstatTcpRecvPacketsTotal = prometheus.NewDesc(
 | 
						tcpStates = []string{
 | 
				
			||||||
		prometheus.BuildFQName(namespace, "netstat", "tcp_receive_packets_total"),
 | 
							"CLOSED", "LISTEN", "SYN_SENT", "SYN_RCVD",
 | 
				
			||||||
		"TCP packets received",
 | 
							"ESTABLISHED", "CLOSE_WAIT", "FIN_WAIT_1", "CLOSING",
 | 
				
			||||||
		nil, nil,
 | 
							"LAST_ACK", "FIN_WAIT_2", "TIME_WAIT",
 | 
				
			||||||
	)
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tcpStatesMetric = prometheus.NewDesc(
 | 
				
			||||||
 | 
							prometheus.BuildFQName(namespace, "netstat", "tcp_connections"),
 | 
				
			||||||
 | 
							"Number of TCP connections per state", []string{"state"}, nil)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						counterMetrics = map[string]*prometheus.Desc{
 | 
				
			||||||
 | 
							// TCP stats
 | 
				
			||||||
 | 
							tcpSendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "tcp_transmit_packets_total"),
 | 
				
			||||||
 | 
								"TCP packets sent", nil, nil),
 | 
				
			||||||
 | 
							tcpRecvTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "tcp_receive_packets_total"),
 | 
				
			||||||
 | 
								"TCP packets received", nil, nil),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// UDP stats
 | 
				
			||||||
 | 
							udpSendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "udp_transmit_packets_total"),
 | 
				
			||||||
 | 
								"UDP packets sent", nil, nil),
 | 
				
			||||||
 | 
							udpRecvTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "udp_receive_packets_total"),
 | 
				
			||||||
 | 
								"UDP packets received", nil, nil),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// IPv4 stats
 | 
				
			||||||
 | 
							ipv4SendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_transmit_packets_total"),
 | 
				
			||||||
 | 
								"IPv4 packets sent from this host", nil, nil),
 | 
				
			||||||
 | 
							ipv4RawSendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_transmit_raw_packets_total"),
 | 
				
			||||||
 | 
								"IPv4 raw packets generated", nil, nil),
 | 
				
			||||||
 | 
							ipv4RecvTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_receive_packets_total"),
 | 
				
			||||||
 | 
								"IPv4 packets received", nil, nil),
 | 
				
			||||||
 | 
							ipv4RecvFragmentsTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_receive_fragments_total"),
 | 
				
			||||||
 | 
								"IPv4 fragments received", nil, nil),
 | 
				
			||||||
 | 
							ipv4ForwardTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_forward_total"),
 | 
				
			||||||
 | 
								"IPv4 packets forwarded", nil, nil),
 | 
				
			||||||
 | 
							ipv4FastForwardTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_fast_forward_total"),
 | 
				
			||||||
 | 
								"IPv4 packets fast forwarded", nil, nil),
 | 
				
			||||||
 | 
							ipv4DeliveredTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip4_delivered_total"),
 | 
				
			||||||
 | 
								"IPv4 packets delivered to the upper layer (packets for this host)", nil, nil),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// IPv6 stats
 | 
				
			||||||
 | 
							ipv6SendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_transmit_packets_total"),
 | 
				
			||||||
 | 
								"IPv6 packets sent from this host", nil, nil),
 | 
				
			||||||
 | 
							ipv6RawSendTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_transmit_raw_packets_total"),
 | 
				
			||||||
 | 
								"IPv6 raw packets generated", nil, nil),
 | 
				
			||||||
 | 
							ipv6RecvTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_receive_packets_total"),
 | 
				
			||||||
 | 
								"IPv6 packets received", nil, nil),
 | 
				
			||||||
 | 
							ipv6RecvFragmentsTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_receive_fragments_total"),
 | 
				
			||||||
 | 
								"IPv6 fragments received", nil, nil),
 | 
				
			||||||
 | 
							ipv6ForwardTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_forward_total"),
 | 
				
			||||||
 | 
								"IPv6 packets forwarded", nil, nil),
 | 
				
			||||||
 | 
							ipv6DeliveredTotal: prometheus.NewDesc(
 | 
				
			||||||
 | 
								prometheus.BuildFQName(namespace, "netstat", "ip6_delivered_total"),
 | 
				
			||||||
 | 
								"IPv6 packets delivered to the upper layer (packets for this host)", nil, nil),
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatData struct {
 | 
				
			||||||
 | 
						structSize int
 | 
				
			||||||
 | 
						sysctl     string
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatMetrics map[string]float64
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatTCPData NetstatData
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func NewTCPStat() *NetstatTCPData {
 | 
				
			||||||
 | 
						return &NetstatTCPData{
 | 
				
			||||||
 | 
							structSize: int(unsafe.Sizeof(C.struct_tcpstat{})),
 | 
				
			||||||
 | 
							sysctl:     "net.inet.tcp.stats",
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (netstatMetric *NetstatTCPData) GetData() (NetstatMetrics, error) {
 | 
				
			||||||
 | 
						data, err := getData(netstatMetric.sysctl, netstatMetric.structSize)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tcpStats := *(*C.struct_tcpstat)(unsafe.Pointer(&data[0]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return NetstatMetrics{
 | 
				
			||||||
 | 
							tcpSendTotal: float64(tcpStats.tcps_sndtotal),
 | 
				
			||||||
 | 
							tcpRecvTotal: float64(tcpStats.tcps_rcvtotal),
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatUDPData NetstatData
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func NewUDPStat() *NetstatUDPData {
 | 
				
			||||||
 | 
						return &NetstatUDPData{
 | 
				
			||||||
 | 
							structSize: int(unsafe.Sizeof(C.struct_udpstat{})),
 | 
				
			||||||
 | 
							sysctl:     "net.inet.udp.stats",
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (netstatMetric *NetstatUDPData) GetData() (NetstatMetrics, error) {
 | 
				
			||||||
 | 
						data, err := getData(netstatMetric.sysctl, netstatMetric.structSize)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						udpStats := *(*C.struct_udpstat)(unsafe.Pointer(&data[0]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return NetstatMetrics{
 | 
				
			||||||
 | 
							udpSendTotal: float64(udpStats.udps_opackets),
 | 
				
			||||||
 | 
							udpRecvTotal: float64(udpStats.udps_ipackets),
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatIPv4Data NetstatData
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func NewIPv4Stat() *NetstatIPv4Data {
 | 
				
			||||||
 | 
						return &NetstatIPv4Data{
 | 
				
			||||||
 | 
							structSize: int(unsafe.Sizeof(C.struct_ipstat{})),
 | 
				
			||||||
 | 
							sysctl:     "net.inet.ip.stats",
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (netstatMetric *NetstatIPv4Data) GetData() (NetstatMetrics, error) {
 | 
				
			||||||
 | 
						data, err := getData(netstatMetric.sysctl, netstatMetric.structSize)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipStats := *(*C.struct_ipstat)(unsafe.Pointer(&data[0]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return NetstatMetrics{
 | 
				
			||||||
 | 
							ipv4SendTotal:          float64(ipStats.ips_localout),
 | 
				
			||||||
 | 
							ipv4RawSendTotal:       float64(ipStats.ips_rawout),
 | 
				
			||||||
 | 
							ipv4RecvTotal:          float64(ipStats.ips_total),
 | 
				
			||||||
 | 
							ipv4RecvFragmentsTotal: float64(ipStats.ips_fragments),
 | 
				
			||||||
 | 
							ipv4ForwardTotal:       float64(ipStats.ips_forward),
 | 
				
			||||||
 | 
							ipv4FastForwardTotal:   float64(ipStats.ips_fastforward),
 | 
				
			||||||
 | 
							ipv4DeliveredTotal:     float64(ipStats.ips_delivered),
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					type NetstatIPv6Data NetstatData
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func NewIPv6Stat() *NetstatIPv6Data {
 | 
				
			||||||
 | 
						return &NetstatIPv6Data{
 | 
				
			||||||
 | 
							structSize: int(unsafe.Sizeof(C.struct_ipstat{})),
 | 
				
			||||||
 | 
							sysctl:     "net.inet6.ip6.stats",
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (netstatMetric *NetstatIPv6Data) GetData() (NetstatMetrics, error) {
 | 
				
			||||||
 | 
						data, err := getData(netstatMetric.sysctl, netstatMetric.structSize)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipStats := *(*C.struct_ip6stat)(unsafe.Pointer(&data[0]))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return NetstatMetrics{
 | 
				
			||||||
 | 
							ipv6SendTotal:          float64(ipStats.ip6s_localout),
 | 
				
			||||||
 | 
							ipv6RawSendTotal:       float64(ipStats.ip6s_rawout),
 | 
				
			||||||
 | 
							ipv6RecvTotal:          float64(ipStats.ip6s_total),
 | 
				
			||||||
 | 
							ipv6RecvFragmentsTotal: float64(ipStats.ip6s_fragments),
 | 
				
			||||||
 | 
							ipv6ForwardTotal:       float64(ipStats.ip6s_forward),
 | 
				
			||||||
 | 
							ipv6DeliveredTotal:     float64(ipStats.ip6s_delivered),
 | 
				
			||||||
 | 
						}, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getData(queryString string, expectedSize int) ([]byte, error) {
 | 
				
			||||||
 | 
						data, err := sysctlRaw(queryString)
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							fmt.Println("Error:", err)
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(data) < expectedSize {
 | 
				
			||||||
 | 
							return nil, errors.New("Data Size mismatch")
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return data, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func getTCPStates() ([]uint64, error) {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// This sysctl returns an array of uint64
 | 
				
			||||||
 | 
						data, err := sysctlRaw("net.inet.tcp.states")
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return nil, err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if len(data)/8 != len(tcpStates) {
 | 
				
			||||||
 | 
							return nil, fmt.Errorf("invalid TCP states data: expected %d entries, found %d", len(tcpStates), len(data)/8)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						states := make([]uint64, 0)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						offset := 0
 | 
				
			||||||
 | 
						for range len(tcpStates) {
 | 
				
			||||||
 | 
							s := data[offset : offset+8]
 | 
				
			||||||
 | 
							offset += 8
 | 
				
			||||||
 | 
							states = append(states, binary.NativeEndian.Uint64(s))
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						return states, nil
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
type netStatCollector struct {
 | 
					type netStatCollector struct {
 | 
				
			||||||
	netStatMetric *prometheus.Desc
 | 
						netStatMetric *prometheus.Desc
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -70,39 +297,120 @@ func (c *netStatCollector) Collect(ch chan<- prometheus.Metric) {
 | 
				
			||||||
	_ = c.Update(ch)
 | 
						_ = c.Update(ch)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func getData(queryString string) ([]byte, error) {
 | 
					 | 
				
			||||||
	data, err := unix.SysctlRaw(queryString)
 | 
					 | 
				
			||||||
	if err != nil {
 | 
					 | 
				
			||||||
		fmt.Println("Error:", err)
 | 
					 | 
				
			||||||
		return nil, err
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if len(data) < int(unsafe.Sizeof(C.struct_tcpstat{})) {
 | 
					 | 
				
			||||||
		return nil, errors.New("Data Size mismatch")
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
	return data, nil
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
func (c *netStatCollector) Update(ch chan<- prometheus.Metric) error {
 | 
					func (c *netStatCollector) Update(ch chan<- prometheus.Metric) error {
 | 
				
			||||||
 | 
						tcpStats, err := NewTCPStat().GetData()
 | 
				
			||||||
	tcpData, err := getData("net.inet.tcp.stats")
 | 
					 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		return err
 | 
							return err
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	tcpStats := *(*C.struct_tcpstat)(unsafe.Pointer(&tcpData[0]))
 | 
						udpStats, err := NewUDPStat().GetData()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ch <- prometheus.MustNewConstMetric(
 | 
						ipv4Stats, err := NewIPv4Stat().GetData()
 | 
				
			||||||
		bsdNetstatTcpSendPacketsTotal,
 | 
						if err != nil {
 | 
				
			||||||
		prometheus.CounterValue,
 | 
							return err
 | 
				
			||||||
		float64(tcpStats.tcps_sndtotal),
 | 
						}
 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	ch <- prometheus.MustNewConstMetric(
 | 
						ipv6Stats, err := NewIPv6Stat().GetData()
 | 
				
			||||||
		bsdNetstatTcpRecvPacketsTotal,
 | 
						if err != nil {
 | 
				
			||||||
		prometheus.CounterValue,
 | 
							return err
 | 
				
			||||||
		float64(tcpStats.tcps_rcvtotal),
 | 
						}
 | 
				
			||||||
	)
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						allStats := make(map[string]float64)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for k, v := range tcpStats {
 | 
				
			||||||
 | 
							allStats[k] = v
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for k, v := range udpStats {
 | 
				
			||||||
 | 
							allStats[k] = v
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for k, v := range ipv4Stats {
 | 
				
			||||||
 | 
							allStats[k] = v
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for k, v := range ipv6Stats {
 | 
				
			||||||
 | 
							allStats[k] = v
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for metricKey, metricData := range counterMetrics {
 | 
				
			||||||
 | 
							ch <- prometheus.MustNewConstMetric(
 | 
				
			||||||
 | 
								metricData,
 | 
				
			||||||
 | 
								prometheus.CounterValue,
 | 
				
			||||||
 | 
								allStats[metricKey],
 | 
				
			||||||
 | 
							)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tcpConnsPerStates, err := getTCPStates()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							return err
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for i, value := range tcpConnsPerStates {
 | 
				
			||||||
 | 
							ch <- prometheus.MustNewConstMetric(tcpStatesMetric, prometheus.GaugeValue, float64(value), tcpStates[i])
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
	return nil
 | 
						return nil
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Used by tests to mock unix.SysctlRaw
 | 
				
			||||||
 | 
					func getFreeBSDDataMock(sysctl string) []byte {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if sysctl == "net.inet.tcp.stats" {
 | 
				
			||||||
 | 
							tcpStats := C.struct_tcpstat{
 | 
				
			||||||
 | 
								tcps_sndtotal: 1234,
 | 
				
			||||||
 | 
								tcps_rcvtotal: 4321,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							size := int(unsafe.Sizeof(C.struct_tcpstat{}))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return unsafe.Slice((*byte)(unsafe.Pointer(&tcpStats)), size)
 | 
				
			||||||
 | 
						} else if sysctl == "net.inet.tcp.states" {
 | 
				
			||||||
 | 
							tcpStatesSlice := make([]byte, 0, len(tcpStates)*8)
 | 
				
			||||||
 | 
							tcpStatesValues := []uint64{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							for _, value := range tcpStatesValues {
 | 
				
			||||||
 | 
								tcpStatesSlice = binary.NativeEndian.AppendUint64(tcpStatesSlice, value)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return tcpStatesSlice
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						} else if sysctl == "net.inet.udp.stats" {
 | 
				
			||||||
 | 
							udpStats := C.struct_udpstat{
 | 
				
			||||||
 | 
								udps_opackets: 1234,
 | 
				
			||||||
 | 
								udps_ipackets: 4321,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							size := int(unsafe.Sizeof(C.struct_udpstat{}))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return unsafe.Slice((*byte)(unsafe.Pointer(&udpStats)), size)
 | 
				
			||||||
 | 
						} else if sysctl == "net.inet.ip.stats" {
 | 
				
			||||||
 | 
							ipStats := C.struct_ipstat{
 | 
				
			||||||
 | 
								ips_localout:    1234,
 | 
				
			||||||
 | 
								ips_rawout:      1235,
 | 
				
			||||||
 | 
								ips_total:       1236,
 | 
				
			||||||
 | 
								ips_fragments:   1237,
 | 
				
			||||||
 | 
								ips_forward:     1238,
 | 
				
			||||||
 | 
								ips_fastforward: 1239,
 | 
				
			||||||
 | 
								ips_delivered:   1240,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							size := int(unsafe.Sizeof(C.struct_ipstat{}))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return unsafe.Slice((*byte)(unsafe.Pointer(&ipStats)), size)
 | 
				
			||||||
 | 
						} else if sysctl == "net.inet6.ip6.stats" {
 | 
				
			||||||
 | 
							ipStats := C.struct_ip6stat{
 | 
				
			||||||
 | 
								ip6s_localout:  1234,
 | 
				
			||||||
 | 
								ip6s_rawout:    1235,
 | 
				
			||||||
 | 
								ip6s_total:     1236,
 | 
				
			||||||
 | 
								ip6s_fragments: 1237,
 | 
				
			||||||
 | 
								ip6s_forward:   1238,
 | 
				
			||||||
 | 
								ip6s_delivered: 1240,
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
							size := int(unsafe.Sizeof(C.struct_ip6stat{}))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return unsafe.Slice((*byte)(unsafe.Pointer(&ipStats)), size)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return make([]byte, 0, 0)
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -18,11 +18,16 @@ package collector
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
	"github.com/prometheus/client_golang/prometheus"
 | 
						"github.com/prometheus/client_golang/prometheus"
 | 
				
			||||||
	"golang.org/x/sys/unix"
 | 
					 | 
				
			||||||
	"testing"
 | 
						"testing"
 | 
				
			||||||
	"unsafe"
 | 
					 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func testSetup() {
 | 
				
			||||||
 | 
						sysctlRaw = func(name string, _ ...int) ([]byte, error) {
 | 
				
			||||||
 | 
							mockData := getFreeBSDDataMock(name)
 | 
				
			||||||
 | 
							return mockData, nil
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestNetStatCollectorDescribe(t *testing.T) {
 | 
					func TestNetStatCollectorDescribe(t *testing.T) {
 | 
				
			||||||
	ch := make(chan *prometheus.Desc, 1)
 | 
						ch := make(chan *prometheus.Desc, 1)
 | 
				
			||||||
	collector := &netStatCollector{
 | 
						collector := &netStatCollector{
 | 
				
			||||||
| 
						 | 
					@ -31,37 +36,149 @@ func TestNetStatCollectorDescribe(t *testing.T) {
 | 
				
			||||||
	collector.Describe(ch)
 | 
						collector.Describe(ch)
 | 
				
			||||||
	desc := <-ch
 | 
						desc := <-ch
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if want, got := "dummy_metric", desc.String(); want != got {
 | 
						expected := "Desc{fqName: \"dummy_metric\", help: \"dummy\", constLabels: {}, variableLabels: {}}"
 | 
				
			||||||
 | 
						if want, got := expected, desc.String(); want != got {
 | 
				
			||||||
		t.Errorf("want %s, got %s", want, got)
 | 
							t.Errorf("want %s, got %s", want, got)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestGetData(t *testing.T) {
 | 
					func TestGetTCPMetrics(t *testing.T) {
 | 
				
			||||||
	data, err := getData("net.inet.tcp.stats")
 | 
						testSetup()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tcpData, err := NewTCPStat().GetData()
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		t.Fatal("unexpected error:", err)
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if got, want := len(data), int(unsafe.Sizeof(unix.TCPStats{})); got < want {
 | 
						sndTotal := tcpData[tcpSendTotal]
 | 
				
			||||||
		t.Errorf("data length too small: want >= %d, got %d", want, got)
 | 
						rcvTotal := tcpData[tcpRecvTotal]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := sndTotal, float64(1234); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected sndTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := rcvTotal, float64(4321); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected rcvTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetTCPStatesMetrics(t *testing.T) {
 | 
				
			||||||
 | 
						testSetup()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tcpData, err := getTCPStates()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						expected := []uint64{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for i, value := range tcpData {
 | 
				
			||||||
 | 
							if got, want := float64(value), float64(expected[i]); got != want {
 | 
				
			||||||
 | 
								t.Errorf("unexpected %s value: want %f, got %f", tcpStates[i], want, got)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetUDPMetrics(t *testing.T) {
 | 
				
			||||||
 | 
						testSetup()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						udpData, err := NewUDPStat().GetData()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sndTotal := udpData[udpSendTotal]
 | 
				
			||||||
 | 
						rcvTotal := udpData[udpRecvTotal]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := sndTotal, float64(1234); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected sndTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := rcvTotal, float64(4321); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected rcvTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetIPv4Metrics(t *testing.T) {
 | 
				
			||||||
 | 
						testSetup()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipv4Data, err := NewIPv4Stat().GetData()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sndTotal := ipv4Data[ipv4SendTotal]
 | 
				
			||||||
 | 
						rcvTotal := ipv4Data[ipv4RecvTotal]
 | 
				
			||||||
 | 
						forwardTotal := ipv4Data[ipv4ForwardTotal]
 | 
				
			||||||
 | 
						deliveredTotal := ipv4Data[ipv4DeliveredTotal]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := sndTotal, float64(1234); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected sndTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := rcvTotal, float64(1236); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected rcvTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := forwardTotal, float64(1238); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected forwardTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := deliveredTotal, float64(1240); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected deliveredTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestGetIPv6Metrics(t *testing.T) {
 | 
				
			||||||
 | 
						testSetup()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ipv6Data, err := NewIPv6Stat().GetData()
 | 
				
			||||||
 | 
						if err != nil {
 | 
				
			||||||
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						sndTotal := ipv6Data[ipv6SendTotal]
 | 
				
			||||||
 | 
						rcvTotal := ipv6Data[ipv6RecvTotal]
 | 
				
			||||||
 | 
						forwardTotal := ipv6Data[ipv6ForwardTotal]
 | 
				
			||||||
 | 
						deliveredTotal := ipv6Data[ipv6DeliveredTotal]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := sndTotal, float64(1234); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected sndTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := rcvTotal, float64(1236); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected rcvTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := forwardTotal, float64(1238); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected forwardTotal value: want %f, got %f", want, got)
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if got, want := deliveredTotal, float64(1240); got != want {
 | 
				
			||||||
 | 
							t.Errorf("unexpected deliveredTotal value: want %f, got %f", want, got)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestNetStatCollectorUpdate(t *testing.T) {
 | 
					func TestNetStatCollectorUpdate(t *testing.T) {
 | 
				
			||||||
	ch := make(chan prometheus.Metric, len(metrics))
 | 
					 | 
				
			||||||
	collector := &netStatCollector{
 | 
						collector := &netStatCollector{
 | 
				
			||||||
		netStatMetric: prometheus.NewDesc("netstat_metric", "NetStat Metric", nil, nil),
 | 
							netStatMetric: prometheus.NewDesc("netstat_metric", "NetStat Metric", nil, nil),
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						totalMetrics := len(counterMetrics) + len(tcpStates)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ch := make(chan prometheus.Metric, totalMetrics)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	err := collector.Update(ch)
 | 
						err := collector.Update(ch)
 | 
				
			||||||
	if err != nil {
 | 
						if err != nil {
 | 
				
			||||||
		t.Fatal("unexpected error:", err)
 | 
							t.Fatal("unexpected error:", err)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if got, want := len(ch), len(metrics); got != want {
 | 
						if got, want := len(ch), totalMetrics; got != want {
 | 
				
			||||||
		t.Errorf("metric count mismatch: want %d, got %d", want, got)
 | 
							t.Fatalf("metric count mismatch: want %d, got %d", want, got)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for range metrics {
 | 
						for range totalMetrics {
 | 
				
			||||||
		<-ch
 | 
							<-ch
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue