mirror of
https://github.com/open-telemetry/opentelemetry-go.git
synced 2025-01-12 02:28:07 +02:00
a1ce7e5f0d
* Combine spatially aggregated precomputed vals Fix #3439 When an attribute filter drops a distinguishing attribute during the aggregation of a precomputed sum add that value to existing, instead of just setting the value as an override (current behavior). * Ignore false positive lint error and test method * Add fix to changelog * Handle edge case of exact set after filter * Fix filter and measure algo for precomp * Add tests for precomp sums * Unify precomputedMap * Adds example from supplimental guide * Fixes for lint * Update sdk/metric/meter_example_test.go * Fix async example test * Reduce duplicate code in TestAsynchronousExample * Clarify naming and documentation * Fix spelling errors * Add a noop filter to default view Co-authored-by: Chester Cheung <cheung.zhy.csu@gmail.com> Co-authored-by: Aaron Clawson <3766680+MadVikingGod@users.noreply.github.com>
328 lines
12 KiB
Go
328 lines
12 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 internal // import "go.opentelemetry.io/otel/sdk/metric/internal"
|
|
|
|
import (
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"go.opentelemetry.io/otel/attribute"
|
|
"go.opentelemetry.io/otel/sdk/metric/metricdata"
|
|
"go.opentelemetry.io/otel/sdk/metric/metricdata/metricdatatest"
|
|
)
|
|
|
|
func TestSum(t *testing.T) {
|
|
t.Cleanup(mockTime(now))
|
|
t.Run("Int64", testSum[int64])
|
|
t.Run("Float64", testSum[float64])
|
|
}
|
|
|
|
func testSum[N int64 | float64](t *testing.T) {
|
|
tester := &aggregatorTester[N]{
|
|
GoroutineN: defaultGoroutines,
|
|
MeasurementN: defaultMeasurements,
|
|
CycleN: defaultCycles,
|
|
}
|
|
|
|
t.Run("Delta", func(t *testing.T) {
|
|
incr, mono := monoIncr, true
|
|
eFunc := deltaExpecter[N](incr, mono)
|
|
t.Run("Monotonic", tester.Run(NewDeltaSum[N](mono), incr, eFunc))
|
|
|
|
incr, mono = nonMonoIncr, false
|
|
eFunc = deltaExpecter[N](incr, mono)
|
|
t.Run("NonMonotonic", tester.Run(NewDeltaSum[N](mono), incr, eFunc))
|
|
})
|
|
|
|
t.Run("Cumulative", func(t *testing.T) {
|
|
incr, mono := monoIncr, true
|
|
eFunc := cumuExpecter[N](incr, mono)
|
|
t.Run("Monotonic", tester.Run(NewCumulativeSum[N](mono), incr, eFunc))
|
|
|
|
incr, mono = nonMonoIncr, false
|
|
eFunc = cumuExpecter[N](incr, mono)
|
|
t.Run("NonMonotonic", tester.Run(NewCumulativeSum[N](mono), incr, eFunc))
|
|
})
|
|
|
|
t.Run("PreComputedDelta", func(t *testing.T) {
|
|
incr, mono := monoIncr, true
|
|
eFunc := preDeltaExpecter[N](incr, mono)
|
|
t.Run("Monotonic", tester.Run(NewPrecomputedDeltaSum[N](mono), incr, eFunc))
|
|
|
|
incr, mono = nonMonoIncr, false
|
|
eFunc = preDeltaExpecter[N](incr, mono)
|
|
t.Run("NonMonotonic", tester.Run(NewPrecomputedDeltaSum[N](mono), incr, eFunc))
|
|
})
|
|
|
|
t.Run("PreComputedCumulative", func(t *testing.T) {
|
|
incr, mono := monoIncr, true
|
|
eFunc := preCumuExpecter[N](incr, mono)
|
|
t.Run("Monotonic", tester.Run(NewPrecomputedCumulativeSum[N](mono), incr, eFunc))
|
|
|
|
incr, mono = nonMonoIncr, false
|
|
eFunc = preCumuExpecter[N](incr, mono)
|
|
t.Run("NonMonotonic", tester.Run(NewPrecomputedCumulativeSum[N](mono), incr, eFunc))
|
|
})
|
|
}
|
|
|
|
func deltaExpecter[N int64 | float64](incr setMap, mono bool) expectFunc {
|
|
sum := metricdata.Sum[N]{Temporality: metricdata.DeltaTemporality, IsMonotonic: mono}
|
|
return func(m int) metricdata.Aggregation {
|
|
sum.DataPoints = make([]metricdata.DataPoint[N], 0, len(incr))
|
|
for a, v := range incr {
|
|
sum.DataPoints = append(sum.DataPoints, point(a, N(v*m)))
|
|
}
|
|
return sum
|
|
}
|
|
}
|
|
|
|
func cumuExpecter[N int64 | float64](incr setMap, mono bool) expectFunc {
|
|
var cycle int
|
|
sum := metricdata.Sum[N]{Temporality: metricdata.CumulativeTemporality, IsMonotonic: mono}
|
|
return func(m int) metricdata.Aggregation {
|
|
cycle++
|
|
sum.DataPoints = make([]metricdata.DataPoint[N], 0, len(incr))
|
|
for a, v := range incr {
|
|
sum.DataPoints = append(sum.DataPoints, point(a, N(v*cycle*m)))
|
|
}
|
|
return sum
|
|
}
|
|
}
|
|
|
|
func preDeltaExpecter[N int64 | float64](incr setMap, mono bool) expectFunc {
|
|
sum := metricdata.Sum[N]{Temporality: metricdata.DeltaTemporality, IsMonotonic: mono}
|
|
last := make(map[attribute.Set]N)
|
|
return func(int) metricdata.Aggregation {
|
|
sum.DataPoints = make([]metricdata.DataPoint[N], 0, len(incr))
|
|
for a, v := range incr {
|
|
l := last[a]
|
|
sum.DataPoints = append(sum.DataPoints, point(a, N(v)-l))
|
|
last[a] = N(v)
|
|
}
|
|
return sum
|
|
}
|
|
}
|
|
|
|
func preCumuExpecter[N int64 | float64](incr setMap, mono bool) expectFunc {
|
|
sum := metricdata.Sum[N]{Temporality: metricdata.CumulativeTemporality, IsMonotonic: mono}
|
|
return func(int) metricdata.Aggregation {
|
|
sum.DataPoints = make([]metricdata.DataPoint[N], 0, len(incr))
|
|
for a, v := range incr {
|
|
sum.DataPoints = append(sum.DataPoints, point(a, N(v)))
|
|
}
|
|
return sum
|
|
}
|
|
}
|
|
|
|
// point returns a DataPoint that started and ended now.
|
|
func point[N int64 | float64](a attribute.Set, v N) metricdata.DataPoint[N] {
|
|
return metricdata.DataPoint[N]{
|
|
Attributes: a,
|
|
StartTime: now(),
|
|
Time: now(),
|
|
Value: N(v),
|
|
}
|
|
}
|
|
|
|
func testDeltaSumReset[N int64 | float64](t *testing.T) {
|
|
t.Cleanup(mockTime(now))
|
|
|
|
a := NewDeltaSum[N](false)
|
|
assert.Nil(t, a.Aggregation())
|
|
|
|
a.Aggregate(1, alice)
|
|
expect := metricdata.Sum[N]{Temporality: metricdata.DeltaTemporality}
|
|
expect.DataPoints = []metricdata.DataPoint[N]{point[N](alice, 1)}
|
|
metricdatatest.AssertAggregationsEqual(t, expect, a.Aggregation())
|
|
|
|
// The attr set should be forgotten once Aggregations is called.
|
|
expect.DataPoints = nil
|
|
assert.Nil(t, a.Aggregation())
|
|
|
|
// Aggregating another set should not affect the original (alice).
|
|
a.Aggregate(1, bob)
|
|
expect.DataPoints = []metricdata.DataPoint[N]{point[N](bob, 1)}
|
|
metricdatatest.AssertAggregationsEqual(t, expect, a.Aggregation())
|
|
}
|
|
|
|
func TestDeltaSumReset(t *testing.T) {
|
|
t.Run("Int64", testDeltaSumReset[int64])
|
|
t.Run("Float64", testDeltaSumReset[float64])
|
|
}
|
|
|
|
func TestPreComputedDeltaSum(t *testing.T) {
|
|
var mono bool
|
|
agg := NewPrecomputedDeltaSum[int64](mono)
|
|
require.Implements(t, (*precomputeAggregator[int64])(nil), agg)
|
|
|
|
attrs := attribute.NewSet(attribute.String("key", "val"))
|
|
agg.Aggregate(1, attrs)
|
|
got := agg.Aggregation()
|
|
want := metricdata.Sum[int64]{
|
|
IsMonotonic: mono,
|
|
Temporality: metricdata.DeltaTemporality,
|
|
DataPoints: []metricdata.DataPoint[int64]{point[int64](attrs, 1)},
|
|
}
|
|
opt := metricdatatest.IgnoreTimestamp()
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Delta values should zero.
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 0)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(1, attrs)
|
|
got = agg.Aggregation()
|
|
// measured(+): 1, previous(-): 1, filtered(+): 1
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 1)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Filtered values should not persist.
|
|
got = agg.Aggregation()
|
|
// measured(+): 1, previous(-): 2, filtered(+): 0
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, -1)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
got = agg.Aggregation()
|
|
// measured(+): 1, previous(-): 1, filtered(+): 0
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 0)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Override set value.
|
|
agg.Aggregate(2, attrs)
|
|
agg.Aggregate(5, attrs)
|
|
// Filtered should add.
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(3, attrs)
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(10, attrs)
|
|
got = agg.Aggregation()
|
|
// measured(+): 5, previous(-): 1, filtered(+): 13
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 17)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Filtered values should not persist.
|
|
agg.Aggregate(5, attrs)
|
|
got = agg.Aggregation()
|
|
// measured(+): 5, previous(-): 18, filtered(+): 0
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, -13)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Order should not affect measure.
|
|
// Filtered should add.
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(3, attrs)
|
|
agg.Aggregate(7, attrs)
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(10, attrs)
|
|
got = agg.Aggregation()
|
|
// measured(+): 7, previous(-): 5, filtered(+): 13
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 15)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
agg.Aggregate(7, attrs)
|
|
got = agg.Aggregation()
|
|
// measured(+): 7, previous(-): 20, filtered(+): 0
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, -13)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
}
|
|
|
|
func TestPreComputedCumulativeSum(t *testing.T) {
|
|
var mono bool
|
|
agg := NewPrecomputedCumulativeSum[int64](mono)
|
|
require.Implements(t, (*precomputeAggregator[int64])(nil), agg)
|
|
|
|
attrs := attribute.NewSet(attribute.String("key", "val"))
|
|
agg.Aggregate(1, attrs)
|
|
got := agg.Aggregation()
|
|
want := metricdata.Sum[int64]{
|
|
IsMonotonic: mono,
|
|
Temporality: metricdata.CumulativeTemporality,
|
|
DataPoints: []metricdata.DataPoint[int64]{point[int64](attrs, 1)},
|
|
}
|
|
opt := metricdatatest.IgnoreTimestamp()
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Cumulative values should persist.
|
|
got = agg.Aggregation()
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(1, attrs)
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 2)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Filtered values should not persist.
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 1)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Override set value.
|
|
agg.Aggregate(5, attrs)
|
|
// Filtered should add.
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(3, attrs)
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(10, attrs)
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 18)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Filtered values should not persist.
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 5)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
|
|
// Order should not affect measure.
|
|
// Filtered should add.
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(3, attrs)
|
|
agg.Aggregate(7, attrs)
|
|
agg.(precomputeAggregator[int64]).aggregateFiltered(10, attrs)
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 20)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
got = agg.Aggregation()
|
|
want.DataPoints = []metricdata.DataPoint[int64]{point[int64](attrs, 7)}
|
|
metricdatatest.AssertAggregationsEqual(t, want, got, opt)
|
|
}
|
|
|
|
func TestEmptySumNilAggregation(t *testing.T) {
|
|
assert.Nil(t, NewCumulativeSum[int64](true).Aggregation())
|
|
assert.Nil(t, NewCumulativeSum[int64](false).Aggregation())
|
|
assert.Nil(t, NewCumulativeSum[float64](true).Aggregation())
|
|
assert.Nil(t, NewCumulativeSum[float64](false).Aggregation())
|
|
assert.Nil(t, NewDeltaSum[int64](true).Aggregation())
|
|
assert.Nil(t, NewDeltaSum[int64](false).Aggregation())
|
|
assert.Nil(t, NewDeltaSum[float64](true).Aggregation())
|
|
assert.Nil(t, NewDeltaSum[float64](false).Aggregation())
|
|
assert.Nil(t, NewPrecomputedCumulativeSum[int64](true).Aggregation())
|
|
assert.Nil(t, NewPrecomputedCumulativeSum[int64](false).Aggregation())
|
|
assert.Nil(t, NewPrecomputedCumulativeSum[float64](true).Aggregation())
|
|
assert.Nil(t, NewPrecomputedCumulativeSum[float64](false).Aggregation())
|
|
assert.Nil(t, NewPrecomputedDeltaSum[int64](true).Aggregation())
|
|
assert.Nil(t, NewPrecomputedDeltaSum[int64](false).Aggregation())
|
|
assert.Nil(t, NewPrecomputedDeltaSum[float64](true).Aggregation())
|
|
assert.Nil(t, NewPrecomputedDeltaSum[float64](false).Aggregation())
|
|
}
|
|
|
|
func BenchmarkSum(b *testing.B) {
|
|
b.Run("Int64", benchmarkSum[int64])
|
|
b.Run("Float64", benchmarkSum[float64])
|
|
}
|
|
|
|
func benchmarkSum[N int64 | float64](b *testing.B) {
|
|
// The monotonic argument is only used to annotate the Sum returned from
|
|
// the Aggregation method. It should not have an effect on operational
|
|
// performance, therefore, only monotonic=false is benchmarked here.
|
|
factory := func() Aggregator[N] { return NewDeltaSum[N](false) }
|
|
b.Run("Delta", benchmarkAggregator(factory))
|
|
factory = func() Aggregator[N] { return NewCumulativeSum[N](false) }
|
|
b.Run("Cumulative", benchmarkAggregator(factory))
|
|
}
|