Skip to content

Commit d4b18c7

Browse files
committed
Add regexes for interface selection
Fixes #8
1 parent 61a4422 commit d4b18c7

File tree

3 files changed

+76
-15
lines changed

3 files changed

+76
-15
lines changed

README.md

+3-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,9 @@ devices:
4444
- 100
4545
interfaces: # optional: Some devices (BNGs) might have thousands of interfaces
4646
- HundredGigE0/0/0 # you can specify the interfaces to be scraped
47-
- GigabitEthernet0
47+
- GigabitEthernet.* # all interfaces are interpreted as regexes, that must match the whole interface name
48+
excluded_interfaces: # optional: to easy limit to broad interface regexes
49+
- GigabitEthernet0/1 # entries are also regexes thet must match full interface names
4850
username: monitoring # required: Username to use for SSH auth
4951
key_file: /path/to/a/private.key # optional: Private key to use for SSH auth
5052
password: correcthorsebatterystaple # optional: Password for SSH auth

config/config.go

+51-12
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,8 @@ import (
44
"github.com/gobwas/glob"
55
"io"
66
"io/ioutil"
7+
"regexp"
8+
"strings"
79

810
"gopkg.in/yaml.v2"
911
)
@@ -55,18 +57,53 @@ func (o OSVersion) String() string {
5557
// DeviceGroupConfig describe how to connect to a remote device and what metrics
5658
// to extract from the remote device.
5759
type DeviceGroupConfig struct {
58-
OSVersion OSVersion
59-
StaticName *string `yaml:"-"`
60-
Matcher glob.Glob `yaml:"-"`
61-
Port int `yaml:"port,omitempty"`
62-
Username string `yaml:"username"`
63-
KeyFile string `yaml:"key_file,omitempty"`
64-
Password string `yaml:"password,omitempty"`
65-
ConnectTimeout int `yaml:"connect_timeout,omitempty"`
66-
CommandTimeout int `yaml:"command_timeout,omitempty"`
67-
EnabledCollectors []string `yaml:"enabled_collectors,flow"`
68-
Interfaces []string `yaml:"interfaces,flow"`
69-
EnabledVLANs []string `yaml:"enabled_vlans,flow"`
60+
OSVersion OSVersion
61+
StaticName *string `yaml:"-"`
62+
Matcher glob.Glob `yaml:"-"`
63+
Port int `yaml:"port,omitempty"`
64+
Username string `yaml:"username"`
65+
KeyFile string `yaml:"key_file,omitempty"`
66+
Password string `yaml:"password,omitempty"`
67+
ConnectTimeout int `yaml:"connect_timeout,omitempty"`
68+
CommandTimeout int `yaml:"command_timeout,omitempty"`
69+
EnabledCollectors []string `yaml:"enabled_collectors,flow"`
70+
Interfaces []string `yaml:"interfaces,flow"`
71+
interfacesRegexp []*regexp.Regexp `yaml:"-"`
72+
ExcludedInterfaces []string `yaml:"excluded_interfaces,flow"`
73+
excludedInterfacesRegexp []*regexp.Regexp `yaml:"-"`
74+
EnabledVLANs []string `yaml:"enabled_vlans,flow"`
75+
}
76+
77+
func normalizeRegex(str string) string {
78+
return "^" + strings.TrimRight(strings.TrimLeft(str, "^"), "$") + "$"
79+
}
80+
81+
func (dgc *DeviceGroupConfig) createInterfaceRegexp() {
82+
dgc.interfacesRegexp = make([]*regexp.Regexp, len(dgc.Interfaces))
83+
for i, str := range dgc.Interfaces {
84+
dgc.interfacesRegexp[i] = regexp.MustCompile(normalizeRegex(str))
85+
}
86+
dgc.excludedInterfacesRegexp = make([]*regexp.Regexp, len(dgc.ExcludedInterfaces))
87+
for i, str := range dgc.ExcludedInterfaces {
88+
dgc.excludedInterfacesRegexp[i] = regexp.MustCompile(normalizeRegex(str))
89+
}
90+
}
91+
92+
func (dgc *DeviceGroupConfig) MatchInterface(ifName string) bool {
93+
match := false
94+
for _, r := range dgc.interfacesRegexp {
95+
if r.MatchString(ifName) {
96+
match = true
97+
break
98+
}
99+
}
100+
for _, r := range dgc.excludedInterfacesRegexp {
101+
if r.MatchString(ifName) {
102+
match = false
103+
break
104+
}
105+
}
106+
return match
70107
}
71108

72109
func newConfig() *Config {
@@ -136,6 +173,8 @@ func Load(reader io.Reader) (*Config, error) {
136173
if groupConfig.Port == 0 {
137174
groupConfig.Port = defaultPort
138175
}
176+
177+
groupConfig.createInterfaceRegexp()
139178
}
140179

141180
return config, nil

interfaces/collector.go

+22-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
package interfaces
22

33
import (
4+
"regexp"
5+
46
"gitlab.com/wobcom/cisco-exporter/collector"
57
"gitlab.com/wobcom/cisco-exporter/connector"
68

@@ -21,6 +23,8 @@ var (
2123
adminStatusDesc *prometheus.Desc
2224
operStatusDesc *prometheus.Desc
2325
errorStatusDesc *prometheus.Desc
26+
27+
interfaceLineRegexp *regexp.Regexp
2428
)
2529

2630
// Collector gathers counters for remote device's interfaces.
@@ -48,6 +52,8 @@ func init() {
4852
adminStatusDesc = prometheus.NewDesc(prefix+"admin_up_info", "Admin operational status", l, nil)
4953
operStatusDesc = prometheus.NewDesc(prefix+"up_info", "Interface operational status", l, nil)
5054
errorStatusDesc = prometheus.NewDesc(prefix+"error_status_info", "Admin and operational status differ", l, nil)
55+
56+
interfaceLineRegexp = regexp.MustCompile(`^\s*\*?\s(\S+)[\s\d-]*$`)
5157
}
5258

5359
// Describe implements the collector.Collector interface's Describe function
@@ -70,8 +76,22 @@ func (c *Collector) Collect(ctx *collector.CollectContext) {
7076
}()
7177

7278
if len(ctx.Connection.Device.Interfaces) > 0 {
73-
for _, interfaceName := range ctx.Connection.Device.Interfaces {
74-
c.collect(ctx, interfaceName)
79+
sshCtx := connector.NewSSHCommandContext("show interface summary")
80+
go ctx.Connection.RunCommand(sshCtx)
81+
82+
IfCollection: for {
83+
select {
84+
case <-sshCtx.Done:
85+
break IfCollection
86+
case line := <-sshCtx.Output:
87+
match := interfaceLineRegexp.FindStringSubmatch(line)
88+
if len(match) > 0 {
89+
ifName := match[1]
90+
if ctx.Connection.Device.MatchInterface(ifName) {
91+
c.collect(ctx, ifName)
92+
}
93+
}
94+
}
7595
}
7696
} else {
7797
c.collect(ctx, "")

0 commit comments

Comments
 (0)