All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.cassandra.index.sai.plan.QueryController Maven / Gradle / Ivy

Go to download

The Apache Cassandra Project develops a highly scalable second-generation distributed database, bringing together Dynamo's fully distributed design and Bigtable's ColumnFamily-based data model.

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.apache.cassandra.index.sai.plan;

import java.io.IOException;
import java.io.UncheckedIOException;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.NavigableSet;
import java.util.stream.Collectors;

import javax.annotation.Nullable;

import com.google.common.collect.Lists;

import org.apache.cassandra.cql3.Operator;
import org.apache.cassandra.db.Clustering;
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.PartitionRangeReadCommand;
import org.apache.cassandra.db.ReadCommand;
import org.apache.cassandra.db.ReadExecutionController;
import org.apache.cassandra.db.SinglePartitionReadCommand;
import org.apache.cassandra.db.filter.ClusteringIndexFilter;
import org.apache.cassandra.db.filter.ClusteringIndexNamesFilter;
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.rows.Row;
import org.apache.cassandra.db.rows.UnfilteredRowIterator;
import org.apache.cassandra.dht.AbstractBounds;
import org.apache.cassandra.index.sai.QueryContext;
import org.apache.cassandra.index.sai.StorageAttachedIndex;
import org.apache.cassandra.index.sai.VectorQueryContext;
import org.apache.cassandra.index.sai.disk.IndexSearchResultIterator;
import org.apache.cassandra.index.sai.disk.SSTableIndex;
import org.apache.cassandra.index.sai.iterators.KeyRangeConcatIterator;
import org.apache.cassandra.index.sai.iterators.KeyRangeIntersectionIterator;
import org.apache.cassandra.index.sai.iterators.KeyRangeIterator;
import org.apache.cassandra.index.sai.iterators.KeyRangeOrderingIterator;
import org.apache.cassandra.index.sai.iterators.KeyRangeUnionIterator;
import org.apache.cassandra.index.sai.utils.PrimaryKey;
import org.apache.cassandra.net.ParamType;
import org.apache.cassandra.schema.TableMetadata;
import org.apache.cassandra.tracing.Tracing;
import org.apache.cassandra.utils.InsertionOrderedNavigableSet;
import org.apache.cassandra.utils.Throwables;

import static org.apache.cassandra.config.CassandraRelevantProperties.SAI_VECTOR_SEARCH_ORDER_CHUNK_SIZE;

public class QueryController
{
    final QueryContext queryContext;

    private final ColumnFamilyStore cfs;
    private final ReadCommand command;
    private final RowFilter indexFilter;
    private final List ranges;
    private final AbstractBounds mergeRange;
    private final PrimaryKey.Factory keyFactory;
    private final PrimaryKey firstPrimaryKey;
    private final PrimaryKey lastPrimaryKey;
    private final int orderChunkSize;

    private final NavigableSet> nextClusterings;

    public QueryController(ColumnFamilyStore cfs,
                           ReadCommand command,
                           RowFilter indexFilter,
                           QueryContext queryContext)
    {
        this.cfs = cfs;
        this.command = command;
        this.queryContext = queryContext;
        this.indexFilter = indexFilter;
        this.ranges = dataRanges(command);
        DataRange first = ranges.get(0);
        DataRange last = ranges.get(ranges.size() - 1);
        this.mergeRange = ranges.size() == 1 ? first.keyRange() : first.keyRange().withNewRight(last.keyRange().right);
        this.keyFactory = new PrimaryKey.Factory(cfs.getPartitioner(), cfs.getComparator());
        this.firstPrimaryKey = keyFactory.create(mergeRange.left.getToken());
        this.lastPrimaryKey = keyFactory.create(mergeRange.right.getToken());
        this.orderChunkSize = SAI_VECTOR_SEARCH_ORDER_CHUNK_SIZE.getInt();
        this.nextClusterings = new InsertionOrderedNavigableSet<>(cfs.metadata().comparator);
    }

    public PrimaryKey.Factory primaryKeyFactory()
    {
        return keyFactory;
    }

    public PrimaryKey firstPrimaryKeyInRange()
    {
        return firstPrimaryKey;
    }

    public PrimaryKey lastPrimaryKeyInRange()
    {
        return lastPrimaryKey;
    }

    public TableMetadata metadata()
    {
        return command.metadata();
    }

    public RowFilter indexFilter()
    {
        return this.indexFilter;
    }
    
    public boolean usesStrictFiltering()
    {
        return command.rowFilter().isStrict();
    }

    /**
     * @return token ranges used in the read command
     */
    public List dataRanges()
    {
        return ranges;
    }

    @Nullable
    public StorageAttachedIndex indexFor(RowFilter.Expression expression)
    {
        return cfs.indexManager.getBestIndexFor(expression, StorageAttachedIndex.class).orElse(null);
    }

    public boolean hasAnalyzer(RowFilter.Expression expression)
    {
        StorageAttachedIndex index = indexFor(expression);
        return index != null && index.hasAnalyzer();
    }

    public UnfilteredRowIterator queryStorage(List keys, ReadExecutionController executionController)
    {
        if (keys.isEmpty())
            throw new IllegalArgumentException("At least one primary key is required!");

        SinglePartitionReadCommand partition = SinglePartitionReadCommand.create(cfs.metadata(),
                                                                                 command.nowInSec(),
                                                                                 command.columnFilter(),
                                                                                 RowFilter.none(),
                                                                                 DataLimits.NONE,
                                                                                 keys.get(0).partitionKey(),
                                                                                 makeFilter(keys));

        return partition.queryMemtableAndDisk(cfs, executionController);
    }

    /**
     * Build a {@link KeyRangeIterator.Builder} from the given list of {@link Expression}s.
     * 

* This is achieved by creating an on-disk view of the query that maps the expressions to * the {@link SSTableIndex}s that will satisfy the expression. *

* Each {@link QueryViewBuilder.QueryExpressionView} is then passed to * {@link IndexSearchResultIterator#build(QueryViewBuilder.QueryExpressionView, AbstractBounds, QueryContext, boolean, Runnable)} * to search the in-memory indexes associated with the expression and the SSTable indexes, the results of * which are unioned and returned. *

* The results from each call to {@link IndexSearchResultIterator#build(QueryViewBuilder.QueryExpressionView, AbstractBounds, QueryContext, boolean, Runnable)} * are added to a {@link KeyRangeIntersectionIterator} and returned if strict filtering is allowed. *

* If strict filtering is not allowed, indexes are split into two groups according to the repaired status of their * backing SSTables. Results from searches over the repaired group are added to a * {@link KeyRangeIntersectionIterator}, which is then added, along with results from searches on the unrepaired * set, to a top-level {@link KeyRangeUnionIterator}, and returned. This is done to ensure that AND queries do not * prematurely filter out matches on un-repaired partial updates. Post-filtering must also take this into * account. (see {@link FilterTree#isSatisfiedBy(DecoratedKey, Row, Row)}) Note that Memtable-attached * indexes are treated as part of the unrepaired set. */ public KeyRangeIterator.Builder getIndexQueryResults(Collection expressions) { // VSTODO move ANN out of expressions and into its own abstraction? That will help get generic ORDER BY support expressions = expressions.stream().filter(e -> e.getIndexOperator() != Expression.IndexOperator.ANN).collect(Collectors.toList()); QueryViewBuilder.QueryView queryView = new QueryViewBuilder(expressions, mergeRange).build(); Runnable onClose = () -> queryView.referencedIndexes.forEach(SSTableIndex::releaseQuietly); KeyRangeIterator.Builder builder = command.rowFilter().isStrict() ? KeyRangeIntersectionIterator.builder(expressions.size(), onClose) : KeyRangeUnionIterator.builder(expressions.size(), onClose); try { maybeTriggerGuardrails(queryView); if (command.rowFilter().isStrict()) { // If strict filtering is enabled, evaluate indexes for both repaired and un-repaired SSTables together. // This usually means we are making this local index query in the context of a user query that reads // from a single replica and thus can safely perform local intersections. for (QueryViewBuilder.QueryExpressionView queryExpressionView : queryView.view) builder.add(IndexSearchResultIterator.build(queryExpressionView, mergeRange, queryContext, true, () -> {})); } else { KeyRangeIterator.Builder repairedBuilder = KeyRangeIntersectionIterator.builder(expressions.size(), () -> {}); for (QueryViewBuilder.QueryExpressionView queryExpressionView : queryView.view) { Expression expression = queryExpressionView.expression; // The initial sizes here reflect little more than an effort to avoid resizing for // partition-restricted searches w/ LCS: List repaired = new ArrayList<>(5); List unrepaired = new ArrayList<>(5); // Split SSTable indexes into repaired and un-reparired: for (SSTableIndex index : queryExpressionView.sstableIndexes) if (index.getSSTable().isRepaired()) repaired.add(index); else unrepaired.add(index); // Always build an iterator for the un-repaired set, given this must include Memtable indexes... IndexSearchResultIterator unrepairedIterator = IndexSearchResultIterator.build(expression, queryExpressionView.memtableIndexes, unrepaired, mergeRange, queryContext, true, () -> {}); // ...but ignore it if our combined results are empty. if (unrepairedIterator.getMaxKeys() > 0) { builder.add(unrepairedIterator); queryContext.hasUnrepairedMatches = true; } else { // We're not going to use this, so release the resources it holds. unrepairedIterator.close(); } // ...then only add an iterator to the repaired intersection if repaired SSTable indexes exist. if (!repaired.isEmpty()) repairedBuilder.add(IndexSearchResultIterator.build(expression, Collections.emptyList(), repaired, mergeRange, queryContext, false, () -> {})); } if (repairedBuilder.rangeCount() > 0) builder.add(repairedBuilder.build()); } } catch (Throwable t) { // all sstable indexes in view have been referenced, need to clean up when exception is thrown builder.cleanup(); throw t; } return builder; } private void maybeTriggerGuardrails(QueryViewBuilder.QueryView queryView) { int referencedIndexes = 0; // We want to make sure that no individual column expression touches too many SSTable-attached indexes: for (QueryViewBuilder.QueryExpressionView expressionSSTables : queryView.view) referencedIndexes = Math.max(referencedIndexes, expressionSSTables.sstableIndexes.size()); if (Guardrails.saiSSTableIndexesPerQuery.failsOn(referencedIndexes, 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(), referencedIndexes, Guardrails.CONFIG_PROVIDER.getOrCreate(null).getSaiSSTableIndexesPerQueryFailThreshold()); Tracing.trace(msg); MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_FAIL, referencedIndexes); throw new QueryReferencingTooManyIndexesException(msg); } else if (Guardrails.saiSSTableIndexesPerQuery.warnsOn(referencedIndexes, null)) { MessageParams.add(ParamType.TOO_MANY_REFERENCED_INDEXES_WARN, referencedIndexes); } } /** * Returns whether this query is not selecting the {@link PrimaryKey}. * The query does not select the key if both of the following statements are false: * 1. The table associated with the query is not using clustering keys * 2. The clustering index filter for the command wants the row. *

* Item 2 is important in paged queries where the {@link org.apache.cassandra.db.filter.ClusteringIndexSliceFilter} for * subsequent paged queries may not select rows that are returned by the index * search because that is initially partition based. * * @param key The {@link PrimaryKey} to be tested * @return true if the key is not selected by the query */ public boolean doesNotSelect(PrimaryKey key) { return key.kind() == PrimaryKey.Kind.WIDE && !command.clusteringIndexFilter(key.partitionKey()).selects(key.clustering()); } // This is an ANN only query public KeyRangeIterator getTopKRows(RowFilter.Expression expression) { assert expression.operator() == Operator.ANN; StorageAttachedIndex index = indexFor(expression); assert index != null; Expression planExpression = Expression.create(index).add(Operator.ANN, expression.getIndexValue().duplicate()); QueryViewBuilder.QueryView queryView = new QueryViewBuilder(Collections.singleton(planExpression), mergeRange).build(); Runnable onClose = () -> queryView.referencedIndexes.forEach(SSTableIndex::releaseQuietly); try { List memtableResults = queryView.view .stream() .map(v -> v.memtableIndexes) .flatMap(Collection::stream) .map(idx -> idx.search(queryContext, planExpression, mergeRange)) .collect(Collectors.toList()); List sstableIntersections = queryView.view .stream() .map(this::createRowIdIterator) .collect(Collectors.toList()); return IndexSearchResultIterator.build(sstableIntersections, memtableResults, queryView.referencedIndexes, queryContext, onClose); } catch (Throwable t) { // all sstable indexes in view have been referenced, need to clean up when exception is thrown onClose.run(); throw t; } } // This is a hybrid query. We apply all other predicates before ordering and limiting. public KeyRangeIterator getTopKRows(KeyRangeIterator source, RowFilter.Expression expression) { return new KeyRangeOrderingIterator(source, orderChunkSize, list -> this.getTopKRows(list, expression)); } private KeyRangeIterator getTopKRows(List rawSourceKeys, RowFilter.Expression expression) { VectorQueryContext vectorQueryContext = queryContext.vectorContext(); // Filter out PKs now. Each PK is passed to every segment of the ANN index, so filtering shadowed keys // eagerly can save some work when going from PK to row id for on disk segments. // Since the result is shared with multiple streams, we use an unmodifiable list. List sourceKeys = rawSourceKeys.stream().filter(vectorQueryContext::shouldInclude).collect(Collectors.toList()); StorageAttachedIndex index = indexFor(expression); assert index != null : "Cannot do ANN ordering on an unindexed column"; Expression planExpression = Expression.create(index); planExpression.add(Operator.ANN, expression.getIndexValue().duplicate()); QueryViewBuilder.QueryView queryView = new QueryViewBuilder(Collections.singleton(planExpression), mergeRange).build(); Runnable onClose = () -> queryView.referencedIndexes.forEach(SSTableIndex::releaseQuietly); try { List memtableResults = queryView.view .stream() .map(v -> v.memtableIndexes) .flatMap(Collection::stream) .map(idx -> idx.limitToTopResults(sourceKeys, planExpression, vectorQueryContext.limit())) .collect(Collectors.toList()); List sstableIntersections = queryView.view .stream() .flatMap(pair -> pair.sstableIndexes.stream()) .map(idx -> { try { return idx.limitToTopKResults(queryContext, sourceKeys, planExpression); } catch (IOException e) { throw new UncheckedIOException(e); } }) .collect(Collectors.toList()); return IndexSearchResultIterator.build(sstableIntersections, memtableResults, queryView.referencedIndexes, queryContext, onClose); } catch (Throwable t) { // all sstable indexes in view have been referenced, need to clean up when exception is thrown onClose.run(); throw t; } } /** * Create row id iterator from different indexes' on-disk searcher of the same sstable */ private KeyRangeIterator createRowIdIterator(QueryViewBuilder.QueryExpressionView indexExpression) { List subIterators = indexExpression.sstableIndexes .stream() .map(index -> { try { List iterators = index.search(indexExpression.expression, mergeRange, queryContext); // concat the result from multiple segments for the same index return KeyRangeConcatIterator.builder(iterators.size()).add(iterators).build(); } catch (Throwable ex) { throw Throwables.cleaned(ex); } }).collect(Collectors.toList()); return KeyRangeUnionIterator.build(subIterators); } // Note: This method assumes that the selects method has already been called for the // key to avoid having to (potentially) call selects twice private ClusteringIndexFilter makeFilter(List keys) { PrimaryKey firstKey = keys.get(0); assert cfs.metadata().comparator.size() == 0 && !firstKey.kind().hasClustering || cfs.metadata().comparator.size() > 0 && firstKey.kind().hasClustering : "PrimaryKey " + firstKey + " clustering does not match table. There should be a clustering of size " + cfs.metadata().comparator.size(); ClusteringIndexFilter clusteringIndexFilter = command.clusteringIndexFilter(firstKey.partitionKey()); // If we have skinny partitions or the key is for a static row then we need to get the partition as // requested by the original query. if (cfs.metadata().comparator.size() == 0 || firstKey.kind() == PrimaryKey.Kind.STATIC) { return clusteringIndexFilter; } else { nextClusterings.clear(); for (PrimaryKey key : keys) nextClusterings.add(key.clustering()); return new ClusteringIndexNamesFilter(nextClusterings, clusteringIndexFilter.isReversed()); } } /** * Returns the {@link DataRange} list covered by the specified {@link ReadCommand}. * * @param command a read command * @return the data ranges covered by {@code command} */ private static List dataRanges(ReadCommand command) { if (command instanceof SinglePartitionReadCommand) { return Lists.newArrayList(command.dataRange()); } else if (command instanceof PartitionRangeReadCommand) { return Lists.newArrayList(command.dataRange()); } else { throw new AssertionError("Unsupported read command type: " + command.getClass().getName()); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy