From 5d903b5b71f2f1b84fe25568139a4c2b0e4e07ac Mon Sep 17 00:00:00 2001 From: Josh Cummings Date: Mon, 8 May 2023 14:03:23 -0600 Subject: [PATCH] Enforce start happens-before stop Closes gh-13133 --- .../ObservationWebFilterChainDecorator.java | 34 +++++++++++++++---- 1 file changed, 28 insertions(+), 6 deletions(-) diff --git a/web/src/main/java/org/springframework/security/web/server/ObservationWebFilterChainDecorator.java b/web/src/main/java/org/springframework/security/web/server/ObservationWebFilterChainDecorator.java index 12c78b9faf3..5d6f5f528b7 100644 --- a/web/src/main/java/org/springframework/security/web/server/ObservationWebFilterChainDecorator.java +++ b/web/src/main/java/org/springframework/security/web/server/ObservationWebFilterChainDecorator.java @@ -21,6 +21,8 @@ import java.util.ListIterator; import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicReference; +import java.util.concurrent.locks.Lock; +import java.util.concurrent.locks.ReentrantLock; import io.micrometer.common.KeyValue; import io.micrometer.common.KeyValues; @@ -382,6 +384,8 @@ private static final class ObservationReference { private static final ObservationReference NOOP = new ObservationReference(Observation.NOOP); + private final Lock lock = new ReentrantLock(); + private final AtomicInteger state = new AtomicInteger(0); private final Observation observation; @@ -391,20 +395,38 @@ private ObservationReference(Observation observation) { } private void start() { - if (this.state.compareAndSet(0, 1)) { - this.observation.start(); + try { + this.lock.lock(); + if (this.state.compareAndSet(0, 1)) { + this.observation.start(); + } + } + finally { + this.lock.unlock(); } } private void error(Throwable ex) { - if (this.state.get() == 1) { - this.observation.error(ex); + try { + this.lock.lock(); + if (this.state.get() == 1) { + this.observation.error(ex); + } + } + finally { + this.lock.unlock(); } } private void stop() { - if (this.state.compareAndSet(1, 2)) { - this.observation.stop(); + try { + this.lock.lock(); + if (this.state.compareAndSet(1, 2)) { + this.observation.stop(); + } + } + finally { + this.lock.unlock(); } }