mirror of
https://github.com/nshttpd/mikrotik-exporter.git
synced 2024-11-16 10:25:12 +01:00
9ceb56fdbf
Thanks. I could have sworn I went through already and done this at one point in time, but I guess not.
125 lines
2.9 KiB
Go
125 lines
2.9 KiB
Go
package collector
|
|
|
|
import (
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
log "github.com/sirupsen/logrus"
|
|
"gopkg.in/routeros.v2/proto"
|
|
)
|
|
|
|
type poeCollector struct {
|
|
currentDesc *prometheus.Desc
|
|
powerDesc *prometheus.Desc
|
|
voltageDesc *prometheus.Desc
|
|
props []string
|
|
}
|
|
|
|
func newPOECollector() routerOSCollector {
|
|
const prefix = "poe"
|
|
|
|
labelNames := []string{"name", "address", "interface"}
|
|
return &poeCollector{
|
|
currentDesc: description(prefix, "current", "current in mA", labelNames),
|
|
powerDesc: description(prefix, "wattage", "Power in W", labelNames),
|
|
voltageDesc: description(prefix, "voltage", "Voltage in V", labelNames),
|
|
props: []string{"poe-out-current", "poe-out-voltage", "poe-out-power"},
|
|
}
|
|
}
|
|
|
|
func (c *poeCollector) describe(ch chan<- *prometheus.Desc) {
|
|
ch <- c.currentDesc
|
|
ch <- c.powerDesc
|
|
ch <- c.voltageDesc
|
|
}
|
|
|
|
func (c *poeCollector) collect(ctx *collectorContext) error {
|
|
reply, err := ctx.client.Run("/interface/ethernet/poe/print", "=.proplist=name")
|
|
if err != nil {
|
|
log.WithFields(log.Fields{
|
|
"device": ctx.device.Name,
|
|
"error": err,
|
|
}).Error("error fetching interface poe metrics")
|
|
return err
|
|
}
|
|
|
|
ifaces := make([]string, 0)
|
|
for _, iface := range reply.Re {
|
|
n := iface.Map["name"]
|
|
ifaces = append(ifaces, n)
|
|
}
|
|
|
|
if len(ifaces) == 0 {
|
|
return nil
|
|
}
|
|
|
|
return c.collectPOEMetricsForInterfaces(ifaces, ctx)
|
|
}
|
|
|
|
func (c *poeCollector) collectPOEMetricsForInterfaces(ifaces []string, ctx *collectorContext) error {
|
|
reply, err := ctx.client.Run("/interface/ethernet/poe/monitor",
|
|
"=numbers="+strings.Join(ifaces, ","),
|
|
"=once=",
|
|
"=.proplist=name,"+strings.Join(c.props, ","))
|
|
if err != nil {
|
|
log.WithFields(log.Fields{
|
|
"device": ctx.device.Name,
|
|
"error": err,
|
|
}).Error("error fetching interface poe monitor metrics")
|
|
return err
|
|
}
|
|
|
|
for _, se := range reply.Re {
|
|
name, ok := se.Map["name"]
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
c.collectMetricsForInterface(name, se, ctx)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *poeCollector) collectMetricsForInterface(name string, se *proto.Sentence, ctx *collectorContext) {
|
|
for _, prop := range c.props {
|
|
v, ok := se.Map[prop]
|
|
if !ok {
|
|
continue
|
|
}
|
|
if v == "" {
|
|
continue
|
|
}
|
|
value, err := strconv.ParseFloat(v, 64)
|
|
if err != nil {
|
|
log.WithFields(log.Fields{
|
|
"device": ctx.device.Name,
|
|
"interface": name,
|
|
"property": prop,
|
|
"error": err,
|
|
}).Error("error parsing interface poe monitor metric")
|
|
return
|
|
}
|
|
|
|
ctx.ch <- prometheus.MustNewConstMetric(c.descForKey(prop), prometheus.GaugeValue, value, ctx.device.Name, ctx.device.Address, name)
|
|
}
|
|
}
|
|
|
|
func (c *poeCollector) valueForKey(name, value string) (float64, error) {
|
|
return strconv.ParseFloat(value, 64)
|
|
}
|
|
|
|
func (c *poeCollector) descForKey(name string) *prometheus.Desc {
|
|
switch name {
|
|
case "poe-out-current":
|
|
return c.currentDesc
|
|
case "poe-out-voltage":
|
|
return c.voltageDesc
|
|
case "poe-out-power":
|
|
return c.powerDesc
|
|
}
|
|
|
|
return nil
|
|
}
|