SOLR-9106: Cache cluster properties on ZkStateReader

This commit is contained in:
Alan Woodward 2016-05-13 22:22:23 +01:00
parent ec4bcf1c70
commit dd23fa4015
18 changed files with 312 additions and 255 deletions

View File

@ -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.

View File

@ -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() {

View File

@ -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.");

View File

@ -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,

View File

@ -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

View File

@ -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";

View File

@ -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);

View File

@ -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());

View File

@ -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);
} }

View File

@ -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");

View File

@ -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;

View File

@ -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;
} }

View File

@ -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";
} }
}); });

View File

@ -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

View File

@ -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();

View File

@ -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

View File

@ -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;
}
}
}

View File

@ -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);