From c2fd1573731f6e711dd39ccbf7ff0f0af6ca13e7 Mon Sep 17 00:00:00 2001 From: Brian Harrington Date: Wed, 13 Mar 2024 16:30:00 -0500 Subject: [PATCH] only consider grouping keys for id mapping When using a group by expression, only consider the tags that are part of the grouping rather than the full set when mapping the id. --- .../spectator/atlas/impl/DataExpr.java | 8 +++++-- .../spectator/atlas/impl/Evaluator.java | 23 +++++++++++++++---- .../spectator/atlas/impl/EvaluatorConfig.java | 6 +++-- .../spectator/atlas/impl/IdMapper.java | 20 ++++++++++------ .../netflix/spectator/atlas/impl/Parser.java | 3 ++- 5 files changed, 43 insertions(+), 17 deletions(-) diff --git a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/DataExpr.java b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/DataExpr.java index f2dd9480e..e21aaf731 100644 --- a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/DataExpr.java +++ b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/DataExpr.java @@ -471,15 +471,19 @@ final class Count implements AggregateFunction { final class GroupBy implements DataExpr { private final AggregateFunction af; - private final List keys; + private final Set keys; /** Create a new instance. */ - GroupBy(AggregateFunction af, List keys) { + GroupBy(AggregateFunction af, Set keys) { Preconditions.checkArg(!keys.isEmpty(), "key list for group by cannot be empty"); this.af = af; this.keys = keys; } + Set keys() { + return keys; + } + @SuppressWarnings("PMD.ReturnEmptyCollectionRatherThanNull") private Map keyTags(Map tags) { Map result = new HashMap<>(); diff --git a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Evaluator.java b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Evaluator.java index 8f843b532..1aed07134 100644 --- a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Evaluator.java +++ b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Evaluator.java @@ -34,8 +34,8 @@ import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; +import java.util.function.BiFunction; import java.util.function.Consumer; -import java.util.function.Function; /** * Evaluates all the expressions for a set of subscriptions. @@ -49,7 +49,7 @@ public class Evaluator { private final Lock lock = new ReentrantLock(); private final Map commonTags; - private final Function> idMapper; + private final BiFunction, Map> idMapper; private final long step; private final boolean delayGaugeAggregation; private final QueryIndex index; @@ -174,11 +174,12 @@ public EvalPayload eval(long timestamp) { final double v = consolidator.value(timestamp); if (!Double.isNaN(v)) { Map tags = Collections.emptyMap(); - if (!(expr instanceof DataExpr.AggregateFunction)) { + if (expr instanceof DataExpr.GroupBy) { // Aggregation functions only use tags based on the expression. Avoid overhead of // considering the tags for the data. - tags = idMapper.apply(entry.getKey()); - tags.putAll(commonTags); + Set keys = ((DataExpr.GroupBy) expr).keys(); + tags = idMapper.apply(entry.getKey(), keys); + putCommonTags(tags, keys); } if (delayGaugeAggr && consolidator.isGauge()) { // When performing a group by, datapoints missing tag used for the grouping @@ -211,6 +212,18 @@ public EvalPayload eval(long timestamp) { return new EvalPayload(timestamp, metrics); } + private void putCommonTags(Map dst, Set keys) { + if (dst.size() < keys.size()) { + // Skip this step unless there is something pending + for (String key : keys) { + String value = commonTags.get(key); + if (value != null) { + dst.put(key, value); + } + } + } + } + private String idHash(Id id) { Hash64 hasher = new Hash64(); hasher.updateString(id.name()); diff --git a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/EvaluatorConfig.java b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/EvaluatorConfig.java index 0b16e3dc8..c7412e872 100644 --- a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/EvaluatorConfig.java +++ b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/EvaluatorConfig.java @@ -20,6 +20,8 @@ import com.netflix.spectator.atlas.AtlasConfig; import java.util.Map; +import java.util.Set; +import java.util.function.BiFunction; import java.util.function.Function; /** @@ -45,7 +47,7 @@ static EvaluatorConfig fromAtlasConfig(AtlasConfig config) { return config.commonTags(); } - @Override public Function> idMapper() { + @Override public BiFunction, Map> idMapper() { return new IdMapper(JsonUtils.createReplacementFunction(config.validTagCharacters())); } }; @@ -59,7 +61,7 @@ static EvaluatorConfig fromAtlasConfig(AtlasConfig config) { Map commonTags(); /** Function to convert an id to a map of key/value pairs. */ - default Function> idMapper() { + default BiFunction, Map> idMapper() { return new IdMapper(Function.identity()); } diff --git a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/IdMapper.java b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/IdMapper.java index c5ca866fe..222789361 100644 --- a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/IdMapper.java +++ b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/IdMapper.java @@ -19,6 +19,8 @@ import java.util.HashMap; import java.util.Map; +import java.util.Set; +import java.util.function.BiFunction; import java.util.function.Function; /** @@ -27,7 +29,7 @@ *

Classes in this package are only intended for use internally within spectator. They may * change at any time and without notice. */ -public final class IdMapper implements Function> { +public final class IdMapper implements BiFunction, Map> { private final Function fixTagString; @@ -37,21 +39,25 @@ public IdMapper(Function fixTagString) { } @Override - public Map apply(Id id) { + public Map apply(Id id, Set keys) { int size = id.size(); - Map tags = new HashMap<>(size); + Map tags = new HashMap<>(keys.size()); // Start at 1 as name will be added last for (int i = 1; i < size; ++i) { String k = fixTagString.apply(id.getKey(i)); - String v = fixTagString.apply(id.getValue(i)); - tags.put(k, v); + if (keys.contains(k)) { + String v = fixTagString.apply(id.getValue(i)); + tags.put(k, v); + } } // Add the name, it is added last so it will have precedence if the user tried to // use a tag key of "name". - String name = fixTagString.apply(id.name()); - tags.put("name", name); + if (keys.contains("name")) { + String name = fixTagString.apply(id.name()); + tags.put("name", name); + } return tags; } diff --git a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Parser.java b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Parser.java index 6272af69e..a8c1a2790 100644 --- a/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Parser.java +++ b/spectator-reg-atlas/src/main/java/com/netflix/spectator/atlas/impl/Parser.java @@ -21,6 +21,7 @@ import java.util.ArrayList; import java.util.Deque; import java.util.HashSet; +import java.util.LinkedHashSet; import java.util.List; /** @@ -196,7 +197,7 @@ private static Object parse(String expr) { case ":by": tmp = (List) stack.pop(); af = (DataExpr.AggregateFunction) stack.pop(); - stack.push(new DataExpr.GroupBy(af, tmp)); + stack.push(new DataExpr.GroupBy(af, new LinkedHashSet<>(tmp))); break; case ":rollup-drop": tmp = (List) stack.pop();