2020-03-24 07:41:10 +02:00
|
|
|
// Copyright The OpenTelemetry Authors
|
2019-11-15 23:01:20 +02:00
|
|
|
//
|
|
|
|
// 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 ungrouped // import "go.opentelemetry.io/otel/sdk/metric/batcher/ungrouped"
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-03-17 01:28:33 +02:00
|
|
|
"errors"
|
2019-11-15 23:01:20 +02:00
|
|
|
|
2020-03-19 21:02:46 +02:00
|
|
|
"go.opentelemetry.io/otel/api/metric"
|
2019-11-15 23:01:20 +02:00
|
|
|
export "go.opentelemetry.io/otel/sdk/export/metric"
|
2020-03-17 01:28:33 +02:00
|
|
|
"go.opentelemetry.io/otel/sdk/export/metric/aggregator"
|
2019-11-15 23:01:20 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type (
|
|
|
|
Batcher struct {
|
|
|
|
selector export.AggregationSelector
|
|
|
|
batchMap batchMap
|
|
|
|
stateful bool
|
|
|
|
}
|
|
|
|
|
|
|
|
batchKey struct {
|
2020-03-19 21:02:46 +02:00
|
|
|
descriptor *metric.Descriptor
|
2019-11-15 23:01:20 +02:00
|
|
|
encoded string
|
|
|
|
}
|
|
|
|
|
|
|
|
batchValue struct {
|
|
|
|
aggregator export.Aggregator
|
|
|
|
labels export.Labels
|
|
|
|
}
|
|
|
|
|
|
|
|
batchMap map[batchKey]batchValue
|
|
|
|
)
|
|
|
|
|
|
|
|
var _ export.Batcher = &Batcher{}
|
|
|
|
var _ export.CheckpointSet = batchMap{}
|
|
|
|
|
|
|
|
func New(selector export.AggregationSelector, stateful bool) *Batcher {
|
|
|
|
return &Batcher{
|
|
|
|
selector: selector,
|
|
|
|
batchMap: batchMap{},
|
|
|
|
stateful: stateful,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-19 21:02:46 +02:00
|
|
|
func (b *Batcher) AggregatorFor(descriptor *metric.Descriptor) export.Aggregator {
|
2019-11-15 23:01:20 +02:00
|
|
|
return b.selector.AggregatorFor(descriptor)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Batcher) Process(_ context.Context, record export.Record) error {
|
|
|
|
desc := record.Descriptor()
|
|
|
|
key := batchKey{
|
|
|
|
descriptor: desc,
|
|
|
|
encoded: record.Labels().Encoded(),
|
|
|
|
}
|
|
|
|
agg := record.Aggregator()
|
|
|
|
value, ok := b.batchMap[key]
|
|
|
|
if ok {
|
2019-12-24 02:38:35 +02:00
|
|
|
// Note: The call to Merge here combines only
|
|
|
|
// identical records. It is required even for a
|
|
|
|
// stateless Batcher because such identical records
|
|
|
|
// may arise in the Meter implementation due to race
|
|
|
|
// conditions.
|
2019-11-15 23:01:20 +02:00
|
|
|
return value.aggregator.Merge(agg, desc)
|
|
|
|
}
|
|
|
|
// If this Batcher is stateful, create a copy of the
|
|
|
|
// Aggregator for long-term storage. Otherwise the
|
|
|
|
// Meter implementation will checkpoint the aggregator
|
|
|
|
// again, overwriting the long-lived state.
|
|
|
|
if b.stateful {
|
|
|
|
tmp := agg
|
2019-12-24 02:38:35 +02:00
|
|
|
// Note: the call to AggregatorFor() followed by Merge
|
|
|
|
// is effectively a Clone() operation.
|
2019-11-15 23:01:20 +02:00
|
|
|
agg = b.AggregatorFor(desc)
|
|
|
|
if err := agg.Merge(tmp, desc); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b.batchMap[key] = batchValue{
|
|
|
|
aggregator: agg,
|
|
|
|
labels: record.Labels(),
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Batcher) CheckpointSet() export.CheckpointSet {
|
|
|
|
return b.batchMap
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Batcher) FinishedCollection() {
|
|
|
|
if !b.stateful {
|
|
|
|
b.batchMap = batchMap{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-17 01:28:33 +02:00
|
|
|
func (c batchMap) ForEach(f func(export.Record) error) error {
|
2019-11-15 23:01:20 +02:00
|
|
|
for key, value := range c {
|
2020-03-17 01:28:33 +02:00
|
|
|
if err := f(export.NewRecord(
|
2019-11-15 23:01:20 +02:00
|
|
|
key.descriptor,
|
|
|
|
value.labels,
|
|
|
|
value.aggregator,
|
2020-03-17 01:28:33 +02:00
|
|
|
)); err != nil && !errors.Is(err, aggregator.ErrNoData) {
|
|
|
|
return err
|
|
|
|
}
|
2019-11-15 23:01:20 +02:00
|
|
|
}
|
2020-03-17 01:28:33 +02:00
|
|
|
return nil
|
2019-11-15 23:01:20 +02:00
|
|
|
}
|