/* * GeoTools - The Open Source Java GIS Toolkit * http://geotools.org * * (C) 2005-2008, Open Source Geospatial Foundation (OSGeo) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. */ package org.geotools.factory; import java.util.*; import java.util.logging.Level; import java.util.logging.LogRecord; import java.util.logging.Logger; import java.lang.ref.Reference; import java.awt.RenderingHints; import javax.imageio.spi.ServiceRegistry; import org.geotools.util.Utilities; import org.geotools.util.logging.Logging; import org.geotools.resources.Classes; import org.geotools.resources.i18n.Errors; import org.geotools.resources.i18n.ErrorKeys; import org.geotools.resources.i18n.Loggings; import org.geotools.resources.i18n.LoggingKeys; /** * A registry for factories, organized by categories (usualy by <strong>interface</strong>). For * example <code>{@linkplain org.opengis.referencing.crs.CRSFactory}.class</code> is a category, * and <code>{@linkplain org.opengis.referencing.operation.MathTransformFactory}.class</code> is * an other category. * <p> * For each category, implementations are registered in a file placed in the * {@code META-INF/services/} directory, as specified in the {@link ServiceRegistry} * javadoc. Those files are usually bundled into the JAR file distributed by the vendor. * If the same {@code META-INF/services/} file appears many time in different JARs, * they are processed as if their content were merged. * <p> * Example use: * <blockquote><code> * Set<Class<?>> categories = * Collections.singleton(new Class<?>[] {<br> * &npsp;&npsp;&npsp;&npsp;MathTransformProvider.class<br> * });<br> * FactoryRegistry registry = new FactoryRegistry(categories);<br> * <br> * // get the providers<br> * Filter filter = null;<br> * Hints hints = null;<br> * Iterator<MathTransform> providers = * registry.getServiceProviders(MathTransformProvider.class, filter, hints);<br> * </code></blockquote> * <p> * <strong>NOTE: This class is not thread safe</strong>. Users are responsable * for synchronisation. This is usually done in an utility class wrapping this * service registry (e.g. {@link org.geotools.referencing.ReferencingFactoryFinder}). * * @since 2.1 * * @source $URL$ * @version $Id$ * @author Martin Desruisseaux * @author Richard Gould * @author Jody Garnett * * @see org.geotools.referencing.ReferencingFactoryFinder * @see org.geotools.coverage.CoverageFactoryFinder */ public class FactoryRegistry extends ServiceRegistry { /** * The logger for all events related to factory registry. */ protected static final Logger LOGGER = Logging.getLogger("org.geotools.factory"); /** * The logger level for debug messages. */ private static final Level DEBUG_LEVEL = Level.FINEST; /** * A copy of the global configuration defined through {@link FactoryIteratorProviders} * static methods. We keep a copy in every {@code FactoryRegistry} instance in order to * compare against the master {@link FactoryIteratorProviders#GLOBAL} and detect if the * configuration changed since the last time this registry was used. * * @see #synchronizeIteratorProviders */ private final FactoryIteratorProviders globalConfiguration = new FactoryIteratorProviders(); /** * The set of category that need to be scanned for plugins, or {@code null} if none. * On initialization, all categories need to be scanned for plugins. After a category * has been first used, it is removed from this set so we don't scan for plugins again. */ private Set<Class<?>> needScanForPlugins; /** * Categories under scanning. This is used by {@link #scanForPlugins(Collection,Class)} * as a guard against infinite recursivity (i.e. when a factory to be scanned request * an other dependency of the same category). */ private final RecursionCheckingHelper scanningCategories = new RecursionCheckingHelper(); /** * Factories under testing for availability. This is used by * {@link #isAvailable} as a guard against infinite recursivity. */ private final RecursionCheckingHelper testingAvailability = new RecursionCheckingHelper(); /** * Factories under testing for hints compatibility. This is used by * {@link #usesAcceptableHints} as a guard against infinite recursivity. */ private final RecursionCheckingHelper testingHints = new RecursionCheckingHelper(); /** * Constructs a new registry for the specified category. * * @param category The single category. * * @since 2.4 */ @SuppressWarnings("unchecked") public FactoryRegistry(final Class<?> category) { this((Collection) Collections.singleton(category)); } /** * Constructs a new registry for the specified categories. * * @param categories The categories. * * @since 2.4 */ public FactoryRegistry(final Class<?>[] categories) { this(Arrays.asList(categories)); } /** * Constructs a new registry for the specified categories. * * @param categories The categories. */ public FactoryRegistry(final Collection<Class<?>> categories) { super(categories.iterator()); for (final Iterator<Class<?>> it=getCategories(); it.hasNext();) { if (needScanForPlugins == null) { needScanForPlugins = new HashSet<Class<?>>(); } needScanForPlugins.add(it.next()); } } /** * Returns the providers in the registry for the specified category, filter and hints. * Providers that are not {@linkplain OptionalFactory#isAvailable available} will be * ignored. This method will {@linkplain #scanForPlugins() scan for plugins} the first * time it is invoked for the given category. * * @param <T> The class represented by the {@code category} argument. * @param category The category to look for. Usually an interface class * (not the actual implementation class). * @param filter The optional filter, or {@code null}. * @param hints The optional user requirements, or {@code null}. * @return Factories ready to use for the specified category, filter and hints. * * @since 2.3 */ public synchronized <T> Iterator<T> getServiceProviders(final Class<T> category, final Filter filter, final Hints hints) { /* * The implementation of this method is very similar to the 'getUnfilteredProviders' * one except for filter handling. See the comments in 'getUnfilteredProviders' for * more implementation details. */ if (scanningCategories.contains(category)) { // Please note you will get errors here if you accidentally allow // more than one thread to use your FactoryRegistry at a time. throw new RecursiveSearchException(category); } final Filter hintsFilter = new Filter() { public boolean filter(final Object provider) { return isAcceptable(category.cast(provider), category, hints, filter); } }; synchronizeIteratorProviders(); scanForPluginsIfNeeded(category); return getServiceProviders(category, hintsFilter, true); } /** * Implementation of {@link #getServiceProviders(Class, Filter, Hints)} without the filtering * applied by the {@link #isAcceptable(Object, Class, Hints, Filter)} method. If this filtering * is not already presents in the filter given to this method, then it must be applied on the * elements returned by the iterator. The later is preferrable when: * <p> * <ul> * <li>There is some cheaper tests to perform before {@code isAcceptable}.</li> * <li>We don't want a restrictive filter in order to avoid trigging a classpath * scan if this method doesn't found any element to iterate.</li> * </ul> * <p> * <b>Note:</b> * {@link #synchronizeIteratorProviders} should also be invoked once before this method. * * @todo Use Hints to match Constructor. */ final <T> Iterator<T> getUnfilteredProviders(final Class<T> category) { /* * If the map is not empty, then this mean that a scanning is under progress, i.e. * 'scanForPlugins' is currently being executed. This is okay as long as the user * is not asking for one of the categories under scanning. Otherwise, the answer * returned by 'getServiceProviders' would be incomplete because not all plugins * have been found yet. This can lead to some bugs hard to spot because this methoud * could complete normally but return the wrong plugin. It is safer to thrown an * exception so the user is advised that something is wrong. */ if (scanningCategories.contains(category)) { throw new RecursiveSearchException(category); } scanForPluginsIfNeeded(category); return getServiceProviders(category, true); } /** * Returns the first provider in the registry for the specified category, using the specified * map of hints (if any). This method may {@linkplain #scanForPlugins scan for plugins} the * first time it is invoked. Except as a result of this scan, no new provider instance is * created by the default implementation of this method. The {@link FactoryCreator} class * change this behavior however. * * @param <T> The class represented by the {@code category} argument. * @param category The category to look for. Must be one of the categories declared to the * constructor. Usually an interface class (not the actual implementation * class). * @param filter An optional filter, or {@code null} if none. * This is used for example in order to select the first factory for some * {@linkplain org.opengis.referencing.AuthorityFactory#getAuthority authority}. * @param hints A {@linkplain Hints map of hints}, or {@code null} if none. * @param key The key to use for looking for a user-provided instance in the hints, or * {@code null} if none. * @return A factory {@linkplain OptionalFactory#isAvailable available} for use for the * specified category and hints. The returns type is {@code Object} instead of * {@link Factory} because the factory implementation doesn't need to be a Geotools one. * @throws FactoryNotFoundException if no factory was found for the specified category, filter * and hints. * @throws FactoryRegistryException if a factory can't be returned for some other reason. * * @see #getServiceProviders(Class, Filter, Hints) * @see FactoryCreator#getServiceProvider */ public <T> T getServiceProvider(final Class<T> category, final Filter filter, Hints hints, final Hints.Key key) throws FactoryRegistryException { synchronizeIteratorProviders(); final boolean debug = LOGGER.isLoggable(DEBUG_LEVEL); if (debug) { /* * We are not required to insert the method name ("GetServiceProvider") in the * message because it is part of the informations already stored by LogRecord, * and formatted by the default java.util.logging.SimpleFormatter. * * Conventions for the message part according java.util.logging.Logger javadoc: * - "ENTRY" at the begining of a method. * - "RETURN" at the end of a method, if successful. * - "THROW" in case of failure. * - "CHECK" ... is our own addition to Sun's convention for this method ... */ debug("ENTRY", category, key, null, null); } Class<?> implementation = null; if (key != null) { /* * Sanity check: make sure that the key class is appropriate for the category. */ final Class<?> valueClass = key.getValueClass(); if (!category.isAssignableFrom(valueClass)) { if (debug) { debug("THROW", category, key, "unexpected type:", valueClass); } throw new IllegalArgumentException(Errors.format(ErrorKeys.ILLEGAL_KEY_$1, key)); } if (hints != null) { final Object hint = hints.get(key); if (hint != null) { if (debug) { debug("CHECK", category, key, "user provided a", hint.getClass()); } if (category.isInstance(hint)) { /* * The factory implementation was given explicitly by the user. * Nothing to do; we are done. */ if (debug) { debug("RETURN", category, key, "return hint as provided.", null); } return category.cast(hint); } /* * Before to pass the hints to the private 'getServiceImplementation' method, * remove the hint for the user-supplied key. This is because this hint has * been processed by this public 'getServiceProvider' method, and the policy * is to remove the processed hints before to pass them to child dependencies * (see the "Check recursively in factory dependencies" comment elswhere in * this class). * * Use case: DefaultDataSourceTest invokes indirectly 'getServiceProvider' * with a "CRS_AUTHORITY_FACTORY = ThreadedEpsgFactory.class" hint. However * ThreadedEpsgFactory (in the org.geotools.referencing.factory.epsg package) * is a wrapper around DirectEpsgFactory, and defines this dependency through * a "CRS_AUTHORITY_FACTORY = DirectEpsgFactory.class" hint. There is no way * to match this hint for both factories in same time. Since we must choose * one, we assume that the user is interrested in the most top level one and * discart this particular hint for the dependencies. */ hints = new Hints(hints); if (hints.remove(key) != hint) { // Should never happen except on concurrent modification in an other thread. throw new AssertionError(key); } /* * If the user accepts many implementation classes, then try all of them in * the preference order given by the user. The last class (or the singleton * if the hint was not an array) will be tried using the "normal" path * (oustide the loop) in order to get the error message in case of failure. */ if (hint instanceof Class<?>[]) { final Class<?>[] types = (Class<?>[]) hint; final int length = types.length; for (int i=0; i<length-1; i++) { final Class<?> type = types[i]; if (debug) { debug("CHECK", category, key, "consider hint[" + i + ']', type); } final T candidate = getServiceImplementation(category, type, filter, hints); if (candidate != null) { if (debug) { debug("RETURN", category, key, "found implementation", candidate.getClass()); } return candidate; } } if (length != 0) { implementation = types[length-1]; // Last try to be done below. } } else { implementation = (Class<?>) hint; } } } } if (debug && implementation != null) { debug("CHECK", category, key, "consider hint[last]", implementation); } final T candidate = getServiceImplementation(category, implementation, filter, hints); if (candidate != null) { if (debug) { debug("RETURN", category, key, "found implementation", candidate.getClass()); } return candidate; } if (debug) { debug("THROW", category, key, "could not find implementation.", null); } throw new FactoryNotFoundException(Errors.format(ErrorKeys.FACTORY_NOT_FOUND_$1, implementation!=null ? implementation : category)); } /** * Logs a debug message for {@link #getServiceProvider} method. Note: we are not required * to insert the method name ({@code "GetServiceProvider"}) in the message because it is * part of the informations already stored by {@link LogRecord}, and formatted by the * default {@link java.util.logging.SimpleFormatter}. * * @param status {@code "ENTRY"}, {@code "RETURN"} or {@code "THROW"}, * according {@link Logger} conventions. * @param category The category given to the {@link #getServiceProvider} method. * @param key The key being examined, or {@code null}. * @param message Optional message, or {@code null} if none. * @param type Optional class to format after the message, or {@code null}. */ private static void debug(final String status, final Class<?> category, final Hints.Key key, final String message, final Class type) { final StringBuilder buffer = new StringBuilder(status); buffer.append(Utilities.spaces(Math.max(1, 7-status.length()))) .append('(').append(Classes.getShortName(category)); if (key != null) { buffer.append(", ").append(key); } buffer.append(')'); if (message != null) { buffer.append(": ").append(message); } if (type != null) { buffer.append(' ').append(Classes.getShortName(type)).append('.'); } final LogRecord record = new LogRecord(DEBUG_LEVEL, buffer.toString()); record.setSourceClassName(FactoryRegistry.class.getName()); record.setSourceMethodName("getServiceProvider"); record.setLoggerName(LOGGER.getName()); LOGGER.log(record); } /** * Searchs the first implementation in the registery matching the specified conditions. * This method is invoked only by the {@link #getServiceProvider(Class, Filter, Hints, * Hints.Key)} public method above; there is no recursivity there. This method do not * creates new instance if no matching factory is found. * * @param category The category to look for. Usually an interface class. * @param implementation The desired class for the implementation, or {@code null} if none. * @param filter An optional filter, or {@code null} if none. * @param hints A {@linkplain Hints map of hints}, or {@code null} if none. * @return A factory for the specified category and hints, or {@code null} if none. */ private <T> T getServiceImplementation(final Class<T> category, final Class<?> implementation, final Filter filter, final Hints hints) { for (final Iterator<T> it=getUnfilteredProviders(category); it.hasNext();) { final T candidate = it.next(); // Implementation class must be tested before 'isAcceptable' // in order to avoid StackOverflowError in some situations. if (implementation!=null && !implementation.isInstance(candidate)) { continue; } if (!isAcceptable(candidate, category, hints, filter)) { continue; } return candidate; } final List<Reference<T>> cached = getCachedProviders(category); if (cached != null) { /* * Checks if a factory previously created by FactoryCreator could fit. This * block should never be executed if this instance is not a FactoryCreator. */ for (final Iterator<Reference<T>> it=cached.iterator(); it.hasNext();) { final T candidate = it.next().get(); if (candidate == null) { it.remove(); continue; } if (implementation!=null && !implementation.isInstance(candidate)) { continue; } if (!isAcceptable(candidate, category, hints, filter)) { continue; } return candidate; } } return null; } /** * Returns the providers available in the cache, or {@code null} if none. * To be overridden by {@link FactoryCreator} only. */ <T> List<Reference<T>> getCachedProviders(final Class<T> category) { return null; } /** * Returns {@code true} is the specified {@code factory} meets the requirements specified by * a map of {@code hints} and the filter. This method is the entry point for the following * public methods: * <ul> * <li>Singleton {@link #getServiceProvider (Class category, Filter, Hints, Hints.Key)}</li> * <li>Iterator {@link #getServiceProviders(Class category, Filter, Hints)}</li> * </ul> * * @param candidate The factory to checks. * @param category The factory category. Usually an interface. * @param hints The optional user requirements, or {@code null}. * @param filter The optional filter, or {@code null}. * @return {@code true} if the {@code factory} meets the user requirements. */ final <T> boolean isAcceptable(final T candidate, final Class<T> category, final Hints hints, final Filter filter) { if (filter!=null && !filter.filter(candidate)) { return false; } /* * Note: isAvailable(...) must be tested before checking the hints, because in current * Geotools implementation (especially DeferredAuthorityFactory), some hints computation * are deferred until a connection to the database is etablished (which 'isAvailable' * does in order to test the connection). */ if (!isAvailable(candidate)) { return false; } if (hints != null) { if (candidate instanceof Factory) { if (!usesAcceptableHints((Factory) candidate, category, hints, (Set<Factory>) null)) { return false; } } } /* * Checks for optional user conditions supplied in FactoryRegistry subclasses. */ return isAcceptable(candidate, category, hints); } /** * Returns {@code true} is the specified {@code factory} meets the requirements specified * by a map of {@code hints}. This method checks only the hints; it doesn't check the * {@link Filter}, the {@linkplain OptionalFactory#isAvailable availability} or the * user-overrideable {@link #isAcceptable(Object, Class, Hints)} method. This method * invokes itself recursively. * * @param factory The factory to checks. * @param category The factory category. Usually an interface. * @param hints The user requirements ({@code null} not allowed). * @param alreadyDone Should be {@code null} except on recursive calls (for internal use only). * @return {@code true} if the {@code factory} meets the hints requirements. */ private boolean usesAcceptableHints(final Factory factory, final Class<?> category, final Hints hints, Set<Factory> alreadyDone) { /* * Ask for implementation hints with special care against infinite recursivity. * Some implementations use deferred algorithms fetching dependencies only when * first needed. The call to getImplementationHints() is sometime a trigger for * fetching dependencies (in order to return accurate hints). For example the * BufferedCoordinateOperationFactory implementation asks for an other instance * of CoordinateOperationFactory, the instance to cache behind a buffer, which * should not be itself. Of course BufferedCoordinateOperation will checks that * it is not caching itself, but its test happen too late for preventing a never- * ending loop if we don't put a 'testingHints' guard here. It is also a safety * against broken factory implementations. */ if (!testingHints.addAndCheck(factory)) { return false; } final Map<RenderingHints.Key, ?> implementationHints; try { implementationHints = Hints.stripNonKeys(factory.getImplementationHints()); } finally { testingHints.removeAndCheck(factory); } if (implementationHints == null) { // factory was bad and did not meet contract - assume it used no Hints return true; } /* * We got the implementation hints. Now tests their compatibility. */ Hints remaining = null; for (final Map.Entry<?,?> entry : implementationHints.entrySet()) { final Object key = entry.getKey(); final Object value = entry.getValue(); final Object expected = hints.get(key); if (expected != null) { /* * We have found a hint that matter. Check if the * available factory meets the user's criterions. */ if (expected instanceof Class<?>) { if (!((Class<?>) expected).isInstance(value)) { return false; } } else if (expected instanceof Class<?>[]) { final Class<?>[] types = (Class<?>[]) expected; int i = 0; do if (i >= types.length) return false; while (!types[i++].isInstance(value)); } else if (!expected.equals(value)) { return false; } } /* * Checks recursively in factory dependencies, if any. Note that the dependencies * will be checked against a subset of user's hints. More specifically, all hints * processed by the current pass will NOT be passed to the factories dependencies. * This is because the same hint may appears in the "parent" factory and a "child" * dependency with different value. For example the FORCE_LONGITUDE_FIRST_AXIS_ORDER * hint has the value TRUE in OrderedAxisAuthorityFactory, but the later is basically * a wrapper around the ThreadedEpsgFactory (typically), which has the value FALSE * for the same hint. * * Additional note: The 'alreadyDone' set is a safety against cyclic dependencies, * in order to protect ourself against never-ending loops. This is not the same * kind of dependencies than 'testingHints'. It is a "factory A depends on factory * B which depends on factory A" loop, which is legal. */ if (value instanceof Factory) { final Factory dependency = (Factory) value; if (alreadyDone == null) { alreadyDone = new HashSet<Factory>(); } if (!alreadyDone.contains(dependency)) { alreadyDone.add(factory); if (remaining == null) { remaining = new Hints(hints); remaining.keySet().removeAll(implementationHints.keySet()); } final Class<?> type; if (key instanceof Hints.Key) { type = ((Hints.Key) key).getValueClass(); } else { type = Factory.class; // Kind of unknown factory type... } // Recursive call to this method for scanning dependencies. if (!usesAcceptableHints(dependency, type, remaining, alreadyDone)) { return false; } } } } return true; } /** * Returns {@code true} if the specified {@code provider} meets the requirements specified by * a map of {@code hints}. The default implementation always returns {@code true}. There is no * need to override this method for {@link AbstractFactory} implementations, since their hints * are automatically checked. Override this method for non-Geotools implementations. * For example a JTS geometry factory finder may overrides this method in order to check * if a {@link com.vividsolutions.jts.geom.GeometryFactory} uses the required * {@link com.vividsolutions.jts.geom.CoordinateSequenceFactory}. Such method should be * implemented as below, since this method may be invoked for various kind of objects: * * <blockquote><pre> * if (provider instanceof GeometryFactory) { * // ... Check the GeometryFactory state here. * } * </pre></blockquote> * * @param <T> The class represented by the {@code category} argument. * @param provider The provider to checks. * @param category The factory category. Usually an interface. * @param hints The user requirements, or {@code null} if none. * @return {@code true} if the {@code provider} meets the user requirements. */ protected <T> boolean isAcceptable(final T provider, final Class<T> category, final Hints hints) { return true; } /** * Returns {@code true} if the specified factory is available. */ private boolean isAvailable(final Object provider) { if (!(provider instanceof OptionalFactory)) { return true; } final OptionalFactory factory = (OptionalFactory) provider; final Class<? extends OptionalFactory> type = factory.getClass(); if (!testingAvailability.addAndCheck(type)) { throw new RecursiveSearchException(type); } try { return factory.isAvailable(); } finally { testingAvailability.removeAndCheck(type); } } /** * Returns all class loaders to be used for scanning plugins. Current implementation * returns the following class loaders: * <p> * <ul> * <li>{@linkplain Class#getClassLoader This object class loader}</li> * <li>{@linkplain Thread#getContextClassLoader The thread context class loader}</li> * <li>{@linkplain ClassLoader#getSystemClassLoader The system class loader}</li> * </ul> * * The actual number of class loaders may be smaller if redundancies was found. * If some more classloaders should be scanned, they shall be added into the code * of this method. * * @return All classloaders to be used for scanning plugins. */ public final Set<ClassLoader> getClassLoaders() { final Set<ClassLoader> loaders = new HashSet<ClassLoader>(); for (int i=0; i<4; i++) { final ClassLoader loader; try { switch (i) { case 0: loader = getClass().getClassLoader(); break; case 1: loader = FactoryRegistry.class.getClassLoader(); break; case 2: loader = Thread.currentThread().getContextClassLoader(); break; case 3: loader = ClassLoader.getSystemClassLoader(); break; // Add any supplementary class loaders here, if needed. default: throw new AssertionError(i); // Should never happen. } } catch (SecurityException exception) { // We are not allowed to get a class loader. // Continue; some other class loader may be available. continue; } loaders.add(loader); } loaders.remove(null); loaders.addAll(GeoTools.getClassLoaders()); /* * We now have a set of class loaders with duplicated object already removed * (e.g. system classloader == context classloader). However, we may still * have an other form of redundancie. A class loader may be the parent of an * other one. Try to remove those dependencies. */ final ClassLoader[] asArray = loaders.toArray(new ClassLoader[loaders.size()]); for (int i=0; i<asArray.length; i++) { ClassLoader loader = asArray[i]; try { while ((loader=loader.getParent()) != null) { loaders.remove(loader); } } catch (SecurityException exception) { // We are not allowed to fetch the parent class loader. // Ignore (some redundancies may remains). } } if (loaders.isEmpty()) { LOGGER.warning("No class loaders available."); } return loaders; } /** * Scans for factory plug-ins on the application class path. This method is * needed because the application class path can theoretically change, or * additional plug-ins may become available. Rather than re-scanning the * classpath on every invocation of the API, the class path is scanned * automatically only on the first invocation. Clients can call this * method to prompt a re-scan. Thus this method need only be invoked by * sophisticated applications which dynamically make new plug-ins * available at runtime. */ public void scanForPlugins() { final Set<ClassLoader> loaders = getClassLoaders(); for (final Iterator<Class<?>> categories=getCategories(); categories.hasNext();) { final Class<?> category = categories.next(); scanForPlugins(loaders, category); } } /** * Scans for factory plug-ins of the given category, with guard against recursivities. * The recursivity check make debugging easier than inspecting a {@link StackOverflowError}. * * @param loader The class loader to use. * @param category The category to scan for plug-ins. */ private <T> void scanForPlugins(final Collection<ClassLoader> loaders, final Class<T> category) { if (!scanningCategories.addAndCheck(category)) { throw new RecursiveSearchException(category); } try { final StringBuilder message = getLogHeader(category); boolean newServices = false; /* * First, scan META-INF/services directories (the default mechanism). */ for (final ClassLoader loader : loaders) { newServices |= register(lookupProviders(category, loader), category, message); newServices |= registerFromSystemProperty(loader, category, message); } /* * Next, query the user-provider iterators, if any. */ final FactoryIteratorProvider[] fip = FactoryIteratorProviders.getIteratorProviders(); for (int i=0; i<fip.length; i++) { final Iterator<T> it = fip[i].iterator(category); if (it != null) { newServices |= register(it, category, message); } } /* * Finally, log the list of registered factories. */ if (newServices) { log("scanForPlugins", message); } } finally { scanningCategories.removeAndCheck(category); } } /** * Scans the given category for plugins only if needed. After this method has been * invoked once for a given category, it will no longer scan for that category. */ private <T> void scanForPluginsIfNeeded(final Class<?> category) { if (needScanForPlugins != null && needScanForPlugins.remove(category)) { if (needScanForPlugins.isEmpty()) { needScanForPlugins = null; } scanForPlugins(getClassLoaders(), category); } } /** * {@linkplain #registerServiceProvider Registers} all factories given by the * supplied iterator. * * @param factories The factories (or "service providers") to register. * @param category the category under which to register the providers. * @param message A buffer where to write the logging message. * @return {@code true} if at least one factory has been registered. */ private <T> boolean register(final Iterator<T> factories, final Class<T> category, final StringBuilder message) { boolean newServices = false; final String lineSeparator = System.getProperty("line.separator", "\n"); while (factories.hasNext()) { T factory; try { factory = factories.next(); } catch (OutOfMemoryError error) { // Makes sure that we don't try to handle this error. throw error; } catch (NoClassDefFoundError error) { /* * A provider can't be registered because of some missing dependencies. * This occurs for example when trying to register the WarpTransform2D * math transform on a machine without JAI installation. Since the factory * may not be essential (this is the case of WarpTransform2D), just skip it. */ loadingFailure(category, error, false); continue; } catch (ExceptionInInitializerError error) { /* * If an exception occured during class initialization, log the cause. * The ExceptionInInitializerError alone doesn't help enough. */ final Throwable cause = error.getCause(); if (cause != null) { loadingFailure(category, cause, true); } throw error; } catch (Error error) { if (!Classes.getShortClassName(error).equals("ServiceConfigurationError")) { // We want to handle sun.misc.ServiceConfigurationError only. Unfortunatly, we // need to rely on reflection because this error class is not a commited API. // TODO: Check if the error is catchable with JSE 6. throw error; } /* * Failed to register a factory for a reason probably related to the plugin * initialisation. It may be some factory-dependent missing resources. */ loadingFailure(category, error, true); continue; } final Class<? extends T> factoryClass = factory.getClass().asSubclass(category); /* * If the factory implements more than one interface and an instance were * already registered, reuse the same instance instead of duplicating it. */ final T replacement = getServiceProviderByClass(factoryClass); if (replacement != null) { factory = replacement; // Need to register anyway, because the category may not be the same. } if (registerServiceProvider(factory, category)) { /* * The factory is now registered. Add it to the message to be logged. We will log * all factories together in a single log event because some registration (e.g. * MathTransformProviders) would be otherwise quite verbose. */ message.append(lineSeparator); message.append(" "); message.append(factoryClass.getName()); newServices = true; } } return newServices; } /** * If a system property was setup, load the class (if not already registered) * and move it in front of any other factory. This is done for compatibility * with legacy {@code FactoryFinder} implementation. * * @param loader The class loader to use. * @param category The category to scan for plug-ins. * @param message A buffer where to write the logging message. * @return {@code true} if at least one factory has been registered. */ private <T> boolean registerFromSystemProperty(final ClassLoader loader, final Class<T> category, final StringBuilder message) { boolean newServices = false; try { final String classname = System.getProperty(category.getName()); if (classname != null) try { final Class<?> candidate = loader.loadClass(classname); if (category.isAssignableFrom(candidate)) { final Class<? extends T> factoryClass = candidate.asSubclass(category); T factory = getServiceProviderByClass(factoryClass); if (factory == null) try { factory = factoryClass.newInstance(); if (registerServiceProvider(factory, category)) { message.append(System.getProperty("line.separator", "\n")); message.append(" "); message.append(factoryClass.getName()); newServices = true; } } catch (IllegalAccessException exception) { throw new FactoryRegistryException(Errors.format( ErrorKeys.CANT_CREATE_FACTORY_$1, classname), exception); } catch (InstantiationException exception) { throw new FactoryRegistryException(Errors.format( ErrorKeys.CANT_CREATE_FACTORY_$1, classname), exception); } /* * Put this factory in front of every other factories (including the ones loaded * in previous class loaders, which is why we don't inline this ordering in the * 'register' loop). Note: if some factories were not yet registered, they will * not be properly ordered. Since this code exists more for compatibility reasons * than as a commited API, we ignore this short comming for now. */ for (final Iterator<T> it=getServiceProviders(category, false); it.hasNext();) { final T other = it.next(); if (other != factory) { setOrdering(category, factory, other); } } } } catch (ClassNotFoundException exception) { // The class has not been found, maybe because we are not using the appropriate // class loader. Ignore (do not thrown an exception), in order to give a chance // to the caller to invokes this method again with a different class loader. } } catch (SecurityException exception) { // We are not allowed to read property, probably // because we are running in an applet. Ignore... } return newServices; } /** * Invoked when a factory can't be loaded. Log a warning, but do not stop the process. */ private static void loadingFailure(final Class<?> category, final Throwable error, final boolean showStackTrace) { final String name = Classes.getShortName(category); final StringBuilder cause = new StringBuilder(Classes.getShortClassName(error)); final String message = error.getLocalizedMessage(); if (message != null) { cause.append(": "); cause.append(message); } final LogRecord record = Loggings.format(Level.WARNING, LoggingKeys.CANT_LOAD_SERVICE_$2, name, cause.toString()); if (showStackTrace) { record.setThrown(error); } record.setSourceClassName(FactoryRegistry.class.getName()); record.setSourceMethodName("scanForPlugins"); record.setLoggerName(LOGGER.getName()); LOGGER.log(record); } /** * Prepares a message to be logged if any provider has been registered. */ private static StringBuilder getLogHeader(final Class<?> category) { return new StringBuilder(Loggings.getResources(null).getString( LoggingKeys.FACTORY_IMPLEMENTATIONS_$1, category)); } /** * Log the specified message after all provider for a given category have been registered. */ private static void log(final String method, final StringBuilder message) { final LogRecord record = new LogRecord(Level.CONFIG, message.toString()); record.setSourceClassName(FactoryRegistry.class.getName()); record.setSourceMethodName(method); record.setLoggerName(LOGGER.getName()); LOGGER.log(record); } /** * Synchronizes the content of the {@link #globalConfiguration} with * {@link FactoryIteratorProviders#GLOBAL}. New providers are {@linkplain #register registered} * immediately. Note that this method is typically invoked in a different thread than * {@link FactoryIteratorProviders} method calls. * * @see FactoryIteratorProviders#addFactoryIteratorProvider */ private void synchronizeIteratorProviders() { final FactoryIteratorProvider[] newProviders = globalConfiguration.synchronizeIteratorProviders(); if (newProviders == null) { return; } for (final Iterator<Class<?>> categories=getCategories(); categories.hasNext();) { final Class<?> category = categories.next(); if (needScanForPlugins == null || !needScanForPlugins.contains(category)) { /* * Register immediately the factories only if some other factories were already * registered for this category, because in such case scanForPlugin() will not * be invoked automatically. If no factory are registered for this category, do * nothing - we will rely on the lazy invocation of scanForPlugins() when first * needed. We perform this check because getServiceProviders(category).hasNext() * is the criterion used by FactoryRegistry in order to decide if it should invoke * automatically scanForPlugins(). */ for (int i=0; i<newProviders.length; i++) { register(newProviders[i], category); } } } } /** * Registers every factories from the specified provider for the given category. */ private <T> void register(final FactoryIteratorProvider provider, final Class<T> category) { final Iterator<T> it = provider.iterator(category); if (it != null) { final StringBuilder message = getLogHeader(category); if (register(it, category, message)) { log("synchronizeIteratorProviders", message); } } } /** * Set pairwise ordering between all factories according a comparator. Calls to * <code>{@linkplain Comparator#compare compare}(factory1, factory2)</code> should returns: * <ul> * <li>{@code -1} if {@code factory1} is preferred to {@code factory2}</li> * <li>{@code +1} if {@code factory2} is preferred to {@code factory1}</li> * <li>{@code 0} if there is no preferred order between {@code factory1} and * {@code factory2}</li> * </ul> * * @param <T> The class represented by the {@code category} argument. * @param category The category to set ordering. * @param comparator The comparator to use for ordering. * @return {@code true} if at least one ordering setting has been modified as a consequence * of this call. */ public <T> boolean setOrdering(final Class<T> category, final Comparator<T> comparator) { boolean set = false; final List<T> previous = new ArrayList<T>(); for (final Iterator<T> it=getServiceProviders(category, false); it.hasNext();) { final T f1 = it.next(); for (int i=previous.size(); --i>=0;) { final T f2 = previous.get(i); final int c; try { c = comparator.compare(f1, f2); } catch (ClassCastException exception) { /* * This exception is expected if the user-supplied comparator follows strictly * the java.util.Comparator specification and has determined that it can't * compare the supplied factories. From ServiceRegistry point of view, it just * means that the ordering between those factories will stay undeterminated. */ continue; } if (c > 0) { set |= setOrdering(category, f1, f2); } else if (c < 0) { set |= setOrdering(category, f2, f1); } } previous.add(f1); } return set; } /** * Sets or unsets a pairwise ordering between all factories meeting a criterion. For example * in the CRS framework ({@link org.geotools.referencing.FactoryFinder}), this is used for * setting ordering between all factories provided by two vendors, or for two authorities. * If one or both factories are not currently registered, or if the desired ordering is * already set/unset, nothing happens and false is returned. * * @param <T> The class represented by the {@code base} argument. * @param base The base category. Only categories {@linkplain Class#isAssignableFrom * assignable} to {@code base} will be processed. * @param set {@code true} for setting the ordering, or {@code false} for unsetting. * @param service1 Filter for the preferred factory. * @param service2 Filter for the factory to which {@code service1} is preferred. * @return {@code true} if the ordering changed as a result of this call. */ public <T> boolean setOrdering(final Class<T> base, final boolean set, final Filter service1, final Filter service2) { boolean done = false; for (final Iterator<Class<?>> categories=getCategories(); categories.hasNext();) { final Class<?> candidate = categories.next(); if (base.isAssignableFrom(candidate)) { final Class<? extends T> category = candidate.asSubclass(base); done |= setOrUnsetOrdering(category, set, service1, service2); } } return done; } /** * Helper method for the above. */ private <T> boolean setOrUnsetOrdering(final Class<T> category, final boolean set, final Filter service1, final Filter service2) { boolean done = false; T impl1 = null; T impl2 = null; for (final Iterator<? extends T> it=getServiceProviders(category, false); it.hasNext();) { final T factory = it.next(); if (service1.filter(factory)) impl1 = factory; if (service2.filter(factory)) impl2 = factory; if (impl1!=null && impl2!=null && impl1!=impl2) { if (set) done |= setOrdering(category, impl1, impl2); else done |= unsetOrdering(category, impl1, impl2); } } return done; } }