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

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 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 fieldValues = Arrays.asList(new Object[suitableIndex.getIndexParts().size()]);
        for (Map.Entry> conditions : fieldConditions.entrySet()) {
            FieldValueCondition condition = conditions.getValue().iterator().next();
            if (selectedOperator == null) {
                selectedOperator = condition.operator();
            } else {
                if (!condition.operator().equals(selectedOperator)) {
                    throw new TarantoolClientException(
                        "Different conditions for index parts are not supported");
                }
            }
            TarantoolFieldMetadata field = selectedFields.get(conditions.getKey());
            int partPosition = suitableIndex
                .getIndexPartPositionByFieldPosition(field.getFieldPosition())
                .orElseThrow(() -> new TarantoolClientException(
                    "Field %s not found in index %s", field.getFieldName(), suitableIndex.getIndexName()));
            fieldValues.set(partPosition, condition.value());
        }
        TarantoolIteratorType iteratorType = selectedOperator != null ?
            selectedOperator.toIteratorType() :
            TarantoolIteratorType.ITER_EQ;
        return new TarantoolIndexQuery(suitableIndex.getIndexId())
            .withIteratorType(descending ? iteratorType.reverse() : iteratorType)
            .withKeyValues(fieldValues);
    }

    private static Optional findCoveringIndex(
        TarantoolMetadataOperations operations,
        TarantoolSpaceMetadata spaceMetadata,
        Collection selectedFields) {
        Map allIndexes = operations.getSpaceIndexes(spaceMetadata.getSpaceName())
            .orElseThrow(() -> new TarantoolClientException(
                "Metadata for space %s not found", spaceMetadata.getSpaceName()));

        Optional coveringIndex = allIndexes.values().stream()
            .map(metadata -> new AbstractMap.SimpleEntry(
                calculateCoverage(metadata, selectedFields), metadata))
            .filter(entry -> entry.getKey() > 0)
            .max(Comparator.comparingLong(AbstractMap.SimpleEntry::getKey))
            .map(AbstractMap.SimpleEntry::getValue);

        return coveringIndex;
    }

    private static long calculateCoverage(
        TarantoolIndexMetadata metadata,
        Collection selectedFields) {
        AtomicBoolean firstFieldIsSet = new AtomicBoolean(false);
        long count = selectedFields.stream()
            .map(f -> metadata.getIndexPartPositionByFieldPosition(f.getFieldPosition()))
            .filter(Optional::isPresent)
            .map(Optional::get)
            .peek(indexPosition -> {
                if (indexPosition == 0) {
                    firstFieldIsSet.set(true);
                }
            })
            .count();
        return firstFieldIsSet.get() ? count : 0;
    }

    private static TarantoolIndexMetadata findSuitableIndex(
        TarantoolMetadataOperations operations,
        TarantoolSpaceMetadata spaceMetadata,
        Collection selectedFields) {
        Map allIndexes = operations.getSpaceIndexes(spaceMetadata.getSpaceName())
            .orElseThrow(() -> new TarantoolClientException(
                "Metadata for space %s not found", spaceMetadata.getSpaceName()));

        TarantoolIndexMetadata suitableIndex = allIndexes.values().stream()
            .filter(metadata -> isSuitableIndex(metadata, selectedFields))
            .min(Comparator.comparingInt(m -> m.getIndexParts().size()))
            .orElseThrow(() -> new TarantoolClientException("No indexes that fit the passed fields are found"));

        return suitableIndex;
    }

    private static boolean isSuitableIndex(
        TarantoolIndexMetadata indexMetadata,
        Collection selectedFields) {
        Map indexParts = indexMetadata.getIndexPartsByPosition();

        if (indexParts.size() < selectedFields.size()) {
            return false;
        }

        for (TarantoolFieldMetadata fieldMetadata : selectedFields) {
            if (!indexParts.containsKey(fieldMetadata.getFieldPosition())) {
                return false;
            }
        }

        return true;
    }

    private static class StartTupleWrapper implements Packable {

        private final T tuple;
        private final ObjectConverter tupleConverter;

        StartTupleWrapper(T tuple, ObjectConverter tupleConverter) {
            this.tuple = tuple;
            this.tupleConverter = tupleConverter;
        }

        @Override
        public Value toMessagePackValue(MessagePackObjectMapper mapper) {
            return tupleConverter.toValue(tuple);
        }
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) {
            return true;
        }
        if (o == null || getClass() != o.getClass()) {
            return false;
        }
        Conditions that = (Conditions) o;
        return isDescending() == that.isDescending() &&
            getLimit() == that.getLimit() &&
            getOffset() == that.getOffset()
            && conditions.equals(that.conditions) &&
            Objects.equals(getStartTuple(), that.getStartTuple());
    }

    @Override
    public int hashCode() {
        return Objects.hash(conditions, isDescending(), getLimit(), getOffset(), getStartTuple());
    }
}