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

com.mongodb.embedded.client.EmbeddedServer Maven / Gradle / Ivy

There is a newer version: 3.8.2
Show newest version
/*
 * Copyright 2008-present MongoDB, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *   http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.mongodb.embedded.client;

import com.mongodb.MongoDriverInformation;
import com.mongodb.MongoException;
import com.mongodb.ServerAddress;
import com.mongodb.async.SingleResultCallback;
import com.mongodb.connection.AsyncConnection;
import com.mongodb.connection.ClusterConnectionMode;
import com.mongodb.connection.Connection;
import com.mongodb.connection.Server;
import com.mongodb.connection.ServerDescription;
import com.mongodb.connection.ServerVersion;
import com.mongodb.diagnostics.logging.Logger;
import com.mongodb.diagnostics.logging.Loggers;
import com.mongodb.event.CommandListener;
import com.mongodb.internal.connection.ClusterClock;
import com.mongodb.internal.connection.ClusterClockAdvancingSessionContext;
import com.mongodb.internal.connection.CommandHelper;
import com.mongodb.internal.connection.CommandProtocol;
import com.mongodb.internal.connection.DefaultServerConnection;
import com.mongodb.internal.connection.DescriptionHelper;
import com.mongodb.internal.connection.InternalConnection;
import com.mongodb.internal.connection.LegacyProtocol;
import com.mongodb.internal.connection.ProtocolExecutor;
import com.mongodb.session.SessionContext;
import com.sun.jna.Pointer;
import org.bson.BsonDocument;
import org.bson.BsonInt32;

import java.io.Closeable;
import java.io.File;

import static com.mongodb.assertions.Assertions.isTrue;
import static com.mongodb.internal.async.ErrorHandlingResultCallback.errorHandlingCallback;
import static com.mongodb.internal.connection.ClientMetadataHelper.createClientMetadataDocument;
import static com.mongodb.internal.event.EventListenerHelper.getCommandListener;
import static java.lang.String.format;

class EmbeddedServer implements Server, Closeable {
    private static final Logger LOGGER = Loggers.getLogger("embedded.client");
    private static final MongoDriverInformation MONGO_DRIVER_INFORMATION = MongoDriverInformation.builder().driverName("embedded").build();
    private final ClusterClock clusterClock;
    private final CommandListener commandListener;
    private final ServerAddress serverAddress;
    private final ServerDescription serverDescription;
    private final EmbeddedInternalConnectionPool connectionPool;
    private volatile boolean isClosed;
    private volatile Pointer instanceStatusPointer;
    private volatile Pointer instancePointer;


    EmbeddedServer(final MongoClientSettings mongoClientSettings) {
        this.instanceStatusPointer = MongoDBCAPIHelper.createStatusPointer();
        this.instancePointer = createInstancePointer(mongoClientSettings);
        this.clusterClock = new ClusterClock();
        this.commandListener =  getCommandListener(mongoClientSettings.getCommandListeners());
        this.serverAddress = new ServerAddress();
        this.connectionPool = new EmbeddedInternalConnectionPool(new EmbeddedInternalConnectionFactory() {
            @Override
            public EmbeddedInternalConnection create() {
                return new EmbeddedInternalConnection(instancePointer, commandListener,
                        createClientMetadataDocument(mongoClientSettings.getApplicationName(), MONGO_DRIVER_INFORMATION));
            }
        });

        this.serverDescription = createServerDescription();
    }

    @Override
    public ServerDescription getDescription() {
        isTrue("open", !isClosed);
        return serverDescription;
    }

    @Override
    public Connection getConnection() {
        isTrue("open", !isClosed);
        return new DefaultServerConnection(connectionPool.get(), new DefaultServerProtocolExecutor(), ClusterConnectionMode.SINGLE);
    }

    @Override
    public void getConnectionAsync(final SingleResultCallback callback) {
        throw new UnsupportedOperationException("Async not supported");
    }

    @Override
    public void close() {
        if (!isClosed) {
            isClosed = true;
            connectionPool.close();
            destroyInstancePointer();
        }
    }

    private Pointer createInstancePointer(final MongoClientSettings mongoClientSettings) {
        File directory = new File(mongoClientSettings.getDbPath());
        try {
            if (directory.mkdirs() && LOGGER.isInfoEnabled()) {
                LOGGER.info(format("Created dbpath directory: %s", mongoClientSettings.getDbPath()));
            }
        } catch (SecurityException e) {
            throw new MongoException(format("Could not validate / create the dbpath: %s", mongoClientSettings.getDbPath()));
        }

        String yamlConfig = createYamlConfig(mongoClientSettings);
        return MongoDBCAPIHelper.instance_create(yamlConfig, instanceStatusPointer);
    }

    private void destroyInstancePointer() {
        MongoDBCAPIHelper.instance_destroy(instancePointer, instanceStatusPointer);
        instancePointer = null;
        MongoDBCAPIHelper.destroyStatusPointer(instanceStatusPointer);
        instanceStatusPointer = null;
    }

    private String createYamlConfig(final MongoClientSettings mongoClientSettings) {
        return format("{ storage: { dbPath: %s } }", mongoClientSettings.getDbPath());
    }

    private class DefaultServerProtocolExecutor implements ProtocolExecutor {

        @Override
        public  T execute(final LegacyProtocol protocol, final InternalConnection connection) {
            protocol.setCommandListener(commandListener);
            return protocol.execute(connection);
        }

        @Override
        public  void executeAsync(final LegacyProtocol protocol, final InternalConnection connection,
                                     final SingleResultCallback callback) {
            protocol.setCommandListener(commandListener);
            protocol.executeAsync(connection, callback);
        }

        @Override
        public  T execute(final CommandProtocol protocol, final InternalConnection connection,
                             final SessionContext sessionContext) {
            protocol.sessionContext(new ClusterClockAdvancingSessionContext(sessionContext, clusterClock));
            return protocol.execute(connection);
        }

        @Override
        public  void executeAsync(final CommandProtocol protocol, final InternalConnection connection,
                                     final SessionContext sessionContext, final SingleResultCallback callback) {
            protocol.sessionContext(new ClusterClockAdvancingSessionContext(sessionContext, clusterClock));
            protocol.executeAsync(connection, errorHandlingCallback(callback, LOGGER));
        }
    }

    private ServerDescription createServerDescription() {
        InternalConnection connection = connectionPool.get();
        try {
            long start = System.nanoTime();
            BsonDocument isMasterResult = CommandHelper.executeCommand("admin", new BsonDocument("ismaster", new BsonInt32(1)),
                    clusterClock, connection);
            return DescriptionHelper.createServerDescription(serverAddress, isMasterResult, new ServerVersion(), System.nanoTime() - start);
        } finally {
            connection.close();
        }
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy