-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathtelemetry.go
201 lines (178 loc) · 6.09 KB
/
telemetry.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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
package telemetry
import (
"fmt"
"io"
"strings"
"time"
"github.com/uber-go/tally/v4"
"github.com/uber-go/tally/v4/prometheus"
)
var reporter = prometheus.NewReporter(prometheus.Options{})
var defaultBucketsForIntegerValues = tally.ValueBuckets{
1,
2,
5,
7,
9,
10,
50,
100,
}
var defaultBucketFactorsForDurations = []float64{
0.001,
0.005,
0.01,
0.05,
0.1,
0.25,
0.5,
0.75,
0.9,
0.95,
0.99,
1,
2.5,
5,
10,
25,
50,
100,
}
// Scope represents the measurements scope for an application.
type Scope struct {
scope tally.Scope
closer io.Closer
}
// NewScope creates a scope for an application. Receives a scope
// argument (a single-word) that is used as a prefix for all measurements.
func NewScope(scope string) *Scope {
s, closer := newRootScope(tally.ScopeOptions{
Prefix: scope,
}, 1*time.Second)
return &Scope{
scope: s,
closer: closer,
}
}
// Close closes the scope and stops reporting.
func (n *Scope) Close() error {
return n.closer.Close()
}
// RecordHit increases a hit counter. This is ideal for counting HTTP requests
// or other events that are incremented by one each time.
//
// RecordHit adds the "_total" suffix to the name of the measurement.
func (n *Scope) RecordHit(measurement string, tags map[string]string) {
record := n.scope.Tagged(tags).Counter(SnakeCasef(measurement + "_total"))
record.Inc(1.0)
}
func (n *Scope) RecordIncrementValue(measurement string, tags map[string]string, value int64) {
record := n.scope.Tagged(tags).Counter(SnakeCasef(measurement + "_total"))
record.Inc(value)
}
// RecordGauge sets the value of a measurement that can go up or down over
// time.
//
// RecordGauge measures a prometheus raw type and no suffix is added to the
// measurement.
func (n *Scope) RecordGauge(measurement string, tags map[string]string, value float64) {
record := n.scope.Tagged(tags).Gauge(SnakeCasef(measurement))
record.Update(value)
}
// RecordSize records a numeric unit-less value that can go up or down. Use it
// when it's more important to know the last value of said size. This is useful
// to measure things like the size of a queue.
//
// RecordSize adds the "_size" prefix to the name of the measurement.
func (n *Scope) RecordSize(measurement string, tags map[string]string, value float64) {
n.RecordGauge(SnakeCasef(measurement+"_size"), tags, value)
}
// RecordIntegerValue records a numeric unit-less value that can go up or down.
// Use it when is important to see how the value evolved over time.
//
// RecordIntegerValue uses an histogram configured with buckets that priorize
// values closer to zero.
func (n *Scope) RecordIntegerValue(measurement string, tags map[string]string, value int) {
record := n.scope.Tagged(tags).
Histogram(SnakeCasef(measurement+"_value"), defaultBucketsForIntegerValues)
record.RecordValue(float64(value))
}
// RecordValue records a numeric unit-less value that can go up or down. Use it
// when is important to see how the value evolved over time.
//
// RecordValue measures a prometheus raw type and no suffix is added to the measurement.
func (n *Scope) RecordValue(measurement string, tags map[string]string, value float64) {
n.RecordValueWithBuckets(SnakeCasef(measurement), tags, value, nil)
}
// RecordValueWithBuckets records a numeric unit-less value that can go up or
// down. Use it when is important to see how the value evolved over time.
//
// RecordValueWithBuckets adds the "_value" suffix to the name of the measurement.
func (n *Scope) RecordValueWithBuckets(measurement string, tags map[string]string, value float64, buckets []float64) {
record := n.scope.Tagged(tags).
Histogram(SnakeCasef(measurement+"_value"), tally.ValueBuckets(buckets))
record.RecordValue(value)
}
// RecordDuration records an elapsed time. Use it when is important to see how
// values. Use it when is important to see how a value evolved over time, for
// instance request durations, the time it takes for a task to finish, etc.
//
// RecordDuration adds the "_duration_seconds" prefix to the name of the
// measurement.
func (n *Scope) RecordDuration(measurement string, tags map[string]string, start time.Time, stop time.Time) {
n.RecordDurationWithResolution(SnakeCasef(measurement), tags, start, stop, 0)
}
// RecordDurationWithResolution records the elapsed duration between two time
// values. Use it when is important to see how a value evolved over time, for
// instance request durations, the time it takes for a task to finish, etc.
//
// The resolution parameter can be any value, this value will be taken as base
// to build buckets.
//
// RecordDurationWithResolution adds the "_duration_seconds" prefix to the name
// of the measurement.
func (n *Scope) RecordDurationWithResolution(measurement string, tags map[string]string, timeA time.Time, timeB time.Time, resolution time.Duration) {
var buckets tally.Buckets
if resolution <= 0 {
resolution = time.Second
}
unit := float64(resolution)
durations := make([]time.Duration, len(defaultBucketFactorsForDurations))
for i := range durations {
durations[i] = time.Duration(int64(unit * defaultBucketFactorsForDurations[i]))
}
buckets = tally.DurationBuckets(durations)
record := n.scope.Tagged(tags).Histogram(
SnakeCasef(measurement+"_duration_seconds"),
buckets,
)
elapsed := timeB.Sub(timeA)
if elapsed < 0 {
elapsed = elapsed * -1
}
record.RecordDuration(elapsed)
}
func (n *Scope) RecordSpan(measurement string, tags map[string]string) tally.Stopwatch {
return n.scope.Timer(SnakeCasef(measurement + "_span")).Start()
}
func newRootScope(opts tally.ScopeOptions, interval time.Duration) (tally.Scope, io.Closer) {
opts.CachedReporter = reporter
opts.Separator = prometheus.DefaultSeparator
opts.SanitizeOptions = &prometheus.DefaultSanitizerOpts
opts.OmitCardinalityMetrics = true
return tally.NewRootScope(opts, interval)
}
func SnakeCasef(format string, args ...any) string {
// TODO: memoize..?
s := strings.ToLower(fmt.Sprintf(format, args...))
if strings.Contains(s, "-") {
s = strings.ReplaceAll(s, "-", "_")
}
if strings.Contains(s, " ") {
s = strings.ReplaceAll(s, " ", "_")
}
if strings.Contains(s, ".") {
s = strings.ReplaceAll(s, ".", "_")
}
return s
}