You've already forked opentelemetry-go
							
							
				mirror of
				https://github.com/open-telemetry/opentelemetry-go.git
				synced 2025-10-31 00:07:40 +02:00 
			
		
		
		
	* Rename metric Handle to Bound Instrument * Rename metric Handle to Bound Instrument * Rename metric Handle to Bound Instrument * chore(meter): renamed from *BoundInstrument to Bound*, renamed AcquireBoundInstrument to Bind * chore(meter): renamed Release to Unbind * Self feedback in doc.go * Rename confusing method name Co-authored-by: Marc Bramaud <sircelsius@users.noreply.github.com>
		
			
				
	
	
		
			143 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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"
 | |
| 
 | |
| 	"go.opentelemetry.io/otel/api/core"
 | |
| 	"go.opentelemetry.io/otel/api/key"
 | |
| 	"go.opentelemetry.io/otel/api/metric"
 | |
| 	export "go.opentelemetry.io/otel/sdk/export/metric"
 | |
| 	"go.opentelemetry.io/otel/sdk/export/metric/aggregator"
 | |
| 	sdk "go.opentelemetry.io/otel/sdk/metric"
 | |
| 	"go.opentelemetry.io/otel/sdk/metric/aggregator/gauge"
 | |
| )
 | |
| 
 | |
| type monotoneBatcher struct {
 | |
| 	t *testing.T
 | |
| 
 | |
| 	collections  int
 | |
| 	currentValue *core.Number
 | |
| 	currentTime  *time.Time
 | |
| }
 | |
| 
 | |
| func (*monotoneBatcher) AggregatorFor(*export.Descriptor) export.Aggregator {
 | |
| 	return gauge.New()
 | |
| }
 | |
| 
 | |
| func (*monotoneBatcher) CheckpointSet() export.CheckpointSet {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (*monotoneBatcher) FinishedCollection() {
 | |
| }
 | |
| 
 | |
| func (m *monotoneBatcher) Process(_ context.Context, record export.Record) error {
 | |
| 	require.Equal(m.t, "my.gauge.name", record.Descriptor().Name())
 | |
| 	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())
 | |
| 
 | |
| 	gauge := record.Aggregator().(*gauge.Aggregator)
 | |
| 	val, ts, err := gauge.LastValue()
 | |
| 	require.Nil(m.t, err)
 | |
| 
 | |
| 	m.currentValue = &val
 | |
| 	m.currentTime = &ts
 | |
| 	m.collections++
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func TestMonotoneGauge(t *testing.T) {
 | |
| 	ctx := context.Background()
 | |
| 	batcher := &monotoneBatcher{
 | |
| 		t: t,
 | |
| 	}
 | |
| 	sdk := sdk.New(batcher, sdk.NewDefaultLabelEncoder())
 | |
| 
 | |
| 	sdk.SetErrorHandler(func(error) { t.Fatal("Unexpected") })
 | |
| 
 | |
| 	gauge := sdk.NewInt64Gauge("my.gauge.name", metric.WithMonotonic(true))
 | |
| 
 | |
| 	handle := gauge.Bind(sdk.Labels(key.String("a", "b")))
 | |
| 
 | |
| 	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.
 | |
| 	var err error
 | |
| 	sdk.SetErrorHandler(func(sdkErr error) {
 | |
| 		err = sdkErr
 | |
| 	})
 | |
| 	handle.Set(ctx, 1)
 | |
| 	require.Equal(t, aggregator.ErrNonMonotoneInput, err)
 | |
| 	sdk.SetErrorHandler(func(error) { t.Fatal("Unexpected") })
 | |
| 
 | |
| 	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))
 | |
| }
 |