Merge remote-tracking branch 'origin/master'

This commit is contained in:
Daniel Czerwonk 2017-09-29 23:38:19 +02:00
commit 25fc6324ef
10 changed files with 170 additions and 150 deletions

33
bgp/bgp_collector.go Normal file
View File

@ -0,0 +1,33 @@
package bgp
import (
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
var exporter map[int]*protocol.GenericProtocolMetricExporter
type BgpCollector struct {
protocols []*protocol.Protocol
}
func init() {
exporter = make(map[int]*protocol.GenericProtocolMetricExporter)
exporter[4] = protocol.NewGenericProtocolMetricExporter("bgp4_session")
exporter[6] = protocol.NewGenericProtocolMetricExporter("bgp6_session")
}
func NewCollector(p []*protocol.Protocol) prometheus.Collector {
return &BgpCollector{protocols: p}
}
func (m *BgpCollector) Describe(ch chan<- *prometheus.Desc) {
exporter[4].Describe(ch)
exporter[6].Describe(ch)
}
func (m *BgpCollector) Collect(ch chan<- prometheus.Metric) {
for _, p := range m.protocols {
exporter[p.IpVersion].Export(p, ch)
}
}

View File

@ -1,55 +0,0 @@
package bgp
import (
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
var descriptions map[int]*desc
type desc struct {
upDesc *prometheus.Desc
importCountDesc *prometheus.Desc
exportCountDesc *prometheus.Desc
filterCountDesc *prometheus.Desc
uptimeDesc *prometheus.Desc
}
type BgpMetric struct {
Protocol *protocol.Protocol
}
func init() {
descriptions = make(map[int]*desc)
descriptions[4] = getDesc("bgp4")
descriptions[6] = getDesc("bgp6")
}
func getDesc(prefix string) *desc {
labels := []string{"name"}
d := &desc{}
d.upDesc = prometheus.NewDesc(prefix+"_session_up", "Protocol is up", labels, nil)
d.importCountDesc = prometheus.NewDesc(prefix+"_session_prefix_count_import", "Number of imported routes", labels, nil)
d.exportCountDesc = prometheus.NewDesc(prefix+"_session_prefix_count_export", "Number of exported routes", labels, nil)
d.filterCountDesc = prometheus.NewDesc(prefix+"_session_prefix_count_filter", "Number of filtered routes", labels, nil)
d.uptimeDesc = prometheus.NewDesc(prefix+"_session_uptime", "Uptime of the protocol in seconds", labels, nil)
return d
}
func (m *BgpMetric) Describe(ch chan<- *prometheus.Desc) {
ch <- descriptions[m.Protocol.IpVersion].upDesc
ch <- descriptions[m.Protocol.IpVersion].importCountDesc
ch <- descriptions[m.Protocol.IpVersion].exportCountDesc
ch <- descriptions[m.Protocol.IpVersion].filterCountDesc
ch <- descriptions[m.Protocol.IpVersion].uptimeDesc
}
func (m *BgpMetric) GetMetrics(ch chan<- prometheus.Metric) {
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].upDesc, prometheus.GaugeValue, float64(m.Protocol.Up), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].importCountDesc, prometheus.GaugeValue, float64(m.Protocol.Imported), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].exportCountDesc, prometheus.GaugeValue, float64(m.Protocol.Exported), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].filterCountDesc, prometheus.GaugeValue, float64(m.Protocol.Filtered), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].uptimeDesc, prometheus.GaugeValue, float64(m.Protocol.Uptime), m.Protocol.Name)
}

View File

@ -1,31 +1,47 @@
package main
import (
"github.com/czerwonk/bird_exporter/bgp"
"github.com/czerwonk/bird_exporter/ospf"
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
type MetricCollector struct {
Protocols []ProtocolMetric
collectors []prometheus.Collector
}
func NewMetricCollectorForProtocols(p []*protocol.Protocol) *MetricCollector {
m := make([]ProtocolMetric, 0)
for _, x := range p {
m = append(m, NewProtocolMetricFromProtocol(x))
func NewMetricCollectorForProtocols(protocols []*protocol.Protocol) *MetricCollector {
b := make([]*protocol.Protocol, 0)
o := make([]*protocol.Protocol, 0)
for _, p := range protocols {
if p.Proto == protocol.BGP {
b = append(b, p)
} else if p.Proto == protocol.OSPF {
o = append(o, p)
}
}
return &MetricCollector{Protocols: m}
c := make([]prometheus.Collector, 0)
if len(b) > 0 {
c = append(c, bgp.NewCollector(b))
}
if len(o) > 0 {
c = append(c, ospf.NewCollector(o))
}
return &MetricCollector{collectors: c}
}
func (m *MetricCollector) Describe(ch chan<- *prometheus.Desc) {
for _, p := range m.Protocols {
p.Describe(ch)
for _, c := range m.collectors {
c.Describe(ch)
}
}
func (m *MetricCollector) Collect(ch chan<- prometheus.Metric) {
for _, p := range m.Protocols {
p.GetMetrics(ch)
for _, c := range m.collectors {
c.Collect(ch)
}
}

54
ospf/ospf_collector.go Normal file
View File

@ -0,0 +1,54 @@
package ospf
import (
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
var descriptions map[int]*desc
var exporter map[int]*protocol.GenericProtocolMetricExporter
type desc struct {
runningDesc *prometheus.Desc
}
type OspfCollector struct {
protocols []*protocol.Protocol
}
func init() {
exporter = make(map[int]*protocol.GenericProtocolMetricExporter)
exporter[4] = protocol.NewGenericProtocolMetricExporter("ospf")
exporter[6] = protocol.NewGenericProtocolMetricExporter("ospfv3")
descriptions = make(map[int]*desc)
descriptions[4] = getDesc("ospf")
descriptions[6] = getDesc("ospfv3")
}
func getDesc(prefix string) *desc {
labels := []string{"name"}
d := &desc{}
d.runningDesc = prometheus.NewDesc(prefix+"_running", "State of OSPF: 0 = Alone, 1 = Running (Neighbor-Adjacencies established)", labels, nil)
return d
}
func NewCollector(p []*protocol.Protocol) prometheus.Collector {
return &OspfCollector{protocols: p}
}
func (m *OspfCollector) Describe(ch chan<- *prometheus.Desc) {
exporter[4].Describe(ch)
exporter[6].Describe(ch)
ch <- descriptions[4].runningDesc
ch <- descriptions[6].runningDesc
}
func (m *OspfCollector) Collect(ch chan<- prometheus.Metric) {
for _, p := range m.protocols {
exporter[p.IpVersion].Export(p, ch)
ch <- prometheus.MustNewConstMetric(descriptions[p.IpVersion].runningDesc, prometheus.GaugeValue, p.Attributes["running"], p.Name)
}
}

View File

@ -1,59 +0,0 @@
package ospf
import (
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
var descriptions map[int]*desc
type desc struct {
upDesc *prometheus.Desc
importCountDesc *prometheus.Desc
exportCountDesc *prometheus.Desc
filterCountDesc *prometheus.Desc
uptimeDesc *prometheus.Desc
runningDesc *prometheus.Desc
}
type OspfMetric struct {
Protocol *protocol.Protocol
}
func init() {
descriptions = make(map[int]*desc)
descriptions[4] = getDesc("ospf")
descriptions[6] = getDesc("ospfv3")
}
func getDesc(prefix string) *desc {
labels := []string{"name"}
d := &desc{}
d.upDesc = prometheus.NewDesc(prefix+"_up", "Protocol is up", labels, nil)
d.importCountDesc = prometheus.NewDesc(prefix+"_prefix_count_import", "Number of imported routes", labels, nil)
d.exportCountDesc = prometheus.NewDesc(prefix+"_prefix_count_export", "Number of exported routes", labels, nil)
d.filterCountDesc = prometheus.NewDesc(prefix+"_prefix_count_filter", "Number of filtered routes", labels, nil)
d.uptimeDesc = prometheus.NewDesc(prefix+"_uptime", "Uptime of the protocol in seconds", labels, nil)
d.runningDesc = prometheus.NewDesc(prefix+"_running", "State of OSPF: 0 = Alone, 1 = Running (Neighbor-Adjacencies established)", labels, nil)
return d
}
func (m *OspfMetric) Describe(ch chan<- *prometheus.Desc) {
ch <- descriptions[m.Protocol.IpVersion].upDesc
ch <- descriptions[m.Protocol.IpVersion].importCountDesc
ch <- descriptions[m.Protocol.IpVersion].exportCountDesc
ch <- descriptions[m.Protocol.IpVersion].filterCountDesc
ch <- descriptions[m.Protocol.IpVersion].uptimeDesc
ch <- descriptions[m.Protocol.IpVersion].runningDesc
}
func (m *OspfMetric) GetMetrics(ch chan<- prometheus.Metric) {
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].upDesc, prometheus.GaugeValue, float64(m.Protocol.Up), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].importCountDesc, prometheus.GaugeValue, float64(m.Protocol.Imported), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].exportCountDesc, prometheus.GaugeValue, float64(m.Protocol.Exported), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].filterCountDesc, prometheus.GaugeValue, float64(m.Protocol.Filtered), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].uptimeDesc, prometheus.GaugeValue, float64(m.Protocol.Uptime), m.Protocol.Name)
ch <- prometheus.MustNewConstMetric(descriptions[m.Protocol.IpVersion].runningDesc, prometheus.GaugeValue, m.Protocol.Attributes["running"], m.Protocol.Name)
}

View File

@ -21,7 +21,7 @@ var (
func init() {
protocolRegex, _ = regexp.Compile("^(?:1002\\-)?([^\\s]+)\\s+(BGP|OSPF)\\s+([^\\s]+)\\s+([^\\s]+)\\s+([^\\s]+)\\s+(.*?)\\s*$")
routeRegex, _ = regexp.Compile("^\\s+Routes:\\s+(\\d+) imported, (?:(\\d+) filtered, )?(\\d+) exported")
routeRegex, _ = regexp.Compile("^\\s+Routes:\\s+(\\d+) imported, (?:(\\d+) filtered, )?(\\d+) exported(?:, (\\d+) preferred)?")
uptimeRegex, _ = regexp.Compile("^(?:((\\d+):(\\d{2}):(\\d{2}))|\\d+)$")
}
@ -88,6 +88,10 @@ func parseLineForRoutes(line string, p *protocol.Protocol) {
if len(match[2]) > 0 {
p.Filtered, _ = strconv.ParseInt(match[2], 10, 64)
}
if len(match[4]) > 0 {
p.Preferred, _ = strconv.ParseInt(match[4], 10, 64)
}
}
}

View File

@ -19,6 +19,7 @@ func TestEstablishedBgpOldTimeFormat(t *testing.T) {
assert.Int64Equal("imported", 12, x.Imported, t)
assert.Int64Equal("exported", 34, x.Exported, t)
assert.Int64Equal("filtered", 1, x.Filtered, t)
assert.Int64Equal("preferred", 100, x.Preferred, t)
assert.IntEqual("ipVersion", 4, x.IpVersion, t)
}
@ -34,6 +35,7 @@ func TestEstablishedBgpCurrentTimeFormat(t *testing.T) {
assert.Int64Equal("imported", 12, x.Imported, t)
assert.Int64Equal("exported", 34, x.Exported, t)
assert.Int64Equal("filtered", 1, x.Filtered, t)
assert.Int64Equal("preferred", 100, x.Preferred, t)
assert.IntEqual("ipVersion", 4, x.IpVersion, t)
assert.IntEqual("uptime", 60, x.Uptime, t)
}
@ -79,6 +81,7 @@ func TestOspfOldTimeFormat(t *testing.T) {
assert.IntEqual("up", 1, x.Up, t)
assert.Int64Equal("imported", 12, x.Imported, t)
assert.Int64Equal("exported", 34, x.Exported, t)
assert.Int64Equal("preferred", 100, x.Preferred, t)
assert.IntEqual("ipVersion", 4, x.IpVersion, t)
}
@ -93,6 +96,7 @@ func TestOspfCurrentTimeFormat(t *testing.T) {
assert.IntEqual("up", 1, x.Up, t)
assert.Int64Equal("imported", 12, x.Imported, t)
assert.Int64Equal("exported", 34, x.Exported, t)
assert.Int64Equal("preferred", 100, x.Preferred, t)
assert.IntEqual("ipVersion", 4, x.IpVersion, t)
assert.IntEqual("uptime", 60, x.Uptime, t)
}

View File

@ -0,0 +1,47 @@
package protocol
import "github.com/prometheus/client_golang/prometheus"
type GenericProtocolMetricExporter struct {
upDesc *prometheus.Desc
importCountDesc *prometheus.Desc
exportCountDesc *prometheus.Desc
filterCountDesc *prometheus.Desc
preferredCountDesc *prometheus.Desc
uptimeDesc *prometheus.Desc
}
func NewGenericProtocolMetricExporter(prefix string) *GenericProtocolMetricExporter {
m := &GenericProtocolMetricExporter{}
m.initDesc(prefix)
return m
}
func (m *GenericProtocolMetricExporter) initDesc(prefix string) {
labels := []string{"name"}
m.upDesc = prometheus.NewDesc(prefix+"_up", "Protocol is up", labels, nil)
m.importCountDesc = prometheus.NewDesc(prefix+"_prefix_count_import", "Number of imported routes", labels, nil)
m.exportCountDesc = prometheus.NewDesc(prefix+"_prefix_count_export", "Number of exported routes", labels, nil)
m.filterCountDesc = prometheus.NewDesc(prefix+"_prefix_count_filter", "Number of filtered routes", labels, nil)
m.preferredCountDesc = prometheus.NewDesc(prefix+"_prefix_count_preferred", "Number of preferred routes", labels, nil)
m.uptimeDesc = prometheus.NewDesc(prefix+"_uptime", "Uptime of the protocol in seconds", labels, nil)
}
func (m *GenericProtocolMetricExporter) Describe(ch chan<- *prometheus.Desc) {
ch <- m.upDesc
ch <- m.importCountDesc
ch <- m.exportCountDesc
ch <- m.filterCountDesc
ch <- m.preferredCountDesc
ch <- m.uptimeDesc
}
func (m *GenericProtocolMetricExporter) Export(protocol *Protocol, ch chan<- prometheus.Metric) {
ch <- prometheus.MustNewConstMetric(m.upDesc, prometheus.GaugeValue, float64(protocol.Up), protocol.Name)
ch <- prometheus.MustNewConstMetric(m.importCountDesc, prometheus.GaugeValue, float64(protocol.Imported), protocol.Name)
ch <- prometheus.MustNewConstMetric(m.exportCountDesc, prometheus.GaugeValue, float64(protocol.Exported), protocol.Name)
ch <- prometheus.MustNewConstMetric(m.filterCountDesc, prometheus.GaugeValue, float64(protocol.Filtered), protocol.Name)
ch <- prometheus.MustNewConstMetric(m.preferredCountDesc, prometheus.GaugeValue, float64(protocol.Preferred), protocol.Name)
ch <- prometheus.MustNewConstMetric(m.uptimeDesc, prometheus.GaugeValue, float64(protocol.Uptime), protocol.Name)
}

View File

@ -14,6 +14,7 @@ type Protocol struct {
Imported int64
Exported int64
Filtered int64
Preferred int64
Uptime int
Attributes map[string]float64
}

View File

@ -1,25 +0,0 @@
package main
import (
"github.com/czerwonk/bird_exporter/bgp"
"github.com/czerwonk/bird_exporter/ospf"
"github.com/czerwonk/bird_exporter/protocol"
"github.com/prometheus/client_golang/prometheus"
)
type ProtocolMetric interface {
Describe(ch chan<- *prometheus.Desc)
GetMetrics(ch chan<- prometheus.Metric)
}
func NewProtocolMetricFromProtocol(p *protocol.Protocol) ProtocolMetric {
if p.Proto == protocol.BGP {
return &bgp.BgpMetric{Protocol: p}
}
if p.Proto == protocol.OSPF {
return &ospf.OspfMetric{Protocol: p}
}
return nil
}