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

ru.yandex.qatools.embed.service.MongoEmbeddedService Maven / Gradle / Ivy

package ru.yandex.qatools.embed.service;

import de.flapdoodle.embed.mongo.*;
import de.flapdoodle.embed.mongo.config.*;
import de.flapdoodle.embed.mongo.distribution.Version;
import de.flapdoodle.embed.process.config.IRuntimeConfig;
import de.flapdoodle.embed.process.config.io.ProcessOutput;
import de.flapdoodle.embed.process.io.IStreamProcessor;
import de.flapdoodle.embed.process.io.NamedOutputStreamProcessor;
import de.flapdoodle.embed.process.runtime.Network;
import org.apache.commons.lang3.StringUtils;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.nio.file.Paths;
import java.util.Collections;
import java.util.HashSet;

import static de.flapdoodle.embed.mongo.distribution.Version.Main.*;
import static de.flapdoodle.embed.process.io.Processors.console;
import static de.flapdoodle.embed.process.io.Processors.namedConsole;
import static java.lang.Integer.parseInt;
import static java.lang.Runtime.getRuntime;
import static java.lang.String.format;
import static java.util.Arrays.asList;
import static jodd.io.FileUtil.delete;
import static org.apache.commons.lang3.StringUtils.isEmpty;
import static org.apache.commons.lang3.StringUtils.join;

/**
 * Embedded MongoDB service
 *
 * @author smecsia
 */
public class MongoEmbeddedService extends AbstractEmbeddedService {
    private static final String HOST_PORT_SPLIT_PATTERN = "(?emptySet(),
                namedConsole("[mongod output]"));
        runtimeConfig = new RuntimeConfigBuilder()
                .defaults(Command.MongoD)
                .processOutput(new ProcessOutput(
                        mongodOutput,
                        namedConsole("[mongod error]"),
                        console()))
                .build();
        runtime = MongodStarter.getInstance(runtimeConfig);

        try {
            final MongoEmbeddedService self = this;
            getRuntime().addShutdownHook(new Thread() {
                public void run() {
                    self.stop();
                }
            });

            startWithAuth();
            if (newDirectory) {
                addAdmin();
            }
            if (useAuth) {
                addUser();
            }
        } catch (Exception e) {
            logger.error("Failed to startup embedded MongoDB", e);
        }
    }

    private boolean isMongo3() {
        return asList(V3_0, V3_1, PRODUCTION, DEVELOPMENT).contains(useVersion);
    }

    private void startWithAuth() throws IOException {
        prepareExecutable(useAuth);
        startMongoProcess();
    }

    private void startMongoProcess() throws IOException {
        mongod = executable.start();
        if (newDirectory && replSetName != null) {
            try {
                initiateReplicaSet();
            } catch (Exception e) {
                logger.error("Failed to initialize replica set", e);
            }
        }
    }

    private void prepareExecutable(boolean authEnabled) throws IOException {
        final MongoCmdOptionsBuilder cmdBuilder = new MongoCmdOptionsBuilder();
        cmdBuilder.enableAuth(authEnabled);
        if (useWiredTiger && isMongo3()) {
            cmdBuilder.useStorageEngine(WIRED_TIGER);
        }
        final IMongoCmdOptions cmdOptions = cmdBuilder.build();
        MongodConfigBuilder builder = new MongodConfigBuilder()
                .version(useVersion)
                .cmdOptions(cmdOptions)
                .net(new Net(host, port, Network.localhostIsIPv6()));
        if (authEnabled && isMongo3()) {
            builder.setParameter("authenticationMechanisms", authMechanisms);
        }
        if (replSetName != null) {
            removeLockFile(builder);
            builder.replication(new Storage(dataDirectory, replSetName, oplogSizeMb));
        }
        mongodConfig = builder.build();
        executable = null;
        executable = runtime.prepare(mongodConfig);
    }

    private void removeLockFile(MongodConfigBuilder builder) {
        final File lockFile = Paths.get(dataDirectory, "mongod.lock").toFile();
        try {
            delete(lockFile);
        } catch (Exception e) {
            logger.warn("No lock file found for embedded mongodb or removal failed: " + e.getMessage());
        }
    }

    @Override
    public void doStop() {
        if (executable != null) {
            executable.stop();
        }
    }

    public void initiateReplicaSet() throws IOException, InterruptedException {
        final String scriptText = join(asList(
                format("rs.initiate({\"_id\":\"%s\",\"members\":[{\"_id\":1,\"host\":\"%s:%s\"}]});",
                        replSetName, host, port),
                "rs.slaveOk();rs.status();"), "");
        runScriptAndWait(scriptText, null, null, null, null, null);
        mongodOutput.waitForResult(INIT_TIMEOUT_MS);
    }

    private void addAdmin() throws IOException {
        final String scriptText = join(
                format("db.createUser(" +
                                "{\"user\":\"%s\",\"pwd\":\"%s\"," +
                                "\"roles\":[" +
                                "\"root\"," +
                                "{\"role\":\"userAdmin\",\"db\":\"admin\"}," +
                                "{\"role\":\"dbAdmin\",\"db\":\"admin\"}," +
                                "{\"role\":\"userAdminAnyDatabase\",\"db\":\"admin\"}," +
                                "{\"role\":\"dbAdminAnyDatabase\",\"db\":\"admin\"}," +
                                "{\"role\":\"clusterAdmin\",\"db\":\"admin\"}," +
                                "{\"role\":\"dbOwner\",\"db\":\"admin\"}," +
                                "]});\n",
                        adminUsername, adminPassword));
        runScriptAndWait(scriptText, USER_ADDED_TOKEN, new String[]{"couldn't add user", "failed to load", "login failed"}, "admin", null, null);
    }

    private void addUser() throws IOException {
        final String scriptText = join(format("db = db.getSiblingDB('%s'); " +
                        "db.createUser({\"user\":\"%s\",\"pwd\":\"%s\",\"roles\":[%s]});\n" +
                        "db.getUser('%s');",
                mongoDBName, username, password, StringUtils.join(roles, ","), username), "");
        runScriptAndWait(scriptText, USER_ADDED_TOKEN, new String[]{"already exists", "failed to load", "login failed"}, "admin", adminUsername, adminPassword);
    }

    private void runScriptAndWait(String scriptText, String token, String[] failures, String dbName, String username, String password) throws IOException {
        IStreamProcessor mongoOutput;
        if (!isEmpty(token)) {
            mongoOutput = new LogWatchStreamProcessor(
                    format(token),
                    (failures != null) ? new HashSet<>(asList(failures)) : Collections.emptySet(),
                    namedConsole("[mongo shell output]"));
        } else {
            mongoOutput = new NamedOutputStreamProcessor("[mongo shell output]", console());
        }
        IRuntimeConfig runtimeConfig = new RuntimeConfigBuilder()
                .defaults(Command.Mongo)
                .processOutput(new ProcessOutput(
                        mongoOutput,
                        namedConsole("[mongo shell error]"),
                        console()))
                .build();
        MongoShellStarter starter = MongoShellStarter.getInstance(runtimeConfig);
        final File scriptFile = writeTmpScriptFile(scriptText);
        final MongoShellConfigBuilder builder = new MongoShellConfigBuilder();
        if (!isEmpty(dbName)) {
            builder.dbName(dbName);
        }
        if (!isEmpty(username)) {
            builder.username(username);
        }
        if (!isEmpty(password)) {
            builder.password(password);
        }
        starter.prepare(builder
                .scriptName(scriptFile.getAbsolutePath())
                .version(mongodConfig.version())
                .net(mongodConfig.net())
                .build()).start();
        if (mongoOutput instanceof LogWatchStreamProcessor) {
            ((LogWatchStreamProcessor) mongoOutput).waitForResult(INIT_TIMEOUT_MS);
        }
    }

    private File writeTmpScriptFile(String scriptText) throws IOException {
        File scriptFile = File.createTempFile("tempfile", ".js");
        scriptFile.deleteOnExit();
        BufferedWriter bw = new BufferedWriter(new FileWriter(scriptFile));
        bw.write(scriptText);
        bw.close();
        return scriptFile;
    }

    public Net net() {
        return mongodConfig.net();
    }


    public String getHost() {
        return host;
    }

    public int getPort() {
        return port;
    }

    public String[] getRoles() {
        return roles;
    }

    public void setRoles(String... roles) {
        this.roles = roles;
    }

    public String getAdminUsername() {
        return adminUsername;
    }

    public void setAdminUsername(String adminUsername) {
        this.adminUsername = adminUsername;
    }

    public String getAdminPassword() {
        return adminPassword;
    }

    public void setAdminPassword(String adminPassword) {
        this.adminPassword = adminPassword;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy