better logging
This commit is contained in:
parent
9d0d9e97f9
commit
ed94bed4da
|
|
@ -5,37 +5,45 @@ import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/davidnewhall/unifi-poller/metrics"
|
"github.com/davidnewhall/unifi-poller/metrics"
|
||||||
|
"github.com/davidnewhall/unifi-poller/promunifi"
|
||||||
)
|
)
|
||||||
|
|
||||||
// ExportMetrics updates the internal metrics provided via
|
// ExportMetrics updates the internal metrics provided via
|
||||||
// HTTP at /metrics for prometheus collection. This is run by Prometheus CollectFn.
|
// HTTP at /metrics for prometheus collection.
|
||||||
|
// This is run by Prometheus as CollectFn.
|
||||||
func (u *UnifiPoller) ExportMetrics() (*metrics.Metrics, error) {
|
func (u *UnifiPoller) ExportMetrics() (*metrics.Metrics, error) {
|
||||||
if u.Config.ReAuth {
|
|
||||||
u.LogDebugf("Re-authenticating to UniFi Controller")
|
|
||||||
// Some users need to re-auth every interval because the cookie times out.
|
|
||||||
if err := u.Unifi.Login(); err != nil {
|
|
||||||
u.LogError(err, "re-authenticating")
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
u.LastCheck = time.Now()
|
u.LastCheck = time.Now()
|
||||||
m, err := u.CollectMetrics()
|
m, err := u.CollectMetrics()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
u.LogErrorf("collecting metrics: %v", err)
|
u.LogErrorf("collecting metrics: %v", err)
|
||||||
return nil, err
|
u.Logf("Re-authenticating to UniFi Controller")
|
||||||
|
if err := u.Unifi.Login(); err != nil {
|
||||||
|
u.LogError(err, "re-authenticating")
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if m, err = u.CollectMetrics(); err != nil {
|
||||||
|
u.LogErrorf("collecting metrics: %v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
u.AugmentMetrics(m)
|
u.AugmentMetrics(m)
|
||||||
return m, nil
|
return m, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// LogExportReport is called after prometheus exports metrics. This is run by Prometheus as LoggingFn
|
// LogExportReport is called after prometheus exports metrics.
|
||||||
func (u *UnifiPoller) LogExportReport(m *metrics.Metrics, count int64) {
|
// This is run by Prometheus as LoggingFn
|
||||||
|
func (u *UnifiPoller) LogExportReport(report *promunifi.Report) {
|
||||||
|
m := report.Metrics
|
||||||
idsMsg := ""
|
idsMsg := ""
|
||||||
if u.Config.CollectIDS {
|
if u.Config.CollectIDS {
|
||||||
idsMsg = fmt.Sprintf(", IDS Events: %d, ", len(m.IDSList))
|
idsMsg = fmt.Sprintf(", IDS Events: %d, ", len(m.IDSList))
|
||||||
}
|
}
|
||||||
|
|
||||||
u.Logf("UniFi Measurements Exported. Sites: %d, Clients: %d, "+
|
u.Logf("UniFi Measurements Exported. Sites: %d, Clients: %d, "+
|
||||||
"Wireless APs: %d, Gateways: %d, Switches: %d%s, Metrics: %d",
|
"Wireless APs: %d, Gateways: %d, Switches: %d%s, Descs: %d, "+
|
||||||
len(m.Sites), len(m.Clients), len(m.UAPs),
|
"Metrics: %d, Errors: %d, Elapsed: %v",
|
||||||
len(m.UDMs)+len(m.USGs), len(m.USWs), idsMsg, count)
|
len(m.Sites), len(m.Clients), len(m.UAPs), len(m.UDMs)+len(m.USGs),
|
||||||
|
len(m.USWs), idsMsg, report.Descs, report.Total, report.Errors,
|
||||||
|
report.Elapsed.Round(time.Millisecond))
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -24,7 +24,7 @@ type UnifiCollectorCnfg struct {
|
||||||
// function to retreive the latest UniFi
|
// function to retreive the latest UniFi
|
||||||
CollectFn func() (*metrics.Metrics, error)
|
CollectFn func() (*metrics.Metrics, error)
|
||||||
// provide a logger function if you want to run a routine *after* prometheus checks in.
|
// provide a logger function if you want to run a routine *after* prometheus checks in.
|
||||||
LoggingFn func(*metrics.Metrics, int64)
|
LoggingFn func(*Report)
|
||||||
// Setting this to true will enable IDS exports.
|
// Setting this to true will enable IDS exports.
|
||||||
CollectIDS bool
|
CollectIDS bool
|
||||||
}
|
}
|
||||||
|
|
@ -47,6 +47,15 @@ type metricExports struct {
|
||||||
Labels []string
|
Labels []string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Report is passed into LoggingFn to log the export metrics to stdout (outside this package).
|
||||||
|
type Report struct {
|
||||||
|
Total int
|
||||||
|
Errors int
|
||||||
|
Descs int
|
||||||
|
Metrics *metrics.Metrics
|
||||||
|
Elapsed time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
// NewUnifiCollector returns a prometheus collector that will export any available
|
// NewUnifiCollector returns a prometheus collector that will export any available
|
||||||
// UniFi metrics. You must provide a collection function in the opts.
|
// UniFi metrics. You must provide a collection function in the opts.
|
||||||
func NewUnifiCollector(opts UnifiCollectorCnfg) prometheus.Collector {
|
func NewUnifiCollector(opts UnifiCollectorCnfg) prometheus.Collector {
|
||||||
|
|
@ -95,7 +104,7 @@ func (u *unifiCollector) Describe(ch chan<- *prometheus.Desc) {
|
||||||
// Collect satisifes the prometheus Collector. This runs the input method to get
|
// Collect satisifes the prometheus Collector. This runs the input method to get
|
||||||
// the current metrics (from another package) then exports them for prometheus.
|
// the current metrics (from another package) then exports them for prometheus.
|
||||||
func (u *unifiCollector) Collect(ch chan<- prometheus.Metric) {
|
func (u *unifiCollector) Collect(ch chan<- prometheus.Metric) {
|
||||||
var count int64
|
start := time.Now()
|
||||||
m, err := u.Config.CollectFn()
|
m, err := u.Config.CollectFn()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ch <- prometheus.NewInvalidMetric(
|
ch <- prometheus.NewInvalidMetric(
|
||||||
|
|
@ -103,19 +112,34 @@ func (u *unifiCollector) Collect(ch chan<- prometheus.Metric) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
descs := make(map[*prometheus.Desc]bool) // used as a counter
|
||||||
|
r := &Report{Metrics: m}
|
||||||
if u.Config.LoggingFn != nil {
|
if u.Config.LoggingFn != nil {
|
||||||
defer func() { u.Config.LoggingFn(m, count) }()
|
defer func() {
|
||||||
|
r.Elapsed = time.Since(start)
|
||||||
|
r.Descs = len(descs)
|
||||||
|
u.Config.LoggingFn(r)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
|
||||||
|
process := func(m []*metricExports, ts time.Time) {
|
||||||
|
count, errors := u.export(ch, m, ts)
|
||||||
|
r.Total += count
|
||||||
|
r.Errors += errors
|
||||||
|
for _, d := range m {
|
||||||
|
descs[d.Desc] = true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, asset := range m.Clients {
|
for _, asset := range m.Clients {
|
||||||
count += u.export(ch, u.exportClient(asset), m.TS)
|
process(u.exportClient(asset), m.TS)
|
||||||
}
|
}
|
||||||
for _, asset := range m.Sites {
|
for _, asset := range m.Sites {
|
||||||
count += u.export(ch, u.exportSite(asset), m.TS)
|
process(u.exportSite(asset), m.TS)
|
||||||
}
|
}
|
||||||
if u.Config.CollectIDS {
|
if u.Config.CollectIDS {
|
||||||
for _, asset := range m.IDSList {
|
for _, asset := range m.IDSList {
|
||||||
count += u.export(ch, u.exportIDS(asset), m.TS)
|
process(u.exportIDS(asset), m.TS)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -124,20 +148,20 @@ func (u *unifiCollector) Collect(ch chan<- prometheus.Metric) {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, asset := range m.Devices.UAPs {
|
for _, asset := range m.Devices.UAPs {
|
||||||
count += u.export(ch, u.exportUAP(asset), m.TS)
|
process(u.exportUAP(asset), m.TS)
|
||||||
}
|
}
|
||||||
for _, asset := range m.Devices.USGs {
|
for _, asset := range m.Devices.USGs {
|
||||||
count += u.export(ch, u.exportUSG(asset), m.TS)
|
process(u.exportUSG(asset), m.TS)
|
||||||
}
|
}
|
||||||
for _, asset := range m.Devices.USWs {
|
for _, asset := range m.Devices.USWs {
|
||||||
count += u.export(ch, u.exportUSW(asset), m.TS)
|
process(u.exportUSW(asset), m.TS)
|
||||||
}
|
}
|
||||||
for _, asset := range m.Devices.UDMs {
|
for _, asset := range m.Devices.UDMs {
|
||||||
count += u.export(ch, u.exportUDM(asset), m.TS)
|
process(u.exportUDM(asset), m.TS)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (u *unifiCollector) export(ch chan<- prometheus.Metric, exports []*metricExports, ts time.Time) (count int64) {
|
func (u *unifiCollector) export(ch chan<- prometheus.Metric, exports []*metricExports, ts time.Time) (count, errors int) {
|
||||||
for _, e := range exports {
|
for _, e := range exports {
|
||||||
var val float64
|
var val float64
|
||||||
switch v := e.Value.(type) {
|
switch v := e.Value.(type) {
|
||||||
|
|
@ -148,6 +172,7 @@ func (u *unifiCollector) export(ch chan<- prometheus.Metric, exports []*metricEx
|
||||||
case unifi.FlexInt:
|
case unifi.FlexInt:
|
||||||
val = v.Val
|
val = v.Val
|
||||||
default:
|
default:
|
||||||
|
errors++
|
||||||
if u.Config.ReportErrors {
|
if u.Config.ReportErrors {
|
||||||
ch <- prometheus.NewInvalidMetric(e.Desc, fmt.Errorf("not a number: %v", e.Value))
|
ch <- prometheus.NewInvalidMetric(e.Desc, fmt.Errorf("not a number: %v", e.Value))
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue