// 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_test import ( "context" "fmt" "math" "sync" "testing" "github.com/stretchr/testify/require" "go.opentelemetry.io/otel" "go.opentelemetry.io/otel/attribute" "go.opentelemetry.io/otel/metric" export "go.opentelemetry.io/otel/sdk/export/metric" "go.opentelemetry.io/otel/sdk/export/metric/aggregation" metricsdk "go.opentelemetry.io/otel/sdk/metric" "go.opentelemetry.io/otel/sdk/metric/processor/processortest" "go.opentelemetry.io/otel/sdk/resource" ) var Must = metric.Must var testResource = resource.NewSchemaless(attribute.String("R", "V")) type handler struct { sync.Mutex err error } func (h *handler) Handle(err error) { h.Lock() h.err = err h.Unlock() } func (h *handler) Reset() { h.Lock() h.err = nil h.Unlock() } func (h *handler) Flush() error { h.Lock() err := h.err h.err = nil h.Unlock() return err } var testHandler *handler func init() { testHandler = new(handler) otel.SetErrorHandler(testHandler) } type testSelector struct { selector export.AggregatorSelector newAggCount int } func (ts *testSelector) AggregatorFor(desc *metric.Descriptor, aggPtrs ...*export.Aggregator) { ts.newAggCount += len(aggPtrs) processortest.AggregatorSelector().AggregatorFor(desc, aggPtrs...) } func newSDK(t *testing.T) (metric.Meter, *metricsdk.Accumulator, *testSelector, *processortest.Processor) { testHandler.Reset() testSelector := &testSelector{selector: processortest.AggregatorSelector()} processor := processortest.NewProcessor( testSelector, attribute.DefaultEncoder(), ) accum := metricsdk.NewAccumulator( processor, testResource, ) meter := metric.WrapMeterImpl(accum, "test") return meter, accum, testSelector, processor } func TestInputRangeCounter(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) counter := Must(meter).NewInt64Counter("name.sum") counter.Add(ctx, -1) require.Equal(t, aggregation.ErrNegativeInput, testHandler.Flush()) checkpointed := sdk.Collect(ctx) require.Equal(t, 0, checkpointed) processor.Reset() counter.Add(ctx, 1) checkpointed = sdk.Collect(ctx) require.Equal(t, map[string]float64{ "name.sum//R=V": 1, }, processor.Values()) require.Equal(t, 1, checkpointed) require.Nil(t, testHandler.Flush()) } func TestInputRangeUpDownCounter(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) counter := Must(meter).NewInt64UpDownCounter("name.sum") counter.Add(ctx, -1) counter.Add(ctx, -1) counter.Add(ctx, 2) counter.Add(ctx, 1) checkpointed := sdk.Collect(ctx) require.Equal(t, map[string]float64{ "name.sum//R=V": 1, }, processor.Values()) require.Equal(t, 1, checkpointed) require.Nil(t, testHandler.Flush()) } func TestInputRangeValueRecorder(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) valuerecorder := Must(meter).NewFloat64ValueRecorder("name.exact") valuerecorder.Record(ctx, math.NaN()) require.Equal(t, aggregation.ErrNaNInput, testHandler.Flush()) checkpointed := sdk.Collect(ctx) require.Equal(t, 0, checkpointed) valuerecorder.Record(ctx, 1) valuerecorder.Record(ctx, 2) processor.Reset() checkpointed = sdk.Collect(ctx) require.Equal(t, map[string]float64{ "name.exact//R=V": 3, }, processor.Values()) require.Equal(t, 1, checkpointed) require.Nil(t, testHandler.Flush()) } func TestDisabledInstrument(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) valuerecorder := Must(meter).NewFloat64ValueRecorder("name.disabled") valuerecorder.Record(ctx, -1) checkpointed := sdk.Collect(ctx) require.Equal(t, 0, checkpointed) require.Equal(t, map[string]float64{}, processor.Values()) } func TestRecordNaN(t *testing.T) { ctx := context.Background() meter, _, _, _ := newSDK(t) c := Must(meter).NewFloat64Counter("name.sum") require.Nil(t, testHandler.Flush()) c.Add(ctx, math.NaN()) require.Error(t, testHandler.Flush()) } func TestSDKLabelsDeduplication(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) counter := Must(meter).NewInt64Counter("name.sum") const ( maxKeys = 21 keySets = 2 repeats = 3 ) var keysA []attribute.Key var keysB []attribute.Key for i := 0; i < maxKeys; i++ { keysA = append(keysA, attribute.Key(fmt.Sprintf("A%03d", i))) keysB = append(keysB, attribute.Key(fmt.Sprintf("B%03d", i))) } allExpect := map[string]float64{} for numKeys := 0; numKeys < maxKeys; numKeys++ { var kvsA []attribute.KeyValue var kvsB []attribute.KeyValue for r := 0; r < repeats; r++ { for i := 0; i < numKeys; i++ { kvsA = append(kvsA, keysA[i].Int(r)) kvsB = append(kvsB, keysB[i].Int(r)) } } var expectA []attribute.KeyValue var expectB []attribute.KeyValue for i := 0; i < numKeys; i++ { expectA = append(expectA, keysA[i].Int(repeats-1)) expectB = append(expectB, keysB[i].Int(repeats-1)) } counter.Add(ctx, 1, kvsA...) counter.Add(ctx, 1, kvsA...) format := func(attrs []attribute.KeyValue) string { str := attribute.DefaultEncoder().Encode(newSetIter(attrs...)) return fmt.Sprint("name.sum/", str, "/R=V") } allExpect[format(expectA)] += 2 if numKeys != 0 { // In this case A and B sets are the same. counter.Add(ctx, 1, kvsB...) counter.Add(ctx, 1, kvsB...) allExpect[format(expectB)] += 2 } } sdk.Collect(ctx) require.EqualValues(t, allExpect, processor.Values()) } func newSetIter(kvs ...attribute.KeyValue) attribute.Iterator { labels := attribute.NewSet(kvs...) return labels.Iter() } func TestDefaultLabelEncoder(t *testing.T) { encoder := attribute.DefaultEncoder() encoded := encoder.Encode(newSetIter(attribute.String("A", "B"), attribute.String("C", "D"))) require.Equal(t, `A=B,C=D`, encoded) encoded = encoder.Encode(newSetIter(attribute.String("A", "B,c=d"), attribute.String(`C\`, "D"))) require.Equal(t, `A=B\,c\=d,C\\=D`, encoded) encoded = encoder.Encode(newSetIter(attribute.String(`\`, `=`), attribute.String(`,`, `\`))) require.Equal(t, `\,=\\,\\=\=`, encoded) // Note: the label encoder does not sort or de-dup values, // that is done in Labels(...). encoded = encoder.Encode(newSetIter( attribute.Int("I", 1), attribute.Int64("I64", 1), attribute.Float64("F64", 1), attribute.Float64("F64", 1), attribute.String("S", "1"), attribute.Bool("B", true), )) require.Equal(t, "B=true,F64=1,I=1,I64=1,S=1", encoded) } func TestObserverCollection(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) mult := 1 _ = Must(meter).NewFloat64ValueObserver("float.valueobserver.lastvalue", func(_ context.Context, result metric.Float64ObserverResult) { result.Observe(float64(mult), attribute.String("A", "B")) // last value wins result.Observe(float64(-mult), attribute.String("A", "B")) result.Observe(float64(-mult), attribute.String("C", "D")) }) _ = Must(meter).NewInt64ValueObserver("int.valueobserver.lastvalue", func(_ context.Context, result metric.Int64ObserverResult) { result.Observe(int64(-mult), attribute.String("A", "B")) result.Observe(int64(mult)) // last value wins result.Observe(int64(mult), attribute.String("A", "B")) result.Observe(int64(mult)) }) _ = Must(meter).NewFloat64SumObserver("float.sumobserver.sum", func(_ context.Context, result metric.Float64ObserverResult) { result.Observe(float64(mult), attribute.String("A", "B")) result.Observe(float64(2*mult), attribute.String("A", "B")) result.Observe(float64(mult), attribute.String("C", "D")) }) _ = Must(meter).NewInt64SumObserver("int.sumobserver.sum", func(_ context.Context, result metric.Int64ObserverResult) { result.Observe(int64(2*mult), attribute.String("A", "B")) result.Observe(int64(mult)) // last value wins result.Observe(int64(mult), attribute.String("A", "B")) result.Observe(int64(mult)) }) _ = Must(meter).NewFloat64UpDownSumObserver("float.updownsumobserver.sum", func(_ context.Context, result metric.Float64ObserverResult) { result.Observe(float64(mult), attribute.String("A", "B")) result.Observe(float64(-2*mult), attribute.String("A", "B")) result.Observe(float64(mult), attribute.String("C", "D")) }) _ = Must(meter).NewInt64UpDownSumObserver("int.updownsumobserver.sum", func(_ context.Context, result metric.Int64ObserverResult) { result.Observe(int64(2*mult), attribute.String("A", "B")) result.Observe(int64(mult)) // last value wins result.Observe(int64(mult), attribute.String("A", "B")) result.Observe(int64(-mult)) }) _ = Must(meter).NewInt64ValueObserver("empty.valueobserver.sum", func(_ context.Context, result metric.Int64ObserverResult) { }) for mult = 0; mult < 3; mult++ { processor.Reset() collected := sdk.Collect(ctx) require.Equal(t, collected, len(processor.Values())) mult := float64(mult) require.EqualValues(t, map[string]float64{ "float.valueobserver.lastvalue/A=B/R=V": -mult, "float.valueobserver.lastvalue/C=D/R=V": -mult, "int.valueobserver.lastvalue//R=V": mult, "int.valueobserver.lastvalue/A=B/R=V": mult, "float.sumobserver.sum/A=B/R=V": 2 * mult, "float.sumobserver.sum/C=D/R=V": mult, "int.sumobserver.sum//R=V": mult, "int.sumobserver.sum/A=B/R=V": mult, "float.updownsumobserver.sum/A=B/R=V": -2 * mult, "float.updownsumobserver.sum/C=D/R=V": mult, "int.updownsumobserver.sum//R=V": -mult, "int.updownsumobserver.sum/A=B/R=V": mult, }, processor.Values()) } } func TestSumObserverInputRange(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) // TODO: these tests are testing for negative values, not for _descending values_. Fix. _ = Must(meter).NewFloat64SumObserver("float.sumobserver.sum", func(_ context.Context, result metric.Float64ObserverResult) { result.Observe(-2, attribute.String("A", "B")) require.Equal(t, aggregation.ErrNegativeInput, testHandler.Flush()) result.Observe(-1, attribute.String("C", "D")) require.Equal(t, aggregation.ErrNegativeInput, testHandler.Flush()) }) _ = Must(meter).NewInt64SumObserver("int.sumobserver.sum", func(_ context.Context, result metric.Int64ObserverResult) { result.Observe(-1, attribute.String("A", "B")) require.Equal(t, aggregation.ErrNegativeInput, testHandler.Flush()) result.Observe(-1) require.Equal(t, aggregation.ErrNegativeInput, testHandler.Flush()) }) collected := sdk.Collect(ctx) require.Equal(t, 0, collected) require.EqualValues(t, map[string]float64{}, processor.Values()) // check that the error condition was reset require.NoError(t, testHandler.Flush()) } func TestObserverBatch(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) var floatValueObs metric.Float64ValueObserver var intValueObs metric.Int64ValueObserver var floatSumObs metric.Float64SumObserver var intSumObs metric.Int64SumObserver var floatUpDownSumObs metric.Float64UpDownSumObserver var intUpDownSumObs metric.Int64UpDownSumObserver var batch = Must(meter).NewBatchObserver( func(_ context.Context, result metric.BatchObserverResult) { result.Observe( []attribute.KeyValue{ attribute.String("A", "B"), }, floatValueObs.Observation(1), floatValueObs.Observation(-1), intValueObs.Observation(-1), intValueObs.Observation(1), floatSumObs.Observation(1000), intSumObs.Observation(100), floatUpDownSumObs.Observation(-1000), intUpDownSumObs.Observation(-100), ) result.Observe( []attribute.KeyValue{ attribute.String("C", "D"), }, floatValueObs.Observation(-1), floatSumObs.Observation(-1), floatUpDownSumObs.Observation(-1), ) result.Observe( nil, intValueObs.Observation(1), intValueObs.Observation(1), intSumObs.Observation(10), floatSumObs.Observation(1.1), intUpDownSumObs.Observation(10), ) }) floatValueObs = batch.NewFloat64ValueObserver("float.valueobserver.lastvalue") intValueObs = batch.NewInt64ValueObserver("int.valueobserver.lastvalue") floatSumObs = batch.NewFloat64SumObserver("float.sumobserver.sum") intSumObs = batch.NewInt64SumObserver("int.sumobserver.sum") floatUpDownSumObs = batch.NewFloat64UpDownSumObserver("float.updownsumobserver.sum") intUpDownSumObs = batch.NewInt64UpDownSumObserver("int.updownsumobserver.sum") collected := sdk.Collect(ctx) require.Equal(t, collected, len(processor.Values())) require.EqualValues(t, map[string]float64{ "float.sumobserver.sum//R=V": 1.1, "float.sumobserver.sum/A=B/R=V": 1000, "int.sumobserver.sum//R=V": 10, "int.sumobserver.sum/A=B/R=V": 100, "int.updownsumobserver.sum/A=B/R=V": -100, "float.updownsumobserver.sum/A=B/R=V": -1000, "int.updownsumobserver.sum//R=V": 10, "float.updownsumobserver.sum/C=D/R=V": -1, "float.valueobserver.lastvalue/A=B/R=V": -1, "float.valueobserver.lastvalue/C=D/R=V": -1, "int.valueobserver.lastvalue//R=V": 1, "int.valueobserver.lastvalue/A=B/R=V": 1, }, processor.Values()) } func TestRecordBatch(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) counter1 := Must(meter).NewInt64Counter("int64.sum") counter2 := Must(meter).NewFloat64Counter("float64.sum") valuerecorder1 := Must(meter).NewInt64ValueRecorder("int64.exact") valuerecorder2 := Must(meter).NewFloat64ValueRecorder("float64.exact") sdk.RecordBatch( ctx, []attribute.KeyValue{ attribute.String("A", "B"), attribute.String("C", "D"), }, counter1.Measurement(1), counter2.Measurement(2), valuerecorder1.Measurement(3), valuerecorder2.Measurement(4), ) sdk.Collect(ctx) require.EqualValues(t, map[string]float64{ "int64.sum/A=B,C=D/R=V": 1, "float64.sum/A=B,C=D/R=V": 2, "int64.exact/A=B,C=D/R=V": 3, "float64.exact/A=B,C=D/R=V": 4, }, processor.Values()) } // TestRecordPersistence ensures that a direct-called instrument that // is repeatedly used each interval results in a persistent record, so // that its encoded labels will be cached across collection intervals. func TestRecordPersistence(t *testing.T) { ctx := context.Background() meter, sdk, selector, _ := newSDK(t) c := Must(meter).NewFloat64Counter("name.sum") b := c.Bind(attribute.String("bound", "true")) uk := attribute.String("bound", "false") for i := 0; i < 100; i++ { c.Add(ctx, 1, uk) b.Add(ctx, 1) sdk.Collect(ctx) } require.Equal(t, 4, selector.newAggCount) } func TestIncorrectInstruments(t *testing.T) { // The Batch observe/record APIs are susceptible to // uninitialized instruments. var counter metric.Int64Counter var observer metric.Int64ValueObserver ctx := context.Background() meter, sdk, _, processor := newSDK(t) // Now try with uninitialized instruments. meter.RecordBatch(ctx, nil, counter.Measurement(1)) meter.NewBatchObserver(func(_ context.Context, result metric.BatchObserverResult) { result.Observe(nil, observer.Observation(1)) }) collected := sdk.Collect(ctx) require.Equal(t, metricsdk.ErrUninitializedInstrument, testHandler.Flush()) require.Equal(t, 0, collected) // Now try with instruments from another SDK. var noopMeter metric.Meter counter = metric.Must(noopMeter).NewInt64Counter("name.sum") observer = metric.Must(noopMeter).NewBatchObserver( func(context.Context, metric.BatchObserverResult) {}, ).NewInt64ValueObserver("observer") meter.RecordBatch(ctx, nil, counter.Measurement(1)) meter.NewBatchObserver(func(_ context.Context, result metric.BatchObserverResult) { result.Observe(nil, observer.Observation(1)) }) collected = sdk.Collect(ctx) require.Equal(t, 0, collected) require.EqualValues(t, map[string]float64{}, processor.Values()) require.Equal(t, metricsdk.ErrUninitializedInstrument, testHandler.Flush()) } func TestSyncInAsync(t *testing.T) { ctx := context.Background() meter, sdk, _, processor := newSDK(t) counter := Must(meter).NewFloat64Counter("counter.sum") _ = Must(meter).NewInt64ValueObserver("observer.lastvalue", func(ctx context.Context, result metric.Int64ObserverResult) { result.Observe(10) counter.Add(ctx, 100) }, ) sdk.Collect(ctx) require.EqualValues(t, map[string]float64{ "counter.sum//R=V": 100, "observer.lastvalue//R=V": 10, }, processor.Values()) }