From 9ed6534477adada8ec6bb7e4fa329ccdc2128cb8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9s=20de=20la=20Pe=C3=B1a?= Date: Tue, 14 Jan 2025 15:06:24 +0000 Subject: [PATCH] CNDB-12041: Add missing check of guardrail sai_sstable_indexes_per_query --- .../cassandra/index/sai/plan/FilterTree.java | 21 +++++++++++++ .../index/sai/plan/QueryController.java | 22 -------------- .../plan/StorageAttachedIndexSearcher.java | 30 +++++++++++++++++++ .../reads/thresholds/WarningsSnapshot.java | 5 +++- ...rdrailNonPartitionRestrictedQueryTest.java | 6 ++-- 5 files changed, 58 insertions(+), 26 deletions(-) diff --git a/src/java/org/apache/cassandra/index/sai/plan/FilterTree.java b/src/java/org/apache/cassandra/index/sai/plan/FilterTree.java index 4017d5ab9a12..84c6d351f95d 100644 --- a/src/java/org/apache/cassandra/index/sai/plan/FilterTree.java +++ b/src/java/org/apache/cassandra/index/sai/plan/FilterTree.java @@ -19,15 +19,18 @@ import java.nio.ByteBuffer; import java.util.ArrayList; +import java.util.HashSet; import java.util.Iterator; import java.util.List; import java.util.ListIterator; +import java.util.Set; import com.google.common.collect.ListMultimap; import org.apache.cassandra.db.DecoratedKey; import org.apache.cassandra.db.rows.Row; import org.apache.cassandra.db.rows.Unfiltered; +import org.apache.cassandra.index.sai.SSTableIndex; import org.apache.cassandra.index.sai.utils.TypeUtil; import org.apache.cassandra.schema.ColumnMetadata; import org.apache.cassandra.schema.ColumnMetadata.Kind; @@ -130,4 +133,22 @@ private boolean localSatisfiedBy(DecoratedKey key, Unfiltered currentCluster, Ro private boolean shouldReturnNow(boolean result) { return (op == OperationType.AND && !result) || (op == OperationType.OR && result); } + + /** + * @return the number of unique SSTable indexes that are referenced by the expressions in this filter tree. + */ + public int numSSTableIndexes() + { + Set referencedIndexes = new HashSet<>(); + sstableIndexes(referencedIndexes); + return referencedIndexes.size(); + } + + private void sstableIndexes(Set indexes) + { + for (Expression expression : expressions.values()) + indexes.addAll(expression.context.getView().getIndexes()); + for (FilterTree child : children) + child.sstableIndexes(indexes); + } } diff --git a/src/java/org/apache/cassandra/index/sai/plan/QueryController.java b/src/java/org/apache/cassandra/index/sai/plan/QueryController.java index 01104d2c195b..ae1fe011b1e9 100644 --- a/src/java/org/apache/cassandra/index/sai/plan/QueryController.java +++ b/src/java/org/apache/cassandra/index/sai/plan/QueryController.java @@ -43,7 +43,6 @@ import org.apache.cassandra.db.ColumnFamilyStore; import org.apache.cassandra.db.DataRange; import org.apache.cassandra.db.DecoratedKey; -import org.apache.cassandra.db.MessageParams; import org.apache.cassandra.db.MultiRangeReadCommand; import org.apache.cassandra.db.PartitionPosition; import org.apache.cassandra.db.PartitionRangeReadCommand; @@ -55,7 +54,6 @@ import org.apache.cassandra.db.filter.ClusteringIndexSliceFilter; import org.apache.cassandra.db.filter.DataLimits; import org.apache.cassandra.db.filter.RowFilter; -import org.apache.cassandra.db.guardrails.Guardrails; import org.apache.cassandra.db.lifecycle.SSTableSet; import org.apache.cassandra.db.marshal.AbstractType; import org.apache.cassandra.db.marshal.CollectionType; @@ -87,7 +85,6 @@ import org.apache.cassandra.index.sai.view.View; import org.apache.cassandra.io.sstable.format.SSTableReader; import org.apache.cassandra.io.util.FileUtils; -import org.apache.cassandra.net.ParamType; import org.apache.cassandra.schema.TableMetadata; import org.apache.cassandra.tracing.Tracing; import org.apache.cassandra.utils.CloseableIterator; @@ -822,25 +819,6 @@ private void closeQueryViews() } } - void maybeTriggerReferencedIndexesGuardrail(int numReferencedIndexes) - { - if (Guardrails.saiSSTableIndexesPerQuery.failsOn(numReferencedIndexes, null)) - { - String msg = String.format("Query %s attempted to read from too many indexes (%s) but max allowed is %s; " + - "query aborted (see sai_sstable_indexes_per_query_fail_threshold)", - command.toCQLString(), - numReferencedIndexes, - Guardrails.CONFIG_PROVIDER.getOrCreate(null).getSaiSSTableIndexesPerQueryFailThreshold()); - Tracing.trace(msg); - MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_FAIL, numReferencedIndexes); - throw new QueryReferencingTooManyIndexesException(msg); - } - else if (Guardrails.saiSSTableIndexesPerQuery.warnsOn(numReferencedIndexes, null)) - { - MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_WARN, numReferencedIndexes); - } - } - /** * Returns the {@link DataRange} list covered by the specified {@link ReadCommand}. * diff --git a/src/java/org/apache/cassandra/index/sai/plan/StorageAttachedIndexSearcher.java b/src/java/org/apache/cassandra/index/sai/plan/StorageAttachedIndexSearcher.java index ceadabbc22af..3d3a21057855 100644 --- a/src/java/org/apache/cassandra/index/sai/plan/StorageAttachedIndexSearcher.java +++ b/src/java/org/apache/cassandra/index/sai/plan/StorageAttachedIndexSearcher.java @@ -42,9 +42,11 @@ import org.apache.cassandra.db.ColumnFamilyStore; import org.apache.cassandra.db.DataRange; import org.apache.cassandra.db.DecoratedKey; +import org.apache.cassandra.db.MessageParams; import org.apache.cassandra.db.PartitionPosition; import org.apache.cassandra.db.ReadCommand; import org.apache.cassandra.db.ReadExecutionController; +import org.apache.cassandra.db.guardrails.Guardrails; import org.apache.cassandra.db.partitions.UnfilteredPartitionIterator; import org.apache.cassandra.db.rows.AbstractUnfilteredRowIterator; import org.apache.cassandra.db.rows.Row; @@ -63,7 +65,9 @@ import org.apache.cassandra.index.sai.utils.RangeUtil; import org.apache.cassandra.index.sai.utils.PrimaryKeyWithSortKey; import org.apache.cassandra.io.util.FileUtils; +import org.apache.cassandra.net.ParamType; import org.apache.cassandra.schema.TableMetadata; +import org.apache.cassandra.tracing.Tracing; import org.apache.cassandra.utils.AbstractIterator; import org.apache.cassandra.utils.Clock; import org.apache.cassandra.utils.CloseableIterator; @@ -109,6 +113,8 @@ public UnfilteredPartitionIterator search(ReadExecutionController executionContr try { FilterTree filterTree = analyzeFilter(); + maybeTriggerReferencedIndexesGuardrail(filterTree); + Plan plan = controller.buildPlan(); Iterator keysIterator = controller.buildIterator(plan); @@ -154,6 +160,30 @@ public UnfilteredPartitionIterator search(ReadExecutionController executionContr } } + private void maybeTriggerReferencedIndexesGuardrail(FilterTree filterTree) + { + if (!Guardrails.saiSSTableIndexesPerQuery.enabled()) + return; + + int numReferencedIndexes = filterTree.numSSTableIndexes(); + + if (Guardrails.saiSSTableIndexesPerQuery.failsOn(numReferencedIndexes, null)) + { + String msg = String.format("Query %s attempted to read from too many indexes (%s) but max allowed is %s; " + + "query aborted (see sai_sstable_indexes_per_query_fail_threshold)", + command.toCQLString(), + numReferencedIndexes, + Guardrails.CONFIG_PROVIDER.getOrCreate(null).getSaiSSTableIndexesPerQueryFailThreshold()); + Tracing.trace(msg); + MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_FAIL, numReferencedIndexes); + throw new QueryReferencingTooManyIndexesException(msg); + } + else if (Guardrails.saiSSTableIndexesPerQuery.warnsOn(numReferencedIndexes, null)) + { + MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_WARN, numReferencedIndexes); + } + } + /** * Converts expressions into filter tree (which is currently just a single AND). *

diff --git a/src/java/org/apache/cassandra/service/reads/thresholds/WarningsSnapshot.java b/src/java/org/apache/cassandra/service/reads/thresholds/WarningsSnapshot.java index 0a07c8360ea9..591b73cc9412 100644 --- a/src/java/org/apache/cassandra/service/reads/thresholds/WarningsSnapshot.java +++ b/src/java/org/apache/cassandra/service/reads/thresholds/WarningsSnapshot.java @@ -194,7 +194,10 @@ public int hashCode() @Override public String toString() { - return "(tombstones=" + tombstones + ", localReadSize=" + localReadSize + ", rowIndexTooLarge=" + rowIndexReadSize + ')'; + return "(tombstones=" + tombstones + + ", localReadSize=" + localReadSize + + ", rowIndexTooLarge=" + rowIndexReadSize + + ", indexReadSSTablesCount=" + indexReadSSTablesCount + ')'; } public static final class Warnings diff --git a/test/distributed/org/apache/cassandra/distributed/test/guardrails/GuardrailNonPartitionRestrictedQueryTest.java b/test/distributed/org/apache/cassandra/distributed/test/guardrails/GuardrailNonPartitionRestrictedQueryTest.java index ab67e3c9936d..70249c2132ec 100644 --- a/test/distributed/org/apache/cassandra/distributed/test/guardrails/GuardrailNonPartitionRestrictedQueryTest.java +++ b/test/distributed/org/apache/cassandra/distributed/test/guardrails/GuardrailNonPartitionRestrictedQueryTest.java @@ -144,11 +144,11 @@ public void testSAIWarnThreshold() assertWarnAborts(0, 0); - // create 3 more SSTables on each node, this will trigger warn threshold (3 > 2 but < 5) + // create 3 more SSTables on each node, this will trigger warn threshold (4 > 2 but < 5) valueToQuery = createSSTables(3); String valueToQueryString = LongType.instance.toCQLString(LongType.instance.decompose(valueToQuery), true); String expectedMessage = tooManyIndexesReadWarnMessage(cluster.size(), - 3, + 4, String.format("SELECT * FROM %s.%s WHERE v1 = %s ALLOW FILTERING", KEYSPACE, tableName, valueToQueryString)); assertThat(getOnlyElement(executeSelect(valueToQuery, false))).contains(expectedMessage); @@ -168,7 +168,7 @@ public void testSAIWarnThreshold() // notice we expect warnings from 2 nodes expectedMessage = tooManyIndexesReadWarnMessage(cluster.size() - 1, - 3, + 4, String.format("SELECT * FROM %s.%s WHERE v1 = %s ALLOW FILTERING", KEYSPACE, tableName, valueToQueryString));