From 8e1ec6d03da70efb085db5d627c0e0465b04a0ef Mon Sep 17 00:00:00 2001 From: Kirill Batalin Date: Wed, 11 Dec 2024 19:43:54 +0000 Subject: [PATCH] fix(qos): Use PolledMeter to prevent metric deletion by Garbage Collector (#4810) Depending on the load and configuration, the metric may be updated infrequently, and therefore the real value is deleted. As a result, instead of the value you see N/A. Replaced with PolledMeter to hold a strong reference Co-authored-by: mergify[bot] <37929162+mergify[bot]@users.noreply.github.com> --- .../spinnaker/orca/qos/ExecutionBufferActuator.kt | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/orca-qos/src/main/kotlin/com/netflix/spinnaker/orca/qos/ExecutionBufferActuator.kt b/orca-qos/src/main/kotlin/com/netflix/spinnaker/orca/qos/ExecutionBufferActuator.kt index a0cdce5e8e..8d7b8f1ee4 100644 --- a/orca-qos/src/main/kotlin/com/netflix/spinnaker/orca/qos/ExecutionBufferActuator.kt +++ b/orca-qos/src/main/kotlin/com/netflix/spinnaker/orca/qos/ExecutionBufferActuator.kt @@ -16,6 +16,7 @@ package com.netflix.spinnaker.orca.qos import com.netflix.spectator.api.Registry +import com.netflix.spectator.api.patterns.PolledMeter import com.netflix.spinnaker.kork.dynamicconfig.DynamicConfigService import com.netflix.spinnaker.orca.annotations.Sync import com.netflix.spinnaker.orca.api.pipeline.models.ExecutionStatus.BUFFERED @@ -29,6 +30,7 @@ import net.logstash.logback.argument.StructuredArguments.value import org.slf4j.LoggerFactory import org.springframework.context.event.EventListener import org.springframework.stereotype.Component +import java.util.concurrent.atomic.AtomicLong /** * Determines if an execution should be buffered. @@ -50,6 +52,11 @@ class ExecutionBufferActuator( private val enqueuedId = registry.createId("qos.executionsEnqueued") private val elapsedTimeId = registry.createId("qos.actuator.elapsedTime") + // have to use PolledMeter because an ordinary metric is deleted by Garbage Collector + private val bufferingEnabled = PolledMeter.using(registry) + .withId(bufferingId) + .monitorValue(AtomicLong(0)) + @Sync @EventListener(BeforeInitialExecutionPersist::class) fun beforeInitialPersist(event: BeforeInitialExecutionPersist) { @@ -61,7 +68,7 @@ class ExecutionBufferActuator( val supplierName = bufferStateSupplier.javaClass.simpleName if (bufferStateSupplier.get() == ACTIVE) { - registry.gauge(bufferingId).set(1.0) + bufferingEnabled.set(1) val execution = event.execution withActionDecision(execution) { @@ -83,7 +90,7 @@ class ExecutionBufferActuator( } } } else { - registry.gauge(bufferingId).set(0.0) + bufferingEnabled.set(0) } }