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

alluxio.extensions.ExtensionFactoryRegistry Maven / Gradle / Ivy

There is a newer version: 313
Show newest version
/*
 * The Alluxio Open Foundation licenses this work under the Apache License, version 2.0
 * (the "License"). You may not use this work except in compliance with the License, which is
 * available at www.apache.org/licenses/LICENSE-2.0
 *
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
 * either express or implied, as more fully set forth in the License.
 *
 * See the NOTICE file distributed with this work for information regarding copyright ownership.
 */

package alluxio.extensions;

import static java.util.stream.Collectors.toList;

import alluxio.conf.AlluxioConfiguration;
import alluxio.conf.PropertyKey;
import alluxio.recorder.Recorder;
import alluxio.underfs.UnderFileSystemFactory;
import alluxio.util.ExtensionUtils;
import alluxio.util.io.PathUtils;

import com.google.common.base.Preconditions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.nio.file.DirectoryStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.ServiceLoader;
import java.util.concurrent.CopyOnWriteArrayList;
import javax.annotation.concurrent.NotThreadSafe;

/**
 * 

* An extension registry that uses the {@link ServiceLoader} mechanism to automatically discover * available factories and provides a central place for obtaining actual extension instances. *

*

Registering New Factories

*

* New factories can be registered either using the {@linkplain ServiceLoader} based automatic * discovery mechanism or manually using the static {@link #register(T)} * method. The down-side of the automatic discovery mechanism is that the discovery order is not * controllable. As a result if your implementation is designed as a replacement for one of the * standard implementations, depending on the order in which the JVM discovers the services your own * implementation may not take priority. You can enable {@code DEBUG} level logging for this class * to see the order in which factories are discovered and which is selected when obtaining a * {@link T} instance for a path. If this shows that your implementation is not * getting discovered or used, you may wish to use the manual registration approach. *

*

Automatic Discovery

*

* To use the {@linkplain ServiceLoader} based mechanism you need to have a file with the same name * as the factory class {@code T} placed in the {@code META-INF\services} directory * of your project. This file should contain the full name of your factory types (one per line), * your factory types must have a public unparameterised constructor available (see * {@link ServiceLoader} for more detail on this). You can enable {@code DEBUG} level logging to see * factories as they are discovered if you wish to check that your implementation gets discovered. *

*

Manual Registration

*

* To manually register a factory, simply pass an instance of your factory to the * {@link #register(T)} method. This can be useful when your factory cannot be * instantiated without arguments or in cases where automatic discovery does not give your factory * priority. Factories registered this way will be registered at the start of the factories list so * will have the first opportunity to indicate whether they support a requested path. *

* @param The type of extension factory * @param the type of configuration to be used when creating the extension */ @NotThreadSafe public class ExtensionFactoryRegistry, S extends AlluxioConfiguration> { private static final Logger LOG = LoggerFactory.getLogger(ExtensionFactoryRegistry.class); public static final String UNKNOWN_VERSION = "unknown"; /** * The base list of factories, which does not include any lib or extension factories. The only * factories in the base list will be built-in factories, and any additional factories * registered by tests. All other factories will be discovered and service loaded when extension * creation occurs. */ private final List mFactories = new CopyOnWriteArrayList<>(); private final String mExtensionPattern; private final Class mFactoryClass; private boolean mInit = false; /** * Constructs a registry for loading extension of a particular type. * @param factoryClass the type of the extension factory * @param extensionPattern the pattern used to select libraries to be loaded */ public ExtensionFactoryRegistry(Class factoryClass, String extensionPattern) { mFactoryClass = Preconditions.checkNotNull(factoryClass, "factoryClass"); mExtensionPattern = extensionPattern; init(); } private synchronized void init() { // Discover and register the available base factories ServiceLoader discoveredFactories = ServiceLoader.load(mFactoryClass, mFactoryClass.getClassLoader()); for (T factory : discoveredFactories) { LOG.debug("Discovered base extension factory implementation {} - {}", factory.getClass(), factory); register(factory); } } /** * Returns a read-only view of the available base factories. * * @return Read-only view of the available base factories */ public List getAvailable() { return Collections.unmodifiableList(mFactories); } /** * Finds all the factories that support the given path and record the detailed execution process. * * @param path path * @param conf configuration of the extension * @param recorder recorder used to record the detailed execution process * @return list of factories that support the given path which may be an empty list */ public List findAllWithRecorder(String path, S conf, Recorder recorder) { Preconditions.checkArgument(path != null, "path may not be null"); List eligibleFactories = scanRegistered(path, conf); if (!eligibleFactories.isEmpty()) { LOG.debug("Find {} eligible items from registered factories for path {}", eligibleFactories.size(), path); return eligibleFactories; } List factories = new ArrayList<>(mFactories); String libDir = PathUtils.concatPath(conf.getString(PropertyKey.HOME), "lib"); String extensionDir = conf.getString(PropertyKey.EXTENSIONS_DIR); scanLibs(factories, libDir); recorder.record("Loaded {} factory core jars from {}", factories.size(), libDir); scanExtensions(factories, extensionDir); recorder.record("Loaded extension jars from {}.%n" + "The total number of loaded factory core jars is {}", extensionDir, factories.size()); if (conf.isSetByUser(PropertyKey.UNDERFS_VERSION)) { recorder.record("alluxio.underfs.version is set by user, target version is {}", conf.getString(PropertyKey.UNDERFS_VERSION)); } else { recorder.record("alluxio.underfs.version is not set by user"); } for (T factory : factories) { // if `getVersion` returns null set the version to "unknown" String version = UNKNOWN_VERSION; if (factory instanceof UnderFileSystemFactory) { version = Optional.ofNullable(((UnderFileSystemFactory) factory) .getVersion()).orElse(UNKNOWN_VERSION); } if (factory.supportsPath(path, conf)) { String message = String.format("Adding factory %s of version %s which supports path %s", factory.getClass().getSimpleName(), version, path); recorder.record(message); LOG.debug(message); eligibleFactories.add(factory); } else { recorder.record("Factory implementation {} of version {} " + "isn't eligible for path {}", factory.getClass().getSimpleName(), version, path); } } if (eligibleFactories.isEmpty()) { String message = String.format("No factory implementation supports the path %s", path); recorder.record(message); LOG.warn(message); } return eligibleFactories; } /** * Finds factories from registered collection. * @param path path * @param conf configuration of the extension * @return list of factories that support the given path which may be an empty list */ public List scanRegistered(String path, S conf) { return mFactories.stream() .filter(factory -> factory.supportsPath(path, conf)) .collect(toList()); } /** * Finds all factory from the extensions directory. * * @param factories list of factories to add to */ private void scanExtensions(List factories, String extensionsDir) { LOG.info("Loading extension jars from {}", extensionsDir); scan(Arrays.asList(ExtensionUtils.listExtensions(extensionsDir)), factories); } /** * Finds all factory from the lib directory. * * @param factories list of factories to add to */ private void scanLibs(List factories, String libDir) { LOG.info("Loading core jars from {}", libDir); List files = new ArrayList<>(); try (DirectoryStream stream = Files.newDirectoryStream(Paths.get(libDir), mExtensionPattern)) { for (Path entry : stream) { if (entry.toFile().isFile()) { files.add(entry.toFile()); } } } catch (IOException e) { LOG.warn("Failed to load libs: {}", e.toString()); } scan(files, factories); } /** * Class-loads jar files that have not been loaded. * * @param files jar files to class-load * @param factories list of factories to add to */ private void scan(List files, List factories) { for (File jar : files) { try { URL extensionURL = jar.toURI().toURL(); String jarPath = extensionURL.toString(); ClassLoader defaultClassLoader = Thread.currentThread().getContextClassLoader() == null ? ClassLoader.getSystemClassLoader() : Thread.currentThread().getContextClassLoader(); ClassLoader extensionsClassLoader = new ExtensionsClassLoader(new URL[] {extensionURL}, defaultClassLoader); ServiceLoader extensionServiceLoader = ServiceLoader.load(mFactoryClass, extensionsClassLoader); for (T factory : extensionServiceLoader) { LOG.debug("Discovered a factory implementation {} - {} in jar {}", factory.getClass(), factory, jarPath); register(factory, factories); // Also add to the cache register(factory); } } catch (Throwable t) { LOG.warn("Failed to load jar {}: {}", jar, t.toString()); } } } /** * Registers a new factory. *

* Factories are registered at the start of the factories list so they can override the existing * automatically discovered factories. Generally if you use the {@link ServiceLoader} mechanism * properly it should be unnecessary to call this, however since ServiceLoader discovery order * may be susceptible to class loader behavioral differences there may be rare cases when you * need to manually register the desired factory. *

* * @param factory factory to register */ public void register(T factory) { register(factory, mFactories); } private void register(T factory, List factories) { if (factory == null) { return; } LOG.debug("Registered factory implementation {} - {}", factory.getClass(), factory); // Insert at start of list so it will take precedence over automatically discovered and // previously registered factories factories.add(0, factory); } /** * Resets the registry to its default state *

* This clears the registry as it stands and rediscovers the available factories. *

*/ public synchronized void reset() { if (mInit) { // Reset state mInit = false; mFactories.clear(); } // Reinitialise init(); } /** * Unregisters an existing factory. * * @param factory factory to unregister */ public void unregister(T factory) { unregister(factory, mFactories); } private void unregister(T factory, List factories) { if (factory == null) { return; } LOG.debug("Unregistered factory implementation {} - {}", factory.getClass(), factory); factories.remove(factory); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy