Skip to content

Commit 8f3254a

Browse files
committed
netstat_freebsd: refactoring
Make the code testable and more maintainable: - Encapsulate all the CGO code so it can be tested - Fix the test file (it wasn't compiling) and improve tests - Mock the data returned by unix.SysctlRaw so the code can be tested without actually calling sysctl. No change in behavior intended
1 parent 43fb05c commit 8f3254a

File tree

2 files changed

+107
-46
lines changed

2 files changed

+107
-46
lines changed

collector/netstat_freebsd.go

Lines changed: 82 additions & 36 deletions
Original file line numberDiff line numberDiff line change
@@ -37,19 +37,63 @@ import (
3737
import "C"
3838

3939
var (
40-
bsdNetstatTcpSendPacketsTotal = prometheus.NewDesc(
41-
prometheus.BuildFQName(namespace, "netstat", "tcp_transmit_packets_total"),
42-
"TCP packets sent",
43-
nil, nil,
44-
)
45-
46-
bsdNetstatTcpRecvPacketsTotal = prometheus.NewDesc(
47-
prometheus.BuildFQName(namespace, "netstat", "tcp_receive_packets_total"),
48-
"TCP packets received",
49-
nil, nil,
50-
)
40+
sysctlRaw = unix.SysctlRaw
41+
tcpSendTotal = "bsdNetstatTcpSendPacketsTotal"
42+
tcpRecvTotal = "bsdNetstatTcpRecvPacketsTotal"
43+
44+
counterMetrics = map[string]*prometheus.Desc{
45+
tcpSendTotal: prometheus.NewDesc(
46+
prometheus.BuildFQName(namespace, "netstat", "tcp_transmit_packets_total"),
47+
"TCP packets sent", nil, nil),
48+
tcpRecvTotal: prometheus.NewDesc(
49+
prometheus.BuildFQName(namespace, "netstat", "tcp_receive_packets_total"),
50+
"TCP packets received", nil, nil),
51+
}
5152
)
5253

54+
type NetstatData struct {
55+
structSize int
56+
sysctl string
57+
}
58+
59+
type NetstatMetrics map[string]float64
60+
61+
type NetstatTCPData NetstatData
62+
63+
func NewTCPStat() *NetstatTCPData {
64+
return &NetstatTCPData{
65+
structSize: int(unsafe.Sizeof(C.struct_tcpstat{})),
66+
sysctl: "net.inet.tcp.stats",
67+
}
68+
}
69+
70+
func (netstatMetric *NetstatTCPData) GetData() (NetstatMetrics, error) {
71+
data, err := getData(netstatMetric.sysctl, netstatMetric.structSize)
72+
if err != nil {
73+
return nil, err
74+
}
75+
76+
tcpStats := *(*C.struct_tcpstat)(unsafe.Pointer(&data[0]))
77+
78+
return NetstatMetrics{
79+
tcpSendTotal: float64(tcpStats.tcps_sndtotal),
80+
tcpRecvTotal: float64(tcpStats.tcps_rcvtotal),
81+
}, nil
82+
}
83+
84+
func getData(queryString string, expectedSize int) ([]byte, error) {
85+
data, err := sysctlRaw(queryString)
86+
if err != nil {
87+
fmt.Println("Error:", err)
88+
return nil, err
89+
}
90+
91+
if len(data) < expectedSize {
92+
return nil, errors.New("Data Size mismatch")
93+
}
94+
return data, nil
95+
}
96+
5397
type netStatCollector struct {
5498
netStatMetric *prometheus.Desc
5599
}
@@ -70,39 +114,41 @@ func (c *netStatCollector) Collect(ch chan<- prometheus.Metric) {
70114
_ = c.Update(ch)
71115
}
72116

73-
func getData(queryString string) ([]byte, error) {
74-
data, err := unix.SysctlRaw(queryString)
117+
func (c *netStatCollector) Update(ch chan<- prometheus.Metric) error {
118+
tcpStats, err := NewTCPStat().GetData()
75119
if err != nil {
76-
fmt.Println("Error:", err)
77-
return nil, err
120+
return err
78121
}
79122

80-
if len(data) < int(unsafe.Sizeof(C.struct_tcpstat{})) {
81-
return nil, errors.New("Data Size mismatch")
82-
}
83-
return data, nil
84-
}
123+
allStats := make(map[string]float64)
85124

86-
func (c *netStatCollector) Update(ch chan<- prometheus.Metric) error {
125+
for k, v := range tcpStats {
126+
allStats[k] = v
127+
}
87128

88-
tcpData, err := getData("net.inet.tcp.stats")
89-
if err != nil {
90-
return err
129+
for metricKey, metricData := range counterMetrics {
130+
ch <- prometheus.MustNewConstMetric(
131+
metricData,
132+
prometheus.CounterValue,
133+
allStats[metricKey],
134+
)
91135
}
92136

93-
tcpStats := *(*C.struct_tcpstat)(unsafe.Pointer(&tcpData[0]))
137+
return nil
138+
}
94139

95-
ch <- prometheus.MustNewConstMetric(
96-
bsdNetstatTcpSendPacketsTotal,
97-
prometheus.CounterValue,
98-
float64(tcpStats.tcps_sndtotal),
99-
)
140+
// Used by tests to mock unix.SysctlRaw
141+
func getFreeBSDDataMock(sysctl string) []byte {
100142

101-
ch <- prometheus.MustNewConstMetric(
102-
bsdNetstatTcpRecvPacketsTotal,
103-
prometheus.CounterValue,
104-
float64(tcpStats.tcps_rcvtotal),
105-
)
143+
if sysctl == "net.inet.tcp.stats" {
144+
tcpStats := C.struct_tcpstat{
145+
tcps_sndtotal: 1234,
146+
tcps_rcvtotal: 4321,
147+
}
148+
size := int(unsafe.Sizeof(C.struct_tcpstat{}))
106149

107-
return nil
150+
return unsafe.Slice((*byte)(unsafe.Pointer(&tcpStats)), size)
151+
}
152+
153+
return make([]byte, 0, 0)
108154
}

collector/netstat_freebsd_test.go

Lines changed: 25 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,16 @@ package collector
1818

1919
import (
2020
"github.com/prometheus/client_golang/prometheus"
21-
"golang.org/x/sys/unix"
2221
"testing"
23-
"unsafe"
2422
)
2523

24+
func testSetup() {
25+
sysctlRaw = func(name string, _ ...int) ([]byte, error) {
26+
mockData := getFreeBSDDataMock(name)
27+
return mockData, nil
28+
}
29+
}
30+
2631
func TestNetStatCollectorDescribe(t *testing.T) {
2732
ch := make(chan *prometheus.Desc, 1)
2833
collector := &netStatCollector{
@@ -31,24 +36,34 @@ func TestNetStatCollectorDescribe(t *testing.T) {
3136
collector.Describe(ch)
3237
desc := <-ch
3338

34-
if want, got := "dummy_metric", desc.String(); want != got {
39+
expected := "Desc{fqName: \"dummy_metric\", help: \"dummy\", constLabels: {}, variableLabels: {}}"
40+
if want, got := expected, desc.String(); want != got {
3541
t.Errorf("want %s, got %s", want, got)
3642
}
3743
}
3844

39-
func TestGetData(t *testing.T) {
40-
data, err := getData("net.inet.tcp.stats")
45+
func TestGetTCPMetrics(t *testing.T) {
46+
testSetup()
47+
48+
tcpData, err := NewTCPStat().GetData()
4149
if err != nil {
4250
t.Fatal("unexpected error:", err)
4351
}
4452

45-
if got, want := len(data), int(unsafe.Sizeof(unix.TCPStats{})); got < want {
46-
t.Errorf("data length too small: want >= %d, got %d", want, got)
53+
sndTotal := tcpData[tcpSendTotal]
54+
rcvTotal := tcpData[tcpRecvTotal]
55+
56+
if got, want := sndTotal, float64(1234); got != want {
57+
t.Errorf("unexpected sndTotal value: want %f, got %f", want, got)
58+
}
59+
60+
if got, want := rcvTotal, float64(4321); got != want {
61+
t.Errorf("unexpected rcvTotal value: want %f, got %f", want, got)
4762
}
4863
}
4964

5065
func TestNetStatCollectorUpdate(t *testing.T) {
51-
ch := make(chan prometheus.Metric, len(metrics))
66+
ch := make(chan prometheus.Metric, len(counterMetrics))
5267
collector := &netStatCollector{
5368
netStatMetric: prometheus.NewDesc("netstat_metric", "NetStat Metric", nil, nil),
5469
}
@@ -57,11 +72,11 @@ func TestNetStatCollectorUpdate(t *testing.T) {
5772
t.Fatal("unexpected error:", err)
5873
}
5974

60-
if got, want := len(ch), len(metrics); got != want {
75+
if got, want := len(ch), len(counterMetrics); got != want {
6176
t.Errorf("metric count mismatch: want %d, got %d", want, got)
6277
}
6378

64-
for range metrics {
79+
for range len(counterMetrics) {
6580
<-ch
6681
}
6782
}

0 commit comments

Comments
 (0)