diff --git a/components/context/src/main/java/datadog/context/ContextHelpers.java b/components/context/src/main/java/datadog/context/ContextHelpers.java new file mode 100644 index 00000000000..12eddf91e99 --- /dev/null +++ b/components/context/src/main/java/datadog/context/ContextHelpers.java @@ -0,0 +1,165 @@ +package datadog.context; + +import static java.lang.Math.max; +import static java.util.Arrays.copyOfRange; +import static java.util.Objects.requireNonNull; + +import java.util.ArrayList; +import java.util.List; +import java.util.function.BinaryOperator; + +/** + * Static helpers to manipulate context collections. + * + *

Typical usages include: + * + *

{@code
+ * // Finding a context value from multiple sources:
+ * Span span = findFirst(spanKey, message, request, CURRENT)
+ * // Find all context values from different sources:
+ * List errors = findAll(errorKey, message, request, CURRENT)
+ * // Capture multiple contexts in a single one:
+ * Context aggregate = combine(message, request, CURRENT)
+ * // Combine multiple contexts into a single one using custom merge rules:
+ * Context combined = combine(
+ *   (current, next) -> {
+ *     var metric = current.get(metricKey);
+ *     var nextMetric = next.get(metricKey);
+ *     return current.with(metricKey, metric.add(nextMetric));
+ *   }, message, request, CURRENT);
+ * }
+ * + * where {@link #CURRENT} denotes a carrier with the current context. + */ +public final class ContextHelpers { + /** A helper object carrying the {@link Context#current()} context. */ + public static final Object CURRENT = new Object(); + + private ContextHelpers() {} + + /** + * Find the first context value from given context carriers. + * + * @param key The key used to store the value. + * @param carriers The carrier to get context and value from. + * @param The type of the value to look for. + * @return The first context value found, {@code null} if not found. + */ + public static T findFirst(ContextKey key, Object... carriers) { + requireNonNull(key, "key cannot be null"); + for (Object carrier : carriers) { + requireNonNull(carrier, "carrier cannot be null"); + Context context = carrier == CURRENT ? Context.current() : Context.from(carrier); + T value = context.get(key); + if (value != null) { + return value; + } + } + return null; + } + + /** + * Find all the context values from the given context carriers. + * + * @param key The key used to store the value. + * @param carriers The carriers to get context and value from. + * @param The type of the values to look for. + * @return A list of all values found, in context order. + */ + public static List findAll(ContextKey key, Object... carriers) { + requireNonNull(key, "key cannot be null"); + List values = new ArrayList<>(carriers.length); + for (Object carrier : carriers) { + requireNonNull(carrier, "carrier cannot be null"); + Context context = carrier == CURRENT ? Context.current() : Context.from(carrier); + T value = context.get(key); + if (value != null) { + values.add(value); + } + } + return values; + } + + /** + * Combine contexts and their values, keeping the first founds. + * + * @param contexts The contexts to combine. + * @return A context containing all the values from all the given context, keeping the first value + * found for a given key. + */ + public static Context combine(Context... contexts) { + return combine(ContextHelpers::combineKeepingFirst, contexts); + } + + /** + * Combine multiple contexts into a single one. + * + * @param combiner The context combiner, taking already combined context as first parameter, any + * following one as second parameter, and returning the combined context. + * @param contexts The contexts to combine. + * @return The combined context. + */ + public static Context combine(BinaryOperator combiner, Context... contexts) { + requireNonNull(combiner, "combiner cannot be null"); + Context result = new IndexedContext(new Object[0]); + for (Context context : contexts) { + requireNonNull(context, "context cannot be null"); + result = combiner.apply(result, context); + } + return result; + } + + private static Context combineKeepingFirst(Context current, Context next) { + if (!(current instanceof IndexedContext)) { + throw new IllegalStateException("Left context is supposed to be an IndexedContext"); + } + IndexedContext currentIndexed = (IndexedContext) current; + if (next instanceof EmptyContext) { + return current; + } else if (next instanceof SingletonContext) { + SingletonContext nextSingleton = (SingletonContext) next; + // Check if the single next value is already define in current so next context can be skipped + if (nextSingleton.index < currentIndexed.store.length + && currentIndexed.store[nextSingleton.index] != null) { + return current; + } + // Always store next value otherwise + Object[] store = + copyOfRange( + currentIndexed.store, 0, max(currentIndexed.store.length, nextSingleton.index + 1)); + store[nextSingleton.index] = nextSingleton.value; + return new IndexedContext(store); + } else if (next instanceof IndexedContext) { + IndexedContext nextIndexed = (IndexedContext) next; + // Don't prematurely allocate store. Only allocate if: + // * nextIndexed has more values that currentIndexed, + // so the additional values will always be kept + // * nextIndexed has values that currentIndexed do not have + Object[] store = null; + // Allocate store if nextIndexed has more elements than currentIndexed + if (nextIndexed.store.length > currentIndexed.store.length) { + store = copyOfRange(currentIndexed.store, 0, nextIndexed.store.length); + } + // Apply nextIndexed values if not set in currentIndexed + for (int i = 0; i < currentIndexed.store.length; i++) { + Object nextValue = nextIndexed.store[i]; + if (nextValue != null && currentIndexed.store[i] == null) { + if (store == null) { + store = copyOfRange(currentIndexed.store, 0, currentIndexed.store.length); + } + store[i] = nextValue; + } + } + // Apply any additional values from nextIndexed if any + for (int i = currentIndexed.store.length; i < nextIndexed.store.length; i++) { + Object nextValue = nextIndexed.store[i]; + if (nextValue != null) { + store[i] = nextValue; + } + } + // If store was not allocated, no value from nextIndexed was taken + return store == null ? current : new IndexedContext(store); + } + throw new IllegalStateException("Unsupported context type: " + next.getClass().getName()); + } +} diff --git a/components/context/src/main/java/datadog/context/IndexedContext.java b/components/context/src/main/java/datadog/context/IndexedContext.java index cadc481d707..d7aa1731b2a 100644 --- a/components/context/src/main/java/datadog/context/IndexedContext.java +++ b/components/context/src/main/java/datadog/context/IndexedContext.java @@ -11,7 +11,7 @@ /** {@link Context} containing many values. */ @ParametersAreNonnullByDefault final class IndexedContext implements Context { - private final Object[] store; + final Object[] store; IndexedContext(Object[] store) { this.store = store; diff --git a/components/context/src/main/java/datadog/context/SingletonContext.java b/components/context/src/main/java/datadog/context/SingletonContext.java index 7a8a4e98b6f..4aa5e3f04cf 100644 --- a/components/context/src/main/java/datadog/context/SingletonContext.java +++ b/components/context/src/main/java/datadog/context/SingletonContext.java @@ -10,8 +10,8 @@ /** {@link Context} containing a single value. */ @ParametersAreNonnullByDefault final class SingletonContext implements Context { - private final int index; - private final Object value; + final int index; + final Object value; SingletonContext(int index, Object value) { this.index = index; diff --git a/components/context/src/test/java/datadog/context/ContextHelpersTest.java b/components/context/src/test/java/datadog/context/ContextHelpersTest.java new file mode 100644 index 00000000000..de2f1610a87 --- /dev/null +++ b/components/context/src/test/java/datadog/context/ContextHelpersTest.java @@ -0,0 +1,239 @@ +package datadog.context; + +import static datadog.context.Context.current; +import static datadog.context.Context.root; +import static datadog.context.ContextHelpers.CURRENT; +import static datadog.context.ContextHelpers.combine; +import static datadog.context.ContextHelpers.findAll; +import static datadog.context.ContextHelpers.findFirst; +import static datadog.context.ContextTest.BOOLEAN_KEY; +import static datadog.context.ContextTest.FLOAT_KEY; +import static datadog.context.ContextTest.LONG_KEY; +import static datadog.context.ContextTest.STRING_KEY; +import static java.util.Arrays.asList; +import static java.util.Collections.emptyList; +import static java.util.Collections.singleton; +import static java.util.logging.Level.ALL; +import static java.util.logging.Level.INFO; +import static java.util.logging.Level.SEVERE; +import static java.util.logging.Level.WARNING; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertIterableEquals; +import static org.junit.jupiter.api.Assertions.assertNotNull; +import static org.junit.jupiter.api.Assertions.assertNull; +import static org.junit.jupiter.api.Assertions.assertThrows; +import static org.junit.jupiter.params.provider.Arguments.arguments; + +import java.util.function.BinaryOperator; +import java.util.logging.Level; +import java.util.stream.Stream; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.Arguments; +import org.junit.jupiter.params.provider.MethodSource; + +class ContextHelpersTest { + private static final Object CARRIER_1 = new Object(); + private static final Object CARRIER_2 = new Object(); + private static final Object UNSET_CARRIER = new Object(); + private static final Object NON_CARRIER = new Object(); + private static final String VALUE_1 = "value1"; + private static final String VALUE_2 = "value2"; + + @BeforeAll + static void init() { + Context context1 = root().with(STRING_KEY, VALUE_1); + context1.attachTo(CARRIER_1); + + Context context2 = root().with(STRING_KEY, VALUE_2); + context2.attachTo(CARRIER_2); + + root().attachTo(UNSET_CARRIER); + } + + @ParameterizedTest + @MethodSource("findFirstArguments") + void testFindFirst(Object[] carriers, String expected) { + assertEquals(expected, findFirst(STRING_KEY, carriers), "Cannot find first value"); + } + + static Stream findFirstArguments() { + return Stream.of( + arguments(emptyArray(), null), + arguments(arrayOf(NON_CARRIER), null), + arguments(arrayOf(UNSET_CARRIER), null), + arguments(arrayOf(CARRIER_1), VALUE_1), + arguments(arrayOf(CARRIER_1, CARRIER_2), VALUE_1), + arguments(arrayOf(NON_CARRIER, CARRIER_1), VALUE_1), + arguments(arrayOf(UNSET_CARRIER, CARRIER_1), VALUE_1), + arguments(arrayOf(CARRIER_1, NON_CARRIER), VALUE_1), + arguments(arrayOf(CARRIER_1, UNSET_CARRIER), VALUE_1)); + } + + @ParameterizedTest + @MethodSource("findAllArguments") + void testFindAll(Object[] carriers, Iterable expected) { + assertIterableEquals(expected, findAll(STRING_KEY, carriers), "Cannot find all values"); + } + + static Stream findAllArguments() { + return Stream.of( + arguments(emptyArray(), emptyList()), + arguments(arrayOf(CARRIER_1), singleton(VALUE_1)), + arguments(arrayOf(CARRIER_1, CARRIER_2), asList(VALUE_1, VALUE_2)), + arguments(arrayOf(NON_CARRIER, CARRIER_1), singleton(VALUE_1)), + arguments(arrayOf(UNSET_CARRIER, CARRIER_1), singleton(VALUE_1)), + arguments(arrayOf(CARRIER_1, NON_CARRIER), singleton(VALUE_1)), + arguments(arrayOf(CARRIER_1, UNSET_CARRIER), singleton(VALUE_1))); + } + + @Test + void testNullCarriers() { + assertThrows( + NullPointerException.class, () -> findFirst(null, CARRIER_1), "Should fail on null key"); + assertThrows( + NullPointerException.class, + () -> findFirst(STRING_KEY, (Object) null), + "Should fail on null context"); + assertThrows( + NullPointerException.class, + () -> findFirst(STRING_KEY, null, CARRIER_1), + "Should fail on null context"); + assertThrows( + NullPointerException.class, () -> findAll(null, CARRIER_1), "Should fail on null key"); + assertThrows( + NullPointerException.class, + () -> findAll(STRING_KEY, (Object) null), + "Should fail on null context"); + assertThrows( + NullPointerException.class, + () -> findAll(STRING_KEY, null, CARRIER_1), + "Should fail on null context"); + } + + @Test + void testCurrent() { + assertEquals(root(), current(), "Current context is already set"); + Context context = root().with(STRING_KEY, VALUE_1); + try (ContextScope ignored = context.attach()) { + assertEquals( + VALUE_1, findFirst(STRING_KEY, CURRENT), "Failed to get value from current context"); + assertIterableEquals( + singleton(VALUE_1), + findAll(STRING_KEY, CURRENT), + "Failed to get value from current context"); + } + assertEquals(root(), current(), "Current context stayed attached"); + } + + @Test + void testCombine() { + // Test general case + Context context1 = root().with(STRING_KEY, VALUE_1).with(BOOLEAN_KEY, true); + Context context2 = root().with(STRING_KEY, VALUE_2).with(FLOAT_KEY, 3.14F); + Context context3 = root(); + Context context4 = root().with(FLOAT_KEY, 567F); + + Context combined = combine(context1, context2, context3, context4); + assertEquals(VALUE_1, combined.get(STRING_KEY), "First duplicate value should be kept"); + assertEquals(true, combined.get(BOOLEAN_KEY), "Values from first context should be kept"); + assertEquals(3.14F, combined.get(FLOAT_KEY), "Values from second context should be kept"); + + // Test SingletonContext optimization + context1 = root().with(STRING_KEY, VALUE_1); + context2 = root().with(STRING_KEY, VALUE_2); + combined = combine(context1, context2); + assertEquals(VALUE_1, combined.get(STRING_KEY), "First duplicate value should be kept"); + + // Test IndexedContext optimization where later context has more elements + context1 = root().with(STRING_KEY, VALUE_1).with(FLOAT_KEY, 3.14F); + context2 = root().with(STRING_KEY, VALUE_2).with(LONG_KEY, 567L); + combined = combine(context1, context2); + assertEquals(VALUE_1, combined.get(STRING_KEY), "First duplicate value should be kept"); + assertEquals(3.14F, combined.get(FLOAT_KEY), "Values from first context should be kept"); + assertEquals(567L, combined.get(LONG_KEY), "Values from first context should be kept"); + + // Test IndexedContext optimization where context has same size but only later context has value + context1 = root().with(LONG_KEY, 567L).with(STRING_KEY, VALUE_1); + context2 = root().with(LONG_KEY, 789L).with(FLOAT_KEY, 3.14F); + combined = combine(context1, context2); + assertEquals(567L, combined.get(LONG_KEY), "First duplicate value should be kept"); + assertEquals(VALUE_1, combined.get(STRING_KEY), "Values from first context should be kept"); + assertEquals(3.14F, combined.get(FLOAT_KEY), "Values from first context should be kept"); + + // Test IndexedContext optimization with same size context and no new values + context1 = root().with(STRING_KEY, VALUE_1).with(LONG_KEY, 567L); + context2 = root().with(STRING_KEY, VALUE_2).with(LONG_KEY, 789L); + combined = combine(context1, context2); + assertEquals(VALUE_1, combined.get(STRING_KEY), "First duplicate value should be kept"); + assertEquals(567L, combined.get(LONG_KEY), "First duplicate value should be kept"); + } + + @Test + void testCombiner() { + ContextKey errorKey = ContextKey.named("error"); + Context context1 = root().with(errorKey, ErrorStats.from(INFO, 12)).with(STRING_KEY, VALUE_1); + Context context2 = root().with(errorKey, ErrorStats.from(SEVERE, 1)).with(FLOAT_KEY, 3.14F); + Context context3 = root().with(errorKey, ErrorStats.from(WARNING, 6)).with(BOOLEAN_KEY, true); + + BinaryOperator errorStatsMerger = + (left, right) -> { + ErrorStats mergedStats = ErrorStats.merge(left.get(errorKey), right.get(errorKey)); + return left.with(errorKey, mergedStats); + }; + Context combined = combine(errorStatsMerger, context1, context2, context3); + ErrorStats combinedStats = combined.get(errorKey); + assertNotNull(combinedStats, "Failed to combined error stats"); + assertEquals(19, combinedStats.errorCount, "Failed to combine error stats"); + assertEquals(SEVERE, combinedStats.maxLevel, "Failed to combine error stats"); + assertNull(combined.get(STRING_KEY), "Combiner should drop any other context values"); + assertNull(combined.get(FLOAT_KEY), "Combiner should drop any other context values"); + assertNull(combined.get(BOOLEAN_KEY), "Combiner should drop any other context values"); + } + + @Test + void testNullCombine() { + assertThrows( + NullPointerException.class, + () -> combine((BinaryOperator) null, root()), + "Should fail on null combiner"); + assertThrows( + NullPointerException.class, + () -> combine((left, right) -> left, (Context) null), + "Should fail on null context"); + } + + private static class ErrorStats { + int errorCount; + Level maxLevel; + + public ErrorStats() { + this.errorCount = 0; + this.maxLevel = ALL; + } + + public static ErrorStats from(Level logLevel, int count) { + ErrorStats stats = new ErrorStats(); + stats.errorCount = count; + stats.maxLevel = logLevel; + return stats; + } + + public static ErrorStats merge(ErrorStats a, ErrorStats b) { + if (a == null) { + return b; + } + Level maxLevel = a.maxLevel.intValue() > b.maxLevel.intValue() ? a.maxLevel : b.maxLevel; + return from(maxLevel, a.errorCount + b.errorCount); + } + } + + private static Object[] emptyArray() { + return new Object[0]; + } + + private static Object[] arrayOf(Object... objects) { + return objects; + } +}