-
Notifications
You must be signed in to change notification settings - Fork 3.8k
/
metric.go
997 lines (861 loc) · 32.2 KB
/
metric.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
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
// Copyright 2015 The Cockroach Authors.
//
// Use of this software is governed by the Business Source License
// included in the file licenses/BSL.txt.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.
package metric
import (
"encoding/json"
"math"
"sort"
"sync/atomic"
"time"
"github.com/cockroachdb/cockroach/pkg/util"
"github.com/cockroachdb/cockroach/pkg/util/envutil"
"github.com/cockroachdb/cockroach/pkg/util/syncutil"
"github.com/cockroachdb/cockroach/pkg/util/timeutil"
"github.com/gogo/protobuf/proto"
"github.com/prometheus/client_golang/prometheus"
prometheusgo "github.com/prometheus/client_model/go"
"github.com/rcrowley/go-metrics"
)
// TestSampleInterval is passed to histograms during tests which don't
// want to concern themselves with supplying a "correct" interval.
const TestSampleInterval = time.Duration(math.MaxInt64)
// Iterable provides a method for synchronized access to interior objects.
type Iterable interface {
// GetName returns the fully-qualified name of the metric.
GetName() string
// GetHelp returns the help text for the metric.
GetHelp() string
// GetMeasurement returns the label for the metric, which describes the entity
// it measures.
GetMeasurement() string
// GetUnit returns the unit that should be used to display the metric
// (e.g. in bytes).
GetUnit() Unit
// GetMetadata returns the metric's metadata, which can be used in charts.
GetMetadata() Metadata
// Inspect calls the given closure with each contained item.
Inspect(func(interface{}))
}
// PrometheusExportable is the standard interface for an individual metric
// that can be exported to prometheus.
type PrometheusExportable interface {
// GetName is a method on Metadata
GetName() string
// GetHelp is a method on Metadata
GetHelp() string
// GetType returns the prometheus type enum for this metric.
GetType() *prometheusgo.MetricType
// GetLabels is a method on Metadata
GetLabels() []*prometheusgo.LabelPair
// ToPrometheusMetric returns a filled-in prometheus metric of the right type
// for the given metric. It does not fill in labels.
// The implementation must return thread-safe data to the caller, i.e.
// usually a copy of internal state.
ToPrometheusMetric() *prometheusgo.Metric
}
// PrometheusIterable is an extension of PrometheusExportable to indicate that
// this metric comprises children metrics which augment the parent's label
// values.
//
// The motivating use-case for this interface is the existence of tenants. We'd
// like to capture per-tenant metrics and expose them to prometheus while not
// polluting the internal tsdb.
type PrometheusIterable interface {
PrometheusExportable
// Each takes a slice of label pairs associated with the parent metric and
// calls the passed function with each of the children metrics.
Each([]*prometheusgo.LabelPair, func(metric *prometheusgo.Metric))
}
// WindowedHistogram represents a histogram with data over recent window of
// time. It's used primarily to record histogram data into CRDB's internal
// time-series database, which does not know how to encode cumulative
// histograms. What it does instead is scrape off sample count, sum of values,
// and values at specific quantiles from "windowed" histograms and record that
// data directly. These windows could be arbitrary and overlapping.
type WindowedHistogram interface {
// TotalWindowed returns the number of samples and their sum (respectively)
// in the current window.
TotalWindowed() (int64, float64)
// Total returns the number of samples and their sum (respectively) in the
// cumulative histogram.
Total() (int64, float64)
// MeanWindowed returns the average of the samples in the current window.
MeanWindowed() float64
// Mean returns the average of the sample in teh cumulative histogram.
Mean() float64
// ValueAtQuantileWindowed takes a quantile value [0,100] and returns the
// interpolated value at that quantile for the windowed histogram.
ValueAtQuantileWindowed(q float64) float64
}
// GetName returns the metric's name.
func (m *Metadata) GetName() string {
return m.Name
}
// GetHelp returns the metric's help string.
func (m *Metadata) GetHelp() string {
return m.Help
}
// GetMeasurement returns the entity measured by the metric.
func (m *Metadata) GetMeasurement() string {
return m.Measurement
}
// GetUnit returns the metric's unit of measurement.
func (m *Metadata) GetUnit() Unit {
return m.Unit
}
// GetLabels returns the metric's labels. For rationale behind the conversion
// from metric.LabelPair to prometheusgo.LabelPair, see the LabelPair comment
// in pkg/util/metric/metric.proto.
func (m *Metadata) GetLabels() []*prometheusgo.LabelPair {
lps := make([]*prometheusgo.LabelPair, len(m.Labels))
// x satisfies the field XXX_unrecognized in prometheusgo.LabelPair.
var x []byte
for i, v := range m.Labels {
lps[i] = &prometheusgo.LabelPair{Name: v.Name, Value: v.Value, XXX_unrecognized: x}
}
return lps
}
// AddLabel adds a label/value pair for this metric.
func (m *Metadata) AddLabel(name, value string) {
m.Labels = append(m.Labels,
&LabelPair{
Name: proto.String(exportedLabel(name)),
Value: proto.String(value),
})
}
var _ Iterable = &Gauge{}
var _ Iterable = &GaugeFloat64{}
var _ Iterable = &Counter{}
var _ json.Marshaler = &Gauge{}
var _ json.Marshaler = &GaugeFloat64{}
var _ json.Marshaler = &Counter{}
var _ json.Marshaler = &Registry{}
var _ PrometheusExportable = &Gauge{}
var _ PrometheusExportable = &GaugeFloat64{}
var _ PrometheusExportable = &Counter{}
type periodic interface {
nextTick() time.Time
tick()
}
var now = timeutil.Now
// TestingSetNow changes the clock used by the metric system. For use by
// testing to precisely control the clock.
func TestingSetNow(f func() time.Time) func() {
origNow := now
now = f
return func() {
now = origNow
}
}
func maybeTick(m periodic) {
for m.nextTick().Before(now()) {
m.tick()
}
}
// useHdrHistogramsEnvVar can be used to switch all histograms to use the
// legacy HDR histograms (except for those that explicitly force the use
// of the newer Prometheus via HistogramModePrometheus). HDR Histograms
// dynamically generate bucket boundaries, which can lead to hundreds of
// buckets. This can cause performance issues with timeseries databases
// like Prometheus.
const useHdrHistogramsEnvVar = "COCKROACH_ENABLE_HDR_HISTOGRAMS"
var hdrEnabled = util.ConstantWithMetamorphicTestBool(useHdrHistogramsEnvVar, envutil.EnvOrDefaultBool(useHdrHistogramsEnvVar, false))
// HdrEnabled returns whether or not the HdrHistogram model is enabled
// in the metric package. Primarily useful in tests where we want to validate
// different outputs depending on whether or not HDR is enabled.
func HdrEnabled() bool {
return hdrEnabled
}
type HistogramMode byte
const (
// HistogramModePrometheus will force the constructed histogram to use
// the Prometheus histogram model, regardless of the value of
// useHdrHistogramsEnvVar. This option should be used for all
// newly defined histograms moving forward.
//
// NB: If neither this mode nor the HistogramModePreferHdrLatency mode
// is set, MaxVal and SigFigs must be defined to maintain backwards
// compatibility with the legacy HdrHistogram model.
HistogramModePrometheus HistogramMode = iota + 1
// HistogramModePreferHdrLatency will cause the returned histogram to
// use the HdrHistgoram model and be configured with suitable defaults
// for latency tracking iff useHdrHistogramsEnvVar is enabled.
//
// NB: If this option is set, no MaxVal or SigFigs are required in the
// HistogramOptions to maintain backwards compatibility with the legacy
// HdrHistogram model, since suitable defaults are used for both.
HistogramModePreferHdrLatency
)
type HistogramOptions struct {
// Metadata is the metric Metadata associated with the histogram.
Metadata Metadata
// Duration is the histogram's window duration.
Duration time.Duration
// MaxVal is only relevant to the HdrHistogram, and represents the
// highest trackable value in the resulting histogram buckets.
MaxVal int64
// SigFigs is only relevant to the HdrHistogram, and represents
// the number of significant figures to be used to determine the
// degree of accuracy used in measurements.
SigFigs int
// Buckets are only relevant to Prometheus histograms, and represent
// the pre-defined histogram bucket boundaries to be used.
Buckets []float64
// Mode defines the type of histogram to be used. See individual
// comments on each HistogramMode value for details.
Mode HistogramMode
}
func NewHistogram(opt HistogramOptions) IHistogram {
if hdrEnabled && opt.Mode != HistogramModePrometheus {
if opt.Mode == HistogramModePreferHdrLatency {
return NewHdrLatency(opt.Metadata, opt.Duration)
} else {
return NewHdrHistogram(opt.Metadata, opt.Duration, opt.MaxVal, opt.SigFigs)
}
} else {
return newHistogram(opt.Metadata, opt.Duration, opt.Buckets)
}
}
// NewHistogram is a prometheus-backed histogram. Depending on the value of
// opts.Buckets, this is suitable for recording any kind of quantity. Common
// sensible choices are {IO,Network}LatencyBuckets.
func newHistogram(meta Metadata, windowDuration time.Duration, buckets []float64) *Histogram {
// TODO(obs-inf): prometheus supports labeled histograms but they require more
// plumbing and don't fit into the PrometheusObservable interface any more.
opts := prometheus.HistogramOpts{
Buckets: buckets,
}
cum := prometheus.NewHistogram(opts)
h := &Histogram{
Metadata: meta,
cum: cum,
}
h.windowed.tickHelper = &tickHelper{
nextT: now(),
tickInterval: windowDuration,
onTick: func() {
h.windowed.prev = h.windowed.cur
h.windowed.cur = prometheus.NewHistogram(opts)
},
}
h.windowed.tickHelper.onTick()
return h
}
var _ periodic = (*Histogram)(nil)
var _ PrometheusExportable = (*Histogram)(nil)
var _ WindowedHistogram = (*Histogram)(nil)
// Histogram is a prometheus-backed histogram. It collects observed values by
// keeping bucketed counts. For convenience, internally two sets of buckets are
// kept: A cumulative set (i.e. data is never evicted) and a windowed set (which
// keeps only recently collected samples).
//
// New buckets are created using TestHistogramBuckets.
type Histogram struct {
Metadata
cum prometheus.Histogram
// TODO(obs-inf): the way we implement windowed histograms is not great. If
// the windowed histogram is pulled right after a tick, it will be mostly
// empty. We could add a third bucket and represent the merged view of the two
// most recent buckets to avoid that. Or we could "just" double the rotation
// interval (so that the histogram really collects for 20s when we expect to
// persist the contents every 10s). Really it would make more sense to
// explicitly rotate the histogram atomically with collecting its contents,
// but that is now how we have set it up right now. It should be doable
// though, since there is only one consumer of windowed histograms - our
// internal timeseries system.
windowed struct {
// prometheus.Histogram is thread safe, so we only
// need an RLock to record into it. But write lock
// is held while rotating.
syncutil.RWMutex
*tickHelper
prev, cur prometheus.Histogram
}
}
type IHistogram interface {
Iterable
PrometheusExportable
WindowedHistogram
RecordValue(n int64)
Total() (int64, float64)
Mean() float64
}
var _ IHistogram = &Histogram{}
func (h *Histogram) nextTick() time.Time {
h.windowed.RLock()
defer h.windowed.RUnlock()
return h.windowed.nextTick()
}
func (h *Histogram) tick() {
h.windowed.Lock()
defer h.windowed.Unlock()
h.windowed.tick()
}
// Windowed returns a copy of the current windowed histogram.
func (h *Histogram) Windowed() prometheus.Histogram {
h.windowed.RLock()
defer h.windowed.RUnlock()
return h.windowed.cur
}
// RecordValue adds the given value to the histogram.
func (h *Histogram) RecordValue(n int64) {
v := float64(n)
h.cum.Observe(v)
h.windowed.RLock()
defer h.windowed.RUnlock()
h.windowed.cur.Observe(v)
}
// GetType returns the prometheus type enum for this metric.
func (h *Histogram) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_HISTOGRAM.Enum()
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (h *Histogram) ToPrometheusMetric() *prometheusgo.Metric {
m := &prometheusgo.Metric{}
if err := h.cum.Write(m); err != nil {
panic(err)
}
return m
}
// ToPrometheusMetricWindowed returns a filled-in prometheus metric of the
// right type.
func (h *Histogram) ToPrometheusMetricWindowed() *prometheusgo.Metric {
h.windowed.Lock()
defer h.windowed.Unlock()
m := &prometheusgo.Metric{}
if err := h.windowed.cur.Write(m); err != nil {
panic(err)
}
return m
}
// ToPrometheusMetricWindowedMerged returns a filled-in prometheus metric of the
// right type, with the previous window's histogram bucket counts and sample
// count added to those of the current one.
// Merging the sample count is necessary to correctly calculate the value at
// a given quantile with the merged bucket counts.
func (h *Histogram) ToPrometheusMetricWindowedMerged() *prometheusgo.Metric {
h.windowed.Lock()
defer h.windowed.Unlock()
cur := &prometheusgo.Metric{}
prev := &prometheusgo.Metric{}
if err := h.windowed.cur.Write(cur); err != nil {
panic(err)
}
if h.windowed.prev != nil {
if err := h.windowed.prev.Write(prev); err != nil {
panic(err)
}
MergeWindowedHistogram(cur.Histogram, prev.Histogram)
}
return cur
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (h *Histogram) GetMetadata() Metadata {
return h.Metadata
}
// Inspect calls the closure.
func (h *Histogram) Inspect(f func(interface{})) {
h.windowed.Lock()
maybeTick(&h.windowed)
h.windowed.Unlock()
f(h)
}
// Total returns the (cumulative) number of samples and the sum of all samples.
func (h *Histogram) Total() (int64, float64) {
pHist := h.ToPrometheusMetric().Histogram
return int64(pHist.GetSampleCount()), pHist.GetSampleSum()
}
// TotalWindowed implements the WindowedHistogram interface.
func (h *Histogram) TotalWindowed() (int64, float64) {
pHist := h.ToPrometheusMetricWindowed().Histogram
return int64(pHist.GetSampleCount()), pHist.GetSampleSum()
}
// Mean returns the (cumulative) mean of samples.
func (h *Histogram) Mean() float64 {
pm := h.ToPrometheusMetric()
return pm.Histogram.GetSampleSum() / float64(pm.Histogram.GetSampleCount())
}
// MeanWindowed implements the WindowedHistogram interface.
func (h *Histogram) MeanWindowed() float64 {
pHist := h.ToPrometheusMetricWindowed().Histogram
return pHist.GetSampleSum() / float64(pHist.GetSampleCount())
}
// ValueAtQuantileWindowed implements the WindowedHistogram interface.
//
// https://github.com/prometheus/prometheus/blob/d9162189/promql/quantile.go#L75
// This function is mostly taken from a prometheus internal function that
// does the same thing. There are a few differences for our use case:
// 1. As a user of the prometheus go client library, we don't have access
// to the implicit +Inf bucket, so we don't need special cases to deal
// with the quantiles that include the +Inf bucket.
// 2. Since the prometheus client library ensures buckets are in a strictly
// increasing order at creation, we do not sort them.
func (h *Histogram) ValueAtQuantileWindowed(q float64) float64 {
return ValueAtQuantileWindowed(h.ToPrometheusMetricWindowedMerged().Histogram,
q)
}
var _ PrometheusExportable = (*ManualWindowHistogram)(nil)
var _ Iterable = (*ManualWindowHistogram)(nil)
var _ WindowedHistogram = (*ManualWindowHistogram)(nil)
// NewManualWindowHistogram is a prometheus-backed histogram. Depending on the
// value of the buckets parameter, this is suitable for recording any kind of
// quantity. The histogram is very similar to Histogram produced by
// NewHistogram with the main difference being that Histogram supports
// collecting values over time using the Histogram.RecordValue whereas this
// histogram provides limited support RecordValue, the caller is responsible
// for calling Rotate, after recording is complete or manually providing the
// cumulative and current windowed histogram via Update. This means that it is
// the responsibility of the creator of this histogram to replace the values by
// either calling ManualWindowHistogram.Update or
// ManualWindowHistogram.RecordValue and ManualWindowHistogram.Rotate. If
// NewManualWindowHistogram is called withRotate as true, only the RecordValue
// and Rotate method may be used; withRotate as false, only Update may be used.
// TODO(kvoli,aadityasondhi): The two ways to use this histogram is a hack and
// "temporary", rationalize the interface. Tracked in #98622.
func NewManualWindowHistogram(
meta Metadata, buckets []float64, withRotate bool,
) *ManualWindowHistogram {
opts := prometheus.HistogramOpts{
Buckets: buckets,
}
cum := prometheus.NewHistogram(opts)
prev := &prometheusgo.Metric{}
if err := cum.Write(prev); err != nil {
panic(err.Error())
}
h := &ManualWindowHistogram{
Metadata: meta,
}
h.mu.rotating = withRotate
h.mu.cum = cum
h.mu.prev = prev.GetHistogram()
return h
}
// ManualWindowHistogram is a prometheus-backed histogram. Internally there are
// three sets of histograms: one is the cumulative set (i.e. data is never
// evicted) which is a prometheus.Histogram, the cumulative histogram value
// when last rotated and the current histogram, which is windowed. Both the
// previous and current histograms are prometheusgo.Histograms. Both histograms
// must be updated by the client by calling either ManualWindowHistogram.Update
// or ManualWindowHistogram.RecordValue and subsequently Rotate.
type ManualWindowHistogram struct {
Metadata
mu struct {
// prometheus.Histogram is thread safe, so we only need an RLock to
// RecordValue. When calling Update or Rotate, we require a WLock since we
// swap out fields.
syncutil.RWMutex
rotating bool
cum prometheus.Histogram
prev, cur *prometheusgo.Histogram
}
}
// Update replaces the cumulative and current windowed histograms.
func (mwh *ManualWindowHistogram) Update(cum prometheus.Histogram, cur *prometheusgo.Histogram) {
mwh.mu.Lock()
defer mwh.mu.Unlock()
if mwh.mu.rotating {
panic("Unexpected call to Update with rotate enabled")
}
mwh.mu.cum = cum
mwh.mu.cur = cur
}
// RecordValue records a value to the cumulative histogram. The value is only
// added to the current window histogram once Rotate is called.
func (mwh *ManualWindowHistogram) RecordValue(val float64) {
mwh.mu.RLock()
defer mwh.mu.RUnlock()
if !mwh.mu.rotating {
panic("Unexpected call to RecordValue with rotate disabled")
}
mwh.mu.cum.Observe(val)
}
// SubtractPrometheusHistograms subtracts the prev histogram from the cur
// histogram, in place modifying the cur histogram. The bucket boundaries must
// be identical for both prev and cur.
func SubtractPrometheusHistograms(cur *prometheusgo.Histogram, prev *prometheusgo.Histogram) {
prevBuckets := prev.GetBucket()
curBuckets := cur.GetBucket()
*cur.SampleCount -= prev.GetSampleCount()
*cur.SampleSum -= prev.GetSampleSum()
for idx, v := range prevBuckets {
if *curBuckets[idx].UpperBound != *v.UpperBound {
panic("Bucket Upperbounds don't match")
}
*curBuckets[idx].CumulativeCount -= *v.CumulativeCount
}
}
// Rotate sets the current windowed histogram (cur) to be the delta of the
// cumulative histogram at the last rotation (prev) and the cumulative
// histogram currently (cum).
func (mwh *ManualWindowHistogram) Rotate() error {
mwh.mu.Lock()
defer mwh.mu.Unlock()
if !mwh.mu.rotating {
panic("Unexpected call to Rotate with rotate disabled")
}
cur := &prometheusgo.Metric{}
if err := mwh.mu.cum.Write(cur); err != nil {
return err
}
SubtractPrometheusHistograms(cur.GetHistogram(), mwh.mu.prev)
mwh.mu.cur = cur.GetHistogram()
prev := &prometheusgo.Metric{}
if err := mwh.mu.cum.Write(prev); err != nil {
return err
}
mwh.mu.prev = prev.GetHistogram()
return nil
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (mwh *ManualWindowHistogram) GetMetadata() Metadata {
return mwh.Metadata
}
// Inspect calls the closure.
func (mwh *ManualWindowHistogram) Inspect(f func(interface{})) {
f(mwh)
}
// GetType returns the prometheus type enum for this metric.
func (mwh *ManualWindowHistogram) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_HISTOGRAM.Enum()
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (mwh *ManualWindowHistogram) ToPrometheusMetric() *prometheusgo.Metric {
mwh.mu.RLock()
defer mwh.mu.RUnlock()
m := &prometheusgo.Metric{}
if err := mwh.mu.cum.Write(m); err != nil {
panic(err)
}
return m
}
// TotalWindowed implements the WindowedHistogram interface.
func (mwh *ManualWindowHistogram) TotalWindowed() (int64, float64) {
mwh.mu.RLock()
defer mwh.mu.RUnlock()
return int64(mwh.mu.cur.GetSampleCount()), mwh.mu.cur.GetSampleSum()
}
// Total implements the WindowedHistogram interface.
func (mwh *ManualWindowHistogram) Total() (int64, float64) {
h := mwh.ToPrometheusMetric().Histogram
return int64(h.GetSampleCount()), h.GetSampleSum()
}
func (mwh *ManualWindowHistogram) MeanWindowed() float64 {
mwh.mu.RLock()
defer mwh.mu.RUnlock()
return mwh.mu.cur.GetSampleSum() / float64(mwh.mu.cur.GetSampleCount())
}
func (mwh *ManualWindowHistogram) Mean() float64 {
h := mwh.ToPrometheusMetric().Histogram
return h.GetSampleSum() / float64(h.GetSampleCount())
}
// ValueAtQuantileWindowed implements the WindowedHistogram interface.
//
// This function is very similar to Histogram.ValueAtQuantileWindowed. Thus see
// Histogram.ValueAtQuantileWindowed for a more in-depth description.
func (mwh *ManualWindowHistogram) ValueAtQuantileWindowed(q float64) float64 {
mwh.mu.RLock()
defer mwh.mu.RUnlock()
if mwh.mu.cur == nil {
return 0
}
return ValueAtQuantileWindowed(mwh.mu.cur, q)
}
// A Counter holds a single mutable atomic value.
type Counter struct {
Metadata
metrics.Counter
}
// NewCounter creates a counter.
func NewCounter(metadata Metadata) *Counter {
return &Counter{metadata, metrics.NewCounter()}
}
// Dec overrides the metric.Counter method. This method should NOT be
// used and serves only to prevent misuse of the metric type.
func (c *Counter) Dec(int64) {
// From https://prometheus.io/docs/concepts/metric_types/#counter
// > Counters should not be used to expose current counts of items
// > whose number can also go down, e.g. the number of currently
// > running goroutines. Use gauges for this use case.
panic("Counter should not be decremented, use a Gauge instead")
}
// GetType returns the prometheus type enum for this metric.
func (c *Counter) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_COUNTER.Enum()
}
// Inspect calls the given closure with the empty string and itself.
func (c *Counter) Inspect(f func(interface{})) { f(c) }
// MarshalJSON marshals to JSON.
func (c *Counter) MarshalJSON() ([]byte, error) {
return json.Marshal(c.Counter.Count())
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (c *Counter) ToPrometheusMetric() *prometheusgo.Metric {
return &prometheusgo.Metric{
Counter: &prometheusgo.Counter{Value: proto.Float64(float64(c.Counter.Count()))},
}
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (c *Counter) GetMetadata() Metadata {
baseMetadata := c.Metadata
baseMetadata.MetricType = prometheusgo.MetricType_COUNTER
return baseMetadata
}
type CounterFloat64 struct {
Metadata
count syncutil.AtomicFloat64
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (c *CounterFloat64) GetMetadata() Metadata {
baseMetadata := c.Metadata
baseMetadata.MetricType = prometheusgo.MetricType_COUNTER
return baseMetadata
}
func (c *CounterFloat64) Clear() {
syncutil.StoreFloat64(&c.count, 0)
}
func (c *CounterFloat64) Count() float64 {
return syncutil.LoadFloat64(&c.count)
}
func (c *CounterFloat64) Inc(i float64) {
syncutil.AddFloat64(&c.count, i)
}
func (c *CounterFloat64) UpdateIfHigher(i float64) {
syncutil.StoreFloat64IfHigher(&c.count, i)
}
func (c *CounterFloat64) Snapshot() *CounterFloat64 {
newCounter := NewCounterFloat64(c.Metadata)
syncutil.StoreFloat64(&newCounter.count, c.Count())
return newCounter
}
// GetType returns the prometheus type enum for this metric.
func (c *CounterFloat64) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_COUNTER.Enum()
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (c *CounterFloat64) ToPrometheusMetric() *prometheusgo.Metric {
return &prometheusgo.Metric{
Counter: &prometheusgo.Counter{Value: proto.Float64(c.Count())},
}
}
// MarshalJSON marshals to JSON.
func (c *CounterFloat64) MarshalJSON() ([]byte, error) {
return json.Marshal(c.Count())
}
// NewCounterFloat64 creates a counter.
func NewCounterFloat64(metadata Metadata) *CounterFloat64 {
return &CounterFloat64{Metadata: metadata}
}
// A Gauge atomically stores a single integer value.
type Gauge struct {
Metadata
value *int64
fn func() int64
}
// NewGauge creates a Gauge.
func NewGauge(metadata Metadata) *Gauge {
return &Gauge{metadata, new(int64), nil}
}
// NewFunctionalGauge creates a Gauge metric whose value is determined when
// asked for by calling the provided function.
// Note that Update, Inc, and Dec should NOT be called on a Gauge returned
// from NewFunctionalGauge.
func NewFunctionalGauge(metadata Metadata, f func() int64) *Gauge {
return &Gauge{metadata, nil, f}
}
// Snapshot returns a read-only copy of the gauge.
func (g *Gauge) Snapshot() metrics.Gauge {
return metrics.GaugeSnapshot(g.Value())
}
// Update updates the gauge's value.
func (g *Gauge) Update(v int64) {
atomic.StoreInt64(g.value, v)
}
// Value returns the gauge's current value.
func (g *Gauge) Value() int64 {
if g.fn != nil {
return g.fn()
}
return atomic.LoadInt64(g.value)
}
// Inc increments the gauge's value.
func (g *Gauge) Inc(i int64) {
atomic.AddInt64(g.value, i)
}
// Dec decrements the gauge's value.
func (g *Gauge) Dec(i int64) {
atomic.AddInt64(g.value, -i)
}
// GetType returns the prometheus type enum for this metric.
func (g *Gauge) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_GAUGE.Enum()
}
// Inspect calls the given closure with the empty string and itself.
func (g *Gauge) Inspect(f func(interface{})) { f(g) }
// MarshalJSON marshals to JSON.
func (g *Gauge) MarshalJSON() ([]byte, error) {
return json.Marshal(g.Value())
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (g *Gauge) ToPrometheusMetric() *prometheusgo.Metric {
return &prometheusgo.Metric{
Gauge: &prometheusgo.Gauge{Value: proto.Float64(float64(g.Value()))},
}
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (g *Gauge) GetMetadata() Metadata {
baseMetadata := g.Metadata
baseMetadata.MetricType = prometheusgo.MetricType_GAUGE
return baseMetadata
}
// A GaugeFloat64 atomically stores a single float64 value.
type GaugeFloat64 struct {
Metadata
bits *uint64
}
// NewGaugeFloat64 creates a GaugeFloat64.
func NewGaugeFloat64(metadata Metadata) *GaugeFloat64 {
return &GaugeFloat64{metadata, new(uint64)}
}
// Snapshot returns a read-only copy of the gauge.
func (g *GaugeFloat64) Snapshot() metrics.GaugeFloat64 {
return metrics.GaugeFloat64Snapshot(g.Value())
}
// Update updates the gauge's value.
func (g *GaugeFloat64) Update(v float64) {
atomic.StoreUint64(g.bits, math.Float64bits(v))
}
// Value returns the gauge's current value.
func (g *GaugeFloat64) Value() float64 {
return math.Float64frombits(atomic.LoadUint64(g.bits))
}
// Inc increments the gauge's value.
func (g *GaugeFloat64) Inc(delta float64) {
for {
oldBits := atomic.LoadUint64(g.bits)
newBits := math.Float64bits(math.Float64frombits(oldBits) + delta)
if atomic.CompareAndSwapUint64(g.bits, oldBits, newBits) {
return
}
}
}
// Dec decrements the gauge's value.
func (g *GaugeFloat64) Dec(delta float64) {
g.Inc(-delta)
}
// GetType returns the prometheus type enum for this metric.
func (g *GaugeFloat64) GetType() *prometheusgo.MetricType {
return prometheusgo.MetricType_GAUGE.Enum()
}
// Inspect calls the given closure with itself.
func (g *GaugeFloat64) Inspect(f func(interface{})) { f(g) }
// MarshalJSON marshals to JSON.
func (g *GaugeFloat64) MarshalJSON() ([]byte, error) {
return json.Marshal(g.Value())
}
// ToPrometheusMetric returns a filled-in prometheus metric of the right type.
func (g *GaugeFloat64) ToPrometheusMetric() *prometheusgo.Metric {
return &prometheusgo.Metric{
Gauge: &prometheusgo.Gauge{Value: proto.Float64(g.Value())},
}
}
// GetMetadata returns the metric's metadata including the Prometheus
// MetricType.
func (g *GaugeFloat64) GetMetadata() Metadata {
baseMetadata := g.Metadata
baseMetadata.MetricType = prometheusgo.MetricType_GAUGE
return baseMetadata
}
// MergeWindowedHistogram adds the bucket counts SampleCount from the
// previous windowed histogram to those of the current windowed histogram.
// NB: Buckets on each histogram must be the same
func MergeWindowedHistogram(cur *prometheusgo.Histogram, prev *prometheusgo.Histogram) {
for i, bucket := range cur.Bucket {
count := *bucket.CumulativeCount + *prev.Bucket[i].CumulativeCount
*bucket.CumulativeCount = count
}
sampleCount := *cur.SampleCount + *prev.SampleCount
*cur.SampleCount = sampleCount
}
// ValueAtQuantileWindowed takes a quantile value [0,100] and returns the
// interpolated value at that quantile for the given histogram.
func ValueAtQuantileWindowed(histogram *prometheusgo.Histogram, q float64) float64 {
buckets := histogram.Bucket
n := float64(*histogram.SampleCount)
if n == 0 {
return 0
}
// NB: The 0.5 is added for rounding purposes; it helps in cases where
// SampleCount is small.
rank := uint64(((q / 100) * n) + 0.5)
// Since we are missing the +Inf bucket, CumulativeCounts may never exceed
// rank. By omitting the highest bucket we have from the search, the failed
// search will land on that last bucket and we don't have to do any special
// checks regarding landing on a non-existent bucket.
b := sort.Search(len(buckets)-1, func(i int) bool { return *buckets[i].CumulativeCount >= rank })
var (
bucketStart float64 // defaults to 0, which we assume is the lower bound of the smallest bucket
bucketEnd = *buckets[b].UpperBound
count = *buckets[b].CumulativeCount
)
// Calculate the linearly interpolated value within the bucket.
if b > 0 {
bucketStart = *buckets[b-1].UpperBound
count -= *buckets[b-1].CumulativeCount
rank -= *buckets[b-1].CumulativeCount
}
val := bucketStart + (bucketEnd-bucketStart)*(float64(rank)/float64(count))
if math.IsNaN(val) || math.IsInf(val, -1) {
return 0
}
// Should not extrapolate past the upper bound of the largest bucket.
//
// NB: SampleCount includes the implicit +Inf bucket but the
// buckets[len(buckets)-1].UpperBound refers to the largest bucket defined
// by us -- the client library doesn't give us access to the +Inf bucket
// which Prometheus uses under the hood. With a high enough quantile, the
// val computed further below surpasses the upper bound of the largest
// bucket. Using that interpolated value feels wrong since we'd be
// extrapolating. Also, for specific metrics if we see our q99 values to be
// hitting the top-most bucket boundary, that's an indication for us to
// choose better buckets for more accuracy. It's also worth noting that the
// prometheus client library does the same thing when the resulting value is
// in the +Inf bucket, whereby they return the upper bound of the second
// last bucket -- see [1].
//
// [1]: https://github.com/prometheus/prometheus/blob/d9162189/promql/quantile.go#L103.
if val > *buckets[len(buckets)-1].UpperBound {
return *buckets[len(buckets)-1].UpperBound
}
return val
}
// Quantile is a quantile along with a string suffix to be attached to the metric
// name upon recording into the internal TSDB.
type Quantile struct {
Suffix string
Quantile float64
}
// RecordHistogramQuantiles are the quantiles at which (windowed) histograms
// are recorded into the internal TSDB.
var RecordHistogramQuantiles = []Quantile{
{"-max", 100},
{"-p99.999", 99.999},
{"-p99.99", 99.99},
{"-p99.9", 99.9},
{"-p99", 99},
{"-p90", 90},
{"-p75", 75},
{"-p50", 50},
}