mirror of https://github.com/apache/lucene.git
SOLR-13822: Isolated Classloading from packages (#957)
SOLR-13822: A Package management system with the following features. A packages.json in ZK to store the configuration, APIs to read/edit them and isolated classloaders to load the classes from those packages if the 'class' attribute is prefixed with `<package-name>:`
This commit is contained in:
parent
3ae8204248
commit
98f08d39aa
|
@ -100,7 +100,11 @@ Upgrade Notes
|
||||||
|
|
||||||
New Features
|
New Features
|
||||||
---------------------
|
---------------------
|
||||||
(No changes)
|
* SOLR-13821: A Package store to store and load package artifacts (noble, Ishan Chattopadhyaya)
|
||||||
|
|
||||||
|
* SOLR-13822: A Package management system with the following features. A packages.json in ZK to store
|
||||||
|
the configuration, APIs to read/edit them and isolated classloaders to load the classes from
|
||||||
|
hose packages if the 'class' attribute is prefixed with `<package-name>:` (noble, Ishan Chattopadhyaya)
|
||||||
|
|
||||||
Improvements
|
Improvements
|
||||||
---------------------
|
---------------------
|
||||||
|
@ -194,8 +198,6 @@ New Features
|
||||||
|
|
||||||
* SOLR-8241: Add CaffeineCache, an efficient implementation of SolrCache.(Ben Manes, Shawn Heisey, David Smiley, Andrzej Bialecki)
|
* SOLR-8241: Add CaffeineCache, an efficient implementation of SolrCache.(Ben Manes, Shawn Heisey, David Smiley, Andrzej Bialecki)
|
||||||
|
|
||||||
* SOLR-13821: A Package store to store and load package artefacts (noble, Ishan Chattopadhyaya)
|
|
||||||
|
|
||||||
* SOLR-13298: Allow zplot to plot matrices (Joel Bernstein)
|
* SOLR-13298: Allow zplot to plot matrices (Joel Bernstein)
|
||||||
|
|
||||||
Improvements
|
Improvements
|
||||||
|
|
|
@ -227,6 +227,7 @@ public class AnnotatedApi extends Api implements PermissionNameProvider {
|
||||||
}
|
}
|
||||||
if (isWrappedInPayloadObj) {
|
if (isWrappedInPayloadObj) {
|
||||||
PayloadObj<Object> payloadObj = new PayloadObj<>(cmd.name, cmd.getCommandData(), o);
|
PayloadObj<Object> payloadObj = new PayloadObj<>(cmd.name, cmd.getCommandData(), o);
|
||||||
|
cmd = payloadObj;
|
||||||
method.invoke(obj, req, rsp, payloadObj);
|
method.invoke(obj, req, rsp, payloadObj);
|
||||||
} else {
|
} else {
|
||||||
method.invoke(obj, req, rsp, o);
|
method.invoke(obj, req, rsp, o);
|
||||||
|
@ -239,10 +240,13 @@ public class AnnotatedApi extends Api implements PermissionNameProvider {
|
||||||
|
|
||||||
|
|
||||||
} catch (SolrException se) {
|
} catch (SolrException se) {
|
||||||
|
log.error("Error executing command ", se);
|
||||||
throw se;
|
throw se;
|
||||||
} catch (InvocationTargetException ite) {
|
} catch (InvocationTargetException ite) {
|
||||||
|
log.error("Error executing command ", ite);
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, ite.getCause());
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, ite.getCause());
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
log.error("Error executing command : ", e);
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, e);
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -101,6 +101,7 @@ import org.apache.solr.metrics.SolrCoreMetricManager;
|
||||||
import org.apache.solr.metrics.SolrMetricManager;
|
import org.apache.solr.metrics.SolrMetricManager;
|
||||||
import org.apache.solr.metrics.SolrMetricProducer;
|
import org.apache.solr.metrics.SolrMetricProducer;
|
||||||
import org.apache.solr.metrics.SolrMetricsContext;
|
import org.apache.solr.metrics.SolrMetricsContext;
|
||||||
|
import org.apache.solr.pkg.PackageLoader;
|
||||||
import org.apache.solr.request.SolrRequestHandler;
|
import org.apache.solr.request.SolrRequestHandler;
|
||||||
import org.apache.solr.request.SolrRequestInfo;
|
import org.apache.solr.request.SolrRequestInfo;
|
||||||
import org.apache.solr.search.SolrFieldCacheBean;
|
import org.apache.solr.search.SolrFieldCacheBean;
|
||||||
|
@ -224,6 +225,7 @@ public class CoreContainer {
|
||||||
protected volatile AutoscalingHistoryHandler autoscalingHistoryHandler;
|
protected volatile AutoscalingHistoryHandler autoscalingHistoryHandler;
|
||||||
|
|
||||||
private PackageStoreAPI packageStoreAPI;
|
private PackageStoreAPI packageStoreAPI;
|
||||||
|
private PackageLoader packageLoader;
|
||||||
|
|
||||||
|
|
||||||
// Bits for the state variable.
|
// Bits for the state variable.
|
||||||
|
@ -582,6 +584,10 @@ public class CoreContainer {
|
||||||
return replayUpdatesExecutor;
|
return replayUpdatesExecutor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PackageLoader getPackageLoader() {
|
||||||
|
return packageLoader;
|
||||||
|
}
|
||||||
|
|
||||||
public PackageStoreAPI getPackageStoreAPI() {
|
public PackageStoreAPI getPackageStoreAPI() {
|
||||||
return packageStoreAPI;
|
return packageStoreAPI;
|
||||||
}
|
}
|
||||||
|
@ -736,8 +742,12 @@ public class CoreContainer {
|
||||||
|
|
||||||
if (isZooKeeperAware()) {
|
if (isZooKeeperAware()) {
|
||||||
metricManager.loadClusterReporters(metricReporters, this);
|
metricManager.loadClusterReporters(metricReporters, this);
|
||||||
|
packageLoader = new PackageLoader(this);
|
||||||
|
containerHandlers.getApiBag().register(new AnnotatedApi(packageLoader.getPackageAPI().editAPI), Collections.EMPTY_MAP);
|
||||||
|
containerHandlers.getApiBag().register(new AnnotatedApi(packageLoader.getPackageAPI().readAPI), Collections.EMPTY_MAP);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// setup executor to load cores in parallel
|
// setup executor to load cores in parallel
|
||||||
ExecutorService coreLoadExecutor = MetricUtils.instrumentedExecutorService(
|
ExecutorService coreLoadExecutor = MetricUtils.instrumentedExecutorService(
|
||||||
ExecutorUtil.newMDCAwareFixedThreadPool(
|
ExecutorUtil.newMDCAwareFixedThreadPool(
|
||||||
|
|
|
@ -29,6 +29,7 @@ import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.function.Supplier;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
import java.util.zip.ZipEntry;
|
import java.util.zip.ZipEntry;
|
||||||
import java.util.zip.ZipInputStream;
|
import java.util.zip.ZipInputStream;
|
||||||
|
@ -43,8 +44,10 @@ import org.apache.solr.common.SolrException;
|
||||||
import org.apache.solr.common.util.StrUtils;
|
import org.apache.solr.common.util.StrUtils;
|
||||||
import org.apache.solr.handler.RequestHandlerBase;
|
import org.apache.solr.handler.RequestHandlerBase;
|
||||||
import org.apache.solr.handler.component.SearchComponent;
|
import org.apache.solr.handler.component.SearchComponent;
|
||||||
|
import org.apache.solr.pkg.PackagePluginHolder;
|
||||||
import org.apache.solr.request.SolrRequestHandler;
|
import org.apache.solr.request.SolrRequestHandler;
|
||||||
import org.apache.solr.update.processor.UpdateRequestProcessorChain;
|
import org.apache.solr.update.processor.UpdateRequestProcessorChain;
|
||||||
|
import org.apache.solr.update.processor.UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder;
|
||||||
import org.apache.solr.update.processor.UpdateRequestProcessorFactory;
|
import org.apache.solr.update.processor.UpdateRequestProcessorFactory;
|
||||||
import org.apache.solr.util.CryptoKeys;
|
import org.apache.solr.util.CryptoKeys;
|
||||||
import org.apache.solr.util.SimplePostTool;
|
import org.apache.solr.util.SimplePostTool;
|
||||||
|
@ -97,7 +100,7 @@ public class PluginBag<T> implements AutoCloseable {
|
||||||
this(klass, core, false);
|
this(klass, core, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void initInstance(Object inst, PluginInfo info) {
|
public static void initInstance(Object inst, PluginInfo info) {
|
||||||
if (inst instanceof PluginInfoInitialized) {
|
if (inst instanceof PluginInfoInitialized) {
|
||||||
((PluginInfoInitialized) inst).init(info);
|
((PluginInfoInitialized) inst).init(info);
|
||||||
} else if (inst instanceof NamedListInitializedPlugin) {
|
} else if (inst instanceof NamedListInitializedPlugin) {
|
||||||
|
@ -138,13 +141,22 @@ public class PluginBag<T> implements AutoCloseable {
|
||||||
log.debug("{} : '{}' created with startup=lazy ", meta.getCleanTag(), info.name);
|
log.debug("{} : '{}' created with startup=lazy ", meta.getCleanTag(), info.name);
|
||||||
return new LazyPluginHolder<T>(meta, info, core, core.getResourceLoader(), false);
|
return new LazyPluginHolder<T>(meta, info, core, core.getResourceLoader(), false);
|
||||||
} else {
|
} else {
|
||||||
T inst = core.createInstance(info.className, (Class<T>) meta.clazz, meta.getCleanTag(), null, core.getResourceLoader());
|
if (info.pkgName != null) {
|
||||||
|
PackagePluginHolder<T> holder = new PackagePluginHolder<>(info, core, meta);
|
||||||
|
return meta.clazz == UpdateRequestProcessorFactory.class ?
|
||||||
|
new PluginHolder(info, new LazyUpdateProcessorFactoryHolder(holder)) :
|
||||||
|
holder;
|
||||||
|
} else {
|
||||||
|
T inst = core.createInstance(info.className, (Class<T>) meta.clazz, meta.getCleanTag(), null, core.getResourceLoader(info.pkgName));
|
||||||
initInstance(inst, info);
|
initInstance(inst, info);
|
||||||
return new PluginHolder<>(info, inst);
|
return new PluginHolder<>(info, inst);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** make a plugin available in an alternate name. This is an internal API and not for public use
|
/**
|
||||||
|
* make a plugin available in an alternate name. This is an internal API and not for public use
|
||||||
|
*
|
||||||
* @param src key in which the plugin is already registered
|
* @param src key in which the plugin is already registered
|
||||||
* @param target the new key in which the plugin should be aliased to. If target exists already, the alias fails
|
* @param target the new key in which the plugin should be aliased to. If target exists already, the alias fails
|
||||||
* @return flag if the operation is successful or not
|
* @return flag if the operation is successful or not
|
||||||
|
@ -340,8 +352,8 @@ public class PluginBag<T> implements AutoCloseable {
|
||||||
* An indirect reference to a plugin. It just wraps a plugin instance.
|
* An indirect reference to a plugin. It just wraps a plugin instance.
|
||||||
* subclasses may choose to lazily load the plugin
|
* subclasses may choose to lazily load the plugin
|
||||||
*/
|
*/
|
||||||
public static class PluginHolder<T> implements AutoCloseable {
|
public static class PluginHolder<T> implements Supplier<T>, AutoCloseable {
|
||||||
private T inst;
|
protected T inst;
|
||||||
protected final PluginInfo pluginInfo;
|
protected final PluginInfo pluginInfo;
|
||||||
boolean registerAPI = false;
|
boolean registerAPI = false;
|
||||||
|
|
||||||
|
|
|
@ -16,14 +16,20 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.core;
|
package org.apache.solr.core;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
import org.apache.solr.common.MapSerializable;
|
import org.apache.solr.common.MapSerializable;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
|
import org.apache.solr.common.util.Pair;
|
||||||
import org.apache.solr.util.DOMUtil;
|
import org.apache.solr.util.DOMUtil;
|
||||||
import org.w3c.dom.Node;
|
import org.w3c.dom.Node;
|
||||||
import org.w3c.dom.NodeList;
|
import org.w3c.dom.NodeList;
|
||||||
|
|
||||||
import java.util.*;
|
|
||||||
|
|
||||||
import static java.util.Arrays.asList;
|
import static java.util.Arrays.asList;
|
||||||
import static java.util.Collections.unmodifiableList;
|
import static java.util.Collections.unmodifiableList;
|
||||||
import static java.util.Collections.unmodifiableMap;
|
import static java.util.Collections.unmodifiableMap;
|
||||||
|
@ -35,27 +41,51 @@ import static org.apache.solr.schema.FieldType.CLASS_NAME;
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class PluginInfo implements MapSerializable {
|
public class PluginInfo implements MapSerializable {
|
||||||
public final String name, className, type;
|
public final String name, className, type, pkgName;
|
||||||
public final NamedList initArgs;
|
public final NamedList initArgs;
|
||||||
public final Map<String, String> attributes;
|
public final Map<String, String> attributes;
|
||||||
public final List<PluginInfo> children;
|
public final List<PluginInfo> children;
|
||||||
private boolean isFromSolrConfig;
|
private boolean isFromSolrConfig;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public PluginInfo(String type, Map<String, String> attrs, NamedList initArgs, List<PluginInfo> children) {
|
public PluginInfo(String type, Map<String, String> attrs, NamedList initArgs, List<PluginInfo> children) {
|
||||||
this.type = type;
|
this.type = type;
|
||||||
this.name = attrs.get(NAME);
|
this.name = attrs.get(NAME);
|
||||||
this.className = attrs.get(CLASS_NAME);
|
Pair<String, String> parsed = parseClassName(attrs.get(CLASS_NAME));
|
||||||
|
this.className = parsed.second();
|
||||||
|
this.pkgName = parsed.first();
|
||||||
this.initArgs = initArgs;
|
this.initArgs = initArgs;
|
||||||
attributes = unmodifiableMap(attrs);
|
attributes = unmodifiableMap(attrs);
|
||||||
this.children = children == null ? Collections.<PluginInfo>emptyList(): unmodifiableList(children);
|
this.children = children == null ? Collections.<PluginInfo>emptyList(): unmodifiableList(children);
|
||||||
isFromSolrConfig = false;
|
isFromSolrConfig = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** class names can be prefixed with package name e.g: my_package:my.pkg.Class
|
||||||
|
* This checks if it is a package name prefixed classname.
|
||||||
|
* the return value has first = package name & second = class name
|
||||||
|
*/
|
||||||
|
static Pair<String,String > parseClassName(String name) {
|
||||||
|
String pkgName = null;
|
||||||
|
String className = name;
|
||||||
|
if (name != null) {
|
||||||
|
int colonIdx = name.indexOf(':');
|
||||||
|
if (colonIdx > -1) {
|
||||||
|
pkgName = name.substring(0, colonIdx);
|
||||||
|
className = name.substring(colonIdx + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new Pair<>(pkgName, className);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public PluginInfo(Node node, String err, boolean requireName, boolean requireClass) {
|
public PluginInfo(Node node, String err, boolean requireName, boolean requireClass) {
|
||||||
type = node.getNodeName();
|
type = node.getNodeName();
|
||||||
name = DOMUtil.getAttr(node, NAME, requireName ? err : null);
|
name = DOMUtil.getAttr(node, NAME, requireName ? err : null);
|
||||||
className = DOMUtil.getAttr(node, CLASS_NAME, requireClass ? err : null);
|
Pair<String, String> parsed = parseClassName(DOMUtil.getAttr(node, CLASS_NAME, requireClass ? err : null));
|
||||||
|
className = parsed.second();
|
||||||
|
pkgName = parsed.first();
|
||||||
initArgs = DOMUtil.childNodesToNamedList(node);
|
initArgs = DOMUtil.childNodesToNamedList(node);
|
||||||
attributes = unmodifiableMap(DOMUtil.toMap(node.getAttributes()));
|
attributes = unmodifiableMap(DOMUtil.toMap(node.getAttributes()));
|
||||||
children = loadSubPlugins(node);
|
children = loadSubPlugins(node);
|
||||||
|
@ -85,7 +115,9 @@ public class PluginInfo implements MapSerializable {
|
||||||
}
|
}
|
||||||
this.type = type;
|
this.type = type;
|
||||||
this.name = (String) m.get(NAME);
|
this.name = (String) m.get(NAME);
|
||||||
this.className = (String) m.get(CLASS_NAME);
|
Pair<String, String> parsed = parseClassName((String) m.get(CLASS_NAME));
|
||||||
|
this.className = parsed.second();
|
||||||
|
this.pkgName = parsed.first();
|
||||||
attributes = unmodifiableMap(m);
|
attributes = unmodifiableMap(m);
|
||||||
this.children = Collections.<PluginInfo>emptyList();
|
this.children = Collections.<PluginInfo>emptyList();
|
||||||
isFromSolrConfig = true;
|
isFromSolrConfig = true;
|
||||||
|
|
|
@ -250,9 +250,18 @@ public class RequestParams implements MapSerializable {
|
||||||
return m1;
|
return m1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param type one of defaults, appends, invariants
|
||||||
|
*/
|
||||||
public VersionedParams getParams(String type) {
|
public VersionedParams getParams(String type) {
|
||||||
return paramsMap.get(type);
|
return paramsMap.get(type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**get the raw map
|
||||||
|
*/
|
||||||
|
public Map<String, Object> get() {
|
||||||
|
return defaults;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class VersionedParams extends MapSolrParams {
|
public static class VersionedParams extends MapSolrParams {
|
||||||
|
|
|
@ -110,6 +110,8 @@ import org.apache.solr.logging.MDCLoggingContext;
|
||||||
import org.apache.solr.metrics.SolrCoreMetricManager;
|
import org.apache.solr.metrics.SolrCoreMetricManager;
|
||||||
import org.apache.solr.metrics.SolrMetricProducer;
|
import org.apache.solr.metrics.SolrMetricProducer;
|
||||||
import org.apache.solr.metrics.SolrMetricsContext;
|
import org.apache.solr.metrics.SolrMetricsContext;
|
||||||
|
import org.apache.solr.pkg.PackageListeners;
|
||||||
|
import org.apache.solr.pkg.PackageLoader;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.request.SolrRequestHandler;
|
import org.apache.solr.request.SolrRequestHandler;
|
||||||
import org.apache.solr.response.BinaryResponseWriter;
|
import org.apache.solr.response.BinaryResponseWriter;
|
||||||
|
@ -238,6 +240,8 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
|
||||||
public volatile boolean indexEnabled = true;
|
public volatile boolean indexEnabled = true;
|
||||||
public volatile boolean readOnly = false;
|
public volatile boolean readOnly = false;
|
||||||
|
|
||||||
|
private PackageListeners packageListeners = new PackageListeners(this);
|
||||||
|
|
||||||
public Set<String> getMetricNames() {
|
public Set<String> getMetricNames() {
|
||||||
return metricNames;
|
return metricNames;
|
||||||
}
|
}
|
||||||
|
@ -262,6 +266,10 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
|
||||||
return restManager;
|
return restManager;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public PackageListeners getPackageListeners() {
|
||||||
|
return packageListeners;
|
||||||
|
}
|
||||||
|
|
||||||
static int boolean_query_max_clause_count = Integer.MIN_VALUE;
|
static int boolean_query_max_clause_count = Integer.MIN_VALUE;
|
||||||
|
|
||||||
private ExecutorService coreAsyncTaskExecutor = ExecutorUtil.newMDCAwareCachedThreadPool("Core Async Task");
|
private ExecutorService coreAsyncTaskExecutor = ExecutorUtil.newMDCAwareCachedThreadPool("Core Async Task");
|
||||||
|
@ -275,6 +283,18 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
|
||||||
return resourceLoader;
|
return resourceLoader;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Gets the SolrResourceLoader for a given package
|
||||||
|
* @param pkg The package name
|
||||||
|
*/
|
||||||
|
public SolrResourceLoader getResourceLoader(String pkg) {
|
||||||
|
if (pkg == null) {
|
||||||
|
return resourceLoader;
|
||||||
|
}
|
||||||
|
PackageLoader.Package aPackage = coreContainer.getPackageLoader().getPackage(pkg);
|
||||||
|
PackageLoader.Package.Version latest = aPackage.getLatest();
|
||||||
|
return latest.getLoader();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the configuration resource name used by this core instance.
|
* Gets the configuration resource name used by this core instance.
|
||||||
*
|
*
|
||||||
|
@ -857,7 +877,7 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
|
||||||
|
|
||||||
public <T extends Object> T createInitInstance(PluginInfo info, Class<T> cast, String msg, String defClassName) {
|
public <T extends Object> T createInitInstance(PluginInfo info, Class<T> cast, String msg, String defClassName) {
|
||||||
if (info == null) return null;
|
if (info == null) return null;
|
||||||
T o = createInstance(info.className == null ? defClassName : info.className, cast, msg, this, getResourceLoader());
|
T o = createInstance(info.className == null ? defClassName : info.className, cast, msg, this, getResourceLoader(info.pkgName));
|
||||||
if (o instanceof PluginInfoInitialized) {
|
if (o instanceof PluginInfoInitialized) {
|
||||||
((PluginInfoInitialized) o).init(info);
|
((PluginInfoInitialized) o).init(info);
|
||||||
} else if (o instanceof NamedListInitializedPlugin) {
|
} else if (o instanceof NamedListInitializedPlugin) {
|
||||||
|
|
|
@ -28,6 +28,7 @@ import java.io.InputStream;
|
||||||
import java.io.OutputStream;
|
import java.io.OutputStream;
|
||||||
import java.lang.invoke.MethodHandles;
|
import java.lang.invoke.MethodHandles;
|
||||||
import java.lang.reflect.Constructor;
|
import java.lang.reflect.Constructor;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.net.URLClassLoader;
|
import java.net.URLClassLoader;
|
||||||
import java.nio.charset.CharacterCodingException;
|
import java.nio.charset.CharacterCodingException;
|
||||||
|
@ -82,8 +83,7 @@ import org.slf4j.LoggerFactory;
|
||||||
/**
|
/**
|
||||||
* @since solr 1.3
|
* @since solr 1.3
|
||||||
*/
|
*/
|
||||||
public class SolrResourceLoader implements ResourceLoader,Closeable
|
public class SolrResourceLoader implements ResourceLoader, Closeable {
|
||||||
{
|
|
||||||
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
static final String project = "solr";
|
static final String project = "solr";
|
||||||
|
@ -96,7 +96,10 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
};
|
};
|
||||||
private static final java.lang.String SOLR_CORE_NAME = "solr.core.name";
|
private static final java.lang.String SOLR_CORE_NAME = "solr.core.name";
|
||||||
private static Set<String> loggedOnce = new ConcurrentSkipListSet<>();
|
private static Set<String> loggedOnce = new ConcurrentSkipListSet<>();
|
||||||
|
private static final Charset UTF_8 = StandardCharsets.UTF_8;
|
||||||
|
|
||||||
|
|
||||||
|
private String name = "";
|
||||||
protected URLClassLoader classLoader;
|
protected URLClassLoader classLoader;
|
||||||
private final Path instanceDir;
|
private final Path instanceDir;
|
||||||
private String dataDir;
|
private String dataDir;
|
||||||
|
@ -104,7 +107,6 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
private final List<SolrCoreAware> waitingForCore = Collections.synchronizedList(new ArrayList<SolrCoreAware>());
|
private final List<SolrCoreAware> waitingForCore = Collections.synchronizedList(new ArrayList<SolrCoreAware>());
|
||||||
private final List<SolrInfoBean> infoMBeans = Collections.synchronizedList(new ArrayList<SolrInfoBean>());
|
private final List<SolrInfoBean> infoMBeans = Collections.synchronizedList(new ArrayList<SolrInfoBean>());
|
||||||
private final List<ResourceLoaderAware> waitingForResources = Collections.synchronizedList(new ArrayList<ResourceLoaderAware>());
|
private final List<ResourceLoaderAware> waitingForResources = Collections.synchronizedList(new ArrayList<ResourceLoaderAware>());
|
||||||
private static final Charset UTF_8 = StandardCharsets.UTF_8;
|
|
||||||
|
|
||||||
private final Properties coreProperties;
|
private final Properties coreProperties;
|
||||||
|
|
||||||
|
@ -134,11 +136,20 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
* found in the "lib/" directory in the specified instance directory.
|
* found in the "lib/" directory in the specified instance directory.
|
||||||
* If the instance directory is not specified (=null), SolrResourceLoader#locateInstanceDir will provide one.
|
* If the instance directory is not specified (=null), SolrResourceLoader#locateInstanceDir will provide one.
|
||||||
*/
|
*/
|
||||||
public SolrResourceLoader(Path instanceDir, ClassLoader parent)
|
public SolrResourceLoader(Path instanceDir, ClassLoader parent) {
|
||||||
{
|
|
||||||
this(instanceDir, parent, null);
|
this(instanceDir, parent, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public SolrResourceLoader(String name, List<Path> classpath, Path instanceDir, ClassLoader parent) throws MalformedURLException {
|
||||||
|
this(instanceDir, parent);
|
||||||
|
this.name = name;
|
||||||
|
for (Path path : classpath) {
|
||||||
|
addToClassLoader(path.toUri().normalize().toURL());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
public SolrResourceLoader(Path instanceDir) {
|
public SolrResourceLoader(Path instanceDir) {
|
||||||
this(instanceDir, null, null);
|
this(instanceDir, null, null);
|
||||||
}
|
}
|
||||||
|
@ -264,6 +275,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility method to get the URLs of all paths under a given directory that match a filter
|
* Utility method to get the URLs of all paths under a given directory that match a filter
|
||||||
|
*
|
||||||
* @param libDir the root directory
|
* @param libDir the root directory
|
||||||
* @param filter the filter
|
* @param filter the filter
|
||||||
* @return all matching URLs
|
* @return all matching URLs
|
||||||
|
@ -296,6 +308,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Utility method to get the URLs of all paths under a given directory that match a regex
|
* Utility method to get the URLs of all paths under a given directory that match a regex
|
||||||
|
*
|
||||||
* @param libDir the root directory
|
* @param libDir the root directory
|
||||||
* @param regex the regex as a String
|
* @param regex the regex as a String
|
||||||
* @return all matching URLs
|
* @return all matching URLs
|
||||||
|
@ -311,7 +324,9 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Ensures a directory name always ends with a '/'. */
|
/**
|
||||||
|
* Ensures a directory name always ends with a '/'.
|
||||||
|
*/
|
||||||
public static String normalizeDir(String path) {
|
public static String normalizeDir(String path) {
|
||||||
return (path != null && (!(path.endsWith("/") || path.endsWith("\\")))) ? path + File.separator : path;
|
return (path != null && (!(path.endsWith("/") || path.endsWith("\\")))) ? path + File.separator : path;
|
||||||
}
|
}
|
||||||
|
@ -341,22 +356,27 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
* EXPERT
|
* EXPERT
|
||||||
* <p>
|
* <p>
|
||||||
* The underlying class loader. Most applications will not need to use this.
|
* The underlying class loader. Most applications will not need to use this.
|
||||||
|
*
|
||||||
* @return The {@link ClassLoader}
|
* @return The {@link ClassLoader}
|
||||||
*/
|
*/
|
||||||
public ClassLoader getClassLoader() {
|
public ClassLoader getClassLoader() {
|
||||||
return classLoader;
|
return classLoader;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Opens a schema resource by its name.
|
/**
|
||||||
|
* Opens a schema resource by its name.
|
||||||
* Override this method to customize loading schema resources.
|
* Override this method to customize loading schema resources.
|
||||||
|
*
|
||||||
* @return the stream for the named schema
|
* @return the stream for the named schema
|
||||||
*/
|
*/
|
||||||
public InputStream openSchema(String name) throws IOException {
|
public InputStream openSchema(String name) throws IOException {
|
||||||
return openResource(name);
|
return openResource(name);
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Opens a config resource by its name.
|
/**
|
||||||
|
* Opens a config resource by its name.
|
||||||
* Override this method to customize loading config resources.
|
* Override this method to customize loading config resources.
|
||||||
|
*
|
||||||
* @return the stream for the named configuration
|
* @return the stream for the named configuration
|
||||||
*/
|
*/
|
||||||
public InputStream openConfig(String name) throws IOException {
|
public InputStream openConfig(String name) throws IOException {
|
||||||
|
@ -373,12 +393,14 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
"; set -Dsolr.allow.unsafe.resourceloading=true to allow unsafe loading");
|
"; set -Dsolr.allow.unsafe.resourceloading=true to allow unsafe loading");
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Opens any resource by its name.
|
/**
|
||||||
|
* Opens any resource by its name.
|
||||||
* By default, this will look in multiple locations to load the resource:
|
* By default, this will look in multiple locations to load the resource:
|
||||||
* $configDir/$resource (if resource is not absolute)
|
* $configDir/$resource (if resource is not absolute)
|
||||||
* $CWD/$resource
|
* $CWD/$resource
|
||||||
* otherwise, it will look for it in any jar accessible through the class loader.
|
* otherwise, it will look for it in any jar accessible through the class loader.
|
||||||
* Override this method to customize loading resources.
|
* Override this method to customize loading resources.
|
||||||
|
*
|
||||||
* @return the stream for the named resource
|
* @return the stream for the named resource
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
|
@ -514,7 +536,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
return Class.forName(c, true, classLoader).asSubclass(expectedType);
|
return Class.forName(c, true, classLoader).asSubclass(expectedType);
|
||||||
} catch (ClassNotFoundException | ClassCastException e) {
|
} catch (ClassNotFoundException | ClassCastException e) {
|
||||||
// this can happen if the legacyAnalysisPattern below caches the wrong thing
|
// this can happen if the legacyAnalysisPattern below caches the wrong thing
|
||||||
log.warn("Unable to load cached class, attempting lookup. name={} shortname={} reason={}", c, cname, e);
|
log.warn( name + " Unable to load cached class, attempting lookup. name={} shortname={} reason={}", c, cname, e);
|
||||||
classNameCache.remove(cname);
|
classNameCache.remove(cname);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -560,7 +582,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new SolrException( SolrException.ErrorCode.SERVER_ERROR, "Error loading class '" + cname + "'", e);
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, name +" Error loading class '" + cname + "'", e);
|
||||||
}
|
}
|
||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -608,8 +630,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
try {
|
try {
|
||||||
Constructor<? extends CoreAdminHandler> ctor = clazz.getConstructor(CoreContainer.class);
|
Constructor<? extends CoreAdminHandler> ctor = clazz.getConstructor(CoreContainer.class);
|
||||||
obj = ctor.newInstance(coreContainer);
|
obj = ctor.newInstance(coreContainer);
|
||||||
}
|
} catch (Exception e) {
|
||||||
catch (Exception e) {
|
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
||||||
"Error instantiating class: '" + clazz.getName() + "'", e);
|
"Error instantiating class: '" + clazz.getName() + "'", e);
|
||||||
}
|
}
|
||||||
|
@ -627,7 +648,6 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
public <T> T newInstance(String cName, Class<T> expectedType, String[] subPackages, Class[] params, Object[] args) {
|
public <T> T newInstance(String cName, Class<T> expectedType, String[] subPackages, Class[] params, Object[] args) {
|
||||||
Class<? extends T> clazz = findClass(cName, expectedType, subPackages);
|
Class<? extends T> clazz = findClass(cName, expectedType, subPackages);
|
||||||
if (clazz == null) {
|
if (clazz == null) {
|
||||||
|
@ -684,8 +704,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
/**
|
/**
|
||||||
* Tell all {@link SolrCoreAware} instances about the SolrCore
|
* Tell all {@link SolrCoreAware} instances about the SolrCore
|
||||||
*/
|
*/
|
||||||
public void inform(SolrCore core)
|
public void inform(SolrCore core) {
|
||||||
{
|
|
||||||
this.dataDir = core.getDataDir();
|
this.dataDir = core.getDataDir();
|
||||||
|
|
||||||
// make a copy to avoid potential deadlock of a callback calling newInstance and trying to
|
// make a copy to avoid potential deadlock of a callback calling newInstance and trying to
|
||||||
|
@ -710,8 +729,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
/**
|
/**
|
||||||
* Tell all {@link ResourceLoaderAware} instances about the loader
|
* Tell all {@link ResourceLoaderAware} instances about the loader
|
||||||
*/
|
*/
|
||||||
public void inform( ResourceLoader loader ) throws IOException
|
public void inform(ResourceLoader loader) throws IOException {
|
||||||
{
|
|
||||||
|
|
||||||
// make a copy to avoid potential deadlock of a callback adding to the list
|
// make a copy to avoid potential deadlock of a callback adding to the list
|
||||||
ResourceLoaderAware[] arr;
|
ResourceLoaderAware[] arr;
|
||||||
|
@ -730,6 +748,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register any {@link SolrInfoBean}s
|
* Register any {@link SolrInfoBean}s
|
||||||
|
*
|
||||||
* @param infoRegistry The Info Registry
|
* @param infoRegistry The Info Registry
|
||||||
*/
|
*/
|
||||||
public void inform(Map<String, SolrInfoBean> infoRegistry) {
|
public void inform(Map<String, SolrInfoBean> infoRegistry) {
|
||||||
|
@ -769,8 +788,9 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
* <li>The system property solr.solr.home</li>
|
* <li>The system property solr.solr.home</li>
|
||||||
* <li>Look in the current working directory for a solr/ directory</li>
|
* <li>Look in the current working directory for a solr/ directory</li>
|
||||||
* </ol>
|
* </ol>
|
||||||
*
|
* <p>
|
||||||
* The return value is normalized. Normalization essentially means it ends in a trailing slash.
|
* The return value is normalized. Normalization essentially means it ends in a trailing slash.
|
||||||
|
*
|
||||||
* @return A normalized solrhome
|
* @return A normalized solrhome
|
||||||
* @see #normalizeDir(String)
|
* @see #normalizeDir(String)
|
||||||
*/
|
*/
|
||||||
|
@ -809,11 +829,12 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Solr allows users to store arbitrary files in a special directory located directly under SOLR_HOME.
|
* Solr allows users to store arbitrary files in a special directory located directly under SOLR_HOME.
|
||||||
*
|
* <p>
|
||||||
* This directory is generally created by each node on startup. Files located in this directory can then be
|
* This directory is generally created by each node on startup. Files located in this directory can then be
|
||||||
* manipulated using select Solr features (e.g. streaming expressions).
|
* manipulated using select Solr features (e.g. streaming expressions).
|
||||||
*/
|
*/
|
||||||
public static final String USER_FILES_DIRECTORY = "userfiles";
|
public static final String USER_FILES_DIRECTORY = "userfiles";
|
||||||
|
|
||||||
public static void ensureUserFilesDataDir(Path solrHome) {
|
public static void ensureUserFilesDataDir(Path solrHome) {
|
||||||
final Path userFilesPath = getUserFilesPath(solrHome);
|
final Path userFilesPath = getUserFilesPath(solrHome);
|
||||||
final File userFilesDirectory = new File(userFilesPath.toString());
|
final File userFilesDirectory = new File(userFilesPath.toString());
|
||||||
|
@ -852,6 +873,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
* Keep a list of classes that are allowed to implement each 'Aware' interface
|
* Keep a list of classes that are allowed to implement each 'Aware' interface
|
||||||
*/
|
*/
|
||||||
private static final Map<Class, Class[]> awareCompatibility;
|
private static final Map<Class, Class[]> awareCompatibility;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
awareCompatibility = new HashMap<>();
|
awareCompatibility = new HashMap<>();
|
||||||
awareCompatibility.put(
|
awareCompatibility.put(
|
||||||
|
@ -886,8 +908,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
/**
|
/**
|
||||||
* Utility function to throw an exception if the class is invalid
|
* Utility function to throw an exception if the class is invalid
|
||||||
*/
|
*/
|
||||||
static void assertAwareCompatibility( Class aware, Object obj )
|
static void assertAwareCompatibility(Class aware, Object obj) {
|
||||||
{
|
|
||||||
Class[] valid = awareCompatibility.get(aware);
|
Class[] valid = awareCompatibility.get(aware);
|
||||||
if (valid == null) {
|
if (valid == null) {
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
||||||
|
@ -912,6 +933,7 @@ public class SolrResourceLoader implements ResourceLoader,Closeable
|
||||||
public void close() throws IOException {
|
public void close() throws IOException {
|
||||||
IOUtils.close(classLoader);
|
IOUtils.close(classLoader);
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<SolrInfoBean> getInfoMBeans() {
|
public List<SolrInfoBean> getInfoMBeans() {
|
||||||
return Collections.unmodifiableList(infoMBeans);
|
return Collections.unmodifiableList(infoMBeans);
|
||||||
}
|
}
|
||||||
|
|
|
@ -354,26 +354,21 @@ public class DistribPackageStore implements PackageStore {
|
||||||
Utils.executeGET(coreContainer.getUpdateShardHandler().getDefaultHttpClient(), url, null);
|
Utils.executeGET(coreContainer.getUpdateShardHandler().getDefaultHttpClient(), url, null);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
log.info("Node: " + node +
|
log.info("Node: " + node +
|
||||||
" failed to respond for blob notification", e);
|
" failed to respond for file fetch notification", e);
|
||||||
//ignore the exception
|
//ignore the exception
|
||||||
// some nodes may be down or not responding
|
// some nodes may be down or not responding
|
||||||
}
|
}
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
new Thread(() -> {
|
coreContainer.getUpdateShardHandler().getUpdateExecutor().submit(() -> {
|
||||||
try {
|
try {
|
||||||
// keep the jar in memory for 10 secs , so that
|
|
||||||
//every node can download it from memory without the file system
|
|
||||||
Thread.sleep(10 * 1000);
|
Thread.sleep(10 * 1000);
|
||||||
} catch (Exception e) {
|
|
||||||
//don't care
|
|
||||||
} finally {
|
} finally {
|
||||||
tmpFiles.remove(entry.getPath());
|
tmpFiles.remove(entry.getPath());
|
||||||
}
|
}
|
||||||
}).start();
|
return null;
|
||||||
|
});
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -198,7 +198,7 @@ public class PackageStoreAPI {
|
||||||
cryptoKeys = new CryptoKeys(keys);
|
cryptoKeys = new CryptoKeys(keys);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
||||||
"Error parsing public keyts in ZooKeeper");
|
"Error parsing public keys in ZooKeeper");
|
||||||
}
|
}
|
||||||
for (String sig : sigs) {
|
for (String sig : sigs) {
|
||||||
if (cryptoKeys.verify(sig, buf) == null) {
|
if (cryptoKeys.verify(sig, buf) == null) {
|
||||||
|
|
|
@ -68,6 +68,7 @@ import org.apache.solr.core.RequestParams;
|
||||||
import org.apache.solr.core.SolrConfig;
|
import org.apache.solr.core.SolrConfig;
|
||||||
import org.apache.solr.core.SolrCore;
|
import org.apache.solr.core.SolrCore;
|
||||||
import org.apache.solr.core.SolrResourceLoader;
|
import org.apache.solr.core.SolrResourceLoader;
|
||||||
|
import org.apache.solr.pkg.PackageListeners;
|
||||||
import org.apache.solr.request.LocalSolrQueryRequest;
|
import org.apache.solr.request.LocalSolrQueryRequest;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.request.SolrRequestHandler;
|
import org.apache.solr.request.SolrRequestHandler;
|
||||||
|
@ -245,10 +246,26 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
|
||||||
if (componentName != null) {
|
if (componentName != null) {
|
||||||
Map map = (Map) val.get(parts.get(1));
|
Map map = (Map) val.get(parts.get(1));
|
||||||
if (map != null) {
|
if (map != null) {
|
||||||
val.put(parts.get(1), makeMap(componentName, map.get(componentName)));
|
Object o = map.get(componentName);
|
||||||
|
val.put(parts.get(1), makeMap(componentName, o));
|
||||||
|
if (req.getParams().getBool("meta", false)) {
|
||||||
|
// meta=true is asking for the package info of the plugin
|
||||||
|
// We go through all the listeners and see if there is one registered for this plugin
|
||||||
|
List<PackageListeners.Listener> listeners = req.getCore().getPackageListeners().getListeners();
|
||||||
|
for (PackageListeners.Listener listener :
|
||||||
|
listeners) {
|
||||||
|
PluginInfo info = listener.pluginInfo();
|
||||||
|
if(info == null) continue;
|
||||||
|
if (info.type.equals(parts.get(1)) && info.name.equals(componentName)) {
|
||||||
|
if (o instanceof Map) {
|
||||||
|
Map m1 = (Map) o;
|
||||||
|
m1.put("_packageinfo_", listener.getPackageVersion());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
resp.add("config", val);
|
resp.add("config", val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -492,6 +509,7 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
|
||||||
}
|
}
|
||||||
List errs = CommandOperation.captureErrors(ops);
|
List errs = CommandOperation.captureErrors(ops);
|
||||||
if (!errs.isEmpty()) {
|
if (!errs.isEmpty()) {
|
||||||
|
log.error("ERROR:" + Utils.toJSONString(errs));
|
||||||
throw new ApiBag.ExceptionWithErrObject(SolrException.ErrorCode.BAD_REQUEST, "error processing commands", errs);
|
throw new ApiBag.ExceptionWithErrObject(SolrException.ErrorCode.BAD_REQUEST, "error processing commands", errs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -499,7 +517,7 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
|
||||||
if (loader instanceof ZkSolrResourceLoader) {
|
if (loader instanceof ZkSolrResourceLoader) {
|
||||||
int latestVersion = ZkController.persistConfigResourceToZooKeeper((ZkSolrResourceLoader) loader, overlay.getZnodeVersion(),
|
int latestVersion = ZkController.persistConfigResourceToZooKeeper((ZkSolrResourceLoader) loader, overlay.getZnodeVersion(),
|
||||||
ConfigOverlay.RESOURCE_NAME, overlay.toByteArray(), true);
|
ConfigOverlay.RESOURCE_NAME, overlay.toByteArray(), true);
|
||||||
log.info("Executed config commands successfully and persisted to ZK {}", ops);
|
log.debug("Executed config commands successfully and persisted to ZK {}", ops);
|
||||||
waitForAllReplicasState(req.getCore().getCoreDescriptor().getCloudDescriptor().getCollectionName(),
|
waitForAllReplicasState(req.getCore().getCoreDescriptor().getCloudDescriptor().getCollectionName(),
|
||||||
req.getCore().getCoreContainer().getZkController(),
|
req.getCore().getCoreContainer().getZkController(),
|
||||||
ConfigOverlay.NAME,
|
ConfigOverlay.NAME,
|
||||||
|
@ -574,6 +592,7 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
|
||||||
try {
|
try {
|
||||||
req.getCore().createInitInstance(new PluginInfo(SolrRequestHandler.TYPE, op.getDataMap()), expected, clz, "");
|
req.getCore().createInitInstance(new PluginInfo(SolrRequestHandler.TYPE, op.getDataMap()), expected, clz, "");
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
log.error("Error checking plugin : ", e);
|
||||||
op.addError(e.getMessage());
|
op.addError(e.getMessage());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -40,6 +40,8 @@ import org.apache.solr.core.CoreContainer;
|
||||||
import org.apache.solr.core.PluginInfo;
|
import org.apache.solr.core.PluginInfo;
|
||||||
import org.apache.solr.core.SolrCore;
|
import org.apache.solr.core.SolrCore;
|
||||||
import org.apache.solr.handler.RequestHandlerBase;
|
import org.apache.solr.handler.RequestHandlerBase;
|
||||||
|
import org.apache.solr.pkg.PackageListeners;
|
||||||
|
import org.apache.solr.pkg.PackageLoader;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
import org.apache.solr.search.SolrQueryTimeoutImpl;
|
import org.apache.solr.search.SolrQueryTimeoutImpl;
|
||||||
|
@ -58,9 +60,7 @@ import static org.apache.solr.common.params.CommonParams.PATH;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* Refer SOLR-281
|
* Refer SOLR-281
|
||||||
*
|
|
||||||
*/
|
*/
|
||||||
public class SearchHandler extends RequestHandlerBase implements SolrCoreAware, PluginInfoInitialized, PermissionNameProvider {
|
public class SearchHandler extends RequestHandlerBase implements SolrCoreAware, PluginInfoInitialized, PermissionNameProvider {
|
||||||
static final String INIT_COMPONENTS = "components";
|
static final String INIT_COMPONENTS = "components";
|
||||||
|
@ -74,8 +74,7 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
|
||||||
private PluginInfo shfInfo;
|
private PluginInfo shfInfo;
|
||||||
private SolrCore core;
|
private SolrCore core;
|
||||||
|
|
||||||
protected List<String> getDefaultComponents()
|
protected List<String> getDefaultComponents() {
|
||||||
{
|
|
||||||
ArrayList<String> names = new ArrayList<>(8);
|
ArrayList<String> names = new ArrayList<>(8);
|
||||||
names.add(QueryComponent.COMPONENT_NAME);
|
names.add(QueryComponent.COMPONENT_NAME);
|
||||||
names.add(FacetComponent.COMPONENT_NAME);
|
names.add(FacetComponent.COMPONENT_NAME);
|
||||||
|
@ -113,8 +112,7 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public void inform(SolrCore core)
|
public void inform(SolrCore core) {
|
||||||
{
|
|
||||||
this.core = core;
|
this.core = core;
|
||||||
List<String> c = (List<String>) initArgs.get(INIT_COMPONENTS);
|
List<String> c = (List<String>) initArgs.get(INIT_COMPONENTS);
|
||||||
Set<String> missing = new HashSet<>(core.getSearchComponents().checkContains(c));
|
Set<String> missing = new HashSet<>(core.getSearchComponents().checkContains(c));
|
||||||
|
@ -143,6 +141,32 @@ public class SearchHandler extends RequestHandlerBase implements SolrCoreAware ,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (core.getCoreContainer().isZooKeeperAware()) {
|
||||||
|
core.getPackageListeners().addListener(new PackageListeners.Listener() {
|
||||||
|
@Override
|
||||||
|
public String packageName() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PluginInfo pluginInfo() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void changed(PackageLoader.Package pkg) {
|
||||||
|
//we could optimize this by listening to only relevant packages,
|
||||||
|
// but it is not worth optimizing as these are lightweight objects
|
||||||
|
components = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PackageLoader.Package.Version getPackageVersion() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void initComponents() {
|
private void initComponents() {
|
||||||
|
|
|
@ -0,0 +1,383 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.pkg;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import com.fasterxml.jackson.databind.ObjectMapper;
|
||||||
|
import org.apache.solr.api.Command;
|
||||||
|
import org.apache.solr.api.EndPoint;
|
||||||
|
import org.apache.solr.api.PayloadObj;
|
||||||
|
import org.apache.solr.client.solrj.SolrRequest;
|
||||||
|
import org.apache.solr.client.solrj.request.beans.Package;
|
||||||
|
import org.apache.solr.common.SolrException;
|
||||||
|
import org.apache.solr.common.cloud.SolrZkClient;
|
||||||
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
|
import org.apache.solr.common.cloud.ZooKeeperException;
|
||||||
|
import org.apache.solr.common.util.CommandOperation;
|
||||||
|
import org.apache.solr.common.util.ReflectMapWriter;
|
||||||
|
import org.apache.solr.common.util.Utils;
|
||||||
|
import org.apache.solr.core.CoreContainer;
|
||||||
|
import org.apache.solr.filestore.PackageStoreAPI;
|
||||||
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
|
import org.apache.zookeeper.KeeperException;
|
||||||
|
import org.apache.zookeeper.WatchedEvent;
|
||||||
|
import org.apache.zookeeper.Watcher;
|
||||||
|
import org.apache.zookeeper.data.Stat;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
import static org.apache.solr.common.cloud.ZkStateReader.SOLR_PKGS_PATH;
|
||||||
|
import static org.apache.solr.security.PermissionNameProvider.Name.PACKAGE_EDIT_PERM;
|
||||||
|
import static org.apache.solr.security.PermissionNameProvider.Name.PACKAGE_READ_PERM;
|
||||||
|
|
||||||
|
public class PackageAPI {
|
||||||
|
public static final String PACKAGES = "packages";
|
||||||
|
public final boolean enablePackages = Boolean.parseBoolean(System.getProperty("enable.packages", "false"));
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
|
final CoreContainer coreContainer;
|
||||||
|
private ObjectMapper mapper = new ObjectMapper();
|
||||||
|
private final PackageLoader packageLoader;
|
||||||
|
Packages pkgs;
|
||||||
|
|
||||||
|
public final Edit editAPI = new Edit();
|
||||||
|
public final Read readAPI = new Read();
|
||||||
|
|
||||||
|
public PackageAPI(CoreContainer coreContainer, PackageLoader loader) {
|
||||||
|
this.coreContainer = coreContainer;
|
||||||
|
this.packageLoader = loader;
|
||||||
|
pkgs = new Packages();
|
||||||
|
SolrZkClient zkClient = coreContainer.getZkController().getZkClient();
|
||||||
|
try {
|
||||||
|
registerListener(zkClient);
|
||||||
|
} catch (KeeperException | InterruptedException e) {
|
||||||
|
SolrZkClient.checkInterrupted(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void registerListener(SolrZkClient zkClient)
|
||||||
|
throws KeeperException, InterruptedException {
|
||||||
|
String path = SOLR_PKGS_PATH;
|
||||||
|
zkClient.exists(path,
|
||||||
|
new Watcher() {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void process(WatchedEvent event) {
|
||||||
|
// session events are not change events, and do not remove the watcher
|
||||||
|
if (Event.EventType.None.equals(event.getType())) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
synchronized (this) {
|
||||||
|
log.debug("Updating [{}] ... ", path);
|
||||||
|
|
||||||
|
// remake watch
|
||||||
|
final Watcher thisWatch = this;
|
||||||
|
final Stat stat = new Stat();
|
||||||
|
final byte[] data = zkClient.getData(path, thisWatch, stat, true);
|
||||||
|
pkgs = readPkgsFromZk(data, stat);
|
||||||
|
packageLoader.refreshPackageConf();
|
||||||
|
}
|
||||||
|
} catch (KeeperException.ConnectionLossException | KeeperException.SessionExpiredException e) {
|
||||||
|
log.warn("ZooKeeper watch triggered, but Solr cannot talk to ZK: [{}]", e.getMessage());
|
||||||
|
} catch (KeeperException e) {
|
||||||
|
log.error("A ZK error has occurred", e);
|
||||||
|
throw new ZooKeeperException(SolrException.ErrorCode.SERVER_ERROR, "", e);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
// Restore the interrupted status
|
||||||
|
Thread.currentThread().interrupt();
|
||||||
|
log.warn("Interrupted", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private Packages readPkgsFromZk(byte[] data, Stat stat) throws KeeperException, InterruptedException {
|
||||||
|
|
||||||
|
if (data == null || stat == null) {
|
||||||
|
stat = new Stat();
|
||||||
|
data = coreContainer.getZkController().getZkClient()
|
||||||
|
.getData(ZkStateReader.CLUSTER_PROPS, null, stat, true);
|
||||||
|
|
||||||
|
}
|
||||||
|
Packages packages = null;
|
||||||
|
if (data == null || data.length == 0) {
|
||||||
|
packages = new Packages();
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
packages = mapper.readValue(data, Packages.class);
|
||||||
|
packages.znodeVersion = stat.getVersion();
|
||||||
|
} catch (IOException e) {
|
||||||
|
//invalid data in packages
|
||||||
|
//TODO handle properly;
|
||||||
|
return new Packages();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return packages;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static class Packages implements ReflectMapWriter {
|
||||||
|
@JsonProperty
|
||||||
|
public int znodeVersion = -1;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
public Map<String, List<PkgVersion>> packages = new LinkedHashMap<>();
|
||||||
|
|
||||||
|
|
||||||
|
public Packages copy() {
|
||||||
|
Packages p = new Packages();
|
||||||
|
p.znodeVersion = this.znodeVersion;
|
||||||
|
p.packages = new LinkedHashMap<>();
|
||||||
|
packages.forEach((s, versions) ->
|
||||||
|
p.packages.put(s, new ArrayList<>(versions)));
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class PkgVersion implements ReflectMapWriter {
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
public String version;
|
||||||
|
|
||||||
|
@JsonProperty
|
||||||
|
public List<String> files;
|
||||||
|
|
||||||
|
public PkgVersion() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public PkgVersion(Package.AddVersion addVersion) {
|
||||||
|
this.version = addVersion.version;
|
||||||
|
this.files = addVersion.files;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean equals(Object obj) {
|
||||||
|
if (obj instanceof PkgVersion) {
|
||||||
|
PkgVersion that = (PkgVersion) obj;
|
||||||
|
return Objects.equals(this.version, that.version)
|
||||||
|
&& Objects.equals(this.files, that.files);
|
||||||
|
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@EndPoint(method = SolrRequest.METHOD.POST,
|
||||||
|
path = "/cluster/package",
|
||||||
|
permission = PACKAGE_EDIT_PERM)
|
||||||
|
public class Edit {
|
||||||
|
|
||||||
|
@Command(name = "refresh")
|
||||||
|
public void refresh(SolrQueryRequest req, SolrQueryResponse rsp, PayloadObj<String> payload) {
|
||||||
|
String p = payload.get();
|
||||||
|
if (p == null) {
|
||||||
|
payload.addError("Package null");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
PackageLoader.Package pkg = coreContainer.getPackageLoader().getPackage(p);
|
||||||
|
if (pkg == null) {
|
||||||
|
payload.addError("No such package: " + p);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (String s : coreContainer.getPackageStoreAPI().shuffledNodes()) {
|
||||||
|
Utils.executeGET(coreContainer.getUpdateShardHandler().getDefaultHttpClient(),
|
||||||
|
coreContainer.getZkController().zkStateReader.getBaseUrlForNodeName(s).replace("/solr", "/api") + "/cluster/package?wt=javabin&omitHeader=true&refreshPackage=" + p,
|
||||||
|
Utils.JAVABINCONSUMER);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Command(name = "add")
|
||||||
|
public void add(SolrQueryRequest req, SolrQueryResponse rsp, PayloadObj<Package.AddVersion> payload) {
|
||||||
|
if (!checkEnabled(payload)) return;
|
||||||
|
Package.AddVersion add = payload.get();
|
||||||
|
if (add.files.isEmpty()) {
|
||||||
|
payload.addError("No files specified");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
PackageStoreAPI packageStoreAPI = coreContainer.getPackageStoreAPI();
|
||||||
|
packageStoreAPI.validateFiles(add.files, true, s -> payload.addError(s));
|
||||||
|
if (payload.hasError()) return;
|
||||||
|
Packages[] finalState = new Packages[1];
|
||||||
|
try {
|
||||||
|
coreContainer.getZkController().getZkClient().atomicUpdate(SOLR_PKGS_PATH, (stat, bytes) -> {
|
||||||
|
Packages packages = null;
|
||||||
|
try {
|
||||||
|
packages = bytes == null ? new Packages() : mapper.readValue(bytes, Packages.class);
|
||||||
|
packages = packages.copy();
|
||||||
|
} catch (IOException e) {
|
||||||
|
log.error("Error deserializing packages.json", e);
|
||||||
|
packages = new Packages();
|
||||||
|
}
|
||||||
|
packages.packages.computeIfAbsent(add.pkg, Utils.NEW_ARRAYLIST_FUN).add(new PkgVersion(add));
|
||||||
|
packages.znodeVersion = stat.getVersion() + 1;
|
||||||
|
finalState[0] = packages;
|
||||||
|
return Utils.toJSON(packages);
|
||||||
|
});
|
||||||
|
} catch (KeeperException | InterruptedException e) {
|
||||||
|
finalState[0] = null;
|
||||||
|
handleZkErr(e);
|
||||||
|
}
|
||||||
|
if (finalState[0] != null) {
|
||||||
|
// succeeded in updating
|
||||||
|
pkgs = finalState[0];
|
||||||
|
notifyAllNodesToSync(pkgs.znodeVersion);
|
||||||
|
packageLoader.refreshPackageConf();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Command(name = "delete")
|
||||||
|
public void del(SolrQueryRequest req, SolrQueryResponse rsp, PayloadObj<Package.DelVersion> payload) {
|
||||||
|
if (!checkEnabled(payload)) return;
|
||||||
|
Package.DelVersion delVersion = payload.get();
|
||||||
|
try {
|
||||||
|
coreContainer.getZkController().getZkClient().atomicUpdate(SOLR_PKGS_PATH, (stat, bytes) -> {
|
||||||
|
Packages packages = null;
|
||||||
|
try {
|
||||||
|
packages = mapper.readValue(bytes, Packages.class);
|
||||||
|
packages = packages.copy();
|
||||||
|
} catch (IOException e) {
|
||||||
|
packages = new Packages();
|
||||||
|
}
|
||||||
|
|
||||||
|
List<PkgVersion> versions = packages.packages.get(delVersion.pkg);
|
||||||
|
if (versions == null || versions.isEmpty()) {
|
||||||
|
payload.addError("No such package: " + delVersion.pkg);
|
||||||
|
return null;// no change
|
||||||
|
}
|
||||||
|
int idxToremove = -1;
|
||||||
|
for (int i = 0; i < versions.size(); i++) {
|
||||||
|
if (Objects.equals(versions.get(i).version, delVersion.version)) {
|
||||||
|
idxToremove = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (idxToremove == -1) {
|
||||||
|
payload.addError("No such version: " + delVersion.version);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
versions.remove(idxToremove);
|
||||||
|
packages.znodeVersion = stat.getVersion() + 1;
|
||||||
|
return Utils.toJSON(packages);
|
||||||
|
});
|
||||||
|
} catch (KeeperException | InterruptedException e) {
|
||||||
|
handleZkErr(e);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean checkEnabled(CommandOperation payload) {
|
||||||
|
if (!enablePackages) {
|
||||||
|
payload.addError("Package loading is not enabled , Start your nodes with -Denable.packages=true");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@EndPoint(
|
||||||
|
method = SolrRequest.METHOD.GET,
|
||||||
|
path = {"/cluster/package/",
|
||||||
|
"/cluster/package/{name}"},
|
||||||
|
permission = PACKAGE_READ_PERM
|
||||||
|
)
|
||||||
|
public class Read {
|
||||||
|
@Command()
|
||||||
|
public void get(SolrQueryRequest req, SolrQueryResponse rsp) {
|
||||||
|
String refresh = req.getParams().get("refreshPackage");
|
||||||
|
if (refresh != null) {
|
||||||
|
packageLoader.notifyListeners(refresh);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int expectedVersion = req.getParams().getInt("expectedVersion", -1);
|
||||||
|
if (expectedVersion != -1) {
|
||||||
|
syncToVersion(expectedVersion);
|
||||||
|
}
|
||||||
|
String name = req.getPathTemplateValues().get("name");
|
||||||
|
if (name == null) {
|
||||||
|
rsp.add("result", pkgs);
|
||||||
|
} else {
|
||||||
|
rsp.add("result", Collections.singletonMap(name, pkgs.packages.get(name)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void syncToVersion(int expectedVersion) {
|
||||||
|
int origVersion = pkgs.znodeVersion;
|
||||||
|
for (int i = 0; i < 10; i++) {
|
||||||
|
log.debug("my version is {} , and expected version {}", pkgs.znodeVersion, expectedVersion);
|
||||||
|
if (pkgs.znodeVersion >= expectedVersion) {
|
||||||
|
if (origVersion < pkgs.znodeVersion) {
|
||||||
|
packageLoader.refreshPackageConf();
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
Thread.sleep(10);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
pkgs = readPkgsFromZk(null, null);
|
||||||
|
} catch (KeeperException | InterruptedException e) {
|
||||||
|
handleZkErr(e);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void notifyAllNodesToSync(int expected) {
|
||||||
|
for (String s : coreContainer.getPackageStoreAPI().shuffledNodes()) {
|
||||||
|
Utils.executeGET(coreContainer.getUpdateShardHandler().getDefaultHttpClient(),
|
||||||
|
coreContainer.getZkController().zkStateReader.getBaseUrlForNodeName(s).replace("/solr", "/api") + "/cluster/package?wt=javabin&omitHeader=true&expectedVersion" + expected,
|
||||||
|
Utils.JAVABINCONSUMER);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void handleZkErr(Exception e) {
|
||||||
|
log.error("Error reading package config from zookeeper", SolrZkClient.checkInterrupted(e));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,111 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.pkg;
|
||||||
|
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.lang.ref.Reference;
|
||||||
|
import java.lang.ref.SoftReference;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.apache.solr.core.PluginInfo;
|
||||||
|
import org.apache.solr.core.SolrCore;
|
||||||
|
import org.apache.solr.logging.MDCLoggingContext;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
public class PackageListeners {
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
|
public static final String PACKAGE_VERSIONS = "PKG_VERSIONS";
|
||||||
|
private SolrCore core;
|
||||||
|
|
||||||
|
public PackageListeners(SolrCore core) {
|
||||||
|
this.core = core;
|
||||||
|
}
|
||||||
|
|
||||||
|
// this registry only keeps a weak reference because it does not want to
|
||||||
|
// cause a memory leak if the listener forgets to unregister itself
|
||||||
|
private List<Reference<Listener>> listeners = new ArrayList<>();
|
||||||
|
|
||||||
|
public synchronized void addListener(Listener listener) {
|
||||||
|
listeners.add(new SoftReference<>(listener));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public synchronized void removeListener(Listener listener) {
|
||||||
|
Iterator<Reference<Listener>> it = listeners.iterator();
|
||||||
|
while (it.hasNext()) {
|
||||||
|
Reference<Listener> ref = it.next();
|
||||||
|
Listener pkgListener = ref.get();
|
||||||
|
if (pkgListener == null || pkgListener == listener) {
|
||||||
|
it.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
synchronized void packagesUpdated(List<PackageLoader.Package> pkgs) {
|
||||||
|
if(core != null) MDCLoggingContext.setCore(core);
|
||||||
|
try {
|
||||||
|
for (PackageLoader.Package pkgInfo : pkgs) {
|
||||||
|
invokeListeners(pkgInfo);
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
if(core != null) MDCLoggingContext.clear();
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private synchronized void invokeListeners(PackageLoader.Package pkg) {
|
||||||
|
for (Reference<Listener> ref : listeners) {
|
||||||
|
Listener listener = ref.get();
|
||||||
|
if(listener == null) continue;
|
||||||
|
if (listener.packageName() == null || listener.packageName().equals(pkg.name())) {
|
||||||
|
listener.changed(pkg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Listener> getListeners() {
|
||||||
|
List<Listener> result = new ArrayList<>();
|
||||||
|
for (Reference<Listener> ref : listeners) {
|
||||||
|
Listener l = ref.get();
|
||||||
|
if (l != null) {
|
||||||
|
result.add(l);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public interface Listener {
|
||||||
|
/**Name of the package or null to loisten to all package changes
|
||||||
|
*/
|
||||||
|
String packageName();
|
||||||
|
|
||||||
|
PluginInfo pluginInfo();
|
||||||
|
|
||||||
|
void changed(PackageLoader.Package pkg);
|
||||||
|
|
||||||
|
PackageLoader.Package.Version getPackageVersion();
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,276 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.pkg;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.nio.file.Path;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.CopyOnWriteArrayList;
|
||||||
|
|
||||||
|
import org.apache.solr.common.MapWriter;
|
||||||
|
import org.apache.solr.common.cloud.ZkStateReader;
|
||||||
|
import org.apache.solr.core.CoreContainer;
|
||||||
|
import org.apache.solr.core.SolrCore;
|
||||||
|
import org.apache.solr.core.SolrResourceLoader;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The class that holds a mapping of various packages and classloaders
|
||||||
|
*/
|
||||||
|
public class PackageLoader {
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
|
||||||
|
private final CoreContainer coreContainer;
|
||||||
|
private final Map<String, Package> packageClassLoaders = new ConcurrentHashMap<>();
|
||||||
|
|
||||||
|
private PackageAPI.Packages myCopy;
|
||||||
|
|
||||||
|
private PackageAPI packageAPI;
|
||||||
|
|
||||||
|
|
||||||
|
public PackageLoader(CoreContainer coreContainer) {
|
||||||
|
this.coreContainer = coreContainer;
|
||||||
|
packageAPI = new PackageAPI(coreContainer, this);
|
||||||
|
myCopy = packageAPI.pkgs;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public PackageAPI getPackageAPI() {
|
||||||
|
return packageAPI;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Package getPackage(String key) {
|
||||||
|
return packageClassLoaders.get(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, Package> getPackages() {
|
||||||
|
return Collections.EMPTY_MAP;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void refreshPackageConf() {
|
||||||
|
log.info("{} updated to version {}", ZkStateReader.SOLR_PKGS_PATH, packageAPI.pkgs.znodeVersion);
|
||||||
|
|
||||||
|
List<Package> updated = new ArrayList<>();
|
||||||
|
Map<String, List<PackageAPI.PkgVersion>> modified = getModified(myCopy, packageAPI.pkgs);
|
||||||
|
|
||||||
|
for (Map.Entry<String, List<PackageAPI.PkgVersion>> e : modified.entrySet()) {
|
||||||
|
if (e.getValue() != null) {
|
||||||
|
Package p = packageClassLoaders.get(e.getKey());
|
||||||
|
if (e.getValue() != null && p == null) {
|
||||||
|
packageClassLoaders.put(e.getKey(), p = new Package(e.getKey()));
|
||||||
|
}
|
||||||
|
p.updateVersions(e.getValue());
|
||||||
|
updated.add(p);
|
||||||
|
} else {
|
||||||
|
Package p = packageClassLoaders.remove(e.getKey());
|
||||||
|
if (p != null) {
|
||||||
|
//other classes are holding to a reference to this objecec
|
||||||
|
// they should know that this is removed
|
||||||
|
p.markDeleted();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (SolrCore core : coreContainer.getCores()) {
|
||||||
|
core.getPackageListeners().packagesUpdated(updated);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, List<PackageAPI.PkgVersion>> getModified(PackageAPI.Packages old, PackageAPI.Packages newPkgs) {
|
||||||
|
Map<String, List<PackageAPI.PkgVersion>> changed = new HashMap<>();
|
||||||
|
for (Map.Entry<String, List<PackageAPI.PkgVersion>> e : newPkgs.packages.entrySet()) {
|
||||||
|
List<PackageAPI.PkgVersion> versions = old.packages.get(e.getKey());
|
||||||
|
if (versions != null) {
|
||||||
|
if (!Objects.equals(e.getValue(), versions)) {
|
||||||
|
log.info("Package {} is modified ", e.getKey());
|
||||||
|
changed.put(e.getKey(), e.getValue());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
log.info("A new package: {} introduced", e.getKey());
|
||||||
|
changed.put(e.getKey(), e.getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//some packages are deleted altogether
|
||||||
|
for (String s : old.packages.keySet()) {
|
||||||
|
if (!newPkgs.packages.keySet().contains(s)) {
|
||||||
|
log.info("Package: {} is removed althogether", s);
|
||||||
|
changed.put(s, null);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return changed;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public void notifyListeners(String pkg) {
|
||||||
|
Package p = packageClassLoaders.get(pkg);
|
||||||
|
if(p != null){
|
||||||
|
List<Package> l = Collections.singletonList(p);
|
||||||
|
for (SolrCore core : coreContainer.getCores()) {
|
||||||
|
core.getPackageListeners().packagesUpdated(l);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* represents a package definition in the packages.json
|
||||||
|
*/
|
||||||
|
public class Package {
|
||||||
|
final String name;
|
||||||
|
final Map<String, Version> myVersions = new ConcurrentHashMap<>();
|
||||||
|
private List<String> sortedVersions = new CopyOnWriteArrayList<>();
|
||||||
|
String latest;
|
||||||
|
private boolean deleted;
|
||||||
|
|
||||||
|
|
||||||
|
Package(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isDeleted() {
|
||||||
|
return deleted;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private synchronized void updateVersions(List<PackageAPI.PkgVersion> modified) {
|
||||||
|
for (PackageAPI.PkgVersion v : modified) {
|
||||||
|
Version version = myVersions.get(v.version);
|
||||||
|
if (version == null) {
|
||||||
|
log.info("A new version: {} added for package: {} with artifacts {}", v.version, this.name, v.files);
|
||||||
|
myVersions.put(v.version, new Version(this, v));
|
||||||
|
sortedVersions.add(v.version);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Set<String> newVersions = new HashSet<>();
|
||||||
|
for (PackageAPI.PkgVersion v : modified) {
|
||||||
|
newVersions.add(v.version);
|
||||||
|
}
|
||||||
|
for (String s : new HashSet<>(myVersions.keySet())) {
|
||||||
|
if (!newVersions.contains(s)) {
|
||||||
|
log.info("version: {} is removed from package: {}", s, this.name);
|
||||||
|
sortedVersions.remove(s);
|
||||||
|
myVersions.remove(s);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sortedVersions.sort(String::compareTo);
|
||||||
|
if (sortedVersions.size() > 0) {
|
||||||
|
String latest = sortedVersions.get(sortedVersions.size() - 1);
|
||||||
|
if (!latest.equals(this.latest)) {
|
||||||
|
log.info("version: {} is the new latest in package: {}", latest, this.name);
|
||||||
|
}
|
||||||
|
this.latest = latest;
|
||||||
|
} else {
|
||||||
|
log.error("latest version: null");
|
||||||
|
latest = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public Version getLatest() {
|
||||||
|
return latest == null ? null : myVersions.get(latest);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Version getLatest(String lessThan) {
|
||||||
|
if (lessThan == null) {
|
||||||
|
return getLatest();
|
||||||
|
}
|
||||||
|
String latest = findBiggest(lessThan, new ArrayList(sortedVersions));
|
||||||
|
return latest == null ? null : myVersions.get(latest);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String name() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void markDeleted() {
|
||||||
|
deleted = true;
|
||||||
|
myVersions.clear();
|
||||||
|
sortedVersions.clear();
|
||||||
|
latest = null;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Version implements MapWriter {
|
||||||
|
private final Package parent;
|
||||||
|
private SolrResourceLoader loader;
|
||||||
|
|
||||||
|
private final PackageAPI.PkgVersion version;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void writeMap(EntryWriter ew) throws IOException {
|
||||||
|
ew.put("package", parent.name());
|
||||||
|
version.writeMap(ew);
|
||||||
|
}
|
||||||
|
|
||||||
|
Version(Package parent, PackageAPI.PkgVersion v) {
|
||||||
|
this.parent = parent;
|
||||||
|
this.version = v;
|
||||||
|
List<Path> paths = new ArrayList<>();
|
||||||
|
for (String file : version.files) {
|
||||||
|
paths.add(coreContainer.getPackageStoreAPI().getPackageStore().getRealpath(file));
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
loader = new SolrResourceLoader(
|
||||||
|
"PACKAGE_LOADER: " + parent.name() + ":" + version,
|
||||||
|
paths,
|
||||||
|
coreContainer.getResourceLoader().getInstancePath(),
|
||||||
|
coreContainer.getResourceLoader().getClassLoader());
|
||||||
|
} catch (MalformedURLException e) {
|
||||||
|
log.error("Could not load classloader ", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getVersion() {
|
||||||
|
return version.version;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Collection getFiles() {
|
||||||
|
return Collections.unmodifiableList(version.files);
|
||||||
|
}
|
||||||
|
|
||||||
|
public SolrResourceLoader getLoader() {
|
||||||
|
return loader;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String findBiggest(String lessThan, List<String> sortedList) {
|
||||||
|
String latest = null;
|
||||||
|
for (String v : sortedList) {
|
||||||
|
if (v.compareTo(lessThan) < 1) {
|
||||||
|
latest = v;
|
||||||
|
} else break;
|
||||||
|
}
|
||||||
|
return latest;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,123 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.pkg;
|
||||||
|
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
|
||||||
|
import org.apache.solr.core.PluginBag;
|
||||||
|
import org.apache.solr.core.PluginInfo;
|
||||||
|
import org.apache.solr.core.RequestParams;
|
||||||
|
import org.apache.solr.core.SolrConfig;
|
||||||
|
import org.apache.solr.core.SolrCore;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
|
public class PackagePluginHolder<T> extends PluginBag.PluginHolder<T> {
|
||||||
|
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
|
||||||
|
public static final String LATEST = "$LATEST";
|
||||||
|
|
||||||
|
private final SolrCore core;
|
||||||
|
private final SolrConfig.SolrPluginInfo pluginMeta;
|
||||||
|
private PackageLoader.Package.Version pkgVersion;
|
||||||
|
private PluginInfo info;
|
||||||
|
|
||||||
|
|
||||||
|
public PackagePluginHolder(PluginInfo info, SolrCore core, SolrConfig.SolrPluginInfo pluginMeta) {
|
||||||
|
super(info);
|
||||||
|
this.core = core;
|
||||||
|
this.pluginMeta = pluginMeta;
|
||||||
|
this.info = info;
|
||||||
|
|
||||||
|
reload(core.getCoreContainer().getPackageLoader().getPackage(info.pkgName));
|
||||||
|
core.getPackageListeners().addListener(new PackageListeners.Listener() {
|
||||||
|
@Override
|
||||||
|
public String packageName() {
|
||||||
|
return info.pkgName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PluginInfo pluginInfo() {
|
||||||
|
return info;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void changed(PackageLoader.Package pkg) {
|
||||||
|
reload(pkg);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PackageLoader.Package.Version getPackageVersion() {
|
||||||
|
return pkgVersion;
|
||||||
|
}
|
||||||
|
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private String maxVersion() {
|
||||||
|
RequestParams.ParamSet p = core.getSolrConfig().getRequestParams().getParams(PackageListeners.PACKAGE_VERSIONS);
|
||||||
|
if (p == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
Object o = p.get().get(info.pkgName);
|
||||||
|
if (o == null || LATEST.equals(o)) return null;
|
||||||
|
return o.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private synchronized void reload(PackageLoader.Package pkg) {
|
||||||
|
String lessThan = maxVersion();
|
||||||
|
PackageLoader.Package.Version newest = pkg.getLatest(lessThan);
|
||||||
|
if (newest == null) {
|
||||||
|
log.error("No latest version available for package : {}", pkg.name());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (lessThan != null) {
|
||||||
|
PackageLoader.Package.Version pkgLatest = pkg.getLatest();
|
||||||
|
if (pkgLatest != newest) {
|
||||||
|
log.info("Using version :{}. latest is {}, params.json has config {} : {}", newest.getVersion(), pkgLatest.getVersion(), pkg.name(), lessThan);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pkgVersion != null) {
|
||||||
|
if (newest == pkgVersion) {
|
||||||
|
//I'm already using the latest classloder in the package. nothing to do
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.info("loading plugin: {} -> {} using package {}:{}",
|
||||||
|
pluginInfo.type, pluginInfo.name, pkg.name(), newest.getVersion());
|
||||||
|
|
||||||
|
Object instance = SolrCore.createInstance(pluginInfo.className,
|
||||||
|
pluginMeta.clazz, pluginMeta.getCleanTag(), core, newest.getLoader());
|
||||||
|
PluginBag.initInstance(instance, pluginInfo);
|
||||||
|
T old = inst;
|
||||||
|
inst = (T) instance;
|
||||||
|
pkgVersion = newest;
|
||||||
|
if (old instanceof AutoCloseable) {
|
||||||
|
AutoCloseable closeable = (AutoCloseable) old;
|
||||||
|
try {
|
||||||
|
closeable.close();
|
||||||
|
} catch (Exception e) {
|
||||||
|
log.error("error closing plugin", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -53,6 +53,9 @@ public interface PermissionNameProvider {
|
||||||
METRICS_HISTORY_READ_PERM("metrics-history-read", null),
|
METRICS_HISTORY_READ_PERM("metrics-history-read", null),
|
||||||
FILESTORE_READ_PERM("filestore-read", null),
|
FILESTORE_READ_PERM("filestore-read", null),
|
||||||
FILESTORE_WRITE_PERM("filestore-write", null),
|
FILESTORE_WRITE_PERM("filestore-write", null),
|
||||||
|
PACKAGE_EDIT_PERM("package-edit", null),
|
||||||
|
PACKAGE_READ_PERM("package-read", null),
|
||||||
|
|
||||||
ALL("all", unmodifiableSet(new HashSet<>(asList("*", null))))
|
ALL("all", unmodifiableSet(new HashSet<>(asList("*", null))))
|
||||||
;
|
;
|
||||||
final String name;
|
final String name;
|
||||||
|
|
|
@ -23,6 +23,7 @@ import java.util.LinkedList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import com.google.common.collect.ImmutableMap;
|
import com.google.common.collect.ImmutableMap;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
|
@ -33,9 +34,12 @@ import org.apache.solr.common.util.StrUtils;
|
||||||
import org.apache.solr.common.util.Utils;
|
import org.apache.solr.common.util.Utils;
|
||||||
import org.apache.solr.core.PluginBag;
|
import org.apache.solr.core.PluginBag;
|
||||||
import org.apache.solr.core.PluginInfo;
|
import org.apache.solr.core.PluginInfo;
|
||||||
|
import org.apache.solr.core.SolrConfig;
|
||||||
import org.apache.solr.core.SolrCore;
|
import org.apache.solr.core.SolrCore;
|
||||||
|
import org.apache.solr.pkg.PackagePluginHolder;
|
||||||
import org.apache.solr.request.SolrQueryRequest;
|
import org.apache.solr.request.SolrQueryRequest;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
import org.apache.solr.response.SolrQueryResponse;
|
||||||
|
import org.apache.solr.update.processor.UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder.LazyUpdateRequestProcessorFactory;
|
||||||
import org.apache.solr.util.plugin.PluginInfoInitialized;
|
import org.apache.solr.util.plugin.PluginInfoInitialized;
|
||||||
import org.apache.solr.util.plugin.SolrCoreAware;
|
import org.apache.solr.util.plugin.SolrCoreAware;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
|
@ -126,8 +130,7 @@ public final class UpdateRequestProcessorChain implements PluginInfoInitialized
|
||||||
|
|
||||||
// wrap in an ArrayList so we know we know we can do fast index lookups
|
// wrap in an ArrayList so we know we know we can do fast index lookups
|
||||||
// and that add(int,Object) is supported
|
// and that add(int,Object) is supported
|
||||||
List<UpdateRequestProcessorFactory> list = new ArrayList<>
|
List<UpdateRequestProcessorFactory> list = createProcessors(info);
|
||||||
(solrCore.initPlugins(info.getChildren("processor"),UpdateRequestProcessorFactory.class,null));
|
|
||||||
|
|
||||||
if(list.isEmpty()){
|
if(list.isEmpty()){
|
||||||
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
|
||||||
|
@ -170,6 +173,23 @@ public final class UpdateRequestProcessorChain implements PluginInfoInitialized
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private List<UpdateRequestProcessorFactory> createProcessors(PluginInfo info) {
|
||||||
|
List<PluginInfo> processors = info.getChildren("processor");
|
||||||
|
return processors.stream().map(it -> {
|
||||||
|
if(it.pkgName == null){
|
||||||
|
return solrCore.createInitInstance(it, UpdateRequestProcessorFactory.class,
|
||||||
|
UpdateRequestProcessorFactory.class.getSimpleName(), null);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
return new LazyUpdateRequestProcessorFactory(new PackagePluginHolder(
|
||||||
|
it,
|
||||||
|
solrCore,
|
||||||
|
SolrConfig.classVsSolrPluginInfo.get(UpdateRequestProcessorFactory.class.getName())));
|
||||||
|
}
|
||||||
|
}).collect(Collectors.toList());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a chain backed directly by the specified list. Modifications to
|
* Creates a chain backed directly by the specified list. Modifications to
|
||||||
* the array will affect future calls to <code>createProcessor</code>
|
* the array will affect future calls to <code>createProcessor</code>
|
||||||
|
@ -328,7 +348,7 @@ public final class UpdateRequestProcessorChain implements PluginInfoInitialized
|
||||||
public static class LazyUpdateProcessorFactoryHolder extends PluginBag.PluginHolder<UpdateRequestProcessorFactory> {
|
public static class LazyUpdateProcessorFactoryHolder extends PluginBag.PluginHolder<UpdateRequestProcessorFactory> {
|
||||||
private volatile UpdateRequestProcessorFactory lazyFactory;
|
private volatile UpdateRequestProcessorFactory lazyFactory;
|
||||||
|
|
||||||
public LazyUpdateProcessorFactoryHolder(final PluginBag.LazyPluginHolder holder) {
|
public LazyUpdateProcessorFactoryHolder(final PluginBag.PluginHolder holder) {
|
||||||
super(holder.getPluginInfo());
|
super(holder.getPluginInfo());
|
||||||
lazyFactory = new LazyUpdateRequestProcessorFactory(holder);
|
lazyFactory = new LazyUpdateRequestProcessorFactory(holder);
|
||||||
}
|
}
|
||||||
|
@ -339,27 +359,20 @@ public final class UpdateRequestProcessorChain implements PluginInfoInitialized
|
||||||
return lazyFactory;
|
return lazyFactory;
|
||||||
}
|
}
|
||||||
|
|
||||||
public class LazyUpdateRequestProcessorFactory extends UpdateRequestProcessorFactory {
|
public static class LazyUpdateRequestProcessorFactory extends UpdateRequestProcessorFactory {
|
||||||
private final PluginBag.LazyPluginHolder holder;
|
private final PluginBag.PluginHolder<UpdateRequestProcessorFactory> holder;
|
||||||
UpdateRequestProcessorFactory delegate;
|
|
||||||
|
|
||||||
public LazyUpdateRequestProcessorFactory(PluginBag.LazyPluginHolder holder) {
|
public LazyUpdateRequestProcessorFactory(PluginBag.PluginHolder holder) {
|
||||||
this.holder = holder;
|
this.holder = holder;
|
||||||
}
|
}
|
||||||
|
|
||||||
public UpdateRequestProcessorFactory getDelegate() {
|
public UpdateRequestProcessorFactory getDelegate() {
|
||||||
return delegate;
|
return holder.get();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public UpdateRequestProcessor getInstance(SolrQueryRequest req, SolrQueryResponse rsp, UpdateRequestProcessor next) {
|
public UpdateRequestProcessor getInstance(SolrQueryRequest req, SolrQueryResponse rsp, UpdateRequestProcessor next) {
|
||||||
if (delegate != null) return delegate.getInstance(req, rsp, next);
|
return holder.get().getInstance(req, rsp, next);
|
||||||
|
|
||||||
synchronized (this) {
|
|
||||||
if (delegate == null)
|
|
||||||
delegate = (UpdateRequestProcessorFactory) holder.get();
|
|
||||||
}
|
|
||||||
return delegate.getInstance(req, rsp, next);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,495 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.pkg;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.Callable;
|
||||||
|
|
||||||
|
import org.apache.commons.codec.digest.DigestUtils;
|
||||||
|
import org.apache.solr.client.solrj.SolrClient;
|
||||||
|
import org.apache.solr.client.solrj.SolrRequest;
|
||||||
|
import org.apache.solr.client.solrj.SolrServerException;
|
||||||
|
import org.apache.solr.client.solrj.embedded.JettySolrRunner;
|
||||||
|
import org.apache.solr.client.solrj.impl.BaseHttpSolrClient;
|
||||||
|
import org.apache.solr.client.solrj.impl.HttpSolrClient;
|
||||||
|
import org.apache.solr.client.solrj.request.CollectionAdminRequest;
|
||||||
|
import org.apache.solr.client.solrj.request.GenericSolrRequest;
|
||||||
|
import org.apache.solr.client.solrj.request.RequestWriter;
|
||||||
|
import org.apache.solr.client.solrj.request.V2Request;
|
||||||
|
import org.apache.solr.client.solrj.request.beans.Package;
|
||||||
|
import org.apache.solr.client.solrj.util.ClientUtils;
|
||||||
|
import org.apache.solr.cloud.ConfigRequest;
|
||||||
|
import org.apache.solr.cloud.MiniSolrCloudCluster;
|
||||||
|
import org.apache.solr.cloud.SolrCloudTestCase;
|
||||||
|
import org.apache.solr.common.MapWriterMap;
|
||||||
|
import org.apache.solr.common.NavigableObject;
|
||||||
|
import org.apache.solr.common.params.MapSolrParams;
|
||||||
|
import org.apache.solr.common.params.ModifiableSolrParams;
|
||||||
|
import org.apache.solr.common.params.SolrParams;
|
||||||
|
import org.apache.solr.common.util.Utils;
|
||||||
|
import org.apache.solr.filestore.TestDistribPackageStore;
|
||||||
|
import org.apache.solr.util.LogLevel;
|
||||||
|
import org.apache.zookeeper.CreateMode;
|
||||||
|
import org.apache.zookeeper.data.Stat;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import static org.apache.solr.common.cloud.ZkStateReader.SOLR_PKGS_PATH;
|
||||||
|
import static org.apache.solr.common.params.CommonParams.JAVABIN;
|
||||||
|
import static org.apache.solr.common.params.CommonParams.WT;
|
||||||
|
import static org.apache.solr.core.TestDynamicLoading.getFileContent;
|
||||||
|
import static org.apache.solr.filestore.TestDistribPackageStore.readFile;
|
||||||
|
|
||||||
|
@LogLevel("org.apache.solr.pkg.PackageLoader=DEBUG;org.apache.solr.pkg.PackageAPI=DEBUG")
|
||||||
|
public class TestPackages extends SolrCloudTestCase {
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testPluginLoading() throws Exception {
|
||||||
|
System.setProperty("enable.packages", "true");
|
||||||
|
MiniSolrCloudCluster cluster =
|
||||||
|
configureCluster(4)
|
||||||
|
.withJettyConfig(jetty -> jetty.enableV2(true))
|
||||||
|
.addConfig("conf", configset("cloud-minimal"))
|
||||||
|
.configure();
|
||||||
|
try {
|
||||||
|
String FILE1 = "/mypkg/runtimelibs.jar";
|
||||||
|
String FILE2 = "/mypkg/runtimelibs_v2.jar";
|
||||||
|
String FILE3 = "/mypkg/runtimelibs_v3.jar";
|
||||||
|
String COLLECTION_NAME = "testPluginLoadingColl";
|
||||||
|
byte[] derFile = readFile("cryptokeys/pub_key512.der");
|
||||||
|
cluster.getZkClient().makePath("/keys/exe", true);
|
||||||
|
cluster.getZkClient().create("/keys/exe/pub_key512.der", derFile, CreateMode.PERSISTENT, true);
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs.jar.bin", FILE1,
|
||||||
|
"L3q/qIGs4NaF6JiO0ZkMUFa88j0OmYc+I6O7BOdNuMct/xoZ4h73aZHZGc0+nmI1f/U3bOlMPINlSOM6LK3JpQ==");
|
||||||
|
|
||||||
|
Package.AddVersion add = new Package.AddVersion();
|
||||||
|
add.version = "1.0";
|
||||||
|
add.pkg = "mypkg";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE1});
|
||||||
|
V2Request req = new V2Request.Builder("/cluster/package")
|
||||||
|
.forceV2(true)
|
||||||
|
.withMethod(SolrRequest.METHOD.POST)
|
||||||
|
.withPayload(Collections.singletonMap("add", add))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
|
||||||
|
CollectionAdminRequest
|
||||||
|
.createCollection(COLLECTION_NAME, "conf", 2, 2)
|
||||||
|
.setMaxShardsPerNode(100)
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
cluster.waitForActiveCollection(COLLECTION_NAME, 2, 4);
|
||||||
|
|
||||||
|
TestDistribPackageStore.assertResponseValues(10,
|
||||||
|
() -> new V2Request.Builder("/cluster/package").
|
||||||
|
withMethod(SolrRequest.METHOD.GET)
|
||||||
|
.build().process(cluster.getSolrClient()),
|
||||||
|
Utils.makeMap(
|
||||||
|
":result:packages:mypkg[0]:version", "1.0",
|
||||||
|
":result:packages:mypkg[0]:files[0]", FILE1
|
||||||
|
));
|
||||||
|
|
||||||
|
String payload = "{\n" +
|
||||||
|
"'create-requesthandler' : { 'name' : '/runtime', 'class': 'mypkg:org.apache.solr.core.RuntimeLibReqHandler' }," +
|
||||||
|
"'create-searchcomponent' : { 'name' : 'get', 'class': 'mypkg:org.apache.solr.core.RuntimeLibSearchComponent' }," +
|
||||||
|
"'create-queryResponseWriter' : { 'name' : 'json1', 'class': 'mypkg:org.apache.solr.core.RuntimeLibResponseWriter' }" +
|
||||||
|
"}";
|
||||||
|
cluster.getSolrClient().request(new ConfigRequest(payload) {
|
||||||
|
@Override
|
||||||
|
public String getCollection() {
|
||||||
|
return COLLECTION_NAME;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "1.0" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "1.0" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "1.0" );
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
executeReq( "/" + COLLECTION_NAME + "/runtime?wt=javabin", cluster.getRandomJetty(random()),
|
||||||
|
Utils.JAVABINCONSUMER,
|
||||||
|
Utils.makeMap("class", "org.apache.solr.core.RuntimeLibReqHandler"));
|
||||||
|
|
||||||
|
executeReq( "/" + COLLECTION_NAME + "/get?wt=json", cluster.getRandomJetty(random()),
|
||||||
|
Utils.JSONCONSUMER,
|
||||||
|
Utils.makeMap("class", "org.apache.solr.core.RuntimeLibSearchComponent",
|
||||||
|
"Version","1"));
|
||||||
|
|
||||||
|
|
||||||
|
executeReq( "/" + COLLECTION_NAME + "/runtime?wt=json1", cluster.getRandomJetty(random()),
|
||||||
|
Utils.JSONCONSUMER,
|
||||||
|
Utils.makeMap("wt", "org.apache.solr.core.RuntimeLibResponseWriter"));
|
||||||
|
|
||||||
|
//now upload the second jar
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs_v2.jar.bin", FILE2,
|
||||||
|
"j+Rflxi64tXdqosIhbusqi6GTwZq8znunC/dzwcWW0/dHlFGKDurOaE1Nz9FSPJuXbHkVLj638yZ0Lp1ssnoYA==");
|
||||||
|
|
||||||
|
//add the version using package API
|
||||||
|
add.version = "1.1";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE2});
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
executeReq( "/" + COLLECTION_NAME + "/get?wt=json", cluster.getRandomJetty(random()),
|
||||||
|
Utils.JSONCONSUMER,
|
||||||
|
Utils.makeMap( "Version","2"));
|
||||||
|
|
||||||
|
|
||||||
|
//now upload the third jar
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs_v3.jar.bin", FILE3,
|
||||||
|
"a400n4T7FT+2gM0SC6+MfSOExjud8MkhTSFylhvwNjtWwUgKdPFn434Wv7Qc4QEqDVLhQoL3WqYtQmLPti0G4Q==");
|
||||||
|
|
||||||
|
add.version = "2.1";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE3});
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
//now let's verify that the classes are updated
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
executeReq( "/" + COLLECTION_NAME + "/runtime?wt=json", cluster.getRandomJetty(random()),
|
||||||
|
Utils.JSONCONSUMER,
|
||||||
|
Utils.makeMap("Version","2"));
|
||||||
|
|
||||||
|
|
||||||
|
Package.DelVersion delVersion = new Package.DelVersion();
|
||||||
|
delVersion.pkg = "mypkg";
|
||||||
|
delVersion.version = "1.0";
|
||||||
|
V2Request delete = new V2Request.Builder("/cluster/package")
|
||||||
|
.withMethod(SolrRequest.METHOD.POST)
|
||||||
|
.forceV2(true)
|
||||||
|
.withPayload(Collections.singletonMap("delete", delVersion))
|
||||||
|
.build();
|
||||||
|
delete.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
// now remove the hughest version. So, it will roll back to the next highest one
|
||||||
|
delVersion.version = "2.1";
|
||||||
|
delete.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
ModifiableSolrParams params = new ModifiableSolrParams();
|
||||||
|
params.add("collection", COLLECTION_NAME);
|
||||||
|
new GenericSolrRequest(SolrRequest.METHOD.POST, "/config/params", params ){
|
||||||
|
@Override
|
||||||
|
public RequestWriter.ContentWriter getContentWriter(String expectedType) {
|
||||||
|
return new RequestWriter.StringPayloadContentWriter("{set:{PKG_VERSIONS:{mypkg : '1.1'}}}",
|
||||||
|
ClientUtils.TEXT_JSON);
|
||||||
|
}
|
||||||
|
}.process(cluster.getSolrClient()) ;
|
||||||
|
|
||||||
|
add.version = "2.1";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE3});
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
//the collections mypkg is set to use version 1.1
|
||||||
|
//so no upgrade
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "1.1" );
|
||||||
|
|
||||||
|
new GenericSolrRequest(SolrRequest.METHOD.POST, "/config/params", params ){
|
||||||
|
@Override
|
||||||
|
public RequestWriter.ContentWriter getContentWriter(String expectedType) {
|
||||||
|
return new RequestWriter.StringPayloadContentWriter("{set:{PKG_VERSIONS:{mypkg : '2.1'}}}",
|
||||||
|
ClientUtils.TEXT_JSON);
|
||||||
|
}
|
||||||
|
}.process(cluster.getSolrClient()) ;
|
||||||
|
|
||||||
|
//now, let's force every collection using 'mypkg' to refresh
|
||||||
|
//so that it uses version 2.1
|
||||||
|
new V2Request.Builder("/cluster/package")
|
||||||
|
.withMethod(SolrRequest.METHOD.POST)
|
||||||
|
.withPayload("{refresh : mypkg}")
|
||||||
|
.forceV2(true)
|
||||||
|
.build()
|
||||||
|
.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "queryResponseWriter", "json1",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "searchComponent", "get",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
verifyCmponent(cluster.getSolrClient(),
|
||||||
|
COLLECTION_NAME, "requestHandler", "/runtime",
|
||||||
|
"mypkg", "2.1" );
|
||||||
|
|
||||||
|
} finally {
|
||||||
|
cluster.shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
private void executeReq(String uri, JettySolrRunner jetty, Utils.InputStreamConsumer parser, Map expected) throws Exception {
|
||||||
|
try(HttpSolrClient client = (HttpSolrClient) jetty.newClient()){
|
||||||
|
TestDistribPackageStore.assertResponseValues(10,
|
||||||
|
() -> {
|
||||||
|
Object o = Utils.executeGET(client.getHttpClient(),
|
||||||
|
jetty.getBaseUrl() + uri, parser);
|
||||||
|
if(o instanceof NavigableObject) return (NavigableObject) o;
|
||||||
|
if(o instanceof Map) return new MapWriterMap((Map) o);
|
||||||
|
throw new RuntimeException("Unknown response");
|
||||||
|
}, expected);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void verifyCmponent(SolrClient client, String COLLECTION_NAME,
|
||||||
|
String componentType, String componentName, String pkg, String version) throws Exception {
|
||||||
|
SolrParams params = new MapSolrParams((Map) Utils.makeMap("collection", COLLECTION_NAME,
|
||||||
|
WT, JAVABIN,
|
||||||
|
"componentName", componentName,
|
||||||
|
"meta", "true"));
|
||||||
|
|
||||||
|
String s = "queryResponseWriter";
|
||||||
|
GenericSolrRequest req1 = new GenericSolrRequest(SolrRequest.METHOD.GET,
|
||||||
|
"/config/" + componentType, params);
|
||||||
|
TestDistribPackageStore.assertResponseValues(10,
|
||||||
|
client,
|
||||||
|
req1, Utils.makeMap(
|
||||||
|
":config:" + componentType + ":" + componentName + ":_packageinfo_:package", pkg,
|
||||||
|
":config:" + componentType + ":" + componentName + ":_packageinfo_:version", version
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testAPI() throws Exception {
|
||||||
|
System.setProperty("enable.packages", "true");
|
||||||
|
MiniSolrCloudCluster cluster =
|
||||||
|
configureCluster(4)
|
||||||
|
.withJettyConfig(jetty -> jetty.enableV2(true))
|
||||||
|
.addConfig("conf", configset("cloud-minimal"))
|
||||||
|
.configure();
|
||||||
|
try {
|
||||||
|
String errPath = "/error/details[0]/errorMessages[0]";
|
||||||
|
String FILE1 = "/mypkg/v.0.12/jar_a.jar";
|
||||||
|
String FILE2 = "/mypkg/v.0.12/jar_b.jar";
|
||||||
|
String FILE3 = "/mypkg/v.0.13/jar_a.jar";
|
||||||
|
|
||||||
|
Package.AddVersion add = new Package.AddVersion();
|
||||||
|
add.version = "0.12";
|
||||||
|
add.pkg = "test_pkg";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE1, FILE2});
|
||||||
|
V2Request req = new V2Request.Builder("/cluster/package")
|
||||||
|
.forceV2(true)
|
||||||
|
.withMethod(SolrRequest.METHOD.POST)
|
||||||
|
.withPayload(Collections.singletonMap("add", add))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
//the files is not yet there. The command should fail with error saying "No such file"
|
||||||
|
expectError(req, cluster.getSolrClient(), errPath, "No such file :");
|
||||||
|
|
||||||
|
|
||||||
|
//post the jar file. No signature is sent
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs.jar.bin", FILE1, null);
|
||||||
|
|
||||||
|
|
||||||
|
add.files = Arrays.asList(new String[]{FILE1});
|
||||||
|
expectError(req, cluster.getSolrClient(), errPath,
|
||||||
|
FILE1 + " has no signature");
|
||||||
|
//now we upload the keys
|
||||||
|
byte[] derFile = readFile("cryptokeys/pub_key512.der");
|
||||||
|
cluster.getZkClient().makePath("/keys/exe", true);
|
||||||
|
cluster.getZkClient().create("/keys/exe/pub_key512.der", derFile, CreateMode.PERSISTENT, true);
|
||||||
|
//and upload the same file with a different name but it has proper signature
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs.jar.bin", FILE2,
|
||||||
|
"L3q/qIGs4NaF6JiO0ZkMUFa88j0OmYc+I6O7BOdNuMct/xoZ4h73aZHZGc0+nmI1f/U3bOlMPINlSOM6LK3JpQ==");
|
||||||
|
// with correct signature
|
||||||
|
//after uploading the file, let's delete the keys to see if we get proper error message
|
||||||
|
cluster.getZkClient().delete("/keys/exe/pub_key512.der", -1, true);
|
||||||
|
add.files = Arrays.asList(new String[]{FILE2});
|
||||||
|
expectError(req, cluster.getSolrClient(), errPath,
|
||||||
|
"ZooKeeper does not have any public keys");
|
||||||
|
|
||||||
|
//Now lets' put the keys back
|
||||||
|
cluster.getZkClient().create("/keys/exe/pub_key512.der", derFile, CreateMode.PERSISTENT, true);
|
||||||
|
|
||||||
|
//this time we have a file with proper signature, public keys are in ZK
|
||||||
|
// so the add {} command should succeed
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
|
||||||
|
//Now verify the data in ZK
|
||||||
|
TestDistribPackageStore.assertResponseValues(1,
|
||||||
|
() -> new MapWriterMap((Map) Utils.fromJSON(cluster.getZkClient().getData(SOLR_PKGS_PATH,
|
||||||
|
null, new Stat(), true))),
|
||||||
|
Utils.makeMap(
|
||||||
|
":packages:test_pkg[0]:version", "0.12",
|
||||||
|
":packages:test_pkg[0]:files[0]", FILE1
|
||||||
|
));
|
||||||
|
|
||||||
|
//post a new jar with a proper signature
|
||||||
|
postFileAndWait(cluster, "runtimecode/runtimelibs_v2.jar.bin", FILE3,
|
||||||
|
"j+Rflxi64tXdqosIhbusqi6GTwZq8znunC/dzwcWW0/dHlFGKDurOaE1Nz9FSPJuXbHkVLj638yZ0Lp1ssnoYA==");
|
||||||
|
|
||||||
|
|
||||||
|
//this time we are adding the second version of the package (0.13)
|
||||||
|
add.version = "0.13";
|
||||||
|
add.pkg = "test_pkg";
|
||||||
|
add.files = Arrays.asList(new String[]{FILE3});
|
||||||
|
|
||||||
|
//this request should succeed
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
//no verify the data (/packages.json) in ZK
|
||||||
|
TestDistribPackageStore.assertResponseValues(1,
|
||||||
|
() -> new MapWriterMap((Map) Utils.fromJSON(cluster.getZkClient().getData(SOLR_PKGS_PATH,
|
||||||
|
null, new Stat(), true))),
|
||||||
|
Utils.makeMap(
|
||||||
|
":packages:test_pkg[1]:version", "0.13",
|
||||||
|
":packages:test_pkg[1]:files[0]", FILE3
|
||||||
|
));
|
||||||
|
|
||||||
|
//Now we will just delete one version
|
||||||
|
Package.DelVersion delVersion = new Package.DelVersion();
|
||||||
|
delVersion.version = "0.1";//this version does not exist
|
||||||
|
delVersion.pkg = "test_pkg";
|
||||||
|
req = new V2Request.Builder("/cluster/package")
|
||||||
|
.forceV2(true)
|
||||||
|
.withMethod(SolrRequest.METHOD.POST)
|
||||||
|
.withPayload(Collections.singletonMap("delete", delVersion))
|
||||||
|
.build();
|
||||||
|
|
||||||
|
//we are expecting an error
|
||||||
|
expectError(req, cluster.getSolrClient(), errPath, "No such version:");
|
||||||
|
|
||||||
|
delVersion.version = "0.12";//correct version. Should succeed
|
||||||
|
req.process(cluster.getSolrClient());
|
||||||
|
//Verify with ZK that the data is correcy
|
||||||
|
TestDistribPackageStore.assertResponseValues(1,
|
||||||
|
() -> new MapWriterMap((Map) Utils.fromJSON(cluster.getZkClient().getData(SOLR_PKGS_PATH,
|
||||||
|
null, new Stat(), true))),
|
||||||
|
Utils.makeMap(
|
||||||
|
":packages:test_pkg[0]:version", "0.13",
|
||||||
|
":packages:test_pkg[0]:files[0]", FILE2
|
||||||
|
));
|
||||||
|
|
||||||
|
|
||||||
|
//So far we have been verifying the details with ZK directly
|
||||||
|
//use the package read API to verify with each node that it has the correct data
|
||||||
|
for (JettySolrRunner jetty : cluster.getJettySolrRunners()) {
|
||||||
|
String path = jetty.getBaseUrl().toString().replace("/solr", "/api") + "/cluster/package?wt=javabin";
|
||||||
|
TestDistribPackageStore.assertResponseValues(10, new Callable<NavigableObject>() {
|
||||||
|
@Override
|
||||||
|
public NavigableObject call() throws Exception {
|
||||||
|
try (HttpSolrClient solrClient = (HttpSolrClient) jetty.newClient()) {
|
||||||
|
return (NavigableObject) Utils.executeGET(solrClient.getHttpClient(), path, Utils.JAVABINCONSUMER);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, Utils.makeMap(
|
||||||
|
":result:packages:test_pkg[0]:version", "0.13",
|
||||||
|
":result:packages:test_pkg[0]:files[0]", FILE3
|
||||||
|
));
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
cluster.shutdown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void postFileAndWait(MiniSolrCloudCluster cluster, String fname, String path, String sig) throws Exception {
|
||||||
|
ByteBuffer fileContent = getFileContent(fname);
|
||||||
|
String sha512 = DigestUtils.sha512Hex(fileContent.array());
|
||||||
|
|
||||||
|
TestDistribPackageStore.postFile(cluster.getSolrClient(),
|
||||||
|
fileContent,
|
||||||
|
path, sig);// has file, but no signature
|
||||||
|
|
||||||
|
TestDistribPackageStore.waitForAllNodesHaveFile(cluster, path, Utils.makeMap(
|
||||||
|
":files:" + path + ":sha512",
|
||||||
|
sha512
|
||||||
|
), false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void expectError(V2Request req, SolrClient client, String errPath, String expectErrorMsg) throws IOException, SolrServerException {
|
||||||
|
try {
|
||||||
|
req.process(client);
|
||||||
|
fail("should have failed with message : " + expectErrorMsg);
|
||||||
|
} catch (BaseHttpSolrClient.RemoteExecutionException e) {
|
||||||
|
String msg = e.getMetaData()._getStr(errPath, "");
|
||||||
|
assertTrue("should have failed with message: " + expectErrorMsg + "actual message : " + msg,
|
||||||
|
msg.contains(expectErrorMsg)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -31,7 +31,7 @@ public class RuntimeUrp extends SimpleUpdateProcessorFactory {
|
||||||
List<String> names = new ArrayList<>();
|
List<String> names = new ArrayList<>();
|
||||||
for (UpdateRequestProcessorFactory p : processorChain.getProcessors()) {
|
for (UpdateRequestProcessorFactory p : processorChain.getProcessors()) {
|
||||||
if (p instanceof UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder.LazyUpdateRequestProcessorFactory) {
|
if (p instanceof UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder.LazyUpdateRequestProcessorFactory) {
|
||||||
p = ((UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder.LazyUpdateRequestProcessorFactory) p).delegate;
|
p = ((UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder.LazyUpdateRequestProcessorFactory) p).getDelegate();
|
||||||
}
|
}
|
||||||
names.add(p.getClass().getSimpleName());
|
names.add(p.getClass().getSimpleName());
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,46 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.apache.solr.client.solrj.request.beans;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
|
import org.apache.solr.common.util.ReflectMapWriter;
|
||||||
|
|
||||||
|
/**Just a container class for POJOs used in Package APIs
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
public class Package {
|
||||||
|
public static class AddVersion implements ReflectMapWriter {
|
||||||
|
@JsonProperty(value = "package", required = true)
|
||||||
|
public String pkg;
|
||||||
|
@JsonProperty(required = true)
|
||||||
|
public String version;
|
||||||
|
@JsonProperty(required = true)
|
||||||
|
public List<String> files;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class DelVersion implements ReflectMapWriter {
|
||||||
|
@JsonProperty(value = "package", required = true)
|
||||||
|
public String pkg;
|
||||||
|
@JsonProperty(required = true)
|
||||||
|
public String version;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,23 @@
|
||||||
|
/*
|
||||||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||||||
|
* contributor license agreements. See the NOTICE file distributed with
|
||||||
|
* this work for additional information regarding copyright ownership.
|
||||||
|
* The ASF licenses this file to You under the Apache License, Version 2.0
|
||||||
|
* (the "License"); you may not use this file except in compliance with
|
||||||
|
* the License. You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data objects used in V2 Requests with jackson bindings
|
||||||
|
*/
|
||||||
|
package org.apache.solr.client.solrj.request.beans;
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,7 @@ import java.nio.file.Path;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
import java.util.concurrent.RejectedExecutionException;
|
import java.util.concurrent.RejectedExecutionException;
|
||||||
|
import java.util.function.BiFunction;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
import java.util.function.Predicate;
|
import java.util.function.Predicate;
|
||||||
import java.util.regex.Pattern;
|
import java.util.regex.Pattern;
|
||||||
|
@ -353,6 +354,10 @@ public class SolrZkClient implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void atomicUpdate(String path, Function<byte[], byte[]> editor) throws KeeperException, InterruptedException {
|
public void atomicUpdate(String path, Function<byte[], byte[]> editor) throws KeeperException, InterruptedException {
|
||||||
|
atomicUpdate(path, (stat, bytes) -> editor.apply(bytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void atomicUpdate(String path, BiFunction<Stat , byte[], byte[]> editor) throws KeeperException, InterruptedException {
|
||||||
for (; ; ) {
|
for (; ; ) {
|
||||||
byte[] modified = null;
|
byte[] modified = null;
|
||||||
byte[] zkData = null;
|
byte[] zkData = null;
|
||||||
|
@ -360,7 +365,7 @@ public class SolrZkClient implements Closeable {
|
||||||
try {
|
try {
|
||||||
if (exists(path, true)) {
|
if (exists(path, true)) {
|
||||||
zkData = getData(path, null, s, true);
|
zkData = getData(path, null, s, true);
|
||||||
modified = editor.apply(zkData);
|
modified = editor.apply(s, zkData);
|
||||||
if (modified == null) {
|
if (modified == null) {
|
||||||
//no change , no need to persist
|
//no change , no need to persist
|
||||||
return;
|
return;
|
||||||
|
@ -368,7 +373,7 @@ public class SolrZkClient implements Closeable {
|
||||||
setData(path, modified, s.getVersion(), true);
|
setData(path, modified, s.getVersion(), true);
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
modified = editor.apply(null);
|
modified = editor.apply(s,null);
|
||||||
if (modified == null) {
|
if (modified == null) {
|
||||||
//no change , no need to persist
|
//no change , no need to persist
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -118,6 +118,7 @@ public class ZkStateReader implements SolrCloseable {
|
||||||
public static final String SOLR_AUTOSCALING_TRIGGER_STATE_PATH = "/autoscaling/triggerState";
|
public static final String SOLR_AUTOSCALING_TRIGGER_STATE_PATH = "/autoscaling/triggerState";
|
||||||
public static final String SOLR_AUTOSCALING_NODE_ADDED_PATH = "/autoscaling/nodeAdded";
|
public static final String SOLR_AUTOSCALING_NODE_ADDED_PATH = "/autoscaling/nodeAdded";
|
||||||
public static final String SOLR_AUTOSCALING_NODE_LOST_PATH = "/autoscaling/nodeLost";
|
public static final String SOLR_AUTOSCALING_NODE_LOST_PATH = "/autoscaling/nodeLost";
|
||||||
|
public static final String SOLR_PKGS_PATH = "/packages.json";
|
||||||
|
|
||||||
public static final String DEFAULT_SHARD_PREFERENCES = "defaultShardPreferences";
|
public static final String DEFAULT_SHARD_PREFERENCES = "defaultShardPreferences";
|
||||||
public static final String REPLICATION_FACTOR = "replicationFactor";
|
public static final String REPLICATION_FACTOR = "replicationFactor";
|
||||||
|
|
|
@ -23,7 +23,7 @@ import java.lang.reflect.Modifier;
|
||||||
|
|
||||||
import com.fasterxml.jackson.annotation.JsonProperty;
|
import com.fasterxml.jackson.annotation.JsonProperty;
|
||||||
import org.apache.solr.common.MapWriter;
|
import org.apache.solr.common.MapWriter;
|
||||||
|
// An implementation of MapWriter which is annotated with Jackson annotations
|
||||||
public interface ReflectMapWriter extends MapWriter {
|
public interface ReflectMapWriter extends MapWriter {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
Loading…
Reference in New Issue