From 18e8fc53fa08e8a65ca2b66d9c3fa0da47ca6a54 Mon Sep 17 00:00:00 2001 From: Yashwant Date: Mon, 19 Aug 2024 19:22:27 +0530 Subject: [PATCH 1/5] removing high cardinality metrics --- .../extensions/MetricViewConfiguration.java | 34 +++++++++++++++++++ .../config/HypertraceAgentConfiguration.java | 10 ++++++ 2 files changed, 44 insertions(+) create mode 100644 otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java diff --git a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java new file mode 100644 index 00000000..2dc47627 --- /dev/null +++ b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java @@ -0,0 +1,34 @@ +package org.hypertrace.agent.otel.extensions; + +import io.opentelemetry.sdk.metrics.View; +import java.util.Arrays; +import java.util.HashSet; +import java.util.Set; + +public class MetricViewConfiguration { + + public static View createHttpServerDurationView() { + // Attributes to exclude + Set excludedAttributes = + new HashSet<>( + Arrays.asList( + "net.sock.peer.addr", + "net.sock.peer.port", + "http.user_agent", + "enduser.id", + "http.client_ip")); + + // Build the view + return View.builder() + .setAttributeFilter( + attributes -> { + for (String attribute : excludedAttributes) { + if (attributes.contains(attribute)) { + return false; + } + } + return true; + }) + .build(); + } +} diff --git a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java index 3262d0ba..ff8a314a 100644 --- a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java +++ b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java @@ -20,6 +20,8 @@ import com.google.common.annotations.VisibleForTesting; import io.opentelemetry.sdk.autoconfigure.spi.AutoConfigurationCustomizer; import io.opentelemetry.sdk.autoconfigure.spi.AutoConfigurationCustomizerProvider; +import io.opentelemetry.sdk.metrics.InstrumentSelector; +import io.opentelemetry.sdk.metrics.View; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -28,6 +30,7 @@ import org.hypertrace.agent.config.v1.Config.MetricReporterType; import org.hypertrace.agent.config.v1.Config.PropagationFormat; import org.hypertrace.agent.config.v1.Config.TraceReporterType; +import org.hypertrace.agent.otel.extensions.MetricViewConfiguration; import org.slf4j.Logger; import org.slf4j.LoggerFactory; @@ -113,5 +116,12 @@ public static String toOtelPropagators(List propagationFormat @Override public void customize(AutoConfigurationCustomizer autoConfigurationCustomizer) { autoConfigurationCustomizer.addPropertiesSupplier(this::getProperties); + autoConfigurationCustomizer.addMeterProviderCustomizer( + (builder, config) -> { + View httpServerDurationView = MetricViewConfiguration.createHttpServerDurationView(); + builder.registerView( + InstrumentSelector.builder().setName("*").build(), httpServerDurationView); + return builder; + }); } } From 33ecf58330da18e7a697dfb66481bfd6c3e902a2 Mon Sep 17 00:00:00 2001 From: Yashwant Date: Mon, 19 Aug 2024 19:28:03 +0530 Subject: [PATCH 2/5] nit fix --- .../extensions/MetricViewConfiguration.java | 18 +++++++++++++++++- .../config/HypertraceAgentConfiguration.java | 2 +- 2 files changed, 18 insertions(+), 2 deletions(-) diff --git a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java index 2dc47627..9bac8c18 100644 --- a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java +++ b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java @@ -1,3 +1,19 @@ +/* + * Copyright The Hypertrace Authors + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + package org.hypertrace.agent.otel.extensions; import io.opentelemetry.sdk.metrics.View; @@ -7,7 +23,7 @@ public class MetricViewConfiguration { - public static View createHttpServerDurationView() { + public static View createView() { // Attributes to exclude Set excludedAttributes = new HashSet<>( diff --git a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java index ff8a314a..955f1098 100644 --- a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java +++ b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/config/HypertraceAgentConfiguration.java @@ -118,7 +118,7 @@ public void customize(AutoConfigurationCustomizer autoConfigurationCustomizer) { autoConfigurationCustomizer.addPropertiesSupplier(this::getProperties); autoConfigurationCustomizer.addMeterProviderCustomizer( (builder, config) -> { - View httpServerDurationView = MetricViewConfiguration.createHttpServerDurationView(); + View httpServerDurationView = MetricViewConfiguration.createView(); builder.registerView( InstrumentSelector.builder().setName("*").build(), httpServerDurationView); return builder; From 6f91389e8ff91409517ca7b6dffffbc76aa7c672 Mon Sep 17 00:00:00 2001 From: Yashwant Date: Mon, 19 Aug 2024 19:51:53 +0530 Subject: [PATCH 3/5] removing content_length as well --- .../agent/otel/extensions/MetricViewConfiguration.java | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java index 9bac8c18..079563fa 100644 --- a/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java +++ b/otel-extensions/src/main/java/org/hypertrace/agent/otel/extensions/MetricViewConfiguration.java @@ -32,7 +32,10 @@ public static View createView() { "net.sock.peer.port", "http.user_agent", "enduser.id", - "http.client_ip")); + "http.client_ip", + "http.request_content_length", + "http.response_content_length", + "user_agent.original")); // Build the view return View.builder() From 637d83f7d0e39aa915965e8b9d38dd8c5e038feb Mon Sep 17 00:00:00 2001 From: Yashwant Date: Tue, 3 Sep 2024 17:25:34 +0530 Subject: [PATCH 4/5] adding smoke test --- .../agent/smoketest/AbstractSmokeTest.java | 37 +++++++++++++++++++ .../agent/smoketest/SpringBootSmokeTest.java | 6 +++ 2 files changed, 43 insertions(+) diff --git a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java index 6360902c..671edf4a 100644 --- a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java +++ b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java @@ -20,11 +20,14 @@ import com.google.protobuf.util.JsonFormat; import io.opentelemetry.proto.collector.metrics.v1.ExportMetricsServiceRequest; import io.opentelemetry.proto.collector.trace.v1.ExportTraceServiceRequest; +import io.opentelemetry.proto.common.v1.KeyValue; +import io.opentelemetry.proto.metrics.v1.Metric; import io.opentelemetry.proto.trace.v1.ScopeSpans; import io.opentelemetry.proto.trace.v1.Span; import java.io.IOException; import java.time.Duration; import java.util.Collection; +import java.util.List; import java.util.concurrent.TimeUnit; import java.util.concurrent.atomic.AtomicLong; import java.util.jar.Attributes; @@ -244,6 +247,40 @@ protected boolean hasMetricNamed( .anyMatch(metric -> metric.getName().equals(metricName)); } + // Checks if a metric with the given name contains the specified attribute + protected boolean hasMetricWithAttribute( + String metricName, String attributeName, Collection metricRequests) { + + return metricRequests.stream() + .flatMap(metricRequest -> metricRequest.getResourceMetricsList().stream()) + .flatMap(resourceMetrics -> resourceMetrics.getScopeMetricsList().stream()) + .flatMap(scopeMetrics -> scopeMetrics.getMetricsList().stream()) + .filter(metric -> metric.getName().equals(metricName)) + .anyMatch(metric -> metricHasAttribute(metric, attributeName)); + } + + private boolean metricHasAttribute(Metric metric, String attributeName) { + switch (metric.getDataCase()) { + case GAUGE: + return metric.getGauge().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + case SUM: + return metric.getSum().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + case HISTOGRAM: + return metric.getHistogram().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + case EXPONENTIAL_HISTOGRAM: + return metric.getExponentialHistogram().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + case SUMMARY: + return metric.getSummary().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + default: + return false; + } + } + + private boolean hasAttribute(List attributes, String attributeName) { + return attributes.stream() + .anyMatch(attribute -> attribute.getKey().equals(attributeName)); + } + public static String getPropertyOrEnv(String propName) { String property = System.getProperty(propName); if (property != null && !property.isEmpty()) { diff --git a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java index eb4c2f15..c462864b 100644 --- a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java +++ b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java @@ -174,6 +174,12 @@ public void postJson() throws IOException, InterruptedException { Assertions.assertTrue(hasMetricNamed("processedSpans", metrics)); Assertions.assertTrue(hasMetricNamed("queueSize", metrics)); Assertions.assertTrue(hasMetricNamed("http.server.duration", metrics)); + + Assertions.assertTrue(hasMetricWithAttribute("http.server.duration", "http.method", metrics)); + Assertions.assertTrue(hasMetricWithAttribute("http.server.duration", "http.route", metrics)); + Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "net.sock.peer.addr", metrics)); + Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "http.request_content_length", metrics)); + Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "http.response_content_length", metrics)); } @Test From 71e4c787991e75f5e5a8a15f0dd353cf913a7785 Mon Sep 17 00:00:00 2001 From: Yashwant Date: Tue, 3 Sep 2024 17:33:59 +0530 Subject: [PATCH 5/5] spotless apply --- .../agent/smoketest/AbstractSmokeTest.java | 32 +++++++++++-------- .../agent/smoketest/SpringBootSmokeTest.java | 9 ++++-- 2 files changed, 25 insertions(+), 16 deletions(-) diff --git a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java index 671edf4a..19693b8b 100644 --- a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java +++ b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/AbstractSmokeTest.java @@ -249,36 +249,42 @@ protected boolean hasMetricNamed( // Checks if a metric with the given name contains the specified attribute protected boolean hasMetricWithAttribute( - String metricName, String attributeName, Collection metricRequests) { + String metricName, + String attributeName, + Collection metricRequests) { return metricRequests.stream() - .flatMap(metricRequest -> metricRequest.getResourceMetricsList().stream()) - .flatMap(resourceMetrics -> resourceMetrics.getScopeMetricsList().stream()) - .flatMap(scopeMetrics -> scopeMetrics.getMetricsList().stream()) - .filter(metric -> metric.getName().equals(metricName)) - .anyMatch(metric -> metricHasAttribute(metric, attributeName)); + .flatMap(metricRequest -> metricRequest.getResourceMetricsList().stream()) + .flatMap(resourceMetrics -> resourceMetrics.getScopeMetricsList().stream()) + .flatMap(scopeMetrics -> scopeMetrics.getMetricsList().stream()) + .filter(metric -> metric.getName().equals(metricName)) + .anyMatch(metric -> metricHasAttribute(metric, attributeName)); } private boolean metricHasAttribute(Metric metric, String attributeName) { switch (metric.getDataCase()) { case GAUGE: - return metric.getGauge().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + return metric.getGauge().getDataPointsList().stream() + .anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); case SUM: - return metric.getSum().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + return metric.getSum().getDataPointsList().stream() + .anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); case HISTOGRAM: - return metric.getHistogram().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + return metric.getHistogram().getDataPointsList().stream() + .anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); case EXPONENTIAL_HISTOGRAM: - return metric.getExponentialHistogram().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + return metric.getExponentialHistogram().getDataPointsList().stream() + .anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); case SUMMARY: - return metric.getSummary().getDataPointsList().stream().anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); + return metric.getSummary().getDataPointsList().stream() + .anyMatch(dataPoint -> hasAttribute(dataPoint.getAttributesList(), attributeName)); default: return false; } } private boolean hasAttribute(List attributes, String attributeName) { - return attributes.stream() - .anyMatch(attribute -> attribute.getKey().equals(attributeName)); + return attributes.stream().anyMatch(attribute -> attribute.getKey().equals(attributeName)); } public static String getPropertyOrEnv(String propName) { diff --git a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java index c462864b..dc76f878 100644 --- a/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java +++ b/smoke-tests/src/test/java/org/hypertrace/agent/smoketest/SpringBootSmokeTest.java @@ -177,9 +177,12 @@ public void postJson() throws IOException, InterruptedException { Assertions.assertTrue(hasMetricWithAttribute("http.server.duration", "http.method", metrics)); Assertions.assertTrue(hasMetricWithAttribute("http.server.duration", "http.route", metrics)); - Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "net.sock.peer.addr", metrics)); - Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "http.request_content_length", metrics)); - Assertions.assertFalse(hasMetricWithAttribute("http.server.duration", "http.response_content_length", metrics)); + Assertions.assertFalse( + hasMetricWithAttribute("http.server.duration", "net.sock.peer.addr", metrics)); + Assertions.assertFalse( + hasMetricWithAttribute("http.server.duration", "http.request_content_length", metrics)); + Assertions.assertFalse( + hasMetricWithAttribute("http.server.duration", "http.response_content_length", metrics)); } @Test