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

io.tarantool.driver.core.AbstractTarantoolClient Maven / Gradle / Ivy

Go to download

Tarantool Cartridge driver for Tarantool versions 1.10+ based on Netty framework

There is a newer version: 0.14.0
Show newest version
package io.tarantool.driver.core;

import java.util.Arrays;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.atomic.AtomicReference;
import java.util.function.Supplier;

import io.netty.bootstrap.Bootstrap;
import io.netty.channel.ChannelOption;
import io.netty.channel.nio.NioEventLoopGroup;
import io.netty.channel.socket.nio.NioSocketChannel;
import org.msgpack.value.Value;

import io.tarantool.driver.TarantoolVersion;
import io.tarantool.driver.api.CallResult;
import io.tarantool.driver.api.MultiValueCallResult;
import io.tarantool.driver.api.SingleValueCallResult;
import io.tarantool.driver.api.TarantoolClient;
import io.tarantool.driver.api.TarantoolClientConfig;
import io.tarantool.driver.api.TarantoolResult;
import io.tarantool.driver.api.connection.ConnectionSelectionStrategyFactory;
import io.tarantool.driver.api.connection.TarantoolConnection;
import io.tarantool.driver.api.connection.TarantoolConnectionListeners;
import io.tarantool.driver.api.metadata.TarantoolMetadataOperations;
import io.tarantool.driver.api.metadata.TarantoolMetadataProvider;
import io.tarantool.driver.api.metadata.TarantoolSpaceMetadata;
import io.tarantool.driver.api.space.TarantoolSpaceOperations;
import io.tarantool.driver.core.connection.TarantoolConnectionFactory;
import io.tarantool.driver.core.connection.TarantoolConnectionManager;
import io.tarantool.driver.core.metadata.SpacesMetadataProvider;
import io.tarantool.driver.core.metadata.TarantoolMetadata;
import io.tarantool.driver.exceptions.TarantoolClientException;
import io.tarantool.driver.exceptions.TarantoolSpaceNotFoundException;
import io.tarantool.driver.mappers.CallResultMapper;
import io.tarantool.driver.mappers.MessagePackMapper;
import io.tarantool.driver.mappers.MessagePackObjectMapper;
import io.tarantool.driver.mappers.MessagePackValueMapper;
import io.tarantool.driver.mappers.converters.ValueConverter;
import io.tarantool.driver.mappers.factories.ResultMapperFactoryFactory;
import io.tarantool.driver.mappers.factories.ResultMapperFactoryFactoryImpl;
import io.tarantool.driver.protocol.Packable;
import io.tarantool.driver.protocol.TarantoolProtocolException;
import io.tarantool.driver.protocol.requests.TarantoolCallRequest;
import io.tarantool.driver.protocol.requests.TarantoolEvalRequest;
import io.tarantool.driver.utils.Assert;

/**
 * Basic Tarantool client implementation. Subclasses must provide the connection manager.
 *
 * @param  target tuple type
 * @param  target tuple collection type
 * @author Alexey Kuzin
 */
public abstract class AbstractTarantoolClient>
    implements TarantoolClient {

    private final NioEventLoopGroup eventLoopGroup;
    private final TarantoolClientConfig config;
    private final Bootstrap bootstrap;
    private final TarantoolConnectionFactory connectionFactory;
    private final TarantoolConnectionListeners listeners;
    private final AtomicReference metadataHolder = new AtomicReference<>();
    private final ResultMapperFactoryFactoryImpl mapperFactoryFactory;

    private final SpacesMetadataProvider metadataProvider;
    private final ScheduledExecutorService timeoutScheduler;
    private TarantoolConnectionManager connectionManager;

    /**
     * Create a client.
     *
     * @param config the client configuration
     * @see TarantoolClientConfig
     */
    public AbstractTarantoolClient(TarantoolClientConfig config) {
        this(config, new TarantoolConnectionListeners());
    }

    /**
     * Create a client, specifying the connection established event listeners.
     *
     * @param config                   the client configuration
     * @param selectionStrategyFactory instantiates strategies which provide the algorithm of selecting connections
     *                                 from the connection pool for performing the next request
     * @param listeners                connection established event listeners
     * @see TarantoolClientConfig
     * @deprecated
     */
    protected AbstractTarantoolClient(
        TarantoolClientConfig config,
        ConnectionSelectionStrategyFactory selectionStrategyFactory,
        TarantoolConnectionListeners listeners) {
        this(config, listeners);
    }

    /**
     * Create a client, specifying the connection established event listeners.
     *
     * @param config    the client configuration
     * @param listeners connection established event listeners
     * @see TarantoolClientConfig
     */
    public AbstractTarantoolClient(TarantoolClientConfig config, TarantoolConnectionListeners listeners) {
        Assert.notNull(config, "Tarantool client config must not be null");
        Assert.notNull(listeners, "Tarantool connection listeners must not be null");

        this.config = config;
        this.mapperFactoryFactory = new ResultMapperFactoryFactoryImpl();
        this.eventLoopGroup = new NioEventLoopGroup(config.getEventLoopThreadsNumber());
        this.bootstrap = new Bootstrap()
            .group(eventLoopGroup)
            .channel(NioSocketChannel.class)
            .option(ChannelOption.SO_REUSEADDR, true)
            .option(ChannelOption.SO_KEEPALIVE, true)
            .option(ChannelOption.TCP_NODELAY, true)
            .option(ChannelOption.CONNECT_TIMEOUT_MILLIS, config.getConnectTimeout());
        this.timeoutScheduler =
            Executors.newSingleThreadScheduledExecutor(new TarantoolDaemonThreadFactory("tarantool-timeout"));
        this.connectionFactory = new TarantoolConnectionFactory(config, this.bootstrap, this.timeoutScheduler);
        this.listeners = listeners;
        this.metadataProvider = new SpacesMetadataProvider(this, config.getMessagePackMapper());
    }

    /**
     * Provides a connection manager for Tarantool server connections
     *
     * @param config            contains Tarantool client configuration options
     * @param connectionFactory provides helper methods for connection instantiation
     * @param listeners         listeners which will be invoked once all connections are established
     * @return connection manager
     */
    protected abstract TarantoolConnectionManager connectionManager(
        TarantoolClientConfig config,
        TarantoolConnectionFactory connectionFactory,
        TarantoolConnectionListeners listeners);

    private TarantoolConnectionManager connectionManager() {
        if (this.connectionManager == null) {
            synchronized (this) {
                if (this.connectionManager == null) {
                    this.connectionManager = connectionManager(config, connectionFactory, listeners);
                }
            }
        }
        return this.connectionManager;
    }

    @Override
    public boolean refresh() {
        return connectionManager().refresh();
    }

    @Override
    public TarantoolVersion getVersion() throws TarantoolClientException {
        try {
            return connectionManager().getConnection().thenApply(TarantoolConnection::getVersion).get();
        } catch (InterruptedException e) {
            throw new TarantoolClientException(e);
        } catch (ExecutionException e) {
            throw new TarantoolClientException(e.getCause());
        }
    }

    @Override
    public TarantoolSpaceOperations space(String spaceName) throws TarantoolClientException {
        Assert.hasText(spaceName, "Space name must not be null or empty");

        TarantoolMetadataOperations metadata = this.metadata();
        Optional meta = metadata.getSpaceByName(spaceName);
        if (!meta.isPresent()) {
            throw new TarantoolSpaceNotFoundException(spaceName);
        }

        return spaceOperations(config, connectionManager(), metadata, meta.get());
    }

    @Override
    public TarantoolSpaceOperations space(int spaceId) throws TarantoolClientException {
        Assert.state(spaceId > 0, "Space ID must be greater than 0");

        TarantoolMetadataOperations metadata = this.metadata();
        Optional meta = metadata.getSpaceById(spaceId);
        if (!meta.isPresent()) {
            throw new TarantoolSpaceNotFoundException(spaceId);
        }

        return spaceOperations(config, connectionManager(), metadata, meta.get());
    }

    /**
     * Creates a space API implementation instance for the specified space
     *
     * @param config            Tarantool client configuration
     * @param connectionManager configured internal connection manager
     * @param metadata          metadata operations
     * @param spaceMetadata     current space metadata
     * @return space API implementation instance
     */
    protected abstract TarantoolSpaceOperations spaceOperations(
        TarantoolClientConfig config,
        TarantoolConnectionManager connectionManager,
        TarantoolMetadataOperations metadata,
        TarantoolSpaceMetadata spaceMetadata);

    @Override
    public TarantoolMetadataOperations metadata() throws TarantoolClientException {
        if (metadataHolder.get() == null) {
            this.metadataHolder.compareAndSet(null, new TarantoolMetadata(metadataProvider()));
        }
        return metadataHolder.get();
    }

    @Override
    public TarantoolMetadataProvider metadataProvider() {
        return metadataProvider;
    }

    @Override
    public CompletableFuture> call(String functionName) throws TarantoolClientException {
        return call(functionName, Collections.emptyList());
    }

    @Override
    public CompletableFuture> call(String functionName, Object... arguments)
        throws TarantoolClientException {
        return call(functionName, Arrays.asList(arguments));
    }

    @Override
    public CompletableFuture> call(String functionName, List arguments)
        throws TarantoolClientException {
        return call(functionName, arguments, config.getMessagePackMapper());
    }

    @Override
    public CompletableFuture> call(String functionName, List arguments, MessagePackMapper mapper)
        throws TarantoolClientException {
        return makeRequest(functionName, arguments, mapper, mapper);
    }

    @Override
    public  CompletableFuture> callForTupleResult(String functionName, Class tupleClass)
        throws TarantoolClientException {
        return callForTupleResult(functionName, Collections.emptyList(), tupleClass);
    }

    @Override
    public  CompletableFuture call(
        String functionName,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return call(functionName, Collections.emptyList(), resultMapper);
    }

    @Override
    public  CompletableFuture> callForTupleResult(
            String functionName, List arguments, Class tupleClass)
        throws TarantoolClientException {
        return callForTupleResult(functionName, arguments, config.getMessagePackMapper(), tupleClass);
    }

    @Override
    public  CompletableFuture call(
        String functionName,
        List arguments,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return call(functionName, arguments, config.getMessagePackMapper(), resultMapper);
    }

    @Override
    public  CompletableFuture> callForTupleResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        Class tupleClass)
        throws TarantoolClientException {
        return call(functionName, arguments, argumentsMapper,
            mapperFactoryFactory.getTarantoolResultMapper(config.getMessagePackMapper(), tupleClass));
    }

    @Override
    public  CompletableFuture call(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, argumentsMapper, resultMapper);
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        Class resultClass)
        throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, config.getMessagePackMapper(), resultClass);
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        ValueConverter valueConverter)
        throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, config.getMessagePackMapper(), valueConverter);
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        CallResultMapper> resultMapper) throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, config.getMessagePackMapper(), resultMapper);
    }

    @Override
    public  CompletableFuture callForSingleResult(String functionName, Class resultClass)
        throws TarantoolClientException {
        return callForSingleResult(functionName, Collections.emptyList(), resultClass);
    }

    @Override
    public  CompletableFuture callForSingleResult(String functionName, ValueConverter valueConverter)
        throws TarantoolClientException {
        return callForSingleResult(functionName, Collections.emptyList(), valueConverter);
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        CallResultMapper> resultMapper) throws TarantoolClientException {
        return callForSingleResult(functionName, Collections.emptyList(), resultMapper);
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        Class resultClass)
        throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, argumentsMapper,
            mapperFactoryFactory.getDefaultSingleValueMapper(config.getMessagePackMapper(), resultClass));
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        ValueConverter valueConverter)
        throws TarantoolClientException {
        return callForSingleResult(functionName, arguments, argumentsMapper,
            mapperFactoryFactory.getSingleValueResultMapper(valueConverter));
    }

    @Override
    public  CompletableFuture callForSingleResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return makeRequestForSingleResult(functionName, arguments, argumentsMapper, resultMapper)
            .thenApply(CallResult::value);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        Supplier resultContainerSupplier,
        Class resultClass)
        throws TarantoolClientException {
        return callForMultiResult(
            functionName, arguments, config.getMessagePackMapper(), resultContainerSupplier, resultClass);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        Supplier resultContainerSupplier,
        ValueConverter valueConverter) throws TarantoolClientException {
        return callForMultiResult(functionName, arguments, config.getMessagePackMapper(),
            resultContainerSupplier, valueConverter);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        CallResultMapper> resultMapper) throws TarantoolClientException {
        return callForMultiResult(functionName, arguments, config.getMessagePackMapper(), resultMapper);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        Supplier resultContainerSupplier,
        Class resultClass)
        throws TarantoolClientException {
        return callForMultiResult(functionName, Collections.emptyList(), resultContainerSupplier, resultClass);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        Supplier resultContainerSupplier,
        ValueConverter valueConverter)
        throws TarantoolClientException {
        return callForMultiResult(functionName, Collections.emptyList(), resultContainerSupplier, valueConverter);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return callForMultiResult(functionName, Collections.emptyList(), resultMapper);
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        Supplier resultContainerSupplier,
        Class resultClass)
        throws TarantoolClientException {
        return callForMultiResult(functionName, arguments, argumentsMapper,
            mapperFactoryFactory.getDefaultMultiValueMapper(config.getMessagePackMapper(), resultClass));
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        Supplier resultContainerSupplier,
        ValueConverter valueConverter)
        throws TarantoolClientException {
        return callForMultiResult(functionName, arguments, argumentsMapper,
            mapperFactoryFactory.getMultiValueResultMapper(resultContainerSupplier, valueConverter));
    }

    @Override
    public > CompletableFuture callForMultiResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        CallResultMapper> resultMapper)
        throws TarantoolClientException {
        return makeRequestForMultiResult(functionName, arguments, argumentsMapper, resultMapper)
            .thenApply(CallResult::value);
    }

    private  CompletableFuture> makeRequestForSingleResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        CallResultMapper> resultMapper) {
        return makeRequest(functionName, arguments, argumentsMapper, resultMapper);
    }

    private > CompletableFuture> makeRequestForMultiResult(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        CallResultMapper> resultMapper) {
        return makeRequest(functionName, arguments, argumentsMapper, resultMapper);
    }

    private  CompletableFuture makeRequest(
        String functionName,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        try {
            TarantoolCallRequest.Builder builder = new TarantoolCallRequest.Builder()
                .withFunctionName(functionName);

            if (arguments.size() > 0) {
                builder.withArguments(arguments);
            }

            TarantoolCallRequest request = builder.build(argumentsMapper);
            return connectionManager().getConnection().thenCompose(c -> c.sendRequest(request, resultMapper));
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public CompletableFuture> eval(String expression) throws TarantoolClientException {
        return eval(expression, Collections.emptyList());
    }

    @Override
    public CompletableFuture> eval(String expression, List arguments)
        throws TarantoolClientException {
        return eval(expression, arguments, config.getMessagePackMapper());
    }

    @Override
    public CompletableFuture> eval(String expression, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        return eval(expression, Collections.emptyList(), resultMapper);
    }

    @Override
    public CompletableFuture> eval(String expression, List arguments, MessagePackValueMapper resultMapper)
        throws TarantoolClientException {
        return eval(expression, arguments, config.getMessagePackMapper(), resultMapper);
    }

    @Override
    public CompletableFuture> eval(
        String expression,
        List arguments,
        MessagePackObjectMapper argumentsMapper,
        MessagePackValueMapper resultMapper) throws TarantoolClientException {
        try {
            TarantoolEvalRequest request = new TarantoolEvalRequest.Builder()
                .withExpression(expression)
                .withArguments(arguments)
                .build(argumentsMapper);
            return connectionManager().getConnection().thenCompose(c -> c.sendRequest(request, resultMapper));
        } catch (TarantoolProtocolException e) {
            throw new TarantoolClientException(e);
        }
    }

    @Override
    public TarantoolClientConfig getConfig() {
        return config;
    }

    protected Bootstrap getBootstrap() {
        return bootstrap;
    }

    @Override
    public void close() throws Exception {
        try {
            connectionManager().close();
            timeoutScheduler.shutdownNow();
        } finally {
            eventLoopGroup.shutdownGracefully();
        }
    }

    @Override
    public TarantoolConnectionListeners getConnectionListeners() {
        return listeners;
    }

    @Override
    public ResultMapperFactoryFactory getResultMapperFactoryFactory() {
        return mapperFactoryFactory;
    }
}