169 lines
4.0 KiB
Go
169 lines
4.0 KiB
Go
/*
|
|
Copyright (c) 2012, Matt T. Proud
|
|
All rights reserved.
|
|
|
|
Use of this source code is governed by a BSD-style
|
|
license that can be found in the LICENSE file.
|
|
*/
|
|
|
|
package metrics
|
|
|
|
import (
|
|
"fmt"
|
|
"github.com/matttproud/golang_instrumentation/maths"
|
|
"math"
|
|
"sync"
|
|
)
|
|
|
|
const (
|
|
lowerThird = 1.0 / 3.0
|
|
upperThird = 2.0 * lowerThird
|
|
)
|
|
|
|
/*
|
|
A TallyingIndexEstimator is responsible for estimating the value of index for
|
|
a given TallyingBucket, even though a TallyingBucket does not possess a
|
|
collection of samples. There are a few strategies listed below for how
|
|
this value should be approximated.
|
|
*/
|
|
type TallyingIndexEstimator func(minimum, maximum float64, index, observations int) float64
|
|
|
|
/*
|
|
Provide a filter for handling empty buckets.
|
|
*/
|
|
func emptyFilter(e TallyingIndexEstimator) TallyingIndexEstimator {
|
|
return func(minimum, maximum float64, index, observations int) float64 {
|
|
if observations == 0 {
|
|
return math.NaN()
|
|
}
|
|
|
|
return e(minimum, maximum, index, observations)
|
|
}
|
|
}
|
|
|
|
/*
|
|
Report the smallest observed value in the bucket.
|
|
*/
|
|
var Minimum TallyingIndexEstimator = emptyFilter(func(minimum, maximum float64, _, observations int) float64 {
|
|
return minimum
|
|
})
|
|
|
|
/*
|
|
Report the largest observed value in the bucket.
|
|
*/
|
|
var Maximum TallyingIndexEstimator = emptyFilter(func(minimum, maximum float64, _, observations int) float64 {
|
|
return maximum
|
|
})
|
|
|
|
/*
|
|
Report the average of the extrema.
|
|
*/
|
|
var Average TallyingIndexEstimator = emptyFilter(func(minimum, maximum float64, _, observations int) float64 {
|
|
return maths.Average([]float64{minimum, maximum})
|
|
})
|
|
|
|
/*
|
|
Report the minimum value of the index is in the lower-third of observations,
|
|
the average if in the middle-third, and the maximum if in the largest third.
|
|
*/
|
|
var Uniform TallyingIndexEstimator = emptyFilter(func(minimum, maximum float64, index, observations int) float64 {
|
|
if observations == 1 {
|
|
return minimum
|
|
}
|
|
|
|
location := float64(index) / float64(observations)
|
|
|
|
if location > upperThird {
|
|
return maximum
|
|
} else if location < lowerThird {
|
|
return minimum
|
|
}
|
|
|
|
return maths.Average([]float64{minimum, maximum})
|
|
})
|
|
|
|
/*
|
|
A TallyingBucket is a Bucket that tallies when an object is added to it.
|
|
Upon insertion, an object is compared against collected extrema and noted
|
|
as a new minimum or maximum if appropriate.
|
|
*/
|
|
type TallyingBucket struct {
|
|
estimator TallyingIndexEstimator
|
|
largestObserved float64
|
|
mutex sync.RWMutex
|
|
observations int
|
|
smallestObserved float64
|
|
}
|
|
|
|
func (b *TallyingBucket) Add(value float64) {
|
|
b.mutex.Lock()
|
|
defer b.mutex.Unlock()
|
|
|
|
b.observations += 1
|
|
b.smallestObserved = math.Min(value, b.smallestObserved)
|
|
b.largestObserved = math.Max(value, b.largestObserved)
|
|
}
|
|
|
|
func (b *TallyingBucket) String() string {
|
|
b.mutex.RLock()
|
|
defer b.mutex.RUnlock()
|
|
|
|
observations := b.observations
|
|
|
|
if observations == 0 {
|
|
return fmt.Sprintf("[TallyingBucket (Empty)]")
|
|
}
|
|
|
|
return fmt.Sprintf("[TallyingBucket (%f, %f); %d items]", b.smallestObserved, b.largestObserved, observations)
|
|
}
|
|
|
|
func (b *TallyingBucket) Observations() int {
|
|
b.mutex.RLock()
|
|
defer b.mutex.RUnlock()
|
|
|
|
return b.observations
|
|
}
|
|
|
|
func (b *TallyingBucket) ValueForIndex(index int) float64 {
|
|
b.mutex.RLock()
|
|
defer b.mutex.RUnlock()
|
|
|
|
return b.estimator(b.smallestObserved, b.largestObserved, index, b.observations)
|
|
}
|
|
|
|
func (b *TallyingBucket) Reset() {
|
|
b.mutex.Lock()
|
|
defer b.mutex.Unlock()
|
|
|
|
b.largestObserved = math.SmallestNonzeroFloat64
|
|
b.observations = 0
|
|
b.smallestObserved = math.MaxFloat64
|
|
}
|
|
|
|
/*
|
|
Produce a TallyingBucket with sane defaults.
|
|
*/
|
|
func DefaultTallyingBucket() TallyingBucket {
|
|
return TallyingBucket{
|
|
estimator: Minimum,
|
|
largestObserved: math.SmallestNonzeroFloat64,
|
|
smallestObserved: math.MaxFloat64,
|
|
}
|
|
}
|
|
|
|
func CustomTallyingBucket(estimator TallyingIndexEstimator) TallyingBucket {
|
|
return TallyingBucket{
|
|
estimator: estimator,
|
|
largestObserved: math.SmallestNonzeroFloat64,
|
|
smallestObserved: math.MaxFloat64,
|
|
}
|
|
}
|
|
|
|
/*
|
|
This is used strictly for testing.
|
|
*/
|
|
func TallyingBucketBuilder() Bucket {
|
|
b := DefaultTallyingBucket()
|
|
return &b
|
|
}
|