io.tarantool.driver.api.conditions.Conditions Maven / Gradle / Ivy
package io.tarantool.driver.api.conditions;
import io.tarantool.driver.api.metadata.TarantoolFieldMetadata;
import io.tarantool.driver.api.metadata.TarantoolIndexMetadata;
import io.tarantool.driver.api.metadata.TarantoolIndexPartMetadata;
import io.tarantool.driver.api.metadata.TarantoolMetadataOperations;
import io.tarantool.driver.api.metadata.TarantoolSpaceMetadata;
import io.tarantool.driver.api.tuple.TarantoolTuple;
import io.tarantool.driver.core.conditions.FieldValueConditionImpl;
import io.tarantool.driver.core.conditions.IdIndexImpl;
import io.tarantool.driver.core.conditions.IndexValueConditionImpl;
import io.tarantool.driver.core.conditions.NamedFieldImpl;
import io.tarantool.driver.core.conditions.NamedIndexImpl;
import io.tarantool.driver.core.conditions.PositionFieldImpl;
import io.tarantool.driver.exceptions.TarantoolClientException;
import io.tarantool.driver.mappers.MessagePackObjectMapper;
import io.tarantool.driver.mappers.converters.ObjectConverter;
import io.tarantool.driver.protocol.Packable;
import io.tarantool.driver.protocol.TarantoolIndexQuery;
import io.tarantool.driver.protocol.TarantoolIteratorType;
import io.tarantool.driver.utils.Assert;
import org.msgpack.value.ArrayValue;
import org.msgpack.value.Value;
import java.io.Serializable;
import java.util.AbstractMap;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Comparator;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.Collectors;
/**
* A collection and a builder for tuple filtering conditions.
* See
* https://github.com/tarantool/crud#select-conditions
*
* @author Alexey Kuzin
*/
public final class Conditions implements Serializable {
private static final long serialVersionUID = 20200708L;
private static final long MAX_LIMIT = 0xff_ff_ff_ffL;
private static final long MAX_OFFSET = 0xff_ff_ff_ffL;
private final List conditions = new LinkedList<>();
private boolean descending;
private long limit = MAX_LIMIT; // 0 is unlimited
private long offset; // 0 is no offset
private Packable startTuple;
private Conditions(boolean descending) {
this.descending = descending;
}
public Conditions(Conditions conditions) {
this.descending = conditions.descending;
this.limit = conditions.limit;
this.offset = conditions.offset;
this.startTuple = conditions.startTuple;
this.conditions.addAll(conditions.conditions);
}
private Conditions(Condition condition) {
conditions.add(condition);
}
private Conditions(long limit, long offset) {
this.limit = limit;
this.offset = offset;
}
private Conditions(Packable startTuple) {
this.startTuple = startTuple;
}
/**
* Get the descending option value
*
* @return false by default
*/
public boolean isDescending() {
return descending;
}
/**
* Create new Conditions instance, returning tuples in the descending order
*
* @return new {@link Conditions} instance
*/
public static Conditions descending() {
return new Conditions(true);
}
/**
* Return tuples in the descending order
*
* @return this {@link Conditions} instance
*/
public Conditions withDescending() {
this.descending = true;
return this;
}
/**
* Create new Conditions instance, returning tuples will in the ascending order
*
* @return new {@link Conditions} instance
*/
public static Conditions ascending() {
return new Conditions(false);
}
/**
* Return tuples will in the ascending order
*
* @return this {@link Conditions} instance
*/
public Conditions withAscending() {
this.descending = false;
return this;
}
/**
* Create new Conditions instance without any filtration.
*
* @return new {@link Conditions} instance
*/
public static Conditions any() {
return ascending();
}
/**
* Limit the number od returned tuples with the specified value
*
* @param limit number of tuples, should be greater than 0
* @return new {@link Conditions} instance
*/
public static Conditions limit(long limit) {
Assert.state(limit >= 0 && limit <= MAX_LIMIT, "Limit mast be a value between 0 and 0xffffffff");
return new Conditions(limit, 0);
}
/**
* Limit the number od returned tuples with the specified value
*
* @param limit number of tuples, should be greater than 0
* @return this {@link Conditions} instance
*/
public Conditions withLimit(long limit) {
Assert.state(limit >= 0 && limit <= MAX_LIMIT, "Limit mast be a value between 0 and 0xffffffff");
this.limit = limit;
return this;
}
/**
* Get the specified limit
*
* @return number of tuples
*/
public long getLimit() {
return limit;
}
/**
* Skip the specified number of tuples before collecting the result.
*
* @param offset number of tuples, should be greater than 0
* @return new {@link Conditions} instance
*/
public static Conditions offset(long offset) {
Assert.state(offset >= 0 && offset <= MAX_OFFSET, "Offset mast be a value between 0 and 0xffffffff");
return new Conditions(0, offset);
}
/**
* Skip the specified number of tuples before collecting the result.
*
* @param offset number of tuples, should be greater than 0
* @return this {@link Conditions} instance
*/
public Conditions withOffset(long offset) {
Assert.state(offset >= 0 && offset <= MAX_OFFSET, "Offset mast be a value between 0 and 0xffffffff");
this.offset = offset;
return this;
}
/**
* Get the specified offset
*
* @return number of tuples
*/
public long getOffset() {
return offset;
}
/**
* Start collecting tuples into result after the specified tuple. The tuple itself will not be added to the result.
*
* @param tuple last tuple value from the previous result, may be null
* @return new {@link Conditions} instance
*/
public static Conditions after(TarantoolTuple tuple) {
return new Conditions(tuple);
}
/**
* Start collecting tuples into result after the specified tuple. The tuple itself will not be added to the result.
*
* @param tuple last tuple value from the previous result, may be null
* @param tupleConverter converter of the specified tuple type into a MessagePack array
* @param tuple type
* @return new {@link Conditions} instance
*/
public static Conditions after(T tuple, ObjectConverter tupleConverter) {
Assert.notNull(tupleConverter, "Tuple to ArrayValue converter should not be null");
return new Conditions(new StartTupleWrapper<>(tuple, tupleConverter));
}
/**
* Start collecting tuples into result after the specified tuple. The tuple itself will not be added to the result.
*
* @param tuple last tuple value from the previous result, may be null
* @return new {@link Conditions} instance
*/
public Conditions startAfter(TarantoolTuple tuple) {
this.startTuple = tuple;
return this;
}
/**
* Start collecting tuples into result after the specified tuple. The tuple itself will not be added to the result.
*
* @param tuple last tuple value from the previous result, may be null
* @param tupleConverter converter of the specified tuple type into a MessagePack array
* @param tuple type
* @return new {@link Conditions} instance
*/
public Conditions startAfter(T tuple, ObjectConverter tupleConverter) {
Assert.notNull(tupleConverter, "Tuple to ArrayValue converter should not be null");
this.startTuple = new StartTupleWrapper<>(tuple, tupleConverter);
return this;
}
/**
* Get the specified index values to start from
*
* @return list of index parts values
*/
public Packable getStartTuple() {
return startTuple;
}
/**
* Create new Conditions instance with filter by the specified index
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexEquals(String indexName, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.EQ, new NamedIndexImpl(indexName), indexPartValues));
}
/**
* Filter tuples by the specified index
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public Conditions andIndexEquals(String indexName, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.EQ, new NamedIndexImpl(indexName), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index
*
* @param indexId index id
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexEquals(int indexId, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.EQ, new IdIndexImpl(indexId), indexPartValues));
}
/**
* Filter tuples by the specified index
*
* @param indexId index id
* @param indexPartValues index parts values
* @return this {@link Conditions} instance
*/
public Conditions andIndexEquals(int indexId, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.EQ, new IdIndexImpl(indexId), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values greater than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexGreaterThan(String indexName, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.GT, new NamedIndexImpl(indexName), indexPartValues));
}
/**
* Filter tuples by the specified index, with values greater than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public Conditions andIndexGreaterThan(String indexName, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.GT, new NamedIndexImpl(indexName), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values greater than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexGreaterThan(int indexId, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.GT, new IdIndexImpl(indexId), indexPartValues));
}
/**
* Filter tuples by the specified index, with values greater than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return this {@link Conditions} instance
*/
public Conditions andIndexGreaterThan(int indexId, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.GT, new IdIndexImpl(indexId), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values greater or equal than the
* specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexGreaterOrEquals(String indexName, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.GE, new NamedIndexImpl(indexName), indexPartValues));
}
/**
* Filter tuples by the specified index, with values greater or equal than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public Conditions andIndexGreaterOrEquals(String indexName, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.GE, new NamedIndexImpl(indexName), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values greater or equal than the
* specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexGreaterOrEquals(int indexId, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.GE, new IdIndexImpl(indexId), indexPartValues));
}
/**
* Filter tuples by the specified index, with values greater or equal than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return this {@link Conditions} instance
*/
public Conditions andIndexGreaterOrEquals(int indexId, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.GE, new IdIndexImpl(indexId), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values less than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexLessThan(String indexName, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.LT, new NamedIndexImpl(indexName), indexPartValues));
}
/**
* Filter tuples by the specified index, with values less than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public Conditions andIndexLessThan(String indexName, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.LT, new NamedIndexImpl(indexName), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values less than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexLessThan(int indexId, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.LT, new IdIndexImpl(indexId), indexPartValues));
}
/**
* Filter tuples by the specified index, with values less than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return this {@link Conditions} instance
*/
public Conditions andIndexLessThan(int indexId, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.LT, new IdIndexImpl(indexId), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values less or equal than the
* specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexLessOrEquals(String indexName, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.LE, new NamedIndexImpl(indexName), indexPartValues));
}
/**
* Filter tuples by the specified index, with values less or equal than the specified value
*
* @param indexName index name
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public Conditions andIndexLessOrEquals(String indexName, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.LE, new NamedIndexImpl(indexName), indexPartValues));
return this;
}
/**
* Create new Conditions instance with filter by the specified index, with values less or equal than the
* specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return new {@link Conditions} instance
*/
public static Conditions indexLessOrEquals(int indexId, List> indexPartValues) {
return new Conditions(new IndexValueConditionImpl(Operator.LE, new IdIndexImpl(indexId), indexPartValues));
}
/**
* Filter tuples by the specified index, with values less or equal than the specified value
*
* @param indexId index id
* @param indexPartValues index parts values
* @return this {@link Conditions} instance
*/
public Conditions andIndexLessOrEquals(int indexId, List> indexPartValues) {
conditions.add(new IndexValueConditionImpl(Operator.LE, new IdIndexImpl(indexId), indexPartValues));
return this;
}
/**
* Filter tuples by the specified field
*
* @param fieldName field name
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions equals(String fieldName, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.EQ, new NamedFieldImpl(fieldName), value));
}
/**
* Filter tuples by the specified field
*
* @param fieldName field name
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andEquals(String fieldName, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.EQ, new NamedFieldImpl(fieldName), value));
return this;
}
/**
* Filter tuples by the specified field
*
* @param fieldPosition field position
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions equals(int fieldPosition, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.EQ, new PositionFieldImpl(fieldPosition), value));
}
/**
* Filter tuples by the specified field
*
* @param fieldPosition field position
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andEquals(int fieldPosition, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.EQ, new PositionFieldImpl(fieldPosition), value));
return this;
}
/**
* Filter tuples by the specified field, with values greater than the specified value
*
* @param fieldName field name
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions greaterThan(String fieldName, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.GT, new NamedFieldImpl(fieldName), value));
}
/**
* Filter tuples by the specified field, with values greater than the specified value
*
* @param fieldName field name
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andGreaterThan(String fieldName, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.GT, new NamedFieldImpl(fieldName), value));
return this;
}
/**
* Filter tuples by the specified field, with values greater than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions greaterThan(int fieldPosition, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.GT, new PositionFieldImpl(fieldPosition), value));
}
/**
* Filter tuples by the specified field, with values greater than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andGreaterThan(int fieldPosition, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.GT, new PositionFieldImpl(fieldPosition), value));
return this;
}
/**
* Filter tuples by the specified field, with values greater or equal than the specified value
*
* @param fieldName field name
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions greaterOrEquals(String fieldName, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.GE, new NamedFieldImpl(fieldName), value));
}
/**
* Filter tuples by the specified field, with values greater or equal than the specified value
*
* @param fieldName field name
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andGreaterOrEquals(String fieldName, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.GE, new NamedFieldImpl(fieldName), value));
return this;
}
/**
* Filter tuples by the specified field, with values greater or equal than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions greaterOrEquals(int fieldPosition, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.GE, new PositionFieldImpl(fieldPosition), value));
}
/**
* Filter tuples by the specified field, with values greater or equal than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andGreaterOrEquals(int fieldPosition, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.GE, new PositionFieldImpl(fieldPosition), value));
return this;
}
/**
* Filter tuples by the specified field, with values less than the specified value
*
* @param fieldName field name
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions lessThan(String fieldName, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.LT, new NamedFieldImpl(fieldName), value));
}
/**
* Filter tuples by the specified field, with values less than the specified value
*
* @param fieldName field name
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andLessThan(String fieldName, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.LT, new NamedFieldImpl(fieldName), value));
return this;
}
/**
* Filter tuples by the specified field, with values less than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions lessThan(int fieldPosition, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.LT, new PositionFieldImpl(fieldPosition), value));
}
/**
* Filter tuples by the specified field, with values less than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andLessThan(int fieldPosition, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.LT, new PositionFieldImpl(fieldPosition), value));
return this;
}
/**
* Filter tuples by the specified field, with values less or equal than the specified value
*
* @param fieldName field name
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions lessOrEquals(String fieldName, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.LE, new NamedFieldImpl(fieldName), value));
}
/**
* Filter tuples by the specified field, with values less or equal than the specified value
*
* @param fieldName field name
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andLessOrEquals(String fieldName, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.LE, new NamedFieldImpl(fieldName), value));
return this;
}
/**
* Filter tuples by the specified field, with values less or equal than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return new {@link Conditions} instance
*/
public static Conditions lessOrEquals(int fieldPosition, Object value) {
return new Conditions(new FieldValueConditionImpl(Operator.LE, new PositionFieldImpl(fieldPosition), value));
}
/**
* Filter tuples by the specified field, with values less or equal than the specified value
*
* @param fieldPosition field position
* @param value field value
* @return this {@link Conditions} instance
*/
public Conditions andLessOrEquals(int fieldPosition, Object value) {
conditions.add(new FieldValueConditionImpl(Operator.LE, new PositionFieldImpl(fieldPosition), value));
return this;
}
public List> toProxyQuery(
TarantoolMetadataOperations operations,
TarantoolSpaceMetadata spaceMetadata) {
if (offset > 0) {
throw new TarantoolClientException("Offset is not supported");
}
final Map> indexConditions = new HashMap<>();
final Map> fieldConditions = new HashMap<>();
final Map selectedFields = new HashMap<>();
for (Condition condition : conditions) {
if (condition instanceof IndexValueConditionImpl) {
TarantoolIndexMetadata indexMetadata =
(TarantoolIndexMetadata) condition.field().metadata(operations, spaceMetadata);
List current = indexConditions.computeIfAbsent(
indexMetadata.getIndexName(), name -> new LinkedList<>());
current.add(convertIndexIfNecessary((IndexValueCondition) condition, indexMetadata.getIndexName()));
} else {
TarantoolFieldMetadata fieldMetadata =
(TarantoolFieldMetadata) condition.field().metadata(operations, spaceMetadata);
List current = fieldConditions
.computeIfAbsent(fieldMetadata.getFieldPosition(), f -> new LinkedList<>());
current.add((FieldValueCondition) condition);
selectedFields.putIfAbsent(fieldMetadata.getFieldPosition(), fieldMetadata);
}
}
if (indexConditions.size() > 1) {
throw new TarantoolClientException("Filtering by more than one index is not supported");
}
List> allConditions = new ArrayList<>();
if (indexConditions.size() > 0) {
allConditions.addAll(
conditionsListToLists(indexConditions.values().iterator().next(), operations, spaceMetadata));
for (List conditionList : fieldConditions.values()) {
allConditions.addAll(conditionsListToLists(conditionList, operations, spaceMetadata));
}
} else {
Optional suitableIndex = findCoveringIndex(
operations, spaceMetadata, selectedFields.values());
if (suitableIndex.isPresent()) {
for (TarantoolIndexPartMetadata part : suitableIndex.get().getIndexParts()) {
List conditions = fieldConditions.get(part.getFieldIndex());
if (conditions != null) {
allConditions.addAll(conditionsListToLists(conditions, operations, spaceMetadata));
fieldConditions.remove(part.getFieldIndex());
}
}
}
for (List conditionList : fieldConditions.values()) {
allConditions.addAll(
conditionsListToLists(conditionList, operations, spaceMetadata));
}
}
return allConditions;
}
private IndexValueCondition convertIndexIfNecessary(
IndexValueCondition condition,
String indexName) {
if (!(condition.field() instanceof IdIndexImpl)) {
return condition;
}
return new IndexValueConditionImpl(condition.operator(), new NamedIndexImpl(indexName), condition.value());
}
private List> conditionsListToLists(
List extends Condition> conditionsList,
TarantoolMetadataOperations operations,
TarantoolSpaceMetadata spaceMetadata) {
return conditionsList.stream().map(c -> c.toList(operations, spaceMetadata)).collect(Collectors.toList());
}
public TarantoolIndexQuery toIndexQuery(
TarantoolMetadataOperations operations,
TarantoolSpaceMetadata spaceMetadata) {
if (startTuple != null) {
throw new TarantoolClientException("'startAfter' is not supported");
}
final Map> indexConditions = new HashMap<>();
final Map selectedIndexes = new HashMap<>();
final Map> fieldConditions = new HashMap<>();
final Map selectedFields = new HashMap<>();
for (Condition condition : conditions) {
if (condition instanceof IndexValueConditionImpl) {
TarantoolIndexMetadata indexMetadata =
(TarantoolIndexMetadata) condition.field().metadata(operations, spaceMetadata);
List current = indexConditions.computeIfAbsent(
indexMetadata.getIndexName(), name -> new LinkedList<>());
if (current.size() > 0) {
throw new TarantoolClientException("Multiple conditions for one index are not supported");
}
current.add((IndexValueCondition) condition);
selectedIndexes.putIfAbsent(indexMetadata.getIndexName(), indexMetadata);
} else {
TarantoolFieldMetadata fieldMetadata =
(TarantoolFieldMetadata) condition.field().metadata(operations, spaceMetadata);
List current = fieldConditions
.computeIfAbsent(fieldMetadata.getFieldName(), f -> new LinkedList<>());
if (current.size() > 0) {
throw new TarantoolClientException("Multiple conditions for one field are not supported");
}
current.add((FieldValueCondition) condition);
selectedFields.putIfAbsent(fieldMetadata.getFieldName(), fieldMetadata);
}
}
if (indexConditions.size() > 1) {
throw new TarantoolClientException("Filtering by more than one index is not supported");
}
TarantoolIndexQuery query;
if (indexConditions.size() > 0) {
if (fieldConditions.size() > 0) {
throw new TarantoolClientException("Filtering simultaneously by index and fields is not supported");
}
query = indexQueryFromIndexValues(indexConditions, selectedIndexes);
} else {
if (selectedFields.size() > 0) {
TarantoolIndexMetadata suitableIndex = findSuitableIndex(
operations, spaceMetadata, selectedFields.values());
query = indexQueryFromFieldValues(suitableIndex, fieldConditions, selectedFields);
} else {
query = new TarantoolIndexQuery(TarantoolIndexQuery.PRIMARY)
.withIteratorType(descending ? TarantoolIteratorType.ITER_REQ : TarantoolIteratorType.ITER_EQ);
}
}
return query;
}
private TarantoolIndexQuery indexQueryFromIndexValues(
Map> indexConditions,
Map selectedIndexes) {
IndexValueCondition condition = indexConditions.values().iterator().next().get(0);
TarantoolIndexMetadata indexMetadata = selectedIndexes.values().iterator().next();
TarantoolIteratorType iteratorType = condition.operator().toIteratorType();
return new TarantoolIndexQuery(indexMetadata.getIndexId())
.withIteratorType(descending ? iteratorType.reverse() : iteratorType)
.withKeyValues(condition.value());
}
private TarantoolIndexQuery indexQueryFromFieldValues(
TarantoolIndexMetadata suitableIndex,
Map> fieldConditions,
Map selectedFields) {
Operator selectedOperator = null;
List
© 2015 - 2025 Weber Informatics LLC | Privacy Policy