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

io.tarantool.driver.core.space.TarantoolSpace Maven / Gradle / Ivy

package io.tarantool.driver.core.space;

import io.tarantool.driver.api.TarantoolClientConfig;
import io.tarantool.driver.api.TarantoolVoidResult;
import io.tarantool.driver.api.conditions.Conditions;
import io.tarantool.driver.api.metadata.TarantoolIndexMetadata;
import io.tarantool.driver.api.metadata.TarantoolMetadataOperations;
import io.tarantool.driver.api.metadata.TarantoolSpaceMetadata;
import io.tarantool.driver.api.space.TarantoolSpaceOperations;
import io.tarantool.driver.api.tuple.operations.TupleOperation;
import io.tarantool.driver.api.tuple.operations.TupleOperations;
import io.tarantool.driver.core.connection.TarantoolConnectionManager;
import io.tarantool.driver.exceptions.TarantoolClientException;
import io.tarantool.driver.exceptions.TarantoolSpaceFieldNotFoundException;
import io.tarantool.driver.exceptions.TarantoolSpaceOperationException;
import io.tarantool.driver.mappers.MessagePackValueMapper;
import io.tarantool.driver.protocol.Packable;
import io.tarantool.driver.protocol.TarantoolIndexQuery;
import io.tarantool.driver.protocol.TarantoolProtocolException;
import io.tarantool.driver.protocol.TarantoolRequest;
import io.tarantool.driver.protocol.requests.TarantoolCallRequest;
import io.tarantool.driver.protocol.requests.TarantoolDeleteRequest;
import io.tarantool.driver.protocol.requests.TarantoolInsertRequest;
import io.tarantool.driver.protocol.requests.TarantoolReplaceRequest;
import io.tarantool.driver.protocol.requests.TarantoolSelectRequest;
import io.tarantool.driver.protocol.requests.TarantoolUpdateRequest;
import io.tarantool.driver.protocol.requests.TarantoolUpsertRequest;
import org.msgpack.value.ArrayValue;

import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.stream.Collectors;

/**
 * Basic implementation for working with spaces via Tarantool protocol requests
 *
 * @author Alexey Kuzin
 */
public abstract class TarantoolSpace>
    implements TarantoolSpaceOperations {

    private final int spaceId;
    private final TarantoolClientConfig config;
    private final TarantoolConnectionManager connectionManager;
    private final TarantoolSpaceMetadata spaceMetadata;
    private final TarantoolMetadataOperations metadataOperations;

    public TarantoolSpace(
        TarantoolClientConfig config,
        TarantoolConnectionManager connectionManager,
        TarantoolMetadataOperations metadataOperations,
        TarantoolSpaceMetadata spaceMetadata) {
        this.spaceId = spaceMetadata.getSpaceId();
        this.config = config;
        this.connectionManager = connectionManager;
        this.spaceMetadata = spaceMetadata;
        this.metadataOperations = metadataOperations;
    }

    @Override
    public CompletableFuture delete(Conditions conditions) throws TarantoolClientException {
        return delete(conditions, arrayTupleResultMapper());
    }

    private CompletableFuture delete(Conditions conditions, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolIndexQuery indexQuery = conditions.toIndexQuery(metadataOperations, spaceMetadata);

            TarantoolDeleteRequest request = new TarantoolDeleteRequest.Builder()
                .withSpaceId(spaceId)
                .withIndexId(indexQuery.getIndexId())
                .withKeyValues(indexQuery.getKeyValues())
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture insert(T tuple) throws TarantoolClientException {
        return insert(tuple, arrayTupleResultMapper());
    }

    @Override
    public CompletableFuture insertMany(Collection tuples) throws TarantoolClientException {
        // TODO: add support with interactive transactions if the node has the MVCC mode enabled
        // TODO: add support in all other cases if https://github.com/tarantool/tarantool/issues/7691 is implemented
        throw new UnsupportedOperationException(
            "Standalone node API does not support inserting several tuples at once yet");
    }

    private CompletableFuture insert(T tuple, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolInsertRequest request = new TarantoolInsertRequest.Builder()
                .withSpaceId(spaceId)
                .withTuple(tuple)
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture replace(T tuple) throws TarantoolClientException {
        return replace(tuple, arrayTupleResultMapper());
    }

    @Override
    public CompletableFuture replaceMany(Collection tuples) throws TarantoolClientException {
        // TODO: add support with interactive transactions if the node has the MVCC mode enabled
        // TODO: add support in all other cases if https://github.com/tarantool/tarantool/issues/7691 is implemented
        throw new UnsupportedOperationException(
            "Standalone node API does not support replacing several tuples at once yet");
    }

    private CompletableFuture replace(T tuple, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolReplaceRequest request = new TarantoolReplaceRequest.Builder()
                .withSpaceId(spaceId)
                .withTuple(tuple)
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture select(Conditions conditions) throws TarantoolClientException {
        return select(conditions, arrayTupleResultMapper());
    }

    private CompletableFuture select(Conditions conditions, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolIndexQuery indexQuery = conditions.toIndexQuery(metadataOperations, spaceMetadata);
            TarantoolSelectRequest request = new TarantoolSelectRequest.Builder()
                .withSpaceId(spaceId)
                .withIndexId(indexQuery.getIndexId())
                .withIteratorType(indexQuery.getIteratorType())
                .withKeyValues(indexQuery.getKeyValues())
                .withLimit(conditions.getLimit())
                .withOffset(conditions.getOffset())
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture update(Conditions conditions, T tuple) {
        return update(conditions, makeOperationsFromTuple(tuple), arrayTupleResultMapper());
    }

    /**
     * Create a {@link TupleOperations} instance from the given tuple of type {@code T}
     *
     * @param tuple tuple of the specified type
     * @return new {@link TupleOperations} instance
     */
    protected abstract TupleOperations makeOperationsFromTuple(T tuple);

    @Override
    public CompletableFuture update(Conditions conditions, TupleOperations operations) {
        return update(conditions, operations, arrayTupleResultMapper());
    }

    private CompletableFuture update(
        Conditions conditions,
        TupleOperations operations,
        MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolIndexQuery indexQuery = conditions.toIndexQuery(metadataOperations, spaceMetadata);

            Optional indexMetadata =
                metadataOperations.getIndexById(spaceId, indexQuery.getIndexId());

            if (!indexMetadata.isPresent() || !indexMetadata.get().isUnique()) {
                throw new TarantoolSpaceOperationException("Index must be primary or unique for update operation");
            }

            TarantoolUpdateRequest request = new TarantoolUpdateRequest.Builder()
                .withSpaceId(spaceId)
                .withIndexId(indexQuery.getIndexId())
                .withKeyValues(indexQuery.getKeyValues())
                .withTupleOperations(fillFieldIndexFromMetadata(operations))
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture upsert(Conditions conditions, T tuple, TupleOperations operations) {
        return upsert(conditions, tuple, operations, arrayTupleResultMapper());
    }

    private CompletableFuture upsert(
        Conditions conditions,
        T tuple,
        TupleOperations operations,
        MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolIndexQuery indexQuery = conditions.toIndexQuery(metadataOperations, spaceMetadata);

            TarantoolUpsertRequest request = new TarantoolUpsertRequest.Builder()
                .withSpaceId(spaceId)
                .withKeyValues(indexQuery.getKeyValues())
                .withTuple(tuple)
                .withTupleOperations(fillFieldIndexFromMetadata(operations))
                .build(config.getMessagePackMapper());

            return sendRequest(request, resultMapper);
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture truncate() throws TarantoolClientException {
        return truncate(arrayTupleResultMapper());
    }

    private CompletableFuture truncate(MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            String spaceName = spaceMetadata.getSpaceName();
            TarantoolCallRequest request = new TarantoolCallRequest.Builder()
                .withFunctionName("box.space." + spaceName + ":truncate")
                .build(config.getMessagePackMapper());
            return sendRequest(request, resultMapper)
                .thenApply(v -> TarantoolVoidResult.INSTANCE.value());
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    /**
     * MessagePack value mapper configured with an ArrayValue to tuple converter corresponding to the selected
     * tuple type
     *
     * @return configured mapper with {@link ArrayValue} to {@code T} converter
     */
    protected abstract MessagePackValueMapper arrayTupleResultMapper();

    private CompletableFuture sendRequest(TarantoolRequest request, MessagePackValueMapper resultMapper) {
        return connectionManager.getConnection().thenCompose(c -> c.sendRequest(request, resultMapper));
    }

    @Override
    public TarantoolSpaceMetadata getMetadata() {
        return spaceMetadata;
    }

    @Override
    public String toString() {
        return String.format(
            "StandaloneTarantoolSpace %s [%d]", spaceMetadata.getSpaceName(), spaceMetadata.getSpaceId());
    }

    private List fillFieldIndexFromMetadata(TupleOperations operations) {
        return operations.asList().stream()
            .map(operation -> {
                if (operation.getFieldIndex() == null) {
                    String fieldName = operation.getFieldName();
                    int fieldMetadataIndex = this.spaceMetadata.getFieldByName(fieldName)
                        .orElseThrow(() -> new TarantoolSpaceFieldNotFoundException(fieldName))
                        .getFieldPosition();
                    return operation.cloneWithIndex(fieldMetadataIndex);
                }
                return operation;
            }).collect(Collectors.toList());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy