// 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 otlp import ( "context" "testing" colmetricpb "github.com/open-telemetry/opentelemetry-proto/gen/go/collector/metrics/v1" commonpb "github.com/open-telemetry/opentelemetry-proto/gen/go/common/v1" metricpb "github.com/open-telemetry/opentelemetry-proto/gen/go/metrics/v1" resourcepb "github.com/open-telemetry/opentelemetry-proto/gen/go/resource/v1" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "go.opentelemetry.io/otel/api/kv" "go.opentelemetry.io/otel/api/label" "go.opentelemetry.io/otel/api/metric" metricsdk "go.opentelemetry.io/otel/sdk/export/metric" "go.opentelemetry.io/otel/sdk/export/metric/aggregator" "go.opentelemetry.io/otel/sdk/metric/aggregator/minmaxsumcount" "go.opentelemetry.io/otel/sdk/metric/aggregator/sum" "go.opentelemetry.io/otel/sdk/resource" "google.golang.org/grpc" ) type metricsServiceClientStub struct { rm []metricpb.ResourceMetrics } func (m *metricsServiceClientStub) Export(ctx context.Context, in *colmetricpb.ExportMetricsServiceRequest, opts ...grpc.CallOption) (*colmetricpb.ExportMetricsServiceResponse, error) { for _, rm := range in.GetResourceMetrics() { if rm == nil { continue } m.rm = append(m.rm, *rm) } return &colmetricpb.ExportMetricsServiceResponse{}, nil } func (m *metricsServiceClientStub) ResourceMetrics() []metricpb.ResourceMetrics { return m.rm } func (m *metricsServiceClientStub) Reset() { m.rm = nil } type checkpointSet struct { records []metricsdk.Record } func (m checkpointSet) ForEach(fn func(metricsdk.Record) error) error { for _, r := range m.records { if err := fn(r); err != nil && err != aggregator.ErrNoData { return err } } return nil } type record struct { name string mKind metric.Kind nKind metric.NumberKind resource *resource.Resource opts []metric.Option labels []kv.KeyValue } var ( baseKeyValues = []kv.KeyValue{kv.String("host", "test.com")} cpuKey = kv.Key("CPU") testInstA = resource.New(kv.String("instance", "tester-a")) testInstB = resource.New(kv.String("instance", "tester-b")) cpu1MD = &metricpb.MetricDescriptor{ Name: "int64-count", Type: metricpb.MetricDescriptor_COUNTER_INT64, Labels: []*commonpb.StringKeyValue{ { Key: "CPU", Value: "1", }, { Key: "host", Value: "test.com", }, }, } cpu2MD = &metricpb.MetricDescriptor{ Name: "int64-count", Type: metricpb.MetricDescriptor_COUNTER_INT64, Labels: []*commonpb.StringKeyValue{ { Key: "CPU", Value: "2", }, { Key: "host", Value: "test.com", }, }, } testerAResource = &resourcepb.Resource{ Attributes: []*commonpb.AttributeKeyValue{ { Key: "instance", Type: commonpb.AttributeKeyValue_STRING, StringValue: "tester-a", }, }, } testerBResource = &resourcepb.Resource{ Attributes: []*commonpb.AttributeKeyValue{ { Key: "instance", Type: commonpb.AttributeKeyValue_STRING, StringValue: "tester-b", }, }, } ) func TestNoGroupingExport(t *testing.T) { runMetricExportTests( t, []record{ { "int64-count", metric.CounterKind, metric.Int64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(2)), }, }, []metricpb.ResourceMetrics{ { Resource: nil, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, { MetricDescriptor: cpu2MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, }, }, }, ) } func TestMeasureMetricGroupingExport(t *testing.T) { r := record{ "measure", metric.MeasureKind, metric.Int64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(1)), } expected := []metricpb.ResourceMetrics{ { Resource: nil, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: &metricpb.MetricDescriptor{ Name: "measure", Type: metricpb.MetricDescriptor_SUMMARY, Labels: []*commonpb.StringKeyValue{ { Key: "CPU", Value: "1", }, { Key: "host", Value: "test.com", }, }, }, SummaryDataPoints: []*metricpb.SummaryDataPoint{ { Count: 2, Sum: 11, PercentileValues: []*metricpb.SummaryDataPoint_ValueAtPercentile{ { Percentile: 0.0, Value: 1.0, }, { Percentile: 100.0, Value: 10.0, }, }, }, { Count: 2, Sum: 11, PercentileValues: []*metricpb.SummaryDataPoint_ValueAtPercentile{ { Percentile: 0.0, Value: 1.0, }, { Percentile: 100.0, Value: 10.0, }, }, }, }, }, }, }, }, }, } runMetricExportTests(t, []record{r, r}, expected) //changing the number kind should make no difference. r.nKind = metric.Uint64NumberKind runMetricExportTests(t, []record{r, r}, expected) r.nKind = metric.Float64NumberKind runMetricExportTests(t, []record{r, r}, expected) } func TestCountInt64MetricGroupingExport(t *testing.T) { r := record{ "int64-count", metric.CounterKind, metric.Int64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(1)), } runMetricExportTests( t, []record{r, r}, []metricpb.ResourceMetrics{ { Resource: nil, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, { Value: 11, }, }, }, }, }, }, }, }, ) } func TestCountUint64MetricGroupingExport(t *testing.T) { r := record{ "uint64-count", metric.CounterKind, metric.Uint64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(1)), } runMetricExportTests( t, []record{r, r}, []metricpb.ResourceMetrics{ { Resource: nil, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: &metricpb.MetricDescriptor{ Name: "uint64-count", Type: metricpb.MetricDescriptor_COUNTER_INT64, Labels: []*commonpb.StringKeyValue{ { Key: "CPU", Value: "1", }, { Key: "host", Value: "test.com", }, }, }, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, { Value: 11, }, }, }, }, }, }, }, }, ) } func TestCountFloat64MetricGroupingExport(t *testing.T) { r := record{ "float64-count", metric.CounterKind, metric.Float64NumberKind, nil, nil, append(baseKeyValues, cpuKey.Int(1)), } runMetricExportTests( t, []record{r, r}, []metricpb.ResourceMetrics{ { Resource: nil, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: &metricpb.MetricDescriptor{ Name: "float64-count", Type: metricpb.MetricDescriptor_COUNTER_DOUBLE, Labels: []*commonpb.StringKeyValue{ { Key: "CPU", Value: "1", }, { Key: "host", Value: "test.com", }, }, }, DoubleDataPoints: []*metricpb.DoubleDataPoint{ { Value: 11, }, { Value: 11, }, }, }, }, }, }, }, }, ) } func TestResourceMetricGroupingExport(t *testing.T) { runMetricExportTests( t, []record{ { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, nil, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, nil, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, nil, append(baseKeyValues, cpuKey.Int(2)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstB, nil, append(baseKeyValues, cpuKey.Int(1)), }, }, []metricpb.ResourceMetrics{ { Resource: testerAResource, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, { Value: 11, }, }, }, { MetricDescriptor: cpu2MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, }, }, { Resource: testerBResource, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, }, }, }, ) } func TestResourceInstLibMetricGroupingExport(t *testing.T) { runMetricExportTests( t, []record{ { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, []metric.Option{ metric.WithLibraryName("couting-lib"), }, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, []metric.Option{ metric.WithLibraryName("couting-lib"), }, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, []metric.Option{ metric.WithLibraryName("couting-lib"), }, append(baseKeyValues, cpuKey.Int(2)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstA, []metric.Option{ metric.WithLibraryName("summing-lib"), }, append(baseKeyValues, cpuKey.Int(1)), }, { "int64-count", metric.CounterKind, metric.Int64NumberKind, testInstB, []metric.Option{ metric.WithLibraryName("couting-lib"), }, append(baseKeyValues, cpuKey.Int(1)), }, }, []metricpb.ResourceMetrics{ { Resource: testerAResource, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { InstrumentationLibrary: &commonpb.InstrumentationLibrary{ Name: "couting-lib", }, Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, { Value: 11, }, }, }, { MetricDescriptor: cpu2MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, { InstrumentationLibrary: &commonpb.InstrumentationLibrary{ Name: "summing-lib", }, Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, }, }, { Resource: testerBResource, InstrumentationLibraryMetrics: []*metricpb.InstrumentationLibraryMetrics{ { InstrumentationLibrary: &commonpb.InstrumentationLibrary{ Name: "couting-lib", }, Metrics: []*metricpb.Metric{ { MetricDescriptor: cpu1MD, Int64DataPoints: []*metricpb.Int64DataPoint{ { Value: 11, }, }, }, }, }, }, }, }, ) } // What works single-threaded should work multi-threaded func runMetricExportTests(t *testing.T, rs []record, expected []metricpb.ResourceMetrics) { t.Run("1 goroutine", func(t *testing.T) { runMetricExportTest(t, NewUnstartedExporter(WorkerCount(1)), rs, expected) }) t.Run("20 goroutines", func(t *testing.T) { runMetricExportTest(t, NewUnstartedExporter(WorkerCount(20)), rs, expected) }) } func runMetricExportTest(t *testing.T, exp *Exporter, rs []record, expected []metricpb.ResourceMetrics) { msc := &metricsServiceClientStub{} exp.metricExporter = msc exp.started = true recs := map[label.Distinct][]metricsdk.Record{} resources := map[label.Distinct]*resource.Resource{} for _, r := range rs { desc := metric.NewDescriptor(r.name, r.mKind, r.nKind, r.opts...) labs := label.NewSet(r.labels...) var agg metricsdk.Aggregator switch r.mKind { case metric.CounterKind: agg = sum.New() default: agg = minmaxsumcount.New(&desc) } ctx := context.Background() switch r.nKind { case metric.Uint64NumberKind: require.NoError(t, agg.Update(ctx, metric.NewUint64Number(1), &desc)) require.NoError(t, agg.Update(ctx, metric.NewUint64Number(10), &desc)) case metric.Int64NumberKind: require.NoError(t, agg.Update(ctx, metric.NewInt64Number(1), &desc)) require.NoError(t, agg.Update(ctx, metric.NewInt64Number(10), &desc)) case metric.Float64NumberKind: require.NoError(t, agg.Update(ctx, metric.NewFloat64Number(1), &desc)) require.NoError(t, agg.Update(ctx, metric.NewFloat64Number(10), &desc)) default: t.Fatalf("invalid number kind: %v", r.nKind) } agg.Checkpoint(ctx, &desc) equiv := r.resource.Equivalent() resources[equiv] = r.resource recs[equiv] = append(recs[equiv], metricsdk.NewRecord(&desc, &labs, agg)) } for equiv, records := range recs { resource := resources[equiv] assert.NoError(t, exp.Export(context.Background(), resource, checkpointSet{records: records})) } // assert.ElementsMatch does not equate nested slices of different order, // therefore this requires the top level slice to be broken down. // Build a map of Resource/InstrumentationLibrary pairs to Metrics, from // that validate the metric elements match for all expected pairs. Finally, // make we saw all expected pairs. type key struct { resource, instrumentationLibrary string } got := map[key][]*metricpb.Metric{} for _, rm := range msc.ResourceMetrics() { for _, ilm := range rm.InstrumentationLibraryMetrics { k := key{ resource: rm.GetResource().String(), instrumentationLibrary: ilm.GetInstrumentationLibrary().String(), } got[k] = ilm.GetMetrics() } } seen := map[key]struct{}{} for _, rm := range expected { for _, ilm := range rm.InstrumentationLibraryMetrics { k := key{ resource: rm.GetResource().String(), instrumentationLibrary: ilm.GetInstrumentationLibrary().String(), } seen[k] = struct{}{} g, ok := got[k] if !ok { t.Errorf("missing metrics for:\n\tResource: %s\n\tInstrumentationLibrary: %s\n", k.resource, k.instrumentationLibrary) continue } assert.ElementsMatch(t, ilm.GetMetrics(), g, "metrics did not match for:\n\tResource: %s\n\tInstrumentationLibrary: %s\n", k.resource, k.instrumentationLibrary) } } for k := range got { if _, ok := seen[k]; !ok { t.Errorf("did not expect metrics for:\n\tResource: %s\n\tInstrumentationLibrary: %s\n", k.resource, k.instrumentationLibrary) } } } func TestEmptyMetricExport(t *testing.T) { msc := &metricsServiceClientStub{} exp := NewUnstartedExporter() exp.metricExporter = msc exp.started = true resource := resource.New(kv.String("R", "S")) for _, test := range []struct { records []metricsdk.Record want []metricpb.ResourceMetrics }{ { []metricsdk.Record(nil), []metricpb.ResourceMetrics(nil), }, { []metricsdk.Record{}, []metricpb.ResourceMetrics(nil), }, } { msc.Reset() require.NoError(t, exp.Export(context.Background(), resource, checkpointSet{records: test.records})) assert.Equal(t, test.want, msc.ResourceMetrics()) } }