mirror of https://github.com/apache/lucene.git
SOLR-9106: Cache cluster properties on ZkStateReader
This commit is contained in:
parent
ec4bcf1c70
commit
dd23fa4015
|
@ -208,6 +208,8 @@ Optimizations
|
||||||
* SOLR-9014: Deprecate and reduce usage of ClusterState methods which may make calls to ZK via
|
* SOLR-9014: Deprecate and reduce usage of ClusterState methods which may make calls to ZK via
|
||||||
the lazy collection reference. (Scott Blum, shalin)
|
the lazy collection reference. (Scott Blum, shalin)
|
||||||
|
|
||||||
|
* SOLR-9106: Cluster properties are now cached on ZkStateReader. (Alan Woodward)
|
||||||
|
|
||||||
Other Changes
|
Other Changes
|
||||||
----------------------
|
----------------------
|
||||||
* SOLR-7516: Improve javadocs for JavaBinCodec, ObjectResolver and enforce the single-usage policy.
|
* SOLR-7516: Improve javadocs for JavaBinCodec, ObjectResolver and enforce the single-usage policy.
|
||||||
|
|
|
@ -1062,8 +1062,9 @@ public class Overseer implements Closeable {
|
||||||
throw new RuntimeException(e);
|
throw new RuntimeException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
public static boolean isLegacy(Map clusterProps) {
|
public static boolean isLegacy(ZkStateReader stateReader) {
|
||||||
return !"false".equals(clusterProps.get(ZkStateReader.LEGACY_CLOUD));
|
String legacyProperty = stateReader.getClusterProperty(ZkStateReader.LEGACY_CLOUD, "true");
|
||||||
|
return !"false".equals(legacyProperty);
|
||||||
}
|
}
|
||||||
|
|
||||||
public ZkStateReader getZkStateReader() {
|
public ZkStateReader getZkStateReader() {
|
||||||
|
|
|
@ -24,7 +24,6 @@ import java.util.Collections;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.TreeMap;
|
import java.util.TreeMap;
|
||||||
|
@ -148,7 +147,7 @@ public class OverseerAutoReplicaFailoverThread implements Runnable, Closeable {
|
||||||
// TODO: extract to configurable strategy class ??
|
// TODO: extract to configurable strategy class ??
|
||||||
ClusterState clusterState = zkStateReader.getClusterState();
|
ClusterState clusterState = zkStateReader.getClusterState();
|
||||||
//check if we have disabled autoAddReplicas cluster wide
|
//check if we have disabled autoAddReplicas cluster wide
|
||||||
String autoAddReplicas = (String) zkStateReader.getClusterProps().get(ZkStateReader.AUTO_ADD_REPLICAS);
|
String autoAddReplicas = zkStateReader.getClusterProperty(ZkStateReader.AUTO_ADD_REPLICAS, (String) null);
|
||||||
if (autoAddReplicas != null && autoAddReplicas.equals("false")) {
|
if (autoAddReplicas != null && autoAddReplicas.equals("false")) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
@ -229,7 +228,7 @@ public class OverseerAutoReplicaFailoverThread implements Runnable, Closeable {
|
||||||
private boolean addReplica(final String collection, DownReplica badReplica) {
|
private boolean addReplica(final String collection, DownReplica badReplica) {
|
||||||
// first find best home - first strategy, sort by number of cores
|
// first find best home - first strategy, sort by number of cores
|
||||||
// hosted where maxCoresPerNode is not violated
|
// hosted where maxCoresPerNode is not violated
|
||||||
final Integer maxCoreCount = (Integer) zkStateReader.getClusterProps().get(ZkStateReader.MAX_CORES_PER_NODE);
|
final Integer maxCoreCount = zkStateReader.getClusterProperty(ZkStateReader.MAX_CORES_PER_NODE, (Integer) null);
|
||||||
final String createUrl = getBestCreateUrl(zkStateReader, badReplica, maxCoreCount);
|
final String createUrl = getBestCreateUrl(zkStateReader, badReplica, maxCoreCount);
|
||||||
if (createUrl == null) {
|
if (createUrl == null) {
|
||||||
log.warn("Could not find a node to create new replica on.");
|
log.warn("Could not find a node to create new replica on.");
|
||||||
|
|
|
@ -1894,7 +1894,7 @@ public class OverseerCollectionMessageHandler implements OverseerMessageHandler
|
||||||
positionVsNodes = identifyNodes(clusterState, nodeList, message, shardNames, repFactor);
|
positionVsNodes = identifyNodes(clusterState, nodeList, message, shardNames, repFactor);
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean isLegacyCloud = Overseer.isLegacy(zkStateReader.getClusterProps());
|
boolean isLegacyCloud = Overseer.isLegacy(zkStateReader);
|
||||||
|
|
||||||
createConfNode(configName, collectionName, isLegacyCloud);
|
createConfNode(configName, collectionName, isLegacyCloud);
|
||||||
|
|
||||||
|
@ -2126,7 +2126,7 @@ public class OverseerCollectionMessageHandler implements OverseerMessageHandler
|
||||||
}
|
}
|
||||||
ModifiableSolrParams params = new ModifiableSolrParams();
|
ModifiableSolrParams params = new ModifiableSolrParams();
|
||||||
|
|
||||||
if (!Overseer.isLegacy(zkStateReader.getClusterProps())) {
|
if (!Overseer.isLegacy(zkStateReader)) {
|
||||||
if (!skipCreateReplicaInClusterState) {
|
if (!skipCreateReplicaInClusterState) {
|
||||||
ZkNodeProps props = new ZkNodeProps(Overseer.QUEUE_OPERATION, ADDREPLICA.toLower(), ZkStateReader.COLLECTION_PROP,
|
ZkNodeProps props = new ZkNodeProps(Overseer.QUEUE_OPERATION, ADDREPLICA.toLower(), ZkStateReader.COLLECTION_PROP,
|
||||||
collection, ZkStateReader.SHARD_ID_PROP, shard, ZkStateReader.CORE_NAME_PROP, coreName,
|
collection, ZkStateReader.SHARD_ID_PROP, shard, ZkStateReader.CORE_NAME_PROP, coreName,
|
||||||
|
|
|
@ -16,7 +16,16 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
import static org.apache.solr.common.params.CommonParams.*;
|
import javax.xml.parsers.ParserConfigurationException;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.nio.file.Paths;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.TimeoutException;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
import org.apache.commons.cli.CommandLine;
|
import org.apache.commons.cli.CommandLine;
|
||||||
import org.apache.commons.cli.CommandLineParser;
|
import org.apache.commons.cli.CommandLineParser;
|
||||||
|
@ -28,26 +37,16 @@ import org.apache.commons.cli.ParseException;
|
||||||
import org.apache.commons.cli.PosixParser;
|
import org.apache.commons.cli.PosixParser;
|
||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.cloud.ClusterProperties;
|
||||||
import org.apache.solr.common.cloud.SolrZkClient;
|
import org.apache.solr.common.cloud.SolrZkClient;
|
||||||
import org.apache.solr.common.cloud.ZkConfigManager;
|
import org.apache.solr.common.cloud.ZkConfigManager;
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
|
||||||
import org.apache.solr.core.CoreContainer;
|
import org.apache.solr.core.CoreContainer;
|
||||||
import org.apache.zookeeper.CreateMode;
|
import org.apache.zookeeper.CreateMode;
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.xml.sax.SAXException;
|
import org.xml.sax.SAXException;
|
||||||
|
|
||||||
import javax.xml.parsers.ParserConfigurationException;
|
import static org.apache.solr.common.params.CommonParams.NAME;
|
||||||
|
import static org.apache.solr.common.params.CommonParams.VALUE_LONG;
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
import java.nio.file.Paths;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.concurrent.TimeoutException;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
public class ZkCLI {
|
public class ZkCLI {
|
||||||
|
|
||||||
|
@ -324,28 +323,12 @@ public class ZkCLI {
|
||||||
//If -val option is missing, we will use the null value. This is required to maintain
|
//If -val option is missing, we will use the null value. This is required to maintain
|
||||||
//compatibility with Collections API.
|
//compatibility with Collections API.
|
||||||
String propertyValue = line.getOptionValue(VALUE_LONG);
|
String propertyValue = line.getOptionValue(VALUE_LONG);
|
||||||
ZkStateReader reader = new ZkStateReader(zkClient);
|
ClusterProperties props = new ClusterProperties(zkClient);
|
||||||
try {
|
try {
|
||||||
reader.setClusterProperty(propertyName, propertyValue);
|
props.setClusterProperty(propertyName, propertyValue);
|
||||||
} catch (SolrException ex) {
|
} catch (IOException ex) {
|
||||||
//This can happen if two concurrent invocations of this command collide
|
System.out.println("Unable to set the cluster property due to following error : " + ex.getLocalizedMessage());
|
||||||
//with each other. Here we are just adding a defensive check to see if
|
|
||||||
//the value is already set to expected value. If yes, then we don't
|
|
||||||
//fail the command.
|
|
||||||
Throwable cause = ex.getCause();
|
|
||||||
if(cause instanceof KeeperException.NodeExistsException
|
|
||||||
|| cause instanceof KeeperException.BadVersionException) {
|
|
||||||
String currentValue = (String)reader.getClusterProps().get(propertyName);
|
|
||||||
if((currentValue == propertyValue) || (currentValue != null && currentValue.equals(propertyValue))) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
System.out.println("Unable to set the cluster property due to following error : " +
|
|
||||||
ex.getLocalizedMessage() +
|
|
||||||
((cause instanceof KeeperException.BadVersionException)?". Try again":""));
|
|
||||||
System.exit(1);
|
System.exit(1);
|
||||||
} finally {
|
|
||||||
reader.close();
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// If not cmd matches
|
// If not cmd matches
|
||||||
|
|
|
@ -151,7 +151,7 @@ public final class ZkController {
|
||||||
private final int localHostPort; // example: 54065
|
private final int localHostPort; // example: 54065
|
||||||
private final String hostName; // example: 127.0.0.1
|
private final String hostName; // example: 127.0.0.1
|
||||||
private final String nodeName; // example: 127.0.0.1:54065_solr
|
private final String nodeName; // example: 127.0.0.1:54065_solr
|
||||||
private final String baseURL; // example: http://127.0.0.1:54065/solr
|
private String baseURL; // example: http://127.0.0.1:54065/solr
|
||||||
|
|
||||||
private final CloudConfig cloudConfig;
|
private final CloudConfig cloudConfig;
|
||||||
|
|
||||||
|
@ -386,8 +386,6 @@ public final class ZkController {
|
||||||
if (cc != null) cc.securityNodeChanged();
|
if (cc != null) cc.securityNodeChanged();
|
||||||
});
|
});
|
||||||
|
|
||||||
this.baseURL = zkStateReader.getBaseUrlForNodeName(this.nodeName);
|
|
||||||
|
|
||||||
init(registerOnReconnect);
|
init(registerOnReconnect);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -642,6 +640,7 @@ public final class ZkController {
|
||||||
try {
|
try {
|
||||||
createClusterZkNodes(zkClient);
|
createClusterZkNodes(zkClient);
|
||||||
zkStateReader.createClusterStateWatchersAndUpdate();
|
zkStateReader.createClusterStateWatchersAndUpdate();
|
||||||
|
this.baseURL = zkStateReader.getBaseUrlForNodeName(this.nodeName);
|
||||||
|
|
||||||
// start the overseer first as following code may need it's processing
|
// start the overseer first as following code may need it's processing
|
||||||
if (!zkRunOnly) {
|
if (!zkRunOnly) {
|
||||||
|
@ -1488,7 +1487,7 @@ public final class ZkController {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void checkStateInZk(CoreDescriptor cd) throws InterruptedException {
|
private void checkStateInZk(CoreDescriptor cd) throws InterruptedException {
|
||||||
if (!Overseer.isLegacy(zkStateReader.getClusterProps())) {
|
if (!Overseer.isLegacy(zkStateReader)) {
|
||||||
CloudDescriptor cloudDesc = cd.getCloudDescriptor();
|
CloudDescriptor cloudDesc = cd.getCloudDescriptor();
|
||||||
String coreNodeName = cloudDesc.getCoreNodeName();
|
String coreNodeName = cloudDesc.getCoreNodeName();
|
||||||
assert coreNodeName != null : "SolrCore: " + cd.getName() + " has no coreNodeName";
|
assert coreNodeName != null : "SolrCore: " + cd.getName() + " has no coreNodeName";
|
||||||
|
|
|
@ -197,7 +197,7 @@ public class ReplicaMutator {
|
||||||
}
|
}
|
||||||
|
|
||||||
public ZkWriteCommand setState(ClusterState clusterState, ZkNodeProps message) {
|
public ZkWriteCommand setState(ClusterState clusterState, ZkNodeProps message) {
|
||||||
if (Overseer.isLegacy(zkStateReader.getClusterProps())) {
|
if (Overseer.isLegacy(zkStateReader)) {
|
||||||
return updateState(clusterState, message);
|
return updateState(clusterState, message);
|
||||||
} else {
|
} else {
|
||||||
return updateStateNew(clusterState, message);
|
return updateStateNew(clusterState, message);
|
||||||
|
|
|
@ -737,7 +737,7 @@ public class CoreContainer {
|
||||||
boolean preExisitingZkEntry = false;
|
boolean preExisitingZkEntry = false;
|
||||||
try {
|
try {
|
||||||
if (getZkController() != null) {
|
if (getZkController() != null) {
|
||||||
if (!Overseer.isLegacy(getZkController().getZkStateReader().getClusterProps())) {
|
if (!Overseer.isLegacy(getZkController().getZkStateReader())) {
|
||||||
if (cd.getCloudDescriptor().getCoreNodeName() == null) {
|
if (cd.getCloudDescriptor().getCoreNodeName() == null) {
|
||||||
throw new SolrException(ErrorCode.SERVER_ERROR, "non legacy mode coreNodeName missing " + parameters.toString());
|
throw new SolrException(ErrorCode.SERVER_ERROR, "non legacy mode coreNodeName missing " + parameters.toString());
|
||||||
|
|
||||||
|
|
|
@ -152,7 +152,7 @@ public class ClusterStatus {
|
||||||
clusterStatus.add("collections", collectionProps);
|
clusterStatus.add("collections", collectionProps);
|
||||||
|
|
||||||
// read cluster properties
|
// read cluster properties
|
||||||
Map clusterProps = zkStateReader.getClusterProps();
|
Map clusterProps = zkStateReader.getClusterProperties();
|
||||||
if (clusterProps != null && !clusterProps.isEmpty()) {
|
if (clusterProps != null && !clusterProps.isEmpty()) {
|
||||||
clusterStatus.add("properties", clusterProps);
|
clusterStatus.add("properties", clusterProps);
|
||||||
}
|
}
|
||||||
|
|
|
@ -51,17 +51,8 @@ import org.apache.solr.cloud.rule.ReplicaAssigner;
|
||||||
import org.apache.solr.cloud.rule.Rule;
|
import org.apache.solr.cloud.rule.Rule;
|
||||||
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;
|
||||||
import org.apache.solr.common.cloud.ClusterState;
|
import org.apache.solr.common.cloud.*;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
|
||||||
import org.apache.solr.common.cloud.ImplicitDocRouter;
|
|
||||||
import org.apache.solr.common.cloud.Replica;
|
|
||||||
import org.apache.solr.common.cloud.Replica.State;
|
import org.apache.solr.common.cloud.Replica.State;
|
||||||
import org.apache.solr.common.cloud.Slice;
|
|
||||||
import org.apache.solr.common.cloud.SolrZkClient;
|
|
||||||
import org.apache.solr.common.cloud.ZkCmdExecutor;
|
|
||||||
import org.apache.solr.common.cloud.ZkCoreNodeProps;
|
|
||||||
import org.apache.solr.common.cloud.ZkNodeProps;
|
|
||||||
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.params.CollectionParams;
|
import org.apache.solr.common.params.CollectionParams;
|
||||||
import org.apache.solr.common.params.CollectionParams.CollectionAction;
|
import org.apache.solr.common.params.CollectionParams.CollectionAction;
|
||||||
|
@ -570,7 +561,8 @@ public class CollectionsHandler extends RequestHandlerBase implements Permission
|
||||||
Map<String, Object> call(SolrQueryRequest req, SolrQueryResponse rsp, CollectionsHandler h) throws Exception {
|
Map<String, Object> call(SolrQueryRequest req, SolrQueryResponse rsp, CollectionsHandler h) throws Exception {
|
||||||
String name = req.getParams().required().get(NAME);
|
String name = req.getParams().required().get(NAME);
|
||||||
String val = req.getParams().get(VALUE_LONG);
|
String val = req.getParams().get(VALUE_LONG);
|
||||||
h.coreContainer.getZkController().getZkStateReader().setClusterProperty(name, val);
|
ClusterProperties cp = new ClusterProperties(h.coreContainer.getZkController().getZkClient());
|
||||||
|
cp.setClusterProperty(name, val);
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -808,7 +800,7 @@ public class CollectionsHandler extends RequestHandlerBase implements Permission
|
||||||
|
|
||||||
String location = req.getParams().get("location");
|
String location = req.getParams().get("location");
|
||||||
if (location == null) {
|
if (location == null) {
|
||||||
location = (String) h.coreContainer.getZkController().getZkStateReader().getClusterProps().get("location");
|
location = h.coreContainer.getZkController().getZkStateReader().getClusterProperty("location", (String) null);
|
||||||
}
|
}
|
||||||
if (location == null) {
|
if (location == null) {
|
||||||
throw new SolrException(ErrorCode.BAD_REQUEST, "'location' is not specified as a query parameter or set as a cluster property");
|
throw new SolrException(ErrorCode.BAD_REQUEST, "'location' is not specified as a query parameter or set as a cluster property");
|
||||||
|
@ -832,7 +824,7 @@ public class CollectionsHandler extends RequestHandlerBase implements Permission
|
||||||
|
|
||||||
String location = req.getParams().get("location");
|
String location = req.getParams().get("location");
|
||||||
if (location == null) {
|
if (location == null) {
|
||||||
location = (String) h.coreContainer.getZkController().getZkStateReader().getClusterProps().get("location");
|
location = h.coreContainer.getZkController().getZkStateReader().getClusterProperty("location", (String) null);
|
||||||
}
|
}
|
||||||
if (location == null) {
|
if (location == null) {
|
||||||
throw new SolrException(ErrorCode.BAD_REQUEST, "'location' is not specified as a query parameter or set as a cluster property");
|
throw new SolrException(ErrorCode.BAD_REQUEST, "'location' is not specified as a query parameter or set as a cluster property");
|
||||||
|
|
|
@ -1320,7 +1320,7 @@ public class CollectionsAPIDistributedZkTest extends AbstractFullDistribZkTestBa
|
||||||
boolean changed = false;
|
boolean changed = false;
|
||||||
while(! timeout.hasTimedOut()){
|
while(! timeout.hasTimedOut()){
|
||||||
Thread.sleep(10);
|
Thread.sleep(10);
|
||||||
changed = Objects.equals(val,client.getZkStateReader().getClusterProps().get(name));
|
changed = Objects.equals(val,client.getZkStateReader().getClusterProperty(name, (String) null));
|
||||||
if(changed) break;
|
if(changed) break;
|
||||||
}
|
}
|
||||||
return changed;
|
return changed;
|
||||||
|
|
|
@ -45,7 +45,6 @@ import org.apache.solr.common.params.CoreAdminParams;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
import org.apache.solr.util.TimeOut;
|
import org.apache.solr.util.TimeOut;
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.junit.Ignore;
|
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import static org.apache.solr.cloud.ReplicaPropertiesBase.verifyUniqueAcrossCollection;
|
import static org.apache.solr.cloud.ReplicaPropertiesBase.verifyUniqueAcrossCollection;
|
||||||
|
@ -336,7 +335,7 @@ public class CollectionsAPISolrJTest extends AbstractFullDistribZkTestBase {
|
||||||
while(! timeout.hasTimedOut()){
|
while(! timeout.hasTimedOut()){
|
||||||
Thread.sleep(10);
|
Thread.sleep(10);
|
||||||
changed = Objects.equals("false",
|
changed = Objects.equals("false",
|
||||||
cloudClient.getZkStateReader().getClusterProps().get(ZkStateReader.LEGACY_CLOUD));
|
cloudClient.getZkStateReader().getClusterProperty(ZkStateReader.LEGACY_CLOUD, "none"));
|
||||||
if(changed) break;
|
if(changed) break;
|
||||||
}
|
}
|
||||||
assertTrue("The Cluster property wasn't set", changed);
|
assertTrue("The Cluster property wasn't set", changed);
|
||||||
|
@ -351,7 +350,7 @@ public class CollectionsAPISolrJTest extends AbstractFullDistribZkTestBase {
|
||||||
changed = false;
|
changed = false;
|
||||||
while(! timeout.hasTimedOut()) {
|
while(! timeout.hasTimedOut()) {
|
||||||
Thread.sleep(10);
|
Thread.sleep(10);
|
||||||
changed = (cloudClient.getZkStateReader().getClusterProps().get(ZkStateReader.LEGACY_CLOUD) == null);
|
changed = (cloudClient.getZkStateReader().getClusterProperty(ZkStateReader.LEGACY_CLOUD, (String) null) == null);
|
||||||
if(changed)
|
if(changed)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,11 +16,17 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
import java.util.concurrent.ArrayBlockingQueue;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
import org.apache.solr.client.solrj.SolrResponse;
|
import org.apache.solr.client.solrj.SolrResponse;
|
||||||
import org.apache.solr.client.solrj.response.QueryResponse;
|
import org.apache.solr.client.solrj.response.QueryResponse;
|
||||||
import org.apache.solr.cloud.OverseerTaskQueue.QueueEvent;
|
|
||||||
import org.apache.solr.cloud.Overseer.LeaderStatus;
|
import org.apache.solr.cloud.Overseer.LeaderStatus;
|
||||||
|
import org.apache.solr.cloud.OverseerTaskQueue.QueueEvent;
|
||||||
import org.apache.solr.common.cloud.ClusterState;
|
import org.apache.solr.common.cloud.ClusterState;
|
||||||
import org.apache.solr.common.cloud.SolrZkClient;
|
import org.apache.solr.common.cloud.SolrZkClient;
|
||||||
import org.apache.solr.common.cloud.ZkNodeProps;
|
import org.apache.solr.common.cloud.ZkNodeProps;
|
||||||
|
@ -48,32 +54,7 @@ import org.junit.Test;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.lang.invoke.MethodHandles;
|
import static org.easymock.EasyMock.*;
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Map.Entry;
|
|
||||||
import java.util.Queue;
|
|
||||||
import java.util.Set;
|
|
||||||
import java.util.concurrent.ArrayBlockingQueue;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
import static org.easymock.EasyMock.anyBoolean;
|
|
||||||
import static org.easymock.EasyMock.anyObject;
|
|
||||||
import static org.easymock.EasyMock.capture;
|
|
||||||
import static org.easymock.EasyMock.createMock;
|
|
||||||
import static org.easymock.EasyMock.expect;
|
|
||||||
import static org.easymock.EasyMock.expectLastCall;
|
|
||||||
import static org.easymock.EasyMock.getCurrentArguments;
|
|
||||||
import static org.easymock.EasyMock.replay;
|
|
||||||
import static org.easymock.EasyMock.reset;
|
|
||||||
import static org.easymock.EasyMock.verify;
|
|
||||||
|
|
||||||
public class OverseerCollectionConfigSetProcessorTest extends SolrTestCaseJ4 {
|
public class OverseerCollectionConfigSetProcessorTest extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
|
@ -284,11 +265,12 @@ public class OverseerCollectionConfigSetProcessorTest extends SolrTestCaseJ4 {
|
||||||
}).anyTimes();
|
}).anyTimes();
|
||||||
|
|
||||||
}
|
}
|
||||||
zkStateReaderMock.getClusterProps();
|
|
||||||
expectLastCall().andAnswer(new IAnswer<Map>() {
|
zkStateReaderMock.getClusterProperty("legacyCloud", "true");
|
||||||
|
expectLastCall().andAnswer(new IAnswer<String>() {
|
||||||
@Override
|
@Override
|
||||||
public Map answer() throws Throwable {
|
public String answer() throws Throwable {
|
||||||
return new HashMap();
|
return "true";
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -16,6 +16,15 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.FileNotFoundException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.lang.invoke.MethodHandles;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.util.Collection;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.commons.io.FileUtils;
|
import org.apache.commons.io.FileUtils;
|
||||||
import org.apache.commons.io.IOUtils;
|
import org.apache.commons.io.IOUtils;
|
||||||
import org.apache.commons.io.filefilter.RegexFileFilter;
|
import org.apache.commons.io.filefilter.RegexFileFilter;
|
||||||
|
@ -23,6 +32,7 @@ import org.apache.commons.io.filefilter.TrueFileFilter;
|
||||||
import org.apache.solr.SolrJettyTestBase;
|
import org.apache.solr.SolrJettyTestBase;
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
import org.apache.solr.common.SolrException;
|
import org.apache.solr.common.SolrException;
|
||||||
|
import org.apache.solr.common.cloud.ClusterProperties;
|
||||||
import org.apache.solr.common.cloud.SolrZkClient;
|
import org.apache.solr.common.cloud.SolrZkClient;
|
||||||
import org.apache.solr.common.cloud.VMParamsAllAndReadonlyDigestZkACLProvider;
|
import org.apache.solr.common.cloud.VMParamsAllAndReadonlyDigestZkACLProvider;
|
||||||
import org.apache.solr.common.cloud.ZkConfigManager;
|
import org.apache.solr.common.cloud.ZkConfigManager;
|
||||||
|
@ -37,15 +47,6 @@ import org.junit.Test;
|
||||||
import org.slf4j.Logger;
|
import org.slf4j.Logger;
|
||||||
import org.slf4j.LoggerFactory;
|
import org.slf4j.LoggerFactory;
|
||||||
|
|
||||||
import java.io.File;
|
|
||||||
import java.io.FileInputStream;
|
|
||||||
import java.io.FileNotFoundException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.lang.invoke.MethodHandles;
|
|
||||||
import java.nio.charset.StandardCharsets;
|
|
||||||
import java.util.Collection;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
// TODO: This test would be a lot faster if it used a solrhome with fewer config
|
// TODO: This test would be a lot faster if it used a solrhome with fewer config
|
||||||
// files - there are a lot of them to upload
|
// files - there are a lot of them to upload
|
||||||
public class ZkCLITest extends SolrTestCaseJ4 {
|
public class ZkCLITest extends SolrTestCaseJ4 {
|
||||||
|
@ -321,22 +322,19 @@ public class ZkCLITest extends SolrTestCaseJ4 {
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testSetClusterProperty() throws Exception {
|
public void testSetClusterProperty() throws Exception {
|
||||||
ZkStateReader reader = new ZkStateReader(zkClient);
|
ClusterProperties properties = new ClusterProperties(zkClient);
|
||||||
try {
|
|
||||||
// add property urlScheme=http
|
// add property urlScheme=http
|
||||||
String[] args = new String[] {"-zkhost", zkServer.getZkAddress(),
|
String[] args = new String[] {"-zkhost", zkServer.getZkAddress(),
|
||||||
"-cmd", "CLUSTERPROP", "-name", "urlScheme", "-val", "http"};
|
"-cmd", "CLUSTERPROP", "-name", "urlScheme", "-val", "http"};
|
||||||
ZkCLI.main(args);
|
ZkCLI.main(args);
|
||||||
assertEquals("http", reader.getClusterProps().get("urlScheme"));
|
assertEquals("http", properties.getClusterProperty("urlScheme", "none"));
|
||||||
|
|
||||||
// remove it again
|
// remove it again
|
||||||
args = new String[] {"-zkhost", zkServer.getZkAddress(),
|
args = new String[] {"-zkhost", zkServer.getZkAddress(),
|
||||||
"-cmd", "CLUSTERPROP", "-name", "urlScheme"};
|
"-cmd", "CLUSTERPROP", "-name", "urlScheme"};
|
||||||
ZkCLI.main(args);
|
ZkCLI.main(args);
|
||||||
assertNull(reader.getClusterProps().get("urlScheme"));
|
assertNull(properties.getClusterProperty("urlScheme", (String) null));
|
||||||
} finally {
|
|
||||||
reader.close();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
@ -16,17 +16,15 @@
|
||||||
*/
|
*/
|
||||||
package org.apache.solr.cloud;
|
package org.apache.solr.cloud;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import org.apache.lucene.util.LuceneTestCase.Slow;
|
import org.apache.lucene.util.LuceneTestCase.Slow;
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
import org.apache.solr.common.cloud.ClusterState;
|
import org.apache.solr.common.cloud.*;
|
||||||
import org.apache.solr.common.cloud.DocCollection;
|
|
||||||
import org.apache.solr.common.cloud.DocRouter;
|
|
||||||
import org.apache.solr.common.cloud.Replica;
|
|
||||||
import org.apache.solr.common.cloud.Slice;
|
|
||||||
import org.apache.solr.common.cloud.SolrZkClient;
|
|
||||||
import org.apache.solr.common.cloud.ZkConfigManager;
|
|
||||||
import org.apache.solr.common.cloud.ZkNodeProps;
|
|
||||||
import org.apache.solr.common.cloud.ZkStateReader;
|
|
||||||
import org.apache.solr.common.util.Utils;
|
import org.apache.solr.common.util.Utils;
|
||||||
import org.apache.solr.core.CloudConfig;
|
import org.apache.solr.core.CloudConfig;
|
||||||
import org.apache.solr.core.CoreContainer;
|
import org.apache.solr.core.CoreContainer;
|
||||||
|
@ -40,12 +38,6 @@ import org.junit.AfterClass;
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
|
||||||
|
|
||||||
@Slow
|
@Slow
|
||||||
@SolrTestCaseJ4.SuppressSSL
|
@SolrTestCaseJ4.SuppressSSL
|
||||||
public class ZkControllerTest extends SolrTestCaseJ4 {
|
public class ZkControllerTest extends SolrTestCaseJ4 {
|
||||||
|
@ -98,8 +90,13 @@ public class ZkControllerTest extends SolrTestCaseJ4 {
|
||||||
AbstractZkTestCase.tryCleanSolrZkNode(server.getZkHost());
|
AbstractZkTestCase.tryCleanSolrZkNode(server.getZkHost());
|
||||||
AbstractZkTestCase.makeSolrZkNode(server.getZkHost());
|
AbstractZkTestCase.makeSolrZkNode(server.getZkHost());
|
||||||
|
|
||||||
ZkStateReader zkStateReader = new ZkStateReader(server.getZkAddress(), TIMEOUT, TIMEOUT);
|
try (SolrZkClient client = new SolrZkClient(server.getZkAddress(), TIMEOUT)) {
|
||||||
try {
|
|
||||||
|
ZkController.createClusterZkNodes(client);
|
||||||
|
|
||||||
|
try (ZkStateReader zkStateReader = new ZkStateReader(client)) {
|
||||||
|
zkStateReader.createClusterStateWatchersAndUpdate();
|
||||||
|
|
||||||
// getBaseUrlForNodeName
|
// getBaseUrlForNodeName
|
||||||
assertEquals("http://zzz.xxx:1234/solr",
|
assertEquals("http://zzz.xxx:1234/solr",
|
||||||
zkStateReader.getBaseUrlForNodeName("zzz.xxx:1234_solr"));
|
zkStateReader.getBaseUrlForNodeName("zzz.xxx:1234_solr"));
|
||||||
|
@ -113,44 +110,51 @@ public class ZkControllerTest extends SolrTestCaseJ4 {
|
||||||
// generateNodeName + getBaseUrlForNodeName
|
// generateNodeName + getBaseUrlForNodeName
|
||||||
assertEquals("http://foo:9876/solr",
|
assertEquals("http://foo:9876/solr",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo","9876","solr")));
|
(ZkController.generateNodeName("foo", "9876", "solr")));
|
||||||
assertEquals("http://foo:9876/solr",
|
assertEquals("http://foo:9876/solr",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo","9876","/solr")));
|
(ZkController.generateNodeName("foo", "9876", "/solr")));
|
||||||
assertEquals("http://foo:9876/solr",
|
assertEquals("http://foo:9876/solr",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo","9876","/solr/")));
|
(ZkController.generateNodeName("foo", "9876", "/solr/")));
|
||||||
assertEquals("http://foo.bar.com:9876/solr/sub_dir",
|
assertEquals("http://foo.bar.com:9876/solr/sub_dir",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo.bar.com","9876","solr/sub_dir")));
|
(ZkController.generateNodeName("foo.bar.com", "9876", "solr/sub_dir")));
|
||||||
assertEquals("http://foo.bar.com:9876/solr/sub_dir",
|
assertEquals("http://foo.bar.com:9876/solr/sub_dir",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo.bar.com","9876","/solr/sub_dir/")));
|
(ZkController.generateNodeName("foo.bar.com", "9876", "/solr/sub_dir/")));
|
||||||
assertEquals("http://foo-bar:9876",
|
assertEquals("http://foo-bar:9876",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo-bar","9876","")));
|
(ZkController.generateNodeName("foo-bar", "9876", "")));
|
||||||
assertEquals("http://foo-bar:9876",
|
assertEquals("http://foo-bar:9876",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo-bar","9876","/")));
|
(ZkController.generateNodeName("foo-bar", "9876", "/")));
|
||||||
assertEquals("http://foo-bar.com:80/some_dir",
|
assertEquals("http://foo-bar.com:80/some_dir",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo-bar.com","80","some_dir")));
|
(ZkController.generateNodeName("foo-bar.com", "80", "some_dir")));
|
||||||
assertEquals("http://foo-bar.com:80/some_dir",
|
assertEquals("http://foo-bar.com:80/some_dir",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo-bar.com","80","/some_dir")));
|
(ZkController.generateNodeName("foo-bar.com", "80", "/some_dir")));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
ClusterProperties cp = new ClusterProperties(client);
|
||||||
|
cp.setClusterProperty("urlScheme", "https");
|
||||||
|
|
||||||
//Verify the URL Scheme is taken into account
|
//Verify the URL Scheme is taken into account
|
||||||
zkStateReader.getZkClient().create(ZkStateReader.CLUSTER_PROPS,
|
|
||||||
Utils.toJSON(Collections.singletonMap("urlScheme", "https")), CreateMode.PERSISTENT, true);
|
try (ZkStateReader zkStateReader = new ZkStateReader(client)) {
|
||||||
|
|
||||||
|
zkStateReader.createClusterStateWatchersAndUpdate();
|
||||||
|
|
||||||
assertEquals("https://zzz.xxx:1234/solr",
|
assertEquals("https://zzz.xxx:1234/solr",
|
||||||
zkStateReader.getBaseUrlForNodeName("zzz.xxx:1234_solr"));
|
zkStateReader.getBaseUrlForNodeName("zzz.xxx:1234_solr"));
|
||||||
|
|
||||||
assertEquals("https://foo-bar.com:80/some_dir",
|
assertEquals("https://foo-bar.com:80/some_dir",
|
||||||
zkStateReader.getBaseUrlForNodeName
|
zkStateReader.getBaseUrlForNodeName
|
||||||
(ZkController.generateNodeName("foo-bar.com","80","/some_dir")));
|
(ZkController.generateNodeName("foo-bar.com", "80", "/some_dir")));
|
||||||
} finally {
|
|
||||||
zkStateReader.close();
|
}
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
server.shutdown();
|
server.shutdown();
|
||||||
|
|
|
@ -21,21 +21,16 @@ import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.solr.SolrTestCaseJ4;
|
import org.apache.solr.SolrTestCaseJ4;
|
||||||
import org.apache.solr.client.solrj.SolrResponse;
|
|
||||||
import org.apache.solr.client.solrj.embedded.JettySolrRunner;
|
import org.apache.solr.client.solrj.embedded.JettySolrRunner;
|
||||||
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
import org.apache.solr.client.solrj.impl.CloudSolrClient;
|
||||||
import org.apache.solr.client.solrj.request.QueryRequest;
|
import org.apache.solr.client.solrj.request.QueryRequest;
|
||||||
import org.apache.solr.client.solrj.request.SolrPing;
|
|
||||||
import org.apache.solr.client.solrj.response.QueryResponse;
|
import org.apache.solr.client.solrj.response.QueryResponse;
|
||||||
import org.apache.solr.client.solrj.response.SolrPingResponse;
|
|
||||||
import org.apache.solr.cloud.MiniSolrCloudCluster;
|
import org.apache.solr.cloud.MiniSolrCloudCluster;
|
||||||
import org.apache.solr.common.util.NamedList;
|
import org.apache.solr.common.util.NamedList;
|
||||||
import org.apache.solr.core.SolrCore;
|
import org.apache.solr.core.SolrCore;
|
||||||
import org.apache.solr.response.SolrQueryResponse;
|
|
||||||
import org.junit.BeforeClass;
|
import org.junit.BeforeClass;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
|
|
||||||
public class SearchHandlerTest extends SolrTestCaseJ4
|
public class SearchHandlerTest extends SolrTestCaseJ4
|
||||||
{
|
{
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
|
|
|
@ -0,0 +1,126 @@
|
||||||
|
package org.apache.solr.common.cloud;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.solr.common.SolrException;
|
||||||
|
import org.apache.solr.common.util.Utils;
|
||||||
|
import org.apache.zookeeper.CreateMode;
|
||||||
|
import org.apache.zookeeper.KeeperException;
|
||||||
|
import org.apache.zookeeper.data.Stat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interact with solr cluster properties
|
||||||
|
*
|
||||||
|
* Note that all methods on this class make calls to ZK on every invocation. For
|
||||||
|
* read-only eventually-consistent uses, clients should instead call
|
||||||
|
* {@link ZkStateReader#getClusterProperty(String, Object)}
|
||||||
|
*/
|
||||||
|
public class ClusterProperties {
|
||||||
|
|
||||||
|
private final SolrZkClient client;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a ClusterProperties object using a provided SolrZkClient
|
||||||
|
*/
|
||||||
|
public ClusterProperties(SolrZkClient client) {
|
||||||
|
this.client = client;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read the value of a cluster property, returning a default if it is not set
|
||||||
|
* @param key the property name
|
||||||
|
* @param defaultValue the default value
|
||||||
|
* @param <T> the type of the property
|
||||||
|
* @return the property value
|
||||||
|
* @throws IOException if there is an error reading the value from the cluster
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public <T> T getClusterProperty(String key, T defaultValue) throws IOException {
|
||||||
|
T value = (T) getClusterProperties().get(key);
|
||||||
|
if (value == null)
|
||||||
|
return defaultValue;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the cluster properties
|
||||||
|
* @throws IOException if there is an error reading properties from the cluster
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public Map<String, Object> getClusterProperties() throws IOException {
|
||||||
|
try {
|
||||||
|
return (Map<String, Object>) Utils.fromJSON(client.getData(ZkStateReader.CLUSTER_PROPS, null, new Stat(), true));
|
||||||
|
} catch (KeeperException.NoNodeException e) {
|
||||||
|
return Collections.emptyMap();
|
||||||
|
} catch (KeeperException | InterruptedException e) {
|
||||||
|
throw new IOException("Error reading cluster property", SolrZkClient.checkInterrupted(e));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This method sets a cluster property.
|
||||||
|
*
|
||||||
|
* @param propertyName The property name to be set.
|
||||||
|
* @param propertyValue The value of the property.
|
||||||
|
* @throws IOException if there is an error writing data to the cluster
|
||||||
|
*/
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
public void setClusterProperty(String propertyName, String propertyValue) throws IOException {
|
||||||
|
|
||||||
|
if (!ZkStateReader.KNOWN_CLUSTER_PROPS.contains(propertyName)) {
|
||||||
|
throw new SolrException(SolrException.ErrorCode.BAD_REQUEST, "Not a known cluster property " + propertyName);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (; ; ) {
|
||||||
|
Stat s = new Stat();
|
||||||
|
try {
|
||||||
|
if (client.exists(ZkStateReader.CLUSTER_PROPS, true)) {
|
||||||
|
Map properties = (Map) Utils.fromJSON(client.getData(ZkStateReader.CLUSTER_PROPS, null, s, true));
|
||||||
|
if (propertyValue == null) {
|
||||||
|
//Don't update ZK unless absolutely necessary.
|
||||||
|
if (properties.get(propertyName) != null) {
|
||||||
|
properties.remove(propertyName);
|
||||||
|
client.setData(ZkStateReader.CLUSTER_PROPS, Utils.toJSON(properties), s.getVersion(), true);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
//Don't update ZK unless absolutely necessary.
|
||||||
|
if (!propertyValue.equals(properties.get(propertyName))) {
|
||||||
|
properties.put(propertyName, propertyValue);
|
||||||
|
client.setData(ZkStateReader.CLUSTER_PROPS, Utils.toJSON(properties), s.getVersion(), true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Map properties = new LinkedHashMap();
|
||||||
|
properties.put(propertyName, propertyValue);
|
||||||
|
client.create(ZkStateReader.CLUSTER_PROPS, Utils.toJSON(properties), CreateMode.PERSISTENT, true);
|
||||||
|
}
|
||||||
|
} catch (KeeperException.BadVersionException | KeeperException.NodeExistsException e) {
|
||||||
|
//race condition
|
||||||
|
continue;
|
||||||
|
} catch (InterruptedException | KeeperException e) {
|
||||||
|
throw new IOException("Error setting cluster property", SolrZkClient.checkInterrupted(e));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -46,7 +46,6 @@ import org.apache.solr.common.SolrException.ErrorCode;
|
||||||
import org.apache.solr.common.util.ExecutorUtil;
|
import org.apache.solr.common.util.ExecutorUtil;
|
||||||
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.zookeeper.CreateMode;
|
|
||||||
import org.apache.zookeeper.KeeperException;
|
import org.apache.zookeeper.KeeperException;
|
||||||
import org.apache.zookeeper.WatchedEvent;
|
import org.apache.zookeeper.WatchedEvent;
|
||||||
import org.apache.zookeeper.Watcher;
|
import org.apache.zookeeper.Watcher;
|
||||||
|
@ -133,6 +132,8 @@ public class ZkStateReader implements Closeable {
|
||||||
|
|
||||||
private volatile Set<String> liveNodes = emptySet();
|
private volatile Set<String> liveNodes = emptySet();
|
||||||
|
|
||||||
|
private volatile Map<String, Object> clusterProperties = Collections.emptyMap();
|
||||||
|
|
||||||
private final ZkConfigManager configManager;
|
private final ZkConfigManager configManager;
|
||||||
|
|
||||||
private ConfigData securityData;
|
private ConfigData securityData;
|
||||||
|
@ -363,6 +364,7 @@ public class ZkStateReader implements Closeable {
|
||||||
}
|
}
|
||||||
|
|
||||||
// on reconnect of SolrZkClient force refresh and re-add watches.
|
// on reconnect of SolrZkClient force refresh and re-add watches.
|
||||||
|
loadClusterProperties();
|
||||||
refreshLiveNodes(new LiveNodeWatcher());
|
refreshLiveNodes(new LiveNodeWatcher());
|
||||||
refreshLegacyClusterState(new LegacyClusterStateWatcher());
|
refreshLegacyClusterState(new LegacyClusterStateWatcher());
|
||||||
refreshStateFormat2Collections();
|
refreshStateFormat2Collections();
|
||||||
|
@ -793,68 +795,46 @@ public class ZkStateReader implements Closeable {
|
||||||
this.aliases = ClusterState.load(data);
|
this.aliases = ClusterState.load(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Map getClusterProps() {
|
@SuppressWarnings("unchecked")
|
||||||
try {
|
public <T> T getClusterProperty(String key, T defaultValue) {
|
||||||
if (getZkClient().exists(ZkStateReader.CLUSTER_PROPS, true)) {
|
T value = (T) clusterProperties.get(key);
|
||||||
return (Map) Utils.fromJSON(getZkClient().getData(ZkStateReader.CLUSTER_PROPS, null, new Stat(), true)) ;
|
if (value == null)
|
||||||
} else {
|
return defaultValue;
|
||||||
return new LinkedHashMap();
|
return value;
|
||||||
}
|
|
||||||
} catch (InterruptedException e) {
|
|
||||||
Thread.currentThread().interrupt();
|
|
||||||
throw new SolrException(ErrorCode.SERVER_ERROR, "Thread interrupted. Error reading cluster properties", e);
|
|
||||||
} catch (KeeperException e) {
|
|
||||||
throw new SolrException(ErrorCode.SERVER_ERROR, "Error reading cluster properties", e);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public Map<String, Object> getClusterProperties() {
|
||||||
* This method sets a cluster property.
|
return Collections.unmodifiableMap(clusterProperties);
|
||||||
*
|
|
||||||
* @param propertyName The property name to be set.
|
|
||||||
* @param propertyValue The value of the property.
|
|
||||||
*/
|
|
||||||
public void setClusterProperty(String propertyName, String propertyValue) {
|
|
||||||
if (!KNOWN_CLUSTER_PROPS.contains(propertyName)) {
|
|
||||||
throw new SolrException(ErrorCode.BAD_REQUEST, "Not a known cluster property " + propertyName);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (; ; ) {
|
private final Watcher clusterPropertiesWatcher = event -> {
|
||||||
Stat s = new Stat();
|
// session events are not change events, and do not remove the watcher
|
||||||
|
if (Watcher.Event.EventType.None.equals(event.getType())) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
loadClusterProperties();
|
||||||
|
};
|
||||||
|
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
private void loadClusterProperties() {
|
||||||
try {
|
try {
|
||||||
if (getZkClient().exists(CLUSTER_PROPS, true)) {
|
while (true) {
|
||||||
Map properties = (Map) Utils.fromJSON(getZkClient().getData(CLUSTER_PROPS, null, s, true));
|
try {
|
||||||
if (propertyValue == null) {
|
byte[] data = zkClient.getData(ZkStateReader.CLUSTER_PROPS, clusterPropertiesWatcher, new Stat(), true);
|
||||||
//Don't update ZK unless absolutely necessary.
|
this.clusterProperties = (Map<String, Object>) Utils.fromJSON(data);
|
||||||
if (properties.get(propertyName) != null) {
|
LOG.info("Loaded cluster properties: {}", this.clusterProperties);
|
||||||
properties.remove(propertyName);
|
return;
|
||||||
getZkClient().setData(CLUSTER_PROPS, Utils.toJSON(properties), s.getVersion(), true);
|
} catch (KeeperException.NoNodeException e) {
|
||||||
}
|
this.clusterProperties = Collections.emptyMap();
|
||||||
} else {
|
LOG.info("Loaded empty cluster properties");
|
||||||
//Don't update ZK unless absolutely necessary.
|
// set an exists watch, and if the node has been created since the last call,
|
||||||
if (!propertyValue.equals(properties.get(propertyName))) {
|
// read the data again
|
||||||
properties.put(propertyName, propertyValue);
|
if (zkClient.exists(ZkStateReader.CLUSTER_PROPS, clusterPropertiesWatcher, true) == null)
|
||||||
getZkClient().setData(CLUSTER_PROPS, Utils.toJSON(properties), s.getVersion(), true);
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} catch (KeeperException | InterruptedException e) {
|
||||||
Map properties = new LinkedHashMap();
|
LOG.error("Error reading cluster properties from zookeeper", SolrZkClient.checkInterrupted(e));
|
||||||
properties.put(propertyName, propertyValue);
|
|
||||||
getZkClient().create(CLUSTER_PROPS, Utils.toJSON(properties), CreateMode.PERSISTENT, true);
|
|
||||||
}
|
|
||||||
} catch (KeeperException.BadVersionException | KeeperException.NodeExistsException e) {
|
|
||||||
LOG.warn("Race condition while trying to set a new cluster prop on current version [{}]", s.getVersion());
|
|
||||||
//race condition
|
|
||||||
continue;
|
|
||||||
} catch (InterruptedException e) {
|
|
||||||
Thread.currentThread().interrupt();
|
|
||||||
LOG.error("Thread Interrupted. Error updating path [{}]", CLUSTER_PROPS, e);
|
|
||||||
throw new SolrException(ErrorCode.SERVER_ERROR, "Thread Interrupted. Error updating cluster property " + propertyName, e);
|
|
||||||
} catch (KeeperException e) {
|
|
||||||
LOG.error("Error updating path [{}]", CLUSTER_PROPS, e);
|
|
||||||
throw new SolrException(ErrorCode.SERVER_ERROR, "Error updating cluster property " + propertyName, e);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -898,10 +878,7 @@ public class ZkStateReader implements Closeable {
|
||||||
final String hostAndPort = nodeName.substring(0,_offset);
|
final String hostAndPort = nodeName.substring(0,_offset);
|
||||||
try {
|
try {
|
||||||
final String path = URLDecoder.decode(nodeName.substring(1+_offset), "UTF-8");
|
final String path = URLDecoder.decode(nodeName.substring(1+_offset), "UTF-8");
|
||||||
String urlScheme = (String) getClusterProps().get(URL_SCHEME);
|
String urlScheme = getClusterProperty(URL_SCHEME, "http");
|
||||||
if(urlScheme == null) {
|
|
||||||
urlScheme = "http";
|
|
||||||
}
|
|
||||||
return urlScheme + "://" + hostAndPort + (path.isEmpty() ? "" : ("/" + path));
|
return urlScheme + "://" + hostAndPort + (path.isEmpty() ? "" : ("/" + path));
|
||||||
} catch (UnsupportedEncodingException e) {
|
} catch (UnsupportedEncodingException e) {
|
||||||
throw new IllegalStateException("JVM Does not seem to support UTF-8", e);
|
throw new IllegalStateException("JVM Does not seem to support UTF-8", e);
|
||||||
|
|
Loading…
Reference in New Issue