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

liquibase.plugin.AbstractPluginFactory Maven / Gradle / Ivy

There is a newer version: 4.30.0
Show newest version
package liquibase.plugin;

import liquibase.Scope;
import liquibase.exception.UnexpectedLiquibaseException;
import liquibase.servicelocator.ServiceLocator;

import java.util.*;

/**
 * Convenience base class for all factories that find correct {@link Plugin} implementations.
 */
public abstract class AbstractPluginFactory implements PluginFactory {

    private Collection allInstances;

    protected AbstractPluginFactory() {

    }

    protected abstract Class getPluginClass();

    /**
     * Returns the priority of the given object based on the passed args array.
     * The args are created as part of the custom public getPlugin method in implementations are passed through {@link #getPlugin(Object...)}
     */
    protected abstract int getPriority(T obj, Object... args);

    /**
     * Finds the plugin for which {@link #getPriority(Plugin, Object...)}  returns the highest value for the given scope and args.
     * This method is called by a public implementation-specific methods.
     * Normally this does not need to be overridden, instead override {@link #getPriority(Plugin, Object...)} to compute the priority of each object for the scope and arguments passed to this method.
     * 

* However, if there is a {@link Scope} key of "liquibase.plugin.${plugin.interface.class.Name}", an instance of that class will always be run first. * * @return null if no plugins are found or have a priority greater than zero. */ protected T getPlugin(final Object... args) { Set applicable = this.getPlugins(args); if (applicable.isEmpty()) { return null; } return applicable.iterator().next(); } protected Set getPlugins(final Object... args) { Optional forcedPlugin = this.getForcedPluginIfAvailable(); if (forcedPlugin.isPresent()) { return new HashSet<>(Collections.singletonList(forcedPlugin.get())); } TreeSet applicable = new TreeSet<>((o1, o2) -> { Integer o1Priority = getPriority(o1, args); Integer o2Priority = getPriority(o2, args); int i = o2Priority.compareTo(o1Priority); if (i == 0) { return o1.getClass().getName().compareTo(o2.getClass().getName()); } return i; }); for (T plugin : findAllInstances()) { if (getPriority(plugin, args) >= 0) { applicable.add(plugin); } } return applicable; } private Optional getForcedPluginIfAvailable() { final String pluginClassName = getPluginClass().getName(); final Class forcedPlugin = Scope.getCurrentScope().get("liquibase.plugin." + pluginClassName, Class.class); if (forcedPlugin != null) { for (T plugin : findAllInstances()) { if (plugin.getClass().equals(forcedPlugin)) { return Optional.of(plugin); } } throw new UnexpectedLiquibaseException("No registered " + pluginClassName + " plugin " + forcedPlugin.getName()); } return Optional.empty(); } public synchronized void register(T plugin) { this.findAllInstances(); this.allInstances.add(plugin); } /** * Finds implementations of the given interface or class and returns instances of them. * Standard implementation uses {@link ServiceLoader} to find implementations and caches results in {@link #allInstances} which means the same objects are always returned. * If the instances should not be treated as singletons, clone the objects before returning them from {@link #getPlugin(Object...)}. */ protected synchronized Collection findAllInstances() { if (this.allInstances == null) { this.allInstances = new ArrayList<>(); ServiceLocator serviceLocator = Scope.getCurrentScope().getServiceLocator(); this.allInstances.addAll(serviceLocator.findInstances(getPluginClass())); } return this.allInstances; } protected synchronized void removeInstance(T instance) { if (this.allInstances == null) { return; } this.allInstances.remove(instance); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy