liquibase.change.ChangeFactory Maven / Gradle / Ivy
package liquibase.change;
import liquibase.ChecksumVersion;
import liquibase.Scope;
import liquibase.database.Database;
import liquibase.exception.UnexpectedLiquibaseException;
import liquibase.plugin.AbstractPluginFactory;
import liquibase.plugin.Plugin;
import liquibase.servicelocator.ServiceLocator;
import java.util.*;
import java.util.concurrent.ConcurrentHashMap;
/**
* Factory class for constructing the correct liquibase.change.Change implementation based on a command name.
* For XML-based changelogs, the tag name is the command name.
* Change implementations are looked up via the {@link ServiceLocator}.
*
* @see liquibase.change.Change
*/
public class ChangeFactory extends AbstractPluginFactory{
private final Map cachedMetadata = new ConcurrentHashMap<>();
private boolean performSupportsDatabaseValidation = true;
private ChangeFactory() {
}
@Override
protected Class getPluginClass() {
return Change.class;
}
@Override
protected int getPriority(Change obj, Object... args) {
String commandName = (String) args[0];
ChangeMetaData changeMetaData = getChangeMetaData(obj);
if (commandName.equals(changeMetaData.getName())) {
return changeMetaData.getPriority();
} else {
return Plugin.PRIORITY_NOT_APPLICABLE;
}
}
public ChangeMetaData getChangeMetaData(String change) {
Change changeObj = create(change);
if (changeObj == null) {
return null;
}
return getChangeMetaData(changeObj);
}
public ChangeMetaData getChangeMetaData(Change change) {
String cacheKey = generateCacheKey(change);
cachedMetadata.computeIfAbsent(cacheKey, c -> change.createChangeMetaData());
return cachedMetadata.get(cacheKey);
}
private String generateCacheKey(Change change) {
String key;
try {
ChecksumVersion version = Scope.getCurrentScope().getChecksumVersion();
if (version == null) {
throw new NullPointerException();
}
key = change.getClass().getName() + version;
} catch (Exception ignored) {
key = change.getClass().getName();
}
return key;
}
/**
* Unregister all instances of a given Change name. Normally used for testing, but can be called manually if needed.
*/
public void unregister(String name) {
for (Change change : new ArrayList<>(findAllInstances())) {
if (getChangeMetaData(change).getName().equals(name)) {
this.removeInstance(change);
}
}
}
/**
* Returns all defined changes in the registry. Returned set is not modifiable.
*/
public SortedSet getDefinedChanges() {
SortedSet names = new TreeSet<>();
for (Change change : findAllInstances()) {
names.add(getChangeMetaData(change).getName());
}
return Collections.unmodifiableSortedSet(names);
}
/**
* Create a new Change implementation for the given change name. The class of the constructed object will be the Change implementation with the highest priority.
* Each call to create will return a new instance of the Change.
*/
public Change create(String name) {
Set plugins = getPlugins(name);
if (plugins.isEmpty()) {
return null;
} else if (plugins.size() > 1) {
Database database = Scope.getCurrentScope().getDatabase();
if (database != null && performSupportsDatabaseValidation) {
plugins.removeIf(a -> !a.supports(database));
if (plugins.isEmpty()) {
throw new UnexpectedLiquibaseException(String.format("No registered %s plugin found for %s database", name, database.getDisplayName()));
}
}
}
try {
return plugins.iterator().next().getClass().getConstructor().newInstance();
} catch (Exception e) {
throw new UnexpectedLiquibaseException(e);
}
}
public Map getParameters(Change change) {
Map returnMap = new HashMap<>();
ChangeMetaData changeMetaData = getChangeMetaData(change);
for (ChangeParameterMetaData param : changeMetaData.getParameters().values()) {
Object currentValue = param.getCurrentValue(change);
if (currentValue != null) {
returnMap.put(param.getParameterName(), currentValue);
}
}
return returnMap;
}
/**
* @deprecated Use {@link liquibase.Scope#getSingleton(Class)}
*/
@Deprecated
public static ChangeFactory getInstance() {
return Scope.getCurrentScope().getSingleton(ChangeFactory.class);
}
/**
* Should the change be checked to see if it supports
* the current database?
* Default is true
*
* @param performSupportsDatabaseValidation
*/
public void setPerformSupportsDatabaseValidation(boolean performSupportsDatabaseValidation) {
this.performSupportsDatabaseValidation = performSupportsDatabaseValidation;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy