From b320efaa35cabfb0328b0dbb21890d1b8a67baa8 Mon Sep 17 00:00:00 2001 From: Vishal Raj Date: Thu, 9 Jan 2025 14:57:33 +0000 Subject: [PATCH 1/2] [connector/signaltometrics]Add collector telemetry as resource attributes --- .../signaltometricsconnector/connector.go | 11 ++- connector/signaltometricsconnector/factory.go | 15 ++- .../internal/model/collectorinstanceinfo.go | 75 +++++++++++++++ .../model/collectorinstanceinfo_test.go | 94 +++++++++++++++++++ .../internal/model/model.go | 6 +- .../logs/exponential_histograms/output.yaml | 18 ++++ .../testdata/logs/histograms/output.yaml | 18 ++++ .../testdata/logs/sum/output.yaml | 18 ++++ .../exponential_histograms/output.yaml | 9 ++ .../testdata/metrics/histograms/output.yaml | 9 ++ .../testdata/metrics/sum/output.yaml | 9 ++ .../traces/exponential_histograms/output.yaml | 18 ++++ .../testdata/traces/histograms/output.yaml | 18 ++++ .../testdata/traces/sum/output.yaml | 18 ++++ 14 files changed, 326 insertions(+), 10 deletions(-) create mode 100644 connector/signaltometricsconnector/internal/model/collectorinstanceinfo.go create mode 100644 connector/signaltometricsconnector/internal/model/collectorinstanceinfo_test.go diff --git a/connector/signaltometricsconnector/connector.go b/connector/signaltometricsconnector/connector.go index b1a9234fe2f5..2253faf6e4e6 100644 --- a/connector/signaltometricsconnector/connector.go +++ b/connector/signaltometricsconnector/connector.go @@ -23,8 +23,9 @@ import ( ) type signalToMetrics struct { - next consumer.Metrics - logger *zap.Logger + next consumer.Metrics + collectorInstanceInfo *model.CollectorInstanceInfo + logger *zap.Logger spanMetricDefs []model.MetricDef[ottlspan.TransformContext] dpMetricDefs []model.MetricDef[ottldatapoint.TransformContext] @@ -75,7 +76,7 @@ func (sm *signalToMetrics) ConsumeTraces(ctx context.Context, td ptrace.Traces) } } - filteredResAttrs := md.FilterResourceAttributes(resourceAttrs) + filteredResAttrs := md.FilterResourceAttributes(resourceAttrs, sm.collectorInstanceInfo) if err := aggregator.Aggregate(ctx, tCtx, md, filteredResAttrs, filteredSpanAttrs, 1); err != nil { return err } @@ -104,7 +105,7 @@ func (sm *signalToMetrics) ConsumeMetrics(ctx context.Context, m pmetric.Metrics metrics := scopeMetric.Metrics() metric := metrics.At(k) for _, md := range sm.dpMetricDefs { - filteredResAttrs := md.FilterResourceAttributes(resourceAttrs) + filteredResAttrs := md.FilterResourceAttributes(resourceAttrs, sm.collectorInstanceInfo) aggregate := func(dp any, dpAttrs pcommon.Map) error { // The transform context is created from original attributes so that the // OTTL expressions are also applied on the original attributes. @@ -230,7 +231,7 @@ func (sm *signalToMetrics) ConsumeLogs(ctx context.Context, logs plog.Logs) erro continue } } - filteredResAttrs := md.FilterResourceAttributes(resourceAttrs) + filteredResAttrs := md.FilterResourceAttributes(resourceAttrs, sm.collectorInstanceInfo) if err := aggregator.Aggregate(ctx, tCtx, md, filteredResAttrs, filteredLogAttrs, 1); err != nil { return err } diff --git a/connector/signaltometricsconnector/factory.go b/connector/signaltometricsconnector/factory.go index 673722dd6ce0..04a3038448c3 100644 --- a/connector/signaltometricsconnector/factory.go +++ b/connector/signaltometricsconnector/factory.go @@ -57,7 +57,10 @@ func createTracesToMetrics( } return &signalToMetrics{ - logger: set.Logger, + logger: set.Logger, + collectorInstanceInfo: model.NewCollectorInstanceInfo( + set.TelemetrySettings, + ), next: nextConsumer, spanMetricDefs: metricDefs, }, nil @@ -85,7 +88,10 @@ func createMetricsToMetrics( } return &signalToMetrics{ - logger: set.Logger, + logger: set.Logger, + collectorInstanceInfo: model.NewCollectorInstanceInfo( + set.TelemetrySettings, + ), next: nextConsumer, dpMetricDefs: metricDefs, }, nil @@ -113,7 +119,10 @@ func createLogsToMetrics( } return &signalToMetrics{ - logger: set.Logger, + logger: set.Logger, + collectorInstanceInfo: model.NewCollectorInstanceInfo( + set.TelemetrySettings, + ), next: nextConsumer, logMetricDefs: metricDefs, }, nil diff --git a/connector/signaltometricsconnector/internal/model/collectorinstanceinfo.go b/connector/signaltometricsconnector/internal/model/collectorinstanceinfo.go new file mode 100644 index 000000000000..f6742fc7271c --- /dev/null +++ b/connector/signaltometricsconnector/internal/model/collectorinstanceinfo.go @@ -0,0 +1,75 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package model // import "github.com/open-telemetry/opentelemetry-collector-contrib/connector/signaltometricsconnector/internal/model" + +import ( + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/pdata/pcommon" + semconv "go.opentelemetry.io/collector/semconv/v1.26.0" + + "github.com/open-telemetry/opentelemetry-collector-contrib/connector/signaltometricsconnector/internal/metadata" +) + +var prefix = metadata.Type.String() + +// CollectorInstanceInfo holds the attributes that could uniquely identify +// the current collector instance. These attributes are initialized from the +// telemetry settings. The CollectorInstanceInfo can copy these attributes, +// with a given prefix, to a provided map. +type CollectorInstanceInfo struct { + size int + serviceInstanceID string + serviceName string + serviceNamespace string +} + +func NewCollectorInstanceInfo( + set component.TelemetrySettings, +) *CollectorInstanceInfo { + var info CollectorInstanceInfo + set.Resource.Attributes().Range(func(k string, v pcommon.Value) bool { + switch k { + case semconv.AttributeServiceInstanceID: + if str := v.Str(); str != "" { + info.serviceInstanceID = str + info.size++ + } + case semconv.AttributeServiceName: + if str := v.Str(); str != "" { + info.serviceName = str + info.size++ + } + case semconv.AttributeServiceNamespace: + if str := v.Str(); str != "" { + info.serviceNamespace = str + info.size++ + } + } + return true + }) + return &info +} + +// Size returns the max number of attributes that defines a collector's +// instance information. Can be used to presize the attributes. +func (info CollectorInstanceInfo) Size() int { + return info.size +} + +func (info CollectorInstanceInfo) Copy(to pcommon.Map) { + to.EnsureCapacity(info.Size()) + if info.serviceInstanceID != "" { + to.PutStr(keyWithPrefix(semconv.AttributeServiceInstanceID), info.serviceInstanceID) + } + if info.serviceName != "" { + to.PutStr(keyWithPrefix(semconv.AttributeServiceName), info.serviceName) + } + if info.serviceNamespace != "" { + to.PutStr(keyWithPrefix(semconv.AttributeServiceNamespace), info.serviceNamespace) + } +} + +func keyWithPrefix(key string) string { + return prefix + "." + key +} diff --git a/connector/signaltometricsconnector/internal/model/collectorinstanceinfo_test.go b/connector/signaltometricsconnector/internal/model/collectorinstanceinfo_test.go new file mode 100644 index 000000000000..c36f6c3d8f7c --- /dev/null +++ b/connector/signaltometricsconnector/internal/model/collectorinstanceinfo_test.go @@ -0,0 +1,94 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package model + +import ( + "reflect" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/component/componenttest" + "go.opentelemetry.io/collector/pdata/pcommon" + semconv "go.opentelemetry.io/collector/semconv/v1.26.0" +) + +func TestCollectorInstanceInfo(t *testing.T) { + for _, tc := range []struct { + name string + input component.TelemetrySettings + expected pcommon.Map + }{ + { + name: "empty", + input: componenttest.NewNopTelemetrySettings(), + expected: pcommon.NewMap(), + }, + { + name: "with_service_instance_id", + input: func() component.TelemetrySettings { + ts := componenttest.NewNopTelemetrySettings() + ts.Resource.Attributes().PutStr(semconv.AttributeServiceInstanceID, "627cc493-f310-47de-96bd-71410b7dec09") + return ts + }(), + expected: func() pcommon.Map { + m := pcommon.NewMap() + m.PutStr( + "signaltometrics."+semconv.AttributeServiceInstanceID, + "627cc493-f310-47de-96bd-71410b7dec09", + ) + return m + }(), + }, + { + name: "with_all_values", + input: func() component.TelemetrySettings { + ts := componenttest.NewNopTelemetrySettings() + ts.Resource.Attributes().PutStr(semconv.AttributeServiceInstanceID, "627cc493-f310-47de-96bd-71410b7dec09") + ts.Resource.Attributes().PutStr(semconv.AttributeServiceName, "signaltometrics") + ts.Resource.Attributes().PutStr(semconv.AttributeServiceNamespace, "test") + return ts + }(), + expected: func() pcommon.Map { + m := pcommon.NewMap() + m.PutStr( + "signaltometrics."+semconv.AttributeServiceInstanceID, + "627cc493-f310-47de-96bd-71410b7dec09", + ) + m.PutStr( + "signaltometrics."+semconv.AttributeServiceName, + "signaltometrics", + ) + m.PutStr( + "signaltometrics."+semconv.AttributeServiceNamespace, + "test", + ) + return m + }(), + }, + } { + t.Run(tc.name, func(t *testing.T) { + ci := NewCollectorInstanceInfo(tc.input) + require.NotNil(t, ci) + + actual := pcommon.NewMap() + ci.Copy(actual) + assert.Equal(t, ci.Size(), actual.Len()) + assertMapEquality(t, tc.expected, actual) + }) + } +} + +func assertMapEquality(t *testing.T, expected, actual pcommon.Map) bool { + t.Helper() + + expectedRaw := expected.AsRaw() + actualRaw := actual.AsRaw() + return assert.True( + t, reflect.DeepEqual(expectedRaw, actualRaw), + "attributes don't match expected: %v, actual: %v", + expectedRaw, actualRaw, + ) +} diff --git a/connector/signaltometricsconnector/internal/model/model.go b/connector/signaltometricsconnector/internal/model/model.go index 16d05b129eb5..05ebdabf47f1 100644 --- a/connector/signaltometricsconnector/internal/model/model.go +++ b/connector/signaltometricsconnector/internal/model/model.go @@ -172,17 +172,19 @@ func (md *MetricDef[K]) FromMetricInfo( // definition. func (md *MetricDef[K]) FilterResourceAttributes( attrs pcommon.Map, + collectorInfo *CollectorInstanceInfo, ) pcommon.Map { var filteredAttributes pcommon.Map switch { case len(md.IncludeResourceAttributes) == 0: filteredAttributes = pcommon.NewMap() - filteredAttributes.EnsureCapacity(attrs.Len()) + filteredAttributes.EnsureCapacity(attrs.Len() + collectorInfo.Size()) attrs.CopyTo(filteredAttributes) default: - expectedLen := len(md.IncludeResourceAttributes) + expectedLen := len(md.IncludeResourceAttributes) + collectorInfo.Size() filteredAttributes = filterAttributes(attrs, md.IncludeResourceAttributes, expectedLen) } + collectorInfo.Copy(filteredAttributes) return filteredAttributes } diff --git a/connector/signaltometricsconnector/testdata/logs/exponential_histograms/output.yaml b/connector/signaltometricsconnector/testdata/logs/exponential_histograms/output.yaml index c0e6e450b560..732e026b85e3 100644 --- a/connector/signaltometricsconnector/testdata/logs/exponential_histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/logs/exponential_histograms/output.yaml @@ -7,6 +7,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Logrecords as exponential histogram with log.duration from attributes @@ -327,6 +336,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Logrecords with resource attribute foo as exponential histogram with log.duration from attributes diff --git a/connector/signaltometricsconnector/testdata/logs/histograms/output.yaml b/connector/signaltometricsconnector/testdata/logs/histograms/output.yaml index 45d0973f3eed..013c4ac1e8d0 100644 --- a/connector/signaltometricsconnector/testdata/logs/histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/logs/histograms/output.yaml @@ -7,6 +7,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Logrecords as histogram with log.duration from attributes @@ -127,6 +136,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Logrecords with resource attribute foo as histogram with log.duration from attributes diff --git a/connector/signaltometricsconnector/testdata/logs/sum/output.yaml b/connector/signaltometricsconnector/testdata/logs/sum/output.yaml index f575a5231e79..3fc35b338c63 100644 --- a/connector/signaltometricsconnector/testdata/logs/sum/output.yaml +++ b/connector/signaltometricsconnector/testdata/logs/sum/output.yaml @@ -7,6 +7,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Count total number of log records @@ -57,6 +66,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Count total number of log records with resource attribute foo diff --git a/connector/signaltometricsconnector/testdata/metrics/exponential_histograms/output.yaml b/connector/signaltometricsconnector/testdata/metrics/exponential_histograms/output.yaml index 42b210383c32..68ed74214770 100644 --- a/connector/signaltometricsconnector/testdata/metrics/exponential_histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/metrics/exponential_histograms/output.yaml @@ -4,6 +4,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: An exponential histogram created from gague values diff --git a/connector/signaltometricsconnector/testdata/metrics/histograms/output.yaml b/connector/signaltometricsconnector/testdata/metrics/histograms/output.yaml index 2cda97a02d15..fd42993a5cde 100644 --- a/connector/signaltometricsconnector/testdata/metrics/histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/metrics/histograms/output.yaml @@ -4,6 +4,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: A histogram created from gague values diff --git a/connector/signaltometricsconnector/testdata/metrics/sum/output.yaml b/connector/signaltometricsconnector/testdata/metrics/sum/output.yaml index ba8e962693e5..899ed0f6dcef 100644 --- a/connector/signaltometricsconnector/testdata/metrics/sum/output.yaml +++ b/connector/signaltometricsconnector/testdata/metrics/sum/output.yaml @@ -7,6 +7,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Count total number of datapoints diff --git a/connector/signaltometricsconnector/testdata/traces/exponential_histograms/output.yaml b/connector/signaltometricsconnector/testdata/traces/exponential_histograms/output.yaml index 7904d70e5f7c..fecd340313c6 100644 --- a/connector/signaltometricsconnector/testdata/traces/exponential_histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/traces/exponential_histograms/output.yaml @@ -4,6 +4,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Spans with resource attribute including resource.foo as a exponential histogram metric @@ -138,6 +147,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Spans with custom count OTTL expression as a exponential histogram metric diff --git a/connector/signaltometricsconnector/testdata/traces/histograms/output.yaml b/connector/signaltometricsconnector/testdata/traces/histograms/output.yaml index a25d144566c1..f4066729d9d8 100644 --- a/connector/signaltometricsconnector/testdata/traces/histograms/output.yaml +++ b/connector/signaltometricsconnector/testdata/traces/histograms/output.yaml @@ -4,6 +4,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Spans with resource attribute including resource.foo as a histogram metric @@ -60,6 +69,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Spans with custom count OTTL expression as a histogram metric diff --git a/connector/signaltometricsconnector/testdata/traces/sum/output.yaml b/connector/signaltometricsconnector/testdata/traces/sum/output.yaml index 5beb92e861f0..07acd1697635 100644 --- a/connector/signaltometricsconnector/testdata/traces/sum/output.yaml +++ b/connector/signaltometricsconnector/testdata/traces/sum/output.yaml @@ -4,6 +4,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Spans with resource attribute including resource.foo as a int sum metric @@ -24,6 +33,15 @@ resourceMetrics: - key: resource.foo value: stringValue: foo + - key: signaltometrics.service.instance.id + value: + stringValue: 627cc493-f310-47de-96bd-71410b7dec09 + - key: signaltometrics.service.name + value: + stringValue: signaltometrics + - key: signaltometrics.service.namespace + value: + stringValue: test scopeMetrics: - metrics: - description: Adjusted count for the span as a sum metric From 70877320553e906aa287a8e5e130196cffd7c111 Mon Sep 17 00:00:00 2001 From: Vishal Raj Date: Thu, 9 Jan 2025 15:24:44 +0000 Subject: [PATCH 2/2] Add changelog --- .chloggen/singlewriter-signaltometrics.yaml | 27 +++++++++++++++++++++ 1 file changed, 27 insertions(+) create mode 100644 .chloggen/singlewriter-signaltometrics.yaml diff --git a/.chloggen/singlewriter-signaltometrics.yaml b/.chloggen/singlewriter-signaltometrics.yaml new file mode 100644 index 000000000000..008b2e384bf6 --- /dev/null +++ b/.chloggen/singlewriter-signaltometrics.yaml @@ -0,0 +1,27 @@ +# Use this changelog template to create an entry for release notes. + +# One of 'breaking', 'deprecation', 'new_component', 'enhancement', 'bug_fix' +change_type: enhancement + +# The name of the component, or a single word describing the area of concern, (e.g. filelogreceiver) +component: signaltometrics + +# A brief description of the change. Surround your text with quotes ("") if it needs to start with a backtick (`). +note: Adds resource attributes based on telemetry settings to the connector to ensure single writer + +# Mandatory: One or more tracking issues related to the change. You can use the PR number here if no issue exists. +issues: [35930] + +# (Optional) One or more lines of additional information to render under the primary note. +# These lines will be padded with 2 spaces and then inserted directly into the document. +# Use pipe (|) for multiline entries. +subtext: + +# If your change doesn't affect end users or the exported elements of any package, +# you should instead start your pull request title with [chore] or use the "Skip Changelog" label. +# Optional: The change log or logs in which this entry should be included. +# e.g. '[user]' or '[user, api]' +# Include 'user' if the change is relevant to end users. +# Include 'api' if there is a change to a library API. +# Default: '[user]' +change_logs: [user]