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.30.0
Show newest version
/**
 *
 */
package liquibase.integration.spring;

import liquibase.Scope;
import liquibase.exception.LiquibaseException;
import liquibase.logging.Logger;
import lombok.Getter;
import lombok.Setter;
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>
 * 
* * @author ladislav.gazo * @see SpringLiquibase */ public class MultiTenantSpringLiquibase implements InitializingBean, ResourceLoaderAware { private final List dataSources = new ArrayList<>(); /** * Defines the location of data sources suitable for multi-tenant environment. */ @Getter @Setter private String jndiBase; /** * Defines a single data source and several schemas for a multi-tenant environment. */ @Getter @Setter private DataSource dataSource; @Getter @Setter private List schemas; private ResourceLoader resourceLoader; @Getter @Setter private String changeLog; @Getter @Setter private String contexts; @Getter @Setter private String labelFilter; @Getter @Setter private Map parameters; @Getter @Setter private String defaultSchema; @Getter @Setter private String liquibaseSchema; @Getter @Setter private String liquibaseTablespace; @Getter @Setter private String databaseChangeLogTable; @Getter @Setter private String databaseChangeLogLockTable; @Getter @Setter private boolean dropFirst; @Getter @Setter private boolean clearCheckSums; @Getter @Setter private boolean shouldRun = true; @Getter @Setter private File rollbackFile; @Override public void afterPropertiesSet() throws Exception { Logger log = Scope.getCurrentScope().getLog(getClass()); 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("Schema based multitenancy enabled"); if ((schemas == null) || schemas.isEmpty()) { log.warning("Schemas not defined, using defaultSchema only"); schemas = new ArrayList<>(); schemas.add(defaultSchema); } runOnAllSchemas(); } } else { log.info("DataSources based multitenancy enabled"); resolveDataSources(); runOnAllDataSources(); } } private void resolveDataSources() throws NamingException { Logger log = Scope.getCurrentScope().getLog(getClass()); 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.fine("Added a data source at " + jndiUrl); } else { log.info("Skipping a resource " + jndiUrl + " not compatible with DataSource."); } } } private void runOnAllDataSources() throws LiquibaseException { Logger log = Scope.getCurrentScope().getLog(getClass()); for (DataSource aDataSource : dataSources) { log.info("Initializing Liquibase for data source " + aDataSource); SpringLiquibase liquibase = getSpringLiquibase(aDataSource); liquibase.afterPropertiesSet(); log.info("Liquibase ran for data source " + aDataSource); } } private void runOnAllSchemas() throws LiquibaseException { Logger log = Scope.getCurrentScope().getLog(getClass()); for (String schema : schemas) { if ("default".equals(schema)) { schema = null; } log.info("Initializing Liquibase for schema " + schema); SpringLiquibase liquibase = getSpringLiquibase(dataSource); liquibase.setDefaultSchema(schema); liquibase.afterPropertiesSet(); log.info("Liquibase ran for schema " + schema); } } private SpringLiquibase getSpringLiquibase(DataSource dataSource) { SpringLiquibase liquibase = new SpringLiquibase(); liquibase.setChangeLog(changeLog); liquibase.setChangeLogParameters(parameters); liquibase.setContexts(contexts); liquibase.setLabelFilter(labelFilter); 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; } /** * @deprecated use {@link #getLabelFilter()} */ @Deprecated public String getLabels() { return getLabelFilter(); } /** * @deprecated use {@link #setLabelFilter(String)} */ @Deprecated public void setLabels(String labels) { setLabelFilter(labels); } @Override public void setResourceLoader(ResourceLoader resourceLoader) { this.resourceLoader = resourceLoader; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy