From 2ea292cfbae6a617a3de82819b0f3619a275270e Mon Sep 17 00:00:00 2001 From: Stuart McCulloch Date: Wed, 18 Dec 2024 18:23:12 +0000 Subject: [PATCH 1/5] Introduce a shared context component, independent of tracing. Co-authored-by: Bruce Bujon --- components/context/build.gradle.kts | 9 + .../main/java/datadog/context/Context.java | 100 ++++++++ .../java/datadog/context/ContextBinder.java | 34 +++ .../main/java/datadog/context/ContextKey.java | 47 ++++ .../java/datadog/context/ContextManager.java | 50 ++++ .../datadog/context/ContextProviders.java | 16 ++ .../java/datadog/context/ContextScope.java | 12 + .../java/datadog/context/EmptyContext.java | 16 ++ .../datadog/context/ImplicitContextKeyed.java | 12 + .../java/datadog/context/IndexedContext.java | 46 ++++ .../datadog/context/SingletonContext.java | 51 ++++ .../context/ThreadLocalContextManager.java | 56 +++++ .../datadog/context/WeakMapContextBinder.java | 29 +++ .../datadog/context/ContextBinderTest.java | 22 ++ .../java/datadog/context/ContextKeyTest.java | 34 +++ .../datadog/context/ContextManagerTest.java | 219 ++++++++++++++++++ .../context/ContextProviderForkedTest.java | 86 +++++++ .../java/datadog/context/ContextTest.java | 113 +++++++++ settings.gradle | 2 + 19 files changed, 954 insertions(+) create mode 100644 components/context/build.gradle.kts create mode 100644 components/context/src/main/java/datadog/context/Context.java create mode 100644 components/context/src/main/java/datadog/context/ContextBinder.java create mode 100644 components/context/src/main/java/datadog/context/ContextKey.java create mode 100644 components/context/src/main/java/datadog/context/ContextManager.java create mode 100644 components/context/src/main/java/datadog/context/ContextProviders.java create mode 100644 components/context/src/main/java/datadog/context/ContextScope.java create mode 100644 components/context/src/main/java/datadog/context/EmptyContext.java create mode 100644 components/context/src/main/java/datadog/context/ImplicitContextKeyed.java create mode 100644 components/context/src/main/java/datadog/context/IndexedContext.java create mode 100644 components/context/src/main/java/datadog/context/SingletonContext.java create mode 100644 components/context/src/main/java/datadog/context/ThreadLocalContextManager.java create mode 100644 components/context/src/main/java/datadog/context/WeakMapContextBinder.java create mode 100644 components/context/src/test/java/datadog/context/ContextBinderTest.java create mode 100644 components/context/src/test/java/datadog/context/ContextKeyTest.java create mode 100644 components/context/src/test/java/datadog/context/ContextManagerTest.java create mode 100644 components/context/src/test/java/datadog/context/ContextProviderForkedTest.java create mode 100644 components/context/src/test/java/datadog/context/ContextTest.java diff --git a/components/context/build.gradle.kts b/components/context/build.gradle.kts new file mode 100644 index 00000000000..4dca7fc3036 --- /dev/null +++ b/components/context/build.gradle.kts @@ -0,0 +1,9 @@ +plugins { + id("me.champeau.jmh") +} + +apply(from = "$rootDir/gradle/java.gradle") + +jmh { + version = "1.28" +} diff --git a/components/context/src/main/java/datadog/context/Context.java b/components/context/src/main/java/datadog/context/Context.java new file mode 100644 index 00000000000..fcdc4d418d1 --- /dev/null +++ b/components/context/src/main/java/datadog/context/Context.java @@ -0,0 +1,100 @@ +package datadog.context; + +import static datadog.context.ContextProviders.binder; +import static datadog.context.ContextProviders.manager; + +import javax.annotation.Nullable; + +/** Immutable context scoped to an execution unit or carrier object. */ +public interface Context { + + /** Returns the root context. */ + static Context root() { + return manager().root(); + } + + /** + * Returns the context attached to the current execution unit. + * + * @return Attached context; {@link #root()} if there is none + */ + static Context current() { + return manager().current(); + } + + /** + * Attaches this context to the current execution unit. + * + * @return Scope to be closed when the context is invalid. + */ + default ContextScope attach() { + return manager().attach(this); + } + + /** + * Swaps this context with the one attached to current execution unit. + * + * @return Previously attached context; {@link #root()} if there was none + */ + default Context swap() { + return manager().swap(this); + } + + /** + * Detaches the context attached to the current execution unit, leaving it context-less. + * + *

WARNING: prefer {@link ContextScope#close()} to properly restore the surrounding context. + * + * @return Previously attached context; {@link #root()} if there was none + */ + static Context detach() { + return manager().detach(); + } + + /** + * Returns the context attached to the given carrier object. + * + * @return Attached context; {@link #root()} if there is none + */ + static Context from(Object carrier) { + return binder().from(carrier); + } + + /** Attaches this context to the given carrier object. */ + default void attachTo(Object carrier) { + binder().attachTo(carrier, this); + } + + /** + * Detaches the context attached to the given carrier object, leaving it context-less. + * + * @return Previously attached context; {@link #root()} if there was none + */ + static Context detachFrom(Object carrier) { + return binder().detachFrom(carrier); + } + + /** + * Gets the value stored in this context under the given key. + * + * @return Value stored under the key; {@code null} if there is no value. + */ + @Nullable + T get(ContextKey key); + + /** + * Creates a new context with the given key-value mapping. + * + * @return New context with the key-value mapping. + */ + Context with(ContextKey key, T value); + + /** + * Creates a new context with a value that has its own implicit key. + * + * @return New context with the implicitly keyed value. + */ + default Context with(ImplicitContextKeyed value) { + return value.storeInto(this); + } +} diff --git a/components/context/src/main/java/datadog/context/ContextBinder.java b/components/context/src/main/java/datadog/context/ContextBinder.java new file mode 100644 index 00000000000..bfcc85a887a --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextBinder.java @@ -0,0 +1,34 @@ +package datadog.context; + +/** Binds context to carrier objects. */ +public interface ContextBinder { + + /** + * Returns the context attached to the given carrier object. + * + * @return Attached context; {@link Context#root()} if there is none + */ + Context from(Object carrier); + + /** Attaches the given context to the given carrier object. */ + void attachTo(Object carrier, Context context); + + /** + * Detaches the context attached to the given carrier object, leaving it context-less. + * + * @return Previously attached context; {@link Context#root()} if there was none + */ + Context detachFrom(Object carrier); + + /** Requests use of a custom {@link ContextBinder}. */ + static void register(ContextBinder binder) { + ContextProviders.customBinder = binder; + } + + final class Provided { + static final ContextBinder INSTANCE = + null != ContextProviders.customBinder + ? ContextProviders.customBinder + : new WeakMapContextBinder(); + } +} diff --git a/components/context/src/main/java/datadog/context/ContextKey.java b/components/context/src/main/java/datadog/context/ContextKey.java new file mode 100644 index 00000000000..41b79a42c52 --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextKey.java @@ -0,0 +1,47 @@ +package datadog.context; + +import java.util.concurrent.atomic.AtomicInteger; + +/** + * Key for indexing values of type {@link T} stored in a {@link Context}. + * + *

Keys are compared by identity rather than by name. Each stored context type should either + * share its key for re-use or implement {@link ImplicitContextKeyed} to keep its key private. + */ +public final class ContextKey { + private static final AtomicInteger NEXT_INDEX = new AtomicInteger(0); + + private final String name; + final int index; + + private ContextKey(String name) { + this.name = name; + this.index = NEXT_INDEX.getAndIncrement(); + } + + /** Creates a new key with the given name. */ + public static ContextKey named(String name) { + return new ContextKey<>(name); + } + + @Override + public int hashCode() { + return index; + } + + @Override + public boolean equals(Object o) { + if (this == o) { + return true; + } else if (o == null || getClass() != o.getClass()) { + return false; + } else { + return index == ((ContextKey) o).index; + } + } + + @Override + public String toString() { + return name; + } +} diff --git a/components/context/src/main/java/datadog/context/ContextManager.java b/components/context/src/main/java/datadog/context/ContextManager.java new file mode 100644 index 00000000000..e8d45a8ec86 --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextManager.java @@ -0,0 +1,50 @@ +package datadog.context; + +/** Manages context across execution units. */ +public interface ContextManager { + + /** Returns the root context. */ + Context root(); + + /** + * Returns the context attached to the current execution unit. + * + * @return Attached context; {@link #root()} if there is none + */ + Context current(); + + /** + * Attaches the given context to the current execution unit. + * + * @return Scope to be closed when the context is invalid. + */ + ContextScope attach(Context context); + + /** + * Swaps the given context with the one attached to current execution unit. + * + * @return Previously attached context; {@link #root()} if there was none + */ + Context swap(Context context); + + /** + * Detaches the context attached to the current execution unit, leaving it context-less. + * + *

WARNING: prefer {@link ContextScope#close()} to properly restore the surrounding context. + * + * @return Previously attached context; {@link #root()} if there was none + */ + Context detach(); + + /** Requests use of a custom {@link ContextManager}. */ + static void register(ContextManager manager) { + ContextProviders.customManager = manager; + } + + final class Provided { + static final ContextManager INSTANCE = + null != ContextProviders.customManager + ? ContextProviders.customManager + : new ThreadLocalContextManager(); + } +} diff --git a/components/context/src/main/java/datadog/context/ContextProviders.java b/components/context/src/main/java/datadog/context/ContextProviders.java new file mode 100644 index 00000000000..e43d989bedc --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextProviders.java @@ -0,0 +1,16 @@ +package datadog.context; + +/** Provides {@link ContextManager} and {@link ContextBinder} implementations. */ +final class ContextProviders { + + static volatile ContextManager customManager; + static volatile ContextBinder customBinder; + + static ContextManager manager() { + return ContextManager.Provided.INSTANCE; // may be overridden by instrumentation + } + + static ContextBinder binder() { + return ContextBinder.Provided.INSTANCE; // may be overridden by instrumentation + } +} diff --git a/components/context/src/main/java/datadog/context/ContextScope.java b/components/context/src/main/java/datadog/context/ContextScope.java new file mode 100644 index 00000000000..3048d00b37a --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextScope.java @@ -0,0 +1,12 @@ +package datadog.context; + +/** Controls the validity of context attached to an execution unit. */ +public interface ContextScope extends AutoCloseable { + + /** Returns the context controlled by this scope. */ + Context context(); + + /** Detaches the context from the execution unit. */ + @Override + void close(); +} diff --git a/components/context/src/main/java/datadog/context/EmptyContext.java b/components/context/src/main/java/datadog/context/EmptyContext.java new file mode 100644 index 00000000000..ff1599fa4ee --- /dev/null +++ b/components/context/src/main/java/datadog/context/EmptyContext.java @@ -0,0 +1,16 @@ +package datadog.context; + +/** {@link Context} containing no values. */ +final class EmptyContext implements Context { + static final Context INSTANCE = new EmptyContext(); + + @Override + public T get(ContextKey key) { + return null; + } + + @Override + public Context with(ContextKey key, T value) { + return new SingletonContext(key.index, value); + } +} diff --git a/components/context/src/main/java/datadog/context/ImplicitContextKeyed.java b/components/context/src/main/java/datadog/context/ImplicitContextKeyed.java new file mode 100644 index 00000000000..a822158984b --- /dev/null +++ b/components/context/src/main/java/datadog/context/ImplicitContextKeyed.java @@ -0,0 +1,12 @@ +package datadog.context; + +/** {@link Context} value that has its own implicit {@link ContextKey}. */ +public interface ImplicitContextKeyed { + + /** + * Creates a new context with this value under its chosen key. + * + * @return New context with the implicitly keyed value. + */ + Context storeInto(Context context); +} diff --git a/components/context/src/main/java/datadog/context/IndexedContext.java b/components/context/src/main/java/datadog/context/IndexedContext.java new file mode 100644 index 00000000000..89bd29b6200 --- /dev/null +++ b/components/context/src/main/java/datadog/context/IndexedContext.java @@ -0,0 +1,46 @@ +package datadog.context; + +import static java.lang.Math.max; +import static java.util.Arrays.copyOfRange; + +import java.util.Arrays; + +/** {@link Context} containing many values. */ +final class IndexedContext implements Context { + private final Object[] store; + + IndexedContext(Object[] store) { + this.store = store; + } + + @Override + @SuppressWarnings("unchecked") + public T get(ContextKey key) { + int index = key.index; + return index < store.length ? (T) store[index] : null; + } + + @Override + public Context with(ContextKey key, T value) { + int index = key.index; + Object[] newStore = copyOfRange(store, 0, max(store.length, index + 1)); + newStore[index] = value; + + return new IndexedContext(newStore); + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + IndexedContext that = (IndexedContext) o; + return Arrays.equals(store, that.store); + } + + @Override + public int hashCode() { + int result = 31; + result = 31 * result + Arrays.hashCode(store); + return result; + } +} diff --git a/components/context/src/main/java/datadog/context/SingletonContext.java b/components/context/src/main/java/datadog/context/SingletonContext.java new file mode 100644 index 00000000000..b65dffc63ae --- /dev/null +++ b/components/context/src/main/java/datadog/context/SingletonContext.java @@ -0,0 +1,51 @@ +package datadog.context; + +import static java.lang.Math.max; + +import java.util.Objects; + +/** {@link Context} containing a single value. */ +final class SingletonContext implements Context { + private final int index; + private final Object value; + + SingletonContext(int index, Object value) { + this.index = index; + this.value = value; + } + + @Override + @SuppressWarnings("unchecked") + public V get(ContextKey key) { + return index == key.index ? (V) value : null; + } + + @Override + public Context with(ContextKey secondKey, V secondValue) { + int secondIndex = secondKey.index; + if (index == secondIndex) { + return new SingletonContext(index, secondValue); + } else { + Object[] store = new Object[max(index, secondIndex) + 1]; + store[index] = value; + store[secondIndex] = secondValue; + return new IndexedContext(store); + } + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + SingletonContext that = (SingletonContext) o; + return index == that.index && Objects.equals(value, that.value); + } + + @Override + public int hashCode() { + int result = 31; + result = 31 * result + index; + result = 31 * result + Objects.hashCode(value); + return result; + } +} diff --git a/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java b/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java new file mode 100644 index 00000000000..d8fab846111 --- /dev/null +++ b/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java @@ -0,0 +1,56 @@ +package datadog.context; + +/** {@link ContextManager} that uses a {@link ThreadLocal} to track context per thread. */ +final class ThreadLocalContextManager implements ContextManager { + + private static final ThreadLocal CURRENT_HOLDER = + ThreadLocal.withInitial(() -> new Context[] {EmptyContext.INSTANCE}); + + @Override + public Context root() { + return EmptyContext.INSTANCE; + } + + @Override + public Context current() { + return CURRENT_HOLDER.get()[0]; + } + + @Override + public ContextScope attach(Context context) { + + Context[] holder = CURRENT_HOLDER.get(); + Context previous = holder[0]; + holder[0] = context; + + return new ContextScope() { + private boolean closed; + + @Override + public Context context() { + return context; + } + + @Override + public void close() { + if (!closed && context == holder[0]) { + holder[0] = previous; + closed = true; + } + } + }; + } + + @Override + public Context swap(Context context) { + Context[] holder = CURRENT_HOLDER.get(); + Context previous = holder[0]; + holder[0] = context; + return previous; + } + + @Override + public Context detach() { + return swap(root()); + } +} diff --git a/components/context/src/main/java/datadog/context/WeakMapContextBinder.java b/components/context/src/main/java/datadog/context/WeakMapContextBinder.java new file mode 100644 index 00000000000..144a62c1c35 --- /dev/null +++ b/components/context/src/main/java/datadog/context/WeakMapContextBinder.java @@ -0,0 +1,29 @@ +package datadog.context; + +import java.util.Collections; +import java.util.Map; +import java.util.WeakHashMap; + +/** {@link ContextBinder} that uses a global weak map of carriers to contexts. */ +final class WeakMapContextBinder implements ContextBinder { + + private static final Map TRACKED = + Collections.synchronizedMap(new WeakHashMap<>()); + + @Override + public Context from(Object carrier) { + Context bound = TRACKED.get(carrier); + return null != bound ? bound : Context.root(); + } + + @Override + public void attachTo(Object carrier, Context context) { + TRACKED.put(carrier, context); + } + + @Override + public Context detachFrom(Object carrier) { + Context previous = TRACKED.remove(carrier); + return null != previous ? previous : Context.root(); + } +} diff --git a/components/context/src/test/java/datadog/context/ContextBinderTest.java b/components/context/src/test/java/datadog/context/ContextBinderTest.java new file mode 100644 index 00000000000..6aabffba095 --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextBinderTest.java @@ -0,0 +1,22 @@ +package datadog.context; + +import static datadog.context.Context.root; +import static datadog.context.ContextTest.STRING_KEY; +import static org.junit.jupiter.api.Assertions.assertEquals; + +import org.junit.jupiter.api.Test; + +class ContextBinderTest { + + @Test + void testAttachAndDetach() { + Context context = root().with(STRING_KEY, "value"); + Object carrier = new Object(); + assertEquals(root(), Context.from(carrier)); + context.attachTo(carrier); + assertEquals(context, Context.from(carrier)); + // Detaching removes all context + assertEquals(context, Context.detachFrom(carrier)); + assertEquals(root(), Context.from(carrier)); + } +} diff --git a/components/context/src/test/java/datadog/context/ContextKeyTest.java b/components/context/src/test/java/datadog/context/ContextKeyTest.java new file mode 100644 index 00000000000..8e65485f6c3 --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextKeyTest.java @@ -0,0 +1,34 @@ +package datadog.context; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertNotEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertNull; + +import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.NullSource; +import org.junit.jupiter.params.provider.ValueSource; + +class ContextKeyTest { + + @ParameterizedTest + @NullSource + @ValueSource(strings = {"", "key"}) + void testConstructor(String name) { + ContextKey key = ContextKey.named(name); + assertNotNull(key); + assertEquals(name, key.toString()); + } + + @Test + void testKeyNameCollision() { + ContextKey key1 = ContextKey.named("same-name"); + ContextKey key2 = ContextKey.named("same-name"); + assertNotEquals(key1, key2); + String value = "value"; + Context context = Context.root().with(key1, value); + assertEquals(value, context.get(key1)); + assertNull(context.get(key2)); + } +} diff --git a/components/context/src/test/java/datadog/context/ContextManagerTest.java b/components/context/src/test/java/datadog/context/ContextManagerTest.java new file mode 100644 index 00000000000..41ca1e6859c --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextManagerTest.java @@ -0,0 +1,219 @@ +package datadog.context; + +import static datadog.context.Context.current; +import static datadog.context.Context.detach; +import static datadog.context.Context.root; +import static datadog.context.ContextTest.STRING_KEY; +import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; +import static org.junit.jupiter.api.Assertions.assertEquals; + +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; +import java.util.concurrent.Future; +import java.util.concurrent.Phaser; +import org.junit.jupiter.api.AfterEach; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; + +class ContextManagerTest { + + @BeforeEach + void init() { + // Ensure no current context prior starting test + assertEquals(root(), current()); + } + + @Test + void testContextAttachment() { + Context context1 = root().with(STRING_KEY, "value1"); + try (ContextScope scope1 = context1.attach()) { + // Test context1 is attached + assertEquals(context1, current()); + assertEquals(context1, scope1.context()); + Context context2 = context1.with(STRING_KEY, "value2"); + try (ContextScope scope2 = context2.attach()) { + // Test context2 is attached + assertEquals(context2, current()); + assertEquals(context2, scope2.context()); + // Can still access context1 from its scope + assertEquals(context1, scope1.context()); + } + // Test context1 is restored + assertEquals(context1, current()); + } + } + + @Test + void testContextSwapping() { + Context context1 = root().with(STRING_KEY, "value1"); + assertEquals(root(), current()); + assertEquals(root(), context1.swap()); + // Test context1 is attached + Context context2 = context1.with(STRING_KEY, "value2"); + assertEquals(context1, current()); + assertEquals(context1, context2.swap()); + // Test context2 is attached + assertEquals(context2, current()); + assertEquals(context2, detach()); + // Test we're now context-less + assertEquals(root(), current()); + } + + @Test + void testAttachSameContextMultipleTimes() { + Context context = root().with(STRING_KEY, "value1"); + try (ContextScope ignored1 = context.attach()) { + assertEquals(context, current()); + try (ContextScope ignored2 = context.attach()) { + try (ContextScope ignored3 = context.attach()) { + assertEquals(context, current()); + } + // Test closing a scope on the current context should not deactivate it if activated + // multiple times + assertEquals(context, current()); + } + } + // Test closing the same number of scope as activation should deactivate the context + assertEquals(root(), current()); + } + + @Test + void testOnlyCurrentScopeCanBeClosed() { + Context context1 = root().with(STRING_KEY, "value1"); + try (ContextScope scope1 = context1.attach()) { + Context context2 = context1.with(STRING_KEY, "value2"); + try (ContextScope ignored = context2.attach()) { + // Try closing the non-current scope + scope1.close(); + // Test context2 is still attached + assertEquals(context2, current()); + } + // Test context1 is restored + assertEquals(context1, current()); + } + } + + @Test + void testClosingMultipleTimes() { + Context context1 = root().with(STRING_KEY, "value1"); + try (ContextScope ignored = context1.attach()) { + Context context2 = context1.with(STRING_KEY, "value2"); + ContextScope scope = context2.attach(); + // Test current context + assertEquals(context2, current()); + // Test current context deactivation + scope.close(); + assertEquals(context1, current()); + // Test multiple context deactivations don’t change current context + scope.close(); + assertEquals(context1, current()); + } + } + + @Test + void testThreadIndependence() { + /* + * This test has 2 executors in addition to the main thread. + * They are synchronized using a Phaser, and arrived before each assert phase. + * If an assert fails in of one the executor, the executor is "deregister" to unblock the test, + * and the exception is restored at the end of the test using "Future.get()". + */ + ExecutorService executor = Executors.newFixedThreadPool(2); + Phaser phaser = new Phaser(3); + /* + * Create first executor. + */ + Future future1 = + executor.submit( + () -> { + try { + // Fist step: check empty context + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Second step: set context on first executor + Context context1 = root().with(STRING_KEY, "executor1"); + try (ContextScope ignored1 = context1.attach()) { + phaser.arriveAndAwaitAdvance(); + assertEquals(context1, current()); + // Third step: set context on second executor + phaser.arriveAndAwaitAdvance(); + assertEquals(context1, current()); + // Fourth step: set child context on first executor + Context context11 = context1.with(STRING_KEY, "executor1.1"); + try (ContextScope ignored11 = context11.attach()) { + phaser.arriveAndAwaitAdvance(); + assertEquals(context11, current()); + } + } + } finally { + // Complete the execution + phaser.arriveAndDeregister(); + } + }); + /* + * Create second executor. + */ + Future future2 = + executor.submit( + () -> { + try { + // First step: check empty context + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Second step: set context on first executor + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Third step: set context on second executor + Context context2 = root().with(STRING_KEY, "executor2"); + try (ContextScope ignored2 = context2.attach()) { + phaser.arriveAndAwaitAdvance(); + assertEquals(context2, current()); + // Fourth step: set child context on first executor + phaser.arriveAndAwaitAdvance(); + assertEquals(context2, current()); + } + } finally { + // Complete the execution + phaser.arriveAndDeregister(); + } + }); + /* + * Run main thread. + */ + // First step: check empty context + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Second step: set context on first executor + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Third step: set context on second executor + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Fourth step: set child context on first executor + phaser.arriveAndAwaitAdvance(); + assertEquals(root(), current()); + // Complete execution and wait for the others + phaser.arriveAndAwaitAdvance(); + executor.shutdown(); + // Check any test error in executors + assertDoesNotThrow(() -> future1.get()); + assertDoesNotThrow(() -> future2.get()); + } + + @Test + void testNonThreadInheritance() { + Context context = root().with(STRING_KEY, "value"); + try (ContextScope ignored = context.attach()) { + // Check new thread don't inherit from current context + ExecutorService executor = Executors.newSingleThreadExecutor(); + Future future = executor.submit(() -> assertEquals(root(), current())); + assertDoesNotThrow(() -> future.get()); + } + } + + @AfterEach + void tearDown() { + // Ensure no current context after ending test + assertEquals(root(), current()); + } +} diff --git a/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java b/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java new file mode 100644 index 00000000000..18059adf059 --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java @@ -0,0 +1,86 @@ +package datadog.context; + +import static datadog.context.Context.root; +import static datadog.context.ContextTest.STRING_KEY; +import static org.junit.jupiter.api.Assertions.assertEquals; + +import org.junit.jupiter.api.Test; + +class ContextProviderForkedTest { + + @Test + void testCustomBinder() { + // register a NOOP context binder + ContextBinder.register(new ContextBinder() { + @Override + public Context from(Object carrier) { + return root(); + } + + @Override + public void attachTo(Object carrier, Context context) { + // no-op + } + + @Override + public Context detachFrom(Object carrier) { + return root(); + } + }); + + Context context = root().with(STRING_KEY, "value"); + + // NOOP binder, context will always be root + Object carrier = new Object(); + context.attachTo(carrier); + assertEquals(root(), Context.from(carrier)); + } + + @Test + void testCustomManager() { + // register a NOOP context manager + ContextManager.register(new ContextManager() { + @Override + public Context root() { + return EmptyContext.INSTANCE; + } + + @Override + public Context current() { + return root(); + } + + @Override + public ContextScope attach(Context context) { + return new ContextScope() { + @Override + public Context context() { + return root(); + } + + @Override + public void close() { + // no-op + } + }; + } + + @Override + public Context swap(Context context) { + return root(); + } + + @Override + public Context detach() { + return root(); + } + }); + + Context context = root().with(STRING_KEY, "value"); + + // NOOP manager, context will always be root + try (ContextScope ignored = context.attach()) { + assertEquals(root(), Context.current()); + } + } +} diff --git a/components/context/src/test/java/datadog/context/ContextTest.java b/components/context/src/test/java/datadog/context/ContextTest.java new file mode 100644 index 00000000000..c1632163bb6 --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextTest.java @@ -0,0 +1,113 @@ +package datadog.context; + +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.api.Assertions.assertTrue; + +import javax.annotation.Nullable; +import org.junit.jupiter.api.Test; + +class ContextTest { + static final ContextKey STRING_KEY = ContextKey.named("string-key"); + static final ContextKey BOOLEAN_KEY = ContextKey.named("boolean-key"); + + // demonstrate how values can hide their context keys + static class ValueWithKey implements ImplicitContextKeyed { + static final ContextKey HIDDEN_KEY = ContextKey.named("hidden-key"); + + @Override + public Context storeInto(Context context) { + return context.with(HIDDEN_KEY, this); + } + + @Nullable + public static ValueWithKey from(Context context) { + return context.get(HIDDEN_KEY); + } + } + + @Test + void testEmpty() { + // Test empty is always the same + Context empty = Context.root(); + assertEquals(empty, Context.root()); + // Test empty is not mutated + String stringValue = "value"; + empty.with(STRING_KEY, stringValue); + assertEquals(empty, Context.root()); + } + + @Test + void testWith() { + Context empty = Context.root(); + // Test accessing non-set value + assertNull(empty.get(STRING_KEY)); + // Test retrieving value + String stringValue = "value"; + Context context1 = empty.with(STRING_KEY, stringValue); + assertEquals(stringValue, context1.get(STRING_KEY)); + // Test overriding value + String stringValue2 = "value2"; + Context context2 = context1.with(STRING_KEY, stringValue2); + assertEquals(stringValue2, context2.get(STRING_KEY)); + // Test clearing value + Context context3 = context2.with(STRING_KEY, null); + assertNull(context3.get(STRING_KEY)); + // Test null key handling + assertThrows(NullPointerException.class, () -> empty.with(null, "test")); + } + + @Test + void testGet() { + // Setup context + Context empty = Context.root(); + String value = "value"; + Context context = empty.with(STRING_KEY, value); + // Test null key handling + assertThrows(NullPointerException.class, () -> context.get(null)); + // Test unset key + assertNull(context.get(BOOLEAN_KEY)); + // Test set key + assertEquals(value, context.get(STRING_KEY)); + } + + @SuppressWarnings({ + "EqualsWithItself", + "SimplifiableAssertion", + "ConstantValue", + "EqualsBetweenInconvertibleTypes" + }) + @Test + void testEqualsAndHashCode() { + // Setup contexts + Context empty = Context.root(); + Context context1 = empty.with(STRING_KEY, "value"); + Context context2 = empty.with(STRING_KEY, "value "); + Context context3 = empty.with(STRING_KEY, "value ".trim()); + // Test equals on self + assertTrue(empty.equals(empty)); + assertTrue(context1.equals(context1)); + // Test equals on null + assertFalse(context1.equals(null)); + // Test equals on different object type + assertFalse(context1.equals("value")); + // Test equals on different contexts with the same values + assertTrue(context1.equals(context3)); + assertEquals(context1.hashCode(), context3.hashCode()); + // Test equals on different contexts + assertFalse(context1.equals(empty)); + assertFalse(context1.equals(context2)); + } + + @Test + void testImplicitKey() { + // Setup context + Context empty = Context.root(); + ValueWithKey valueWithKey = new ValueWithKey(); + Context context = empty.with(valueWithKey); + assertNull(ValueWithKey.from(empty)); + assertEquals(valueWithKey, ValueWithKey.from(context)); + } +} diff --git a/settings.gradle b/settings.gradle index 2d78784b023..67178ac923b 100644 --- a/settings.gradle +++ b/settings.gradle @@ -66,6 +66,7 @@ include ':dd-java-agent:agent-otel:otel-shim' include ':dd-java-agent:agent-otel:otel-tooling' include ':communication' +include ':components:context' include ':components:json' include ':telemetry' include ':remote-config:remote-config-api' @@ -512,3 +513,4 @@ include ':dd-java-agent:benchmark' include ':dd-java-agent:benchmark-integration' include ':dd-java-agent:benchmark-integration:jetty-perftest' include ':dd-java-agent:benchmark-integration:play-perftest' + From 9e3d3a8ac972e30c5603288809e21fee2700d150 Mon Sep 17 00:00:00 2001 From: Stuart McCulloch Date: Thu, 19 Dec 2024 13:52:30 +0000 Subject: [PATCH 2/5] Remove detach, same effect can be achieved with root().swap() --- .../main/java/datadog/context/Context.java | 11 --- .../java/datadog/context/ContextManager.java | 9 --- .../context/ThreadLocalContextManager.java | 5 -- .../datadog/context/ContextManagerTest.java | 3 +- .../context/ContextProviderForkedTest.java | 81 +++++++++---------- 5 files changed, 40 insertions(+), 69 deletions(-) diff --git a/components/context/src/main/java/datadog/context/Context.java b/components/context/src/main/java/datadog/context/Context.java index fcdc4d418d1..94489964cc0 100644 --- a/components/context/src/main/java/datadog/context/Context.java +++ b/components/context/src/main/java/datadog/context/Context.java @@ -40,17 +40,6 @@ default Context swap() { return manager().swap(this); } - /** - * Detaches the context attached to the current execution unit, leaving it context-less. - * - *

WARNING: prefer {@link ContextScope#close()} to properly restore the surrounding context. - * - * @return Previously attached context; {@link #root()} if there was none - */ - static Context detach() { - return manager().detach(); - } - /** * Returns the context attached to the given carrier object. * diff --git a/components/context/src/main/java/datadog/context/ContextManager.java b/components/context/src/main/java/datadog/context/ContextManager.java index e8d45a8ec86..aa571a3a8ba 100644 --- a/components/context/src/main/java/datadog/context/ContextManager.java +++ b/components/context/src/main/java/datadog/context/ContextManager.java @@ -27,15 +27,6 @@ public interface ContextManager { */ Context swap(Context context); - /** - * Detaches the context attached to the current execution unit, leaving it context-less. - * - *

WARNING: prefer {@link ContextScope#close()} to properly restore the surrounding context. - * - * @return Previously attached context; {@link #root()} if there was none - */ - Context detach(); - /** Requests use of a custom {@link ContextManager}. */ static void register(ContextManager manager) { ContextProviders.customManager = manager; diff --git a/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java b/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java index d8fab846111..928098e2cc1 100644 --- a/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java +++ b/components/context/src/main/java/datadog/context/ThreadLocalContextManager.java @@ -48,9 +48,4 @@ public Context swap(Context context) { holder[0] = context; return previous; } - - @Override - public Context detach() { - return swap(root()); - } } diff --git a/components/context/src/test/java/datadog/context/ContextManagerTest.java b/components/context/src/test/java/datadog/context/ContextManagerTest.java index 41ca1e6859c..ed76419d06b 100644 --- a/components/context/src/test/java/datadog/context/ContextManagerTest.java +++ b/components/context/src/test/java/datadog/context/ContextManagerTest.java @@ -1,7 +1,6 @@ package datadog.context; import static datadog.context.Context.current; -import static datadog.context.Context.detach; import static datadog.context.Context.root; import static datadog.context.ContextTest.STRING_KEY; import static org.junit.jupiter.api.Assertions.assertDoesNotThrow; @@ -54,7 +53,7 @@ void testContextSwapping() { assertEquals(context1, context2.swap()); // Test context2 is attached assertEquals(context2, current()); - assertEquals(context2, detach()); + assertEquals(context2, root().swap()); // Test we're now context-less assertEquals(root(), current()); } diff --git a/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java b/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java index 18059adf059..9c15d37da45 100644 --- a/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java +++ b/components/context/src/test/java/datadog/context/ContextProviderForkedTest.java @@ -11,22 +11,23 @@ class ContextProviderForkedTest { @Test void testCustomBinder() { // register a NOOP context binder - ContextBinder.register(new ContextBinder() { - @Override - public Context from(Object carrier) { - return root(); - } + ContextBinder.register( + new ContextBinder() { + @Override + public Context from(Object carrier) { + return root(); + } - @Override - public void attachTo(Object carrier, Context context) { - // no-op - } + @Override + public void attachTo(Object carrier, Context context) { + // no-op + } - @Override - public Context detachFrom(Object carrier) { - return root(); - } - }); + @Override + public Context detachFrom(Object carrier) { + return root(); + } + }); Context context = root().with(STRING_KEY, "value"); @@ -39,42 +40,38 @@ public Context detachFrom(Object carrier) { @Test void testCustomManager() { // register a NOOP context manager - ContextManager.register(new ContextManager() { - @Override - public Context root() { - return EmptyContext.INSTANCE; - } - - @Override - public Context current() { - return root(); - } + ContextManager.register( + new ContextManager() { + @Override + public Context root() { + return EmptyContext.INSTANCE; + } - @Override - public ContextScope attach(Context context) { - return new ContextScope() { @Override - public Context context() { + public Context current() { return root(); } @Override - public void close() { - // no-op + public ContextScope attach(Context context) { + return new ContextScope() { + @Override + public Context context() { + return root(); + } + + @Override + public void close() { + // no-op + } + }; } - }; - } - @Override - public Context swap(Context context) { - return root(); - } - - @Override - public Context detach() { - return root(); - } - }); + @Override + public Context swap(Context context) { + return root(); + } + }); Context context = root().with(STRING_KEY, "value"); From 8c2978b65a7442a29fdb77ba149227a46862206c Mon Sep 17 00:00:00 2001 From: Stuart McCulloch Date: Thu, 19 Dec 2024 14:35:50 +0000 Subject: [PATCH 3/5] Move provider initialization-on-demand types to ContextProviders --- .../java/datadog/context/ContextBinder.java | 7 ------- .../java/datadog/context/ContextManager.java | 7 ------- .../java/datadog/context/ContextProviders.java | 18 ++++++++++++++++-- 3 files changed, 16 insertions(+), 16 deletions(-) diff --git a/components/context/src/main/java/datadog/context/ContextBinder.java b/components/context/src/main/java/datadog/context/ContextBinder.java index bfcc85a887a..db461788942 100644 --- a/components/context/src/main/java/datadog/context/ContextBinder.java +++ b/components/context/src/main/java/datadog/context/ContextBinder.java @@ -24,11 +24,4 @@ public interface ContextBinder { static void register(ContextBinder binder) { ContextProviders.customBinder = binder; } - - final class Provided { - static final ContextBinder INSTANCE = - null != ContextProviders.customBinder - ? ContextProviders.customBinder - : new WeakMapContextBinder(); - } } diff --git a/components/context/src/main/java/datadog/context/ContextManager.java b/components/context/src/main/java/datadog/context/ContextManager.java index aa571a3a8ba..409bc8d1c94 100644 --- a/components/context/src/main/java/datadog/context/ContextManager.java +++ b/components/context/src/main/java/datadog/context/ContextManager.java @@ -31,11 +31,4 @@ public interface ContextManager { static void register(ContextManager manager) { ContextProviders.customManager = manager; } - - final class Provided { - static final ContextManager INSTANCE = - null != ContextProviders.customManager - ? ContextProviders.customManager - : new ThreadLocalContextManager(); - } } diff --git a/components/context/src/main/java/datadog/context/ContextProviders.java b/components/context/src/main/java/datadog/context/ContextProviders.java index e43d989bedc..6895446947a 100644 --- a/components/context/src/main/java/datadog/context/ContextProviders.java +++ b/components/context/src/main/java/datadog/context/ContextProviders.java @@ -6,11 +6,25 @@ final class ContextProviders { static volatile ContextManager customManager; static volatile ContextBinder customBinder; + private static final class ProvidedManager { + static final ContextManager INSTANCE = + null != ContextProviders.customManager + ? ContextProviders.customManager + : new ThreadLocalContextManager(); + } + + private static final class ProvidedBinder { + static final ContextBinder INSTANCE = + null != ContextProviders.customBinder + ? ContextProviders.customBinder + : new WeakMapContextBinder(); + } + static ContextManager manager() { - return ContextManager.Provided.INSTANCE; // may be overridden by instrumentation + return ProvidedManager.INSTANCE; // may be overridden by instrumentation } static ContextBinder binder() { - return ContextBinder.Provided.INSTANCE; // may be overridden by instrumentation + return ProvidedBinder.INSTANCE; // may be overridden by instrumentation } } From 9e9e15609fe19af12bc92c7170da1ead000cba57 Mon Sep 17 00:00:00 2001 From: Stuart McCulloch Date: Fri, 20 Dec 2024 14:06:51 +0000 Subject: [PATCH 4/5] Test coverage --- components/context/build.gradle.kts | 4 ++ .../main/java/datadog/context/ContextKey.java | 11 +--- .../datadog/context/ContextBinderTest.java | 1 + .../java/datadog/context/ContextKeyTest.java | 24 +++++++++ .../java/datadog/context/ContextTest.java | 54 +++++++++++++++++++ 5 files changed, 84 insertions(+), 10 deletions(-) diff --git a/components/context/build.gradle.kts b/components/context/build.gradle.kts index 4dca7fc3036..c737d9fefdd 100644 --- a/components/context/build.gradle.kts +++ b/components/context/build.gradle.kts @@ -7,3 +7,7 @@ apply(from = "$rootDir/gradle/java.gradle") jmh { version = "1.28" } + +val excludedClassesInstructionCoverage by extra { + listOf("datadog.context.ContextProviders") // covered by forked test +} diff --git a/components/context/src/main/java/datadog/context/ContextKey.java b/components/context/src/main/java/datadog/context/ContextKey.java index 41b79a42c52..41f09e914a5 100644 --- a/components/context/src/main/java/datadog/context/ContextKey.java +++ b/components/context/src/main/java/datadog/context/ContextKey.java @@ -29,16 +29,7 @@ public int hashCode() { return index; } - @Override - public boolean equals(Object o) { - if (this == o) { - return true; - } else if (o == null || getClass() != o.getClass()) { - return false; - } else { - return index == ((ContextKey) o).index; - } - } + // we want identity equality, so no need to override equals() @Override public String toString() { diff --git a/components/context/src/test/java/datadog/context/ContextBinderTest.java b/components/context/src/test/java/datadog/context/ContextBinderTest.java index 6aabffba095..6e1b77026c7 100644 --- a/components/context/src/test/java/datadog/context/ContextBinderTest.java +++ b/components/context/src/test/java/datadog/context/ContextBinderTest.java @@ -17,6 +17,7 @@ void testAttachAndDetach() { assertEquals(context, Context.from(carrier)); // Detaching removes all context assertEquals(context, Context.detachFrom(carrier)); + assertEquals(root(), Context.detachFrom(carrier)); assertEquals(root(), Context.from(carrier)); } } diff --git a/components/context/src/test/java/datadog/context/ContextKeyTest.java b/components/context/src/test/java/datadog/context/ContextKeyTest.java index 8e65485f6c3..3ef6236c51e 100644 --- a/components/context/src/test/java/datadog/context/ContextKeyTest.java +++ b/components/context/src/test/java/datadog/context/ContextKeyTest.java @@ -1,9 +1,11 @@ package datadog.context; import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; import static org.junit.jupiter.api.Assertions.assertNotEquals; import static org.junit.jupiter.api.Assertions.assertNotNull; import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertTrue; import org.junit.jupiter.api.Test; import org.junit.jupiter.params.ParameterizedTest; @@ -31,4 +33,26 @@ void testKeyNameCollision() { assertEquals(value, context.get(key1)); assertNull(context.get(key2)); } + + @SuppressWarnings({ + "EqualsWithItself", + "SimplifiableAssertion", + "ConstantValue", + "EqualsBetweenInconvertibleTypes" + }) + @Test + void testEqualsAndHashCode() { + ContextKey key1 = ContextKey.named("same-name"); + ContextKey key2 = ContextKey.named("same-name"); + // Test equals on self + assertTrue(key1.equals(key1)); + assertEquals(key1.hashCode(), key1.hashCode()); + // Test equals on null + assertFalse(key1.equals(null)); + // Test equals on different object type + assertFalse(key1.equals("value")); + // Test equals on different keys with the same name + assertFalse(key1.equals(key2)); + assertNotEquals(key1.hashCode(), key2.hashCode()); + } } diff --git a/components/context/src/test/java/datadog/context/ContextTest.java b/components/context/src/test/java/datadog/context/ContextTest.java index c1632163bb6..a7ea8f42b52 100644 --- a/components/context/src/test/java/datadog/context/ContextTest.java +++ b/components/context/src/test/java/datadog/context/ContextTest.java @@ -2,6 +2,7 @@ import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNotEquals; import static org.junit.jupiter.api.Assertions.assertNull; import static org.junit.jupiter.api.Assertions.assertThrows; import static org.junit.jupiter.api.Assertions.assertTrue; @@ -12,6 +13,8 @@ class ContextTest { static final ContextKey STRING_KEY = ContextKey.named("string-key"); static final ContextKey BOOLEAN_KEY = ContextKey.named("boolean-key"); + static final ContextKey FLOAT_KEY = ContextKey.named("float-key"); + static final ContextKey LONG_KEY = ContextKey.named("long-key"); // demonstrate how values can hide their context keys static class ValueWithKey implements ImplicitContextKeyed { @@ -86,19 +89,33 @@ void testEqualsAndHashCode() { Context context1 = empty.with(STRING_KEY, "value"); Context context2 = empty.with(STRING_KEY, "value "); Context context3 = empty.with(STRING_KEY, "value ".trim()); + Context context4 = empty.with(STRING_KEY, "value").with(BOOLEAN_KEY, true); // Test equals on self assertTrue(empty.equals(empty)); assertTrue(context1.equals(context1)); + assertTrue(context4.equals(context4)); // Test equals on null assertFalse(context1.equals(null)); + assertFalse(context4.equals(null)); // Test equals on different object type assertFalse(context1.equals("value")); + assertFalse(context4.equals("value")); // Test equals on different contexts with the same values assertTrue(context1.equals(context3)); assertEquals(context1.hashCode(), context3.hashCode()); // Test equals on different contexts assertFalse(context1.equals(empty)); + assertNotEquals(context1.hashCode(), empty.hashCode()); assertFalse(context1.equals(context2)); + assertNotEquals(context1.hashCode(), context2.hashCode()); + assertFalse(context1.equals(context4)); + assertNotEquals(context1.hashCode(), context4.hashCode()); + assertFalse(empty.equals(context1)); + assertNotEquals(empty.hashCode(), context1.hashCode()); + assertFalse(context2.equals(context1)); + assertNotEquals(context2.hashCode(), context1.hashCode()); + assertFalse(context4.equals(context1)); + assertNotEquals(context4.hashCode(), context1.hashCode()); } @Test @@ -110,4 +127,41 @@ void testImplicitKey() { assertNull(ValueWithKey.from(empty)); assertEquals(valueWithKey, ValueWithKey.from(context)); } + + @SuppressWarnings({"SimplifiableAssertion"}) + @Test + void testInflation() { + Context empty = Context.root(); + + Context one = empty.with(STRING_KEY, "unset").with(STRING_KEY, "one"); + Context two = one.with(BOOLEAN_KEY, false).with(BOOLEAN_KEY, true); + Context three = two.with(FLOAT_KEY, 0.0f).with(FLOAT_KEY, 3.3f); + + assertNull(empty.get(STRING_KEY)); + assertNull(empty.get(BOOLEAN_KEY)); + assertNull(empty.get(FLOAT_KEY)); + assertNull(empty.get(LONG_KEY)); + + assertEquals("one", one.get(STRING_KEY)); + assertNull(one.get(BOOLEAN_KEY)); + assertNull(one.get(FLOAT_KEY)); + assertNull(one.get(LONG_KEY)); + + assertEquals("one", two.get(STRING_KEY)); + assertEquals(true, two.get(BOOLEAN_KEY)); + assertNull(two.get(FLOAT_KEY)); + assertNull(two.get(LONG_KEY)); + + assertEquals("one", three.get(STRING_KEY)); + assertEquals(true, three.get(BOOLEAN_KEY)); + assertEquals(3.3f, three.get(FLOAT_KEY)); + assertNull(three.get(LONG_KEY)); + + assertFalse(empty.equals(one)); + assertFalse(one.equals(two)); + assertFalse(two.equals(three)); + assertNotEquals(one.hashCode(), empty.hashCode()); + assertNotEquals(two.hashCode(), one.hashCode()); + assertNotEquals(three.hashCode(), two.hashCode()); + } } From f93396d7b61195ce48f018c5af241d8283e29b61 Mon Sep 17 00:00:00 2001 From: Stuart McCulloch Date: Fri, 20 Dec 2024 14:44:22 +0000 Subject: [PATCH 5/5] javadoc --- .../src/main/java/datadog/context/Context.java | 18 ++++++++++++++---- .../main/java/datadog/context/ContextKey.java | 2 +- .../java/datadog/context/ContextManager.java | 6 +++++- 3 files changed, 20 insertions(+), 6 deletions(-) diff --git a/components/context/src/main/java/datadog/context/Context.java b/components/context/src/main/java/datadog/context/Context.java index 94489964cc0..7ca2309bece 100644 --- a/components/context/src/main/java/datadog/context/Context.java +++ b/components/context/src/main/java/datadog/context/Context.java @@ -5,10 +5,20 @@ import javax.annotation.Nullable; -/** Immutable context scoped to an execution unit or carrier object. */ +/** + * Immutable context scoped to an execution unit or carrier object. + * + *

Each element of the context is accessible by its {@link ContextKey}. Keys represents product + * or functional areas and should be created sparingly. Elements in the context may themselves be + * mutable. + */ public interface Context { - /** Returns the root context. */ + /** + * Returns the root context. + * + *

This is the initial local context that all contexts extend. + */ static Context root() { return manager().root(); } @@ -72,14 +82,14 @@ static Context detachFrom(Object carrier) { T get(ContextKey key); /** - * Creates a new context with the given key-value mapping. + * Creates a new context from the same elements, except the key is now mapped to the given value. * * @return New context with the key-value mapping. */ Context with(ContextKey key, T value); /** - * Creates a new context with a value that has its own implicit key. + * Creates a new context from the same elements, except the implicit key is mapped to this value. * * @return New context with the implicitly keyed value. */ diff --git a/components/context/src/main/java/datadog/context/ContextKey.java b/components/context/src/main/java/datadog/context/ContextKey.java index 41f09e914a5..962a1ce28df 100644 --- a/components/context/src/main/java/datadog/context/ContextKey.java +++ b/components/context/src/main/java/datadog/context/ContextKey.java @@ -3,7 +3,7 @@ import java.util.concurrent.atomic.AtomicInteger; /** - * Key for indexing values of type {@link T} stored in a {@link Context}. + * {@link Context} key that maps to a value of type {@link T}. * *

Keys are compared by identity rather than by name. Each stored context type should either * share its key for re-use or implement {@link ImplicitContextKeyed} to keep its key private. diff --git a/components/context/src/main/java/datadog/context/ContextManager.java b/components/context/src/main/java/datadog/context/ContextManager.java index 409bc8d1c94..af5811416fb 100644 --- a/components/context/src/main/java/datadog/context/ContextManager.java +++ b/components/context/src/main/java/datadog/context/ContextManager.java @@ -3,7 +3,11 @@ /** Manages context across execution units. */ public interface ContextManager { - /** Returns the root context. */ + /** + * Returns the root context. + * + *

This is the initial local context that all contexts extend. + */ Context root(); /**