mirror of
https://github.com/weaveworks/scope.git
synced 2026-05-21 08:33:05 +00:00
``` $ gvt delete github.com/weaveworks/common $ gvt fetch --revision 4d96fd8dcf2c7b417912c6219b310112cb4a4626 github.com/weaveworks/common 2018/07/23 15:31:11 Fetching: github.com/weaveworks/common 2018/07/23 15:31:14 · Skipping (existing): github.com/golang/protobuf/ptypes/any 2018/07/23 15:31:14 · Fetching recursive dependency: github.com/pkg/errors 2018/07/23 15:31:16 · Skipping (existing): github.com/aws/aws-sdk-go/aws 2018/07/23 15:31:16 · Fetching recursive dependency: github.com/sirupsen/logrus 2018/07/23 15:31:18 ·· Skipping (existing): golang.org/x/sys/unix 2018/07/23 15:31:18 ·· Skipping (existing): golang.org/x/crypto/ssh/terminal 2018/07/23 15:31:18 · Skipping (existing): google.golang.org/grpc/status 2018/07/23 15:31:18 · Skipping (existing): github.com/gorilla/mux 2018/07/23 15:31:18 · Fetching recursive dependency: github.com/opentracing-contrib/go-stdlib/nethttp 2018/07/23 15:31:20 ·· Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:31:20 ·· Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:31:20 ·· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:31:20 · Skipping (existing): github.com/prometheus/client_golang/prometheus 2018/07/23 15:31:20 · Skipping (existing): google.golang.org/grpc 2018/07/23 15:31:20 · Skipping (existing): github.com/pmezard/go-difflib/difflib 2018/07/23 15:31:20 · Fetching recursive dependency: github.com/go-kit/kit/log 2018/07/23 15:31:23 ·· Fetching recursive dependency: github.com/go-logfmt/logfmt 2018/07/23 15:31:25 ··· Fetching recursive dependency: github.com/kr/logfmt 2018/07/23 15:31:27 ·· Fetching recursive dependency: github.com/go-stack/stack 2018/07/23 15:31:29 · Fetching recursive dependency: google.golang.org/genproto/googleapis/rpc/status 2018/07/23 15:31:37 ·· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:31:37 ·· Skipping (existing): github.com/golang/protobuf/ptypes/any 2018/07/23 15:31:37 · Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:31:37 · Fetching recursive dependency: github.com/sercand/kuberesolver 2018/07/23 15:31:39 ·· Skipping (existing): google.golang.org/grpc/grpclog 2018/07/23 15:31:39 ·· Skipping (existing): google.golang.org/grpc/resolver 2018/07/23 15:31:39 ·· Skipping (existing): golang.org/x/net/context 2018/07/23 15:31:39 · Skipping (existing): google.golang.org/grpc/metadata 2018/07/23 15:31:39 · Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:31:39 · Skipping (existing): github.com/armon/go-socks5 2018/07/23 15:31:39 · Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:31:39 · Skipping (existing): github.com/davecgh/go-spew/spew 2018/07/23 15:31:39 · Skipping (existing): github.com/golang/protobuf/ptypes 2018/07/23 15:31:39 · Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:31:39 · Fetching recursive dependency: github.com/grpc-ecosystem/grpc-opentracing/go/otgrpc 2018/07/23 15:31:41 ·· Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:31:41 ·· Skipping (existing): golang.org/x/net/context 2018/07/23 15:31:41 ·· Skipping (existing): google.golang.org/grpc/codes 2018/07/23 15:31:41 ·· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:31:41 ·· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:31:41 ·· Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:31:41 ·· Skipping (existing): google.golang.org/grpc 2018/07/23 15:31:41 ·· Skipping (existing): google.golang.org/grpc/metadata 2018/07/23 15:31:41 ·· Skipping (existing): google.golang.org/grpc/status 2018/07/23 15:31:41 · Fetching recursive dependency: github.com/uber/jaeger-client-go/config 2018/07/23 15:31:44 ·· Fetching recursive dependency: github.com/uber/jaeger-client-go/internal/throttler/remote 2018/07/23 15:31:44 ··· Fetching recursive dependency: github.com/uber/jaeger-client-go/utils 2018/07/23 15:31:44 ···· Fetching recursive dependency: github.com/uber/jaeger-client-go/thrift 2018/07/23 15:31:44 ···· Fetching recursive dependency: github.com/uber/jaeger-client-go/thrift-gen/agent 2018/07/23 15:31:44 ····· Fetching recursive dependency: github.com/uber/jaeger-client-go/thrift-gen/jaeger 2018/07/23 15:31:44 ····· Fetching recursive dependency: github.com/uber/jaeger-client-go/thrift-gen/zipkincore 2018/07/23 15:31:44 ··· Fetching recursive dependency: github.com/uber/jaeger-client-go 2018/07/23 15:31:44 ···· Fetching recursive dependency: github.com/crossdock/crossdock-go 2018/07/23 15:31:46 ····· Skipping (existing): github.com/davecgh/go-spew/spew 2018/07/23 15:31:46 ····· Skipping (existing): golang.org/x/net/context/ctxhttp 2018/07/23 15:31:46 ····· Skipping (existing): golang.org/x/net/context 2018/07/23 15:31:46 ····· Skipping (existing): github.com/pmezard/go-difflib/difflib 2018/07/23 15:31:46 ···· Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:31:46 ···· Fetching recursive dependency: go.uber.org/zap/zapcore 2018/07/23 15:31:49 ····· Fetching recursive dependency: go.uber.org/atomic 2018/07/23 15:31:51 ····· Fetching recursive dependency: go.uber.org/zap/internal/bufferpool 2018/07/23 15:31:51 ······ Fetching recursive dependency: go.uber.org/zap/buffer 2018/07/23 15:31:51 ····· Fetching recursive dependency: go.uber.org/multierr 2018/07/23 15:31:54 ····· Fetching recursive dependency: go.uber.org/zap/internal/exit 2018/07/23 15:31:54 ····· Fetching recursive dependency: go.uber.org/zap/internal/color 2018/07/23 15:31:54 ···· Fetching recursive dependency: go.uber.org/zap 2018/07/23 15:31:54 ···· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:31:54 ···· Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:31:54 ···· Fetching recursive dependency: github.com/uber/jaeger-lib/metrics 2018/07/23 15:31:56 ····· Fetching recursive dependency: github.com/uber-go/tally 2018/07/23 15:31:58 ······ Fetching recursive dependency: github.com/m3db/prometheus_client_golang/prometheus/promhttp 2018/07/23 15:32:00 ······· Skipping (existing): github.com/prometheus/client_golang/prometheus 2018/07/23 15:32:00 ······· Skipping (existing): github.com/prometheus/common/expfmt 2018/07/23 15:32:00 ······· Skipping (existing): github.com/prometheus/client_model/go 2018/07/23 15:32:00 ······ Fetching recursive dependency: gopkg.in/validator.v2 2018/07/23 15:32:06 ······ Fetching recursive dependency: github.com/cactus/go-statsd-client/statsd 2018/07/23 15:32:08 ······ Skipping (existing): gopkg.in/yaml.v2 2018/07/23 15:32:08 ······ Fetching recursive dependency: github.com/m3db/prometheus_client_golang/prometheus 2018/07/23 15:32:08 ······· Skipping (existing): github.com/prometheus/procfs 2018/07/23 15:32:08 ······· Skipping (existing): github.com/prometheus/client_model/go 2018/07/23 15:32:08 ······· Skipping (existing): github.com/prometheus/common/expfmt 2018/07/23 15:32:08 ······· Skipping (existing): golang.org/x/net/context 2018/07/23 15:32:08 ······· Skipping (existing): github.com/beorn7/perks/quantile 2018/07/23 15:32:08 ······· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:32:08 ······· Skipping (existing): github.com/prometheus/common/model 2018/07/23 15:32:08 ······· Skipping (existing): github.com/prometheus/client_golang/prometheus 2018/07/23 15:32:08 ······ Fetching recursive dependency: github.com/apache/thrift/lib/go/thrift 2018/07/23 15:32:13 ····· Skipping (existing): github.com/stretchr/testify/assert 2018/07/23 15:32:13 ····· Fetching recursive dependency: github.com/go-kit/kit/metrics/influx 2018/07/23 15:32:13 ······ Fetching recursive dependency: github.com/influxdata/influxdb/client/v2 2018/07/23 15:32:17 ······· Fetching recursive dependency: github.com/influxdata/influxdb/models 2018/07/23 15:32:17 ········ Fetching recursive dependency: github.com/influxdata/influxdb/pkg/escape 2018/07/23 15:32:17 ······ Fetching recursive dependency: github.com/go-kit/kit/metrics 2018/07/23 15:32:17 ······· Fetching recursive dependency: github.com/performancecopilot/speed 2018/07/23 15:32:19 ······· Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/aws 2018/07/23 15:32:28 ········ Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/internal/sdk 2018/07/23 15:32:28 ········ Skipping (existing): github.com/go-ini/ini 2018/07/23 15:32:28 ········ Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/service/sts 2018/07/23 15:32:28 ········· Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/private/protocol/query 2018/07/23 15:32:28 ·········· Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/private/protocol 2018/07/23 15:32:28 ········· Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/internal/awsutil 2018/07/23 15:32:28 ·········· Skipping (existing): github.com/jmespath/go-jmespath 2018/07/23 15:32:28 ······· Fetching recursive dependency: github.com/aws/aws-sdk-go-v2/service/cloudwatch 2018/07/23 15:32:29 ······· Skipping (existing): github.com/aws/aws-sdk-go/aws 2018/07/23 15:32:29 ······· Skipping (existing): github.com/prometheus/client_golang/prometheus 2018/07/23 15:32:29 ······· Skipping (existing): github.com/aws/aws-sdk-go/service/cloudwatch 2018/07/23 15:32:29 ······· Skipping (existing): github.com/aws/aws-sdk-go/service/cloudwatch/cloudwatchiface 2018/07/23 15:32:29 ······· Fetching recursive dependency: golang.org/x/sync/errgroup 2018/07/23 15:32:31 ········ Skipping (existing): golang.org/x/net/context 2018/07/23 15:32:31 ······· Fetching recursive dependency: github.com/go-kit/kit/util/conn 2018/07/23 15:32:31 ······· Fetching recursive dependency: github.com/VividCortex/gohistogram 2018/07/23 15:32:33 ····· Skipping (existing): github.com/prometheus/client_golang/prometheus 2018/07/23 15:32:33 ····· Fetching recursive dependency: github.com/codahale/hdrhistogram 2018/07/23 15:32:35 ·· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:32:35 · Fetching recursive dependency: github.com/mwitkow/go-grpc-middleware 2018/07/23 15:32:37 ·· Fetching recursive dependency: github.com/grpc-ecosystem/go-grpc-middleware/logging 2018/07/23 15:32:39 ··· Fetching recursive dependency: github.com/grpc-ecosystem/go-grpc-middleware 2018/07/23 15:32:39 ···· Fetching recursive dependency: github.com/golang/protobuf/jsonpb 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/ptypes/timestamp 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/ptypes/duration 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/ptypes/any 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/ptypes/struct 2018/07/23 15:32:42 ····· Skipping (existing): github.com/golang/protobuf/ptypes/wrappers 2018/07/23 15:32:42 ···· Skipping (existing): google.golang.org/grpc/metadata 2018/07/23 15:32:42 ···· Fetching recursive dependency: github.com/stretchr/testify/suite 2018/07/23 15:32:45 ····· Skipping (existing): github.com/stretchr/testify/assert 2018/07/23 15:32:45 ····· Fetching recursive dependency: github.com/stretchr/testify/require 2018/07/23 15:32:45 ······ Skipping (existing): github.com/stretchr/testify/assert 2018/07/23 15:32:45 ···· Skipping (existing): google.golang.org/grpc/peer 2018/07/23 15:32:45 ···· Skipping (existing): golang.org/x/net/context 2018/07/23 15:32:45 ···· Skipping (existing): golang.org/x/net/trace 2018/07/23 15:32:45 ···· Fetching recursive dependency: github.com/gogo/protobuf/gogoproto 2018/07/23 15:32:48 ····· Fetching recursive dependency: github.com/gogo/protobuf/protoc-gen-gogo/descriptor 2018/07/23 15:32:48 ······ Skipping (existing): github.com/gogo/protobuf/proto 2018/07/23 15:32:48 ····· Skipping (existing): github.com/gogo/protobuf/proto 2018/07/23 15:32:48 ···· Skipping (existing): google.golang.org/grpc/credentials 2018/07/23 15:32:48 ···· Skipping (existing): google.golang.org/grpc 2018/07/23 15:32:48 ···· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:32:48 ···· Skipping (existing): google.golang.org/grpc/codes 2018/07/23 15:32:48 ···· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:32:48 ···· Skipping (existing): google.golang.org/grpc/grpclog 2018/07/23 15:32:48 ···· Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:32:48 ···· Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:32:48 ··· Skipping (existing): golang.org/x/net/context 2018/07/23 15:32:48 ··· Skipping (existing): google.golang.org/grpc 2018/07/23 15:32:48 ··· Skipping (existing): google.golang.org/grpc/grpclog 2018/07/23 15:32:48 ··· Skipping (existing): google.golang.org/grpc/codes 2018/07/23 15:32:48 ··· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:32:48 ·· Skipping (existing): github.com/opentracing/opentracing-go 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc 2018/07/23 15:32:48 ·· Skipping (existing): golang.org/x/net/context 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc/codes 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc/grpclog 2018/07/23 15:32:48 ·· Skipping (existing): github.com/opentracing/opentracing-go/log 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc/metadata 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc/peer 2018/07/23 15:32:48 ·· Skipping (existing): google.golang.org/grpc/credentials 2018/07/23 15:32:48 ·· Skipping (existing): github.com/golang/protobuf/proto 2018/07/23 15:32:48 ·· Skipping (existing): golang.org/x/net/trace 2018/07/23 15:32:48 ·· Skipping (existing): github.com/opentracing/opentracing-go/ext 2018/07/23 15:32:48 · Fetching recursive dependency: github.com/weaveworks/promrus 2018/07/23 15:32:53 ·· Skipping (existing): gopkg.in/yaml.v2 2018/07/23 15:32:53 ·· Skipping (existing): golang.org/x/net/context/ctxhttp 2018/07/23 15:32:53 ·· Fetching recursive dependency: github.com/stretchr/objx 2018/07/23 15:32:55 ·· Fetching recursive dependency: gopkg.in/alecthomas/kingpin.v2 2018/07/23 15:32:58 ··· Fetching recursive dependency: github.com/alecthomas/units 2018/07/23 15:33:00 ··· Fetching recursive dependency: github.com/alecthomas/template 2018/07/23 15:33:02 ·· Fetching recursive dependency: github.com/julienschmidt/httprouter 2018/07/23 15:33:05 ·· Skipping (existing): golang.org/x/net/context 2018/07/23 15:33:05 · Skipping (existing): github.com/aws/aws-sdk-go/aws/credentials 2018/07/23 15:33:05 · Skipping (existing): github.com/golang/protobuf/ptypes/empty 2018/07/23 15:33:05 · Skipping (existing): golang.org/x/net/context 2018/07/23 15:33:05 · Skipping (existing): golang.org/x/tools/cover 2018/07/23 15:33:05 · Skipping (existing): github.com/mgutz/ansi ```
476 lines
11 KiB
Go
476 lines
11 KiB
Go
// Copyright (c) 2018 Uber Technologies, Inc.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
// THE SOFTWARE.
|
|
|
|
package tally
|
|
|
|
import (
|
|
"math"
|
|
"sort"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
)
|
|
|
|
var (
|
|
capabilitiesNone = &capabilities{
|
|
reporting: false,
|
|
tagging: false,
|
|
}
|
|
capabilitiesReportingNoTagging = &capabilities{
|
|
reporting: true,
|
|
tagging: false,
|
|
}
|
|
capabilitiesReportingTagging = &capabilities{
|
|
reporting: true,
|
|
tagging: true,
|
|
}
|
|
)
|
|
|
|
type capabilities struct {
|
|
reporting bool
|
|
tagging bool
|
|
}
|
|
|
|
func (c *capabilities) Reporting() bool {
|
|
return c.reporting
|
|
}
|
|
|
|
func (c *capabilities) Tagging() bool {
|
|
return c.tagging
|
|
}
|
|
|
|
type counter struct {
|
|
prev int64
|
|
curr int64
|
|
cachedCount CachedCount
|
|
}
|
|
|
|
func newCounter(cachedCount CachedCount) *counter {
|
|
return &counter{cachedCount: cachedCount}
|
|
}
|
|
|
|
func (c *counter) Inc(v int64) {
|
|
atomic.AddInt64(&c.curr, v)
|
|
}
|
|
|
|
func (c *counter) value() int64 {
|
|
curr := atomic.LoadInt64(&c.curr)
|
|
|
|
prev := atomic.LoadInt64(&c.prev)
|
|
if prev == curr {
|
|
return 0
|
|
}
|
|
atomic.StoreInt64(&c.prev, curr)
|
|
return curr - prev
|
|
}
|
|
|
|
func (c *counter) report(name string, tags map[string]string, r StatsReporter) {
|
|
delta := c.value()
|
|
if delta == 0 {
|
|
return
|
|
}
|
|
|
|
r.ReportCounter(name, tags, delta)
|
|
}
|
|
|
|
func (c *counter) cachedReport() {
|
|
delta := c.value()
|
|
if delta == 0 {
|
|
return
|
|
}
|
|
|
|
c.cachedCount.ReportCount(delta)
|
|
}
|
|
|
|
func (c *counter) snapshot() int64 {
|
|
return atomic.LoadInt64(&c.curr) - atomic.LoadInt64(&c.prev)
|
|
}
|
|
|
|
type gauge struct {
|
|
updated uint64
|
|
curr uint64
|
|
cachedGauge CachedGauge
|
|
}
|
|
|
|
func newGauge(cachedGauge CachedGauge) *gauge {
|
|
return &gauge{cachedGauge: cachedGauge}
|
|
}
|
|
|
|
func (g *gauge) Update(v float64) {
|
|
atomic.StoreUint64(&g.curr, math.Float64bits(v))
|
|
atomic.StoreUint64(&g.updated, 1)
|
|
}
|
|
|
|
func (g *gauge) value() float64 {
|
|
return math.Float64frombits(atomic.LoadUint64(&g.curr))
|
|
}
|
|
|
|
func (g *gauge) report(name string, tags map[string]string, r StatsReporter) {
|
|
if atomic.SwapUint64(&g.updated, 0) == 1 {
|
|
r.ReportGauge(name, tags, g.value())
|
|
}
|
|
}
|
|
|
|
func (g *gauge) cachedReport() {
|
|
if atomic.SwapUint64(&g.updated, 0) == 1 {
|
|
g.cachedGauge.ReportGauge(g.value())
|
|
}
|
|
}
|
|
|
|
func (g *gauge) snapshot() float64 {
|
|
return math.Float64frombits(atomic.LoadUint64(&g.curr))
|
|
}
|
|
|
|
// NB(jra3): timers are a little special because they do no aggregate any data
|
|
// at the timer level. The reporter buffers may timer entries and periodically
|
|
// flushes.
|
|
type timer struct {
|
|
name string
|
|
tags map[string]string
|
|
reporter StatsReporter
|
|
cachedTimer CachedTimer
|
|
unreported timerValues
|
|
}
|
|
|
|
type timerValues struct {
|
|
sync.RWMutex
|
|
values []time.Duration
|
|
}
|
|
|
|
func newTimer(
|
|
name string,
|
|
tags map[string]string,
|
|
r StatsReporter,
|
|
cachedTimer CachedTimer,
|
|
) *timer {
|
|
t := &timer{
|
|
name: name,
|
|
tags: tags,
|
|
reporter: r,
|
|
cachedTimer: cachedTimer,
|
|
}
|
|
if r == nil {
|
|
t.reporter = &timerNoReporterSink{timer: t}
|
|
}
|
|
return t
|
|
}
|
|
|
|
func (t *timer) Record(interval time.Duration) {
|
|
if t.cachedTimer != nil {
|
|
t.cachedTimer.ReportTimer(interval)
|
|
} else {
|
|
t.reporter.ReportTimer(t.name, t.tags, interval)
|
|
}
|
|
}
|
|
|
|
func (t *timer) Start() Stopwatch {
|
|
return NewStopwatch(globalNow(), t)
|
|
}
|
|
|
|
func (t *timer) RecordStopwatch(stopwatchStart time.Time) {
|
|
d := globalNow().Sub(stopwatchStart)
|
|
t.Record(d)
|
|
}
|
|
|
|
func (t *timer) snapshot() []time.Duration {
|
|
t.unreported.RLock()
|
|
snap := make([]time.Duration, len(t.unreported.values))
|
|
for i := range t.unreported.values {
|
|
snap[i] = t.unreported.values[i]
|
|
}
|
|
t.unreported.RUnlock()
|
|
return snap
|
|
}
|
|
|
|
type timerNoReporterSink struct {
|
|
sync.RWMutex
|
|
timer *timer
|
|
}
|
|
|
|
func (r *timerNoReporterSink) ReportCounter(
|
|
name string,
|
|
tags map[string]string,
|
|
value int64,
|
|
) {
|
|
}
|
|
|
|
func (r *timerNoReporterSink) ReportGauge(
|
|
name string,
|
|
tags map[string]string,
|
|
value float64,
|
|
) {
|
|
}
|
|
|
|
func (r *timerNoReporterSink) ReportTimer(
|
|
name string,
|
|
tags map[string]string,
|
|
interval time.Duration,
|
|
) {
|
|
r.timer.unreported.Lock()
|
|
r.timer.unreported.values = append(r.timer.unreported.values, interval)
|
|
r.timer.unreported.Unlock()
|
|
}
|
|
|
|
func (r *timerNoReporterSink) ReportHistogramValueSamples(
|
|
name string,
|
|
tags map[string]string,
|
|
buckets Buckets,
|
|
bucketLowerBound,
|
|
bucketUpperBound float64,
|
|
samples int64,
|
|
) {
|
|
}
|
|
|
|
func (r *timerNoReporterSink) ReportHistogramDurationSamples(
|
|
name string,
|
|
tags map[string]string,
|
|
buckets Buckets,
|
|
bucketLowerBound,
|
|
bucketUpperBound time.Duration,
|
|
samples int64,
|
|
) {
|
|
}
|
|
|
|
func (r *timerNoReporterSink) Capabilities() Capabilities {
|
|
return capabilitiesReportingTagging
|
|
}
|
|
|
|
func (r *timerNoReporterSink) Flush() {
|
|
}
|
|
|
|
type histogram struct {
|
|
htype histogramType
|
|
name string
|
|
tags map[string]string
|
|
reporter StatsReporter
|
|
specification Buckets
|
|
buckets []histogramBucket
|
|
lookupByValue []float64
|
|
lookupByDuration []int
|
|
}
|
|
|
|
type histogramType int
|
|
|
|
const (
|
|
valueHistogramType histogramType = iota
|
|
durationHistogramType
|
|
)
|
|
|
|
func newHistogram(
|
|
name string,
|
|
tags map[string]string,
|
|
reporter StatsReporter,
|
|
buckets Buckets,
|
|
cachedHistogram CachedHistogram,
|
|
) *histogram {
|
|
htype := valueHistogramType
|
|
if _, ok := buckets.(DurationBuckets); ok {
|
|
htype = durationHistogramType
|
|
}
|
|
|
|
pairs := BucketPairs(buckets)
|
|
|
|
h := &histogram{
|
|
htype: htype,
|
|
name: name,
|
|
tags: tags,
|
|
reporter: reporter,
|
|
specification: buckets,
|
|
buckets: make([]histogramBucket, 0, len(pairs)),
|
|
lookupByValue: make([]float64, 0, len(pairs)),
|
|
lookupByDuration: make([]int, 0, len(pairs)),
|
|
}
|
|
|
|
for _, pair := range pairs {
|
|
h.addBucket(newHistogramBucket(h,
|
|
pair.LowerBoundValue(), pair.UpperBoundValue(),
|
|
pair.LowerBoundDuration(), pair.UpperBoundDuration(),
|
|
cachedHistogram))
|
|
}
|
|
|
|
return h
|
|
}
|
|
|
|
func (h *histogram) addBucket(b histogramBucket) {
|
|
h.buckets = append(h.buckets, b)
|
|
h.lookupByValue = append(h.lookupByValue, b.valueUpperBound)
|
|
h.lookupByDuration = append(h.lookupByDuration, int(b.durationUpperBound))
|
|
}
|
|
|
|
func (h *histogram) report(name string, tags map[string]string, r StatsReporter) {
|
|
for i := range h.buckets {
|
|
samples := h.buckets[i].samples.value()
|
|
if samples == 0 {
|
|
continue
|
|
}
|
|
switch h.htype {
|
|
case valueHistogramType:
|
|
r.ReportHistogramValueSamples(name, tags, h.specification,
|
|
h.buckets[i].valueLowerBound, h.buckets[i].valueUpperBound,
|
|
samples)
|
|
case durationHistogramType:
|
|
r.ReportHistogramDurationSamples(name, tags, h.specification,
|
|
h.buckets[i].durationLowerBound, h.buckets[i].durationUpperBound,
|
|
samples)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (h *histogram) cachedReport() {
|
|
for i := range h.buckets {
|
|
samples := h.buckets[i].samples.value()
|
|
if samples == 0 {
|
|
continue
|
|
}
|
|
switch h.htype {
|
|
case valueHistogramType:
|
|
h.buckets[i].cachedValueBucket.ReportSamples(samples)
|
|
case durationHistogramType:
|
|
h.buckets[i].cachedDurationBucket.ReportSamples(samples)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (h *histogram) RecordValue(value float64) {
|
|
// Find the highest inclusive of the bucket upper bound
|
|
// and emit directly to it. Since we use BucketPairs to derive
|
|
// buckets there will always be an inclusive bucket as
|
|
// we always have a math.MaxFloat64 bucket.
|
|
idx := sort.SearchFloat64s(h.lookupByValue, value)
|
|
h.buckets[idx].samples.Inc(1)
|
|
}
|
|
|
|
func (h *histogram) RecordDuration(value time.Duration) {
|
|
// Find the highest inclusive of the bucket upper bound
|
|
// and emit directly to it. Since we use BucketPairs to derive
|
|
// buckets there will always be an inclusive bucket as
|
|
// we always have a math.MaxInt64 bucket.
|
|
idx := sort.SearchInts(h.lookupByDuration, int(value))
|
|
h.buckets[idx].samples.Inc(1)
|
|
}
|
|
|
|
func (h *histogram) Start() Stopwatch {
|
|
return NewStopwatch(globalNow(), h)
|
|
}
|
|
|
|
func (h *histogram) RecordStopwatch(stopwatchStart time.Time) {
|
|
d := globalNow().Sub(stopwatchStart)
|
|
h.RecordDuration(d)
|
|
}
|
|
|
|
func (h *histogram) snapshotValues() map[float64]int64 {
|
|
if h.htype == durationHistogramType {
|
|
return nil
|
|
}
|
|
|
|
vals := make(map[float64]int64, len(h.buckets))
|
|
for i := range h.buckets {
|
|
vals[h.buckets[i].valueUpperBound] = h.buckets[i].samples.value()
|
|
}
|
|
|
|
return vals
|
|
}
|
|
|
|
func (h *histogram) snapshotDurations() map[time.Duration]int64 {
|
|
if h.htype == valueHistogramType {
|
|
return nil
|
|
}
|
|
|
|
durations := make(map[time.Duration]int64, len(h.buckets))
|
|
for i := range h.buckets {
|
|
durations[h.buckets[i].durationUpperBound] = h.buckets[i].samples.value()
|
|
}
|
|
|
|
return durations
|
|
}
|
|
|
|
type histogramBucket struct {
|
|
h *histogram
|
|
samples *counter
|
|
valueLowerBound float64
|
|
valueUpperBound float64
|
|
durationLowerBound time.Duration
|
|
durationUpperBound time.Duration
|
|
cachedValueBucket CachedHistogramBucket
|
|
cachedDurationBucket CachedHistogramBucket
|
|
}
|
|
|
|
func newHistogramBucket(
|
|
h *histogram,
|
|
valueLowerBound,
|
|
valueUpperBound float64,
|
|
durationLowerBound,
|
|
durationUpperBound time.Duration,
|
|
cachedHistogram CachedHistogram,
|
|
) histogramBucket {
|
|
bucket := histogramBucket{
|
|
samples: newCounter(nil),
|
|
valueLowerBound: valueLowerBound,
|
|
valueUpperBound: valueUpperBound,
|
|
durationLowerBound: durationLowerBound,
|
|
durationUpperBound: durationUpperBound,
|
|
}
|
|
if cachedHistogram != nil {
|
|
bucket.cachedValueBucket = cachedHistogram.ValueBucket(
|
|
bucket.valueLowerBound, bucket.valueUpperBound,
|
|
)
|
|
bucket.cachedDurationBucket = cachedHistogram.DurationBucket(
|
|
bucket.durationLowerBound, bucket.durationUpperBound,
|
|
)
|
|
}
|
|
return bucket
|
|
}
|
|
|
|
// NullStatsReporter is an implementation of StatsReporter than simply does nothing.
|
|
var NullStatsReporter StatsReporter = nullStatsReporter{}
|
|
|
|
func (r nullStatsReporter) ReportCounter(name string, tags map[string]string, value int64) {
|
|
}
|
|
func (r nullStatsReporter) ReportGauge(name string, tags map[string]string, value float64) {
|
|
}
|
|
func (r nullStatsReporter) ReportTimer(name string, tags map[string]string, interval time.Duration) {
|
|
}
|
|
func (r nullStatsReporter) ReportHistogramValueSamples(
|
|
name string,
|
|
tags map[string]string,
|
|
buckets Buckets,
|
|
bucketLowerBound,
|
|
bucketUpperBound float64,
|
|
samples int64,
|
|
) {
|
|
}
|
|
|
|
func (r nullStatsReporter) ReportHistogramDurationSamples(
|
|
name string,
|
|
tags map[string]string,
|
|
buckets Buckets,
|
|
bucketLowerBound,
|
|
bucketUpperBound time.Duration,
|
|
samples int64,
|
|
) {
|
|
}
|
|
func (r nullStatsReporter) Capabilities() Capabilities {
|
|
return capabilitiesNone
|
|
}
|
|
func (r nullStatsReporter) Flush() {
|
|
}
|
|
|
|
type nullStatsReporter struct{}
|