2019-10-29 22:27:22 +02:00
|
|
|
// Copyright 2019, 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"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
|
2019-11-01 20:40:29 +02:00
|
|
|
"go.opentelemetry.io/otel/api/core"
|
|
|
|
"go.opentelemetry.io/otel/api/key"
|
|
|
|
"go.opentelemetry.io/otel/api/metric"
|
2019-11-05 23:08:55 +02:00
|
|
|
export "go.opentelemetry.io/otel/sdk/export/metric"
|
2019-11-15 23:01:20 +02:00
|
|
|
"go.opentelemetry.io/otel/sdk/export/metric/aggregator"
|
2019-11-01 20:40:29 +02:00
|
|
|
sdk "go.opentelemetry.io/otel/sdk/metric"
|
|
|
|
"go.opentelemetry.io/otel/sdk/metric/aggregator/gauge"
|
2019-10-29 22:27:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type monotoneBatcher struct {
|
2020-01-06 20:08:40 +02:00
|
|
|
// currentValue needs to be aligned for 64-bit atomic operations.
|
2019-10-29 22:27:22 +02:00
|
|
|
currentValue *core.Number
|
2020-01-06 20:08:40 +02:00
|
|
|
collections int
|
2019-10-29 22:27:22 +02:00
|
|
|
currentTime *time.Time
|
2020-01-06 20:08:40 +02:00
|
|
|
|
|
|
|
t *testing.T
|
2019-10-29 22:27:22 +02:00
|
|
|
}
|
|
|
|
|
2019-11-15 23:01:20 +02:00
|
|
|
func (*monotoneBatcher) AggregatorFor(*export.Descriptor) export.Aggregator {
|
2019-10-29 22:27:22 +02:00
|
|
|
return gauge.New()
|
|
|
|
}
|
|
|
|
|
2019-11-15 23:01:20 +02:00
|
|
|
func (*monotoneBatcher) CheckpointSet() export.CheckpointSet {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (*monotoneBatcher) FinishedCollection() {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *monotoneBatcher) Process(_ context.Context, record export.Record) error {
|
2019-10-29 22:27:22 +02:00
|
|
|
require.Equal(m.t, "my.gauge.name", record.Descriptor().Name())
|
2019-11-15 23:01:20 +02:00
|
|
|
require.Equal(m.t, 1, record.Labels().Len())
|
|
|
|
require.Equal(m.t, "a", string(record.Labels().Ordered()[0].Key))
|
|
|
|
require.Equal(m.t, "b", record.Labels().Ordered()[0].Value.Emit())
|
2019-10-29 22:27:22 +02:00
|
|
|
|
2019-11-15 23:01:20 +02:00
|
|
|
gauge := record.Aggregator().(*gauge.Aggregator)
|
|
|
|
val, ts, err := gauge.LastValue()
|
|
|
|
require.Nil(m.t, err)
|
2019-10-29 22:27:22 +02:00
|
|
|
|
|
|
|
m.currentValue = &val
|
|
|
|
m.currentTime = &ts
|
|
|
|
m.collections++
|
2019-11-15 23:01:20 +02:00
|
|
|
return nil
|
2019-10-29 22:27:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestMonotoneGauge(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
|
|
|
batcher := &monotoneBatcher{
|
|
|
|
t: t,
|
|
|
|
}
|
2019-11-22 06:46:05 +02:00
|
|
|
sdk := sdk.New(batcher, sdk.NewDefaultLabelEncoder())
|
2019-11-15 23:01:20 +02:00
|
|
|
|
|
|
|
sdk.SetErrorHandler(func(error) { t.Fatal("Unexpected") })
|
2019-10-29 22:27:22 +02:00
|
|
|
|
|
|
|
gauge := sdk.NewInt64Gauge("my.gauge.name", metric.WithMonotonic(true))
|
|
|
|
|
2019-12-28 02:30:19 +02:00
|
|
|
handle := gauge.Bind(sdk.Labels(key.String("a", "b")))
|
2019-10-29 22:27:22 +02:00
|
|
|
|
|
|
|
require.Nil(t, batcher.currentTime)
|
|
|
|
require.Nil(t, batcher.currentValue)
|
|
|
|
|
|
|
|
before := time.Now()
|
|
|
|
|
|
|
|
handle.Set(ctx, 1)
|
|
|
|
|
|
|
|
// Until collection, expect nil.
|
|
|
|
require.Nil(t, batcher.currentTime)
|
|
|
|
require.Nil(t, batcher.currentValue)
|
|
|
|
|
|
|
|
sdk.Collect(ctx)
|
|
|
|
|
|
|
|
require.NotNil(t, batcher.currentValue)
|
|
|
|
require.Equal(t, core.NewInt64Number(1), *batcher.currentValue)
|
|
|
|
require.True(t, before.Before(*batcher.currentTime))
|
|
|
|
|
|
|
|
before = *batcher.currentTime
|
|
|
|
|
|
|
|
// Collect would ordinarily flush the record, except we're using a handle.
|
|
|
|
sdk.Collect(ctx)
|
|
|
|
|
|
|
|
require.Equal(t, 2, batcher.collections)
|
|
|
|
|
|
|
|
// Increase the value to 2.
|
|
|
|
handle.Set(ctx, 2)
|
|
|
|
|
|
|
|
sdk.Collect(ctx)
|
|
|
|
|
|
|
|
require.Equal(t, 3, batcher.collections)
|
|
|
|
require.Equal(t, core.NewInt64Number(2), *batcher.currentValue)
|
|
|
|
require.True(t, before.Before(*batcher.currentTime))
|
|
|
|
|
|
|
|
before = *batcher.currentTime
|
|
|
|
|
|
|
|
sdk.Collect(ctx)
|
|
|
|
require.Equal(t, 4, batcher.collections)
|
|
|
|
|
|
|
|
// Try to lower the value to 1, it will fail.
|
2019-11-15 23:01:20 +02:00
|
|
|
var err error
|
|
|
|
sdk.SetErrorHandler(func(sdkErr error) {
|
|
|
|
err = sdkErr
|
|
|
|
})
|
2019-10-29 22:27:22 +02:00
|
|
|
handle.Set(ctx, 1)
|
2019-11-15 23:01:20 +02:00
|
|
|
require.Equal(t, aggregator.ErrNonMonotoneInput, err)
|
|
|
|
sdk.SetErrorHandler(func(error) { t.Fatal("Unexpected") })
|
|
|
|
|
2019-10-29 22:27:22 +02:00
|
|
|
sdk.Collect(ctx)
|
|
|
|
|
|
|
|
// The value and timestamp are both unmodified
|
|
|
|
require.Equal(t, 5, batcher.collections)
|
|
|
|
require.Equal(t, core.NewInt64Number(2), *batcher.currentValue)
|
|
|
|
require.Equal(t, before, *batcher.currentTime)
|
|
|
|
|
|
|
|
// Update with the same value, update the timestamp.
|
|
|
|
handle.Set(ctx, 2)
|
|
|
|
sdk.Collect(ctx)
|
|
|
|
|
|
|
|
require.Equal(t, 6, batcher.collections)
|
|
|
|
require.Equal(t, core.NewInt64Number(2), *batcher.currentValue)
|
|
|
|
require.True(t, before.Before(*batcher.currentTime))
|
|
|
|
}
|