From bdf4d541e315856bfa368d09c630bb80f4c42ede Mon Sep 17 00:00:00 2001 From: Bingkun Guo Date: Fri, 5 Jun 2015 01:01:23 -0500 Subject: [PATCH] Fix issue #1016 that if user specifies extension coordinates, the local extensions will be loaded twice, which could cause Guice duplicate binding errors. Add unit test to replicate duplicate extension issue. Update documents. Add an package accessible getter for loadersMap for testing only. Notice that extensions explicitly specified in druid.extensions.coordinates have a higher priority than ones included in the classpath. Extension modules that don't have a canonical class name will be ignored. --- docs/content/configuration/index.md | 2 +- .../druid/initialization/Initialization.java | 63 ++++++++++++++----- .../initialization/InitializationTest.java | 51 ++++++++++++--- 3 files changed, 92 insertions(+), 24 deletions(-) diff --git a/docs/content/configuration/index.md b/docs/content/configuration/index.md index 42938ca649d..b1ef3ea5521 100644 --- a/docs/content/configuration/index.md +++ b/docs/content/configuration/index.md @@ -23,7 +23,7 @@ Many of Druid's external dependencies can be plugged in as modules. Extensions c |--------|-----------|-------| |`druid.extensions.remoteRepositories`|This is a JSON Array list of remote repositories to load dependencies from. If this is not set to '[]', Druid will try to download extensions at the specified remote repository.|["http://repo1.maven.org/maven2/", "https://metamx.artifactoryonline.com/metamx/pub-libs-releases-local"]| |`druid.extensions.localRepository`|. The way maven gets dependencies is that it downloads them to a "local repository" on your local disk and then collects the paths to each of the jars. This specifies the directory to consider the "local repository". If this is set, remoteRepositories is not required.|`~/.m2/repository`| -|`druid.extensions.coordinates`|This is a JSON array of "groupId:artifactId[:version]" maven coordinates. For artifacts without version specified, Druid will append the default version.|[]| +|`druid.extensions.coordinates`|This is a JSON array of "groupId:artifactId[:version]" maven coordinates. For artifacts without version specified, Druid will append the default version. Notice: extensions explicitly specified in this property will have precedence over ones included in the classpath when Druid loads extensions. If there are duplicate extensions, Druid will only load ones explicitly specified here|[]| |`druid.extensions.defaultVersion`|Version to use for extension artifacts without version information.|`druid-server` artifact version.| |`druid.extensions.searchCurrentClassloader`|This is a boolean flag that determines if Druid will search the main classloader for extensions. It defaults to true but can be turned off if you have reason to not automatically add all modules on the classpath.|true| diff --git a/server/src/main/java/io/druid/initialization/Initialization.java b/server/src/main/java/io/druid/initialization/Initialization.java index a6dbee5bcf6..e98c2d99236 100644 --- a/server/src/main/java/io/druid/initialization/Initialization.java +++ b/server/src/main/java/io/druid/initialization/Initialization.java @@ -118,33 +118,52 @@ public class Initialization /** * Used for testing only */ - protected static void clearLoadedModules() + static void clearLoadedModules() { extensionsMap.clear(); } + /** + * Used for testing only + */ + static Map getLoadersMap() + { + return loadersMap; + } + + /** + * Look for extension modules for the given class from both classpath and druid.extensions.coordinates. + * Extensions explicitly specified in druid.extensions.coordinates will be loaded first, if there is a duplicate + * extension from classpath, it will be ignored. + * + * @param config Extensions configuration + * @param clazz The class of extension module (e.g., DruidModule) + * + * @return A collection that contains distinct extension modules + */ public synchronized static Collection getFromExtensions(ExtensionsConfig config, Class clazz) { final TeslaAether aether = getAetherClient(config); - Set retVal = Sets.newHashSet(); - - if (config.searchCurrentClassloader()) { - for (T module : ServiceLoader.load(clazz, Initialization.class.getClassLoader())) { - log.info("Adding local module[%s]", module.getClass()); - retVal.add(module); - } - } + final Set retVal = Sets.newHashSet(); + final Set extensionNames = Sets.newHashSet(); for (String coordinate : config.getCoordinates()) { log.info("Loading extension[%s] for class[%s]", coordinate, clazz.getName()); try { URLClassLoader loader = getClassLoaderForCoordinates(aether, coordinate, config.getDefaultVersion()); - final ServiceLoader serviceLoader = ServiceLoader.load(clazz, loader); - - for (T module : serviceLoader) { - log.info("Adding extension module[%s] for class[%s]", module.getClass(), clazz.getName()); - retVal.add(module); + for (T module : ServiceLoader.load(clazz, loader)) { + String moduleName = module.getClass().getCanonicalName(); + if (moduleName == null) { + log.warn( + "Extension module [%s] was ignored because it doesn't have a canonical name, is it a local or anonymous class?", + module.getClass().getName() + ); + } else if (!extensionNames.contains(moduleName)) { + log.info("Adding remote extension module[%s] for class[%s]", moduleName, clazz.getName()); + extensionNames.add(moduleName); + retVal.add(module); + } } } catch (Exception e) { @@ -152,6 +171,22 @@ public class Initialization } } + if (config.searchCurrentClassloader()) { + for (T module : ServiceLoader.load(clazz, Initialization.class.getClassLoader())) { + String moduleName = module.getClass().getCanonicalName(); + if (moduleName == null) { + log.warn( + "Extension module [%s] was ignored because it doesn't have a canonical name, is it a local or anonymous class?", + module.getClass().getName() + ); + } else if (!extensionNames.contains(moduleName)) { + log.info("Adding local extension module[%s] for class[%s]", moduleName, clazz.getName()); + extensionNames.add(moduleName); + retVal.add(module); + } + } + } + // update the map with currently loaded modules extensionsMap.put(clazz, retVal); diff --git a/server/src/test/java/io/druid/initialization/InitializationTest.java b/server/src/test/java/io/druid/initialization/InitializationTest.java index 5b0fe60da9e..7f58808d589 100644 --- a/server/src/test/java/io/druid/initialization/InitializationTest.java +++ b/server/src/test/java/io/druid/initialization/InitializationTest.java @@ -19,6 +19,7 @@ package io.druid.initialization; import com.fasterxml.jackson.databind.Module; import com.fasterxml.jackson.databind.ObjectMapper; +import com.google.api.client.util.Sets; import com.google.common.base.Function; import com.google.common.collect.Collections2; import com.google.common.collect.ImmutableList; @@ -36,6 +37,7 @@ import org.junit.Test; import org.junit.runners.MethodSorters; import javax.annotation.Nullable; +import java.net.URLClassLoader; import java.util.Collection; import java.util.List; import java.util.Set; @@ -68,14 +70,14 @@ public class InitializationTest Injector startupInjector = GuiceInjectors.makeStartupInjector(); Function fnClassName = new Function() - { - @Nullable - @Override - public String apply(@Nullable DruidModule input) - { - return input.getClass().getCanonicalName(); - } - }; + { + @Nullable + @Override + public String apply(@Nullable DruidModule input) + { + return input.getClass().getCanonicalName(); + } + }; Assert.assertFalse( "modules does not contain TestDruidModule", @@ -96,7 +98,38 @@ public class InitializationTest } @Test - public void test04MakeInjectorWithModules() throws Exception + public void test04DuplicateClassLoaderExtensions() throws Exception + { + Initialization.getLoadersMap().put("xyz", (URLClassLoader) Initialization.class.getClassLoader()); + + Collection modules = Initialization.getFromExtensions( + new ExtensionsConfig() + { + @Override + public List getCoordinates() + { + return ImmutableList.of("xyz"); + } + + @Override + public List getRemoteRepositories() + { + return ImmutableList.of(); + } + }, DruidModule.class + ); + + Set loadedModuleNames = Sets.newHashSet(); + for (DruidModule module : modules) { + Assert.assertFalse("Duplicate extensions are loaded", loadedModuleNames.contains(module.getClass().getName())); + loadedModuleNames.add(module.getClass().getName()); + } + + Initialization.getLoadersMap().clear(); + } + + @Test + public void test05MakeInjectorWithModules() throws Exception { Injector startupInjector = GuiceInjectors.makeStartupInjector(); Injector injector = Initialization.makeInjectorWithModules(