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

liquibase.integration.spring.MultiTenantSpringLiquibase Maven / Gradle / Ivy

There is a newer version: 4.31.0
Show newest version
/**
 * 
 */
package liquibase.integration.spring;

import liquibase.exception.LiquibaseException;
import liquibase.logging.LogService;
import liquibase.logging.LogType;
import liquibase.logging.Logger;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ResourceLoaderAware;
import org.springframework.core.io.ResourceLoader;

import javax.naming.*;
import javax.sql.DataSource;
import java.io.File;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;

/**
 * A wrapper of Liquibase suitable in multi-tenant environments where multiple
 * data sources represent tenants. It utilizes {@link SpringLiquibase} per each
 * data source. All the parameters are the same as for {@link SpringLiquibase}
 * except of the data source definition - in this case it is a list of data
 * sources available under specified JNDI subtree. You have to define the
 * subtree with {@link #jndiBase} property.
*
* The wrapper scans the subtree for all data sources and creates * {@link SpringLiquibase} instances.
*
* Example:
*
 * <bean id="liquibase" class="liquibase.integration.spring.MultiTenantSpringLiquibase">
 *	<property name="jndiBase" value="java:comp/env/jdbc/db" />
 *	<property name="changeLog" value="classpath:db/migration/db-changelog.xml" />	
 * </bean>
 * 
* * @see SpringLiquibase * * @author ladislav.gazo */ public class MultiTenantSpringLiquibase implements InitializingBean, ResourceLoaderAware { private final List dataSources = new ArrayList<>(); private Logger log = LogService.getLog(MultiTenantSpringLiquibase.class); /** Defines the location of data sources suitable for multi-tenant environment. */ private String jndiBase; /** Defines a single data source and several schemas for a multi-tenant environment. */ private DataSource dataSource; private List schemas; private ResourceLoader resourceLoader; private String changeLog; private String contexts; private String labels; private Map parameters; private String defaultSchema; private String liquibaseSchema; private String liquibaseTablespace; private String databaseChangeLogTable; private String databaseChangeLogLockTable; private boolean dropFirst; private boolean clearCheckSums; private boolean shouldRun = true; private File rollbackFile; @Override public void afterPropertiesSet() throws Exception { if((dataSource != null) || (schemas != null)) { if((dataSource == null) && (schemas != null)) { throw new LiquibaseException("When schemas are defined you should also define a base dataSource"); }else if(dataSource!=null){ log.info(LogType.LOG, "Schema based multitenancy enabled"); if((schemas == null) || schemas.isEmpty()) { log.warning(LogType.LOG, "Schemas not defined, using defaultSchema only"); schemas = new ArrayList<>(); schemas.add(defaultSchema); } runOnAllSchemas(); } }else { log.info(LogType.LOG, "DataSources based multitenancy enabled"); resolveDataSources(); runOnAllDataSources(); } } private void resolveDataSources() throws NamingException { Context context = new InitialContext(); int lastIndexOf = jndiBase.lastIndexOf("/"); String jndiRoot = jndiBase.substring(0, lastIndexOf); String jndiParent = jndiBase.substring(lastIndexOf + 1); Context base = (Context) context.lookup(jndiRoot); NamingEnumeration list = base.list(jndiParent); while(list.hasMoreElements()) { NameClassPair entry = list.nextElement(); String name = entry.getName(); String jndiUrl; if(entry.isRelative()) { jndiUrl = jndiBase + "/" + name; } else { jndiUrl = name; } Object lookup = context.lookup(jndiUrl); if(lookup instanceof DataSource) { dataSources.add((DataSource) lookup); log.debug(LogType.LOG, "Added a data source at " + jndiUrl); } else { log.info(LogType.LOG, "Skipping a resource " + jndiUrl + " not compatible with DataSource."); } } } private void runOnAllDataSources() throws LiquibaseException { for(DataSource aDataSource : dataSources) { log.info(LogType.LOG, "Initializing Liquibase for data source " + aDataSource); SpringLiquibase liquibase = getSpringLiquibase(aDataSource); liquibase.afterPropertiesSet(); log.info(LogType.LOG, "Liquibase ran for data source " + aDataSource); } } private void runOnAllSchemas() throws LiquibaseException { for(String schema : schemas) { if("default".equals(schema)) { schema = null; } log.info(LogType.LOG, "Initializing Liquibase for schema " + schema); SpringLiquibase liquibase = getSpringLiquibase(dataSource); liquibase.setDefaultSchema(schema); liquibase.afterPropertiesSet(); log.info(LogType.LOG, "Liquibase ran for schema " + schema); } } private SpringLiquibase getSpringLiquibase(DataSource dataSource) { SpringLiquibase liquibase = new SpringLiquibase(); liquibase.setChangeLog(changeLog); liquibase.setChangeLogParameters(parameters); liquibase.setContexts(contexts); liquibase.setLabels(labels); liquibase.setDropFirst(dropFirst); liquibase.setClearCheckSums(clearCheckSums); liquibase.setShouldRun(shouldRun); liquibase.setRollbackFile(rollbackFile); liquibase.setResourceLoader(resourceLoader); liquibase.setDataSource(dataSource); liquibase.setDefaultSchema(defaultSchema); liquibase.setLiquibaseSchema(liquibaseSchema); liquibase.setLiquibaseTablespace(liquibaseTablespace); liquibase.setDatabaseChangeLogTable(databaseChangeLogTable); liquibase.setDatabaseChangeLogLockTable(databaseChangeLogLockTable); return liquibase; } public String getJndiBase() { return jndiBase; } public void setJndiBase(String jndiBase) { this.jndiBase = jndiBase; } public String getChangeLog() { return changeLog; } public void setChangeLog(String changeLog) { this.changeLog = changeLog; } public String getContexts() { return contexts; } public void setContexts(String contexts) { this.contexts = contexts; } public String getLabels() { return labels; } public void setLabels(String labels) { this.labels = labels; } public Map getParameters() { return parameters; } public void setParameters(Map parameters) { this.parameters = parameters; } public String getDefaultSchema() { return defaultSchema; } public void setDefaultSchema(String defaultSchema) { this.defaultSchema = defaultSchema; } public String getLiquibaseSchema() { return liquibaseSchema; } public void setLiquibaseSchema(String liquibaseSchema) { this.liquibaseSchema = liquibaseSchema; } public String getLiquibaseTablespace() { return liquibaseTablespace; } public void setLiquibaseTablespace(String liquibaseTablespace) { this.liquibaseTablespace = liquibaseTablespace; } public String getDatabaseChangeLogTable() { return databaseChangeLogTable; } public void setDatabaseChangeLogTable(String databaseChangeLogTable) { this.databaseChangeLogTable = databaseChangeLogTable; } public String getDatabaseChangeLogLockTable() { return databaseChangeLogLockTable; } public void setDatabaseChangeLogLockTable(String databaseChangeLogLockTable) { this.databaseChangeLogLockTable = databaseChangeLogLockTable; } public boolean isDropFirst() { return dropFirst; } public void setDropFirst(boolean dropFirst) { this.dropFirst = dropFirst; } public boolean isClearCheckSums() { return clearCheckSums; } public void setClearCheckSums(boolean clearCheckSums) { this.clearCheckSums = clearCheckSums; } public boolean isShouldRun() { return shouldRun; } public void setShouldRun(boolean shouldRun) { this.shouldRun = shouldRun; } public File getRollbackFile() { return rollbackFile; } public void setRollbackFile(File rollbackFile) { this.rollbackFile = rollbackFile; } @Override public void setResourceLoader(ResourceLoader resourceLoader) { this.resourceLoader = resourceLoader; } public List getSchemas() { return schemas; } public void setSchemas(List schemas) { this.schemas = schemas; } public DataSource getDataSource() { return dataSource; } public void setDataSource(DataSource dataSource) { this.dataSource = dataSource; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy