// 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_test import ( "bytes" "context" "errors" "io" "io/ioutil" "testing" "github.com/stretchr/testify/require" "go.opentelemetry.io/otel/api/core" "go.opentelemetry.io/otel/api/global" "go.opentelemetry.io/otel/api/global/internal" "go.opentelemetry.io/otel/api/key" "go.opentelemetry.io/otel/api/metric" "go.opentelemetry.io/otel/exporters/metric/stdout" metrictest "go.opentelemetry.io/otel/internal/metric" ) // Note: Maybe this should be factored into ../../../internal/metric? type measured struct { Name string LibraryName string Labels map[core.Key]core.Value Number core.Number } func asStructs(batches []metrictest.Batch) []measured { var r []measured for _, batch := range batches { for _, m := range batch.Measurements { r = append(r, measured{ Name: m.Instrument.Descriptor().Name(), LibraryName: m.Instrument.Descriptor().LibraryName(), Labels: asMap(batch.Labels...), Number: m.Number, }) } } return r } func asMap(kvs ...core.KeyValue) map[core.Key]core.Value { m := map[core.Key]core.Value{} for _, kv := range kvs { m[kv.Key] = kv.Value } return m } var asInt = core.NewInt64Number var asFloat = core.NewFloat64Number func TestDirect(t *testing.T) { internal.ResetForTest() ctx := context.Background() meter1 := global.Meter("test1") meter2 := global.Meter("test2") labels1 := []core.KeyValue{key.String("A", "B")} labels2 := []core.KeyValue{key.String("C", "D")} labels3 := []core.KeyValue{key.String("E", "F")} counter := Must(meter1).NewInt64Counter("test.counter") counter.Add(ctx, 1, labels1...) counter.Add(ctx, 1, labels1...) measure := Must(meter1).NewFloat64Measure("test.measure") measure.Record(ctx, 1, labels1...) measure.Record(ctx, 2, labels1...) _ = Must(meter1).RegisterFloat64Observer("test.observer.float", func(result metric.Float64ObserverResult) { result.Observe(1., labels1...) result.Observe(2., labels2...) }) _ = Must(meter1).RegisterInt64Observer("test.observer.int", func(result metric.Int64ObserverResult) { result.Observe(1, labels1...) result.Observe(2, labels2...) }) second := Must(meter2).NewFloat64Measure("test.second") second.Record(ctx, 1, labels3...) second.Record(ctx, 2, labels3...) mock, provider := metrictest.NewProvider() global.SetMeterProvider(provider) counter.Add(ctx, 1, labels1...) measure.Record(ctx, 3, labels1...) second.Record(ctx, 3, labels3...) mock.RunAsyncInstruments() measurements := asStructs(mock.MeasurementBatches) require.EqualValues(t, []measured{ { Name: "test.counter", LibraryName: "test1", Labels: asMap(labels1...), Number: asInt(1), }, { Name: "test.measure", LibraryName: "test1", Labels: asMap(labels1...), Number: asFloat(3), }, { Name: "test.second", LibraryName: "test2", Labels: asMap(labels3...), Number: asFloat(3), }, { Name: "test.observer.float", LibraryName: "test1", Labels: asMap(labels1...), Number: asFloat(1), }, { Name: "test.observer.float", LibraryName: "test1", Labels: asMap(labels2...), Number: asFloat(2), }, { Name: "test.observer.int", LibraryName: "test1", Labels: asMap(labels1...), Number: asInt(1), }, { Name: "test.observer.int", LibraryName: "test1", Labels: asMap(labels2...), Number: asInt(2), }, }, measurements, ) } func TestBound(t *testing.T) { internal.ResetForTest() // Note: this test uses opposite Float64/Int64 number kinds // vs. the above, to cover all the instruments. ctx := context.Background() glob := global.Meter("test") labels1 := []core.KeyValue{key.String("A", "B")} counter := Must(glob).NewFloat64Counter("test.counter") boundC := counter.Bind(labels1...) boundC.Add(ctx, 1) boundC.Add(ctx, 1) measure := Must(glob).NewInt64Measure("test.measure") boundM := measure.Bind(labels1...) boundM.Record(ctx, 1) boundM.Record(ctx, 2) mock, provider := metrictest.NewProvider() global.SetMeterProvider(provider) boundC.Add(ctx, 1) boundM.Record(ctx, 3) require.EqualValues(t, []measured{ { Name: "test.counter", LibraryName: "test", Labels: asMap(labels1...), Number: asFloat(1), }, { Name: "test.measure", LibraryName: "test", Labels: asMap(labels1...), Number: asInt(3), }, }, asStructs(mock.MeasurementBatches)) boundC.Unbind() boundM.Unbind() } func TestUnbind(t *testing.T) { // Tests Unbind with SDK never installed. internal.ResetForTest() glob := global.Meter("test") labels1 := []core.KeyValue{key.String("A", "B")} counter := Must(glob).NewFloat64Counter("test.counter") boundC := counter.Bind(labels1...) measure := Must(glob).NewInt64Measure("test.measure") boundM := measure.Bind(labels1...) boundC.Unbind() boundM.Unbind() } func TestDefaultSDK(t *testing.T) { internal.ResetForTest() ctx := context.Background() meter1 := global.Meter("builtin") labels1 := []core.KeyValue{key.String("A", "B")} counter := Must(meter1).NewInt64Counter("test.builtin") counter.Add(ctx, 1, labels1...) counter.Add(ctx, 1, labels1...) in, out := io.Pipe() pusher, err := stdout.InstallNewPipeline(stdout.Config{ Writer: out, DoNotPrintTime: true, }) if err != nil { panic(err) } counter.Add(ctx, 1, labels1...) ch := make(chan string) go func() { data, _ := ioutil.ReadAll(in) ch <- string(data) }() pusher.Stop() out.Close() require.Equal(t, `{"updates":[{"name":"test.builtin{A=B}","sum":1}]} `, <-ch) } func TestUnbindThenRecordOne(t *testing.T) { internal.ResetForTest() ctx := context.Background() mock, provider := metrictest.NewProvider() meter := global.Meter("test") counter := Must(meter).NewInt64Counter("test.counter") boundC := counter.Bind() global.SetMeterProvider(provider) boundC.Unbind() require.NotPanics(t, func() { boundC.Add(ctx, 1) }) require.Equal(t, 0, len(mock.MeasurementBatches)) } type meterProviderWithConstructorError struct { metric.Provider } type meterWithConstructorError struct { metric.Meter } func (m *meterProviderWithConstructorError) Meter(name string) metric.Meter { return &meterWithConstructorError{m.Provider.Meter(name)} } func (m *meterWithConstructorError) NewInt64Counter(name string, opts ...metric.Option) (metric.Int64Counter, error) { return metric.Int64Counter{}, errors.New("constructor error") } func TestErrorInDeferredConstructor(t *testing.T) { internal.ResetForTest() ctx := context.Background() meter := global.MeterProvider().Meter("builtin") c1 := Must(meter).NewInt64Counter("test") c2 := Must(meter).NewInt64Counter("test") _, provider := metrictest.NewProvider() sdk := &meterProviderWithConstructorError{provider} require.Panics(t, func() { global.SetMeterProvider(sdk) }) c1.Add(ctx, 1) c2.Add(ctx, 2) } func TestImplementationIndirection(t *testing.T) { internal.ResetForTest() // Test that Implementation() does the proper indirection, i.e., // returns the implementation interface not the global, after // registered. meter1 := global.Meter("test1") // Sync: no SDK yet counter := Must(meter1).NewInt64Counter("interface.counter") ival := counter.Measurement(1).SyncImpl().Implementation() require.NotNil(t, ival) _, ok := ival.(*metrictest.Sync) require.False(t, ok) // Async: no SDK yet observer := Must(meter1).RegisterFloat64Observer( "interface.observer", func(result metric.Float64ObserverResult) {}, ) ival = observer.AsyncImpl().Implementation() require.NotNil(t, ival) _, ok = ival.(*metrictest.Async) require.False(t, ok) // Register the SDK _, provider := metrictest.NewProvider() global.SetMeterProvider(provider) // Repeat the above tests // Sync ival = counter.Measurement(1).SyncImpl().Implementation() require.NotNil(t, ival) _, ok = ival.(*metrictest.Sync) require.True(t, ok) // Async ival = observer.AsyncImpl().Implementation() require.NotNil(t, ival) _, ok = ival.(*metrictest.Async) require.True(t, ok) } func TestRecordBatchMock(t *testing.T) { internal.ResetForTest() meter := global.MeterProvider().Meter("builtin") counter := Must(meter).NewInt64Counter("test.counter") meter.RecordBatch(context.Background(), nil, counter.Measurement(1)) mock, provider := metrictest.NewProvider() global.SetMeterProvider(provider) meter.RecordBatch(context.Background(), nil, counter.Measurement(1)) require.EqualValues(t, []measured{ { Name: "test.counter", LibraryName: "builtin", Labels: asMap(), Number: asInt(1), }, }, asStructs(mock.MeasurementBatches)) } func TestRecordBatchRealSDK(t *testing.T) { internal.ResetForTest() meter := global.MeterProvider().Meter("builtin") counter := Must(meter).NewInt64Counter("test.counter") meter.RecordBatch(context.Background(), nil, counter.Measurement(1)) var buf bytes.Buffer pusher, err := stdout.InstallNewPipeline(stdout.Config{ Writer: &buf, DoNotPrintTime: true, }) if err != nil { t.Fatal(err) } global.SetMeterProvider(pusher) meter.RecordBatch(context.Background(), nil, counter.Measurement(1)) pusher.Stop() require.Equal(t, `{"updates":[{"name":"test.counter","sum":1}]} `, buf.String()) }