From 862b410b58b0d223e39fb3b1f773963d06f6d26d Mon Sep 17 00:00:00 2001 From: Kakuya Ando Date: Mon, 19 Feb 2024 23:36:31 +0900 Subject: [PATCH] Refactoring of Scrape process, fixing multiple module issues (#1111) * Add SNMPScraper Interface for easier maintenance * Moved SNMP-related operations to the scraper side and simplified the ScrapeTarget function * Remove version retrieval from scraper and update to fetch from auth information * Add error handling for scraper used in worker * Add unit test for ScrapeTarget function using a mock SNMP scraper * Add test cases for scraping targets with different configurations --------- Signed-off-by: Kakuya Ando --- collector/collector.go | 190 +++++++++++++++++------------------- collector/collector_test.go | 108 ++++++++++++++++++++ scraper/gosnmp.go | 117 ++++++++++++++++++++++ scraper/mock.go | 84 ++++++++++++++++ scraper/scraper.go | 26 +++++ 5 files changed, 425 insertions(+), 100 deletions(-) create mode 100644 scraper/gosnmp.go create mode 100644 scraper/mock.go create mode 100644 scraper/scraper.go diff --git a/collector/collector.go b/collector/collector.go index 00f5b01f..cb1faed0 100644 --- a/collector/collector.go +++ b/collector/collector.go @@ -31,6 +31,7 @@ import ( "github.com/prometheus/client_golang/prometheus" "github.com/prometheus/snmp_exporter/config" + "github.com/prometheus/snmp_exporter/scraper" ) var ( @@ -77,74 +78,17 @@ func listToOid(l []int) string { } type ScrapeResults struct { - pdus []gosnmp.SnmpPDU - packets uint64 - retries uint64 + pdus []gosnmp.SnmpPDU } -func ScrapeTarget(ctx context.Context, target string, auth *config.Auth, module *config.Module, logger log.Logger, metrics Metrics) (ScrapeResults, error) { +func ScrapeTarget(snmp scraper.SNMPScraper, target string, auth *config.Auth, module *config.Module, logger log.Logger, metrics Metrics) (ScrapeResults, error) { results := ScrapeResults{} - // Set the options. - snmp := gosnmp.GoSNMP{} - snmp.Context = ctx - snmp.MaxRepetitions = module.WalkParams.MaxRepetitions - snmp.Retries = *module.WalkParams.Retries - snmp.Timeout = module.WalkParams.Timeout - snmp.UseUnconnectedUDPSocket = module.WalkParams.UseUnconnectedUDPSocket - snmp.LocalAddr = *srcAddress - - // Allow a set of OIDs that aren't in a strictly increasing order - if module.WalkParams.AllowNonIncreasingOIDs { - snmp.AppOpts = make(map[string]interface{}) - snmp.AppOpts["c"] = true - } - - var sent time.Time - snmp.OnSent = func(x *gosnmp.GoSNMP) { - sent = time.Now() - metrics.SNMPPackets.Inc() - results.packets++ - } - snmp.OnRecv = func(x *gosnmp.GoSNMP) { - metrics.SNMPDuration.Observe(time.Since(sent).Seconds()) - } - snmp.OnRetry = func(x *gosnmp.GoSNMP) { - metrics.SNMPRetries.Inc() - results.retries++ - } - - // Configure target. - if err := configureTarget(&snmp, target); err != nil { - return results, err - } - - // Configure auth. - auth.ConfigureSNMP(&snmp) - - // Do the actual walk. - getInitialStart := time.Now() - err := snmp.Connect() - if err != nil { - if err == context.Canceled { - return results, fmt.Errorf("scrape cancelled after %s (possible timeout) connecting to target %s", - time.Since(getInitialStart), snmp.Target) - } - return results, fmt.Errorf("error connecting to target %s: %s", target, err) - } - defer snmp.Conn.Close() - // Evaluate rules. newGet := module.Get newWalk := module.Walk for _, filter := range module.Filters { - var pdus []gosnmp.SnmpPDU allowedList := []string{} - - if snmp.Version == gosnmp.Version1 { - pdus, err = snmp.WalkAll(filter.Oid) - } else { - pdus, err = snmp.BulkWalkAll(filter.Oid) - } + pdus, err := snmp.WalkAll(filter.Oid) // Do not try to filter anything if we had errors. if err != nil { level.Info(logger).Log("msg", "Error getting OID, won't do any filter on this oid", "oid", filter.Oid) @@ -165,10 +109,11 @@ func ScrapeTarget(ctx context.Context, target string, auth *config.Auth, module newGet = newCfg } + version := auth.Version getOids := newGet maxOids := int(module.WalkParams.MaxRepetitions) // Max Repetition can be 0, maxOids cannot. SNMPv1 can only report one OID error per call. - if maxOids == 0 || snmp.Version == gosnmp.Version1 { + if maxOids == 0 || version == 1 { maxOids = 1 } for len(getOids) > 0 { @@ -177,19 +122,12 @@ func ScrapeTarget(ctx context.Context, target string, auth *config.Auth, module oids = maxOids } - level.Debug(logger).Log("msg", "Getting OIDs", "oids", oids) - getStart := time.Now() packet, err := snmp.Get(getOids[:oids]) if err != nil { - if err == context.Canceled { - return results, fmt.Errorf("scrape cancelled after %s (possible timeout) getting target %s", - time.Since(getInitialStart), snmp.Target) - } - return results, fmt.Errorf("error getting target %s: %s", snmp.Target, err) + return results, err } - level.Debug(logger).Log("msg", "Get of OIDs completed", "oids", oids, "duration_seconds", time.Since(getStart)) // SNMPv1 will return packet error for unsupported OIDs. - if packet.Error == gosnmp.NoSuchName && snmp.Version == gosnmp.Version1 { + if packet.Error == gosnmp.NoSuchName && version == 1 { level.Debug(logger).Log("msg", "OID not supported by target", "oids", getOids[0]) getOids = getOids[oids:] continue @@ -197,7 +135,7 @@ func ScrapeTarget(ctx context.Context, target string, auth *config.Auth, module // Response received with errors. // TODO: "stringify" gosnmp errors instead of showing error code. if packet.Error != gosnmp.NoError { - return results, fmt.Errorf("error reported by target %s: Error Status %d", snmp.Target, packet.Error) + return results, fmt.Errorf("error reported by target %s: Error Status %d", target, packet.Error) } for _, v := range packet.Variables { if v.Type == gosnmp.NoSuchObject || v.Type == gosnmp.NoSuchInstance { @@ -210,23 +148,10 @@ func ScrapeTarget(ctx context.Context, target string, auth *config.Auth, module } for _, subtree := range newWalk { - var pdus []gosnmp.SnmpPDU - level.Debug(logger).Log("msg", "Walking subtree", "oid", subtree) - walkStart := time.Now() - if snmp.Version == gosnmp.Version1 { - pdus, err = snmp.WalkAll(subtree) - } else { - pdus, err = snmp.BulkWalkAll(subtree) - } + pdus, err := snmp.WalkAll(subtree) if err != nil { - if err == context.Canceled { - return results, fmt.Errorf("scrape canceled after %s (possible timeout) walking target %s", - time.Since(getInitialStart), snmp.Target) - } - return results, fmt.Errorf("error walking target %s: %s", snmp.Target, err) + return results, err } - level.Debug(logger).Log("msg", "Walk of subtree completed", "oid", subtree, "duration_seconds", time.Since(walkStart)) - results.pdus = append(results.pdus, pdus...) } return results, nil @@ -384,11 +309,44 @@ func (c Collector) Describe(ch chan<- *prometheus.Desc) { ch <- prometheus.NewDesc("dummy", "dummy", nil, nil) } -func (c Collector) collect(ch chan<- prometheus.Metric, module *NamedModule) { - logger := log.With(c.logger, "module", module.name) +func (c Collector) collect(ch chan<- prometheus.Metric, logger log.Logger, client scraper.SNMPScraper, module *NamedModule) { + var ( + packets uint64 + retries uint64 + ) + client.SetOptions( + // Set the metrics options. + func(g *gosnmp.GoSNMP) { + var sent time.Time + g.OnSent = func(x *gosnmp.GoSNMP) { + sent = time.Now() + c.metrics.SNMPPackets.Inc() + packets++ + } + g.OnRecv = func(x *gosnmp.GoSNMP) { + c.metrics.SNMPDuration.Observe(time.Since(sent).Seconds()) + } + g.OnRetry = func(x *gosnmp.GoSNMP) { + c.metrics.SNMPRetries.Inc() + retries++ + } + }, + // Set the Walk options. + func(g *gosnmp.GoSNMP) { + g.Retries = *module.WalkParams.Retries + g.Timeout = module.WalkParams.Timeout + g.MaxRepetitions = module.WalkParams.MaxRepetitions + g.UseUnconnectedUDPSocket = module.WalkParams.UseUnconnectedUDPSocket + if module.WalkParams.AllowNonIncreasingOIDs { + g.AppOpts = map[string]interface{}{ + "c": true, + } + } + }, + ) start := time.Now() - results, err := ScrapeTarget(c.ctx, c.target, c.auth, module.Module, logger, c.metrics) moduleLabel := prometheus.Labels{"module": module.name} + results, err := ScrapeTarget(client, c.target, c.auth, module.Module, logger, c.metrics) if err != nil { level.Info(logger).Log("msg", "Error scraping target", "err", err) ch <- prometheus.NewInvalidMetric(prometheus.NewDesc("snmp_error", "Error scraping target", nil, moduleLabel), err) @@ -401,15 +359,16 @@ func (c Collector) collect(ch chan<- prometheus.Metric, module *NamedModule) { ch <- prometheus.MustNewConstMetric( prometheus.NewDesc("snmp_scrape_packets_sent", "Packets sent for get, bulkget, and walk; including retries.", nil, moduleLabel), prometheus.GaugeValue, - float64(results.packets)) + float64(packets)) ch <- prometheus.MustNewConstMetric( prometheus.NewDesc("snmp_scrape_packets_retried", "Packets retried for get, bulkget, and walk.", nil, moduleLabel), prometheus.GaugeValue, - float64(results.retries)) + float64(retries)) ch <- prometheus.MustNewConstMetric( prometheus.NewDesc("snmp_scrape_pdus_returned", "PDUs returned from get, bulkget, and walk.", nil, moduleLabel), prometheus.GaugeValue, float64(len(results.pdus))) + oidToPdu := make(map[string]gosnmp.SnmpPDU, len(results.pdus)) for _, pdu := range results.pdus { oidToPdu[pdu.Name[1:]] = pdu @@ -417,7 +376,6 @@ func (c Collector) collect(ch chan<- prometheus.Metric, module *NamedModule) { metricTree := buildMetricTree(module.Metrics) // Look for metrics that match each pdu. -PduLoop: for oid, pdu := range oidToPdu { head := metricTree oidList := oidToList(oid) @@ -425,7 +383,7 @@ PduLoop: var ok bool head, ok = head.children[o] if !ok { - continue PduLoop + break } if head.metric != nil { // Found a match. @@ -450,25 +408,57 @@ func (c Collector) Collect(ch chan<- prometheus.Metric) { if workerCount < 1 { workerCount = 1 } + ctx, cancel := context.WithCancel(c.ctx) + defer cancel() workerChan := make(chan *NamedModule) for i := 0; i < workerCount; i++ { wg.Add(1) - go func() { + go func(i int) { defer wg.Done() + logger := log.With(c.logger, "worker", i) + client, err := scraper.NewGoSNMP(logger, c.target, *srcAddress) + if err != nil { + level.Info(logger).Log("msg", err) + cancel() + ch <- prometheus.NewInvalidMetric(prometheus.NewDesc("snmp_error", "Error during initialisation of the Worker", nil, nil), err) + return + } + // Set the options. + client.SetOptions(func(g *gosnmp.GoSNMP) { + g.Context = ctx + c.auth.ConfigureSNMP(g) + }) + if err = client.Connect(); err != nil { + level.Info(logger).Log("msg", "Error connecting to target", "err", err) + ch <- prometheus.NewInvalidMetric(prometheus.NewDesc("snmp_error", "Error connecting to target", nil, nil), err) + cancel() + return + } + defer client.Close() for m := range workerChan { - logger := log.With(c.logger, "module", m.name) - level.Debug(logger).Log("msg", "Starting scrape") + _logger := log.With(logger, "module", m.name) + level.Debug(_logger).Log("msg", "Starting scrape") start := time.Now() - c.collect(ch, m) + c.collect(ch, _logger, client, m) duration := time.Since(start).Seconds() - level.Debug(logger).Log("msg", "Finished scrape", "duration_seconds", duration) + level.Debug(_logger).Log("msg", "Finished scrape", "duration_seconds", duration) c.metrics.SNMPCollectionDuration.WithLabelValues(m.name).Observe(duration) } - }() + }(i) } + done := false for _, module := range c.modules { - workerChan <- module + if done { + break + } + select { + case <-ctx.Done(): + done = true + level.Debug(c.logger).Log("msg", "Context canceled", "err", ctx.Err(), "module", module.name) + case workerChan <- module: + level.Debug(c.logger).Log("msg", "Sent module to worker", "module", module.name) + } } close(workerChan) wg.Wait() diff --git a/collector/collector_test.go b/collector/collector_test.go index 2f0fa377..5582ad4e 100644 --- a/collector/collector_test.go +++ b/collector/collector_test.go @@ -26,6 +26,7 @@ import ( io_prometheus_client "github.com/prometheus/client_model/go" "github.com/prometheus/snmp_exporter/config" + "github.com/prometheus/snmp_exporter/scraper" ) func TestPduToSample(t *testing.T) { @@ -1401,3 +1402,110 @@ func TestAddAllowedIndices(t *testing.T) { } } } + +func TestScrapeTarget(t *testing.T) { + cases := []struct { + name string + module *config.Module + getResponse map[string]gosnmp.SnmpPDU + walkResponses map[string][]gosnmp.SnmpPDU + expectPdus []gosnmp.SnmpPDU + getCall []string + walkCall []string + }{ + { + name: "basic", + module: &config.Module{ + Get: []string{"1.3.6.1.2.1.1.1.0"}, + Walk: []string{"1.3.6.1.2.1.2.2.1.2", "1.3.6.1.2.1.31.1.1.1.18"}, + }, + getResponse: map[string]gosnmp.SnmpPDU{ + "1.3.6.1.2.1.1.1.0": {Type: gosnmp.OctetString, Name: "1.3.6.1.2.1.1.1.0", Value: "Test Device"}, // sysDescr + }, + walkResponses: map[string][]gosnmp.SnmpPDU{ + "1.3.6.1.2.1.2.2.1.2": { + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.1", Value: "lo"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.2", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.3", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + }, + "1.3.6.1.2.1.31.1.1.1.18": { + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.1", Value: "lo"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.2", Value: "eth0"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.3", Value: "swp1"}, + }, + }, + expectPdus: []gosnmp.SnmpPDU{ + {Type: gosnmp.OctetString, Name: "1.3.6.1.2.1.1.1.0", Value: "Test Device"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.1", Value: "lo"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.2", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.3", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.1", Value: "lo"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.2", Value: "eth0"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.3", Value: "swp1"}, + }, + getCall: []string{"1.3.6.1.2.1.1.1.0"}, + walkCall: []string{"1.3.6.1.2.1.2.2.1.2", "1.3.6.1.2.1.31.1.1.1.18"}, + }, + { + name: "dynamic filter", + module: &config.Module{ + Get: []string{}, + Walk: []string{"1.3.6.1.2.1.31.1.1.1.18"}, + Filters: []config.DynamicFilter{ + { + Oid: "1.3.6.1.2.1.2.2.1.2", + Targets: []string{ + "1.3.6.1.2.1.31.1.1.1.18", + }, + Values: []string{"Intel Corporation 82540EM Gigabit Ethernet Controller"}, + }, + }, + }, + getResponse: map[string]gosnmp.SnmpPDU{ + "1.3.6.1.2.1.31.1.1.1.18.2": {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.2", Value: "eth0"}, + "1.3.6.1.2.1.31.1.1.1.18.3": {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.3", Value: "swp1"}, + }, + walkResponses: map[string][]gosnmp.SnmpPDU{ + "1.3.6.1.2.1.2.2.1.2": { + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.1", Value: "lo"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.2", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.2.2.1.2.3", Value: "Intel Corporation 82540EM Gigabit Ethernet Controller"}, + }, + }, + expectPdus: []gosnmp.SnmpPDU{ + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.2", Value: "eth0"}, + {Type: gosnmp.OctetString, Name: ".1.3.6.1.2.1.31.1.1.1.18.3", Value: "swp1"}, + }, + getCall: []string{"1.3.6.1.2.1.31.1.1.1.18.2", "1.3.6.1.2.1.31.1.1.1.18.3"}, + walkCall: []string{"1.3.6.1.2.1.2.2.1.2"}, + }, + } + + auth := &config.Auth{Version: 2} + for _, c := range cases { + tt := c + t.Run(tt.name, func(t *testing.T) { + mock := scraper.NewMockSNMPScraper(tt.getResponse, tt.walkResponses) + results, err := ScrapeTarget(mock, "someTarget", auth, tt.module, log.NewNopLogger(), Metrics{}) + if err != nil { + t.Errorf("ScrapeTarget returned an error: %v", err) + } + if !reflect.DeepEqual(mock.CallGet(), tt.getCall) { + t.Errorf("Expected get call %v, got %v", tt.getCall, mock.CallGet()) + } + if !reflect.DeepEqual(mock.CallWalk(), tt.walkCall) { + t.Errorf("Expected walk call %v, got %v", tt.walkCall, mock.CallWalk()) + } + expectedPdusLen := len(tt.expectPdus) + if len(results.pdus) != expectedPdusLen { + t.Fatalf("Expected %d PDUs, got %d", expectedPdusLen, len(results.pdus)) + } + + for i, pdu := range tt.expectPdus { + if !reflect.DeepEqual(pdu, results.pdus[i]) { + t.Errorf("Expected %v, got %v", pdu, results.pdus[i]) + } + } + }) + } +} diff --git a/scraper/gosnmp.go b/scraper/gosnmp.go new file mode 100644 index 00000000..83166898 --- /dev/null +++ b/scraper/gosnmp.go @@ -0,0 +1,117 @@ +// Copyright 2024 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 scraper + +import ( + "context" + "fmt" + "net" + "strconv" + "strings" + "time" + + "github.com/go-kit/log" + "github.com/go-kit/log/level" + "github.com/gosnmp/gosnmp" +) + +type GoSNMPWrapper struct { + c *gosnmp.GoSNMP + logger log.Logger +} + +func NewGoSNMP(logger log.Logger, target, srcAddress string) (*GoSNMPWrapper, error) { + transport := "udp" + if s := strings.SplitN(target, "://", 2); len(s) == 2 { + transport = s[0] + target = s[1] + } + port := uint16(161) + if host, _port, err := net.SplitHostPort(target); err == nil { + target = host + p, err := strconv.Atoi(_port) + if err != nil { + return nil, fmt.Errorf("error converting port number to int for target %q: %w", target, err) + } + port = uint16(p) + } + g := &gosnmp.GoSNMP{ + Transport: transport, + Target: target, + Port: port, + LocalAddr: srcAddress, + } + return &GoSNMPWrapper{c: g, logger: logger}, nil +} + +func (g *GoSNMPWrapper) SetOptions(fns ...func(*gosnmp.GoSNMP)) { + for _, fn := range fns { + fn(g.c) + } +} + +func (g *GoSNMPWrapper) Connect() error { + st := time.Now() + err := g.c.Connect() + if err != nil { + if err == context.Canceled { + return fmt.Errorf("scrape cancelled after %s (possible timeout) connecting to target %s", + time.Since(st), g.c.Target) + } + return fmt.Errorf("error connecting to target %s: %s", g.c.Target, err) + } + return nil +} + +func (g *GoSNMPWrapper) Close() error { + return g.c.Conn.Close() +} + +func (g *GoSNMPWrapper) Get(oids []string) (results *gosnmp.SnmpPacket, err error) { + level.Debug(g.logger).Log("msg", "Getting OIDs", "oids", oids) + st := time.Now() + results, err = g.c.Get(oids) + if err != nil { + if err == context.Canceled { + err = fmt.Errorf("scrape cancelled after %s (possible timeout) getting target %s", + time.Since(st), g.c.Target) + } else { + err = fmt.Errorf("error getting target %s: %s", g.c.Target, err) + } + return + } + level.Debug(g.logger).Log("msg", "Get of OIDs completed", "oids", oids, "duration_seconds", time.Since(st)) + return +} + +func (g *GoSNMPWrapper) WalkAll(oid string) (results []gosnmp.SnmpPDU, err error) { + level.Debug(g.logger).Log("msg", "Walking subtree", "oid", oid) + st := time.Now() + if g.c.Version == gosnmp.Version1 { + results, err = g.c.WalkAll(oid) + } else { + results, err = g.c.BulkWalkAll(oid) + } + if err != nil { + if err == context.Canceled { + err = fmt.Errorf("scrape canceled after %s (possible timeout) walking target %s", + time.Since(st), g.c.Target) + } else { + err = fmt.Errorf("error walking target %s: %s", g.c.Target, err) + } + return + } + level.Debug(g.logger).Log("msg", "Walk of subtree completed", "oid", oid, "duration_seconds", time.Since(st)) + return +} diff --git a/scraper/mock.go b/scraper/mock.go new file mode 100644 index 00000000..a2be8595 --- /dev/null +++ b/scraper/mock.go @@ -0,0 +1,84 @@ +// Copyright 2024 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 scraper + +import ( + "github.com/gosnmp/gosnmp" +) + +func NewMockSNMPScraper(get map[string]gosnmp.SnmpPDU, walk map[string][]gosnmp.SnmpPDU) *mockSNMPScraper { + return &mockSNMPScraper{ + GetResponses: get, + WalkResponses: walk, + callGet: make([]string, 0), + callWalk: make([]string, 0), + } +} + +type mockSNMPScraper struct { + GetResponses map[string]gosnmp.SnmpPDU + WalkResponses map[string][]gosnmp.SnmpPDU + ConnectError error + CloseError error + + callGet []string + callWalk []string +} + +func (m *mockSNMPScraper) CallGet() []string { + return m.callGet +} + +func (m *mockSNMPScraper) CallWalk() []string { + return m.callWalk +} + +func (m *mockSNMPScraper) Get(oids []string) (*gosnmp.SnmpPacket, error) { + pdus := make([]gosnmp.SnmpPDU, 0, len(oids)) + for _, oid := range oids { + if response, exists := m.GetResponses[oid]; exists { + pdus = append(pdus, response) + } else { + pdus = append(pdus, gosnmp.SnmpPDU{ + Name: oid, + Type: gosnmp.NoSuchObject, + Value: nil, + }) + } + m.callGet = append(m.callGet, oid) + } + return &gosnmp.SnmpPacket{ + Variables: pdus, + Error: gosnmp.NoError, + }, nil +} + +func (m *mockSNMPScraper) WalkAll(baseOID string) ([]gosnmp.SnmpPDU, error) { + m.callWalk = append(m.callWalk, baseOID) + if pdus, exists := m.WalkResponses[baseOID]; exists { + return pdus, nil + } + return nil, nil +} + +func (m *mockSNMPScraper) Connect() error { + return m.ConnectError +} + +func (m *mockSNMPScraper) Close() error { + return m.CloseError +} + +func (m *mockSNMPScraper) SetOptions(...func(*gosnmp.GoSNMP)) { +} diff --git a/scraper/scraper.go b/scraper/scraper.go new file mode 100644 index 00000000..0092023b --- /dev/null +++ b/scraper/scraper.go @@ -0,0 +1,26 @@ +// Copyright 2024 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 scraper + +import ( + "github.com/gosnmp/gosnmp" +) + +type SNMPScraper interface { + Get([]string) (*gosnmp.SnmpPacket, error) + WalkAll(string) ([]gosnmp.SnmpPDU, error) + Connect() error + Close() error + SetOptions(...func(*gosnmp.GoSNMP)) +}