SOLR-13650: Support for named global classloaders

This commit is contained in:
Noble Paul 2019-08-20 05:00:38 +10:00 committed by noble
parent ee50c9d5ac
commit b0a11ba5c7
26 changed files with 560 additions and 252 deletions

View File

@ -20,12 +20,12 @@ import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import org.apache.lucene.index.PointValues; import org.apache.lucene.index.PointValues;
import org.apache.lucene.search.BooleanClause.Occur;
import org.apache.lucene.search.BooleanQuery; import org.apache.lucene.search.BooleanQuery;
import org.apache.lucene.search.BoostQuery; import org.apache.lucene.search.BoostQuery;
import org.apache.lucene.search.PointInSetQuery; import org.apache.lucene.search.PointInSetQuery;
import org.apache.lucene.search.PointRangeQuery; import org.apache.lucene.search.PointRangeQuery;
import org.apache.lucene.search.Query; import org.apache.lucene.search.Query;
import org.apache.lucene.search.BooleanClause.Occur;
import org.apache.lucene.util.BytesRef; import org.apache.lucene.util.BytesRef;
import org.apache.lucene.util.NumericUtils; import org.apache.lucene.util.NumericUtils;

View File

@ -16,9 +16,34 @@
*/ */
package org.apache.lucene.queryparser.xml; package org.apache.lucene.queryparser.xml;
import javax.xml.XMLConstants;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import java.io.InputStream;
import java.util.Locale;
import org.apache.lucene.analysis.Analyzer; import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.queryparser.classic.QueryParser; import org.apache.lucene.queryparser.classic.QueryParser;
import org.apache.lucene.queryparser.xml.builders.*; import org.apache.lucene.queryparser.xml.builders.BooleanQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.BoostingTermBuilder;
import org.apache.lucene.queryparser.xml.builders.ConstantScoreQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.DisjunctionMaxQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.MatchAllDocsQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.PointRangeQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.RangeQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanFirstBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanNearBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanNotBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanOrBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanOrTermsBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanPositionRangeBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.SpanQueryBuilderFactory;
import org.apache.lucene.queryparser.xml.builders.SpanTermBuilder;
import org.apache.lucene.queryparser.xml.builders.TermQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.TermsQueryBuilder;
import org.apache.lucene.queryparser.xml.builders.UserInputQueryBuilder;
import org.apache.lucene.search.Query; import org.apache.lucene.search.Query;
import org.apache.lucene.search.spans.SpanQuery; import org.apache.lucene.search.spans.SpanQuery;
import org.w3c.dom.Document; import org.w3c.dom.Document;
@ -27,14 +52,6 @@ import org.xml.sax.EntityResolver;
import org.xml.sax.ErrorHandler; import org.xml.sax.ErrorHandler;
import org.xml.sax.SAXException; import org.xml.sax.SAXException;
import javax.xml.XMLConstants;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import java.io.InputStream;
import java.util.Locale;
/** /**
* Assembles a QueryBuilder which uses only core Lucene Query objects * Assembles a QueryBuilder which uses only core Lucene Query objects
*/ */

View File

@ -16,6 +16,9 @@
*/ */
package org.apache.lucene.queryparser.xml; package org.apache.lucene.queryparser.xml;
import java.io.IOException;
import java.io.InputStream;
import org.apache.lucene.analysis.Analyzer; import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.analysis.MockAnalyzer; import org.apache.lucene.analysis.MockAnalyzer;
import org.apache.lucene.analysis.MockTokenFilter; import org.apache.lucene.analysis.MockTokenFilter;
@ -32,9 +35,6 @@ import org.apache.lucene.util.LuceneTestCase;
import org.junit.AfterClass; import org.junit.AfterClass;
import org.xml.sax.SAXException; import org.xml.sax.SAXException;
import java.io.IOException;
import java.io.InputStream;
public class TestCoreParser extends LuceneTestCase { public class TestCoreParser extends LuceneTestCase {
final private static String defaultField = "contents"; final private static String defaultField = "contents";

View File

@ -24,13 +24,13 @@ import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
import java.util.List; import java.util.List;
import java.util.Locale;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.TreeMap; import java.util.TreeMap;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicLong;
import java.util.Locale;
import org.apache.solr.client.solrj.cloud.SolrCloudManager; import org.apache.solr.client.solrj.cloud.SolrCloudManager;
import org.apache.solr.client.solrj.cloud.autoscaling.ReplicaInfo; import org.apache.solr.client.solrj.cloud.autoscaling.ReplicaInfo;

View File

@ -219,7 +219,7 @@ public class CoreContainer {
protected volatile AutoscalingHistoryHandler autoscalingHistoryHandler; protected volatile AutoscalingHistoryHandler autoscalingHistoryHandler;
private final LibListener clusterPropertiesListener = new LibListener(this); private final PackageManager clusterPropertiesListener = new PackageManager(this);
// Bits for the state variable. // Bits for the state variable.
@ -1802,7 +1802,7 @@ public class CoreContainer {
return containerHandlers; return containerHandlers;
} }
public LibListener getClusterPropertiesListener(){ public PackageManager getPackageManager(){
return clusterPropertiesListener; return clusterPropertiesListener;
} }

View File

@ -54,6 +54,14 @@ public class MemClassLoader extends ClassLoader implements AutoCloseable, Resour
this.libs = libs; this.libs = libs;
} }
public int getZnodeVersion(){
int result = -1;
for (RuntimeLib lib : libs) {
if(lib.znodeVersion > result) result = lib.znodeVersion;
}
return result;
}
synchronized void loadRemoteJars() { synchronized void loadRemoteJars() {
if (allJarsLoaded) return; if (allJarsLoaded) return;
int count = 0; int count = 0;

View File

@ -17,8 +17,8 @@
package org.apache.solr.core; package org.apache.solr.core;
import java.io.IOException;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.Collections; import java.util.Collections;
import java.util.HashMap; import java.util.HashMap;
@ -27,11 +27,9 @@ import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.stream.Collectors; import java.util.stream.Collectors;
import com.google.common.collect.ImmutableMap;
import org.apache.lucene.analysis.util.ResourceLoader; import org.apache.lucene.analysis.util.ResourceLoader;
import org.apache.solr.api.Api; import org.apache.solr.api.Api;
import org.apache.solr.api.V2HttpCall; import org.apache.solr.api.V2HttpCall;
import org.apache.solr.common.IteratorWriter;
import org.apache.solr.common.MapWriter; import org.apache.solr.common.MapWriter;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.cloud.ClusterPropertiesListener; import org.apache.solr.common.cloud.ClusterPropertiesListener;
@ -50,43 +48,83 @@ import org.apache.solr.util.plugin.PluginInfoInitialized;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import static org.apache.solr.common.params.CommonParams.NAME;
import static org.apache.solr.common.params.CommonParams.PACKAGE;
import static org.apache.solr.common.params.CommonParams.VERSION; import static org.apache.solr.common.params.CommonParams.VERSION;
import static org.apache.solr.core.RuntimeLib.SHA512; import static org.apache.solr.core.RuntimeLib.SHA512;
public class LibListener implements ClusterPropertiesListener { public class PackageManager implements ClusterPropertiesListener {
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
private final CoreContainer coreContainer; private final CoreContainer coreContainer;
Map<String, RuntimeLib> runtimeLibs = new HashMap<>(); private Map<String, Package> pkgs = new HashMap<>();
MemClassLoader memClassLoader;
final ExtHandler extHandler; final ExtHandler extHandler;
private int myversion = -1; private int myversion = -1;
LibListener(CoreContainer coreContainer) { public int getZNodeVersion(String pkg) {
Package p = pkgs.get(pkg);
return p == null ? -1 : p.lib.getZnodeVersion();
}
static class Package implements MapWriter {
final RuntimeLib lib;
final MemClassLoader loader;
final String name;
@Override
public void writeMap(EntryWriter ew) throws IOException {
lib.writeMap(ew);
}
Package(RuntimeLib lib, MemClassLoader loader, int zkVersion, String name) {
this.lib = lib;
this.loader = loader;
this.name = name;
}
public String getName() {
return name;
}
public boolean isModified(Map map) {
if ((Objects.equals(lib.getSha512(), (map).get(SHA512)) &&
Objects.equals(lib.getSig(), (map).get(SHA512)))) {
return false;
} else {
return true;
}
}
}
PackageManager(CoreContainer coreContainer) {
this.coreContainer = coreContainer; this.coreContainer = coreContainer;
extHandler = new ExtHandler(this); extHandler = new ExtHandler(this);
} }
public <T> T newInstance(String cName, Class<T> expectedType) { public <T> T newInstance(String cName, Class<T> expectedType, String pkg) {
try { try {
return coreContainer.getResourceLoader().newInstance(cName, expectedType, return coreContainer.getResourceLoader().newInstance(cName, expectedType,
null, new Class[]{CoreContainer.class}, new Object[]{coreContainer}); null, new Class[]{CoreContainer.class}, new Object[]{coreContainer});
} catch (SolrException e) { } catch (SolrException e) {
if (memClassLoader != null) { Package p = pkgs.get(pkg);
if (p != null) {
try { try {
Class<? extends T> klas = memClassLoader.findClass(cName, expectedType); Class<? extends T> klas = p.loader.findClass(cName, expectedType);
try { try {
return klas.getConstructor(CoreContainer.class).newInstance(coreContainer); return klas.getConstructor(CoreContainer.class).newInstance(coreContainer);
} catch (NoSuchMethodException ex) { } catch (NoSuchMethodException ex) {
return klas.getConstructor().newInstance(); return klas.getConstructor().newInstance();
} }
} catch (Exception ex) { } catch (Exception ex) {
if (!memClassLoader.getErrors().isEmpty()) { if (!p.loader.getErrors().isEmpty()) {
//some libraries were no loaded due to some errors. May the class was there in those libraries //some libraries were no loaded due to some errors. May the class was there in those libraries
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, throw new SolrException(SolrException.ErrorCode.SERVER_ERROR,
"There were errors loading some libraries: " + StrUtils.join(memClassLoader.getErrors(), ','), ex); "There were errors loading some libraries: " + StrUtils.join(p.loader.getErrors(), ','), ex);
} }
//there were no errors in loading any libraries. The class was probably not suppoed to be there in those libraries //there were no errors in loading any libraries. The class was probably not suppoed to be there in those libraries
// so throw the original exception // so throw the original exception
@ -101,59 +139,72 @@ public class LibListener implements ClusterPropertiesListener {
@Override @Override
public boolean onChange(Map<String, Object> properties) { public boolean onChange(Map<String, Object> properties) {
log.info("clusterprops.json changed , version {}", coreContainer.getZkController().getZkStateReader().getClusterPropsVersion()); log.info("clusterprops.json changed , version {}", coreContainer.getZkController().getZkStateReader().getClusterPropsVersion());
boolean forceReload = updateRuntimeLibs(properties); int v = coreContainer.getZkController().getZkStateReader().getClusterPropsVersion();
extHandler.updateReqHandlers(properties, forceReload); boolean modified = updatePackages(properties, v);
extHandler.updateReqHandlers(properties, modified);
for (SolrCore core : coreContainer.solrCores.getCores()) { for (SolrCore core : coreContainer.solrCores.getCores()) {
core.globalClassLoaderChanged(); pkgs.forEach((s, pkg) -> core.packageUpdated(pkg.lib));
} }
myversion = coreContainer.getZkController().getZkStateReader().getClusterPropsVersion(); myversion = v;
return false; return false;
} }
private boolean updateRuntimeLibs(Map<String, Object> properties) { private boolean updatePackages(Map<String, Object> properties, int ver) {
Map m = (Map) properties.getOrDefault(RuntimeLib.TYPE, Collections.emptyMap()); Map m = (Map) properties.getOrDefault(PACKAGE, Collections.emptyMap());
if (runtimeLibs.isEmpty() && m.isEmpty()) return false; if (pkgs.isEmpty() && m.isEmpty()) return false;
boolean needsReload[] = new boolean[1]; boolean needsReload[] = new boolean[1];
if (m.size() == runtimeLibs.size()) { if (m.size() == pkgs.size()) {
m.forEach((k, v) -> { m.forEach((k, v) -> {
if (v instanceof Map) { if (v instanceof Map) {
if (!runtimeLibs.containsKey(k)) needsReload[0] = true; Package pkg = pkgs.get(k);
RuntimeLib rtl = runtimeLibs.get(k); if (pkg == null || pkg.isModified((Map) v)) {
if (rtl == null || !Objects.equals(rtl.getSha512(), ((Map) v).get(SHA512))) {
needsReload[0] = true; needsReload[0] = true;
} }
} }
}); });
} else { } else {
needsReload[0] = true; needsReload[0] = true;
} }
if (needsReload[0]) { if (needsReload[0]) {
createNewClassLoader(m); createNewClassLoader(m, ver);
} }
return needsReload[0]; return needsReload[0];
} }
public ResourceLoader getResourceLoader() {
return memClassLoader == null ? coreContainer.getResourceLoader() : memClassLoader; public ResourceLoader getResourceLoader(String pkg) {
Package p = pkgs.get(pkg);
return p == null ? coreContainer.getResourceLoader() : p.loader;
} }
void createNewClassLoader(Map m) {
void createNewClassLoader(Map m, int ver) {
boolean[] loadedAll = new boolean[1]; boolean[] loadedAll = new boolean[1];
loadedAll[0] = true; loadedAll[0] = true;
Map<String, RuntimeLib> libMap = new LinkedHashMap<>(); Map<String, Package> newPkgs = new LinkedHashMap<>();
m.forEach((k, v) -> { m.forEach((k, v) -> {
if (v instanceof Map) { if (v instanceof Map) {
Map map = new HashMap((Map) v); Map map = new HashMap((Map) v);
map.put(CoreAdminParams.NAME, String.valueOf(k)); map.put(CoreAdminParams.NAME, String.valueOf(k));
String name = (String) k;
Package existing = pkgs.get(name);
if (existing != null && !existing.isModified(map)) {
//this package has not changed
newPkgs.put(name, existing);
}
RuntimeLib lib = new RuntimeLib(coreContainer); RuntimeLib lib = new RuntimeLib(coreContainer);
lib.znodeVersion = ver;
try { try {
lib.init(new PluginInfo(null, map)); lib.init(new PluginInfo(RuntimeLib.TYPE, map));
if (lib.getUrl() == null) { if (lib.getUrl() == null) {
log.error("Unable to initialize runtimeLib : " + Utils.toJSONString(v)); log.error("Unable to initialize runtimeLib : " + Utils.toJSONString(v));
loadedAll[0] = false; loadedAll[0] = false;
} }
lib.loadJar(); lib.loadJar();
libMap.put(lib.getName(), lib);
newPkgs.put(name, new Package(lib,
new MemClassLoader(Collections.singletonList(lib), coreContainer.getResourceLoader()),
ver, name));
} catch (Exception e) { } catch (Exception e) {
log.error("error loading a runtimeLib " + Utils.toJSONString(v), e); log.error("error loading a runtimeLib " + Utils.toJSONString(v), e);
loadedAll[0] = false; loadedAll[0] = false;
@ -163,20 +214,20 @@ public class LibListener implements ClusterPropertiesListener {
}); });
if (loadedAll[0]) { if (loadedAll[0]) {
log.info("Libraries changed. New memclassloader created with jars {}", libMap.values().stream().map(runtimeLib -> runtimeLib.getUrl()).collect(Collectors.toList())); log.info("Libraries changed. New memclassloader created with jars {}",
this.memClassLoader = new MemClassLoader(new ArrayList<>(libMap.values()), coreContainer.getResourceLoader()); newPkgs.values().stream().map(it -> it.lib.getUrl()).collect(Collectors.toList()));
this.runtimeLibs = libMap; this.pkgs = newPkgs;
} }
} }
static class ExtHandler extends RequestHandlerBase implements PermissionNameProvider { static class ExtHandler extends RequestHandlerBase implements PermissionNameProvider {
final LibListener libListener; final PackageManager packageManager;
private Map<String, SolrRequestHandler> customHandlers = new HashMap<>(); private Map<String, Handler> customHandlers = new HashMap<>();
ExtHandler(LibListener libListener) { ExtHandler(PackageManager packageManager) {
this.libListener = libListener; this.packageManager = packageManager;
} }
@ -184,15 +235,19 @@ public class LibListener implements ClusterPropertiesListener {
public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) { public void handleRequestBody(SolrQueryRequest req, SolrQueryResponse rsp) {
int v = req.getParams().getInt(ConfigOverlay.ZNODEVER, -1); int v = req.getParams().getInt(ConfigOverlay.ZNODEVER, -1);
if (v >= 0) { if (v >= 0) {
log.debug("expected version : {} , my version {}", v, libListener.myversion ); log.debug("expected version : {} , my version {}", v, packageManager.myversion);
ZkStateReader zkStateReader = libListener.coreContainer.getZkController().getZkStateReader(); ZkStateReader zkStateReader = packageManager.coreContainer.getZkController().getZkStateReader();
try {
zkStateReader.forceRefreshClusterProps(v); zkStateReader.forceRefreshClusterProps(v);
} catch (SolrException e) {
log.error("Error refreshing state ", e);
throw e;
}
} }
rsp.add("metadata", (MapWriter) ew -> ew.putIfNotNull(VERSION, rsp.add("metadata", (MapWriter) ew -> ew.putIfNotNull(VERSION,
libListener.coreContainer.getZkController().zkStateReader.getClusterPropsVersion())); packageManager.coreContainer.getZkController().zkStateReader.getClusterPropsVersion()));
rsp.add(RuntimeLib.TYPE, libListener.runtimeLibs.values()); rsp.add(RuntimeLib.TYPE, packageManager.pkgs.values());
rsp.add(SolrRequestHandler.TYPE, rsp.add(SolrRequestHandler.TYPE, customHandlers.values());
(IteratorWriter) iw -> customHandlers.forEach((s, h) -> iw.addNoEx(ImmutableMap.of(s, h.getClass().getName()))));
} }
@ -206,13 +261,13 @@ public class LibListener implements ClusterPropertiesListener {
handleRequestBody(req, rsp); handleRequestBody(req, rsp);
return; return;
} }
SolrRequestHandler handler = customHandlers.get(name); Handler handler = customHandlers.get(name);
if (handler == null) { if (handler == null) {
String err = StrUtils.formatString(" No such handler: {0}, available handlers : {1}" , name, customHandlers.keySet()); String err = StrUtils.formatString(" No such handler: {0}, available handlers : {1}", name, customHandlers.keySet());
log.error(err); log.error(err);
throw new SolrException(SolrException.ErrorCode.NOT_FOUND, err); throw new SolrException(SolrException.ErrorCode.NOT_FOUND, err);
} }
handler.handleRequest(req, rsp); handler.handler.handleRequest(req, rsp);
} }
}); });
} }
@ -224,24 +279,37 @@ public class LibListener implements ClusterPropertiesListener {
if (customHandlers.size() == m.size() && customHandlers.keySet().containsAll(m.keySet())) hasChanged = false; if (customHandlers.size() == m.size() && customHandlers.keySet().containsAll(m.keySet())) hasChanged = false;
if (forceReload || hasChanged) { if (forceReload || hasChanged) {
log.debug("RequestHandlers being reloaded : {}", m.keySet()); log.debug("RequestHandlers being reloaded : {}", m.keySet());
Map<String, SolrRequestHandler> newCustomHandlers = new HashMap<>(); Map<String, Handler> newCustomHandlers = new HashMap<>();
m.forEach((k, v) -> { m.forEach((k, v) -> {
if (v instanceof Map) { if (v instanceof Map) {
String klas = (String) ((Map) v).get(FieldType.CLASS_NAME); Map metaData = (Map) v;
Handler existing = customHandlers.get(k);
String name = (String) k;
if (existing == null || existing.shouldReload(metaData, packageManager.pkgs)) {
String klas = (String) metaData.get(FieldType.CLASS_NAME);
if (klas != null) { if (klas != null) {
SolrRequestHandler inst = libListener.newInstance(klas, SolrRequestHandler.class); String pkg = (String) metaData.get(PACKAGE);
SolrRequestHandler inst = packageManager.newInstance(klas, SolrRequestHandler.class, pkg);
if (inst instanceof PluginInfoInitialized) { if (inst instanceof PluginInfoInitialized) {
((PluginInfoInitialized) inst).init(new PluginInfo(SolrRequestHandler.TYPE, (Map) v)); ((PluginInfoInitialized) inst).init(new PluginInfo(SolrRequestHandler.TYPE, metaData));
} }
newCustomHandlers.put((String) k, inst); Package p = packageManager.pkgs.get(pkg);
newCustomHandlers.put(name, new Handler(inst, pkg, p == null ? -1 : p.lib.getZnodeVersion(), metaData, name));
} else {
log.error("Invalid requestHandler {}", Utils.toJSONString(v));
} }
} else {
newCustomHandlers.put(name, existing);
}
} else { } else {
log.error("Invalid data for requestHandler : {} , {}", k, v); log.error("Invalid data for requestHandler : {} , {}", k, v);
} }
}); });
log.debug("Registering request handlers {} ", newCustomHandlers.keySet()); log.debug("Registering request handlers {} ", newCustomHandlers.keySet());
Map<String, SolrRequestHandler> old = customHandlers; Map<String, Handler> old = customHandlers;
customHandlers = newCustomHandlers; customHandlers = newCustomHandlers;
old.forEach((s, h) -> PluginBag.closeQuietly(h)); old.forEach((s, h) -> PluginBag.closeQuietly(h));
} }
@ -265,9 +333,41 @@ public class LibListener implements ClusterPropertiesListener {
@Override @Override
public Name getPermissionName(AuthorizationContext request) { public Name getPermissionName(AuthorizationContext request) {
if(request.getResource().endsWith("/node/ext")) return Name.COLL_READ_PERM; if (request.getResource().endsWith("/node/ext")) return Name.COLL_READ_PERM;
return Name.CUSTOM_PERM; return Name.CUSTOM_PERM;
} }
static class Handler implements MapWriter {
final SolrRequestHandler handler;
final String pkg;
final int zkversion;
final Map meta;
final String name;
@Override
public void writeMap(EntryWriter ew) throws IOException {
ew.put(NAME, name);
ew.put(ConfigOverlay.ZNODEVER, zkversion);
meta.forEach(ew.getBiConsumer());
}
Handler(SolrRequestHandler handler, String pkg, int version, Map meta, String name) {
this.handler = handler;
this.pkg = pkg;
this.zkversion = version;
this.meta = Utils.getDeepCopy(meta, 3);
this.name = name;
}
public boolean shouldReload(Map metaData, Map<String, Package> pkgs) {
Package p = pkgs.get(pkg);
if (meta.equals(metaData) && p != null && p.lib.getZnodeVersion() <= zkversion) {
//the metadata is same and the package has not changed since we last loaded
return false;
}
return true;
}
}
} }
} }

View File

@ -34,6 +34,7 @@ import org.apache.solr.api.Api;
import org.apache.solr.api.ApiBag; import org.apache.solr.api.ApiBag;
import org.apache.solr.api.ApiSupport; import org.apache.solr.api.ApiSupport;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.params.CommonParams;
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;
@ -337,15 +338,15 @@ public class PluginBag<T> implements AutoCloseable {
} }
public PluginHolder<T> createPlugin(PluginInfo info) { public PluginHolder<T> createPlugin(PluginInfo info) {
String runtimeType = info.getRuntimeLibType(); String pkg = info.attributes.get(CommonParams.PACKAGE);
if ("global".equals(runtimeType)) { if (pkg != null) {
log.debug(" {} : '{}' created with runtimeLib=global ", meta.getCleanTag(), info.name); log.debug(" {} : '{}' created with package={} ", meta.getCleanTag(), info.name, pkg);
PluginHolder<T> holder = new GlobalLoaderPluginHolder<T>(info, core, meta); PluginHolder<T> holder = new PackagePluginHolder<T>(info, core, meta);
return meta.clazz == UpdateRequestProcessorFactory.class ? return meta.clazz == UpdateRequestProcessorFactory.class ?
(PluginHolder<T>) new UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder((PluginHolder<UpdateRequestProcessorFactory>) holder) : (PluginHolder<T>) new UpdateRequestProcessorChain.LazyUpdateProcessorFactoryHolder((PluginHolder<UpdateRequestProcessorFactory>) holder) :
holder; holder;
} else if ("core".equals(String.valueOf(runtimeType))) { } else if (info.isRuntimePlugin()) {
log.debug(" {} : '{}' created with runtimeLib=true ", meta.getCleanTag(), info.name); log.debug(" {} : '{}' created with runtimeLib=true ", meta.getCleanTag(), info.name);
LazyPluginHolder<T> holder = new LazyPluginHolder<>(meta, info, core, RuntimeLib.isEnabled() ? LazyPluginHolder<T> holder = new LazyPluginHolder<>(meta, info, core, RuntimeLib.isEnabled() ?
core.getMemClassLoader() : core.getMemClassLoader() :
@ -478,58 +479,37 @@ public class PluginBag<T> implements AutoCloseable {
loader.loadJars(); loader.loadJars();
} }
lazyInst = createInitInstance(pluginInfo,pluginMeta,core,resourceLoader, isRuntimeLib); lazyInst = createInitInstance(pluginInfo,pluginMeta,core,resourceLoader, isRuntimeLib);
/* Class<T> clazz = (Class<T>) pluginMeta.clazz;
T localInst = null;
try {
localInst = core.createInstance(pluginInfo.className, clazz, pluginMeta.getCleanTag(), null, resourceLoader);
} catch (SolrException e) {
if (isRuntimeLib && !(resourceLoader instanceof MemClassLoader)) {
throw new SolrException(SolrException.ErrorCode.getErrorCode(e.code()),
e.getMessage() + ". runtime library loading is not enabled, start Solr with -Denable.runtime.lib=true",
e.getCause());
}
throw e;
}
initInstance(localInst, pluginInfo);
if (localInst instanceof SolrCoreAware) {
SolrResourceLoader.assertAwareCompatibility(SolrCoreAware.class, localInst);
((SolrCoreAware) localInst).inform(core);
}
if (localInst instanceof ResourceLoaderAware) {
SolrResourceLoader.assertAwareCompatibility(ResourceLoaderAware.class, localInst);
try {
((ResourceLoaderAware) localInst).inform(core.getResourceLoader());
} catch (IOException e) {
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, "error initializing component", e);
}
}
lazyInst = localInst; // only assign the volatile until after the plugin is completely ready to use*/
return true; return true;
} }
} }
public class GlobalLoaderPluginHolder<T> extends PluginHolder<T> { public class PackagePluginHolder<T> extends PluginHolder<T> {
private final SolrCore core; private final SolrCore core;
private final SolrConfig.SolrPluginInfo pluginMeta; private final SolrConfig.SolrPluginInfo pluginMeta;
private final LibListener libListener; private final PackageManager packageManager;
private final String pkg;
private int znodeVersion =-1;
public GlobalLoaderPluginHolder(PluginInfo info, SolrCore core, SolrConfig.SolrPluginInfo pluginMeta) { public PackagePluginHolder(PluginInfo info, SolrCore core, SolrConfig.SolrPluginInfo pluginMeta) {
super(info); super(info);
this.core = core; this.core = core;
this.pluginMeta = pluginMeta; this.pluginMeta = pluginMeta;
this.core.addGlobalClassLoaderListener(() -> reload()); this.pkg = info.attributes.get(CommonParams.PACKAGE);
this.libListener = core.getCoreContainer().getClusterPropertiesListener(); this.core.addPackageListener(pkg, (lib) -> {
if(lib.getZnodeVersion() > znodeVersion) reload();
});
this.packageManager = core.getCoreContainer().getPackageManager();
reload(); reload();
} }
private void reload() { private void reload() {
if(inst == null) log.info("reloading plugin {} ", pluginInfo.name); if(inst == null) log.info("reloading plugin {} ", pluginInfo.name);
inst = createInitInstance(pluginInfo, pluginMeta, core, core.getCoreContainer().getClusterPropertiesListener().getResourceLoader(), true); inst = createInitInstance(pluginInfo, pluginMeta,
core, packageManager.getResourceLoader(this.pkg), true);
znodeVersion = packageManager.getZNodeVersion(pkg);
} }

View File

@ -25,6 +25,7 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import org.apache.solr.common.MapSerializable; import org.apache.solr.common.MapSerializable;
import org.apache.solr.common.params.CommonParams;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
import org.apache.solr.util.DOMUtil; import org.apache.solr.util.DOMUtil;
import org.slf4j.Logger; import org.slf4j.Logger;
@ -40,7 +41,6 @@ import static org.apache.solr.schema.FieldType.CLASS_NAME;
/** /**
* An Object which represents a Plugin of any type * An Object which represents a Plugin of any type
*
*/ */
public class PluginInfo implements MapSerializable { public class PluginInfo implements MapSerializable {
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
@ -57,7 +57,7 @@ public class PluginInfo implements MapSerializable {
this.className = attrs.get(CLASS_NAME); this.className = attrs.get(CLASS_NAME);
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;
} }
@ -72,7 +72,7 @@ public class PluginInfo implements MapSerializable {
isFromSolrConfig = true; isFromSolrConfig = true;
} }
public PluginInfo(String type, Map<String,Object> map) { public PluginInfo(String type, Map<String, Object> map) {
LinkedHashMap m = new LinkedHashMap<>(map); LinkedHashMap m = new LinkedHashMap<>(map);
initArgs = new NamedList(); initArgs = new NamedList();
for (Map.Entry<String, Object> entry : map.entrySet()) { for (Map.Entry<String, Object> entry : map.entrySet()) {
@ -127,7 +127,7 @@ public class PluginInfo implements MapSerializable {
return sb.toString(); return sb.toString();
} }
public boolean isEnabled(){ public boolean isEnabled() {
String enable = attributes.get("enable"); String enable = attributes.get("enable");
return enable == null || Boolean.parseBoolean(enable); return enable == null || Boolean.parseBoolean(enable);
} }
@ -136,19 +136,19 @@ public class PluginInfo implements MapSerializable {
return Boolean.parseBoolean(attributes.get("default")); return Boolean.parseBoolean(attributes.get("default"));
} }
public PluginInfo getChild(String type){ public PluginInfo getChild(String type) {
List<PluginInfo> l = getChildren(type); List<PluginInfo> l = getChildren(type);
return l.isEmpty() ? null:l.get(0); return l.isEmpty() ? null : l.get(0);
} }
public Map<String, Object> toMap(Map<String, Object> map) { public Map<String, Object> toMap(Map<String, Object> map) {
map.putAll(attributes); map.putAll(attributes);
Map m = map; Map m = map;
if(initArgs!=null ) m.putAll(initArgs.asMap(3)); if (initArgs != null) m.putAll(initArgs.asMap(3));
if(children != null){ if (children != null) {
for (PluginInfo child : children) { for (PluginInfo child : children) {
Object old = m.get(child.name); Object old = m.get(child.name);
if(old == null){ if (old == null) {
m.put(child.name, child.toMap(new LinkedHashMap<>())); m.put(child.name, child.toMap(new LinkedHashMap<>()));
} else if (old instanceof List) { } else if (old instanceof List) {
List list = (List) old; List list = (List) old;
@ -157,7 +157,7 @@ public class PluginInfo implements MapSerializable {
ArrayList l = new ArrayList(); ArrayList l = new ArrayList();
l.add(old); l.add(old);
l.add(child.toMap(new LinkedHashMap<>())); l.add(child.toMap(new LinkedHashMap<>()));
m.put(child.name,l); m.put(child.name, l);
} }
} }
@ -165,17 +165,20 @@ public class PluginInfo implements MapSerializable {
return m; return m;
} }
/**Filter children by type /**
* Filter children by type
*
* @param type The type name. must not be null * @param type The type name. must not be null
* @return The mathcing children * @return The mathcing children
*/ */
public List<PluginInfo> getChildren(String type){ public List<PluginInfo> getChildren(String type) {
if(children.isEmpty()) return children; if (children.isEmpty()) return children;
List<PluginInfo> result = new ArrayList<>(); List<PluginInfo> result = new ArrayList<>();
for (PluginInfo child : children) if(type.equals(child.type)) result.add(child); for (PluginInfo child : children) if (type.equals(child.type)) result.add(child);
return result; return result;
} }
public static final PluginInfo EMPTY_INFO = new PluginInfo("",Collections.<String,String>emptyMap(), new NamedList(),Collections.<PluginInfo>emptyList());
public static final PluginInfo EMPTY_INFO = new PluginInfo("", Collections.<String, String>emptyMap(), new NamedList(), Collections.<PluginInfo>emptyList());
private static final HashSet<String> NL_TAGS = new HashSet<> private static final HashSet<String> NL_TAGS = new HashSet<>
(asList("lst", "arr", (asList("lst", "arr",
@ -187,25 +190,21 @@ public class PluginInfo implements MapSerializable {
public static final String APPENDS = "appends"; public static final String APPENDS = "appends";
public static final String INVARIANTS = "invariants"; public static final String INVARIANTS = "invariants";
public boolean isFromSolrConfig(){ public boolean isFromSolrConfig() {
return isFromSolrConfig; return isFromSolrConfig;
} }
public PluginInfo copy() { public PluginInfo copy() {
PluginInfo result = new PluginInfo(type, attributes, PluginInfo result = new PluginInfo(type, attributes,
initArgs != null ? initArgs.clone() : null, children); initArgs != null ? initArgs.clone() : null, children);
result.isFromSolrConfig = isFromSolrConfig; result.isFromSolrConfig = isFromSolrConfig;
return result; return result;
} }
public String getRuntimeLibType(){
Object runtimeType = attributes.get(RuntimeLib.TYPE); public boolean isRuntimePlugin() {
if(runtimeType == null || "false".equals(runtimeType) || Boolean.FALSE.equals(runtimeType)) return null; return "true".equals(String.valueOf(attributes.get(RuntimeLib.TYPE)))
runtimeType = runtimeType.toString(); || (attributes.get(CommonParams.PACKAGE) != null);
if ("true".equals(String.valueOf(runtimeType))) {
log.warn("runtimeLib = true is deprecated, use runtimeLib=core/global");
runtimeType = "core";
}
return runtimeType.toString();
} }
} }

View File

@ -50,6 +50,7 @@ public class RuntimeLib implements PluginInfoInitialized, AutoCloseable, MapWrit
private String name, version, sig, sha512, url; private String name, version, sig, sha512, url;
private BlobRepository.BlobContentRef<ByteBuffer> jarContent; private BlobRepository.BlobContentRef<ByteBuffer> jarContent;
private boolean verified = false; private boolean verified = false;
int znodeVersion = -1;
@Override @Override
public void writeMap(EntryWriter ew) throws IOException { public void writeMap(EntryWriter ew) throws IOException {
@ -58,6 +59,12 @@ public class RuntimeLib implements PluginInfoInitialized, AutoCloseable, MapWrit
ew.putIfNotNull(version, version); ew.putIfNotNull(version, version);
ew.putIfNotNull(sha512, sha512); ew.putIfNotNull(sha512, sha512);
ew.putIfNotNull("sig", sig); ew.putIfNotNull("sig", sig);
if (znodeVersion > -1) {
ew.put(ConfigOverlay.ZNODEVER, znodeVersion);
}
}
public int getZnodeVersion(){
return znodeVersion;
} }
public RuntimeLib(CoreContainer coreContainer) { public RuntimeLib(CoreContainer coreContainer) {

View File

@ -56,6 +56,7 @@ import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.locks.ReentrantLock; import java.util.concurrent.locks.ReentrantLock;
import java.util.function.Consumer;
import com.codahale.metrics.Counter; import com.codahale.metrics.Counter;
import com.codahale.metrics.MetricRegistry; import com.codahale.metrics.MetricRegistry;
@ -94,6 +95,7 @@ import org.apache.solr.common.util.ExecutorUtil;
import org.apache.solr.common.util.IOUtils; import org.apache.solr.common.util.IOUtils;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
import org.apache.solr.common.util.ObjectReleaseTracker; import org.apache.solr.common.util.ObjectReleaseTracker;
import org.apache.solr.common.util.Pair;
import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.common.util.SimpleOrderedMap;
import org.apache.solr.common.util.Utils; import org.apache.solr.common.util.Utils;
import org.apache.solr.core.DirectoryFactory.DirContext; import org.apache.solr.core.DirectoryFactory.DirContext;
@ -238,7 +240,8 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
public volatile boolean searchEnabled = true; public volatile boolean searchEnabled = true;
public volatile boolean indexEnabled = true; public volatile boolean indexEnabled = true;
public volatile boolean readOnly = false; public volatile boolean readOnly = false;
private List<Runnable> globalClassLoaderListeners = new ArrayList<>(); private List<Pair<String ,Consumer<RuntimeLib>>> packageListeners = new ArrayList<>();
public Set<String> getMetricNames() { public Set<String> getMetricNames() {
return metricNames; return metricNames;
@ -353,14 +356,14 @@ public final class SolrCore implements SolrInfoBean, SolrMetricProducer, Closeab
.getPath(); .getPath();
} }
} }
void globalClassLoaderChanged(){
for (Runnable r : globalClassLoaderListeners) {
r.run();
void packageUpdated(RuntimeLib lib) {
for (Pair<String, Consumer<RuntimeLib>> pair : packageListeners) {
if(lib.equals(pair.first())) pair.second().accept(lib);
} }
} }
void addGlobalClassLoaderListener(Runnable r){ public void addPackageListener(String pkg, Consumer<RuntimeLib> r){
globalClassLoaderListeners.add(r); packageListeners.add(new Pair<>(pkg, r));
} }
@ -868,9 +871,10 @@ 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;
ResourceLoader resourceLoader = "global".equals(info.getRuntimeLibType())? String pkg = info.attributes.get(CommonParams.PACKAGE);
coreContainer.getClusterPropertiesListener(). ResourceLoader resourceLoader = pkg != null?
getResourceLoader(): getResourceLoader(); coreContainer.getPackageManager().getResourceLoader(pkg):
getResourceLoader();
T o = createInstance(info.className == null ? defClassName : info.className, cast, msg, this, resourceLoader); T o = createInstance(info.className == null ? defClassName : info.className, cast, msg, this, resourceLoader);
if (o instanceof PluginInfoInitialized) { if (o instanceof PluginInfoInitialized) {

View File

@ -217,7 +217,7 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
timer.getTime(), prop, expectedVersion, collection); timer.getTime(), prop, expectedVersion, collection);
} }
public static void execInparallel( List<? extends PerReplicaCallable> concurrentTasks, Consumer<ExecutorService> fun) { public static void execInparallel(List<? extends PerReplicaCallable> concurrentTasks, Consumer<ExecutorService> fun) {
int poolSize = Math.min(concurrentTasks.size(), 10); int poolSize = Math.min(concurrentTasks.size(), 10);
ExecutorService parallelExecutor = ExecutorService parallelExecutor =
ExecutorUtil.newMDCAwareFixedThreadPool(poolSize, new DefaultSolrThreadFactory("solrHandlerExecutor")); ExecutorUtil.newMDCAwareFixedThreadPool(poolSize, new DefaultSolrThreadFactory("solrHandlerExecutor"));
@ -601,7 +601,7 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
} }
} catch (Exception e) { } catch (Exception e) {
log.error( "error executing commands "+ Utils.toJSONString(ops) ,e); log.error("error executing commands " + Utils.toJSONString(ops), e);
resp.setException(e); resp.setException(e);
resp.add(CommandOperation.ERR_MSGS, singletonList(SchemaManager.getErrorStr(e))); resp.add(CommandOperation.ERR_MSGS, singletonList(SchemaManager.getErrorStr(e)));
} }
@ -817,13 +817,13 @@ public class SolrConfigHandler extends RequestHandlerBase implements SolrCoreAwa
private boolean verifyClass(CommandOperation op, String clz, SolrConfig.SolrPluginInfo pluginMeta) { private boolean verifyClass(CommandOperation op, String clz, SolrConfig.SolrPluginInfo pluginMeta) {
if (clz == null) return true; if (clz == null) return true;
PluginInfo info = new PluginInfo(pluginMeta.getCleanTag(), op.getDataMap()); PluginInfo info = new PluginInfo(pluginMeta.getCleanTag(), op.getDataMap());
if(info.getRuntimeLibType() != null && !RuntimeLib.isEnabled()){
if (info.isRuntimePlugin() && !RuntimeLib.isEnabled()) {
op.addError("node not started with enable.runtime.lib=true"); op.addError("node not started with enable.runtime.lib=true");
return false; return false;
} }
if (!"true".equals(String.valueOf(op.getStr(RuntimeLib.TYPE, null)))) {
if ( !"true".equals(String.valueOf(op.getStr(RuntimeLib.TYPE, null)))) {
//this is not dynamically loaded so we can verify the class right away //this is not dynamically loaded so we can verify the class right away
try { try {
req.getCore().createInitInstance(new PluginInfo(SolrRequestHandler.TYPE, op.getDataMap()), pluginMeta.clazz, clz, ""); req.getCore().createInitInstance(new PluginInfo(SolrRequestHandler.TYPE, op.getDataMap()), pluginMeta.clazz, clz, "");

View File

@ -95,9 +95,9 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
//The following APIs have only V2 implementations //The following APIs have only V2 implementations
addApi(apiMapping, Meta.GET_NODES, CollectionHandlerApi::getNodes); addApi(apiMapping, Meta.GET_NODES, CollectionHandlerApi::getNodes);
addApi(apiMapping, Meta.SET_CLUSTER_PROPERTY_OBJ, CollectionHandlerApi::setClusterObj); addApi(apiMapping, Meta.SET_CLUSTER_PROPERTY_OBJ, CollectionHandlerApi::setClusterObj);
addApi(apiMapping, Meta.ADD_RUNTIME_LIB, wrap(CollectionHandlerApi::addUpdateRuntimeLib)); addApi(apiMapping, Meta.ADD_PACKAGE, wrap(CollectionHandlerApi::addUpdatePackage));
addApi(apiMapping, Meta.UPDATE_RUNTIME_LIB, wrap(CollectionHandlerApi::addUpdateRuntimeLib)); addApi(apiMapping, Meta.UPDATE_PACKAGE, wrap(CollectionHandlerApi::addUpdatePackage));
addApi(apiMapping, Meta.DELETE_RUNTIME_LIB, wrap(CollectionHandlerApi::deleteRuntimeLib)); addApi(apiMapping, Meta.DELETE_RUNTIME_LIB, wrap(CollectionHandlerApi::deletePackage));
addApi(apiMapping, Meta.ADD_REQ_HANDLER, wrap(CollectionHandlerApi::addRequestHandler)); addApi(apiMapping, Meta.ADD_REQ_HANDLER, wrap(CollectionHandlerApi::addRequestHandler));
addApi(apiMapping, Meta.DELETE_REQ_HANDLER, wrap(CollectionHandlerApi::deleteReqHandler)); addApi(apiMapping, Meta.DELETE_REQ_HANDLER, wrap(CollectionHandlerApi::deleteReqHandler));
@ -117,9 +117,17 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
if (modified) { if (modified) {
Stat stat = new Stat(); Stat stat = new Stat();
Map<String, Object> clusterProperties = new ClusterProperties(cc.getZkController().getZkClient()).getClusterProperties(stat); Map<String, Object> clusterProperties = new ClusterProperties(cc.getZkController().getZkClient()).getClusterProperties(stat);
cc.getClusterPropertiesListener().onChange(clusterProperties); try {
cc.getPackageManager().onChange(clusterProperties);
} catch (SolrException e) {
log.error("error executing command : " + info.op.jsonStr(), e);
throw e;
} catch (Exception e) {
log.error("error executing command : " + info.op.jsonStr(), e);
throw new SolrException(ErrorCode.SERVER_ERROR, "error executing command : ", e);
}
log.info("current version of clusterprops.json is {} , trying to get every node to update ", stat.getVersion()); log.info("current version of clusterprops.json is {} , trying to get every node to update ", stat.getVersion());
log.debug("The current clusterprops.json: {}",clusterProperties ); log.debug("The current clusterprops.json: {}", clusterProperties);
((CollectionHandlerApi) info.apiHandler).waitForStateSync(stat.getVersion(), cc); ((CollectionHandlerApi) info.apiHandler).waitForStateSync(stat.getVersion(), cc);
} }
@ -165,7 +173,8 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
clusterProperties.setClusterProperties(m); clusterProperties.setClusterProperties(m);
return true; return true;
} }
private static boolean deleteRuntimeLib(ApiInfo params) throws Exception {
private static boolean deletePackage(ApiInfo params) throws Exception {
if (!RuntimeLib.isEnabled()) { if (!RuntimeLib.isEnabled()) {
params.op.addError("node not started with enable.runtime.lib=true"); params.op.addError("node not started with enable.runtime.lib=true");
return false; return false;
@ -173,9 +182,9 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
String name = params.op.getStr(CommandOperation.ROOT_OBJ); String name = params.op.getStr(CommandOperation.ROOT_OBJ);
ClusterProperties clusterProperties = new ClusterProperties(((CollectionHandlerApi) params.apiHandler).handler.coreContainer.getZkController().getZkClient()); ClusterProperties clusterProperties = new ClusterProperties(((CollectionHandlerApi) params.apiHandler).handler.coreContainer.getZkController().getZkClient());
Map<String, Object> props = clusterProperties.getClusterProperties(); Map<String, Object> props = clusterProperties.getClusterProperties();
List<String> pathToLib = asList(RuntimeLib.TYPE, name); List<String> pathToLib = asList(CommonParams.PACKAGE, name);
Map existing = (Map) Utils.getObjectByPath(props, false, pathToLib); Map existing = (Map) Utils.getObjectByPath(props, false, pathToLib);
if(existing == null){ if (existing == null) {
params.op.addError("No such runtimeLib : " + name); params.op.addError("No such runtimeLib : " + name);
return false; return false;
} }
@ -185,7 +194,7 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
return true; return true;
} }
private static boolean addUpdateRuntimeLib(ApiInfo params) throws Exception { private static boolean addUpdatePackage(ApiInfo params) throws Exception {
if (!RuntimeLib.isEnabled()) { if (!RuntimeLib.isEnabled()) {
params.op.addError("node not started with enable.runtime.lib=true"); params.op.addError("node not started with enable.runtime.lib=true");
return false; return false;
@ -197,9 +206,9 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
String name = op.getStr("name"); String name = op.getStr("name");
ClusterProperties clusterProperties = new ClusterProperties(((CollectionHandlerApi) params.apiHandler).handler.coreContainer.getZkController().getZkClient()); ClusterProperties clusterProperties = new ClusterProperties(((CollectionHandlerApi) params.apiHandler).handler.coreContainer.getZkController().getZkClient());
Map<String, Object> props = clusterProperties.getClusterProperties(); Map<String, Object> props = clusterProperties.getClusterProperties();
List<String> pathToLib = asList(RuntimeLib.TYPE, name); List<String> pathToLib = asList(CommonParams.PACKAGE, name);
Map existing = (Map) Utils.getObjectByPath(props, false, pathToLib); Map existing = (Map) Utils.getObjectByPath(props, false, pathToLib);
if (Meta.ADD_RUNTIME_LIB.commandName.equals(op.name)) { if (Meta.ADD_PACKAGE.commandName.equals(op.name)) {
if (existing != null) { if (existing != null) {
op.addError(StrUtils.formatString("The jar with a name ''{0}'' already exists ", name)); op.addError(StrUtils.formatString("The jar with a name ''{0}'' already exists ", name));
return false; return false;
@ -209,7 +218,7 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
op.addError(StrUtils.formatString("The jar with a name ''{0}'' does not exist", name)); op.addError(StrUtils.formatString("The jar with a name ''{0}'' does not exist", name));
return false; return false;
} }
if(Objects.equals( existing.get(SHA512) , op.getDataMap().get(SHA512))){ if (Objects.equals(existing.get(SHA512), op.getDataMap().get(SHA512))) {
op.addError("Trying to update a jar with the same sha512"); op.addError("Trying to update a jar with the same sha512");
return false; return false;
} }
@ -323,6 +332,7 @@ public class CollectionHandlerApi extends BaseHandlerApiSupport {
public static class PerNodeCallable extends SolrConfigHandler.PerReplicaCallable { public static class PerNodeCallable extends SolrConfigHandler.PerReplicaCallable {
static final List<String> path = Arrays.asList("metadata", CommonParams.VERSION); static final List<String> path = Arrays.asList("metadata", CommonParams.VERSION);
PerNodeCallable(String baseUrl, int expectedversion, int waitTime) { PerNodeCallable(String baseUrl, int expectedversion, int waitTime) {
super(baseUrl, ConfigOverlay.ZNODEVER, expectedversion, waitTime); super(baseUrl, ConfigOverlay.ZNODEVER, expectedversion, waitTime);
} }

View File

@ -149,10 +149,10 @@ import static org.apache.solr.common.params.CollectionParams.CollectionAction.*;
import static org.apache.solr.common.params.CommonAdminParams.ASYNC; import static org.apache.solr.common.params.CommonAdminParams.ASYNC;
import static org.apache.solr.common.params.CommonAdminParams.IN_PLACE_MOVE; import static org.apache.solr.common.params.CommonAdminParams.IN_PLACE_MOVE;
import static org.apache.solr.common.params.CommonAdminParams.NUM_SUB_SHARDS; import static org.apache.solr.common.params.CommonAdminParams.NUM_SUB_SHARDS;
import static org.apache.solr.common.params.CommonAdminParams.SPLIT_BY_PREFIX;
import static org.apache.solr.common.params.CommonAdminParams.SPLIT_FUZZ; import static org.apache.solr.common.params.CommonAdminParams.SPLIT_FUZZ;
import static org.apache.solr.common.params.CommonAdminParams.SPLIT_METHOD; import static org.apache.solr.common.params.CommonAdminParams.SPLIT_METHOD;
import static org.apache.solr.common.params.CommonAdminParams.WAIT_FOR_FINAL_STATE; import static org.apache.solr.common.params.CommonAdminParams.WAIT_FOR_FINAL_STATE;
import static org.apache.solr.common.params.CommonAdminParams.SPLIT_BY_PREFIX;
import static org.apache.solr.common.params.CommonParams.NAME; import static org.apache.solr.common.params.CommonParams.NAME;
import static org.apache.solr.common.params.CommonParams.TIMING; import static org.apache.solr.common.params.CommonParams.TIMING;
import static org.apache.solr.common.params.CommonParams.VALUE_LONG; import static org.apache.solr.common.params.CommonParams.VALUE_LONG;

View File

@ -79,8 +79,8 @@ import org.apache.solr.common.params.SolrParams;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
import org.apache.solr.common.util.SimpleOrderedMap; import org.apache.solr.common.util.SimpleOrderedMap;
import org.apache.solr.common.util.StrUtils; import org.apache.solr.common.util.StrUtils;
import org.apache.solr.core.XmlConfigFile;
import org.apache.solr.core.SolrCore; import org.apache.solr.core.SolrCore;
import org.apache.solr.core.XmlConfigFile;
import org.apache.solr.request.SolrQueryRequest; import org.apache.solr.request.SolrQueryRequest;
import org.apache.solr.response.transform.ElevatedMarkerFactory; import org.apache.solr.response.transform.ElevatedMarkerFactory;
import org.apache.solr.response.transform.ExcludedMarkerFactory; import org.apache.solr.response.transform.ExcludedMarkerFactory;

View File

@ -16,12 +16,11 @@
*/ */
package org.apache.solr.rest.schema; package org.apache.solr.rest.schema;
import java.util.List;
import java.util.Map;
import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory; import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException; import javax.xml.parsers.ParserConfigurationException;
import java.util.List;
import java.util.Map;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.SolrException.ErrorCode; import org.apache.solr.common.SolrException.ErrorCode;

View File

@ -21,12 +21,17 @@ import javax.xml.xpath.XPathConstants;
import java.io.IOException; import java.io.IOException;
import java.util.HashMap; import java.util.HashMap;
import java.util.LinkedHashMap; import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map; import java.util.Map;
import org.apache.lucene.analysis.util.ResourceLoader; import org.apache.lucene.analysis.util.ResourceLoader;
import org.apache.solr.common.MapWriter; import org.apache.solr.common.MapWriter;
import org.apache.solr.common.SolrException; import org.apache.solr.common.SolrException;
import org.apache.solr.common.params.CommonParams;
import org.apache.solr.common.util.StrUtils;
import org.apache.solr.common.util.Utils;
import org.apache.solr.core.ConfigOverlay; import org.apache.solr.core.ConfigOverlay;
import org.apache.solr.core.MemClassLoader;
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.util.DOMUtil; import org.apache.solr.util.DOMUtil;
@ -40,7 +45,7 @@ public class CacheConfig implements MapWriter {
private CacheRegenerator defRegen; private CacheRegenerator defRegen;
private final String name; private final String name;
private String cacheImpl, regenImpl; private String cacheImpl, regenImpl;
private Object[] persistence = new Object[1]; Object[] persistence = new Object[1];
public CacheConfig(Map<String, String> args) { public CacheConfig(Map<String, String> args) {
@ -49,8 +54,9 @@ public class CacheConfig implements MapWriter {
this.cacheImpl = args.getOrDefault("class", "solr.LRUCache"); this.cacheImpl = args.getOrDefault("class", "solr.LRUCache");
this.regenImpl = args.get("regenerator"); this.regenImpl = args.get("regenerator");
} }
static Map<String,String> copyValsAsString(Map m){
Map<String,String> copy = new LinkedHashMap(m.size()); static Map<String, String> copyValsAsString(Map m) {
Map<String, String> copy = new LinkedHashMap(m.size());
m.forEach((k, v) -> copy.put(String.valueOf(k), String.valueOf(v))); m.forEach((k, v) -> copy.put(String.valueOf(k), String.valueOf(v)));
return copy; return copy;
} }
@ -60,6 +66,10 @@ public class CacheConfig implements MapWriter {
if (node == null || !"true".equals(DOMUtil.getAttrOrDefault(node, "enabled", "true"))) { if (node == null || !"true".equals(DOMUtil.getAttrOrDefault(node, "enabled", "true"))) {
Map<String, String> m = solrConfig.getOverlay().getEditableSubProperties(xpath); Map<String, String> m = solrConfig.getOverlay().getEditableSubProperties(xpath);
if (m == null) return null; if (m == null) return null;
List<String> pieces = StrUtils.splitSmart(xpath, '/');
String name = pieces.get(pieces.size() - 1);
m = Utils.getDeepCopy(m, 2);
m.put(NAME, name);
return new CacheConfig(m); return new CacheConfig(m);
} else { } else {
Map<String, String> attrs = DOMUtil.toMap(node.getAttributes()); Map<String, String> attrs = DOMUtil.toMap(node.getAttributes());
@ -100,28 +110,45 @@ public class CacheConfig implements MapWriter {
public <K, V> SolrCacheHolder<K, V> newInstance(SolrCore core) { public <K, V> SolrCacheHolder<K, V> newInstance(SolrCore core) {
ResourceLoader loader = core.getResourceLoader(); return new SolrCacheHolder<>(new CacheInfo(this, core));
}
SolrCache inst = null; static class CacheInfo {
final CacheConfig cfg;
SolrCore core;
SolrCache cache = null;
int znodeVersion = -1;
String pkg;
CacheRegenerator regen = null; CacheRegenerator regen = null;
CacheInfo(CacheConfig cfg, SolrCore core) {
this.core = core;
this.cfg = cfg;
pkg = cfg.args.get(CommonParams.PACKAGE);
ResourceLoader loader = pkg == null ? core.getResourceLoader() :
core.getCoreContainer().getPackageManager().getResourceLoader(pkg);
try { try {
inst = loader.findClass(cacheImpl, SolrCache.class).getConstructor().newInstance(); cache = loader.findClass(cfg.cacheImpl, SolrCache.class).getConstructor().newInstance();
regen = null; regen = null;
if (regenImpl != null) { if (cfg.regenImpl != null) {
regen = loader.findClass(regenImpl, CacheRegenerator.class).getConstructor().newInstance(); regen = loader.findClass(cfg.regenImpl, CacheRegenerator.class).getConstructor().newInstance();
} }
} catch (Exception e) { } catch (Exception e) {
throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, "Error loading cache " + jsonStr(), e); throw new SolrException(SolrException.ErrorCode.SERVER_ERROR, "Error loading cache " + cfg.jsonStr(), e);
}
if (regen == null && cfg.defRegen != null) regen = cfg.defRegen;
cfg.persistence[0] = cache.init(cfg.args, cfg.persistence[0], regen);
if (loader instanceof MemClassLoader) {
MemClassLoader memClassLoader = (MemClassLoader) loader;
znodeVersion = memClassLoader.getZnodeVersion();
}
} }
if (regen == null && defRegen != null) regen = defRegen;
persistence[0] = inst.init(args, persistence[0], regen);
return new SolrCacheHolder<>(inst, this);
} }
public void setDefaultRegenerator(CacheRegenerator regen) { public void setDefaultRegenerator(CacheRegenerator regen) {
this.defRegen = regen; this.defRegen = regen;
} }

View File

@ -0,0 +1,35 @@
/*
* 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 runtimecode;
import org.apache.lucene.document.Document;
import org.apache.lucene.document.StoredField;
import org.apache.solr.search.LRUCache;
public class MyDocCache<K,V> extends LRUCache<K,V> {
static String fld_name= "my_synthetic_fld_s";
@Override
public V put(K key, V value) {
if(value instanceof Document){
Document d = (Document) value;
d.add(new StoredField(fld_name, "version_2"));
}
return super.put(key, value);
}
}

Binary file not shown.

Binary file not shown.

View File

@ -63,3 +63,11 @@ bc5ce45ad281b6a08fb7e529b1eb475040076834816570902acb6ebdd809410e31006efdeaa7f78a
openssl dgst -sha512 runtimelibs_v3.jar.bin openssl dgst -sha512 runtimelibs_v3.jar.bin
60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922 60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922
openssl dgst -sha512 cache.jar.bin
1a3739b629ce85895c9b2a8c12dd7d98161ff47634b0693f1e1c5b444fb38343f95c6ee955cd99103bd24cfde6c205234b63823818660ac08392cdc626caf585
openssl dgst -sha512 cache_v2.jar.bin
aa3f42fb640636dd8126beca36ac389486d0fcb1c3a2e2c387d043d57637535ce8db3b17983853322f78bb8f447ed75fe7b405675debe652ed826ee95e8ce328

View File

@ -22,20 +22,26 @@ import java.io.InputStream;
import java.io.Reader; import java.io.Reader;
import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodHandles;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collections;
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.function.Predicate; import java.util.function.Predicate;
import com.google.common.collect.ImmutableMap;
import org.apache.solr.SolrTestCaseJ4; import org.apache.solr.SolrTestCaseJ4;
import org.apache.solr.client.solrj.ResponseParser; import org.apache.solr.client.solrj.ResponseParser;
import org.apache.solr.client.solrj.SolrClient; import org.apache.solr.client.solrj.SolrClient;
import org.apache.solr.client.solrj.SolrQuery;
import org.apache.solr.client.solrj.SolrRequest; import org.apache.solr.client.solrj.SolrRequest;
import org.apache.solr.client.solrj.SolrResponse; import org.apache.solr.client.solrj.SolrResponse;
import org.apache.solr.client.solrj.SolrServerException; import org.apache.solr.client.solrj.SolrServerException;
import org.apache.solr.client.solrj.impl.BaseHttpSolrClient; import org.apache.solr.client.solrj.impl.BaseHttpSolrClient;
import org.apache.solr.client.solrj.request.AbstractUpdateRequest;
import org.apache.solr.client.solrj.request.CollectionAdminRequest; import org.apache.solr.client.solrj.request.CollectionAdminRequest;
import org.apache.solr.client.solrj.request.GenericSolrRequest; import org.apache.solr.client.solrj.request.GenericSolrRequest;
import org.apache.solr.client.solrj.request.QueryRequest;
import org.apache.solr.client.solrj.request.UpdateRequest;
import org.apache.solr.client.solrj.request.V2Request; import org.apache.solr.client.solrj.request.V2Request;
import org.apache.solr.client.solrj.response.V2Response; import org.apache.solr.client.solrj.response.V2Response;
import org.apache.solr.cloud.ConfigRequest; import org.apache.solr.cloud.ConfigRequest;
@ -48,6 +54,7 @@ import org.apache.solr.common.params.SolrParams;
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.common.util.Pair;
import org.apache.solr.common.util.Utils; import org.apache.solr.common.util.Utils;
import org.apache.solr.core.ConfigOverlay;
import org.apache.solr.core.MemClassLoader; import org.apache.solr.core.MemClassLoader;
import org.apache.solr.core.RuntimeLib; import org.apache.solr.core.RuntimeLib;
import org.apache.solr.request.SolrRequestHandler; import org.apache.solr.request.SolrRequestHandler;
@ -60,8 +67,6 @@ import org.junit.Test;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import com.google.common.collect.ImmutableMap;
import static org.apache.solr.cloud.TestCryptoKeys.readFile; import static org.apache.solr.cloud.TestCryptoKeys.readFile;
import static org.apache.solr.common.params.CommonParams.JAVABIN; import static org.apache.solr.common.params.CommonParams.JAVABIN;
import static org.apache.solr.common.params.CommonParams.WT; import static org.apache.solr.common.params.CommonParams.WT;
@ -70,7 +75,7 @@ import static org.apache.solr.core.TestDynamicLoading.getFileContent;
import static org.apache.solr.core.TestDynamicLoadingUrl.runHttpServer; import static org.apache.solr.core.TestDynamicLoadingUrl.runHttpServer;
@SolrTestCaseJ4.SuppressSSL @SolrTestCaseJ4.SuppressSSL
@LogLevel("org.apache.solr.common.cloud.ZkStateReader=DEBUG;org.apache.solr.handler.admin.CollectionHandlerApi=DEBUG;org.apache.solr.core.LibListener=DEBUG;org.apache.solr.common.cloud.ClusterProperties=DEBUG") @LogLevel("org.apache.solr.common.cloud.ZkStateReader=DEBUG;org.apache.solr.handler.admin.CollectionHandlerApi=DEBUG;org.apache.solr.core.PackageManager=DEBUG;org.apache.solr.common.cloud.ClusterProperties=DEBUG")
public class TestContainerReqHandler extends SolrCloudTestCase { public class TestContainerReqHandler extends SolrCloudTestCase {
private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass()); private static final Logger log = LoggerFactory.getLogger(MethodHandles.lookup().lookupClass());
@ -81,12 +86,13 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
static void assertResponseValues(int repeats, SolrClient client, SolrRequest req, Map vals) throws Exception { static SolrResponse assertResponseValues(int repeats, SolrClient client, SolrRequest req, Map vals) throws Exception {
SolrResponse rsp = null;
for (int i = 0; i < repeats; i++) { for (int i = 0; i < repeats; i++) {
if (i > 0) { if (i > 0) {
Thread.sleep(100); Thread.sleep(100);
} }
SolrResponse rsp = null;
try { try {
rsp = req.process(client); rsp = req.process(client);
} catch (Exception e) { } catch (Exception e) {
@ -101,7 +107,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
return Objects.equals(val, o); return Objects.equals(val, o);
}; };
boolean isPass = p.test(rsp.getResponse()._get(key, null)); boolean isPass = p.test(rsp.getResponse()._get(key, null));
if(isPass) return; if (isPass) return rsp;
else if (i >= repeats - 1) { else if (i >= repeats - 1) {
fail("attempt: " + i + " Mismatch for value : '" + key + "' in response " + Utils.toJSONString(rsp)); fail("attempt: " + i + " Mismatch for value : '" + key + "' in response " + Utils.toJSONString(rsp));
} }
@ -109,8 +115,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
} }
return rsp;
} }
private static Map<String, Object> assertVersionInSync(SolrZkClient zkClient, SolrClient solrClient) throws SolrServerException, IOException { private static Map<String, Object> assertVersionInSync(SolrZkClient zkClient, SolrClient solrClient) throws SolrServerException, IOException {
@ -127,7 +132,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
@Test @Test
public void testRuntimeLib() throws Exception { public void testPackageAPI() throws Exception {
Map<String, Object> jars = Utils.makeMap( Map<String, Object> jars = Utils.makeMap(
"/jar1.jar", getFileContent("runtimecode/runtimelibs.jar.bin"), "/jar1.jar", getFileContent("runtimecode/runtimelibs.jar.bin"),
"/jar2.jar", getFileContent("runtimecode/runtimelibs_v2.jar.bin"), "/jar2.jar", getFileContent("runtimecode/runtimelibs_v2.jar.bin"),
@ -139,7 +144,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
try { try {
String payload = null; String payload = null;
try { try {
payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sha512 : 'wrong-sha512'}}"; "sha512 : 'wrong-sha512'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
@ -151,7 +156,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
try { try {
payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar0.jar', " + payload = "{add-package:{name : 'foo', url: 'http://localhost:" + port + "/jar0.jar', " +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
@ -162,18 +167,18 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
assertTrue("Actual output : " + Utils.toJSONString(e.getMetaData()), e.getMetaData()._getStr("error/details[0]/errorMessages[0]", "").contains("no such resource available: foo")); assertTrue("Actual output : " + Utils.toJSONString(e.getMetaData()), e.getMetaData()._getStr("error/details[0]/errorMessages[0]", "").contains("no such resource available: foo"));
} }
payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler'}}") .withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler', package : global}}")
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties(); Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties();
@ -192,14 +197,14 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class"))); getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class")));
payload = "{update-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar3.jar', " + payload = "{update-package:{name : 'global', url: 'http://localhost:" + port + "/jar3.jar', " +
"sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}"; "sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
request = new V2Request.Builder("/node/ext/bar") request = new V2Request.Builder("/node/ext/bar")
@ -222,7 +227,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
assertResponseValues(10, cluster.getSolrClient(), request, ImmutableMap.of(SolrRequestHandler.TYPE, assertResponseValues(10, cluster.getSolrClient(), request, ImmutableMap.of(SolrRequestHandler.TYPE,
(Predicate<Object>) o -> o instanceof List && ((List) o).isEmpty())); (Predicate<Object>) o -> o instanceof List && ((List) o).isEmpty()));
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload("{delete-runtimelib : 'foo'}") .withPayload("{delete-package : 'global'}")
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertResponseValues(10, cluster.getSolrClient(), request, ImmutableMap.of(RuntimeLib.TYPE, assertResponseValues(10, cluster.getSolrClient(), request, ImmutableMap.of(RuntimeLib.TYPE,
@ -254,7 +259,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
String signature = "NaTm3+i99/ZhS8YRsLc3NLz2Y6VuwEbu7DihY8GAWwWIGm+jpXgn1JiuaenfxFCcfNKCC9WgZmEgbTZTzmV/OZMVn90u642YJbF3vTnzelW1pHB43ZRAJ1iesH0anM37w03n3es+vFWQtuxc+2Go888fJoMkUX2C6Zk6Jn116KE45DWjeyPM4mp3vvGzwGvdRxP5K9Q3suA+iuI/ULXM7m9mV4ruvs/MZvL+ELm5Jnmk1bBtixVJhQwJP2z++8tQKJghhyBxPIC/2fkAHobQpkhZrXu56JjP+v33ul3Ku4bbvfVMY/LVwCAEnxlvhk+C6uRCKCeFMrzQ/k5inasXLw=="; String signature = "NaTm3+i99/ZhS8YRsLc3NLz2Y6VuwEbu7DihY8GAWwWIGm+jpXgn1JiuaenfxFCcfNKCC9WgZmEgbTZTzmV/OZMVn90u642YJbF3vTnzelW1pHB43ZRAJ1iesH0anM37w03n3es+vFWQtuxc+2Go888fJoMkUX2C6Zk6Jn116KE45DWjeyPM4mp3vvGzwGvdRxP5K9Q3suA+iuI/ULXM7m9mV4ruvs/MZvL+ELm5Jnmk1bBtixVJhQwJP2z++8tQKJghhyBxPIC/2fkAHobQpkhZrXu56JjP+v33ul3Ku4bbvfVMY/LVwCAEnxlvhk+C6uRCKCeFMrzQ/k5inasXLw==";
String payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + String payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sig : 'EdYkvRpMZbvElN93/xUmyKXcj6xHP16AVk71TlTascEwCb5cFQ2AeKhPIlwYpkLWXEOcLZKfeXoWwOLaV5ZNhg==' ," + "sig : 'EdYkvRpMZbvElN93/xUmyKXcj6xHP16AVk71TlTascEwCb5cFQ2AeKhPIlwYpkLWXEOcLZKfeXoWwOLaV5ZNhg==' ," +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
try { try {
@ -268,7 +273,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sig : '" + signature + "'," + "sig : '" + signature + "'," +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
@ -276,11 +281,11 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler'}}") .withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler' package : global}}")
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties(); Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties();
@ -298,7 +303,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
assertEquals("org.apache.solr.core.RuntimeLibReqHandler", assertEquals("org.apache.solr.core.RuntimeLibReqHandler",
getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class"))); getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class")));
payload = "{update-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar3.jar', " + payload = "{update-package:{name : 'global', url: 'http://localhost:" + port + "/jar3.jar', " +
"sig : 'YxFr6SpYrDwG85miDfRWHTjU9UltjtIWQZEhcV55C2rczRUVowCYBxmsDv5mAM8j0CTv854xpI1DtBT86wpoTdbF95LQuP9FJId4TS1j8bZ9cxHP5Cqyz1uBHFfUUNUrnpzTHQkVTp02O9NAjh3c2W41bL4U7j6jQ32+4CW2M+x00TDG0y0H75rQDR8zbLt31oWCz+sBOdZ3rGKJgAvdoGm/wVCTmsabZN+xoz4JaDeBXF16O9Uk9SSq4G0dz5YXFuLxHK7ciB5t0+q6pXlF/tdlDqF76Abze0R3d2/0MhXBzyNp3UxJmj6DiprgysfB0TbQtJG0XGfdSmx0VChvcA==' ," + "sig : 'YxFr6SpYrDwG85miDfRWHTjU9UltjtIWQZEhcV55C2rczRUVowCYBxmsDv5mAM8j0CTv854xpI1DtBT86wpoTdbF95LQuP9FJId4TS1j8bZ9cxHP5Cqyz1uBHFfUUNUrnpzTHQkVTp02O9NAjh3c2W41bL4U7j6jQ32+4CW2M+x00TDG0y0H75rQDR8zbLt31oWCz+sBOdZ3rGKJgAvdoGm/wVCTmsabZN+xoz4JaDeBXF16O9Uk9SSq4G0dz5YXFuLxHK7ciB5t0+q6pXlF/tdlDqF76Abze0R3d2/0MhXBzyNp3UxJmj6DiprgysfB0TbQtJG0XGfdSmx0VChvcA==' ," +
"sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}"; "sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}";
@ -306,8 +311,8 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
request = new V2Request.Builder("/node/ext/bar") request = new V2Request.Builder("/node/ext/bar")
@ -345,7 +350,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
String signature = "L3q/qIGs4NaF6JiO0ZkMUFa88j0OmYc+I6O7BOdNuMct/xoZ4h73aZHZGc0+nmI1f/U3bOlMPINlSOM6LK3JpQ=="; String signature = "L3q/qIGs4NaF6JiO0ZkMUFa88j0OmYc+I6O7BOdNuMct/xoZ4h73aZHZGc0+nmI1f/U3bOlMPINlSOM6LK3JpQ==";
String payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + String payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sig : '" + signature + "'," + "sig : '" + signature + "'," +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
@ -353,11 +358,11 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler'}}") .withPayload("{add-requesthandler:{name : 'bar', class : 'org.apache.solr.core.RuntimeLibReqHandler' package : global }}")
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties(); Map<String, Object> map = new ClusterProperties(cluster.getZkClient()).getClusterProperties();
@ -375,7 +380,7 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
assertEquals("org.apache.solr.core.RuntimeLibReqHandler", assertEquals("org.apache.solr.core.RuntimeLibReqHandler",
getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class"))); getObjectByPath(map, true, Arrays.asList("requestHandler", "bar", "class")));
payload = "{update-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar3.jar', " + payload = "{update-package:{name : 'global', url: 'http://localhost:" + port + "/jar3.jar', " +
"sig : 'a400n4T7FT+2gM0SC6+MfSOExjud8MkhTSFylhvwNjtWwUgKdPFn434Wv7Qc4QEqDVLhQoL3WqYtQmLPti0G4Q==' ," + "sig : 'a400n4T7FT+2gM0SC6+MfSOExjud8MkhTSFylhvwNjtWwUgKdPFn434Wv7Qc4QEqDVLhQoL3WqYtQmLPti0G4Q==' ," +
"sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}"; "sha512 : '60ec88c2a2e9b409f7afc309273383810a0d07a078b482434eda9674f7e25b8adafa8a67c9913c996cbfb78a7f6ad2b9db26dbd4fe0ca4068f248d5db563f922'}}";
@ -383,8 +388,8 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
request = new V2Request.Builder("/node/ext/bar") request = new V2Request.Builder("/node/ext/bar")
@ -457,20 +462,20 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
cluster.waitForActiveCollection(COLLECTION_NAME, 2, 2); cluster.waitForActiveCollection(COLLECTION_NAME, 2, 2);
String payload = "{add-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar1.jar', " + String payload = "{add-package:{name : 'global', url: 'http://localhost:" + port + "/jar1.jar', " +
"sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}"; "sha512 : 'd01b51de67ae1680a84a813983b1de3b592fc32f1a22b662fc9057da5953abd1b72476388ba342cad21671cd0b805503c78ab9075ff2f3951fdf75fa16981420'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
payload = "{\n" + payload = "{\n" +
"'create-requesthandler' : { 'name' : '/runtime', 'class': 'org.apache.solr.core.RuntimeLibReqHandler' , 'runtimeLib':global }," + "'create-requesthandler' : { 'name' : '/runtime', 'class': 'org.apache.solr.core.RuntimeLibReqHandler' , 'package':global }," +
"'create-searchcomponent' : { 'name' : 'get', 'class': 'org.apache.solr.core.RuntimeLibSearchComponent' , 'runtimeLib':global }," + "'create-searchcomponent' : { 'name' : 'get', 'class': 'org.apache.solr.core.RuntimeLibSearchComponent' , 'package':global }," +
"'create-queryResponseWriter' : { 'name' : 'json1', 'class': 'org.apache.solr.core.RuntimeLibResponseWriter' , 'runtimeLib':global }" + "'create-queryResponseWriter' : { 'name' : 'json1', 'class': 'org.apache.solr.core.RuntimeLibResponseWriter' , 'package':global }" +
"}"; "}";
cluster.getSolrClient().request(new ConfigRequest(payload) { cluster.getSolrClient().request(new ConfigRequest(payload) {
@Override @Override
@ -528,14 +533,14 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
"loader", MemClassLoader.class.getName())); "loader", MemClassLoader.class.getName()));
payload = "{update-runtimelib:{name : 'foo', url: 'http://localhost:" + port + "/jar2.jar', " + payload = "{update-package:{name : 'global', url: 'http://localhost:" + port + "/jar2.jar', " +
"sha512 : 'bc5ce45ad281b6a08fb7e529b1eb475040076834816570902acb6ebdd809410e31006efdeaa7f78a6c35574f3504963f5f7e4d92247d0eb4db3fc9abdda5d417'}}"; "sha512 : 'bc5ce45ad281b6a08fb7e529b1eb475040076834816570902acb6ebdd809410e31006efdeaa7f78a6c35574f3504963f5f7e4d92247d0eb4db3fc9abdda5d417'}}";
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
.withPayload(payload) .withPayload(payload)
.withMethod(SolrRequest.METHOD.POST) .withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient()); .build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-runtimelib/sha512"), assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "runtimeLib/foo/sha512")); getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/global/sha512"));
try { try {
new V2Request.Builder("/cluster") new V2Request.Builder("/cluster")
@ -562,4 +567,89 @@ public class TestContainerReqHandler extends SolrCloudTestCase {
} }
public void testCacheFromGlobalLoader() throws Exception {
String COLLECTION_NAME = "globalCacheColl";
Map<String, Object> jars = Utils.makeMap(
"/jar1.jar", getFileContent("runtimecode/cache.jar.bin"),
"/jar2.jar", getFileContent("runtimecode/cache_v2.jar.bin"));
Pair<Server, Integer> server = runHttpServer(jars);
int port = server.second();
String overlay = "{" +
" \"props\":{\"query\":{\"documentCache\":{\n" +
" \"class\":\"org.apache.solr.core.MyDocCache\",\n" +
" \"size\":\"512\",\n" +
" \"initialSize\":\"512\" , \"package\":\"cache_pkg\"}}}}";
MiniSolrCloudCluster cluster = configureCluster(4)
.addConfig("conf", configset("cloud-minimal"),
Collections.singletonMap(ConfigOverlay.RESOURCE_NAME, overlay.getBytes()))
.configure();
try {
String payload = "{add-package:{name : 'cache_pkg', url: 'http://localhost:" + port + "/jar1.jar', " +
"sha512 : '1a3739b629ce85895c9b2a8c12dd7d98161ff47634b0693f1e1c5b444fb38343f95c6ee955cd99103bd24cfde6c205234b63823818660ac08392cdc626caf585'}}";
new V2Request.Builder("/cluster")
.withPayload(payload)
.withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "add-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/cache_pkg/sha512"));
CollectionAdminRequest
.createCollection(COLLECTION_NAME, "conf", 2, 1)
.setMaxShardsPerNode(100)
.process(cluster.getSolrClient());
cluster.waitForActiveCollection(COLLECTION_NAME, 2, 2);
SolrParams params = new MapSolrParams((Map) Utils.makeMap("collection", COLLECTION_NAME, WT, JAVABIN));
NamedList<Object> rsp = cluster.getSolrClient().request(new GenericSolrRequest(SolrRequest.METHOD.GET, "/config/overlay", params));
assertEquals("org.apache.solr.core.MyDocCache", rsp._getStr("overlay/props/query/documentCache/class", null));
UpdateRequest req = new UpdateRequest();
req.add("id", "1", "desc_s", "document 1")
.setAction(AbstractUpdateRequest.ACTION.COMMIT, true, true)
.setWaitSearcher(true);
cluster.getSolrClient().request(req, COLLECTION_NAME);
SolrQuery solrQuery = new SolrQuery("q", "*:*", "collection", COLLECTION_NAME);
assertResponseValues(10,
cluster.getSolrClient(),
new QueryRequest(solrQuery),
Utils.makeMap("response[0]/id", "1"));
payload = "{update-package:{name : 'cache_pkg', url: 'http://localhost:" + port + "/jar2.jar', " +
"sha512 : 'aa3f42fb640636dd8126beca36ac389486d0fcb1c3a2e2c387d043d57637535ce8db3b17983853322f78bb8f447ed75fe7b405675debe652ed826ee95e8ce328'}}";
new V2Request.Builder("/cluster")
.withPayload(payload)
.withMethod(SolrRequest.METHOD.POST)
.build().process(cluster.getSolrClient());
assertEquals(getObjectByPath(Utils.fromJSONString(payload), true, "update-package/sha512"),
getObjectByPath(new ClusterProperties(cluster.getZkClient()).getClusterProperties(), true, "package/cache_pkg/sha512"));
req = new UpdateRequest();
req.add("id", "2", "desc_s", "document 1")
.setAction(AbstractUpdateRequest.ACTION.COMMIT, true, true)
.setWaitSearcher(true);
cluster.getSolrClient().request(req, COLLECTION_NAME);
solrQuery = new SolrQuery("q", "id:2", "collection", COLLECTION_NAME);
SolrResponse result = assertResponseValues(10,
cluster.getSolrClient(),
new QueryRequest(solrQuery),
Utils.makeMap("response[0]/my_synthetic_fld_s", "version_2"));
} finally {
cluster.deleteAllCollections();
cluster.shutdown();
server.first().stop();
}
}
} }

View File

@ -208,15 +208,15 @@ public class CollectionApiMapping {
null, null,
"set-obj-property", null), "set-obj-property", null),
ADD_RUNTIME_LIB(CLUSTER_CMD, ADD_PACKAGE(CLUSTER_CMD,
POST,null, POST,null,
"add-runtimelib",null ), "add-package",null ),
UPDATE_RUNTIME_LIB(CLUSTER_CMD, UPDATE_PACKAGE(CLUSTER_CMD,
POST,null, POST,null,
"update-runtimelib",null ), "update-package",null ),
DELETE_RUNTIME_LIB(CLUSTER_CMD, DELETE_RUNTIME_LIB(CLUSTER_CMD,
POST,null, POST,null,
"delete-runtimelib",null ), "delete-package",null ),
ADD_REQ_HANDLER(CLUSTER_CMD, ADD_REQ_HANDLER(CLUSTER_CMD,
POST,null, POST,null,
"add-requesthandler",null ), "add-requesthandler",null ),

View File

@ -296,5 +296,7 @@ public interface CommonParams {
String JAVABIN_MIME = "application/javabin"; String JAVABIN_MIME = "application/javabin";
String PACKAGE = "package";
} }

View File

@ -170,17 +170,17 @@
"name" "name"
] ]
}, },
"add-runtimelib": { "add-package": {
"documentation": "", "documentation": "",
"description" : "Add a remote jar to the classpath", "description" : "Add a package to the classpath",
"#include": "cluster.Commands.runtimelib.properties" "#include": "cluster.Commands.runtimelib.properties"
}, },
"update-runtimelib": { "update-package": {
"documentation": "", "documentation": "",
"description" : "Update the jar details", "description" : "Update the jar details",
"#include": "cluster.Commands.runtimelib.properties" "#include": "cluster.Commands.runtimelib.properties"
}, },
"delete-runtimelib": { "delete-package": {
"documentation": "", "documentation": "",
"description" : "delete a lib", "description" : "delete a lib",
"type": "string" "type": "string"
@ -197,6 +197,10 @@
"class": { "class": {
"type": "string", "type": "string",
"description": "The class name" "description": "The class name"
},
"package" : {
"type": "string",
"description": " The package from where the plugin can be loaded from"
} }
}, },
"required": ["name", "class"], "required": ["name", "class"],

View File

@ -54,11 +54,14 @@ import org.apache.solr.common.cloud.SolrZkClient;
import org.apache.solr.common.cloud.ZkStateReader; import org.apache.solr.common.cloud.ZkStateReader;
import org.apache.solr.common.params.CollectionAdminParams; import org.apache.solr.common.params.CollectionAdminParams;
import org.apache.solr.common.util.NamedList; import org.apache.solr.common.util.NamedList;
import org.apache.zookeeper.CreateMode;
import org.junit.AfterClass; import org.junit.AfterClass;
import org.junit.Before; import org.junit.Before;
import org.slf4j.Logger; import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import org.slf4j.LoggerFactory;
import static org.apache.solr.common.cloud.ZkConfigManager.CONFIGS_ZKNODE;
/** /**
* Base class for SolrCloud tests * Base class for SolrCloud tests
* *
@ -86,10 +89,12 @@ public class SolrCloudTestCase extends SolrTestCaseJ4 {
private static class Config { private static class Config {
final String name; final String name;
final Path path; final Path path;
final Map<String,byte[]> extraConfig;
private Config(String name, Path path) { private Config(String name, Path path, Map<String,byte[]> extraConfig) {
this.name = name; this.name = name;
this.path = path; this.path = path;
this.extraConfig = extraConfig;
} }
} }
@ -178,7 +183,12 @@ public class SolrCloudTestCase extends SolrTestCaseJ4 {
* @param configPath the path to the config files * @param configPath the path to the config files
*/ */
public Builder addConfig(String configName, Path configPath) { public Builder addConfig(String configName, Path configPath) {
this.configs.add(new Config(configName, configPath)); this.configs.add(new Config(configName, configPath, null));
return this;
}
public Builder addConfig(String configName, Path configPath, Map<String, byte[]> extraConfig) {
this.configs.add(new Config(configName, configPath, extraConfig));
return this; return this;
} }
@ -214,6 +224,14 @@ public class SolrCloudTestCase extends SolrTestCaseJ4 {
CloudSolrClient client = cluster.getSolrClient(); CloudSolrClient client = cluster.getSolrClient();
for (Config config : configs) { for (Config config : configs) {
((ZkClientClusterStateProvider)client.getClusterStateProvider()).uploadConfig(config.path, config.name); ((ZkClientClusterStateProvider)client.getClusterStateProvider()).uploadConfig(config.path, config.name);
if(config.extraConfig!= null){
for (Map.Entry<String, byte[]> e : config.extraConfig.entrySet()) {
((ZkClientClusterStateProvider)client.getClusterStateProvider()).getZkStateReader().getZkClient()
.create(CONFIGS_ZKNODE + "/" + config.name+ "/"+ e.getKey(), e.getValue(), CreateMode.PERSISTENT, true);
}
}
} }
if (clusterProperties.size() > 0) { if (clusterProperties.size() > 0) {