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

com.arpnetworking.utility.Database Maven / Gradle / Ivy

/*
 * Copyright 2015 Groupon.com
 *
 * 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.arpnetworking.utility;

import com.arpnetworking.clusteraggregator.configuration.DatabaseConfiguration;
import com.arpnetworking.logback.annotations.Loggable;
import com.google.common.base.MoreObjects;
import com.zaxxer.hikari.HikariConfig;
import com.zaxxer.hikari.HikariDataSource;
import io.ebean.DatabaseFactory;
import io.ebean.config.DatabaseConfig;
import org.flywaydb.core.Flyway;
import org.flywaydb.core.api.configuration.ClassicConfiguration;

import java.util.Optional;
import java.util.Set;
import javax.persistence.Entity;
import javax.sql.DataSource;

/**
 * Database instance abstraction across database technologies: HikariCP, Flyway and EBean.
 *
 * NOTE: The current configuration process of mapping to classes and then instantiating components from "safe"
 * configuration is a complete falacy. Only by attempting to apply (e.g. restart) the system with the new configuration
 * can you determine if it is truly safe. Once rewritten that way configuration classes can be eliminated in many cases
 * and the raw Configuration object mapped directly to the consumer of that configuration. For example, in this class we
 * could read a subkey for Hikari thus decoupling our configuration from Hikari's. The same can be done for Flyway by
 * automatically converting a subkey into a Properties instance and giving it to Flyway.
 *
 * @author Ville Koskela (ville dot koskela at inscopemetrics dot com)
 */
@Loggable
public class Database implements Launchable {

    /**
     * Public constructor.
     *
     * @param name The name of the database instance.
     * @param configuration Configuration for the database instance.
     */
    public Database(final String name, final DatabaseConfiguration configuration) {
        _name = name;
        _configuration = configuration;
    }

    @Override
    public void launch() {
        _dataSource = createDataSource();
        _flyway = createFlyway();
        _ebeanServer = createEbeanServer();

        if (_flyway.isPresent()) {
            _flyway.get().migrate();
        }
    }

    @Override
    public void shutdown() {
        _ebeanServer.shutdown(false, false);
        _dataSource.close();
    }

    public String getName() {
        return _name;
    }

    public DataSource getDataSource() {
        return _dataSource;
    }

    public io.ebean.Database getEbeanServer() {
        return _ebeanServer;
    }

    public Optional getFlyway() {
        return _flyway;
    }

    @Override
    public String toString() {
        return MoreObjects.toStringHelper(this)
                .add("id", Integer.toHexString(System.identityHashCode(this)))
                .add("Name", _name)
                .add("DataSource", _dataSource)
                .add("EbeanServer", _ebeanServer)
                .add("Flyway", _flyway)
                .toString();
    }

    private HikariDataSource createDataSource() {
        final HikariConfig hikariConfiguration = new HikariConfig();
        hikariConfiguration.setPoolName(_name);
        hikariConfiguration.setJdbcUrl(_configuration.getJdbcUrl());
        hikariConfiguration.setDriverClassName(_configuration.getDriverName());
        hikariConfiguration.setUsername(_configuration.getUsername());
        hikariConfiguration.setPassword(_configuration.getPassword());
        hikariConfiguration.setMaximumPoolSize(_configuration.getMaximumPoolSize());
        hikariConfiguration.setMinimumIdle(_configuration.getMinimumIdle());
        hikariConfiguration.setIdleTimeout(_configuration.getIdleTimeout());
        hikariConfiguration.setConnectionTestQuery("SELECT 1");
        return new HikariDataSource(hikariConfiguration);
    }

    private Optional createFlyway() {
        Flyway flyway = null;
        if (!_configuration.getMigrationLocations().isEmpty()) {
            final ClassicConfiguration flywayConfiguration = new ClassicConfiguration();
            flywayConfiguration.setLocationsAsStrings(_configuration.getMigrationLocations().toArray(new String[0]));
            flywayConfiguration.setSchemas(_configuration.getMigrationSchemas().toArray(new String[0]));
            flywayConfiguration.setDataSource(_dataSource);
            flyway = new Flyway(flywayConfiguration);
        }
        return Optional.ofNullable(flyway);
    }

    private io.ebean.Database createEbeanServer() {
        final DatabaseConfig ebeanConfiguration = new DatabaseConfig();
        ebeanConfiguration.setDefaultServer("default".equalsIgnoreCase(_name));
        ebeanConfiguration.setDdlGenerate(false);
        ebeanConfiguration.setDdlRun(false);
        ebeanConfiguration.setName(_name);
        ebeanConfiguration.setDataSource(_dataSource);
//        ebeanConfiguration.setAllQuotedIdentifiers(true);
        final Set> entityClasses = ReflectionsDatabase.newInstance().findClassesWithAnnotation(Entity.class);
        for (final String modelPackage : _configuration.getModelPackages()) {
            final String safePackage = modelPackage.endsWith(".*") ? modelPackage.substring(0, modelPackage.length() - 2) : modelPackage;
            ebeanConfiguration.addPackage(modelPackage);
            entityClasses.stream().filter(c -> c.getPackage().getName().startsWith(safePackage)).forEach(ebeanConfiguration::addClass);
        }

        return DatabaseFactory.create(ebeanConfiguration);
    }

    private final String _name;
    private final DatabaseConfiguration _configuration;
    private HikariDataSource _dataSource;
    private io.ebean.Database _ebeanServer;
    private Optional _flyway;
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy