mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-22 03:38:42 +02:00
697d245b9c
* update bucket default bounds to match the specification * add changelog entry * test custom boundaries with valid histogram Co-authored-by: Tyler Yahn <MrAlias@users.noreply.github.com>
214 lines
8.2 KiB
Go
214 lines
8.2 KiB
Go
// Copyright The OpenTelemetry Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package metric // import "go.opentelemetry.io/otel/sdk/metric"
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"go.opentelemetry.io/otel/internal/global"
|
|
"go.opentelemetry.io/otel/sdk/metric/aggregation"
|
|
"go.opentelemetry.io/otel/sdk/metric/metricdata"
|
|
"go.opentelemetry.io/otel/sdk/metric/view"
|
|
)
|
|
|
|
// errDuplicateRegister is logged by a Reader when an attempt to registered it
|
|
// more than once occurs.
|
|
var errDuplicateRegister = fmt.Errorf("duplicate reader registration")
|
|
|
|
// ErrReaderNotRegistered is returned if Collect or Shutdown are called before
|
|
// the reader is registered with a MeterProvider.
|
|
var ErrReaderNotRegistered = fmt.Errorf("reader is not registered")
|
|
|
|
// ErrReaderShutdown is returned if Collect or Shutdown are called after a
|
|
// reader has been Shutdown once.
|
|
var ErrReaderShutdown = fmt.Errorf("reader is shutdown")
|
|
|
|
// Reader is the interface used between the SDK and an
|
|
// exporter. Control flow is bi-directional through the
|
|
// Reader, since the SDK initiates ForceFlush and Shutdown
|
|
// while the initiates collection. The Register() method here
|
|
// informs the Reader that it can begin reading, signaling the
|
|
// start of bi-directional control flow.
|
|
//
|
|
// Typically, push-based exporters that are periodic will
|
|
// implement PeroidicExporter themselves and construct a
|
|
// PeriodicReader to satisfy this interface.
|
|
//
|
|
// Pull-based exporters will typically implement Register
|
|
// themselves, since they read on demand.
|
|
type Reader interface {
|
|
// register registers a Reader with a MeterProvider.
|
|
// The producer argument allows the Reader to signal the sdk to collect
|
|
// and send aggregated metric measurements.
|
|
register(producer)
|
|
|
|
// temporality reports the Temporality for the instrument kind provided.
|
|
temporality(view.InstrumentKind) metricdata.Temporality
|
|
|
|
// aggregation returns what Aggregation to use for an instrument kind.
|
|
aggregation(view.InstrumentKind) aggregation.Aggregation // nolint:revive // import-shadow for method scoped by type.
|
|
|
|
// Collect gathers and returns all metric data related to the Reader from
|
|
// the SDK. An error is returned if this is called after Shutdown.
|
|
Collect(context.Context) (metricdata.ResourceMetrics, error)
|
|
|
|
// ForceFlush flushes all metric measurements held in an export pipeline.
|
|
//
|
|
// This deadline or cancellation of the passed context are honored. An appropriate
|
|
// error will be returned in these situations. There is no guaranteed that all
|
|
// telemetry be flushed or all resources have been released in these
|
|
// situations.
|
|
ForceFlush(context.Context) error
|
|
|
|
// Shutdown flushes all metric measurements held in an export pipeline and releases any
|
|
// held computational resources.
|
|
//
|
|
// This deadline or cancellation of the passed context are honored. An appropriate
|
|
// error will be returned in these situations. There is no guaranteed that all
|
|
// telemetry be flushed or all resources have been released in these
|
|
// situations.
|
|
//
|
|
// After Shutdown is called, calls to Collect will perform no operation and instead will return
|
|
// an error indicating the shutdown state.
|
|
Shutdown(context.Context) error
|
|
}
|
|
|
|
// producer produces metrics for a Reader.
|
|
type producer interface {
|
|
// produce returns aggregated metrics from a single collection.
|
|
//
|
|
// This method is safe to call concurrently.
|
|
produce(context.Context) (metricdata.ResourceMetrics, error)
|
|
}
|
|
|
|
// produceHolder is used as an atomic.Value to wrap the non-concrete producer
|
|
// type.
|
|
type produceHolder struct {
|
|
produce func(context.Context) (metricdata.ResourceMetrics, error)
|
|
}
|
|
|
|
// shutdownProducer produces an ErrReaderShutdown error always.
|
|
type shutdownProducer struct{}
|
|
|
|
// produce returns an ErrReaderShutdown error.
|
|
func (p shutdownProducer) produce(context.Context) (metricdata.ResourceMetrics, error) {
|
|
return metricdata.ResourceMetrics{}, ErrReaderShutdown
|
|
}
|
|
|
|
// ReaderOption applies a configuration option value to either a ManualReader or
|
|
// a PeriodicReader.
|
|
type ReaderOption interface {
|
|
ManualReaderOption
|
|
PeriodicReaderOption
|
|
}
|
|
|
|
// TemporalitySelector selects the temporality to use based on the InstrumentKind.
|
|
type TemporalitySelector func(view.InstrumentKind) metricdata.Temporality
|
|
|
|
// DefaultTemporalitySelector is the default TemporalitySelector used if
|
|
// WithTemporalitySelector is not provided. CumulativeTemporality will be used
|
|
// for all instrument kinds if this TemporalitySelector is used.
|
|
func DefaultTemporalitySelector(view.InstrumentKind) metricdata.Temporality {
|
|
return metricdata.CumulativeTemporality
|
|
}
|
|
|
|
// WithTemporalitySelector sets the TemporalitySelector a reader will use to
|
|
// determine the Temporality of an instrument based on its kind. If this
|
|
// option is not used, the reader will use the DefaultTemporalitySelector.
|
|
func WithTemporalitySelector(selector TemporalitySelector) ReaderOption {
|
|
return temporalitySelectorOption{selector: selector}
|
|
}
|
|
|
|
type temporalitySelectorOption struct {
|
|
selector func(instrument view.InstrumentKind) metricdata.Temporality
|
|
}
|
|
|
|
// applyManual returns a manualReaderConfig with option applied.
|
|
func (t temporalitySelectorOption) applyManual(mrc manualReaderConfig) manualReaderConfig {
|
|
mrc.temporalitySelector = t.selector
|
|
return mrc
|
|
}
|
|
|
|
// applyPeriodic returns a periodicReaderConfig with option applied.
|
|
func (t temporalitySelectorOption) applyPeriodic(prc periodicReaderConfig) periodicReaderConfig {
|
|
prc.temporalitySelector = t.selector
|
|
return prc
|
|
}
|
|
|
|
// AggregationSelector selects the aggregation and the parameters to use for
|
|
// that aggregation based on the InstrumentKind.
|
|
type AggregationSelector func(view.InstrumentKind) aggregation.Aggregation
|
|
|
|
// DefaultAggregationSelector returns the default aggregation and parameters
|
|
// that will be used to summarize measurement made from an instrument of
|
|
// InstrumentKind. This AggregationSelector using the following selection
|
|
// mapping: Counter ⇨ Sum, Asynchronous Counter ⇨ Sum, UpDownCounter ⇨ Sum,
|
|
// Asynchronous UpDownCounter ⇨ Sum, Asynchronous Gauge ⇨ LastValue,
|
|
// Histogram ⇨ ExplicitBucketHistogram.
|
|
func DefaultAggregationSelector(ik view.InstrumentKind) aggregation.Aggregation {
|
|
switch ik {
|
|
case view.SyncCounter, view.SyncUpDownCounter, view.AsyncCounter, view.AsyncUpDownCounter:
|
|
return aggregation.Sum{}
|
|
case view.AsyncGauge:
|
|
return aggregation.LastValue{}
|
|
case view.SyncHistogram:
|
|
return aggregation.ExplicitBucketHistogram{
|
|
Boundaries: []float64{0, 5, 10, 25, 50, 75, 100, 250, 500, 750, 1000, 2500, 5000, 7500, 10000},
|
|
NoMinMax: false,
|
|
}
|
|
}
|
|
panic("unknown instrument kind")
|
|
}
|
|
|
|
// WithAggregationSelector sets the AggregationSelector a reader will use to
|
|
// determine the aggregation to use for an instrument based on its kind. If
|
|
// this option is not used, the reader will use the DefaultAggregationSelector
|
|
// or the aggregation explicitly passed for a view matching an instrument.
|
|
func WithAggregationSelector(selector AggregationSelector) ReaderOption {
|
|
// Deep copy and validate before using.
|
|
wrapped := func(ik view.InstrumentKind) aggregation.Aggregation {
|
|
a := selector(ik)
|
|
cpA := a.Copy()
|
|
if err := cpA.Err(); err != nil {
|
|
cpA = DefaultAggregationSelector(ik)
|
|
global.Error(
|
|
err, "using default aggregation instead",
|
|
"aggregation", a,
|
|
"replacement", cpA,
|
|
)
|
|
}
|
|
return cpA
|
|
}
|
|
|
|
return aggregationSelectorOption{selector: wrapped}
|
|
}
|
|
|
|
type aggregationSelectorOption struct {
|
|
selector AggregationSelector
|
|
}
|
|
|
|
// applyManual returns a manualReaderConfig with option applied.
|
|
func (t aggregationSelectorOption) applyManual(c manualReaderConfig) manualReaderConfig {
|
|
c.aggregationSelector = t.selector
|
|
return c
|
|
}
|
|
|
|
// applyPeriodic returns a periodicReaderConfig with option applied.
|
|
func (t aggregationSelectorOption) applyPeriodic(c periodicReaderConfig) periodicReaderConfig {
|
|
c.aggregationSelector = t.selector
|
|
return c
|
|
}
|