From 97bee5df0dd73950a1e1ebaf88a15fbadb03aec1 Mon Sep 17 00:00:00 2001 From: Nathan Slaughter <28688390+nslaughter@users.noreply.github.com> Date: Thu, 6 Jun 2024 18:34:03 -0500 Subject: [PATCH 1/2] [azureeventhubreceiver] port PR --- .../components/azureeventhubreceiver/Makefile | 1 + .../azureeventhubreceiver/README.md | 125 ++++++ .../azureresourcelogs_unmarshaler.go | 36 ++ .../azureresourcelogs_unmarshaler_test.go | 118 ++++++ .../azureresourcemetrics_unmarshaler.go | 160 +++++++ .../azureeventhubreceiver/config.go | 57 +++ .../azureeventhubreceiver/config_test.go | 73 ++++ .../components/azureeventhubreceiver/doc.go | 7 + .../azureeventhubreceiver/eventhubhandler.go | 299 +++++++++++++ .../eventhubhandler_test.go | 117 +++++ .../azureeventhubreceiver/factory.go | 132 ++++++ .../azureeventhubreceiver/factory_test.go | 40 ++ .../generated_component_test.go | 81 ++++ .../components/azureeventhubreceiver/go.mod | 140 ++++++ .../components/azureeventhubreceiver/go.sum | 401 ++++++++++++++++++ .../internal/metadata/generated_status.go | 26 ++ .../azureeventhubreceiver/metadata.yaml | 13 + .../azureeventhubreceiver/persister.go | 103 +++++ .../azureeventhubreceiver/persister_test.go | 100 +++++ .../rawlogs_unmarshaler.go | 35 ++ .../azureeventhubreceiver/receiver.go | 161 +++++++ .../testdata/config.yaml | 22 + .../testdata/log-maximum.json | 31 ++ .../testdata/log-minimum-2.json | 16 + .../testdata/log-minimum.json | 10 + 25 files changed, 2304 insertions(+) create mode 100644 collector/components/azureeventhubreceiver/Makefile create mode 100644 collector/components/azureeventhubreceiver/README.md create mode 100644 collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler.go create mode 100644 collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler_test.go create mode 100644 collector/components/azureeventhubreceiver/azureresourcemetrics_unmarshaler.go create mode 100644 collector/components/azureeventhubreceiver/config.go create mode 100644 collector/components/azureeventhubreceiver/config_test.go create mode 100644 collector/components/azureeventhubreceiver/doc.go create mode 100644 collector/components/azureeventhubreceiver/eventhubhandler.go create mode 100644 collector/components/azureeventhubreceiver/eventhubhandler_test.go create mode 100644 collector/components/azureeventhubreceiver/factory.go create mode 100644 collector/components/azureeventhubreceiver/factory_test.go create mode 100644 collector/components/azureeventhubreceiver/generated_component_test.go create mode 100644 collector/components/azureeventhubreceiver/go.mod create mode 100644 collector/components/azureeventhubreceiver/go.sum create mode 100644 collector/components/azureeventhubreceiver/internal/metadata/generated_status.go create mode 100644 collector/components/azureeventhubreceiver/metadata.yaml create mode 100644 collector/components/azureeventhubreceiver/persister.go create mode 100644 collector/components/azureeventhubreceiver/persister_test.go create mode 100644 collector/components/azureeventhubreceiver/rawlogs_unmarshaler.go create mode 100644 collector/components/azureeventhubreceiver/receiver.go create mode 100644 collector/components/azureeventhubreceiver/testdata/config.yaml create mode 100644 collector/components/azureeventhubreceiver/testdata/log-maximum.json create mode 100644 collector/components/azureeventhubreceiver/testdata/log-minimum-2.json create mode 100644 collector/components/azureeventhubreceiver/testdata/log-minimum.json diff --git a/collector/components/azureeventhubreceiver/Makefile b/collector/components/azureeventhubreceiver/Makefile new file mode 100644 index 0000000..ded7a36 --- /dev/null +++ b/collector/components/azureeventhubreceiver/Makefile @@ -0,0 +1 @@ +include ../../Makefile.Common diff --git a/collector/components/azureeventhubreceiver/README.md b/collector/components/azureeventhubreceiver/README.md new file mode 100644 index 0000000..a8e1bfc --- /dev/null +++ b/collector/components/azureeventhubreceiver/README.md @@ -0,0 +1,125 @@ +# Azure Event Hub Receiver + + +| Status | | +| ------------- |-----------| +| Stability | [alpha]: metrics, logs | +| Distributions | [contrib], [observiq], [splunk], [sumo] | +| Issues | [![Open issues](https://img.shields.io/github/issues-search/open-telemetry/opentelemetry-collector-contrib?query=is%3Aissue%20is%3Aopen%20label%3Areceiver%2Fazureeventhub%20&label=open&color=orange&logo=opentelemetry)](https://github.com/open-telemetry/opentelemetry-collector-contrib/issues?q=is%3Aopen+is%3Aissue+label%3Areceiver%2Fazureeventhub) [![Closed issues](https://img.shields.io/github/issues-search/open-telemetry/opentelemetry-collector-contrib?query=is%3Aissue%20is%3Aclosed%20label%3Areceiver%2Fazureeventhub%20&label=closed&color=blue&logo=opentelemetry)](https://github.com/open-telemetry/opentelemetry-collector-contrib/issues?q=is%3Aclosed+is%3Aissue+label%3Areceiver%2Fazureeventhub) | +| [Code Owners](https://github.com/open-telemetry/opentelemetry-collector-contrib/blob/main/CONTRIBUTING.md#becoming-a-code-owner) | [@atoulme](https://www.github.com/atoulme), [@djaglowski](https://www.github.com/djaglowski) | + +[alpha]: https://github.com/open-telemetry/opentelemetry-collector#alpha +[contrib]: https://github.com/open-telemetry/opentelemetry-collector-releases/tree/main/distributions/otelcol-contrib +[observiq]: https://github.com/observIQ/observiq-otel-collector +[splunk]: https://github.com/signalfx/splunk-otel-collector +[sumo]: https://github.com/SumoLogic/sumologic-otel-collector + + +## Overview +Azure resources and services can be +[configured](https://learn.microsoft.com/en-us/azure/azure-monitor/essentials/diagnostic-settings) +to send their logs to an Azure Event Hub. The Azure Event Hub receiver pulls logs from an Azure +Event Hub, transforms them, and pushes them through the collector pipeline. + +## Configuration + +### connection (Required) +A string describing the connection to an Azure event hub. + +### group (Optional) +The Consumer Group to read from. If empty will default to the default Consumer Group $Default + +### partition (Optional) +The partition to watch. If empty, it will watch explicitly all partitions. + +Default: "" + +### offset (Optional) +The offset at which to start watching the event hub. If empty, it starts with the latest offset. + +Default: "" + +### format (Optional) +Determines how to transform the Event Hub messages into OpenTelemetry logs. See the "Format" +section below for details. + +Default: "azure" + +### Example Configuration + +```yaml +receivers: + azureeventhub: + connection: Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName + partition: foo + group: bar + offset: "1234-5566" + format: "azure" +``` + +This component can persist its state using the [storage extension]. + +## Format + +### raw + +The "raw" format maps the AMQP properties and data into the +attributes and body of an OpenTelemetry LogRecord, respectively. +The body is represented as a raw byte array. + +This format is not supported for Metrics. + +### azure + +The "azure" format extracts the Azure log records from the AMQP +message data, parses them, and maps the fields to OpenTelemetry +attributes. The table below summarizes the mapping between the +[Azure common log format](https://learn.microsoft.com/en-us/azure/azure-monitor/essentials/resource-logs-schema) +and the OpenTelemetry attributes. + + +| Azure | OpenTelemetry | +|----------------------------------|----------------------------------------| +| callerIpAddress (optional) | net.sock.peer.addr (attribute) | +| correlationId (optional) | azure.correlation.id (attribute) | +| category (optional) | azure.category (attribute) | +| durationMs (optional) | azure.duration (attribute) | +| Level (optional) | severity_number, severity_text (field) | +| location (optional) | cloud.region (attribute) | +| — | cloud.provider (attribute) | +| operationName (required) | azure.operation.name (attribute) | +| operationVersion (optional) | azure.operation.version (attribute) | +| properties (optional) | azure.properties (attribute, nested) | +| resourceId (required) | azure.resource.id (resource attribute) | +| resultDescription (optional) | azure.result.description (attribute) | +| resultSignature (optional) | azure.result.signature (attribute) | +| resultType (optional) | azure.result.type (attribute) | +| tenantId (required, tenant logs) | azure.tenant.id (attribute) | +| time or timeStamp (required) | time_unix_nano (time takes precedence) | +| identity (optional) | azure.identity (attribute, nested) | + +Notes: +* JSON does not distinguish between fixed and floating point numbers. All +JSON numbers are encoded as doubles. + +For Metrics the Azure Metric Records are an array +of "records" with the following fields. + +| Azure | Open Telemetry | +|------------|---------------------------------------------| +| time | time_unix_nano (field) | +| resourceId | azure.resource.id (resource attribute) | +| metricName | | +| timeGrain | start_time_unix_nano (field) | +| total | mapped to datapoint metricName + "_TOTAL" | +| count | mapped to datapoint metricName + "_COUNT" | +| minimum | mapped to datapoint metricName + "_MINIMUM" | +| maximum | mapped to datapoint metricName + "_MAXIMUM" | +| average | mapped to datapoint metricName + "_AVERAGE" | + +From this data a Metric of type Gauge is created +with a Data Points that represents the values +for the Metric including: Total, Minimum, Maximum, +Average and Count. + +[storage extension]: https://github.com/open-telemetry/opentelemetry-collector-contrib/tree/main/extension/storage diff --git a/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler.go b/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler.go new file mode 100644 index 0000000..01073fc --- /dev/null +++ b/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler.go @@ -0,0 +1,36 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/pdata/plog" + "go.uber.org/zap" + + "github.com/open-telemetry/opentelemetry-collector-contrib/pkg/translator/azure" +) + +type AzureResourceLogsEventUnmarshaler struct { + unmarshaler *azure.ResourceLogsUnmarshaler +} + +func newAzureResourceLogsUnmarshaler(buildInfo component.BuildInfo, logger *zap.Logger) eventLogsUnmarshaler { + return AzureResourceLogsEventUnmarshaler{ + unmarshaler: &azure.ResourceLogsUnmarshaler{ + Version: buildInfo.Version, + Logger: logger, + }, + } +} + +// UnmarshalLogs takes a byte array containing a JSON-encoded +// payload with Azure log records and transforms it into +// an OpenTelemetry plog.Logs object. The data in the Azure +// log record appears as fields and attributes in the +// OpenTelemetry representation; the bodies of the +// OpenTelemetry log records are empty. +func (r AzureResourceLogsEventUnmarshaler) UnmarshalLogs(event *azeventhubs.ReceivedEventData) (plog.Logs, error) { + return r.unmarshaler.UnmarshalLogs(event.Body) +} diff --git a/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler_test.go b/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler_test.go new file mode 100644 index 0000000..82a1f2a --- /dev/null +++ b/collector/components/azureeventhubreceiver/azureresourcelogs_unmarshaler_test.go @@ -0,0 +1,118 @@ +// Code generated by mdatagen. DO NOT EDIT. + +package azureeventhubreceiver + +import ( + "testing" + "time" + + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + "github.com/stretchr/testify/require" + "go.uber.org/zap" +) + +// TestUnmarshalLogs_Body should succeed regardless of body content type +func TestUnmarshalLogs_Body(t *testing.T) { + logger := zap.NewNop() + unmarshaler := newRawLogsUnmarshaler(logger) + + testCases := []struct { + name string + body []byte + expect []byte + }{ + { + name: "empty body", + body: []byte(""), + // note that zero length body sets otlp Body to []byte(nil) not []byte{} + expect: []byte(nil), + }, + { + name: "nil body", + body: []byte(nil), + expect: []byte(nil), + }, + { + name: "invalid json", + body: []byte("{malformed-json"), + expect: []byte("{malformed-json"), + }, + { + name: "valid json", + body: []byte(`{"key": "value"}`), + expect: []byte(`{"key": "value"}`), + }, + } + + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + event := &azeventhubs.ReceivedEventData{ + EventData: azeventhubs.EventData{ + Body: tc.body, + Properties: map[string]interface{}{"someKey": "someValue"}, + }, + } + logs, err := unmarshaler.UnmarshalLogs(event) + require.NoError(t, err, "Expected no error for valid event data but got one") + require.Equal(t, 1, logs.ResourceLogs().Len(), "Expected 1 ResourceLog") + otlpBody := logs.ResourceLogs().At(0).ScopeLogs().At(0).LogRecords().At(0).Body().Bytes().AsRaw() + require.Equal(t, otlpBody, tc.expect) + }) + } +} + +func TestUnmarshalLogs_Attributes(t *testing.T) { + logger := zap.NewNop() + unmarshaler := newRawLogsUnmarshaler(logger) + + testCases := []struct { + name string + properties map[string]any + expect map[string]any + }{ + { + name: "empty properties", + properties: map[string]any{}, + expect: map[string]any(nil), + }, + { + name: "nil properties", + properties: map[string]any(nil), + expect: map[string]any(nil), + }, + { + name: "single property", + properties: map[string]interface{}{"someKey": "someValue"}, + expect: map[string]interface{}{"someKey": "someValue"}, + }, + { + name: "multiple properties", + properties: map[string]interface{}{"someKey": "someValue", "anotherKey": "anotherValue"}, + expect: map[string]interface{}{"someKey": "someValue", "anotherKey": "anotherValue"}, + }, + } + + var et = time.Now() + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + event := &azeventhubs.ReceivedEventData{ + EnqueuedTime: &et, + SystemProperties: map[string]interface{}{"syskey1": "sysval1", "syskey2": "sysval2"}, + EventData: azeventhubs.EventData{ + Body: []byte(""), + Properties: tc.properties, + }, + } + logs, err := unmarshaler.UnmarshalLogs(event) + require.NoError(t, err, "Expected no error for valid event data but got one") + resAttrs := logs.ResourceLogs().At(0).ScopeLogs().At(0).LogRecords().At(0).Attributes().AsRaw() + // this is because + require.Equal(t, len(resAttrs), len(tc.expect), "Expected %d attributes, got %d", len(tc.expect), len(resAttrs)) + if len(resAttrs) == 0 { + return + } + + require.Equal(t, resAttrs, tc.expect, "Not equal:\nexpected: %v\nactual: %v", tc.expect, resAttrs) + }) + } +} diff --git a/collector/components/azureeventhubreceiver/azureresourcemetrics_unmarshaler.go b/collector/components/azureeventhubreceiver/azureresourcemetrics_unmarshaler.go new file mode 100644 index 0000000..2e739d1 --- /dev/null +++ b/collector/components/azureeventhubreceiver/azureresourcemetrics_unmarshaler.go @@ -0,0 +1,160 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "bytes" + "fmt" + "strings" + "time" + + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + jsoniter "github.com/json-iterator/go" + "github.com/relvacode/iso8601" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/pdata/pcommon" + "go.opentelemetry.io/collector/pdata/pmetric" + conventions "go.opentelemetry.io/collector/semconv/v1.13.0" + "go.uber.org/zap" +) + +const ( + azureResourceID = "azure.resource.id" + scopeName = "otelcol/azureresourcemetrics" +) + +type azureResourceMetricsUnmarshaler struct { + buildInfo component.BuildInfo + logger *zap.Logger +} + +// azureMetricRecords represents an array of Azure metric records +// as exported via an Azure Event Hub +type azureMetricRecords struct { + Records []azureMetricRecord `json:"records"` +} + +// azureMetricRecord represents a single Azure Metric following +// the common schema does not exist (yet): +type azureMetricRecord struct { + Time string `json:"time"` + ResourceID string `json:"resourceId"` + MetricName string `json:"metricName"` + TimeGrain string `json:"timeGrain"` + Total float64 `json:"total"` + Count float64 `json:"count"` + Minimum float64 `json:"minimum"` + Maximum float64 `json:"maximum"` + Average float64 `json:"average"` +} + +func newAzureResourceMetricsUnmarshaler(buildInfo component.BuildInfo, logger *zap.Logger) azureResourceMetricsUnmarshaler { + return azureResourceMetricsUnmarshaler{ + buildInfo: buildInfo, + logger: logger, + } +} + +// UnmarshalMetrics takes a byte array containing a JSON-encoded +// payload with Azure metric records and transforms it into +// an OpenTelemetry pmetric.Metrics object. The data in the Azure +// metric record appears as fields and attributes in the +// OpenTelemetry representation; +func (r azureResourceMetricsUnmarshaler) UnmarshalMetrics(event *azeventhubs.ReceivedEventData) (pmetric.Metrics, error) { + + md := pmetric.NewMetrics() + + var azureMetrics azureMetricRecords + decoder := jsoniter.NewDecoder(bytes.NewReader(event.EventData.Body)) + err := decoder.Decode(&azureMetrics) + if err != nil { + return md, err + } + + resourceMetrics := md.ResourceMetrics().AppendEmpty() + resource := resourceMetrics.Resource() + resource.Attributes().PutStr(conventions.AttributeTelemetrySDKName, receiverScopeName) + resource.Attributes().PutStr(conventions.AttributeTelemetrySDKLanguage, conventions.AttributeTelemetrySDKLanguageGo) + resource.Attributes().PutStr(conventions.AttributeTelemetrySDKVersion, r.buildInfo.Version) + resource.Attributes().PutStr(conventions.AttributeCloudProvider, conventions.AttributeCloudProviderAzure) + + scopeMetrics := resourceMetrics.ScopeMetrics().AppendEmpty() + + metrics := scopeMetrics.Metrics() + metrics.EnsureCapacity(len(azureMetrics.Records) * 5) + + resourceID := "" + for _, azureMetric := range azureMetrics.Records { + if resourceID == "" && azureMetric.ResourceID != "" { + resourceID = azureMetric.ResourceID + } + + nanos, err := asTimestamp(azureMetric.Time) + if err != nil { + r.logger.Warn("Invalid Timestamp", zap.String("time", azureMetric.Time)) + continue + } + + var startTimestamp pcommon.Timestamp + if azureMetric.TimeGrain == "PT1M" { + startTimestamp = pcommon.NewTimestampFromTime(nanos.AsTime().Add(-time.Minute)) + } else { + r.logger.Warn("Unhandled Time Grain", zap.String("timegrain", azureMetric.TimeGrain)) + continue + } + + metricTotal := metrics.AppendEmpty() + metricTotal.SetName(strings.ToLower(fmt.Sprintf("%s_%s", strings.ReplaceAll(azureMetric.MetricName, " ", "_"), "Total"))) + dpTotal := metricTotal.SetEmptyGauge().DataPoints().AppendEmpty() + dpTotal.SetStartTimestamp(startTimestamp) + dpTotal.SetTimestamp(nanos) + dpTotal.SetDoubleValue(azureMetric.Total) + + metricCount := metrics.AppendEmpty() + metricCount.SetName(strings.ToLower(fmt.Sprintf("%s_%s", strings.ReplaceAll(azureMetric.MetricName, " ", "_"), "Count"))) + dpCount := metricCount.SetEmptyGauge().DataPoints().AppendEmpty() + dpCount.SetStartTimestamp(startTimestamp) + dpCount.SetTimestamp(nanos) + dpCount.SetDoubleValue(azureMetric.Count) + + metricMin := metrics.AppendEmpty() + metricMin.SetName(strings.ToLower(fmt.Sprintf("%s_%s", strings.ReplaceAll(azureMetric.MetricName, " ", "_"), "Minimum"))) + dpMin := metricMin.SetEmptyGauge().DataPoints().AppendEmpty() + dpMin.SetStartTimestamp(startTimestamp) + dpMin.SetTimestamp(nanos) + dpMin.SetDoubleValue(azureMetric.Minimum) + + metricMax := metrics.AppendEmpty() + metricMax.SetName(strings.ToLower(fmt.Sprintf("%s_%s", strings.ReplaceAll(azureMetric.MetricName, " ", "_"), "Maximum"))) + dpMax := metricMax.SetEmptyGauge().DataPoints().AppendEmpty() + dpMax.SetStartTimestamp(startTimestamp) + dpMax.SetTimestamp(nanos) + dpMax.SetDoubleValue(azureMetric.Maximum) + + metricAverage := metrics.AppendEmpty() + metricAverage.SetName(strings.ToLower(fmt.Sprintf("%s_%s", strings.ReplaceAll(azureMetric.MetricName, " ", "_"), "Average"))) + dpAverage := metricAverage.SetEmptyGauge().DataPoints().AppendEmpty() + dpAverage.SetStartTimestamp(startTimestamp) + dpAverage.SetTimestamp(nanos) + dpAverage.SetDoubleValue(azureMetric.Average) + } + + if resourceID != "" { + resourceMetrics.Resource().Attributes().PutStr(azureResourceID, resourceID) + } else { + r.logger.Warn("No ResourceID Set on Metrics!") + } + + return md, nil +} + +// asTimestamp will parse an ISO8601 string into an OpenTelemetry nanosecond timestamp. +// If the string cannot be parsed, it will return zero and the error. +func asTimestamp(s string) (pcommon.Timestamp, error) { + t, err := iso8601.ParseString(s) + if err != nil { + return 0, err + } + return pcommon.Timestamp(t.UnixNano()), nil +} diff --git a/collector/components/azureeventhubreceiver/config.go b/collector/components/azureeventhubreceiver/config.go new file mode 100644 index 0000000..9ccb992 --- /dev/null +++ b/collector/components/azureeventhubreceiver/config.go @@ -0,0 +1,57 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "errors" + "fmt" + + "github.com/Azure/azure-amqp-common-go/v4/conn" + "go.opentelemetry.io/collector/component" +) + +type logFormat string + +const ( + defaultLogFormat logFormat = "" + rawLogFormat logFormat = "raw" + azureLogFormat logFormat = "azure" +) + +var ( + validFormats = []logFormat{defaultLogFormat, rawLogFormat, azureLogFormat} + errMissingConnection = errors.New("missing connection") +) + +type Config struct { + Connection string `mapstructure:"connection"` + Partition string `mapstructure:"partition"` + Offset string `mapstructure:"offset"` + StorageID *component.ID `mapstructure:"storage"` + Format string `mapstructure:"format"` + ConsumerGroup string `mapstructure:"group"` +} + +func isValidFormat(format string) bool { + for _, validFormat := range validFormats { + if logFormat(format) == validFormat { + return true + } + } + return false +} + +// Validate config +func (config *Config) Validate() error { + if config.Connection == "" { + return errMissingConnection + } + if _, err := conn.ParsedConnectionFromStr(config.Connection); err != nil { + return err + } + if !isValidFormat(config.Format) { + return fmt.Errorf("invalid format; must be one of %#v", validFormats) + } + return nil +} diff --git a/collector/components/azureeventhubreceiver/config_test.go b/collector/components/azureeventhubreceiver/config_test.go new file mode 100644 index 0000000..7d4dcf4 --- /dev/null +++ b/collector/components/azureeventhubreceiver/config_test.go @@ -0,0 +1,73 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "path/filepath" + "testing" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/otelcol/otelcoltest" + + "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver/internal/metadata" +) + +func TestLoadConfig(t *testing.T) { + factories, err := otelcoltest.NopFactories() + assert.NoError(t, err) + + factory := NewFactory() + factories.Receivers[metadata.Type] = factory + cfg, err := otelcoltest.LoadConfigAndValidate(filepath.Join("testdata", "config.yaml"), factories) + + require.NoError(t, err) + require.NotNil(t, cfg) + + assert.Equal(t, len(cfg.Receivers), 2) + + r0 := cfg.Receivers[component.NewID(metadata.Type)] + assert.Equal(t, "Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName", r0.(*Config).Connection) + assert.Equal(t, "", r0.(*Config).Offset) + assert.Equal(t, "", r0.(*Config).Partition) + assert.Equal(t, defaultLogFormat, logFormat(r0.(*Config).Format)) + + r1 := cfg.Receivers[component.NewIDWithName(metadata.Type, "all")] + assert.Equal(t, "Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName", r1.(*Config).Connection) + assert.Equal(t, "1234-5566", r1.(*Config).Offset) + assert.Equal(t, "foo", r1.(*Config).Partition) + assert.Equal(t, rawLogFormat, logFormat(r1.(*Config).Format)) +} + +func TestMissingConnection(t *testing.T) { + factory := NewFactory() + cfg := factory.CreateDefaultConfig() + err := component.ValidateConfig(cfg) + assert.EqualError(t, err, "missing connection") +} + +func TestInvalidConnectionString(t *testing.T) { + factory := NewFactory() + cfg := factory.CreateDefaultConfig() + cfg.(*Config).Connection = "foo" + err := component.ValidateConfig(cfg) + assert.EqualError(t, err, "failed parsing connection string due to unmatched key value separated by '='") +} + +func TestIsValidFormat(t *testing.T) { + for _, format := range []logFormat{defaultLogFormat, rawLogFormat, azureLogFormat} { + assert.True(t, isValidFormat(string(format))) + } + assert.False(t, isValidFormat("invalid-format")) +} + +func TestInvalidFormat(t *testing.T) { + factory := NewFactory() + cfg := factory.CreateDefaultConfig() + cfg.(*Config).Connection = "Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName" + cfg.(*Config).Format = "invalid" + err := component.ValidateConfig(cfg) + assert.ErrorContains(t, err, "invalid format; must be one of") +} diff --git a/collector/components/azureeventhubreceiver/doc.go b/collector/components/azureeventhubreceiver/doc.go new file mode 100644 index 0000000..c00414c --- /dev/null +++ b/collector/components/azureeventhubreceiver/doc.go @@ -0,0 +1,7 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +//go:generate mdatagen metadata.yaml + +// Package azureeventhubreceiver listens to logs emitted by Azure Event hubs. +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" diff --git a/collector/components/azureeventhubreceiver/eventhubhandler.go b/collector/components/azureeventhubreceiver/eventhubhandler.go new file mode 100644 index 0000000..786ed08 --- /dev/null +++ b/collector/components/azureeventhubreceiver/eventhubhandler.go @@ -0,0 +1,299 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "errors" + "strings" + "sync" + "time" + + "github.com/Azure/azure-sdk-for-go/sdk/azcore/to" + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/receiver" + "go.uber.org/zap" +) + +const ( + batchCount = 100 +) + +type eventHandler interface { + run(ctx context.Context, host component.Host) error + close(ctx context.Context) error + setDataConsumer(dataConsumer dataConsumer) +} + +type consumerClientWrapper interface { + GetEventHubProperties(ctx context.Context, options *azeventhubs.GetEventHubPropertiesOptions) (azeventhubs.EventHubProperties, error) + GetPartitionProperties(ctx context.Context, partitionID string, options *azeventhubs.GetPartitionPropertiesOptions) (azeventhubs.PartitionProperties, error) + NewConsumer(ctx context.Context, options *azeventhubs.ConsumerClientOptions) (*azeventhubs.ConsumerClient, error) + NewPartitionClient(partitionID string, options *azeventhubs.PartitionClientOptions) (*azeventhubs.PartitionClient, error) + Close(ctx context.Context) error +} + +type consumerClientWrapperImpl struct { + consumerClient *azeventhubs.ConsumerClient +} + +func newConsumerClientWrapperImplementation(cfg *Config) (*consumerClientWrapperImpl, error) { + splits := strings.Split(cfg.Connection, "/") + eventhubName := splits[len(splits)-1] + // if that didn't work it's ok as the SDK will try to parse it to create the client + + consumerClient, err := azeventhubs.NewConsumerClientFromConnectionString(cfg.Connection, eventhubName, cfg.ConsumerGroup, nil) + if err != nil { + return nil, err + } + return &consumerClientWrapperImpl{ + consumerClient: consumerClient, + }, nil +} + +func (c *consumerClientWrapperImpl) GetEventHubProperties(ctx context.Context, options *azeventhubs.GetEventHubPropertiesOptions) (azeventhubs.EventHubProperties, error) { + return c.consumerClient.GetEventHubProperties(ctx, options) +} + +func (c *consumerClientWrapperImpl) GetPartitionProperties(ctx context.Context, partitionID string, options *azeventhubs.GetPartitionPropertiesOptions) (azeventhubs.PartitionProperties, error) { + return c.consumerClient.GetPartitionProperties(ctx, partitionID, options) +} + +func (c *consumerClientWrapperImpl) NewConsumer(_ context.Context, _ *azeventhubs.ConsumerClientOptions) (*azeventhubs.ConsumerClient, error) { + return c.consumerClient, nil +} + +func (c *consumerClientWrapperImpl) NewPartitionClient(partitionID string, options *azeventhubs.PartitionClientOptions) (*azeventhubs.PartitionClient, error) { + return c.consumerClient.NewPartitionClient(partitionID, options) +} + +func (c *consumerClientWrapperImpl) Close(ctx context.Context) error { + return c.consumerClient.Close(ctx) +} + +type eventhubHandler struct { + consumerClient consumerClientWrapper + dataConsumer dataConsumer + config *Config + settings receiver.CreateSettings + cancel context.CancelFunc + useProcessor bool +} + +var _ eventHandler = (*eventhubHandler)(nil) + +// newEventhubHandler creates a handler for Azure Event Hub. This version is enhanced to handle mock configurations for testing. +func newEventhubHandler(config *Config, settings receiver.CreateSettings) *eventhubHandler { + // Check if the configuration is meant for testing. This can be done by checking a specific field or a pattern in the connection string. + if strings.Contains(config.Connection, "fake.servicebus.windows.net") { + return nil + } + + return &eventhubHandler{ + config: config, + settings: settings, + useProcessor: true, + } +} + +func (h *eventhubHandler) init(ctx context.Context) error { + _, h.cancel = context.WithCancel(ctx) + consumerClient, err := newConsumerClientWrapperImplementation(h.config) + if err != nil { + return err + } + h.consumerClient = consumerClient + return nil +} + +func (h *eventhubHandler) run(ctx context.Context, host component.Host) error { + ctx, h.cancel = context.WithCancel(ctx) + if h.useProcessor { + return h.runWithProcessor(ctx, host) + } + return h.runWithConsumerClient(ctx, host) +} +func (h *eventhubHandler) runWithProcessor(ctx context.Context, host component.Host) error { + checkpointStore, err := createCheckpointStore(ctx, host, h.config, h.settings) + if err != nil { + h.settings.Logger.Debug("Error creating CheckpointStore", zap.Error(err)) + return err + } + + consumerClientImpl, ok := h.consumerClient.(*consumerClientWrapperImpl) + if !ok { + // we're in a testing environment + return nil + } + + processor, err := azeventhubs.NewProcessor(consumerClientImpl.consumerClient, checkpointStore, nil) + if err != nil { + h.settings.Logger.Debug("Error creating Processor", zap.Error(err)) + return err + } + + processorCtx, processorCancel := context.WithCancel(ctx) + go h.dispatchPartitionClients(processor) + defer processorCancel() + + return processor.Run(processorCtx) +} + +func (h *eventhubHandler) dispatchPartitionClients(processor *azeventhubs.Processor) { + var wg sync.WaitGroup + for { + partitionClient := processor.NextPartitionClient(context.TODO()) + + if partitionClient == nil { + break + } + + wg.Add(1) + go func(pc *azeventhubs.ProcessorPartitionClient) { + defer wg.Done() + if err := h.processEventsForPartition(pc); err != nil { + h.settings.Logger.Error("Error processing partition", zap.Error(err)) + } + }(partitionClient) + } + wg.Wait() +} + +func (h *eventhubHandler) processEventsForPartition(partitionClient *azeventhubs.ProcessorPartitionClient) error { + defer partitionClient.Close(context.TODO()) + + for { + receiveCtx, cancelReceive := context.WithTimeout(context.TODO(), time.Minute) + events, err := partitionClient.ReceiveEvents(receiveCtx, 100, nil) + cancelReceive() + + if err != nil && !errors.Is(err, context.DeadlineExceeded) { + var eventHubError *azeventhubs.Error + if errors.As(err, &eventHubError) && eventHubError.Code == azeventhubs.ErrorCodeOwnershipLost { + return nil + } + return err + } + + if len(events) == 0 { + continue + } + + for _, event := range events { + if err := h.newMessageHandler(context.TODO(), event); err != nil { + h.settings.Logger.Error("Error handling event", zap.Error(err)) + } + } + + if err := partitionClient.UpdateCheckpoint(context.TODO(), events[len(events)-1], nil); err != nil { + h.settings.Logger.Error("Error updating checkpoint", zap.Error(err)) + } + } +} + +func (h *eventhubHandler) runWithConsumerClient(ctx context.Context, _ component.Host) error { + if h.consumerClient == nil { + if err := h.init(ctx); err != nil { + return err + } + } + if h.config.Partition == "" { + properties, err := h.consumerClient.GetEventHubProperties(ctx, nil) + if err != nil { + h.settings.Logger.Debug("Error getting Event Hub properties", zap.Error(err)) + return err + } + + for _, partitionID := range properties.PartitionIDs { + err = h.setupPartition(ctx, partitionID) + if err != nil { + h.settings.Logger.Debug("Error setting up partition", zap.Error(err)) + return err + } + } + } else { + err := h.setupPartition(ctx, h.config.Partition) + if err != nil { + h.settings.Logger.Debug("Error setting up partition", zap.Error(err)) + return err + } + } + return nil +} + +func (h *eventhubHandler) setupPartition(ctx context.Context, partitionID string) error { + cc, err := h.consumerClient.NewConsumer(ctx, nil) + if err != nil { + return err + } + if cc == nil { + return errors.New("failed to initialize consumer client") + } + defer cc.Close(ctx) + + pcOpts := &azeventhubs.PartitionClientOptions{ + StartPosition: azeventhubs.StartPosition{ + Earliest: to.Ptr(true), + }, + } + + pc, err := cc.NewPartitionClient(partitionID, pcOpts) + if err != nil { + return err + } + if pc == nil { + return errors.New("failed to initialize partition client") + } + defer func() { + if pc != nil { + pc.Close(ctx) + } + }() + + go h.receivePartitionEvents(ctx, pc) + + return nil +} + +func (h *eventhubHandler) receivePartitionEvents(ctx context.Context, pc *azeventhubs.PartitionClient) { + for { + rcvCtx, rcvCtxCancel := context.WithTimeout(context.TODO(), time.Second*10) + events, err := pc.ReceiveEvents(rcvCtx, batchCount, nil) + rcvCtxCancel() + if err != nil && !errors.Is(err, context.DeadlineExceeded) { + h.settings.Logger.Error("Error receiving events", zap.Error(err)) + } + + for _, event := range events { + if err := h.newMessageHandler(ctx, event); err != nil { + h.settings.Logger.Error("Error handling event", zap.Error(err)) + } + } + } +} + +func (h *eventhubHandler) newMessageHandler(ctx context.Context, event *azeventhubs.ReceivedEventData) error { + err := h.dataConsumer.consume(ctx, event) + if err != nil { + h.settings.Logger.Error("Error decoding message", zap.Error(err)) + return err + } + return nil +} + +func (h *eventhubHandler) close(ctx context.Context) error { + if h.consumerClient != nil { + err := h.consumerClient.Close(ctx) + if err != nil { + return err + } + h.consumerClient = nil + } + return nil +} + +func (h *eventhubHandler) setDataConsumer(dataConsumer dataConsumer) { + h.dataConsumer = dataConsumer +} diff --git a/collector/components/azureeventhubreceiver/eventhubhandler_test.go b/collector/components/azureeventhubreceiver/eventhubhandler_test.go new file mode 100644 index 0000000..8bf74ec --- /dev/null +++ b/collector/components/azureeventhubreceiver/eventhubhandler_test.go @@ -0,0 +1,117 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "testing" + "time" + + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/mock" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/component/componenttest" + "go.opentelemetry.io/collector/consumer" + "go.opentelemetry.io/collector/receiver" + "go.uber.org/zap" +) + +type MockConsumerClientWrapper struct { + mock.Mock +} + +func (m *MockConsumerClientWrapper) GetEventHubProperties(ctx context.Context, options *azeventhubs.GetEventHubPropertiesOptions) (azeventhubs.EventHubProperties, error) { + args := m.Called(ctx, options) + return args.Get(0).(azeventhubs.EventHubProperties), args.Error(1) +} + +func (m *MockConsumerClientWrapper) GetPartitionProperties(ctx context.Context, partitionID string, options *azeventhubs.GetPartitionPropertiesOptions) (azeventhubs.PartitionProperties, error) { + args := m.Called(ctx, partitionID, options) + return args.Get(0).(azeventhubs.PartitionProperties), args.Error(1) +} + +func (m *MockConsumerClientWrapper) NewConsumer(ctx context.Context, options *azeventhubs.ConsumerClientOptions) (*azeventhubs.ConsumerClient, error) { + args := m.Called(ctx, options) + return args.Get(0).(*azeventhubs.ConsumerClient), args.Error(1) +} + +func (m *MockConsumerClientWrapper) NewPartitionClient(partitionID string, options *azeventhubs.PartitionClientOptions) (*azeventhubs.PartitionClient, error) { + args := m.Called(partitionID, options) + return args.Get(0).(*azeventhubs.PartitionClient), args.Error(1) +} + +func (m *MockConsumerClientWrapper) Close(ctx context.Context) error { + args := m.Called(ctx) + return args.Error(1) +} + +func TestEventHubHandler_Start(t *testing.T) { + logger := zap.NewNop() + settings := receiver.CreateSettings{ + TelemetrySettings: component.TelemetrySettings{Logger: logger}, + } + config := &Config{Connection: "Endpoint=sb://namespace.servicebus.windows.net/;EntityPath=hubName", ConsumerGroup: "$Default"} + + mockConsumerClient := new(MockConsumerClientWrapper) + + handler := newEventhubHandler(config, settings) + handler.consumerClient = mockConsumerClient + + host := componenttest.NewNopHost() + ctx, cancel := context.WithCancel(context.Background()) + defer cancel() + + err := handler.run(ctx, host) + assert.NoError(t, err) + + mockConsumerClient.AssertExpectations(t) +} + +func TestEventHubHandler_HandleEvent(t *testing.T) { + logger := zap.NewNop() + settings := receiver.CreateSettings{ + TelemetrySettings: component.TelemetrySettings{Logger: logger}, + } + config := &Config{Connection: "Endpoint=sb://namespace.servicebus.windows.net/;EntityPath=hubName", ConsumerGroup: "$Default"} + + mockConsumerClient := new(MockConsumerClientWrapper) + event := &azeventhubs.ReceivedEventData{ + EventData: azeventhubs.EventData{ + Body: []byte(`{"message":"test"}`), + }, + } + + mockDataConsumer := new(MockDataConsumer) + handler := newEventhubHandler(config, settings) + handler.consumerClient = mockConsumerClient + handler.dataConsumer = mockDataConsumer + + ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) + defer cancel() + + // expect + mockDataConsumer.On("consume", mock.Anything, mock.Anything).Return(nil) + + // act + err := handler.newMessageHandler(ctx, event) + assert.NoError(t, err) +} + +type MockDataConsumer struct { + mock.Mock +} + +func (m *MockDataConsumer) consume(ctx context.Context, event *azeventhubs.ReceivedEventData) error { + args := m.Called(ctx, event) + return args.Error(0) +} + +func (m *MockDataConsumer) setNextLogsConsumer(c consumer.Logs) { + _ = m.Called(c) +} + +func (m *MockDataConsumer) setNextMetricsConsumer(c consumer.Metrics) { + _ = m.Called(c) +} diff --git a/collector/components/azureeventhubreceiver/factory.go b/collector/components/azureeventhubreceiver/factory.go new file mode 100644 index 0000000..a9d0bfe --- /dev/null +++ b/collector/components/azureeventhubreceiver/factory.go @@ -0,0 +1,132 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "errors" + + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/consumer" + "go.opentelemetry.io/collector/receiver" + + "github.com/open-telemetry/opentelemetry-collector-contrib/internal/sharedcomponent" + "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver/internal/metadata" +) + +var ( + // The receiver scope name + receiverScopeName = "otelcol/" + metadata.Type.String() + "receiver" +) + +var ( + errUnexpectedConfigurationType = errors.New("failed to cast configuration to azure event hub config") +) + +type eventhubReceiverFactory struct { + receivers *sharedcomponent.SharedComponents +} + +// NewFactory creates a factory for the Azure Event Hub receiver. +func NewFactory() receiver.Factory { + f := &eventhubReceiverFactory{ + receivers: sharedcomponent.NewSharedComponents(), + } + + return receiver.NewFactory( + metadata.Type, + createDefaultConfig, + receiver.WithLogs(f.createLogsReceiver, metadata.LogsStability), + receiver.WithMetrics(f.createMetricsReceiver, metadata.MetricsStability)) +} + +func createDefaultConfig() component.Config { + return &Config{} +} + +func (f *eventhubReceiverFactory) createLogsReceiver( + _ context.Context, + settings receiver.CreateSettings, + cfg component.Config, + nextConsumer consumer.Logs, +) (receiver.Logs, error) { + + receiver, err := f.getReceiver(component.DataTypeLogs, cfg, settings) + if err != nil { + return nil, err + } + + receiver.(dataConsumer).setNextLogsConsumer(nextConsumer) + + return receiver, nil +} + +func (f *eventhubReceiverFactory) createMetricsReceiver( + _ context.Context, + settings receiver.CreateSettings, + cfg component.Config, + nextConsumer consumer.Metrics, +) (receiver.Metrics, error) { + + receiver, err := f.getReceiver(component.DataTypeMetrics, cfg, settings) + if err != nil { + return nil, err + } + + receiver.(dataConsumer).setNextMetricsConsumer(nextConsumer) + + return receiver, nil +} + +func (f *eventhubReceiverFactory) getReceiver( + receiverType component.Type, + cfg component.Config, + settings receiver.CreateSettings, +) (component.Component, error) { + + var err error + r := f.receivers.GetOrAdd(cfg, func() component.Component { + receiverConfig, ok := cfg.(*Config) + if !ok { + err = errUnexpectedConfigurationType + return nil + } + + var logsUnmarshaler eventLogsUnmarshaler + var metricsUnmarshaler eventMetricsUnmarshaler + switch receiverType { + case component.DataTypeLogs: + if logFormat(receiverConfig.Format) == rawLogFormat { + logsUnmarshaler = newRawLogsUnmarshaler(settings.Logger) + } else { + logsUnmarshaler = newAzureResourceLogsUnmarshaler(settings.BuildInfo, settings.Logger) + } + case component.DataTypeMetrics: + if logFormat(receiverConfig.Format) == rawLogFormat { + metricsUnmarshaler = nil + err = errors.New("raw format not supported for Metrics") + } else { + metricsUnmarshaler = newAzureResourceMetricsUnmarshaler(settings.BuildInfo, settings.Logger) + } + case component.DataTypeTraces: + err = errors.New("unsupported traces data") + } + + if err != nil { + return nil + } + + eventHandler := newEventhubHandler(receiverConfig, settings) + + var receiver component.Component + receiver, err = newReceiver(receiverType, logsUnmarshaler, metricsUnmarshaler, eventHandler, settings) + return receiver + }) + + if err != nil { + return nil, err + } + + return r.Unwrap(), err +} diff --git a/collector/components/azureeventhubreceiver/factory_test.go b/collector/components/azureeventhubreceiver/factory_test.go new file mode 100644 index 0000000..a809213 --- /dev/null +++ b/collector/components/azureeventhubreceiver/factory_test.go @@ -0,0 +1,40 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com.open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "testing" + + "github.com/stretchr/testify/assert" + "go.opentelemetry.io/collector/consumer/consumertest" + "go.opentelemetry.io/collector/receiver/receivertest" + + "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver/internal/metadata" +) + +func Test_NewFactory(t *testing.T) { + f := NewFactory() + assert.Equal(t, metadata.Type, f.Type()) +} + +func Test_NewLogsReceiver(t *testing.T) { + f := NewFactory() + config := createDefaultConfig().(*Config) + config.Connection = "Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234;EntityPath=hubName" + + receiver, err := f.CreateLogsReceiver(context.Background(), receivertest.NewNopCreateSettings(), config, consumertest.NewNop()) + assert.NoError(t, err) + assert.NotNil(t, receiver) +} + +func Test_NewMetricsReceiver(t *testing.T) { + f := NewFactory() + config := createDefaultConfig().(*Config) + config.Connection = "Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234;EntityPath=hubName" + + receiver, err := f.CreateMetricsReceiver(context.Background(), receivertest.NewNopCreateSettings(), config, consumertest.NewNop()) + assert.NoError(t, err) + assert.NotNil(t, receiver) +} diff --git a/collector/components/azureeventhubreceiver/generated_component_test.go b/collector/components/azureeventhubreceiver/generated_component_test.go new file mode 100644 index 0000000..f533f0b --- /dev/null +++ b/collector/components/azureeventhubreceiver/generated_component_test.go @@ -0,0 +1,81 @@ +// Code generated by mdatagen. DO NOT EDIT. + +package azureeventhubreceiver + +import ( + "context" + "testing" + + "github.com/stretchr/testify/require" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/component/componenttest" + "go.opentelemetry.io/collector/confmap/confmaptest" + "go.opentelemetry.io/collector/consumer/consumertest" + "go.opentelemetry.io/collector/receiver" + "go.opentelemetry.io/collector/receiver/receivertest" +) + +// assertNoErrorHost implements a component.Host that asserts that there were no errors. +type assertNoErrorHost struct { + component.Host + *testing.T +} + +var _ component.Host = (*assertNoErrorHost)(nil) + +func TestComponentLifecycle(t *testing.T) { + factory := NewFactory() + + tests := []struct { + name string + createFn func(ctx context.Context, set receiver.CreateSettings, cfg component.Config) (component.Component, error) + }{ + + { + name: "logs", + createFn: func(ctx context.Context, set receiver.CreateSettings, cfg component.Config) (component.Component, error) { + return factory.CreateLogsReceiver(ctx, set, cfg, consumertest.NewNop()) + }, + }, + + { + name: "metrics", + createFn: func(ctx context.Context, set receiver.CreateSettings, cfg component.Config) (component.Component, error) { + return factory.CreateMetricsReceiver(ctx, set, cfg, consumertest.NewNop()) + }, + }, + } + + cm, err := confmaptest.LoadConf("metadata.yaml") + require.NoError(t, err) + cfg := factory.CreateDefaultConfig() + sub, err := cm.Sub("tests::config") + require.NoError(t, err) + require.NoError(t, component.UnmarshalConfig(sub, cfg)) + + for _, test := range tests { + t.Run(test.name+"-shutdown", func(t *testing.T) { + c, err := test.createFn(context.Background(), receivertest.NewNopCreateSettings(), cfg) + require.NoError(t, err) + err = c.Shutdown(context.Background()) + require.NoError(t, err) + }) + + t.Run(test.name+"-lifecycle", func(t *testing.T) { + + // TODO support lifecycle + t.SkipNow() + + firstRcvr, err := test.createFn(context.Background(), receivertest.NewNopCreateSettings(), cfg) + require.NoError(t, err) + host := componenttest.NewNopHost() + require.NoError(t, err) + require.NoError(t, firstRcvr.Start(context.Background(), host)) + require.NoError(t, firstRcvr.Shutdown(context.Background())) + secondRcvr, err := test.createFn(context.Background(), receivertest.NewNopCreateSettings(), cfg) + require.NoError(t, err) + require.NoError(t, secondRcvr.Start(context.Background(), host)) + require.NoError(t, secondRcvr.Shutdown(context.Background())) + }) + } +} diff --git a/collector/components/azureeventhubreceiver/go.mod b/collector/components/azureeventhubreceiver/go.mod new file mode 100644 index 0000000..e1c68ef --- /dev/null +++ b/collector/components/azureeventhubreceiver/go.mod @@ -0,0 +1,140 @@ +module github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver + +go 1.21.0 + +toolchain go1.22.4 + +require ( + github.com/Azure/azure-amqp-common-go/v4 v4.2.0 + github.com/Azure/azure-sdk-for-go/sdk/azcore v1.11.1 + github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs v1.1.0 + github.com/json-iterator/go v1.1.12 + github.com/open-telemetry/opentelemetry-collector-contrib/internal/sharedcomponent v0.101.0 + github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza v0.101.0 + github.com/open-telemetry/opentelemetry-collector-contrib/pkg/translator/azure v0.101.0 + github.com/relvacode/iso8601 v1.4.0 + github.com/stretchr/testify v1.9.0 + go.opentelemetry.io/collector/component v0.101.0 + go.opentelemetry.io/collector/confmap v0.101.0 + go.opentelemetry.io/collector/consumer v0.101.0 + go.opentelemetry.io/collector/extension v0.101.0 + go.opentelemetry.io/collector/otelcol v0.101.0 + go.opentelemetry.io/collector/pdata v1.8.0 + go.opentelemetry.io/collector/receiver v0.101.0 + go.opentelemetry.io/collector/semconv v0.101.0 + go.opentelemetry.io/otel/metric v1.26.0 + go.opentelemetry.io/otel/trace v1.26.0 + go.uber.org/zap v1.27.0 +) + +require ( + github.com/Azure/azure-sdk-for-go/sdk/internal v1.5.2 // indirect + github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v1.3.2 // indirect + github.com/Azure/go-amqp v1.0.5 // indirect + github.com/beorn7/perks v1.0.1 // indirect + github.com/cenkalti/backoff/v4 v4.3.0 // indirect + github.com/cespare/xxhash/v2 v2.3.0 // indirect + github.com/davecgh/go-spew v1.1.1 // indirect + github.com/expr-lang/expr v1.16.7 // indirect + github.com/go-logr/logr v1.4.1 // indirect + github.com/go-logr/stdr v1.2.2 // indirect + github.com/go-ole/go-ole v1.2.6 // indirect + github.com/go-viper/mapstructure/v2 v2.0.0-alpha.1 // indirect + github.com/gogo/protobuf v1.3.2 // indirect + github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect + github.com/google/uuid v1.6.0 // indirect + github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.1 // indirect + github.com/hashicorp/go-version v1.6.0 // indirect + github.com/inconshreveable/mousetrap v1.1.0 // indirect + github.com/influxdata/go-syslog/v3 v3.0.1-0.20230911200830-875f5bc594a4 // indirect + github.com/knadh/koanf/maps v0.1.1 // indirect + github.com/knadh/koanf/providers/confmap v0.1.0 // indirect + github.com/knadh/koanf/v2 v2.1.1 // indirect + github.com/leodido/ragel-machinery v0.0.0-20181214104525-299bdde78165 // indirect + github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0 // indirect + github.com/mitchellh/copystructure v1.2.0 // indirect + github.com/mitchellh/reflectwalk v1.0.2 // indirect + github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect + github.com/modern-go/reflect2 v1.0.2 // indirect + github.com/open-telemetry/opentelemetry-collector-contrib/internal/coreinternal v0.101.0 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c // indirect + github.com/prometheus/client_golang v1.19.1 // indirect + github.com/prometheus/client_model v0.6.1 // indirect + github.com/prometheus/common v0.53.0 // indirect + github.com/prometheus/procfs v0.12.0 // indirect + github.com/shirou/gopsutil/v3 v3.24.4 // indirect + github.com/shoenig/go-m1cpu v0.1.6 // indirect + github.com/spf13/cobra v1.8.0 // indirect + github.com/spf13/pflag v1.0.5 // indirect + github.com/stretchr/objx v0.5.2 // indirect + github.com/tklauser/go-sysconf v0.3.12 // indirect + github.com/tklauser/numcpus v0.6.1 // indirect + github.com/valyala/fastjson v1.6.4 // indirect + github.com/yusufpapurcu/wmi v1.2.4 // indirect + go.opencensus.io v0.24.0 // indirect + go.opentelemetry.io/collector v0.101.0 // indirect + go.opentelemetry.io/collector/config/configtelemetry v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/converter/expandconverter v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/provider/envprovider v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/provider/fileprovider v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/provider/httpprovider v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/provider/httpsprovider v0.101.0 // indirect + go.opentelemetry.io/collector/confmap/provider/yamlprovider v0.101.0 // indirect + go.opentelemetry.io/collector/connector v0.101.0 // indirect + go.opentelemetry.io/collector/exporter v0.101.0 // indirect + go.opentelemetry.io/collector/featuregate v1.8.0 // indirect + go.opentelemetry.io/collector/pdata/testdata v0.101.0 // indirect + go.opentelemetry.io/collector/processor v0.101.0 // indirect + go.opentelemetry.io/collector/service v0.101.0 // indirect + go.opentelemetry.io/contrib/config v0.6.0 // indirect + go.opentelemetry.io/contrib/propagators/b3 v1.26.0 // indirect + go.opentelemetry.io/otel v1.26.0 // indirect + go.opentelemetry.io/otel/bridge/opencensus v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/prometheus v0.48.0 // indirect + go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.26.0 // indirect + go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.26.0 // indirect + go.opentelemetry.io/otel/sdk v1.26.0 // indirect + go.opentelemetry.io/otel/sdk/metric v1.26.0 // indirect + go.opentelemetry.io/proto/otlp v1.2.0 // indirect + go.uber.org/multierr v1.11.0 // indirect + golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842 // indirect + golang.org/x/net v0.24.0 // indirect + golang.org/x/sys v0.20.0 // indirect + golang.org/x/text v0.15.0 // indirect + gonum.org/v1/gonum v0.15.0 // indirect + google.golang.org/genproto/googleapis/api v0.0.0-20240227224415-6ceb2ff114de // indirect + google.golang.org/genproto/googleapis/rpc v0.0.0-20240401170217-c3f982113cda // indirect + google.golang.org/grpc v1.63.2 // indirect + google.golang.org/protobuf v1.34.1 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect +) + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/extension/storage => ../../extension/storage + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza => ../../pkg/stanza + +retract ( + v0.76.2 + v0.76.1 + v0.65.0 +) + +replace github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver => ../azureeventhubreceiver + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/internal/coreinternal => ../../internal/coreinternal + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatatest => ../../pkg/pdatatest + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatautil => ../../pkg/pdatautil + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/internal/sharedcomponent => ../../internal/sharedcomponent + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/pkg/translator/azure => ../../pkg/translator/azure + +// replace github.com/open-telemetry/opentelemetry-collector-contrib/pkg/golden => ../../pkg/golden diff --git a/collector/components/azureeventhubreceiver/go.sum b/collector/components/azureeventhubreceiver/go.sum new file mode 100644 index 0000000..d92dc1e --- /dev/null +++ b/collector/components/azureeventhubreceiver/go.sum @@ -0,0 +1,401 @@ +cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= +github.com/Azure/azure-amqp-common-go/v4 v4.2.0 h1:q/jLx1KJ8xeI8XGfkOWMN9XrXzAfVTkyvCxPvHCjd2I= +github.com/Azure/azure-amqp-common-go/v4 v4.2.0/go.mod h1:GD3m/WPPma+621UaU6KNjKEo5Hl09z86viKwQjTpV0Q= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.11.1 h1:E+OJmp2tPvt1W+amx48v1eqbjDYsgN+RzP4q16yV5eM= +github.com/Azure/azure-sdk-for-go/sdk/azcore v1.11.1/go.mod h1:a6xsAQUZg+VsS3TJ05SRp524Hs4pZ/AeFSr5ENf0Yjo= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.5.1 h1:sO0/P7g68FrryJzljemN+6GTssUXdANk6aJ7T1ZxnsQ= +github.com/Azure/azure-sdk-for-go/sdk/azidentity v1.5.1/go.mod h1:h8hyGFDsU5HMivxiS2iYFZsgDbU9OnnJ163x5UGVKYo= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.5.2 h1:LqbJ/WzJUwBf8UiaSzgX7aMclParm9/5Vgp+TY51uBQ= +github.com/Azure/azure-sdk-for-go/sdk/internal v1.5.2/go.mod h1:yInRyqWXAuaPrgI7p70+lDDgh3mlBohis29jGMISnmc= +github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs v1.1.0 h1:vEe09cdSBy7evqoVUvuitnsjyozsSzI4TbGgwu01+TI= +github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs v1.1.0/go.mod h1:PgOlzIlvwIagKI8N6hCsfFDpAijHCmlHqOwA5GsSh9w= +github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/eventhub/armeventhub v1.0.0 h1:BWeAAEzkCnL0ABVJqs+4mYudNch7oFGPtTlSmIWL8ms= +github.com/Azure/azure-sdk-for-go/sdk/resourcemanager/eventhub/armeventhub v1.0.0/go.mod h1:Y3gnVwfaz8h6L1YHar+NfWORtBoVUSB5h4GlGkdeF7Q= +github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v1.3.2 h1:YUUxeiOWgdAQE3pXt2H7QXzZs0q8UBjgRbl56qo8GYM= +github.com/Azure/azure-sdk-for-go/sdk/storage/azblob v1.3.2/go.mod h1:dmXQgZuiSubAecswZE+Sm8jkvEa7kQgTPVRvwL/nd0E= +github.com/Azure/go-amqp v1.0.5 h1:po5+ljlcNSU8xtapHTe8gIc8yHxCzC03E8afH2g1ftU= +github.com/Azure/go-amqp v1.0.5/go.mod h1:vZAogwdrkbyK3Mla8m/CxSc/aKdnTZ4IbPxl51Y5WZE= +github.com/AzureAD/microsoft-authentication-library-for-go v1.2.1 h1:DzHpqpoJVaCgOUdVHxE8QB52S6NiVdDQvGlny1qvPqA= +github.com/AzureAD/microsoft-authentication-library-for-go v1.2.1/go.mod h1:wP83P5OoQ5p6ip3ScPr0BAq0BvuPAvacpEuSzyouqAI= +github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= +github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= +github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= +github.com/cenkalti/backoff/v4 v4.3.0 h1:MyRJ/UdXutAwSAT+s3wNd7MfTIcy71VQueUuFK343L8= +github.com/cenkalti/backoff/v4 v4.3.0/go.mod h1:Y3VNntkOUPxTVeUxJ/G5vcM//AlwfmyYozVcomhLiZE= +github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash/v2 v2.3.0 h1:UL815xU9SqsFlibzuggzjXhog7bL6oX9BbNZnL2UFvs= +github.com/cespare/xxhash/v2 v2.3.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= +github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= +github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= +github.com/cpuguy83/go-md2man/v2 v2.0.3/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= +github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= +github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= +github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= +github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/expr-lang/expr v1.16.7 h1:gCIiHt5ODA0xIaDbD0DPKyZpM9Drph3b3lolYAYq2Kw= +github.com/expr-lang/expr v1.16.7/go.mod h1:8/vRC7+7HBzESEqt5kKpYXxrxkr31SaO8r40VO/1IT4= +github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw= +github.com/fortytw2/leaktest v1.3.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= +github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/logr v1.4.1 h1:pKouT5E8xu9zeFC39JXRDukb6JFQPXM5p5I91188VAQ= +github.com/go-logr/logr v1.4.1/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY= +github.com/go-logr/stdr v1.2.2 h1:hSWxHoqTgW2S2qGc0LTAI563KZ5YKYRhT3MFKZMbjag= +github.com/go-logr/stdr v1.2.2/go.mod h1:mMo/vtBO5dYbehREoey6XUKy/eSumjCCveDpRre4VKE= +github.com/go-ole/go-ole v1.2.6 h1:/Fpf6oFPoeFik9ty7siob0G6Ke8QvQEuVcuChpwXzpY= +github.com/go-ole/go-ole v1.2.6/go.mod h1:pprOEPIfldk/42T2oK7lQ4v4JSDwmV0As9GaiUsvbm0= +github.com/go-viper/mapstructure/v2 v2.0.0-alpha.1 h1:TQcrn6Wq+sKGkpyPvppOz99zsMBaUOKXq6HSv655U1c= +github.com/go-viper/mapstructure/v2 v2.0.0-alpha.1/go.mod h1:oJDH3BJKyqBA2TXFhDsKDGDTlndYOZ6rGS0BRZIxGhM= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= +github.com/golang-jwt/jwt/v5 v5.2.0 h1:d/ix8ftRUorsN+5eMIlF4T6J8CAt9rch3My2winC1Jw= +github.com/golang-jwt/jwt/v5 v5.2.0/go.mod h1:pqrtFR0X4osieyHYxtmOUWsAWrfe1Q5UVIyoH402zdk= +github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= +github.com/golang/mock v1.6.0 h1:ErTB+efbowRARo13NNdxyJji2egdxLGQhRaY+DUumQc= +github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= +github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= +github.com/golang/protobuf v1.4.0-rc.1/go.mod h1:ceaxUfeHdC40wWswd/P6IGgMaK3YpKi5j83Wpe3EHw8= +github.com/golang/protobuf v1.4.0-rc.1.0.20200221234624-67d41d38c208/go.mod h1:xKAWHe0F5eneWXFV3EuXVDTCmh+JuBKY0li0aMyXATA= +github.com/golang/protobuf v1.4.0-rc.2/go.mod h1:LlEzMj4AhA7rCAGe4KMBDvJI+AwstrUpVNzEA03Pprs= +github.com/golang/protobuf v1.4.0-rc.4.0.20200313231945-b860323f09d0/go.mod h1:WU3c8KckQ9AFe+yFwt9sWVRKCVIyN9cPHBJSNnbL67w= +github.com/golang/protobuf v1.4.0/go.mod h1:jodUvKwWbYaEsadDk5Fwe5c77LiNKVO9IDvqG2KuDX0= +github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QDs8UjoX8= +github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= +github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= +github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= +github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI= +github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= +github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0= +github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.1 h1:/c3QmbOGMGTOumP2iT/rCwB7b0QDGLKzqOmktBjT+Is= +github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.1/go.mod h1:5SN9VR2LTsRFsrEC6FHgRbTWrTHu6tqPeKxEQv15giM= +github.com/hashicorp/go-version v1.6.0 h1:feTTfFNnjP967rlCxM/I9g701jU+RN74YKx2mOkIeek= +github.com/hashicorp/go-version v1.6.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= +github.com/inconshreveable/mousetrap v1.1.0 h1:wN+x4NVGpMsO7ErUn/mUI3vEoE6Jt13X2s0bqwp9tc8= +github.com/inconshreveable/mousetrap v1.1.0/go.mod h1:vpF70FUmC8bwa3OWnCshd2FqLfsEA9PFc4w1p2J65bw= +github.com/influxdata/go-syslog/v3 v3.0.1-0.20230911200830-875f5bc594a4 h1:2r2WiFeAwiJ/uyx1qIKnV1L4C9w/2V8ehlbJY4gjFaM= +github.com/influxdata/go-syslog/v3 v3.0.1-0.20230911200830-875f5bc594a4/go.mod h1:1yEQhaLb/cETXCqQmdh7lDjupNAReO7c83AHyK2dJ48= +github.com/joho/godotenv v1.5.1 h1:7eLL/+HRGLY0ldzfGMeQkb7vMd0as4CfYvUVzLqw0N0= +github.com/joho/godotenv v1.5.1/go.mod h1:f4LDr5Voq0i2e/R5DDNOoa2zzDfwtkZa6DnEwAbqwq4= +github.com/json-iterator/go v1.1.12 h1:PV8peI4a0ysnczrg+LtxykD8LfKY9ML6u2jnxaEnrnM= +github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= +github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/knadh/koanf/maps v0.1.1 h1:G5TjmUh2D7G2YWf5SQQqSiHRJEjaicvU0KpypqB3NIs= +github.com/knadh/koanf/maps v0.1.1/go.mod h1:npD/QZY3V6ghQDdcQzl1W4ICNVTkohC8E73eI2xW4yI= +github.com/knadh/koanf/providers/confmap v0.1.0 h1:gOkxhHkemwG4LezxxN8DMOFopOPghxRVp7JbIvdvqzU= +github.com/knadh/koanf/providers/confmap v0.1.0/go.mod h1:2uLhxQzJnyHKfxG927awZC7+fyHFdQkd697K4MdLnIU= +github.com/knadh/koanf/v2 v2.1.1 h1:/R8eXqasSTsmDCsAyYj+81Wteg8AqrV9CP6gvsTsOmM= +github.com/knadh/koanf/v2 v2.1.1/go.mod h1:4mnTRbZCK+ALuBXHZMjDfG9y714L7TykVnZkXbMU3Es= +github.com/kr/pretty v0.3.1 h1:flRD4NNwYAUpkphVc1HcthR4KEIFJ65n8Mw5qdRn3LE= +github.com/kr/pretty v0.3.1/go.mod h1:hoEshYVHaxMs3cyo3Yncou5ZscifuDolrwPKZanG3xk= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/kylelemons/godebug v1.1.0 h1:RPNrshWIDI6G2gRW9EHilWtl7Z6Sb1BR0xunSBf0SNc= +github.com/kylelemons/godebug v1.1.0/go.mod h1:9/0rRGxNHcop5bhtWyNeEfOS8JIWk580+fNqagV/RAw= +github.com/leodido/ragel-machinery v0.0.0-20181214104525-299bdde78165 h1:bCiVCRCs1Heq84lurVinUPy19keqGEe4jh5vtK37jcg= +github.com/leodido/ragel-machinery v0.0.0-20181214104525-299bdde78165/go.mod h1:WZxr2/6a/Ar9bMDc2rN/LJrE/hF6bXE4LPyDSIxwAfg= +github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0 h1:6E+4a0GO5zZEnZ81pIr0yLvtUWk2if982qA3F3QD6H4= +github.com/lufia/plan9stats v0.0.0-20211012122336-39d0f177ccd0/go.mod h1:zJYVVT2jmtg6P3p1VtQj7WsuWi/y4VnjVBn7F8KPB3I= +github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= +github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= +github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= +github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx0jmZXqmk4esnw= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v1.0.2 h1:xBagoLtFs94CBntxluKeaWgTMpvLxC4ur3nMaC9Gz0M= +github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= +github.com/open-telemetry/opentelemetry-collector-contrib/extension/storage v0.101.0 h1:oYMkHmAFSXO1UThrb6fK/1ixGN3dROpcgKsWIz5PtXs= +github.com/open-telemetry/opentelemetry-collector-contrib/extension/storage v0.101.0/go.mod h1:QDF6x3uIeJRkPGgw4Ss3i4NbxzB/QFCFnXInsXf4C08= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/common v0.101.0 h1:NjRf0D9Of6WlqEIXKhOIbBs4YudrPAtDn6+Rez1MqzA= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/common v0.101.0/go.mod h1:ofU9nc88bSz7EbFY70TBgrcpWFPpeP+vMpVycda+ZrI= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/coreinternal v0.101.0 h1:X+FXRfxLK2mH813tMyZmX93Mt/3l6F8X5aFi7QPBQDI= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/coreinternal v0.101.0/go.mod h1:j/pizzitn+kpiTNTxsgpaGqAW3qh3pRSbSTUsIeQcLE= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/sharedcomponent v0.101.0 h1:7SoekZMs3TKUetCoRehFJc3ks8QeJ4vIVgJ049wllvA= +github.com/open-telemetry/opentelemetry-collector-contrib/internal/sharedcomponent v0.101.0/go.mod h1:w7QGLrJse2b3gQiA0gETf00LMRhjCaPpIEfDX+fS+1E= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatatest v0.101.0 h1:TCQYvGS2MKTotOTQDnHUSd4ljEzXRzHXopdv71giKWU= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatatest v0.101.0/go.mod h1:Nl2d4DSK/IbaWnnBxYyhMNUW6C9sb5/4idVZrSW/5Ps= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatautil v0.101.0 h1:dVINhi/nne11lG+Xnwuy9t/N4xyaH2Om2EU+5lphCA4= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/pdatautil v0.101.0/go.mod h1:kjyfpKOuBfkx3UsJQsbQ5eTJM3yQWiRYaYxs47PpxvI= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza v0.101.0 h1:r7ue2vHBAH5v1AiNsC3TWDSysSdG/nhZ8HFnhOE+dbw= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza v0.101.0/go.mod h1:l+8+GK6bzSjK4bLTfbkU0hj+9y8wbpaDr42tmqOEDr0= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/translator/azure v0.101.0 h1:oaI1kjrUO6S3GxuoUVe2O0dFqzvOVRbl3Fyq0M8P5y4= +github.com/open-telemetry/opentelemetry-collector-contrib/pkg/translator/azure v0.101.0/go.mod h1:Oi6xicwrWTzOhLw4eX4Zwn2ZsR9g184MpCv8csY/P7Q= +github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c h1:+mdjkGKdHQG3305AYmdv1U2eRNDiU2ErMBj1gwrq8eQ= +github.com/pkg/browser v0.0.0-20240102092130-5ac0b6a4141c/go.mod h1:7rwL4CYBLnjLxUqIJNnCWiEdr3bn6IUYi15bNlnbCCU= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= +github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c h1:ncq/mPwQF4JjgDlrVEn3C11VoGHZN7m8qihwgMEtzYw= +github.com/power-devops/perfstat v0.0.0-20210106213030-5aafc221ea8c/go.mod h1:OmDBASR4679mdNQnz2pUhc2G8CO2JrUAVFDRBDP/hJE= +github.com/prometheus/client_golang v1.19.1 h1:wZWJDwK+NameRJuPGDhlnFgx8e8HN3XHQeLaYJFJBOE= +github.com/prometheus/client_golang v1.19.1/go.mod h1:mP78NwGzrVks5S2H6ab8+ZZGJLZUq1hoULYBAYBw1Ho= +github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.6.1 h1:ZKSh/rekM+n3CeS952MLRAdFwIKqeY8b62p8ais2e9E= +github.com/prometheus/client_model v0.6.1/go.mod h1:OrxVMOVHjw3lKMa8+x6HeMGkHMQyHDk9E3jmP2AmGiY= +github.com/prometheus/common v0.53.0 h1:U2pL9w9nmJwJDa4qqLQ3ZaePJ6ZTwt7cMD3AG3+aLCE= +github.com/prometheus/common v0.53.0/go.mod h1:BrxBKv3FWBIGXw89Mg1AeBq7FSyRzXWI3l3e7W3RN5U= +github.com/prometheus/procfs v0.12.0 h1:jluTpSng7V9hY0O2R9DzzJHYb2xULk9VTR1V1R/k6Bo= +github.com/prometheus/procfs v0.12.0/go.mod h1:pcuDEFsWDnvcgNzo4EEweacyhjeA9Zk3cnaOZAZEfOo= +github.com/relvacode/iso8601 v1.4.0 h1:GsInVSEJfkYuirYFxa80nMLbH2aydgZpIf52gYZXUJs= +github.com/relvacode/iso8601 v1.4.0/go.mod h1:FlNp+jz+TXpyRqgmM7tnzHHzBnz776kmAH2h3sZCn0I= +github.com/rogpeppe/go-internal v1.11.0 h1:cWPaGQEPrBb5/AsnsZesgZZ9yb1OQ+GOISoDNXVBh4M= +github.com/rogpeppe/go-internal v1.11.0/go.mod h1:ddIwULY96R17DhadqLgMfk9H9tvdUzkipdSkR5nkCZA= +github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/shirou/gopsutil/v3 v3.24.4 h1:dEHgzZXt4LMNm+oYELpzl9YCqV65Yr/6SfrvgRBtXeU= +github.com/shirou/gopsutil/v3 v3.24.4/go.mod h1:lTd2mdiOspcqLgAnr9/nGi71NkeMpWKdmhuxm9GusH8= +github.com/shoenig/go-m1cpu v0.1.6 h1:nxdKQNcEB6vzgA2E2bvzKIYRuNj7XNJ4S/aRSwKzFtM= +github.com/shoenig/go-m1cpu v0.1.6/go.mod h1:1JJMcUBvfNwpq05QDQVAnx3gUHr9IYF7GNg9SUEw2VQ= +github.com/shoenig/test v0.6.4 h1:kVTaSd7WLz5WZ2IaoM0RSzRsUD+m8wRR+5qvntpn4LU= +github.com/shoenig/test v0.6.4/go.mod h1:byHiCGXqrVaflBLAMq/srcZIHynQPQgeyvkvXnjqq0k= +github.com/spf13/cobra v1.8.0 h1:7aJaZx1B85qltLMc546zn58BxxfZdR/W22ej9CFoEf0= +github.com/spf13/cobra v1.8.0/go.mod h1:WXLWApfZ71AjXPya3WOlMsY9yMs7YeiHhFVlvLyhcho= +github.com/spf13/pflag v1.0.5 h1:iy+VFUOCP1a+8yFto/drg2CJ5u0yRoB7fZw3DKv/JXA= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= +github.com/stretchr/objx v0.5.2 h1:xuMeJ0Sdp5ZMRXx/aWO6RZxdr3beISkG5/G/aIRr3pY= +github.com/stretchr/objx v0.5.2/go.mod h1:FRsXN1f5AsAjCGJKqEizvkpNtU+EGNCLh3NxZ/8L+MA= +github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.4/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= +github.com/stretchr/testify v1.9.0 h1:HtqpIVDClZ4nwg75+f6Lvsy/wHu+3BoSGCbBAcpTsTg= +github.com/stretchr/testify v1.9.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY= +github.com/tklauser/go-sysconf v0.3.12 h1:0QaGUFOdQaIVdPgfITYzaTegZvdCjmYO52cSFAEVmqU= +github.com/tklauser/go-sysconf v0.3.12/go.mod h1:Ho14jnntGE1fpdOqQEEaiKRpvIavV0hSfmBq8nJbHYI= +github.com/tklauser/numcpus v0.6.1 h1:ng9scYS7az0Bk4OZLvrNXNSAO2Pxr1XXRAPyjhIx+Fk= +github.com/tklauser/numcpus v0.6.1/go.mod h1:1XfjsgE2zo8GVw7POkMbHENHzVg3GzmoZ9fESEdAacY= +github.com/valyala/fastjson v1.6.4 h1:uAUNq9Z6ymTgGhcm0UynUAB6tlbakBrz6CQFax3BXVQ= +github.com/valyala/fastjson v1.6.4/go.mod h1:CLCAqky6SMuOcxStkYQvblddUtoRxhYMGLrsQns1aXY= +github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= +github.com/yusufpapurcu/wmi v1.2.4 h1:zFUKzehAFReQwLys1b/iSMl+JQGSCSjtVqQn9bBrPo0= +github.com/yusufpapurcu/wmi v1.2.4/go.mod h1:SBZ9tNy3G9/m5Oi98Zks0QjeHVDvuK0qfxQmPyzfmi0= +go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= +go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= +go.opentelemetry.io/collector v0.101.0 h1:jnCI/JZgpEYONWy4LCvif4CjMM7cPS4XvGHp3OrZpYo= +go.opentelemetry.io/collector v0.101.0/go.mod h1:N0xja/N3NUDIC55SjjNzyyIoxE6YoCEZC3aXQ39yIVs= +go.opentelemetry.io/collector/component v0.101.0 h1:2sILYgE8cZJj0Vseh6LUjS9iXPyqDPTx/R8yf8IPu+4= +go.opentelemetry.io/collector/component v0.101.0/go.mod h1:OB1uBpQZ2Ba6wVui/sthh6j+CPxVQIy2ou5rzZPINQQ= +go.opentelemetry.io/collector/config/confignet v0.101.0 h1:Mdb9e/EpCSac4Ccg7w4UchS/o4yY1WoIc9X5o7fTu9E= +go.opentelemetry.io/collector/config/confignet v0.101.0/go.mod h1:3naWoPss70RhDHhYjGACi7xh4NcVRvs9itzIRVWyu1k= +go.opentelemetry.io/collector/config/configretry v0.101.0 h1:5QggLq/lZiZXry1Ut52IOTbrdz1RbGoL29Io/wWdE4g= +go.opentelemetry.io/collector/config/configretry v0.101.0/go.mod h1:uRdmPeCkrW9Zsadh2WEbQ1AGXGYJ02vCfmmT+0g69nY= +go.opentelemetry.io/collector/config/configtelemetry v0.101.0 h1:G9RerNdBUm6rYW6wrJoKzleBiDsCGaCjtQx5UYr0hzw= +go.opentelemetry.io/collector/config/configtelemetry v0.101.0/go.mod h1:YV5PaOdtnU1xRomPcYqoHmyCr48tnaAREeGO96EZw8o= +go.opentelemetry.io/collector/confmap v0.101.0 h1:pGXZRBKnZqys1HgNECGSi8Pec5RBGa9vVCfrpcvW+kA= +go.opentelemetry.io/collector/confmap v0.101.0/go.mod h1:BWKPIpYeUzSG6ZgCJMjF7xsLvyrvJCfYURl57E5vhiQ= +go.opentelemetry.io/collector/confmap/converter/expandconverter v0.101.0 h1:/uUZlzzxO8QknVCslpYVlQGSq5EG3Dzr6l4w2xW4u2A= +go.opentelemetry.io/collector/confmap/converter/expandconverter v0.101.0/go.mod h1:GKgiSuL5+ATJE1lrAQVpfSBFX/3XqGyc2qrfQBKpVd8= +go.opentelemetry.io/collector/confmap/provider/envprovider v0.101.0 h1:I7oSi6hTTaDMbh9k6nxF4YhLqB/t0xXTgrBXGCT53vw= +go.opentelemetry.io/collector/confmap/provider/envprovider v0.101.0/go.mod h1:DxCK400//MGnFyLSnIjme+R7qZwfDQtHYERIQtEt7Cg= +go.opentelemetry.io/collector/confmap/provider/fileprovider v0.101.0 h1:uMjJyxN3q+DaKR+GOJcERuVD8rKEe1PvTOUaMs66gCM= +go.opentelemetry.io/collector/confmap/provider/fileprovider v0.101.0/go.mod h1:3WLJtHisH4/7K/IU//OMoCTJYZ8o96/YfjP6J+Q/kBo= +go.opentelemetry.io/collector/confmap/provider/httpprovider v0.101.0 h1:U+/Mmd1+DHl94R/i+LxTjReEtFh3qV4QJ8XxqhCdezk= +go.opentelemetry.io/collector/confmap/provider/httpprovider v0.101.0/go.mod h1:EJ5t47HWbDGxUjVax+BBJ3ySpHBHQz/Ys689+R6OXis= +go.opentelemetry.io/collector/confmap/provider/httpsprovider v0.101.0 h1:W0Xw+OgRCbdKWJy3VSZKPCcf4fFZlFF6L+mMWleq1LY= +go.opentelemetry.io/collector/confmap/provider/httpsprovider v0.101.0/go.mod h1:0I8UtPWeXbuwe/UMQ+LmoFWoNo6NCxxNocHPrLox0X8= +go.opentelemetry.io/collector/confmap/provider/yamlprovider v0.101.0 h1:N5yNF24hxUOO5Ps5mWwwQaYWyBPcqqSh4h10kULDT3c= +go.opentelemetry.io/collector/confmap/provider/yamlprovider v0.101.0/go.mod h1:sPzwdCKCXYXUR8U7eAHshDZPnfbF7B7I/BFyUWTvvKQ= +go.opentelemetry.io/collector/connector v0.101.0 h1:OedmwrzrxC3wrYkp8Mpfcf30bJmlxC9TuwNLnpj4V8M= +go.opentelemetry.io/collector/connector v0.101.0/go.mod h1:skAILMO4ye4Y3s2DUo7k/8uZFwG22fpwjIYXO/pv/JQ= +go.opentelemetry.io/collector/consumer v0.101.0 h1:9tDxaeHe1+Uovf3fhdx7T4pV5mo/Dc0hniH7O5H3RBA= +go.opentelemetry.io/collector/consumer v0.101.0/go.mod h1:ud5k64on9m7hHTrhjEeLhWbLkd8+Gp06rDt3p86TKNs= +go.opentelemetry.io/collector/exporter v0.101.0 h1:zAxQBfaWO+PEHL3nDglgMGaWsqLsj1lJHPaBnO8PeDo= +go.opentelemetry.io/collector/exporter v0.101.0/go.mod h1:ZFwUWCmnM2ZbEty71Q13qME9QhvIKMgyYrS3s8vJPM8= +go.opentelemetry.io/collector/extension v0.101.0 h1:A4hq/aci9+/Pxi8sJfyYgbeHjSIL7JFZR81IlSOTla4= +go.opentelemetry.io/collector/extension v0.101.0/go.mod h1:14gQMuybTcppfTTM9AwqeoFrNCLv/ds/c0A4Z0hWuLI= +go.opentelemetry.io/collector/extension/zpagesextension v0.101.0 h1:hZIkGTgKeVvVlqoPw72G/RkIhp0QSqY5PNRjf38mf2k= +go.opentelemetry.io/collector/extension/zpagesextension v0.101.0/go.mod h1:sllOMdEbNg2UnMxTO8jSx2OEfAcYX3ud6smuXhN6pbA= +go.opentelemetry.io/collector/featuregate v1.8.0 h1:p/bAuk5LiSfdYS88yFl/Jzao9bHEYqCh7YvZJ+L+IZg= +go.opentelemetry.io/collector/featuregate v1.8.0/go.mod h1:w7nUODKxEi3FLf1HslCiE6YWtMtOOrMnSwsDam8Mg9w= +go.opentelemetry.io/collector/otelcol v0.101.0 h1:6kF2dcXpu5NjxK2j0ksCRzZhqigxCGrP/u7n57FSMOg= +go.opentelemetry.io/collector/otelcol v0.101.0/go.mod h1:qGrb+hlZXId/hJj0y28vq0YkMd6Xsoz2w7mZkXJOw68= +go.opentelemetry.io/collector/pdata v1.8.0 h1:d/QQgZxB4Y+d3mqLVh2ozvzujUhloD3P/fk7X+In764= +go.opentelemetry.io/collector/pdata v1.8.0/go.mod h1:/W7clu0wFC4WSRp94Ucn6Vm36Wkrt+tmtlDb1aiNZCY= +go.opentelemetry.io/collector/pdata/testdata v0.101.0 h1:JzeUtg5RN1iIFgY8DakGlqBkGxOTJlkaYlLausnEGKY= +go.opentelemetry.io/collector/pdata/testdata v0.101.0/go.mod h1:ZGobfCus4fWo5RduZ7ENI0+HD9BewgKuO6qU2rBVnUg= +go.opentelemetry.io/collector/processor v0.101.0 h1:VU77ImECho43O/7p3w6KUnNvzg/TQ4/WxjZzvI/TNm0= +go.opentelemetry.io/collector/processor v0.101.0/go.mod h1:uLvPw3SNAJbqkdWAzDs4F7S3FobHNG6fJQaO9q3aGVQ= +go.opentelemetry.io/collector/receiver v0.101.0 h1:+YJQvcAw5Es15Ub8hYqqZumKbe7D0SMU8XCgGRxc25M= +go.opentelemetry.io/collector/receiver v0.101.0/go.mod h1:JFVHAkIIz9uOk85u9pHsYRcyFj1ZAUpw59ahNZ28+ko= +go.opentelemetry.io/collector/semconv v0.101.0 h1:tOe9iTe9dDCnvz/bqgfNRr4w80kXG8505tQJ5h5v08Q= +go.opentelemetry.io/collector/semconv v0.101.0/go.mod h1:8ElcRZ8Cdw5JnvhTOQOdYizkJaQ10Z2fS+R6djOnj6A= +go.opentelemetry.io/collector/service v0.101.0 h1:My2NrH25WYmJ6vMWwT3csglyiTkf0XP3nPgj0mX1yFw= +go.opentelemetry.io/collector/service v0.101.0/go.mod h1:XowYC9FyNGmWClh0aObztKdTfQNLAr6mubpvh27ee+Q= +go.opentelemetry.io/contrib/config v0.6.0 h1:M1SRD1Z15XHPGk61tMLI1up77XT5FdrqQSRrlH0fYuk= +go.opentelemetry.io/contrib/config v0.6.0/go.mod h1:t+/kzmRWLN7J+4F/dD4fFvlYCmCO63WYwy/B00IC++c= +go.opentelemetry.io/contrib/propagators/b3 v1.26.0 h1:wgFbVA+bK2k+fGVfDOCOG4cfDAoppyr5sI2dVlh8MWM= +go.opentelemetry.io/contrib/propagators/b3 v1.26.0/go.mod h1:DDktFXxA+fyItAAM0Sbl5OBH7KOsCTjvbBdPKtoIf/k= +go.opentelemetry.io/contrib/zpages v0.51.0 h1:psVr4JTWd0qtISPj9EA6AODGJ09bvsOxWiuKqiGdSCA= +go.opentelemetry.io/contrib/zpages v0.51.0/go.mod h1:PKtp+NEp1gTTLmFHpynYgYCSkKtisPntOb9S1mQjFKg= +go.opentelemetry.io/otel v1.26.0 h1:LQwgL5s/1W7YiiRwxf03QGnWLb2HW4pLiAhaA5cZXBs= +go.opentelemetry.io/otel v1.26.0/go.mod h1:UmLkJHUAidDval2EICqBMbnAd0/m2vmpf/dAM+fvFs4= +go.opentelemetry.io/otel/bridge/opencensus v1.26.0 h1:DZzxj9QjznMVoehskOJnFP2gsTCWtDTFBDvFhPAY7nc= +go.opentelemetry.io/otel/bridge/opencensus v1.26.0/go.mod h1:rJiX0KrF5m8Tm1XE8jLczpAv5zUaDcvhKecFG0ZoFG4= +go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.26.0 h1:+hm+I+KigBy3M24/h1p/NHkUx/evbLH0PNcjpMyCHc4= +go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetricgrpc v1.26.0/go.mod h1:NjC8142mLvvNT6biDpaMjyz78kyEHIwAJlSX0N9P5KI= +go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.26.0 h1:HGZWGmCVRCVyAs2GQaiHQPbDHo+ObFWeUEOd+zDnp64= +go.opentelemetry.io/otel/exporters/otlp/otlpmetric/otlpmetrichttp v1.26.0/go.mod h1:SaH+v38LSCHddyk7RGlU9uZyQoRrKao6IBnJw6Kbn+c= +go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.26.0 h1:1u/AyyOqAWzy+SkPxDpahCNZParHV8Vid1RnI2clyDE= +go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.26.0/go.mod h1:z46paqbJ9l7c9fIPCXTqTGwhQZ5XoTIsfeFYWboizjs= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.26.0 h1:Waw9Wfpo/IXzOI8bCB7DIk+0JZcqqsyn1JFnAc+iam8= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.26.0/go.mod h1:wnJIG4fOqyynOnnQF/eQb4/16VlX2EJAHhHgqIqWfAo= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.26.0 h1:1wp/gyxsuYtuE/JFxsQRtcCDtMrO2qMvlfXALU5wkzI= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracehttp v1.26.0/go.mod h1:gbTHmghkGgqxMomVQQMur1Nba4M0MQ8AYThXDUjsJ38= +go.opentelemetry.io/otel/exporters/prometheus v0.48.0 h1:sBQe3VNGUjY9IKWQC6z2lNqa5iGbDSxhs60ABwK4y0s= +go.opentelemetry.io/otel/exporters/prometheus v0.48.0/go.mod h1:DtrbMzoZWwQHyrQmCfLam5DZbnmorsGbOtTbYHycU5o= +go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.26.0 h1:5fnmgteaar1VcAA69huatudPduNFz7guRtCmfZCooZI= +go.opentelemetry.io/otel/exporters/stdout/stdoutmetric v1.26.0/go.mod h1:lsPccfZiz1cb1AhBPmicWM2E4F1VynFXEvD8SEBS4TM= +go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.26.0 h1:0W5o9SzoR15ocYHEQfvfipzcNog1lBxOLfnex91Hk6s= +go.opentelemetry.io/otel/exporters/stdout/stdouttrace v1.26.0/go.mod h1:zVZ8nz+VSggWmnh6tTsJqXQ7rU4xLwRtna1M4x5jq58= +go.opentelemetry.io/otel/metric v1.26.0 h1:7S39CLuY5Jgg9CrnA9HHiEjGMF/X2VHvoXGgSllRz30= +go.opentelemetry.io/otel/metric v1.26.0/go.mod h1:SY+rHOI4cEawI9a7N1A4nIg/nTQXe1ccCNWYOJUrpX4= +go.opentelemetry.io/otel/sdk v1.26.0 h1:Y7bumHf5tAiDlRYFmGqetNcLaVUZmh4iYfmGxtmz7F8= +go.opentelemetry.io/otel/sdk v1.26.0/go.mod h1:0p8MXpqLeJ0pzcszQQN4F0S5FVjBLgypeGSngLsmirs= +go.opentelemetry.io/otel/sdk/metric v1.26.0 h1:cWSks5tfriHPdWFnl+qpX3P681aAYqlZHcAyHw5aU9Y= +go.opentelemetry.io/otel/sdk/metric v1.26.0/go.mod h1:ClMFFknnThJCksebJwz7KIyEDHO+nTB6gK8obLy8RyE= +go.opentelemetry.io/otel/trace v1.26.0 h1:1ieeAUb4y0TE26jUFrCIXKpTuVK7uJGN9/Z/2LP5sQA= +go.opentelemetry.io/otel/trace v1.26.0/go.mod h1:4iDxvGDQuUkHve82hJJ8UqrwswHYsZuWCBllGV2U2y0= +go.opentelemetry.io/proto/otlp v1.2.0 h1:pVeZGk7nXDC9O2hncA6nHldxEjm6LByfA2aN8IOkz94= +go.opentelemetry.io/proto/otlp v1.2.0/go.mod h1:gGpR8txAl5M03pDhMC79G6SdqNV26naRm/KDsgaHD8A= +go.uber.org/goleak v1.3.0 h1:2K3zAYmnTNqV73imy9J1T3WC+gmCePx2hEGkimedGto= +go.uber.org/goleak v1.3.0/go.mod h1:CoHD4mav9JJNrW/WLlf7HGZPjdw8EucARQHekz1X6bE= +go.uber.org/multierr v1.11.0 h1:blXXJkSxSSfBVBlC76pxqeO+LN3aDfLQo+309xJstO0= +go.uber.org/multierr v1.11.0/go.mod h1:20+QtiLqy0Nd6FdQB9TLXag12DsQkrbs3htMFfDN80Y= +go.uber.org/zap v1.27.0 h1:aJMhYGrd5QSmlpLMr2MftRKl7t8J8PTZPA732ud/XR8= +go.uber.org/zap v1.27.0/go.mod h1:GB2qFLM7cTU87MWRP2mPIjqfIDnGu+VIO4V/SdhGo2E= +golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.22.0 h1:g1v0xeRhjcugydODzvb3mEM9SQ0HGp9s/nh3COQ/C30= +golang.org/x/crypto v0.22.0/go.mod h1:vr6Su+7cTlO45qkww3VDJlzDn0ctJvRgYbC2NvXHt+M= +golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= +golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842 h1:vr/HnozRka3pE4EsMEg1lgkXJkTFJCVUX+S/ZT6wYzM= +golang.org/x/exp v0.0.0-20240506185415-9bf2ced13842/go.mod h1:XtvwrStGgqGPLc4cjQfWqZHG1YFdYs6swckp8vpsjnc= +golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= +golang.org/x/lint v0.0.0-20190227174305-5b3e6a55c961/go.mod h1:wehouNa3lNwaWXcvxsM5YxQ5yQlVC4a0KAMCusXpPoU= +golang.org/x/lint v0.0.0-20190313153728-d0100b6bd8b3/go.mod h1:6SW0HCj/g11FgYtHlgUYUwCkIfeOF89ocIRzGO/8vkc= +golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.6.0-dev.0.20220419223038-86c51ed26bb4/go.mod h1:jJ57K6gSWd91VN4djpZkiMVwK6gcyfeH4XE8wZrZaV4= +golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug+ECip1KBveYUHfp+8e9klMJ9c= +golang.org/x/net v0.24.0 h1:1PcaxkF854Fu3+lvBIx5SYn9wRlBzzcnHZSiaFFAb0w= +golang.org/x/net v0.24.0/go.mod h1:2Q7sJY5mzlzWjKtYUEXSlBWCdyaioyXzRB2RtU8KVE8= +golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20220722155255-886fb9371eb4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190916202348-b4ddaad3f8a3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201204225414-ed752295db88/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20220722155257-8c9f86f7a55f/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.8.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.11.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.19.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/sys v0.20.0 h1:Od9JTbYCk261bKm4M/mw7AklTlFYIa0bIp9BgSm1S8Y= +golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA= +golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= +golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= +golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= +golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ= +golang.org/x/text v0.15.0 h1:h1V/4gjBv8v9cjcR6+AR5+/cIYK5N/WAgiv4xlsEtAk= +golang.org/x/text v0.15.0/go.mod h1:18ZOQIKpY8NJVqYksKHtTdi31H5itFRjB5/qKTNYzSU= +golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= +golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= +golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= +golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= +gonum.org/v1/gonum v0.15.0 h1:2lYxjRbTYyxkJxlhC+LvJIx3SsANPdRybu1tGj9/OrQ= +gonum.org/v1/gonum v0.15.0/go.mod h1:xzZVBJBtS+Mz4q0Yl2LJTk+OxOg4jiXZ7qBoM0uISGo= +google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= +google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= +google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= +google.golang.org/genproto v0.0.0-20240227224415-6ceb2ff114de h1:F6qOa9AZTYJXOUEr4jDysRDLrm4PHePlge4v4TGAlxY= +google.golang.org/genproto v0.0.0-20240227224415-6ceb2ff114de/go.mod h1:VUhTRKeHn9wwcdrk73nvdC9gF178Tzhmt/qyaFcPLSo= +google.golang.org/genproto/googleapis/api v0.0.0-20240227224415-6ceb2ff114de h1:jFNzHPIeuzhdRwVhbZdiym9q0ory/xY3sA+v2wPg8I0= +google.golang.org/genproto/googleapis/api v0.0.0-20240227224415-6ceb2ff114de/go.mod h1:5iCWqnniDlqZHrd3neWVTOwvh/v6s3232omMecelax8= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240401170217-c3f982113cda h1:LI5DOvAxUPMv/50agcLLoo+AdWc1irS9Rzz4vPuD1V4= +google.golang.org/genproto/googleapis/rpc v0.0.0-20240401170217-c3f982113cda/go.mod h1:WtryC6hu0hhx87FDGxWCDptyssuo68sk10vYjF+T9fY= +google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= +google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= +google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= +google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= +google.golang.org/grpc v1.63.2 h1:MUeiw1B2maTVZthpU5xvASfTh3LDbxHd6IJ6QQVU+xM= +google.golang.org/grpc v1.63.2/go.mod h1:WAX/8DgncnokcFUldAxq7GeB5DXHDbMF+lLvDomNkRA= +google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= +google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= +google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= +google.golang.org/protobuf v1.20.1-0.20200309200217-e05f789c0967/go.mod h1:A+miEFZTKqfCUM6K7xSMQL9OKL/b6hQv+e19PK+JZNE= +google.golang.org/protobuf v1.21.0/go.mod h1:47Nbq4nVaFHyn7ilMalzfO3qCViNmqZ2kzikPIcrTAo= +google.golang.org/protobuf v1.22.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= +google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= +google.golang.org/protobuf v1.34.1 h1:9ddQBjfCyZPOHPUiPxpYESBLc+T8P3E+Vo4IbKZgFWg= +google.golang.org/protobuf v1.34.1/go.mod h1:c6P6GXX6sHbq/GpV6MGZEdwhWPcYBgnhAHhKbcUYpos= +gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +honnef.co/go/tools v0.0.0-20190523083050-ea95bdfd59fc/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= +nhooyr.io/websocket v1.8.10 h1:mv4p+MnGrLDcPlBoWsvPP7XCzTYMXP9F9eIGoKbgx7Q= +nhooyr.io/websocket v1.8.10/go.mod h1:rN9OFWIUwuxg4fR5tELlYC04bXYowCP9GX47ivo2l+c= diff --git a/collector/components/azureeventhubreceiver/internal/metadata/generated_status.go b/collector/components/azureeventhubreceiver/internal/metadata/generated_status.go new file mode 100644 index 0000000..23cb579 --- /dev/null +++ b/collector/components/azureeventhubreceiver/internal/metadata/generated_status.go @@ -0,0 +1,26 @@ +// Code generated by mdatagen. DO NOT EDIT. + +package metadata + +import ( + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/otel/metric" + "go.opentelemetry.io/otel/trace" +) + +var ( + Type = component.MustNewType("azureeventhub") +) + +const ( + MetricsStability = component.StabilityLevelAlpha + LogsStability = component.StabilityLevelAlpha +) + +func Meter(settings component.TelemetrySettings) metric.Meter { + return settings.MeterProvider.Meter("otelcol/azureeventhubreceiver") +} + +func Tracer(settings component.TelemetrySettings) trace.Tracer { + return settings.TracerProvider.Tracer("otelcol/azureeventhubreceiver") +} diff --git a/collector/components/azureeventhubreceiver/metadata.yaml b/collector/components/azureeventhubreceiver/metadata.yaml new file mode 100644 index 0000000..b090eda --- /dev/null +++ b/collector/components/azureeventhubreceiver/metadata.yaml @@ -0,0 +1,13 @@ +type: azureeventhub + +status: + class: receiver + stability: + alpha: [metrics, logs] + distributions: [contrib, splunk, observiq, sumo] + codeowners: + active: [atoulme, djaglowski] + +tests: + config: + skip_lifecycle: true diff --git a/collector/components/azureeventhubreceiver/persister.go b/collector/components/azureeventhubreceiver/persister.go new file mode 100644 index 0000000..9db7cd0 --- /dev/null +++ b/collector/components/azureeventhubreceiver/persister.go @@ -0,0 +1,103 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "fmt" + + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + jsoniter "github.com/json-iterator/go" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/extension/experimental/storage" + "go.opentelemetry.io/collector/receiver" + + "github.com/open-telemetry/opentelemetry-collector-contrib/pkg/stanza/adapter" +) + +// "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs/checkpoints" +// "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs/checkpoints" + +const ( + // storageKeyFormat the format of the key used to store the checkpoint in the storage. + storageKeyFormat = "%s/%s/%s/%s" + // StartOfStream is a constant defined to represent the start of a partition stream in EventHub. + StartOfStream = "-1" + + // EndOfStream is a constant defined to represent the current end of a partition stream in EventHub. + // This can be used as an offset argument in receiver creation to start receiving from the latest + // event, instead of a specific offset or point in time. + EndOfStream = "@latest" +) + +// The Checkpoint type is now maintained here to eliminate the dependence on the deprecated eventhub SDK for this datatype. +// Preserving the previously used structure and tags keeps the receiver compatible with existing checkpoints and avoids +// any need to migrate data. +type Checkpoint struct { + Offset string `json:"offset"` + SequenceNumber int64 `json:"sequenceNumber"` + EnqueueTime string `json:"enqueueTime"` // ": "0001-01-01T00:00:00Z" +} + +type storageCheckpointPersister struct { + storageClient storage.Client +} + +func (s *storageCheckpointPersister) Write(namespace, name, consumerGroup, partitionID string, checkpoint Checkpoint) error { + b, err := jsoniter.Marshal(checkpoint) + if err != nil { + return err + } + return s.storageClient.Set(context.Background(), fmt.Sprintf(storageKeyFormat, namespace, name, consumerGroup, partitionID), b) +} + +func (s *storageCheckpointPersister) Read(namespace, name, consumerGroup, partitionID string) (Checkpoint, error) { + var checkpoint Checkpoint + bytes, err := s.storageClient.Get(context.Background(), fmt.Sprintf(storageKeyFormat, namespace, name, consumerGroup, partitionID)) + if err != nil { + // error reading checkpoint + return Checkpoint{}, err + } else if len(bytes) == 0 { + // nil or empty checkpoint + return NewCheckpointFromStartOfStream(), nil + } + err = jsoniter.Unmarshal(bytes, &checkpoint) + return checkpoint, err +} + +// wrappers and stubs to implement azeventhubs.CheckpointStore +func (s *storageCheckpointPersister) ClaimOwnership(_ context.Context, _ []azeventhubs.Ownership, _ *azeventhubs.ClaimOwnershipOptions) ([]azeventhubs.Ownership, error) { + return nil, nil +} + +func (s *storageCheckpointPersister) ListCheckpoints(_ context.Context, _ string, _ string, _ string, _ *azeventhubs.ListCheckpointsOptions) ([]azeventhubs.Checkpoint, error) { + return nil, nil +} + +func (s *storageCheckpointPersister) ListOwnership(_ context.Context, _ string, _ string, _ string, _ *azeventhubs.ListOwnershipOptions) ([]azeventhubs.Ownership, error) { + return nil, nil +} + +func (s *storageCheckpointPersister) SetCheckpoint(_ context.Context, _ azeventhubs.Checkpoint, _ *azeventhubs.SetCheckpointOptions) error { + return nil +} + +var _ azeventhubs.CheckpointStore = &storageCheckpointPersister{} + +// NewCheckpointFromStartOfStream returns a checkpoint for the start of the stream +func NewCheckpointFromStartOfStream() Checkpoint { + return Checkpoint{ + Offset: StartOfStream, + } +} + +func createCheckpointStore(ctx context.Context, host component.Host, cfg *Config, s receiver.CreateSettings) (*storageCheckpointPersister, error) { + storageClient, err := adapter.GetStorageClient(ctx, host, cfg.StorageID, s.ID) + if err != nil { + return nil, err + } + return &storageCheckpointPersister{ + storageClient: storageClient, + }, nil +} diff --git a/collector/components/azureeventhubreceiver/persister_test.go b/collector/components/azureeventhubreceiver/persister_test.go new file mode 100644 index 0000000..aa2c52b --- /dev/null +++ b/collector/components/azureeventhubreceiver/persister_test.go @@ -0,0 +1,100 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "errors" + "sync" + "testing" + + "github.com/stretchr/testify/assert" + "go.opentelemetry.io/collector/extension/experimental/storage" +) + +func TestStorageOffsetPersisterUnknownCheckpoint(t *testing.T) { + client := newMockClient() + s := storageCheckpointPersister{storageClient: client} + // check we have no match + checkpoint, err := s.Read("foo", "bar", "foobar", "foobarfoo") + assert.NoError(t, err) + assert.NotNil(t, checkpoint) + assert.Equal(t, "-1", checkpoint.Offset) +} + +func TestStorageOffsetPersisterWithKnownCheckpoint(t *testing.T) { + client := newMockClient() + s := storageCheckpointPersister{storageClient: client} + checkpoint := Checkpoint{ + Offset: "foo", + SequenceNumber: 2, + EnqueueTime: "0001-01-01T00:00:00Z", + } + err := s.Write("foo", "bar", "foobar", "foobarfoo", checkpoint) + assert.NoError(t, err) + read, err := s.Read("foo", "bar", "foobar", "foobarfoo") + assert.NoError(t, err) + assert.Equal(t, checkpoint.Offset, read.Offset) + assert.Equal(t, checkpoint.SequenceNumber, read.SequenceNumber) + assert.True(t, checkpoint.EnqueueTime == read.EnqueueTime) +} + +// copied from pkg/stanza/adapter/mocks_test.go +type mockClient struct { + cache map[string][]byte + cacheMux sync.Mutex +} + +func newMockClient() *mockClient { + return &mockClient{ + cache: make(map[string][]byte), + } +} + +func (p *mockClient) Get(_ context.Context, key string) ([]byte, error) { + p.cacheMux.Lock() + defer p.cacheMux.Unlock() + return p.cache[key], nil +} + +func (p *mockClient) Set(_ context.Context, key string, value []byte) error { + p.cacheMux.Lock() + defer p.cacheMux.Unlock() + p.cache[key] = value + return nil +} + +func (p *mockClient) Delete(_ context.Context, key string) error { + p.cacheMux.Lock() + defer p.cacheMux.Unlock() + delete(p.cache, key) + return nil +} + +func (p *mockClient) Batch(_ context.Context, ops ...storage.Operation) error { + p.cacheMux.Lock() + defer p.cacheMux.Unlock() + + for _, op := range ops { + switch op.Type { + case storage.Get: + op.Value = p.cache[op.Key] + case storage.Set: + p.cache[op.Key] = op.Value + case storage.Delete: + delete(p.cache, op.Key) + default: + return errors.New("wrong operation type") + } + } + + return nil +} + +func (p *mockClient) Close(_ context.Context) error { + p.cacheMux.Lock() + defer p.cacheMux.Unlock() + p.cache = nil + return nil +} diff --git a/collector/components/azureeventhubreceiver/rawlogs_unmarshaler.go b/collector/components/azureeventhubreceiver/rawlogs_unmarshaler.go new file mode 100644 index 0000000..672c8cd --- /dev/null +++ b/collector/components/azureeventhubreceiver/rawlogs_unmarshaler.go @@ -0,0 +1,35 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + // "go.opentelemetry.io/collector/pdata/pcommon" + "go.opentelemetry.io/collector/pdata/plog" + "go.uber.org/zap" +) + +type rawLogsUnmarshaler struct { + logger *zap.Logger +} + +func newRawLogsUnmarshaler(logger *zap.Logger) eventLogsUnmarshaler { + return rawLogsUnmarshaler{ + logger: logger, + } +} + +func (r rawLogsUnmarshaler) UnmarshalLogs(event *azeventhubs.ReceivedEventData) (plog.Logs, error) { + r.logger.Debug("started unmarshaling logs", zap.Any("eventBody", event.Body)) + l := plog.NewLogs() + lr := l.ResourceLogs().AppendEmpty().ScopeLogs().AppendEmpty().LogRecords().AppendEmpty() + slice := lr.Body().SetEmptyBytes() + slice.Append(event.Body...) + if err := lr.Attributes().FromRaw(event.Properties); err != nil { + r.logger.Error("failed extracting attributes from raw event properties", zap.Error(err)) + return l, err + } + r.logger.Debug("successfully unmarshaled logs", zap.Any("logRecords", lr)) + return l, nil +} diff --git a/collector/components/azureeventhubreceiver/receiver.go b/collector/components/azureeventhubreceiver/receiver.go new file mode 100644 index 0000000..adef6f8 --- /dev/null +++ b/collector/components/azureeventhubreceiver/receiver.go @@ -0,0 +1,161 @@ +// Copyright The OpenTelemetry Authors +// SPDX-License-Identifier: Apache-2.0 + +package azureeventhubreceiver // import "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver" + +import ( + "context" + "errors" + "fmt" + + "github.com/Azure/azure-sdk-for-go/sdk/messaging/azeventhubs" + "go.opentelemetry.io/collector/component" + "go.opentelemetry.io/collector/consumer" + "go.opentelemetry.io/collector/pdata/plog" + "go.opentelemetry.io/collector/pdata/pmetric" + "go.opentelemetry.io/collector/receiver" + "go.opentelemetry.io/collector/receiver/receiverhelper" + "go.uber.org/zap" + + "github.com/open-telemetry/opentelemetry-collector-contrib/receiver/azureeventhubreceiver/internal/metadata" +) + +type dataConsumer interface { + consume(ctx context.Context, event *azeventhubs.ReceivedEventData) error + setNextLogsConsumer(nextLogsConsumer consumer.Logs) + setNextMetricsConsumer(nextLogsConsumer consumer.Metrics) +} + +type eventLogsUnmarshaler interface { + UnmarshalLogs(event *azeventhubs.ReceivedEventData) (plog.Logs, error) +} + +type eventMetricsUnmarshaler interface { + UnmarshalMetrics(event *azeventhubs.ReceivedEventData) (pmetric.Metrics, error) +} + +type eventhubReceiver struct { + eventHandler eventHandler + dataType component.Type + logger *zap.Logger + logsUnmarshaler eventLogsUnmarshaler + metricsUnmarshaler eventMetricsUnmarshaler + nextLogsConsumer consumer.Logs + nextMetricsConsumer consumer.Metrics + obsrecv *receiverhelper.ObsReport +} + +func (receiver *eventhubReceiver) Start(ctx context.Context, host component.Host) error { + + err := receiver.eventHandler.run(ctx, host) + return err +} + +func (receiver *eventhubReceiver) Shutdown(ctx context.Context) error { + + return receiver.eventHandler.close(ctx) +} + +func (receiver *eventhubReceiver) setNextLogsConsumer(nextLogsConsumer consumer.Logs) { + + receiver.nextLogsConsumer = nextLogsConsumer +} + +func (receiver *eventhubReceiver) setNextMetricsConsumer(nextMetricsConsumer consumer.Metrics) { + + receiver.nextMetricsConsumer = nextMetricsConsumer +} + +func (receiver *eventhubReceiver) consume(ctx context.Context, event *azeventhubs.ReceivedEventData) error { + + switch receiver.dataType { + case component.DataTypeLogs: + return receiver.consumeLogs(ctx, event) + case component.DataTypeMetrics: + return receiver.consumeMetrics(ctx, event) + case component.DataTypeTraces: + fallthrough + default: + return fmt.Errorf("invalid data type: %v", receiver.dataType) + } +} + +func (receiver *eventhubReceiver) consumeLogs(ctx context.Context, event *azeventhubs.ReceivedEventData) error { + + if receiver.nextLogsConsumer == nil { + return nil + } + + if receiver.logsUnmarshaler == nil { + return errors.New("unable to unmarshal logs with configured format") + } + + logsContext := receiver.obsrecv.StartLogsOp(ctx) + + logs, err := receiver.logsUnmarshaler.UnmarshalLogs(event) + if err != nil { + return fmt.Errorf("failed to unmarshal logs: %w", err) + } + + receiver.logger.Debug("Log Records", zap.Any("logs", logs)) + err = receiver.nextLogsConsumer.ConsumeLogs(logsContext, logs) + receiver.obsrecv.EndLogsOp(logsContext, metadata.Type.String(), 1, err) + + return err +} + +func (receiver *eventhubReceiver) consumeMetrics(ctx context.Context, event *azeventhubs.ReceivedEventData) error { + + if receiver.nextMetricsConsumer == nil { + return nil + } + + if receiver.metricsUnmarshaler == nil { + return errors.New("unable to unmarshal metrics with configured format") + } + + metricsContext := receiver.obsrecv.StartMetricsOp(ctx) + + metrics, err := receiver.metricsUnmarshaler.UnmarshalMetrics(event) + if err != nil { + return fmt.Errorf("failed to unmarshal metrics: %w", err) + } + + receiver.logger.Debug("Metric Records", zap.Any("metrics", metrics)) + err = receiver.nextMetricsConsumer.ConsumeMetrics(metricsContext, metrics) + + receiver.obsrecv.EndMetricsOp(metricsContext, metadata.Type.String(), 1, err) + + return err +} + +func newReceiver( + receiverType component.Type, + logsUnmarshaler eventLogsUnmarshaler, + metricsUnmarshaler eventMetricsUnmarshaler, + eventHandler eventHandler, + settings receiver.CreateSettings, +) (component.Component, error) { + + obsrecv, err := receiverhelper.NewObsReport(receiverhelper.ObsReportSettings{ + ReceiverID: settings.ID, + Transport: "event", + ReceiverCreateSettings: settings, + }) + if err != nil { + return nil, err + } + + eventhubReceiver := &eventhubReceiver{ + dataType: receiverType, + eventHandler: eventHandler, + logger: settings.Logger, + logsUnmarshaler: logsUnmarshaler, + metricsUnmarshaler: metricsUnmarshaler, + obsrecv: obsrecv, + } + + eventHandler.setDataConsumer(eventhubReceiver) + + return eventhubReceiver, nil +} diff --git a/collector/components/azureeventhubreceiver/testdata/config.yaml b/collector/components/azureeventhubreceiver/testdata/config.yaml new file mode 100644 index 0000000..4afddc4 --- /dev/null +++ b/collector/components/azureeventhubreceiver/testdata/config.yaml @@ -0,0 +1,22 @@ +receivers: + azureeventhub: + connection: Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName + + azureeventhub/all: + connection: Endpoint=sb://namespace.servicebus.windows.net/;SharedAccessKeyName=RootManageSharedAccessKey;SharedAccessKey=superSecret1234=;EntityPath=hubName + partition: foo + offset: "1234-5566" + format: "raw" + +processors: + nop: + +exporters: + nop: + +service: + pipelines: + logs: + receivers: [azureeventhub, azureeventhub/all] + processors: [nop] + exporters: [nop] diff --git a/collector/components/azureeventhubreceiver/testdata/log-maximum.json b/collector/components/azureeventhubreceiver/testdata/log-maximum.json new file mode 100644 index 0000000..caef4ec --- /dev/null +++ b/collector/components/azureeventhubreceiver/testdata/log-maximum.json @@ -0,0 +1,31 @@ +{ + "records": [ + { + "time": "2022-11-11T04:48:27.6767145Z", + "resourceId": "/RESOURCE_ID", + "tenantId": "/TENANT_ID", + "operationName": "SecretGet", + "operationVersion": "7.0", + "category": "AuditEvent", + "resultType": "Success", + "resultSignature": "Signature", + "resultDescription": "Description", + "durationMs": "1234", + "callerIpAddress": "127.0.0.1", + "correlationId": "607964b6-41a5-4e24-a5db-db7aab3b9b34", + "Level": "Warning", + "location": "ukso", + "identity": { + "claim": { + "oid": "607964b6-41a5-4e24-a5db-db7aab3b9b34" + } + }, + "properties": { + "string": "string", + "int": 429, + "float": 3.14, + "bool": false + } + } + ] +} diff --git a/collector/components/azureeventhubreceiver/testdata/log-minimum-2.json b/collector/components/azureeventhubreceiver/testdata/log-minimum-2.json new file mode 100644 index 0000000..6eac63f --- /dev/null +++ b/collector/components/azureeventhubreceiver/testdata/log-minimum-2.json @@ -0,0 +1,16 @@ +{ + "records": [ + { + "time": "2022-11-11T04:48:27.6767145Z", + "resourceId": "/RESOURCE_ID", + "operationName": "SecretGet", + "category": "AuditEvent" + }, + { + "time": "2022-11-11T04:48:27.6767145Z", + "resourceId": "/RESOURCE_ID", + "operationName": "SecretGet", + "category": "AuditEvent" + } + ] +} diff --git a/collector/components/azureeventhubreceiver/testdata/log-minimum.json b/collector/components/azureeventhubreceiver/testdata/log-minimum.json new file mode 100644 index 0000000..16d4f2e --- /dev/null +++ b/collector/components/azureeventhubreceiver/testdata/log-minimum.json @@ -0,0 +1,10 @@ +{ + "records": [ + { + "time": "2022-11-11T04:48:27.6767145Z", + "resourceId": "/RESOURCE_ID", + "operationName": "SecretGet", + "category": "AuditEvent" + } + ] +} From 0190b8c5c4ee042c6e78792f23897b04d46d8744 Mon Sep 17 00:00:00 2001 From: Nathan Slaughter <28688390+nslaughter@users.noreply.github.com> Date: Wed, 12 Jun 2024 17:53:02 -0500 Subject: [PATCH 2/2] rm Makefile --- collector/components/azureeventhubreceiver/Makefile | 1 - 1 file changed, 1 deletion(-) delete mode 100644 collector/components/azureeventhubreceiver/Makefile diff --git a/collector/components/azureeventhubreceiver/Makefile b/collector/components/azureeventhubreceiver/Makefile deleted file mode 100644 index ded7a36..0000000 --- a/collector/components/azureeventhubreceiver/Makefile +++ /dev/null @@ -1 +0,0 @@ -include ../../Makefile.Common