From 1cf7d553cf684af1c1760e7c359a5d00b2964db7 Mon Sep 17 00:00:00 2001 From: Matt Richardson Date: Wed, 27 Nov 2024 09:52:03 +1100 Subject: [PATCH] Add support for OTEL metrics export So we can diagnose dropped spans --- .../opentelemetry/common/PluginConstants.java | 2 + server/build.gradle | 2 + .../endpoints/IOTELEndpointHandler.java | 4 ++ .../custom/CustomOTELEndpointHandler.java | 8 +++ .../HoneycombOTELEndpointHandler.java | 17 ++++++ ...mbProjectConfigurationSettingsRequest.java | 3 ++ .../zipkin/ZipKinOTELEndpointHandler.java | 8 +++ .../HelperPerBuildOTELHelperFactory.java | 6 +-- .../server/helpers/OTELHelperImpl.java | 53 ++++++++++++++++++- ...projectConfigurationSettingsHoneycomb.jspf | 6 +++ .../opentelemetry/server/OTELHelperTest.java | 3 +- .../server/TeamCityBuildListenerTest.java | 3 +- 12 files changed, 109 insertions(+), 6 deletions(-) diff --git a/common/src/main/java/com/octopus/teamcity/opentelemetry/common/PluginConstants.java b/common/src/main/java/com/octopus/teamcity/opentelemetry/common/PluginConstants.java index 7a608eb..4aae058 100644 --- a/common/src/main/java/com/octopus/teamcity/opentelemetry/common/PluginConstants.java +++ b/common/src/main/java/com/octopus/teamcity/opentelemetry/common/PluginConstants.java @@ -13,6 +13,8 @@ private PluginConstants() {} public static final String PROPERTY_KEY_HONEYCOMB_TEAM = "octopus.teamcity.opentelemetry.plugin.honeycomb.team"; public static final String PROPERTY_KEY_HONEYCOMB_DATASET = "octopus.teamcity.opentelemetry.plugin.honeycomb.dataset"; public static final String PROPERTY_KEY_HONEYCOMB_APIKEY = "octopus.teamcity.opentelemetry.plugin.honeycomb.apikey"; + public static final String PROPERTY_KEY_HONEYCOMB_METRICS_ENABLED = "octopus.teamcity.opentelemetry.plugin.honeycomb.metrics.enabled"; + public static final String ATTRIBUTE_SERVICE_NAME = "service_name"; public static final String ATTRIBUTE_NAME = "name"; diff --git a/server/build.gradle b/server/build.gradle index 9ad0610..114879f 100644 --- a/server/build.gradle +++ b/server/build.gradle @@ -18,6 +18,8 @@ dependencies { implementation('io.opentelemetry:opentelemetry-api:1.44.1') implementation('io.opentelemetry:opentelemetry-sdk:1.44.1') implementation('io.opentelemetry:opentelemetry-exporter-otlp:1.44.1') + implementation('io.opentelemetry:opentelemetry-exporter-logging:1.44.1') + implementation 'io.opentelemetry:opentelemetry-sdk-metrics:1.28.0' implementation('io.opentelemetry:opentelemetry-semconv:1.31.0-alpha-SNAPSHOT') implementation('io.opentelemetry:opentelemetry-exporter-zipkin') implementation 'io.grpc:grpc-netty-shaded:1.68.1' diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/IOTELEndpointHandler.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/IOTELEndpointHandler.java index ef9893b..d8eef1c 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/IOTELEndpointHandler.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/IOTELEndpointHandler.java @@ -1,10 +1,12 @@ package com.octopus.teamcity.opentelemetry.server.endpoints; import com.octopus.teamcity.opentelemetry.server.SetProjectConfigurationSettingsRequest; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import jetbrains.buildServer.serverSide.SBuild; import org.springframework.web.servlet.ModelAndView; +import javax.annotation.Nullable; import javax.servlet.http.HttpServletRequest; import java.util.Map; @@ -12,6 +14,8 @@ public interface IOTELEndpointHandler { ModelAndView getBuildOverviewModelAndView(SBuild build, Map params, String traceId); SpanProcessor buildSpanProcessor(String endpoint, Map params); + @Nullable + MetricExporter buildMetricsExporter(String endpoint, Map params); SetProjectConfigurationSettingsRequest getSetProjectConfigurationSettingsRequest(HttpServletRequest request); diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/custom/CustomOTELEndpointHandler.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/custom/CustomOTELEndpointHandler.java index ebdeed8..ca52990 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/custom/CustomOTELEndpointHandler.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/custom/CustomOTELEndpointHandler.java @@ -4,6 +4,7 @@ import com.octopus.teamcity.opentelemetry.server.endpoints.IOTELEndpointHandler; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporter; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporterBuilder; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import io.opentelemetry.sdk.trace.export.BatchSpanProcessor; import io.opentelemetry.sdk.trace.export.SpanExporter; @@ -11,6 +12,7 @@ import jetbrains.buildServer.serverSide.crypt.EncryptUtil; import jetbrains.buildServer.web.openapi.PluginDescriptor; import org.apache.log4j.Logger; +import org.jetbrains.annotations.Nullable; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; @@ -47,6 +49,12 @@ public SpanProcessor buildSpanProcessor(String endpoint, Map par return buildGrpcSpanProcessor(headers, endpoint); } + @Nullable + @Override + public MetricExporter buildMetricsExporter(String endpoint, Map params) { + return null; + } + @Override public SetProjectConfigurationSettingsRequest getSetProjectConfigurationSettingsRequest(HttpServletRequest request) { return new SetCustomProjectConfigurationSettingsRequest(request); diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/HoneycombOTELEndpointHandler.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/HoneycombOTELEndpointHandler.java index 7e998a8..665f6cb 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/HoneycombOTELEndpointHandler.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/HoneycombOTELEndpointHandler.java @@ -2,8 +2,10 @@ import com.octopus.teamcity.opentelemetry.server.*; import com.octopus.teamcity.opentelemetry.server.endpoints.IOTELEndpointHandler; +import io.opentelemetry.exporter.otlp.metrics.OtlpGrpcMetricExporter; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporter; import io.opentelemetry.exporter.otlp.trace.OtlpGrpcSpanExporterBuilder; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import io.opentelemetry.sdk.trace.export.BatchSpanProcessor; import io.opentelemetry.sdk.trace.export.SpanExporter; @@ -15,6 +17,7 @@ import org.jetbrains.annotations.NotNull; import org.springframework.web.servlet.ModelAndView; +import javax.annotation.Nullable; import javax.servlet.http.HttpServletRequest; import java.util.ArrayList; import java.util.Date; @@ -62,6 +65,19 @@ public SpanProcessor buildSpanProcessor(String endpoint, Map par return buildGrpcSpanProcessor(headers, endpoint); } + @Override + @Nullable + public MetricExporter buildMetricsExporter(String endpoint, Map params) { + if (params.getOrDefault(PROPERTY_KEY_HONEYCOMB_METRICS_ENABLED, "false").equals("true")) { + return OtlpGrpcMetricExporter.builder() + .setEndpoint(endpoint) + .addHeader("x-honeycomb-team", EncryptUtil.unscramble(params.get(PROPERTY_KEY_HONEYCOMB_APIKEY))) + .addHeader("x-honeycomb-dataset", params.get(PROPERTY_KEY_HONEYCOMB_DATASET)) + .build(); + } + return null; + } + private SpanProcessor buildGrpcSpanProcessor(Map headers, String exporterEndpoint) { OtlpGrpcSpanExporterBuilder spanExporterBuilder = OtlpGrpcSpanExporter.builder(); @@ -85,6 +101,7 @@ public void mapParamsToModel(Map params, Map mod model.put("otelEndpoint", params.get(PROPERTY_KEY_ENDPOINT)); model.put("otelHoneycombTeam", params.get(PROPERTY_KEY_HONEYCOMB_TEAM)); model.put("otelHoneycombDataset", params.get(PROPERTY_KEY_HONEYCOMB_DATASET)); + model.put("otelHoneycombMetricsEnabled", params.get(PROPERTY_KEY_HONEYCOMB_METRICS_ENABLED)); if (params.get(PROPERTY_KEY_HONEYCOMB_APIKEY) == null) { model.put("otelHoneycombApiKey", null); } diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/SetHoneycombProjectConfigurationSettingsRequest.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/SetHoneycombProjectConfigurationSettingsRequest.java index 5398e95..b20f6de 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/SetHoneycombProjectConfigurationSettingsRequest.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/honeycomb/SetHoneycombProjectConfigurationSettingsRequest.java @@ -17,11 +17,13 @@ public class SetHoneycombProjectConfigurationSettingsRequest extends SetProjectC public final String honeycombTeam; public final String honeycombDataset; public final String honeycombApiKey; + public final String honeycombMetricsEnabled; public SetHoneycombProjectConfigurationSettingsRequest(HttpServletRequest request) { super(request); this.honeycombTeam = request.getParameter("honeycombTeam"); this.honeycombDataset = request.getParameter("honeycombDataset"); + this.honeycombMetricsEnabled = request.getParameter("honeycombMetricsEnabled"); this.honeycombApiKey = RSACipher.decryptWebRequestData(request.getParameter("encryptedHoneycombApiKey")); } @@ -39,6 +41,7 @@ protected void serviceSpecificValidate(ActionErrors errors) { protected void mapServiceSpecificParams(HashMap params, ArrayList headers) { params.put(PROPERTY_KEY_HONEYCOMB_DATASET, honeycombDataset); params.put(PROPERTY_KEY_HONEYCOMB_TEAM, honeycombTeam); + params.put(PROPERTY_KEY_HONEYCOMB_METRICS_ENABLED, honeycombMetricsEnabled); params.put(PROPERTY_KEY_HONEYCOMB_APIKEY, EncryptUtil.scramble(honeycombApiKey)); } } diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/zipkin/ZipKinOTELEndpointHandler.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/zipkin/ZipKinOTELEndpointHandler.java index 316cd36..1dc5c33 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/zipkin/ZipKinOTELEndpointHandler.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/endpoints/zipkin/ZipKinOTELEndpointHandler.java @@ -3,11 +3,13 @@ import com.octopus.teamcity.opentelemetry.server.SetProjectConfigurationSettingsRequest; import com.octopus.teamcity.opentelemetry.server.endpoints.IOTELEndpointHandler; import io.opentelemetry.exporter.zipkin.ZipkinSpanExporter; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import io.opentelemetry.sdk.trace.export.BatchSpanProcessor; import jetbrains.buildServer.serverSide.SBuild; import jetbrains.buildServer.web.openapi.PluginDescriptor; import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; @@ -38,6 +40,12 @@ public SpanProcessor buildSpanProcessor(String endpoint, Map par return buildZipkinSpanProcessor(endpoint); } + @Nullable + @Override + public MetricExporter buildMetricsExporter(String endpoint, Map params) { + return null; + } + private SpanProcessor buildZipkinSpanProcessor(String exporterEndpoint) { String endpoint = String.format("%s/api/v2/spans", exporterEndpoint); ZipkinSpanExporter zipkinExporter = ZipkinSpanExporter.builder() diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/HelperPerBuildOTELHelperFactory.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/HelperPerBuildOTELHelperFactory.java index 02c53ee..8d58409 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/HelperPerBuildOTELHelperFactory.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/HelperPerBuildOTELHelperFactory.java @@ -43,13 +43,13 @@ public OTELHelper getOTELHelper(BuildPromotion build) { var params = feature.getParameters(); if (params.get(PROPERTY_KEY_ENABLED).equals("true")) { var endpoint = params.get(PROPERTY_KEY_ENDPOINT); - SpanProcessor spanProcessor; var otelHandler = otelEndpointFactory.getOTELEndpointHandler(params.get(PROPERTY_KEY_SERVICE)); - spanProcessor = otelHandler.buildSpanProcessor(endpoint, params); + var spanProcessor = otelHandler.buildSpanProcessor(endpoint, params); + var metricsExporter = otelHandler.buildMetricsExporter(endpoint, params); long startTime = System.nanoTime(); - var otelHelper = new OTELHelperImpl(spanProcessor, String.valueOf(buildId)); + var otelHelper = new OTELHelperImpl(spanProcessor, metricsExporter, String.valueOf(buildId)); long endTime = System.nanoTime(); long duration = (endTime - startTime); diff --git a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/OTELHelperImpl.java b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/OTELHelperImpl.java index 9d87b58..4597b60 100644 --- a/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/OTELHelperImpl.java +++ b/server/src/main/java/com/octopus/teamcity/opentelemetry/server/helpers/OTELHelperImpl.java @@ -1,21 +1,29 @@ package com.octopus.teamcity.opentelemetry.server.helpers; import com.octopus.teamcity.opentelemetry.common.PluginConstants; +import io.opentelemetry.api.GlobalOpenTelemetry; import io.opentelemetry.api.OpenTelemetry; import io.opentelemetry.api.common.Attributes; +import io.opentelemetry.api.metrics.Meter; import io.opentelemetry.api.trace.Span; import io.opentelemetry.api.trace.Tracer; import io.opentelemetry.api.trace.propagation.W3CTraceContextPropagator; import io.opentelemetry.context.Context; import io.opentelemetry.context.propagation.ContextPropagators; +import io.opentelemetry.exporter.logging.LoggingMetricExporter; import io.opentelemetry.sdk.OpenTelemetrySdk; +import io.opentelemetry.sdk.metrics.SdkMeterProvider; +import io.opentelemetry.sdk.metrics.export.MetricExporter; +import io.opentelemetry.sdk.metrics.export.PeriodicMetricReader; import io.opentelemetry.sdk.resources.Resource; import io.opentelemetry.sdk.trace.SdkTracerProvider; import io.opentelemetry.sdk.trace.SpanProcessor; import io.opentelemetry.semconv.resource.attributes.ResourceAttributes; import org.apache.log4j.Logger; +import org.jetbrains.annotations.NotNull; import javax.annotation.Nullable; +import java.time.Duration; import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.TimeUnit; @@ -27,10 +35,16 @@ public class OTELHelperImpl implements OTELHelper { private final SdkTracerProvider sdkTracerProvider; private final String helperName; - public OTELHelperImpl(SpanProcessor spanProcessor, String helperName) { + public OTELHelperImpl( + @NotNull SpanProcessor spanProcessor, + @Nullable MetricExporter metricExporter, + @NotNull String helperName) { this.helperName = helperName; Resource serviceNameResource = Resource .create(Attributes.of(ResourceAttributes.SERVICE_NAME, PluginConstants.SERVICE_NAME)); + + configureMetricsExport(metricExporter, serviceNameResource); + this.sdkTracerProvider = SdkTracerProvider.builder() .setResource(Resource.getDefault().merge(serviceNameResource)) .addSpanProcessor(spanProcessor) @@ -43,6 +57,30 @@ public OTELHelperImpl(SpanProcessor spanProcessor, String helperName) { this.spanMap = new ConcurrentHashMap<>(); } + private static void configureMetricsExport(@Nullable MetricExporter metricExporter, Resource serviceNameResource) { + var loggingMetricExporter = LoggingMetricExporter.create(); + var meterProviderBuilder = SdkMeterProvider.builder() + .setResource(Resource.getDefault().merge(serviceNameResource)) + .registerMetricReader(PeriodicMetricReader.builder(loggingMetricExporter) + .setInterval(Duration.ofSeconds(10)) + .build()); + if (metricExporter != null) { + meterProviderBuilder + .registerMetricReader(PeriodicMetricReader.builder(metricExporter) + .setInterval(Duration.ofSeconds(10)) + .build()); + } + var sdkMeterProvider = meterProviderBuilder.build(); + var globalOpenTelemetry = OpenTelemetrySdk.builder() + .setMeterProvider(sdkMeterProvider) + .build(); + + GlobalOpenTelemetry.set(globalOpenTelemetry); + + // Shutdown hooks to close resources properly + Runtime.getRuntime().addShutdownHook(new Thread(sdkMeterProvider::close)); + } + @Override public boolean isReady() { return this.openTelemetry != null && this.tracer != null && this.spanMap != null; @@ -56,6 +94,7 @@ public Span getOrCreateParentSpan(String buildId) { @Override public Span createSpan(String spanName, Span parentSpan, String parentSpanName) { LOG.info("Creating child span " + spanName + " under parent " + parentSpanName); + IncrementCounter("spans-created", "Number of spans created"); return this.spanMap.computeIfAbsent(spanName, key -> this.tracer .spanBuilder(spanName) .setParent(Context.current().with(parentSpan)) @@ -64,6 +103,7 @@ public Span createSpan(String spanName, Span parentSpan, String parentSpanName) @Override public Span createTransientSpan(String spanName, Span parentSpan, long startTime) { + IncrementCounter("spans-created", "Number of spans created"); return this.tracer.spanBuilder(spanName) .setParent(Context.current().with(parentSpan)) .setStartTimestamp(startTime, TimeUnit.MILLISECONDS) @@ -94,4 +134,15 @@ public void release(String helperName) { this.sdkTracerProvider.close(); this.spanMap.clear(); } + + public void IncrementCounter(String counterName, String counterDescription) + { + //likely temporary while I confirm that metrics is working as expected + Meter meter = openTelemetry.getMeter("teamcity-opentelemetry-plugin"); + meter.counterBuilder(counterName) + .setDescription(counterDescription) + .setUnit("1") + .build() + .add(1); + } } diff --git a/server/src/main/resources/buildServerResources/projectConfigurationSettingsHoneycomb.jspf b/server/src/main/resources/buildServerResources/projectConfigurationSettingsHoneycomb.jspf index d79a488..b990f99 100644 --- a/server/src/main/resources/buildServerResources/projectConfigurationSettingsHoneycomb.jspf +++ b/server/src/main/resources/buildServerResources/projectConfigurationSettingsHoneycomb.jspf @@ -19,3 +19,9 @@ +style="display: none"> + + +   + + diff --git a/server/src/test/java/com/octopus/teamcity/opentelemetry/server/OTELHelperTest.java b/server/src/test/java/com/octopus/teamcity/opentelemetry/server/OTELHelperTest.java index a681496..84a1149 100644 --- a/server/src/test/java/com/octopus/teamcity/opentelemetry/server/OTELHelperTest.java +++ b/server/src/test/java/com/octopus/teamcity/opentelemetry/server/OTELHelperTest.java @@ -6,6 +6,7 @@ import io.opentelemetry.api.OpenTelemetry; import io.opentelemetry.api.trace.Span; import io.opentelemetry.api.trace.Tracer; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import jetbrains.buildServer.serverSide.BuildPromotion; import jetbrains.buildServer.serverSide.SRunningBuild; @@ -32,7 +33,7 @@ class OTELHelperTest { @BeforeEach void setUp() { GlobalOpenTelemetry.resetForTest(); - this.otelHelper = new OTELHelperImpl(mock(SpanProcessor.class, RETURNS_DEEP_STUBS), "helperNamr"); + this.otelHelper = new OTELHelperImpl(mock(SpanProcessor.class, RETURNS_DEEP_STUBS), mock(MetricExporter.class, RETURNS_DEEP_STUBS), "helperNamr"); } diff --git a/server/src/test/java/com/octopus/teamcity/opentelemetry/server/TeamCityBuildListenerTest.java b/server/src/test/java/com/octopus/teamcity/opentelemetry/server/TeamCityBuildListenerTest.java index edab60a..e420c6e 100644 --- a/server/src/test/java/com/octopus/teamcity/opentelemetry/server/TeamCityBuildListenerTest.java +++ b/server/src/test/java/com/octopus/teamcity/opentelemetry/server/TeamCityBuildListenerTest.java @@ -5,6 +5,7 @@ import com.octopus.teamcity.opentelemetry.server.helpers.OTELHelperImpl; import io.opentelemetry.api.GlobalOpenTelemetry; import io.opentelemetry.api.trace.Span; +import io.opentelemetry.sdk.metrics.export.MetricExporter; import io.opentelemetry.sdk.trace.SpanProcessor; import jetbrains.buildServer.serverSide.BuildPromotion; import jetbrains.buildServer.serverSide.BuildServerListener; @@ -33,7 +34,7 @@ class TeamCityBuildListenerTest { @BeforeEach void setUp(@Mock EventDispatcher buildServerListenerEventDispatcher) { GlobalOpenTelemetry.resetForTest(); - this.otelHelper = new OTELHelperImpl(mock(SpanProcessor.class, RETURNS_DEEP_STUBS), "helper"); + this.otelHelper = new OTELHelperImpl(mock(SpanProcessor.class, RETURNS_DEEP_STUBS), mock(MetricExporter.class, RETURNS_DEEP_STUBS), "helper"); this.factory = mock(OTELHelperFactory.class, RETURNS_DEEP_STUBS); var buildStorageManager = mock(BuildStorageManager.class, RETURNS_DEEP_STUBS);