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

org.duracloud.snapshot.db.DatabaseInitializer Maven / Gradle / Ivy

/*
 * The contents of this file are subject to the license and copyright
 * detailed in the LICENSE and NOTICE files at the root of the source
 * tree and available online at
 *
 *     http://duracloud.org/license/
 */
package org.duracloud.snapshot.db;

import java.util.List;

import org.apache.commons.dbcp2.BasicDataSource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.core.io.Resource;
import org.springframework.jdbc.datasource.init.DataSourceInitializer;
import org.springframework.jdbc.datasource.init.DatabasePopulator;
import org.springframework.jdbc.datasource.init.ResourceDatabasePopulator;

/**
 * This class is responsible for initializing the database.  This class has been
 * added as a replacement for jdbc:initialize tags in the spring config in order to 
 * enable us to lazily initialize the database.
 * @author Daniel Bernstein
 *         Date: Feb 12, 2014
 */
public class DatabaseInitializer implements ApplicationContextAware{

    private static final Logger LOGGER =
        LoggerFactory.getLogger(DatabaseInitializer.class);

    private List dropSchemas;

    private List createSchemas;
    
    private List storedProcedureDefinitions;
    
    private BasicDataSource dataSource;
    
    private ApplicationContext context;

    
    public DatabaseInitializer(BasicDataSource dataSource, List dropSchemas, List createSchemas, List storedProcedureDefinitions){
        this.dataSource = dataSource;
        this.dropSchemas = dropSchemas;
        this.createSchemas = createSchemas;
        this.storedProcedureDefinitions = storedProcedureDefinitions;
    }

    public void init(DatabaseConfig databaseConfig) {
        final DataSourceInitializer initializer = new DataSourceInitializer();
        final DataSourceInitializer storedProcInitializer = new DataSourceInitializer();

        dataSource.setUrl(databaseConfig.getUrl());
        dataSource.setUsername(databaseConfig.getUsername());
        dataSource.setPassword(databaseConfig.getPassword());
        initializer.setDataSource(dataSource);
        initializer.setDatabasePopulator(databasePopulator(databaseConfig));
        storedProcInitializer.setDataSource(dataSource);
        storedProcInitializer.setDatabasePopulator(databaseStoreProcedurePopulator(databaseConfig));

        try {
            initializer.afterPropertiesSet();
            storedProcInitializer.afterPropertiesSet();

        } catch (Exception e) {
            Throwable rootCause = getRootCause(e);

            // The database initialization SQL scripts create the necessary
            // tables.  If the exception indicates that the database already
            // contains tables then ignore the exception and continue on,
            // otherwise throw the exception.
            if(rootCause.getMessage().contains("already exists")) {
                LOGGER.info("Database initialization - tables already exist: {}",
                            rootCause.getMessage());
            } else {
                throw e;
            }
        }
    }
    
    /* (non-Javadoc)
     * @see org.springframework.context.ApplicationContextAware#setApplicationContext(org.springframework.context.ApplicationContext)
     */
    @Override
    public void setApplicationContext(ApplicationContext applicationContext)
        throws BeansException {
        this.context = applicationContext;
    }

    private DatabasePopulator databasePopulator(DatabaseConfig databaseConfig) {
        final ResourceDatabasePopulator populator = new ResourceDatabasePopulator();
        if(databaseConfig.isClean()){
            for(Resource schema : dropSchemas){
                populator.addScript(schema);
            }
        }

        for(Resource schema : createSchemas){
            populator.addScript(schema);
        }

        return populator;
    }
    
    private DatabasePopulator databaseStoreProcedurePopulator(DatabaseConfig databaseConfig) {
        final ResourceDatabasePopulator populator = new ResourceDatabasePopulator();
        

        for(Resource schema : storedProcedureDefinitions){
            populator.addScript(schema);
        }

        //stored procedure scripts (in snapshot-common-db) are delimited with the # sign
        //to get around this problem: http://stackoverflow.com/questions/15486516/using-springs-jdbcinitialize-database-how-do-i-run-a-script-with-a-stored-p
        populator.setSeparator("#");
        return populator;
    }

    private Throwable getRootCause(Throwable throwable) {
        if(throwable.getCause() != null) {
            return getRootCause(throwable.getCause());
        }
        return throwable;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy