-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmulticollector.go
62 lines (49 loc) · 1.29 KB
/
multicollector.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
package main
import (
"context"
"log"
"runtime"
"time"
"github.com/prometheus/client_golang/prometheus"
"golang.org/x/sync/errgroup"
)
type multiCollectorMember interface {
describe(chan<- *prometheus.Desc)
collect(context.Context, chan<- prometheus.Metric) error
}
type multiCollector struct {
// Impose a timeout on collection if non-zero.
timeout time.Duration
members []multiCollectorMember
}
var _ prometheus.Collector = (*multiCollector)(nil)
func newMultiCollector(m ...multiCollectorMember) *multiCollector {
return &multiCollector{
members: m,
}
}
func (c *multiCollector) Describe(ch chan<- *prometheus.Desc) {
for _, i := range c.members {
i.describe(ch)
}
}
func (c *multiCollector) Collect(ch chan<- prometheus.Metric) {
ctx := context.Background()
if c.timeout != 0 {
var cancel context.CancelFunc
ctx, cancel = context.WithTimeout(ctx, c.timeout)
defer cancel()
}
g, ctx := errgroup.WithContext(ctx)
g.SetLimit(runtime.GOMAXPROCS(0))
for _, i := range c.members {
collect := i.collect
g.Go(func() error { return collect(ctx, ch) })
}
if err := g.Wait(); err != nil {
log.Printf("Metrics collection failed: %v", err.Error())
ch <- prometheus.NewInvalidMetric(
prometheus.NewDesc("paperless_error", "Metrics collection failed", nil, nil),
err)
}
}