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

io.stargate.db.query.builder.WhereProcessor Maven / Gradle / Ivy

package io.stargate.db.query.builder;

import static com.datastax.oss.driver.shaded.guava.common.base.Preconditions.checkArgument;
import static java.lang.String.format;

import io.stargate.db.query.PartitionKey;
import io.stargate.db.query.Predicate;
import io.stargate.db.query.PrimaryKey;
import io.stargate.db.query.RowsImpacted;
import io.stargate.db.query.RowsImpacted.Ranges;
import io.stargate.db.query.RowsRange;
import io.stargate.db.query.RowsRange.Bound;
import io.stargate.db.query.TypedValue;
import io.stargate.db.query.TypedValue.Codec;
import io.stargate.db.schema.Column;
import io.stargate.db.schema.Column.ColumnType;
import io.stargate.db.schema.Column.Type;
import io.stargate.db.schema.Table;
import java.util.ArrayList;
import java.util.List;
import java.util.stream.Collectors;
import javax.annotation.Nullable;
import org.javatuples.Pair;

/**
 * Processes built WHERE clauses when bind values are provided.
 *
 * 

This currently only fully handles where clauses that results in a {@link RowsImpacted}, that * is WHERE clause that either selects a set of rows (all primary keys have conditions and they are * equals/IN) or select a partition slice (all partition keys have conditions and they are equals, * and clustering columns may or may not have conditions). Also, conditions on columns that are not * primary key column are ignored by this processor. Do note that this processor can be called on a * where clause that don't respect those conditions, but the {@link #process} method may return * {@code null}. */ abstract class WhereProcessor { private enum SelectionKind { KEYS, SINGLE_PARTITION_SLICE, PARTITION_RANGE } private final Table table; private final Codec valueCodec; private final List primaryKeys; private PKCondition[] pkConditions; private boolean shouldIgnorePkConditions; WhereProcessor(Table table, Codec valueCodec) { this.table = table; this.valueCodec = valueCodec; this.primaryKeys = table.primaryKeyColumns(); this.pkConditions = new PKCondition[primaryKeys.size()]; } protected abstract TypedValue handleValue(String name, ColumnType type, Value value); protected void onNonColumnNameLHS(BuiltCondition.LHS lhs) {} protected void onNonPrimaryKeyCondition(Column column) {} protected void onInequalityConditionOnPartitionKey(Column column, BuiltCondition condition) {} @Nullable RowsImpacted process(List whereClause) { preProcess(whereClause); if (shouldIgnorePkConditions) { return null; } switch (kind()) { case KEYS: return createKeys(); case SINGLE_PARTITION_SLICE: return createRange(); default: return null; } } private SelectionKind kind() { for (int i = 0; i < primaryKeys.size(); i++) { Column column = primaryKeys.get(i); PKCondition pkCondition = pkConditions[i]; if (pkCondition == null || !pkCondition.isEqOrIn()) { return column.isPartitionKey() ? SelectionKind.PARTITION_RANGE : SelectionKind.SINGLE_PARTITION_SLICE; } } return SelectionKind.KEYS; } private Ranges createRange() { Pair>, Integer> p = populateEqAndIn(primaryKeys, pkConditions); int firstNonEq = p.getValue1(); assert firstNonEq < primaryKeys.size(); assert firstNonEq >= table.partitionKeyColumns().size() : firstNonEq + ": " + table.partitionKeyColumns(); List> pkValues = p.getValue0(); PKCondition condition = pkConditions[firstNonEq]; int partitionKeys = table.partitionKeyColumns().size(); List ranges = new ArrayList<>(pkValues.size()); for (List pkPrefix : pkValues) { PartitionKey partitionKey = new PartitionKey(table, pkPrefix.subList(0, partitionKeys)); pkPrefix = pkPrefix.subList(partitionKeys, pkPrefix.size()); List startValues = pkPrefix; boolean startInclusive = true; List endValues = pkPrefix; boolean endInclusive = true; if (condition != null) { startValues = new ArrayList<>(pkPrefix); startValues.add(condition.values[0]); startInclusive = condition.isInclusive[0]; endValues = new ArrayList<>(pkPrefix); endValues.add(condition.values[1]); endInclusive = condition.isInclusive[1]; } RowsRange.Bound start = new Bound(startValues, startInclusive); RowsRange.Bound end = new Bound(endValues, endInclusive); ranges.add(new RowsRange(partitionKey, start, end)); } return new Ranges(ranges); } private RowsImpacted.Keys createKeys() { Pair>, Integer> p = populateEqAndIn(primaryKeys, pkConditions); // All keys must have been consumed, or we had a unexpected condition checkArgument( p.getValue1() == primaryKeys.size(), "Invalid condition combinations on primary key columns"); List> pkValues = p.getValue0(); List keys = new ArrayList<>(pkValues.size()); for (List pkValue : pkValues) { keys.add(new PrimaryKey(table, pkValue)); } return new RowsImpacted.Keys(keys); } private Pair>, Integer> populateEqAndIn( List primaryKeys, PKCondition[] pkConditions) { List> pkValues = new ArrayList<>(); pkValues.add(new ArrayList<>()); for (int i = 0; i < primaryKeys.size(); i++) { Column column = primaryKeys.get(i); PKCondition condition = pkConditions[i]; if (condition == null || !condition.isEqOrIn()) { return Pair.with(pkValues, i); } TypedValue value = condition.values[0]; if (condition.isEq()) { // Adds the new value to all keys. for (List pk : pkValues) { pk.add(value); } } else { // It's a IN assert value.javaValue() instanceof List; // Things would have failed before otherwise List inValues = (List) value.javaValue(); List inTypedValues = inValues.stream() .map(o -> TypedValue.forJavaValue(valueCodec, column.name(), column.type(), o)) .collect(Collectors.toList()); // For each existing primary keys, creates #inValues new keys corresponding to the // previous key plus the new value. // TODO: note that if we're not careful with the generated queries, we can have a // combinatorial explosion here. I could swear C* had limits for this, rejecting queries // that would create too many keys, but I can't find it right now, so maybe not. We // may want to add in any case, but it's unclear what a good number is. List> currentValues = pkValues; pkValues = new ArrayList<>(); for (List currentValue : currentValues) { for (TypedValue newValue : inTypedValues) { List newValues = new ArrayList<>(currentValue); newValues.add(newValue); pkValues.add(newValues); } } } } return Pair.with(pkValues, primaryKeys.size()); } void preProcess(List whereClause) { for (BuiltCondition condition : whereClause) { BuiltCondition.LHS lhs = condition.lhs(); Column column = table.existingColumn(lhs.columnName()); if (!column.isPrimaryKeyComponent()) { onNonPrimaryKeyCondition(column); continue; } if (!lhs.isColumnName()) { onNonColumnNameLHS(lhs); this.shouldIgnorePkConditions = true; return; } int idx = table.primaryKeyColumnIndex(column); PKCondition pkCondition = compute(column, pkConditions[idx], condition); if (pkCondition == PKCondition.INVALID) { this.shouldIgnorePkConditions = true; return; } pkConditions[idx] = pkCondition; } } private static boolean isEqOrIn(BuiltCondition condition) { return condition.predicate() == Predicate.EQ || condition.predicate() == Predicate.IN; } private static int rangeIdx(BuiltCondition condition) { switch (condition.predicate()) { case GT: case GTE: return 0; case LT: case LTE: return 1; default: throw new IllegalArgumentException( "Invalidate condition on primary key column: " + condition); } } private static boolean isInclusive(BuiltCondition condition) { switch (condition.predicate()) { case GTE: case LTE: return true; case GT: case LT: return false; default: // This should be called after rangeIdx, which already rejected those. throw new AssertionError(); } } private PKCondition compute(Column column, PKCondition existing, BuiltCondition condition) { TypedValue v; if (condition.predicate() == Predicate.IN) { v = handleValue( format("in(%s)", column.name()), Type.List.of(column.type()), condition.value()); } else { v = handleValue(column.name(), column.type(), condition.value()); } checkArgument( v.bytes() != null, "Cannot use a null value for primary key column %s in table %s", column.cqlName(), table.cqlQualifiedName()); if (v.isUnset()) { return existing; } if (isEqOrIn(condition)) { if (existing != null) { throw new IllegalArgumentException( format("Incompatible conditions %s and %s", existing.firstCondition, condition)); } PKCondition pkCondition = new PKCondition(condition); pkCondition.values[0] = v; return pkCondition; } if (column.isPartitionKey()) { onInequalityConditionOnPartitionKey(column, condition); return PKCondition.INVALID; } if (existing == null) { existing = new PKCondition(condition); } else { checkArgument( !existing.isEqOrIn(), "Incompatible conditions %s and %s", existing.firstCondition, condition); } int idx = rangeIdx(condition); checkArgument( existing.values[idx] == null, "Incompatible conditions %s and %s", existing.firstCondition, condition); existing.values[idx] = v; existing.isInclusive[idx] = isInclusive(condition); return existing; } private static class PKCondition { private static final PKCondition INVALID = new PKCondition(null); private final BuiltCondition firstCondition; // for Eq/IN, only 0 is set, for ranges, 0 = open, 1 = close private final TypedValue[] values = new TypedValue[2]; // Only set for ranges, 0 = open, 1 = close private final boolean[] isInclusive = new boolean[2]; private PKCondition(BuiltCondition firstCondition) { this.firstCondition = firstCondition; } private boolean isEq() { return firstCondition.predicate() == Predicate.EQ; } private boolean isEqOrIn() { return WhereProcessor.isEqOrIn(firstCondition); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy